FairEmail/app/src/main/java/eu/faircode/email/Log.java

881 lines
37 KiB
Java
Raw Normal View History

2018-12-24 12:27:45 +00:00
package eu.faircode.email;
2018-12-31 08:04:33 +00:00
/*
This file is part of FairEmail.
FairEmail is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
FairEmail is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with FairEmail. If not, see <http://www.gnu.org/licenses/>.
Copyright 2018-2019 by Marcel Bokhorst (M66B)
*/
2019-07-22 07:18:03 +00:00
import android.app.ActivityManager;
2019-05-12 17:14:34 +00:00
import android.app.usage.UsageStatsManager;
import android.content.Context;
2019-01-25 14:40:46 +00:00
import android.content.Intent;
2019-05-12 17:14:34 +00:00
import android.content.SharedPreferences;
import android.content.res.Configuration;
import android.graphics.Point;
import android.net.ConnectivityManager;
import android.net.Network;
import android.net.NetworkCapabilities;
import android.net.NetworkInfo;
2019-10-29 06:57:39 +00:00
import android.os.BadParcelableException;
2019-05-12 17:14:34 +00:00
import android.os.Build;
2019-01-25 14:40:46 +00:00
import android.os.Bundle;
2019-10-12 11:09:14 +00:00
import android.os.DeadObjectException;
2019-07-22 07:18:03 +00:00
import android.os.Debug;
2019-05-12 17:14:34 +00:00
import android.os.PowerManager;
2019-08-12 11:17:55 +00:00
import android.os.RemoteException;
2019-01-25 14:40:46 +00:00
import android.text.TextUtils;
2019-05-12 17:14:34 +00:00
import android.view.Display;
2019-08-12 11:17:55 +00:00
import android.view.OrientationEventListener;
2019-05-12 17:14:34 +00:00
import android.view.WindowManager;
2019-08-12 11:17:55 +00:00
import androidx.annotation.NonNull;
2019-05-12 17:14:34 +00:00
import androidx.preference.PreferenceManager;
2019-01-25 14:40:46 +00:00
2019-08-12 11:17:55 +00:00
import com.bugsnag.android.BeforeNotify;
import com.bugsnag.android.BeforeSend;
2019-08-12 11:07:14 +00:00
import com.bugsnag.android.BreadcrumbType;
2019-05-10 06:53:45 +00:00
import com.bugsnag.android.Bugsnag;
import com.bugsnag.android.Callback;
2019-08-12 11:17:55 +00:00
import com.bugsnag.android.Client;
import com.bugsnag.android.Error;
import com.bugsnag.android.Report;
2019-05-10 06:53:45 +00:00
import com.bugsnag.android.Severity;
2019-08-12 11:17:55 +00:00
import com.sun.mail.iap.ProtocolException;
2019-05-10 06:53:45 +00:00
2019-05-12 17:14:34 +00:00
import org.json.JSONException;
import org.json.JSONObject;
import java.io.BufferedOutputStream;
import java.io.BufferedReader;
import java.io.File;
2019-08-12 11:17:55 +00:00
import java.io.FileNotFoundException;
2019-05-12 17:14:34 +00:00
import java.io.FileOutputStream;
2019-08-12 11:17:55 +00:00
import java.io.FileWriter;
2019-05-12 17:14:34 +00:00
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.UnsupportedEncodingException;
2019-04-24 16:39:32 +00:00
import java.lang.reflect.Array;
2019-08-12 11:17:55 +00:00
import java.lang.reflect.Field;
2019-05-12 17:14:34 +00:00
import java.text.DateFormat;
2019-08-12 11:17:55 +00:00
import java.util.ArrayList;
import java.util.Arrays;
2019-05-12 17:14:34 +00:00
import java.util.Date;
import java.util.List;
import java.util.Map;
2019-01-25 14:40:46 +00:00
import java.util.Set;
2019-08-12 11:17:55 +00:00
import java.util.UUID;
import java.util.concurrent.TimeoutException;
2019-01-25 14:40:46 +00:00
2019-05-12 17:14:34 +00:00
import javax.mail.Address;
2019-08-12 11:17:55 +00:00
import javax.mail.MessagingException;
2019-06-29 06:52:15 +00:00
import javax.mail.Part;
2019-05-12 17:14:34 +00:00
import javax.mail.internet.InternetAddress;
2018-12-24 12:27:45 +00:00
public class Log {
2019-08-23 13:22:35 +00:00
private static final int MAX_CRASH_REPORTS = 5;
2019-08-12 11:33:39 +00:00
private static final String TAG = "fairemail";
2018-12-24 12:27:45 +00:00
2019-08-11 13:49:14 +00:00
public static int d(String msg) {
if (BuildConfig.DEBUG)
return android.util.Log.d(TAG, msg);
else
return 0;
}
2018-12-24 12:27:45 +00:00
public static int i(String msg) {
2018-12-28 09:56:40 +00:00
if (BuildConfig.BETA_RELEASE)
return android.util.Log.i(TAG, msg);
else
return 0;
2018-12-24 12:27:45 +00:00
}
public static int w(String msg) {
return android.util.Log.w(TAG, msg);
}
public static int e(String msg) {
if (BuildConfig.BETA_RELEASE) {
List<StackTraceElement> ss = new ArrayList<>(Arrays.asList(new Throwable().getStackTrace()));
ss.remove(0);
Bugsnag.notify("Internal error", msg, ss.toArray(new StackTraceElement[0]), new Callback() {
@Override
public void beforeNotify(@NonNull Report report) {
report.getError().setSeverity(Severity.ERROR);
}
});
}
2018-12-24 12:27:45 +00:00
return android.util.Log.e(TAG, msg);
}
2019-05-14 08:07:05 +00:00
public static int i(Throwable ex) {
2019-06-09 13:51:24 +00:00
return android.util.Log.i(TAG, ex + "\n" + android.util.Log.getStackTraceString(ex));
2019-05-14 08:07:05 +00:00
}
2018-12-24 12:27:45 +00:00
public static int w(Throwable ex) {
2019-05-15 19:20:18 +00:00
if (BuildConfig.BETA_RELEASE)
Bugsnag.notify(ex, Severity.INFO);
2018-12-24 12:27:45 +00:00
return android.util.Log.w(TAG, ex + "\n" + android.util.Log.getStackTraceString(ex));
}
public static int e(Throwable ex) {
2019-05-15 19:20:18 +00:00
if (BuildConfig.BETA_RELEASE)
Bugsnag.notify(ex, Severity.WARNING);
2018-12-24 12:27:45 +00:00
return android.util.Log.e(TAG, ex + "\n" + android.util.Log.getStackTraceString(ex));
}
public static int w(String prefix, Throwable ex) {
2019-05-15 19:20:18 +00:00
if (BuildConfig.BETA_RELEASE)
Bugsnag.notify(ex, Severity.INFO);
2018-12-24 12:27:45 +00:00
return android.util.Log.w(TAG, prefix + " " + ex + "\n" + android.util.Log.getStackTraceString(ex));
}
public static int e(String prefix, Throwable ex) {
2019-05-15 19:20:18 +00:00
if (BuildConfig.BETA_RELEASE)
Bugsnag.notify(ex, Severity.WARNING);
2018-12-24 12:27:45 +00:00
return android.util.Log.e(TAG, prefix + " " + ex + "\n" + android.util.Log.getStackTraceString(ex));
}
2019-01-25 14:40:46 +00:00
2019-08-12 11:17:55 +00:00
static void setCrashReporting(boolean enabled) {
if (enabled)
Bugsnag.startSession();
else
Bugsnag.stopSession();
}
2019-08-12 11:07:14 +00:00
static void breadcrumb(String name, Map<String, String> crumb) {
2019-08-12 11:17:55 +00:00
Bugsnag.leaveBreadcrumb(name, BreadcrumbType.LOG, crumb);
}
static void setupBugsnag(Context context) {
// https://docs.bugsnag.com/platforms/android/sdk/
com.bugsnag.android.Configuration config =
new com.bugsnag.android.Configuration("9d2d57476a0614974449a3ec33f2604a");
if (BuildConfig.DEBUG)
config.setReleaseStage("debug");
else {
String type = "other";
if (Helper.hasValidFingerprint(context))
if (BuildConfig.PLAY_STORE_RELEASE)
type = "play";
else
type = "full";
config.setReleaseStage(type + (BuildConfig.BETA_RELEASE ? "/beta" : ""));
}
config.setAutoCaptureSessions(false);
2019-09-14 11:04:48 +00:00
config.setDetectAnrs(false);
2019-08-12 11:17:55 +00:00
config.setDetectNdkCrashes(false);
List<String> ignore = new ArrayList<>();
ignore.add("com.sun.mail.util.MailConnectException");
ignore.add("android.accounts.OperationCanceledException");
ignore.add("android.app.RemoteServiceException");
ignore.add("java.lang.NoClassDefFoundError");
ignore.add("java.lang.UnsatisfiedLinkError");
ignore.add("java.nio.charset.MalformedInputException");
ignore.add("java.net.ConnectException");
ignore.add("java.net.SocketException");
ignore.add("java.net.SocketTimeoutException");
ignore.add("java.net.UnknownHostException");
ignore.add("javax.mail.AuthenticationFailedException");
ignore.add("javax.mail.FolderClosedException");
ignore.add("javax.mail.internet.AddressException");
ignore.add("javax.mail.MessageRemovedException");
ignore.add("javax.mail.ReadOnlyFolderException");
ignore.add("javax.mail.StoreClosedException");
ignore.add("org.xmlpull.v1.XmlPullParserException");
config.setIgnoreClasses(ignore.toArray(new String[0]));
final SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
config.beforeSend(new BeforeSend() {
2019-08-12 11:17:55 +00:00
@Override
public boolean run(@NonNull Report report) {
2019-09-12 16:32:15 +00:00
// opt-in
boolean crash_reports = prefs.getBoolean("crash_reports", false);
2019-08-29 14:57:36 +00:00
if (!crash_reports)
return false;
Throwable ex = report.getError().getException();
2019-10-28 17:50:39 +00:00
return shouldNotify(ex);
}
2019-08-12 11:17:55 +00:00
2019-10-28 17:50:39 +00:00
private boolean shouldNotify(Throwable ex) {
2019-08-12 11:17:55 +00:00
if (ex instanceof MessagingException &&
(ex.getCause() instanceof IOException ||
ex.getCause() instanceof ProtocolException))
// IOException includes SocketException, SocketTimeoutException
// ProtocolException includes ConnectionException
return false;
if (ex instanceof MessagingException &&
("connection failure".equals(ex.getMessage()) ||
"failed to create new store connection".equals(ex.getMessage()) ||
"Failed to fetch headers".equals(ex.getMessage()) ||
"Failed to load IMAP envelope".equals(ex.getMessage()) ||
"Unable to load BODYSTRUCTURE".equals(ex.getMessage())))
return false;
if (ex instanceof IllegalStateException &&
("Not connected".equals(ex.getMessage()) ||
"This operation is not allowed on a closed folder".equals(ex.getMessage())))
return false;
if (ex instanceof FileNotFoundException &&
ex.getMessage() != null &&
(ex.getMessage().startsWith("Download image failed") ||
ex.getMessage().startsWith("https://ipinfo.io/") ||
ex.getMessage().startsWith("https://autoconfig.thunderbird.net/")))
return false;
2019-11-02 16:15:41 +00:00
if (ex instanceof IOException &&
"Resetting to invalid mark".equals(ex.getMessage()))
return false;
2019-09-12 16:32:15 +00:00
// Rate limit
2019-10-28 17:50:39 +00:00
int count = prefs.getInt("crash_report_count", 0) + 1;
2019-09-12 16:32:15 +00:00
prefs.edit().putInt("crash_report_count", count).apply();
2019-10-28 17:50:39 +00:00
return (count <= MAX_CRASH_REPORTS);
}
});
Bugsnag.init(context, config);
Client client = Bugsnag.getClient();
try {
Log.i("Disabling orientation listener");
Field fOrientationListener = Client.class.getDeclaredField("orientationListener");
fOrientationListener.setAccessible(true);
OrientationEventListener orientationListener = (OrientationEventListener) fOrientationListener.get(client);
orientationListener.disable();
Log.i("Disabled orientation listener");
} catch (Throwable ex) {
Log.e(ex);
}
String uuid = prefs.getString("uuid", null);
if (uuid == null) {
uuid = UUID.randomUUID().toString();
prefs.edit().putString("uuid", uuid).apply();
}
Log.i("uuid=" + uuid);
client.setUserId(uuid);
if (prefs.getBoolean("crash_reports", false))
Bugsnag.startSession();
final String installer = context.getPackageManager().getInstallerPackageName(BuildConfig.APPLICATION_ID);
final boolean fingerprint = Helper.hasValidFingerprint(context);
2019-11-08 07:58:19 +00:00
final Boolean ignoringOptimizations = Helper.isIgnoringOptimizations(context);
Bugsnag.beforeNotify(new BeforeNotify() {
@Override
public boolean run(@NonNull Error error) {
error.addToTab("extra", "installer", installer == null ? "-" : installer);
error.addToTab("extra", "fingerprint", fingerprint);
2019-11-09 13:13:18 +00:00
error.addToTab("extra", "thread", Thread.currentThread().getName() + ":" + Thread.currentThread().getId());
error.addToTab("extra", "free", Log.getFreeMemMb());
2019-11-08 07:58:19 +00:00
error.addToTab("extra", "optimizing", (ignoringOptimizations != null && !ignoringOptimizations));
String theme = prefs.getString("theme", "light");
error.addToTab("extra", "theme", theme);
error.addToTab("extra", "package", BuildConfig.APPLICATION_ID);
return true;
2019-08-12 11:17:55 +00:00
}
});
2019-08-12 11:07:14 +00:00
}
2019-05-15 19:20:18 +00:00
static void logExtras(Intent intent) {
2019-01-25 14:40:46 +00:00
if (intent != null)
logBundle(intent.getExtras());
}
2019-05-15 19:20:18 +00:00
static void logBundle(Bundle data) {
2019-10-01 07:19:44 +00:00
for (String extra : getExtras(data))
i(extra);
}
static List<String> getExtras(Bundle data) {
List<String> result = new ArrayList<>();
if (data == null)
return result;
2019-10-29 06:57:39 +00:00
try {
Set<String> keys = data.keySet();
for (String key : keys) {
Object v = data.get(key);
Object value = v;
if (v != null && v.getClass().isArray()) {
int length = Array.getLength(v);
if (length <= 10) {
String[] elements = new String[length];
for (int i = 0; i < length; i++) {
Object element = Array.get(v, i);
if (element instanceof Long)
elements[i] = "0x" + Long.toHexString((Long) element);
else
elements[i] = (element == null ? null : element.toString());
}
value = TextUtils.join(",", elements);
2019-04-24 16:39:32 +00:00
}
2019-10-29 06:57:39 +00:00
} else if (v instanceof Long)
value = "0x" + Long.toHexString((Long) v);
2019-10-01 07:19:44 +00:00
2019-10-29 06:57:39 +00:00
result.add(key + "=" + value + (value == null ? "" : " (" + v.getClass().getSimpleName() + ")"));
}
} catch (BadParcelableException ex) {
// android.os.BadParcelableException: ClassNotFoundException when unmarshalling: ...
Log.e(ex);
2019-01-25 14:40:46 +00:00
}
2019-10-01 07:19:44 +00:00
return result;
2019-01-25 14:40:46 +00:00
}
2019-05-12 17:14:34 +00:00
2019-08-12 11:17:55 +00:00
static void logMemory(Context context, String message) {
ActivityManager.MemoryInfo mi = new ActivityManager.MemoryInfo();
ActivityManager activityManager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
activityManager.getMemoryInfo(mi);
int mb = Math.round(mi.availMem / 0x100000L);
int perc = Math.round(mi.availMem / (float) mi.totalMem * 100.0f);
Log.i(message + " " + mb + " MB" + " " + perc + " %");
}
2019-08-12 14:18:41 +00:00
static boolean isOwnFault(Throwable ex) {
2019-09-02 12:52:49 +00:00
if (!isSupportedDevice())
return false;
2019-08-12 11:17:55 +00:00
if (ex instanceof OutOfMemoryError)
return false;
if (ex instanceof RemoteException)
return false;
/*
java.lang.NoSuchMethodError: No direct method ()V in class Landroid/security/IKeyChainService$Stub; or its super classes (declaration of 'android.security.IKeyChainService$Stub' appears in /system/framework/framework.jar!classes2.dex)
java.lang.NoSuchMethodError: No direct method ()V in class Landroid/security/IKeyChainService$Stub; or its super classes (declaration of 'android.security.IKeyChainService$Stub' appears in /system/framework/framework.jar!classes2.dex)
at com.android.keychain.KeyChainService$1.(KeyChainService.java:95)
at com.android.keychain.KeyChainService.(KeyChainService.java:95)
at java.lang.Class.newInstance(Native Method)
at android.app.AppComponentFactory.instantiateService(AppComponentFactory.java:103)
*/
if (ex instanceof NoSuchMethodError)
return false;
2019-09-03 06:18:30 +00:00
if (ex instanceof IllegalStateException &&
"Drag shadow dimensions must be positive".equals(ex.getMessage()))
/*
Android 9 only
java.lang.IllegalStateException: Drag shadow dimensions must be positive
java.lang.IllegalStateException: Drag shadow dimensions must be positive
at android.view.View.startDragAndDrop(View.java:24027)
at android.widget.Editor.startDragAndDrop(Editor.java:1165)
at android.widget.Editor.performLongClick(Editor.java:1191)
at android.widget.TextView.performLongClick(TextView.java:11346)
at android.view.View.performLongClick(View.java:6653)
at android.view.View$CheckForLongPress.run(View.java:25855)
at android.os.Handler.handleCallback(Handler.java:873)
*/
return false;
2019-09-19 08:32:55 +00:00
if (ex instanceof IllegalArgumentException &&
ex.getMessage() != null &&
ex.getMessage().startsWith("Tmp detached view should be removed from RecyclerView before it can be recycled"))
/*
Android 9 only?
java.lang.IllegalArgumentException: Tmp detached view should be removed from RecyclerView before it can be recycled: ViewHolder{e3b70bd position=0 id=1, oldPos=-1, pLpos:-1 update tmpDetached no parent} androidx.recyclerview.widget.RecyclerView{f0fe5b1 VFED..... ......ID 0,0-641,456 #7f090293 app:id/rvAccount}, adapter:eu.faircode.email.AdapterNavAccount@9a6ea96, layout:androidx.recyclerview.widget.LinearLayoutManager@d8fc617, context:eu.faircode.email.ActivityView@82a6ec4
at androidx.recyclerview.widget.RecyclerView$Recycler.recycleViewHolderInternal(SourceFile:6435)
at androidx.recyclerview.widget.RecyclerView.removeAnimatingView(SourceFile:1456)
at androidx.recyclerview.widget.RecyclerView$ItemAnimatorRestoreListener.onAnimationFinished(SourceFile:12690)
at androidx.recyclerview.widget.RecyclerView$ItemAnimator.dispatchAnimationFinished(SourceFile:13190)
at androidx.recyclerview.widget.SimpleItemAnimator.dispatchChangeFinished(SourceFile:317)
at androidx.recyclerview.widget.DefaultItemAnimator$8.onAnimationEnd(SourceFile:391)
at android.view.ViewPropertyAnimator$AnimatorEventListener.onAnimationEnd(ViewPropertyAnimator.java:1122)
*/
return false;
2019-08-12 11:17:55 +00:00
if (ex.getMessage() != null &&
(ex.getMessage().startsWith("Bad notification posted") ||
ex.getMessage().contains("ActivityRecord not found") ||
ex.getMessage().startsWith("Unable to create layer") ||
2019-11-10 08:26:39 +00:00
ex.getMessage().startsWith("Illegal meta data value") ||
ex.getMessage().startsWith("Context.startForegroundService")))
2019-08-12 11:17:55 +00:00
return false;
if (ex instanceof TimeoutException &&
ex.getMessage() != null &&
2019-09-19 08:32:55 +00:00
ex.getMessage().contains("finalize"))
2019-08-12 11:17:55 +00:00
return false;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N)
2019-10-12 11:09:14 +00:00
if (ex instanceof RuntimeException && ex.getCause() instanceof DeadObjectException)
2019-08-12 11:17:55 +00:00
return false;
if (BuildConfig.BETA_RELEASE)
return true;
while (ex != null) {
for (StackTraceElement ste : ex.getStackTrace())
if (ste.getClassName().startsWith(BuildConfig.APPLICATION_ID))
return true;
ex = ex.getCause();
}
return false;
}
2019-08-12 14:18:41 +00:00
static void writeCrashLog(Context context, Throwable ex) {
2019-08-12 11:17:55 +00:00
File file = new File(context.getCacheDir(), "crash.log");
Log.w("Writing exception to " + file);
try (FileWriter out = new FileWriter(file, true)) {
out.write(BuildConfig.VERSION_NAME + " " + new Date() + "\r\n");
out.write(ex + "\r\n" + android.util.Log.getStackTraceString(ex) + "\r\n");
} catch (IOException e) {
Log.e(e);
}
}
2019-05-12 17:14:34 +00:00
static EntityMessage getDebugInfo(Context context, int title, Throwable ex, String log) throws IOException {
StringBuilder sb = new StringBuilder();
sb.append(context.getString(title)).append("\n\n\n\n");
sb.append(getAppInfo(context));
if (ex != null)
sb.append(ex.toString()).append("\n").append(android.util.Log.getStackTraceString(ex));
if (log != null)
sb.append(log);
2019-09-26 19:52:47 +00:00
String body = "<pre>" + TextUtils.htmlEncode(sb.toString()) + "</pre>";
2019-05-12 17:14:34 +00:00
EntityMessage draft;
DB db = DB.getInstance(context);
try {
db.beginTransaction();
2019-09-23 08:04:46 +00:00
List<TupleIdentityEx> identities = db.identity().getComposableIdentities(null);
if (identities == null || identities.size() == 0)
throw new IllegalArgumentException(context.getString(R.string.title_no_identities));
EntityIdentity identity = identities.get(0);
EntityFolder drafts = db.folder().getFolderByType(identity.account, EntityFolder.DRAFTS);
2019-05-12 17:14:34 +00:00
draft = new EntityMessage();
draft.account = drafts.account;
draft.folder = drafts.id;
2019-09-23 08:04:46 +00:00
draft.identity = identity.id;
2019-05-12 17:14:34 +00:00
draft.msgid = EntityMessage.generateMessageId();
draft.thread = draft.msgid;
draft.to = new Address[]{myAddress()};
draft.subject = context.getString(R.string.app_name) + " " + BuildConfig.VERSION_NAME + " debug info";
draft.received = new Date().getTime();
draft.seen = true;
draft.ui_seen = true;
draft.id = db.message().insertMessage(draft);
Helper.writeText(draft.getFile(context), body);
db.message().setMessageContent(draft.id,
true,
false,
HtmlHelper.getPreview(body),
null);
2019-07-21 18:09:55 +00:00
attachSettings(context, draft.id, 1);
attachAccounts(context, draft.id, 2);
attachNetworkInfo(context, draft.id, 3);
attachLog(context, draft.id, 4);
attachOperations(context, draft.id, 5);
attachLogcat(context, draft.id, 6);
EntityOperation.queue(context, draft, EntityOperation.ADD);
2019-05-12 17:14:34 +00:00
db.setTransactionSuccessful();
} finally {
db.endTransaction();
}
2019-05-12 17:14:34 +00:00
return draft;
}
private static StringBuilder getAppInfo(Context context) {
StringBuilder sb = new StringBuilder();
// Get version info
String installer = context.getPackageManager().getInstallerPackageName(BuildConfig.APPLICATION_ID);
sb.append(String.format("%s: %s/%s %s/%s%s%s%s\r\n",
context.getString(R.string.app_name),
BuildConfig.APPLICATION_ID,
installer,
BuildConfig.VERSION_NAME,
Helper.hasValidFingerprint(context) ? "1" : "3",
BuildConfig.PLAY_STORE_RELEASE ? "p" : "",
BuildConfig.DEBUG ? "d" : "",
2019-08-13 08:27:17 +00:00
ActivityBilling.isPro(context) ? "+" : ""));
2019-05-12 17:14:34 +00:00
sb.append(String.format("Android: %s (SDK %d)\r\n", Build.VERSION.RELEASE, Build.VERSION.SDK_INT));
sb.append("\r\n");
// Get device info
2019-06-18 16:30:01 +00:00
sb.append(String.format("uid: %s\r\n", android.os.Process.myUid()));
2019-05-12 17:14:34 +00:00
sb.append(String.format("Brand: %s\r\n", Build.BRAND));
sb.append(String.format("Manufacturer: %s\r\n", Build.MANUFACTURER));
sb.append(String.format("Model: %s\r\n", Build.MODEL));
sb.append(String.format("Product: %s\r\n", Build.PRODUCT));
sb.append(String.format("Device: %s\r\n", Build.DEVICE));
sb.append(String.format("Host: %s\r\n", Build.HOST));
sb.append(String.format("Display: %s\r\n", Build.DISPLAY));
sb.append(String.format("Id: %s\r\n", Build.ID));
sb.append("\r\n");
2019-07-22 07:18:03 +00:00
sb.append(String.format("Processors: %d\r\n", Runtime.getRuntime().availableProcessors()));
ActivityManager am = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
sb.append(String.format("Memory class: %d\r\n", am.getMemoryClass()));
2019-10-19 19:53:19 +00:00
sb.append(String.format("Storage space: %s\r\n",
Helper.humanReadableByteCount(Helper.getStorageSpace(), true)));
2019-07-22 07:18:03 +00:00
Runtime rt = Runtime.getRuntime();
long hused = (rt.totalMemory() - rt.freeMemory()) / 1024L;
long hmax = rt.maxMemory() / 1024L;
long nheap = Debug.getNativeHeapAllocatedSize() / 1024L;
sb.append(String.format("Heap usage: %s/%s KiB native: %s KiB\r\n", hused, hmax, nheap));
2019-05-12 17:14:34 +00:00
WindowManager wm = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
Display display = wm.getDefaultDisplay();
Point size = new Point();
display.getSize(size);
float density = context.getResources().getDisplayMetrics().density;
2019-07-15 07:18:03 +00:00
sb.append(String.format("Density %f resolution: %.2f x %.2f dp %b\r\n",
density,
2019-05-12 17:14:34 +00:00
size.x / density, size.y / density,
context.getResources().getConfiguration().isLayoutSizeAtLeast(Configuration.SCREENLAYOUT_SIZE_NORMAL)));
PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
boolean ignoring = true;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M)
ignoring = pm.isIgnoringBatteryOptimizations(BuildConfig.APPLICATION_ID);
sb.append(String.format("Battery optimizations: %b\r\n", !ignoring));
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
UsageStatsManager usm = (UsageStatsManager) context.getSystemService(Context.USAGE_STATS_SERVICE);
int bucket = usm.getAppStandbyBucket();
sb.append(String.format("Standby bucket: %d\r\n", bucket));
}
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
ConnectivityManager cm = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);
boolean saving = (cm.getRestrictBackgroundStatus() == ConnectivityManager.RESTRICT_BACKGROUND_STATUS_ENABLED);
sb.append(String.format("Data saving: %b\r\n", saving));
}
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
boolean reporting = prefs.getBoolean("crash_reports", false);
if (reporting) {
String uuid = prefs.getString("uuid", null);
sb.append(String.format("UUID: %s\r\n", uuid == null ? "-" : uuid));
}
2019-05-12 17:14:34 +00:00
sb.append("\r\n");
sb.append(new Date().toString()).append("\r\n");
sb.append("\r\n");
return sb;
}
private static void attachSettings(Context context, long id, int sequence) throws IOException {
DB db = DB.getInstance(context);
EntityAttachment attachment = new EntityAttachment();
attachment.message = id;
attachment.sequence = sequence;
attachment.name = "settings.txt";
attachment.type = "text/plain";
2019-06-29 06:52:15 +00:00
attachment.disposition = Part.ATTACHMENT;
2019-05-12 17:14:34 +00:00
attachment.size = null;
attachment.progress = 0;
attachment.id = db.attachment().insertAttachment(attachment);
2019-09-28 16:36:07 +00:00
long size = 0;
2019-05-12 17:14:34 +00:00
File file = attachment.getFile(context);
try (OutputStream os = new BufferedOutputStream(new FileOutputStream(file))) {
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
Map<String, ?> settings = prefs.getAll();
for (String key : settings.keySet())
size += write(os, key + "=" + settings.get(key) + "\r\n");
}
2019-09-28 16:36:07 +00:00
db.attachment().setDownloaded(attachment.id, size);
2019-05-12 17:14:34 +00:00
}
private static void attachAccounts(Context context, long id, int sequence) throws IOException {
DB db = DB.getInstance(context);
EntityAttachment attachment = new EntityAttachment();
attachment.message = id;
attachment.sequence = sequence;
attachment.name = "accounts.txt";
attachment.type = "text/plain";
2019-06-29 06:52:15 +00:00
attachment.disposition = Part.ATTACHMENT;
2019-05-12 17:14:34 +00:00
attachment.size = null;
attachment.progress = 0;
attachment.id = db.attachment().insertAttachment(attachment);
2019-09-28 16:36:07 +00:00
long size = 0;
2019-05-12 17:14:34 +00:00
File file = attachment.getFile(context);
try (OutputStream os = new BufferedOutputStream(new FileOutputStream(file))) {
List<EntityAccount> accounts = db.account().getAccounts();
for (EntityAccount account : accounts)
try {
JSONObject jaccount = account.toJSON();
jaccount.remove("user");
jaccount.remove("password");
size += write(os, "==========\r\n");
size += write(os, jaccount.toString(2) + "\r\n");
List<EntityIdentity> identities = db.identity().getIdentities(account.id);
for (EntityIdentity identity : identities)
try {
JSONObject jidentity = identity.toJSON();
jidentity.remove("user");
jidentity.remove("password");
size += write(os, "----------\r\n");
size += write(os, jidentity.toString(2) + "\r\n");
} catch (JSONException ex) {
size += write(os, ex.toString() + "\r\n");
}
} catch (JSONException ex) {
size += write(os, ex.toString() + "\r\n");
}
}
2019-09-28 16:36:07 +00:00
db.attachment().setDownloaded(attachment.id, size);
2019-05-12 17:14:34 +00:00
}
private static void attachNetworkInfo(Context context, long id, int sequence) throws IOException {
DB db = DB.getInstance(context);
EntityAttachment attachment = new EntityAttachment();
attachment.message = id;
attachment.sequence = sequence;
attachment.name = "network.txt";
attachment.type = "text/plain";
2019-06-29 06:52:15 +00:00
attachment.disposition = Part.ATTACHMENT;
2019-05-12 17:14:34 +00:00
attachment.size = null;
attachment.progress = 0;
attachment.id = db.attachment().insertAttachment(attachment);
2019-09-28 16:36:07 +00:00
long size = 0;
2019-05-12 17:14:34 +00:00
File file = attachment.getFile(context);
try (OutputStream os = new BufferedOutputStream(new FileOutputStream(file))) {
ConnectivityManager cm = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo ani = cm.getActiveNetworkInfo();
if (ani != null)
size += write(os, ani.toString() + " metered=" + cm.isActiveNetworkMetered() + "\r\n\r\n");
2019-05-12 17:14:34 +00:00
Network active = null;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M)
active = cm.getActiveNetwork();
for (Network network : cm.getAllNetworks()) {
NetworkCapabilities caps = cm.getNetworkCapabilities(network);
size += write(os, (network.equals(active) ? "active=" : "network=") + network + " capabilities=" + caps + "\r\n\r\n");
}
}
2019-09-28 16:36:07 +00:00
db.attachment().setDownloaded(attachment.id, size);
2019-05-12 17:14:34 +00:00
}
private static void attachLog(Context context, long id, int sequence) throws IOException {
DB db = DB.getInstance(context);
EntityAttachment attachment = new EntityAttachment();
attachment.message = id;
attachment.sequence = sequence;
attachment.name = "log.txt";
attachment.type = "text/plain";
2019-06-29 06:52:15 +00:00
attachment.disposition = Part.ATTACHMENT;
2019-05-12 17:14:34 +00:00
attachment.size = null;
attachment.progress = 0;
attachment.id = db.attachment().insertAttachment(attachment);
2019-09-28 16:36:07 +00:00
long size = 0;
2019-05-12 17:14:34 +00:00
File file = attachment.getFile(context);
try (OutputStream os = new BufferedOutputStream(new FileOutputStream(file))) {
long from = new Date().getTime() - 24 * 3600 * 1000L;
2019-07-15 19:28:25 +00:00
DateFormat TF = Helper.getTimeInstance(context);
2019-05-12 17:14:34 +00:00
for (EntityLog entry : db.log().getLogs(from))
2019-07-15 19:28:25 +00:00
size += write(os, String.format("%s %s\r\n", TF.format(entry.time), entry.data));
2019-05-12 17:14:34 +00:00
}
2019-09-28 16:36:07 +00:00
db.attachment().setDownloaded(attachment.id, size);
2019-05-12 17:14:34 +00:00
}
private static void attachOperations(Context context, long id, int sequence) throws IOException {
DB db = DB.getInstance(context);
EntityAttachment attachment = new EntityAttachment();
attachment.message = id;
attachment.sequence = sequence;
attachment.name = "operations.txt";
attachment.type = "text/plain";
2019-06-29 06:52:15 +00:00
attachment.disposition = Part.ATTACHMENT;
2019-05-12 17:14:34 +00:00
attachment.size = null;
attachment.progress = 0;
attachment.id = db.attachment().insertAttachment(attachment);
2019-09-28 16:36:07 +00:00
long size = 0;
2019-05-12 17:14:34 +00:00
File file = attachment.getFile(context);
try (OutputStream os = new BufferedOutputStream(new FileOutputStream(file))) {
2019-07-15 19:28:25 +00:00
DateFormat TF = Helper.getTimeInstance(context);
2019-05-12 17:14:34 +00:00
for (EntityOperation op : db.operation().getOperations())
size += write(os, String.format("%s %d %s %s %s\r\n",
2019-07-15 19:28:25 +00:00
TF.format(op.created),
2019-05-12 17:14:34 +00:00
op.message == null ? -1 : op.message,
op.name,
op.args,
op.error));
}
2019-09-28 16:36:07 +00:00
db.attachment().setDownloaded(attachment.id, size);
2019-05-12 17:14:34 +00:00
}
private static void attachLogcat(Context context, long id, int sequence) throws IOException {
DB db = DB.getInstance(context);
EntityAttachment attachment = new EntityAttachment();
attachment.message = id;
attachment.sequence = sequence;
attachment.name = "logcat.txt";
attachment.type = "text/plain";
2019-06-29 06:52:15 +00:00
attachment.disposition = Part.ATTACHMENT;
2019-05-12 17:14:34 +00:00
attachment.size = null;
attachment.progress = 0;
attachment.id = db.attachment().insertAttachment(attachment);
Process proc = null;
File file = attachment.getFile(context);
try (OutputStream os = new BufferedOutputStream(new FileOutputStream(file))) {
String[] cmd = new String[]{"logcat",
"-d",
"-v", "threadtime",
//"-t", "1000",
Log.TAG + ":I"};
proc = Runtime.getRuntime().exec(cmd);
long size = 0;
try (BufferedReader br = new BufferedReader(new InputStreamReader(proc.getInputStream()))) {
String line;
while ((line = br.readLine()) != null)
size += write(os, line + "\r\n");
}
db.attachment().setDownloaded(attachment.id, size);
} finally {
if (proc != null)
proc.destroy();
}
}
private static int write(OutputStream os, String text) throws IOException {
byte[] bytes = text.getBytes();
os.write(bytes);
return bytes.length;
}
2019-05-22 13:41:54 +00:00
2019-08-12 11:33:39 +00:00
private static long getFreeMem() {
2019-05-22 13:41:54 +00:00
Runtime rt = Runtime.getRuntime();
long used = (rt.totalMemory() - rt.freeMemory());
long max = rt.maxMemory();
return (max - used);
}
static int getFreeMemMb() {
return (int) (getFreeMem() / 1024L / 1024L);
}
2019-05-12 17:14:34 +00:00
static InternetAddress myAddress() throws UnsupportedEncodingException {
return new InternetAddress("marcel+fairemail@faircode.eu", "FairCode");
}
2019-09-02 12:52:49 +00:00
static boolean isSupportedDevice() {
if ("Amazon".equals(Build.BRAND) && Build.VERSION.SDK_INT < Build.VERSION_CODES.M) {
/*
java.lang.IllegalArgumentException: Comparison method violates its general contract!
java.lang.IllegalArgumentException: Comparison method violates its general contract!
at java.util.TimSort.mergeHi(TimSort.java:864)
at java.util.TimSort.mergeAt(TimSort.java:481)
at java.util.TimSort.mergeCollapse(TimSort.java:406)
at java.util.TimSort.sort(TimSort.java:210)
at java.util.TimSort.sort(TimSort.java:169)
at java.util.Arrays.sort(Arrays.java:2010)
at java.util.Collections.sort(Collections.java:1883)
at android.view.ViewGroup$ChildListForAccessibility.init(ViewGroup.java:7181)
at android.view.ViewGroup$ChildListForAccessibility.obtain(ViewGroup.java:7138)
at android.view.ViewGroup.dispatchPopulateAccessibilityEventInternal(ViewGroup.java:2734)
at android.view.View.dispatchPopulateAccessibilityEvent(View.java:5617)
at android.view.View.sendAccessibilityEventUncheckedInternal(View.java:5582)
at android.view.View.sendAccessibilityEventUnchecked(View.java:5566)
at android.view.View.sendAccessibilityEventInternal(View.java:5543)
at android.view.View.sendAccessibilityEvent(View.java:5512)
at android.view.View.onFocusChanged(View.java:5449)
at android.view.View.handleFocusGainInternal(View.java:5229)
at android.view.ViewGroup.handleFocusGainInternal(ViewGroup.java:651)
at android.view.View.requestFocusNoSearch(View.java:7950)
at android.view.View.requestFocus(View.java:7929)
at android.view.ViewGroup.requestFocus(ViewGroup.java:2612)
at android.view.ViewGroup.onRequestFocusInDescendants(ViewGroup.java:2657)
at android.view.ViewGroup.requestFocus(ViewGroup.java:2613)
at android.view.View.requestFocus(View.java:7896)
at android.view.View.requestFocus(View.java:7875)
at androidx.recyclerview.widget.RecyclerView.recoverFocusFromState(SourceFile:3788)
at androidx.recyclerview.widget.RecyclerView.dispatchLayoutStep3(SourceFile:4023)
at androidx.recyclerview.widget.RecyclerView.dispatchLayout(SourceFile:3652)
at androidx.recyclerview.widget.RecyclerView.consumePendingUpdateOperations(SourceFile:1877)
at androidx.recyclerview.widget.RecyclerView$w.run(SourceFile:5044)
at android.view.Choreographer$CallbackRecord.run(Choreographer.java:781)
at android.view.Choreographer.doCallbacks(Choreographer.java:592)
at android.view.Choreographer.doFrame(Choreographer.java:559)
at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:767)
*/
return false;
}
return true;
}
static boolean isXiaomi() {
return "Xiaomi".equalsIgnoreCase(Build.MANUFACTURER);
}
2018-12-24 12:27:45 +00:00
}