Added disconnect.me intent

This commit is contained in:
M66B 2020-07-20 08:25:25 +02:00
parent 674d5a0180
commit 3c2ae7bae8
5 changed files with 94 additions and 46 deletions

6
FAQ.md
View File

@ -3150,6 +3150,12 @@ Tracking images will be disabled only if the corresponding main 'disable' option
Tracking images will not be recognized when the domain is classified as '*Content*', Tracking images will not be recognized when the domain is classified as '*Content*',
see [here](https://disconnect.me/trackerprotection#trackers-we-dont-block) for more information. see [here](https://disconnect.me/trackerprotection#trackers-we-dont-block) for more information.
This command can be sent to FairEmail from an automation app to update the protection lists:
```
(adb shell) am startservice -a eu.faircode.email.DISCONNECT.ME
```
<br /> <br />
## Support ## Support

View File

@ -322,6 +322,7 @@
<action android:name="${applicationId}.POLL" /> <action android:name="${applicationId}.POLL" />
<action android:name="${applicationId}.ENABLE" /> <action android:name="${applicationId}.ENABLE" />
<action android:name="${applicationId}.DISABLE" /> <action android:name="${applicationId}.DISABLE" />
<action android:name="${applicationId}.DISCONNECT.ME" />
</intent-filter> </intent-filter>
</service> </service>

View File

@ -20,8 +20,11 @@ package eu.faircode.email;
*/ */
import android.content.Context; import android.content.Context;
import android.content.SharedPreferences;
import android.os.SystemClock; import android.os.SystemClock;
import androidx.preference.PreferenceManager;
import org.json.JSONArray; import org.json.JSONArray;
import org.json.JSONException; import org.json.JSONException;
import org.json.JSONObject; import org.json.JSONObject;
@ -31,6 +34,7 @@ import java.io.IOException;
import java.net.URL; import java.net.URL;
import java.nio.charset.StandardCharsets; import java.nio.charset.StandardCharsets;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap; import java.util.HashMap;
import java.util.Iterator; import java.util.Iterator;
import java.util.List; import java.util.List;
@ -122,6 +126,9 @@ public class DisconnectBlacklist {
connection.disconnect(); connection.disconnect();
} }
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
prefs.edit().putLong("disconnect_last", new Date().getTime()).apply();
init(file); init(file);
} }
@ -157,9 +164,4 @@ public class DisconnectBlacklist {
private static File getFile(Context context) { private static File getFile(Context context) {
return new File(context.getFilesDir(), "disconnect-blacklist.json"); return new File(context.getFilesDir(), "disconnect-blacklist.json");
} }
static Long getTime(Context context) {
File file = getFile(context);
return (file.exists() ? file.lastModified() : null);
}
} }

View File

@ -376,9 +376,10 @@ public class FragmentOptionsPrivacy extends FragmentBase implements SharedPrefer
swSecure.setChecked(prefs.getBoolean("secure", false)); swSecure.setChecked(prefs.getBoolean("secure", false));
swSafeBrowsing.setChecked(prefs.getBoolean("safe_browsing", false)); swSafeBrowsing.setChecked(prefs.getBoolean("safe_browsing", false));
Long time = DisconnectBlacklist.getTime(getContext()); long time = prefs.getLong("disconnect_last", -1);
DateFormat DF = SimpleDateFormat.getDateTimeInstance(); DateFormat DF = SimpleDateFormat.getDateTimeInstance();
tvDisconnectBlacklistTime.setText(time == null ? null : DF.format(time)); tvDisconnectBlacklistTime.setText(time < 0 ? null : DF.format(time));
tvDisconnectBlacklistTime.setVisibility(time < 0 ? View.GONE : View.VISIBLE);
swDisconnectLinks.setChecked(prefs.getBoolean("disconnect_links", true)); swDisconnectLinks.setChecked(prefs.getBoolean("disconnect_links", true));
swDisconnectImages.setChecked(prefs.getBoolean("disconnect_images", false)); swDisconnectImages.setChecked(prefs.getBoolean("disconnect_images", false));

View File

