mirror of
https://github.com/M66B/FairEmail.git
synced 2026-01-04 11:54:10 +01:00
Cached disconnect list
This commit is contained in:
@@ -5957,36 +5957,11 @@ public class AdapterMessage extends RecyclerView.Adapter<AdapterMessage.ViewHold
|
|||||||
uriTitle.getHost().equalsIgnoreCase(uri.getHost())
|
uriTitle.getHost().equalsIgnoreCase(uri.getHost())
|
||||||
? View.GONE : View.VISIBLE);
|
? View.GONE : View.VISIBLE);
|
||||||
|
|
||||||
Bundle args = new Bundle();
|
List<String> categories = DisconnectBlacklist.getCategories(uri.getHost());
|
||||||
args.putParcelable("uri", uri);
|
if (categories != null)
|
||||||
|
tvDisconnectCategories.setText(TextUtils.join(", ", categories));
|
||||||
new SimpleTask<List<String>>() {
|
tvDisconnect.setVisibility(categories == null ? View.GONE : View.VISIBLE);
|
||||||
@Override
|
tvDisconnectCategories.setVisibility(categories == null ? View.GONE : View.VISIBLE);
|
||||||
protected void onPreExecute(Bundle args) {
|
|
||||||
tvDisconnect.setVisibility(View.GONE);
|
|
||||||
tvDisconnectCategories.setVisibility(View.GONE);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected List<String> onExecute(Context context, Bundle args) throws Throwable {
|
|
||||||
Uri uri = args.getParcelable("uri");
|
|
||||||
return DisconnectBlacklist.getCategories(uri.getHost(), context);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onExecuted(Bundle args, List<String> data) {
|
|
||||||
if (data != null) {
|
|
||||||
tvDisconnectCategories.setText(TextUtils.join(", ", data));
|
|
||||||
tvDisconnect.setVisibility(View.VISIBLE);
|
|
||||||
tvDisconnectCategories.setVisibility(View.VISIBLE);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void onException(Bundle args, Throwable ex) {
|
|
||||||
Log.unexpectedError(getParentFragmentManager(), ex);
|
|
||||||
}
|
|
||||||
}.execute(getContext(), getViewLifecycleOwner(), args, "disconnect");
|
|
||||||
|
|
||||||
final Context context = getContext();
|
final Context context = getContext();
|
||||||
|
|
||||||
|
|||||||
@@ -114,6 +114,8 @@ public class ApplicationEx extends Application {
|
|||||||
MessageHelper.setSystemProperties(this);
|
MessageHelper.setSystemProperties(this);
|
||||||
ContactInfo.init(this);
|
ContactInfo.init(this);
|
||||||
|
|
||||||
|
DisconnectBlacklist.init(this);
|
||||||
|
|
||||||
WorkerWatchdog.init(this);
|
WorkerWatchdog.init(this);
|
||||||
WorkerCleanup.queue(this);
|
WorkerCleanup.queue(this);
|
||||||
|
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ package eu.faircode.email;
|
|||||||
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
|
import android.os.SystemClock;
|
||||||
|
|
||||||
import org.json.JSONArray;
|
import org.json.JSONArray;
|
||||||
import org.json.JSONException;
|
import org.json.JSONException;
|
||||||
@@ -31,15 +32,86 @@ 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.HashMap;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Locale;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.concurrent.ExecutorService;
|
||||||
|
|
||||||
import javax.net.ssl.HttpsURLConnection;
|
import javax.net.ssl.HttpsURLConnection;
|
||||||
|
|
||||||
public class DisconnectBlacklist {
|
public class DisconnectBlacklist {
|
||||||
|
private static final Map<String, List<String>> map = new HashMap<>();
|
||||||
|
private static final ExecutorService executor = Helper.getBackgroundExecutor(1, "disconnect");
|
||||||
|
|
||||||
private final static int FETCH_TIMEOUT = 20 * 1000; // milliseconds
|
private final static int FETCH_TIMEOUT = 20 * 1000; // milliseconds
|
||||||
private final static String LIST = "https://raw.githubusercontent.com/disconnectme/disconnect-tracking-protection/master/services.json";
|
private final static String LIST = "https://raw.githubusercontent.com/disconnectme/disconnect-tracking-protection/master/services.json";
|
||||||
|
|
||||||
|
static void init(Context context) {
|
||||||
|
final File file = getFile(context);
|
||||||
|
|
||||||
|
executor.submit(new Runnable() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
try {
|
||||||
|
init(file);
|
||||||
|
} catch (Throwable ex) {
|
||||||
|
Log.e(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void init(File file) throws IOException, JSONException {
|
||||||
|
synchronized (map) {
|
||||||
|
long start = SystemClock.elapsedRealtime();
|
||||||
|
|
||||||
|
map.clear();
|
||||||
|
|
||||||
|
String json = Helper.readText(file);
|
||||||
|
JSONObject jdisconnect = new JSONObject(json);
|
||||||
|
JSONObject jcategories = (JSONObject) jdisconnect.get("categories");
|
||||||
|
Iterator<String> categories = jcategories.keys();
|
||||||
|
while (categories.hasNext()) {
|
||||||
|
String category = categories.next();
|
||||||
|
JSONArray jcategory = jcategories.getJSONArray(category);
|
||||||
|
for (int c = 0; c < jcategory.length(); c++) {
|
||||||
|
JSONObject jblock = (JSONObject) jcategory.get(c);
|
||||||
|
Iterator<String> names = jblock.keys();
|
||||||
|
if (names.hasNext()) {
|
||||||
|
String name = names.next();
|
||||||
|
JSONObject jsites = (JSONObject) jblock.get(name);
|
||||||
|
Iterator<String> sites = jsites.keys();
|
||||||
|
if (sites.hasNext()) {
|
||||||
|
List<String> domains = new ArrayList<>();
|
||||||
|
|
||||||
|
String site = sites.next();
|
||||||
|
String host = Uri.parse(site).getHost();
|
||||||
|
if (host != null)
|
||||||
|
domains.add(host.toLowerCase(Locale.ROOT));
|
||||||
|
|
||||||
|
JSONArray jdomains = jsites.getJSONArray(site);
|
||||||
|
for (int d = 0; d < jdomains.length(); d++)
|
||||||
|
domains.add(jdomains.getString(d).toLowerCase(Locale.ROOT));
|
||||||
|
|
||||||
|
for (String domain : domains) {
|
||||||
|
if (!map.containsKey(domain))
|
||||||
|
map.put(domain, new ArrayList<>());
|
||||||
|
List<String> list = map.get(domain);
|
||||||
|
if (!list.contains(category))
|
||||||
|
list.add(category);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
long elapsed = SystemClock.elapsedRealtime() - start;
|
||||||
|
Log.i("Disconnect domains=" + map.size() + " elapsed=" + elapsed + " ms");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void download(Context context) throws IOException, JSONException {
|
static void download(Context context) throws IOException, JSONException {
|
||||||
File file = getFile(context);
|
File file = getFile(context);
|
||||||
|
|
||||||
@@ -57,53 +129,18 @@ public class DisconnectBlacklist {
|
|||||||
} finally {
|
} finally {
|
||||||
connection.disconnect();
|
connection.disconnect();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
init(file);
|
||||||
}
|
}
|
||||||
|
|
||||||
static List<String> getCategories(String domain, Context context) throws IOException, JSONException {
|
static List<String> getCategories(String domain) {
|
||||||
if (domain == null)
|
if (domain == null)
|
||||||
return null;
|
return null;
|
||||||
|
|
||||||
File file = getFile(context);
|
synchronized (map) {
|
||||||
if (!file.exists())
|
List<String> result = map.get(domain.toLowerCase(Locale.ROOT));
|
||||||
return null;
|
return (result == null || result.size() == 0 ? null : result);
|
||||||
|
|
||||||
List<String> result = new ArrayList<>();
|
|
||||||
|
|
||||||
String json = Helper.readText(file);
|
|
||||||
JSONObject jdisconnect = new JSONObject(json);
|
|
||||||
JSONObject jcategories = (JSONObject) jdisconnect.get("categories");
|
|
||||||
Iterator<String> categories = jcategories.keys();
|
|
||||||
while (categories.hasNext()) {
|
|
||||||
String category = categories.next();
|
|
||||||
JSONArray jcategory = jcategories.getJSONArray(category);
|
|
||||||
for (int c = 0; c < jcategory.length(); c++) {
|
|
||||||
JSONObject jblock = (JSONObject) jcategory.get(c);
|
|
||||||
Iterator<String> names = jblock.keys();
|
|
||||||
if (names.hasNext()) {
|
|
||||||
String name = names.next();
|
|
||||||
JSONObject jsites = (JSONObject) jblock.get(name);
|
|
||||||
Iterator<String> sites = jsites.keys();
|
|
||||||
if (sites.hasNext()) {
|
|
||||||
List<String> domains = new ArrayList<>();
|
|
||||||
|
|
||||||
String site = sites.next();
|
|
||||||
String host = Uri.parse(site).getHost();
|
|
||||||
if (host != null)
|
|
||||||
domains.add(host);
|
|
||||||
|
|
||||||
JSONArray jdomains = jsites.getJSONArray(site);
|
|
||||||
for (int d = 0; d < jdomains.length(); d++)
|
|
||||||
domains.add(jdomains.getString(d));
|
|
||||||
|
|
||||||
for (String d : domains)
|
|
||||||
if (domain.equalsIgnoreCase(d) && !result.contains(category))
|
|
||||||
result.add(category);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return (result.size() == 0 ? null : result);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static File getFile(Context context) {
|
private static File getFile(Context context) {
|
||||||
|
|||||||
Reference in New Issue
Block a user