@ -29,6 +29,10 @@ import androidx.annotation.Nullable;
import androidx.core.app.NotificationCompat; import androidx.core.app.NotificationCompat;
import androidx.preference.PreferenceManager; import androidx.preference.PreferenceManager;
import org.json.JSONException;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Collections; import java.util.Collections;
import java.util.List; import java.util.List;
import java.util.concurrent.ExecutorService; import java.util.concurrent.ExecutorService;
@ -37,10 +41,12 @@ public class ServiceExternal extends Service {
private static final String ACTION_POLL = BuildConfig.APPLICATION_ID + ".POLL"; private static final String ACTION_POLL = BuildConfig.APPLICATION_ID + ".POLL";
private static final String ACTION_ENABLE = BuildConfig.APPLICATION_ID + ".ENABLE"; private static final String ACTION_ENABLE = BuildConfig.APPLICATION_ID + ".ENABLE";
private static final String ACTION_DISABLE = BuildConfig.APPLICATION_ID + ".DISABLE"; private static final String ACTION_DISABLE = BuildConfig.APPLICATION_ID + ".DISABLE";
private static final String ACTION_DISCONNECT_ME = BuildConfig.APPLICATION_ID + ".DISCONNECT.ME";
// adb shell am startservice -a eu.faircode.email.POLL --es account Gmail // adb shell am startservice -a eu.faircode.email.POLL --es account Gmail
// adb shell am startservice -a eu.faircode.email.ENABLE --es account Gmail // adb shell am startservice -a eu.faircode.email.ENABLE --es account Gmail
// adb shell am startservice -a eu.faircode.email.DISABLE --es account Gmail // adb shell am startservice -a eu.faircode.email.DISABLE --es account Gmail
// adb shell am startservice -a eu.faircode.email.DISCONNECT
private static final ExecutorService executor = private static final ExecutorService executor =
Helper.getBackgroundExecutor(1, "external"); Helper.getBackgroundExecutor(1, "external");
@ -72,52 +78,35 @@ public class ServiceExternal extends Service {
if (intent == null) if (intent == null)
return START_NOT_STICKY; return START_NOT_STICKY;
if (!ActivityBilling.isPro(this)) final String action = intent.getAction();
boolean pro = ActivityBilling.isPro(this);
EntityLog.log(this, action + " pro=" + pro);
if (!pro)
return START_NOT_STICKY; return START_NOT_STICKY;
final Context context = getApplicationContext(); final Context context = getApplicationContext();
final String accountName = intent.getStringExtra("account");
final Boolean enabled;
String action = intent.getAction();
if (ACTION_ENABLE.equals(action))
enabled = true;
else if (ACTION_DISABLE.equals(action))
enabled = false;
else // poll
enabled = null;
executor.submit(new Runnable() { executor.submit(new Runnable() {
@Override @Override
public void run() { public void run() {
DB db = DB.getInstance(context); try {
switch (action) {
if (enabled == null) { case ACTION_POLL:
List<EntityAccount> accounts = db.account().getSynchronizingAccounts(); poll(context, intent);
for (EntityAccount account : accounts) break;
if (accountName == null || accountName.equals(account.name)) { case ACTION_ENABLE:
List<EntityFolder> folders = db.folder().getSynchronizingFolders(account.id); case ACTION_DISABLE:
if (folders.size() > 0) set(context, intent);
Collections.sort(folders, folders.get(0).getComparator(context)); break;
for (EntityFolder folder : folders) case ACTION_DISCONNECT_ME:
EntityOperation.sync(context, folder.id, false); disconnect(context, intent);
} break;
ServiceSynchronize.eval(context, "external poll account=" + accountName); default:
} else { throw new IllegalArgumentException(action);
if (accountName == null) {
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
prefs.edit().putBoolean("enabled", enabled).apply();
ServiceSynchronize.eval(context, "external enabled=" + enabled);
} else {
EntityAccount account = db.account().getAccount(accountName);
if (account == null) {
EntityLog.log(context, "Account not found name=" + accountName);
return;
}
db.account().setAccountSynchronize(account.id, enabled);
ServiceSynchronize.eval(context, "external account=" + accountName + " enabled=" + enabled);
} }
} catch (Throwable ex) {
Log.e(ex);
EntityLog.log(context, Log.formatThrowable(ex));
} }
} }
}); });
@ -149,4 +138,53 @@ public class ServiceExternal extends Service {
return builder; return builder;
} }
private static void poll(Context context, Intent intent) {
String accountName = intent.getStringExtra("account");
DB db = DB.getInstance(context);
List<EntityAccount> accounts;
if (accountName == null)
accounts = db.account().getSynchronizingAccounts();
else {
EntityAccount account = db.account().getAccount(accountName);
if (account == null)
throw new IllegalArgumentException("Account not found name=" + accountName);
accounts = new ArrayList<>();
accounts.add(account);
}
for (EntityAccount account : accounts) {
List<EntityFolder> folders = db.folder().getSynchronizingFolders(account.id);
if (folders.size() > 0)
Collections.sort(folders, folders.get(0).getComparator(context));
for (EntityFolder folder : folders)
EntityOperation.sync(context, folder.id, false);
}
ServiceSynchronize.eval(context, "external poll account=" + accountName);
}
private static void set(Context context, Intent intent) {
String accountName = intent.getStringExtra("account");
boolean enabled = ACTION_ENABLE.equals(intent.getAction());
DB db = DB.getInstance(context);
if (accountName == null) {
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
prefs.edit().putBoolean("enabled", enabled).apply();
ServiceSynchronize.eval(context, "external enabled=" + enabled);
} else {
EntityAccount account = db.account().getAccount(accountName);
if (account == null)
throw new IllegalArgumentException("Account not found name=" + accountName);
db.account().setAccountSynchronize(account.id, enabled);
ServiceSynchronize.eval(context, "external account=" + accountName + " enabled=" + enabled);
}
}
private static void disconnect(Context context, Intent intent) throws IOException, JSONException {
DisconnectBlacklist.download(context);
}
} }