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/>.
|
|
|
|
|
|
2020-01-05 17:32:53 +00:00
|
|
|
|
Copyright 2018-2020 by Marcel Bokhorst (M66B)
|
2018-12-31 08:04:33 +00:00
|
|
|
|
*/
|
|
|
|
|
|
2019-07-22 07:18:03 +00:00
|
|
|
|
import android.app.ActivityManager;
|
2019-12-06 07:50:46 +00:00
|
|
|
|
import android.app.Dialog;
|
2019-05-12 17:14:34 +00:00
|
|
|
|
import android.app.usage.UsageStatsManager;
|
|
|
|
|
import android.content.Context;
|
2019-12-06 07:50:46 +00:00
|
|
|
|
import android.content.DialogInterface;
|
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;
|
2019-10-22 06:55:27 +00:00
|
|
|
|
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-12-06 07:50:46 +00:00
|
|
|
|
import android.widget.Toast;
|
2019-05-12 17:14:34 +00:00
|
|
|
|
|
2019-08-12 11:17:55 +00:00
|
|
|
|
import androidx.annotation.NonNull;
|
2019-12-06 07:50:46 +00:00
|
|
|
|
import androidx.annotation.Nullable;
|
|
|
|
|
import androidx.appcompat.app.AlertDialog;
|
|
|
|
|
import androidx.fragment.app.FragmentManager;
|
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;
|
2019-10-28 17:46:53 +00:00
|
|
|
|
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;
|
2019-09-24 15:05:33 +00:00
|
|
|
|
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-12-06 07:50:46 +00:00
|
|
|
|
import com.sun.mail.iap.BadCommandException;
|
|
|
|
|
import com.sun.mail.iap.ConnectionException;
|
2019-08-12 11:17:55 +00:00
|
|
|
|
import com.sun.mail.iap.ProtocolException;
|
2019-12-06 07:50:46 +00:00
|
|
|
|
import com.sun.mail.util.FolderClosedIOException;
|
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-12-06 08:00:31 +00:00
|
|
|
|
import java.net.SocketException;
|
2020-02-07 08:58:33 +00:00
|
|
|
|
import java.security.cert.CertPathValidatorException;
|
2019-05-12 17:14:34 +00:00
|
|
|
|
import java.text.DateFormat;
|
2019-08-12 11:17:55 +00:00
|
|
|
|
import java.util.ArrayList;
|
2019-09-24 15:05:33 +00:00
|
|
|
|
import java.util.Arrays;
|
2020-04-03 06:59:22 +00:00
|
|
|
|
import java.util.Collections;
|
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-12-06 08:00:31 +00:00
|
|
|
|
import javax.mail.AuthenticationFailedException;
|
2019-12-06 07:50:46 +00:00
|
|
|
|
import javax.mail.FolderClosedException;
|
|
|
|
|
import javax.mail.MessageRemovedException;
|
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-12-09 07:37:08 +00:00
|
|
|
|
import javax.mail.StoreClosedException;
|
2019-05-12 17:14:34 +00:00
|
|
|
|
import javax.mail.internet.InternetAddress;
|
2020-02-07 08:58:33 +00:00
|
|
|
|
import javax.net.ssl.SSLHandshakeException;
|
2020-02-02 14:24:24 +00:00
|
|
|
|
import javax.net.ssl.SSLPeerUnverifiedException;
|
2019-05-12 17:14:34 +00:00
|
|
|
|
|
2018-12-24 12:27:45 +00:00
|
|
|
|
public class Log {
|
2020-04-05 12:10:42 +00:00
|
|
|
|
private static boolean debug = false;
|
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
|
|
|
|
|
2020-04-05 12:10:42 +00:00
|
|
|
|
public static void setDebug(boolean value) {
|
|
|
|
|
debug = value;
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-11 13:49:14 +00:00
|
|
|
|
public static int d(String msg) {
|
2020-04-05 12:10:42 +00:00
|
|
|
|
if (debug)
|
2019-08-11 13:49:14 +00:00
|
|
|
|
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) {
|
2019-12-14 16:01:27 +00:00
|
|
|
|
if (BuildConfig.BETA_RELEASE)
|
|
|
|
|
try {
|
|
|
|
|
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);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
} catch (Throwable ex) {
|
|
|
|
|
ex.printStackTrace();
|
|
|
|
|
}
|
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)
|
2019-12-14 16:01:27 +00:00
|
|
|
|
try {
|
|
|
|
|
Bugsnag.notify(ex, Severity.INFO);
|
|
|
|
|
} catch (Throwable ex1) {
|
|
|
|
|
ex1.printStackTrace();
|
|
|
|
|
}
|
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)
|
2019-12-14 16:01:27 +00:00
|
|
|
|
try {
|
|
|
|
|
Bugsnag.notify(ex, Severity.WARNING);
|
|
|
|
|
} catch (Throwable ex1) {
|
|
|
|
|
ex1.printStackTrace();
|
|
|
|
|
}
|
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)
|
2019-12-14 16:01:27 +00:00
|
|
|
|
try {
|
2020-03-26 07:55:02 +00:00
|
|
|
|
Bugsnag.notify(ex, Severity.INFO);
|
2019-12-14 16:01:27 +00:00
|
|
|
|
} catch (Throwable ex1) {
|
|
|
|
|
ex1.printStackTrace();
|
|
|
|
|
}
|
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)
|
2019-12-14 16:01:27 +00:00
|
|
|
|
try {
|
2020-03-26 07:55:02 +00:00
|
|
|
|
Bugsnag.notify(ex, Severity.WARNING);
|
2019-12-14 16:01:27 +00:00
|
|
|
|
} catch (Throwable ex1) {
|
|
|
|
|
ex1.printStackTrace();
|
|
|
|
|
}
|
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) {
|
2019-12-14 16:01:27 +00:00
|
|
|
|
try {
|
|
|
|
|
if (enabled)
|
|
|
|
|
Bugsnag.startSession();
|
|
|
|
|
else
|
|
|
|
|
Bugsnag.stopSession();
|
|
|
|
|
} catch (Throwable ex) {
|
|
|
|
|
ex.printStackTrace();
|
|
|
|
|
}
|
2019-08-12 11:17:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-08-12 11:07:14 +00:00
|
|
|
|
static void breadcrumb(String name, Map<String, String> crumb) {
|
2019-12-14 16:01:27 +00:00
|
|
|
|
try {
|
|
|
|
|
Bugsnag.leaveBreadcrumb(name, BreadcrumbType.LOG, crumb);
|
|
|
|
|
} catch (Throwable ex) {
|
|
|
|
|
ex.printStackTrace();
|
|
|
|
|
}
|
2019-08-12 11:17:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-04-05 12:10:42 +00:00
|
|
|
|
static void setup(Context context) {
|
|
|
|
|
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
|
|
|
|
|
debug = prefs.getBoolean("debug", false);
|
|
|
|
|
|
|
|
|
|
setupBugsnag(context);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static void setupBugsnag(Context context) {
|
2019-08-12 11:17:55 +00:00
|
|
|
|
// 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 {
|
2020-03-28 08:39:02 +00:00
|
|
|
|
String type;
|
|
|
|
|
if (Helper.hasValidFingerprint(context)) {
|
2019-08-12 11:17:55 +00:00
|
|
|
|
if (BuildConfig.PLAY_STORE_RELEASE)
|
|
|
|
|
type = "play";
|
|
|
|
|
else
|
|
|
|
|
type = "full";
|
2020-03-28 08:39:02 +00:00
|
|
|
|
} else {
|
|
|
|
|
if (BuildConfig.APPLICATION_ID.startsWith("eu.faircode.email"))
|
|
|
|
|
type = "other";
|
|
|
|
|
else
|
|
|
|
|
type = "clone";
|
|
|
|
|
}
|
2019-08-12 11:17:55 +00:00
|
|
|
|
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");
|
|
|
|
|
|
2020-01-31 07:58:55 +00:00
|
|
|
|
ignore.add("android.accounts.AuthenticatorException");
|
2019-08-12 11:17:55 +00:00
|
|
|
|
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.internet.AddressException");
|
2020-02-09 13:17:43 +00:00
|
|
|
|
ignore.add("javax.mail.internet.ParseException");
|
2019-08-12 11:17:55 +00:00
|
|
|
|
ignore.add("javax.mail.MessageRemovedException");
|
2020-01-31 07:58:55 +00:00
|
|
|
|
ignore.add("javax.mail.FolderNotFoundException");
|
2019-08-12 11:17:55 +00:00
|
|
|
|
ignore.add("javax.mail.ReadOnlyFolderException");
|
2020-01-31 07:58:55 +00:00
|
|
|
|
ignore.add("javax.mail.FolderClosedException");
|
2020-02-07 08:58:33 +00:00
|
|
|
|
ignore.add("com.sun.mail.util.FolderClosedIOException");
|
2019-08-12 11:17:55 +00:00
|
|
|
|
ignore.add("javax.mail.StoreClosedException");
|
|
|
|
|
|
|
|
|
|
ignore.add("org.xmlpull.v1.XmlPullParserException");
|
|
|
|
|
|
|
|
|
|
config.setIgnoreClasses(ignore.toArray(new String[0]));
|
|
|
|
|
|
|
|
|
|
final SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
|
|
|
|
|
|
2020-02-01 14:52:53 +00:00
|
|
|
|
String no_internet = context.getString(R.string.title_no_internet);
|
|
|
|
|
|
2019-10-28 17:46:53 +00:00
|
|
|
|
config.beforeSend(new BeforeSend() {
|
2019-08-12 11:17:55 +00:00
|
|
|
|
@Override
|
2019-10-28 17:46:53 +00:00
|
|
|
|
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;
|
|
|
|
|
|
2019-10-28 17:46:53 +00:00
|
|
|
|
Throwable ex = report.getError().getException();
|
2020-03-25 20:29:30 +00:00
|
|
|
|
boolean should = shouldNotify(ex);
|
|
|
|
|
return should;
|
2019-10-28 17:50:39 +00:00
|
|
|
|
}
|
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 &&
|
2020-02-01 14:52:53 +00:00
|
|
|
|
(no_internet.equals(ex.getMessage()) ||
|
|
|
|
|
"Not connected".equals(ex.getMessage()) ||
|
|
|
|
|
"This operation is not allowed on a closed folder".equals(ex.getMessage())))
|
2019-08-12 11:17:55 +00:00
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
if (ex instanceof FileNotFoundException &&
|
|
|
|
|
ex.getMessage() != null &&
|
|
|
|
|
(ex.getMessage().startsWith("Download image failed") ||
|
2020-02-04 08:40:49 +00:00
|
|
|
|
ex.getMessage().startsWith("http://") ||
|
2020-02-09 13:23:41 +00:00
|
|
|
|
ex.getMessage().startsWith("https://") ||
|
|
|
|
|
ex.getMessage().startsWith("content://")))
|
2019-08-12 11:17:55 +00:00
|
|
|
|
return false;
|
|
|
|
|
|
2020-02-12 08:16:24 +00:00
|
|
|
|
if (ex instanceof IOException &&
|
|
|
|
|
ex.getCause() instanceof MessageRemovedException)
|
|
|
|
|
return false;
|
|
|
|
|
|
2019-11-02 16:15:41 +00:00
|
|
|
|
if (ex instanceof IOException &&
|
2020-02-04 11:42:25 +00:00
|
|
|
|
ex.getMessage() != null &&
|
|
|
|
|
(ex.getMessage().startsWith("HTTP status=") ||
|
|
|
|
|
"NetworkError".equals(ex.getMessage()) || // account manager
|
2020-02-09 13:23:41 +00:00
|
|
|
|
"Resetting to invalid mark".equals(ex.getMessage()) ||
|
|
|
|
|
"Mark has been invalidated.".equals(ex.getMessage())))
|
2019-11-02 16:15:41 +00:00
|
|
|
|
return false;
|
|
|
|
|
|
2020-02-02 14:24:24 +00:00
|
|
|
|
if (ex instanceof SSLPeerUnverifiedException ||
|
|
|
|
|
ex instanceof EmailService.UntrustedException)
|
|
|
|
|
return false;
|
|
|
|
|
|
2020-02-07 08:58:33 +00:00
|
|
|
|
if (ex instanceof SSLHandshakeException &&
|
|
|
|
|
ex.getCause() instanceof CertPathValidatorException)
|
|
|
|
|
return false; // checkUpdate!
|
|
|
|
|
|
2020-02-09 13:23:41 +00:00
|
|
|
|
if (ex instanceof RuntimeException &&
|
|
|
|
|
"Illegal meta data value: the child service doesn't exist".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:46:44 +00:00
|
|
|
|
|
2019-10-28 17:50:39 +00:00
|
|
|
|
return (count <= MAX_CRASH_REPORTS);
|
2019-10-28 17:46:53 +00:00
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
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);
|
2019-10-28 17:46:53 +00:00
|
|
|
|
|
|
|
|
|
Bugsnag.beforeNotify(new BeforeNotify() {
|
|
|
|
|
@Override
|
|
|
|
|
public boolean run(@NonNull Error error) {
|
2019-10-28 17:46:44 +00:00
|
|
|
|
error.addToTab("extra", "installer", installer == null ? "-" : installer);
|
2020-03-28 08:25:46 +00:00
|
|
|
|
error.addToTab("extra", "installed", new Date(Helper.getInstallTime(context)));
|
2019-10-28 17:46:44 +00:00
|
|
|
|
error.addToTab("extra", "fingerprint", fingerprint);
|
2019-11-09 13:13:18 +00:00
|
|
|
|
error.addToTab("extra", "thread", Thread.currentThread().getName() + ":" + Thread.currentThread().getId());
|
2019-10-28 17:46:44 +00:00
|
|
|
|
error.addToTab("extra", "free", Log.getFreeMemMb());
|
2019-11-08 07:58:19 +00:00
|
|
|
|
error.addToTab("extra", "optimizing", (ignoringOptimizations != null && !ignoringOptimizations));
|
2019-10-28 17:46:44 +00:00
|
|
|
|
|
|
|
|
|
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)
|
2019-12-09 21:03:03 +00:00
|
|
|
|
elements[i] = element.toString() + " (0x" + Long.toHexString((Long) element) + ")";
|
2019-10-29 06:57:39 +00:00
|
|
|
|
else
|
|
|
|
|
elements[i] = (element == null ? null : element.toString());
|
|
|
|
|
}
|
|
|
|
|
value = TextUtils.join(",", elements);
|
2019-11-30 08:22:16 +00:00
|
|
|
|
} else
|
|
|
|
|
value = "[" + length + "]";
|
2019-10-29 06:57:39 +00:00
|
|
|
|
} else if (v instanceof Long)
|
2019-12-09 21:03:03 +00:00
|
|
|
|
value = v.toString() + " (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;
|
|
|
|
|
|
2020-03-09 12:07:22 +00:00
|
|
|
|
if (ex instanceof UnsatisfiedLinkError ||
|
|
|
|
|
ex.getCause() instanceof UnsatisfiedLinkError)
|
2020-03-25 09:05:51 +00:00
|
|
|
|
/*
|
|
|
|
|
java.lang.UnsatisfiedLinkError: dlopen failed: couldn't map "/mnt/asec/eu.faircode.email-1/base.apk!/lib/arm64-v8a/libsqlite3x.so" segment 0: Permission denied
|
|
|
|
|
at java.lang.Runtime.loadLibrary0(Runtime.java:1016)
|
|
|
|
|
at java.lang.System.loadLibrary(System.java:1657)
|
|
|
|
|
at io.requery.android.database.sqlite.SQLiteDatabase.<clinit>(SourceFile:91)
|
|
|
|
|
*/
|
2020-03-08 17:12:21 +00:00
|
|
|
|
return false;
|
|
|
|
|
|
2020-04-01 12:38:49 +00:00
|
|
|
|
if (ex instanceof InternalError &&
|
|
|
|
|
"Thread starting during runtime shutdown".equals(ex.getMessage()))
|
|
|
|
|
/*
|
|
|
|
|
java.lang.InternalError: Thread starting during runtime shutdown
|
|
|
|
|
java.lang.InternalError: Thread starting during runtime shutdown
|
|
|
|
|
at java.lang.Thread.nativeCreate(Native Method)
|
|
|
|
|
at java.lang.Thread.start(Thread.java:1063)
|
|
|
|
|
at java.util.concurrent.ThreadPoolExecutor.addWorker(ThreadPoolExecutor.java:921)
|
|
|
|
|
at java.util.concurrent.ThreadPoolExecutor.processWorkerExit(ThreadPoolExecutor.java:989)
|
|
|
|
|
at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1131)
|
|
|
|
|
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:588)
|
|
|
|
|
at java.lang.Thread.run(Thread.java:818)
|
|
|
|
|
*/
|
|
|
|
|
return false;
|
|
|
|
|
|
2020-01-23 11:46:11 +00:00
|
|
|
|
if ("android.app.RemoteServiceException".equals(ex.getClass().getName()))
|
2020-03-25 09:05:51 +00:00
|
|
|
|
/*
|
|
|
|
|
android.app.RemoteServiceException: Bad notification for startForeground: java.util.ConcurrentModificationException
|
|
|
|
|
android.app.RemoteServiceException: Bad notification for startForeground: java.util.ConcurrentModificationException
|
|
|
|
|
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2204)
|
|
|
|
|
*/
|
2020-01-23 11:46:11 +00:00
|
|
|
|
return false;
|
|
|
|
|
|
2019-08-12 11:17:55 +00:00
|
|
|
|
if (ex instanceof NoSuchMethodError)
|
2020-03-25 09:05:51 +00:00
|
|
|
|
/*
|
|
|
|
|
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)
|
|
|
|
|
*/
|
2019-08-12 11:17:55 +00:00
|
|
|
|
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;
|
|
|
|
|
|
2020-03-19 16:54:17 +00:00
|
|
|
|
if (ex instanceof IllegalArgumentException &&
|
|
|
|
|
ex.getCause() instanceof RemoteException)
|
|
|
|
|
/*
|
|
|
|
|
java.lang.IllegalArgumentException
|
|
|
|
|
at android.os.Parcel.createException(Parcel.java:1954)
|
|
|
|
|
at android.os.Parcel.readException(Parcel.java:1918)
|
|
|
|
|
at android.os.Parcel.readException(Parcel.java:1868)
|
|
|
|
|
at android.view.IWindowSession$Stub$Proxy.addToDisplay(IWindowSession.java:826)
|
|
|
|
|
at android.view.ViewRootImpl.setView(ViewRootImpl.java:758)
|
|
|
|
|
at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:356)
|
|
|
|
|
at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:93)
|
|
|
|
|
at android.app.ActivityThread.handleResumeActivity(ActivityThread.java:3906)
|
|
|
|
|
at android.app.servertransaction.ResumeActivityItem.execute(ResumeActivityItem.java:51)
|
|
|
|
|
at android.app.servertransaction.TransactionExecutor.executeLifecycleState(TransactionExecutor.java:145)
|
|
|
|
|
at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:70)
|
|
|
|
|
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1816)
|
|
|
|
|
at android.os.Handler.dispatchMessage(Handler.java:106)
|
|
|
|
|
at android.os.Looper.loop(Looper.java:193)
|
|
|
|
|
at android.app.ActivityThread.main(ActivityThread.java:6718)
|
|
|
|
|
at java.lang.reflect.Method.invoke(Native Method)
|
|
|
|
|
at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:491)
|
|
|
|
|
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:858)
|
|
|
|
|
Caused by: android.os.RemoteException: Remote stack trace:
|
|
|
|
|
at android.view.SurfaceControl.nativeCreate(Native Method)
|
|
|
|
|
at android.view.SurfaceControl.<init>(SurfaceControl.java:630)
|
|
|
|
|
at android.view.SurfaceControl.<init>(SurfaceControl.java:60)
|
|
|
|
|
at android.view.SurfaceControl$Builder.build(SurfaceControl.java:386)
|
|
|
|
|
at com.android.server.wm.WindowContainer.onParentSet(WindowContainer.java:184)
|
|
|
|
|
*/
|
|
|
|
|
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-12-05 12:25:59 +00:00
|
|
|
|
if (ex instanceof RuntimeException &&
|
|
|
|
|
ex.getMessage() != null &&
|
2020-03-06 07:12:33 +00:00
|
|
|
|
(ex.getMessage().startsWith("Could not get application info") ||
|
2020-03-25 09:05:51 +00:00
|
|
|
|
ex.getMessage().startsWith("Unable to create service") ||
|
2020-03-25 09:47:08 +00:00
|
|
|
|
ex.getMessage().startsWith("Failure delivering result")))
|
2019-12-05 12:25:59 +00:00
|
|
|
|
return false;
|
2020-03-25 09:05:51 +00:00
|
|
|
|
/*
|
2019-12-05 12:25:59 +00:00
|
|
|
|
java.lang.RuntimeException: Could not get application info.
|
|
|
|
|
java.lang.RuntimeException: Could not get application info.
|
|
|
|
|
at CH0.a(PG:11)
|
|
|
|
|
at org.chromium.content.browser.ChildProcessLauncherHelperImpl.a(PG:34)
|
|
|
|
|
at Fn2.run(PG:5)
|
|
|
|
|
at android.os.Handler.handleCallback(Handler.java:874)
|
|
|
|
|
at android.os.Handler.dispatchMessage(Handler.java:100)
|
|
|
|
|
at android.os.Looper.loop(Looper.java:198)
|
|
|
|
|
at android.os.HandlerThread.run(HandlerThread.java:65)
|
2020-03-06 07:12:33 +00:00
|
|
|
|
|
|
|
|
|
java.lang.RuntimeException: Unable to create service eu.faircode.email.ServiceSynchronize: java.lang.NullPointerException: Attempt to invoke interface method 'java.util.List android.os.IUserManager.getProfiles(int, boolean)' on a null object reference
|
|
|
|
|
java.lang.RuntimeException: Unable to create service eu.faircode.email.ServiceSynchronize: java.lang.NullPointerException: Attempt to invoke interface method 'java.util.List android.os.IUserManager.getProfiles(int, boolean)' on a null object reference
|
|
|
|
|
at android.app.ActivityThread.handleCreateService(ActivityThread.java:2739)
|
2020-03-25 09:05:51 +00:00
|
|
|
|
|
|
|
|
|
java.lang.RuntimeException: Failure delivering result ResultInfo{who=@android:autoFillAuth:, request=2162688, result=-1, data=Intent { (has extras) }} to activity {eu.faircode.email/eu.faircode.email.ActivitySetup}: java.lang.NullPointerException: Attempt to invoke interface method 'java.lang.Object java.util.List.get(int)' on a null object reference
|
|
|
|
|
at android.app.ActivityThread.deliverResults(ActivityThread.java:4469)
|
|
|
|
|
at android.app.ActivityThread.handleSendResult(ActivityThread.java:4511)
|
|
|
|
|
at android.app.servertransaction.ActivityResultItem.execute(ActivityResultItem.java:49)
|
|
|
|
|
at android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:108)
|
|
|
|
|
at android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:68)
|
|
|
|
|
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1821)
|
|
|
|
|
at android.os.Handler.dispatchMessage(Handler.java:106)
|
|
|
|
|
at android.os.Looper.loop(Looper.java:193)
|
|
|
|
|
at android.app.ActivityThread.main(ActivityThread.java:6874)
|
|
|
|
|
at java.lang.reflect.Method.invoke(Native Method)
|
|
|
|
|
at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:493)
|
|
|
|
|
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:861)
|
|
|
|
|
Caused by: java.lang.NullPointerException: Attempt to invoke interface method 'java.lang.Object java.util.List.get(int)' on a null object reference
|
|
|
|
|
at android.os.Parcel.createException(Parcel.java:1956)
|
|
|
|
|
at android.os.Parcel.readException(Parcel.java:1918)
|
|
|
|
|
at android.os.Parcel.readException(Parcel.java:1868)
|
|
|
|
|
at android.view.autofill.IAutoFillManager$Stub$Proxy.setAuthenticationResult(IAutoFillManager.java:729)
|
|
|
|
|
at android.view.autofill.AutofillManager.onAuthenticationResult(AutofillManager.java:1474)
|
|
|
|
|
at android.app.Activity.dispatchActivityResult(Activity.java:7497)
|
|
|
|
|
at android.app.ActivityThread.deliverResults(ActivityThread.java:4462)
|
|
|
|
|
... 11 more
|
|
|
|
|
Caused by: android.os.RemoteException: Remote stack trace:
|
|
|
|
|
at com.android.server.autofill.Session.setAuthenticationResultLocked(Session.java:1005)
|
|
|
|
|
at com.android.server.autofill.AutofillManagerServiceImpl.setAuthenticationResultLocked(AutofillManagerServiceImpl.java:325)
|
|
|
|
|
at com.android.server.autofill.AutofillManagerService$AutoFillManagerServiceStub.setAuthenticationResult(AutofillManagerService.java:863)
|
|
|
|
|
at android.view.autofill.IAutoFillManager$Stub.onTransact(IAutoFillManager.java:289)
|
|
|
|
|
at android.os.Binder.execTransact(Binder.java:731)
|
|
|
|
|
*/
|
2019-12-05 12:25:59 +00:00
|
|
|
|
|
2020-03-01 19:43:27 +00:00
|
|
|
|
if (ex instanceof RuntimeException &&
|
|
|
|
|
"InputChannel is not initialized.".equals(ex.getMessage()))
|
|
|
|
|
return false;
|
2020-03-25 09:05:51 +00:00
|
|
|
|
/*
|
2020-03-01 19:43:27 +00:00
|
|
|
|
java.lang.RuntimeException: InputChannel is not initialized.
|
|
|
|
|
java.lang.RuntimeException: InputChannel is not initialized.
|
|
|
|
|
at android.view.InputEventReceiver.nativeInit(Native Method)
|
|
|
|
|
at android.view.InputEventReceiver.<init>(InputEventReceiver.java:72)
|
|
|
|
|
at android.view.ViewRootImpl$WindowInputEventReceiver.<init>(ViewRootImpl.java:7612)
|
|
|
|
|
at android.view.ViewRootImpl.setView(ViewRootImpl.java:957)
|
|
|
|
|
at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:387)
|
|
|
|
|
at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:96)
|
|
|
|
|
at android.widget.Toast$TN.handleShow(Toast.java:514)
|
|
|
|
|
at android.widget.Toast$TN$1.handleMessage(Toast.java:417)
|
|
|
|
|
at android.os.Handler.dispatchMessage(Handler.java:107)
|
|
|
|
|
at android.os.Looper.loop(Looper.java:214)
|
|
|
|
|
at android.app.ActivityThread.main(ActivityThread.java:7397)
|
2020-03-25 09:05:51 +00:00
|
|
|
|
*/
|
2020-03-01 19:43:27 +00:00
|
|
|
|
|
2020-03-03 11:00:53 +00:00
|
|
|
|
if (ex instanceof RuntimeException &&
|
|
|
|
|
"Failure from system".equals(ex.getMessage()))
|
|
|
|
|
/*
|
|
|
|
|
java.lang.RuntimeException: Failure from system
|
|
|
|
|
at android.app.ContextImpl.bindServiceCommon(ContextImpl.java:1327)
|
|
|
|
|
at android.app.ContextImpl.bindService(ContextImpl.java:1286)
|
|
|
|
|
at android.content.ContextWrapper.bindService(ContextWrapper.java:604)
|
|
|
|
|
at android.content.ContextWrapper.bindService(ContextWrapper.java:604)
|
|
|
|
|
at hq.run(PG:15)
|
|
|
|
|
at java.lang.Thread.run(Thread.java:818)
|
|
|
|
|
Caused by: android.os.DeadObjectException
|
|
|
|
|
at android.os.BinderProxy.transactNative(Native Method)
|
|
|
|
|
at android.os.BinderProxy.transact(Binder.java:503)
|
|
|
|
|
at android.app.ActivityManagerProxy.bindService(ActivityManagerNative.java:3783)
|
|
|
|
|
at android.app.ContextImpl.bindServiceCommon(ContextImpl.java:1317)
|
|
|
|
|
*/
|
|
|
|
|
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") ||
|
2019-09-23 06:31:04 +00:00
|
|
|
|
ex.getMessage().startsWith("Unable to create layer") ||
|
2019-11-10 08:26:39 +00:00
|
|
|
|
ex.getMessage().startsWith("Illegal meta data value") ||
|
2019-11-17 11:49:54 +00:00
|
|
|
|
ex.getMessage().startsWith("Context.startForegroundService") ||
|
|
|
|
|
ex.getMessage().startsWith("PARAGRAPH span must start at paragraph boundary")))
|
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-12-06 07:50:46 +00:00
|
|
|
|
static String formatThrowable(Throwable ex) {
|
|
|
|
|
return formatThrowable(ex, true);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static String formatThrowable(Throwable ex, boolean santize) {
|
|
|
|
|
return formatThrowable(ex, " ", santize);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static String formatThrowable(Throwable ex, String separator, boolean sanitize) {
|
|
|
|
|
if (sanitize) {
|
|
|
|
|
if (ex instanceof MessageRemovedException)
|
|
|
|
|
return null;
|
|
|
|
|
|
2019-12-06 08:00:31 +00:00
|
|
|
|
if (ex instanceof AuthenticationFailedException &&
|
|
|
|
|
ex.getCause() instanceof SocketException)
|
|
|
|
|
return null;
|
|
|
|
|
|
2019-12-06 08:09:14 +00:00
|
|
|
|
if (ex instanceof MessagingException &&
|
|
|
|
|
("connection failure".equals(ex.getMessage()) ||
|
|
|
|
|
"failed to create new store connection".equals(ex.getMessage())))
|
|
|
|
|
return null;
|
|
|
|
|
|
2019-12-06 07:50:46 +00:00
|
|
|
|
if (ex instanceof MessagingException &&
|
|
|
|
|
ex.getCause() instanceof ConnectionException &&
|
|
|
|
|
ex.getCause().getMessage() != null &&
|
|
|
|
|
(ex.getCause().getMessage().contains("Read error") ||
|
2019-12-06 08:02:11 +00:00
|
|
|
|
ex.getCause().getMessage().contains("Write error") ||
|
2020-01-09 09:49:53 +00:00
|
|
|
|
ex.getCause().getMessage().contains("Unexpected end of ZLIB input stream") ||
|
2019-12-06 08:02:11 +00:00
|
|
|
|
ex.getCause().getMessage().contains("Socket is closed")))
|
2019-12-06 07:50:46 +00:00
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
// javax.mail.MessagingException: AU3 BAD User is authenticated but not connected.;
|
|
|
|
|
// nested exception is:
|
|
|
|
|
// com.sun.mail.iap.BadCommandException: AU3 BAD User is authenticated but not connected.
|
|
|
|
|
// javax.mail.MessagingException: AU3 BAD User is authenticated but not connected.;
|
|
|
|
|
// nested exception is:
|
|
|
|
|
// com.sun.mail.iap.BadCommandException: AU3 BAD User is authenticated but not connected.
|
|
|
|
|
// at com.sun.mail.imap.IMAPFolder.logoutAndThrow(SourceFile:1156)
|
|
|
|
|
// at com.sun.mail.imap.IMAPFolder.open(SourceFile:1063)
|
|
|
|
|
// at com.sun.mail.imap.IMAPFolder.open(SourceFile:977)
|
|
|
|
|
// at eu.faircode.email.ServiceSynchronize.monitorAccount(SourceFile:890)
|
|
|
|
|
// at eu.faircode.email.ServiceSynchronize.access$1500(SourceFile:85)
|
|
|
|
|
// at eu.faircode.email.ServiceSynchronize$7$1.run(SourceFile:627)
|
|
|
|
|
// at java.lang.Thread.run(Thread.java:764)
|
|
|
|
|
// Caused by: com.sun.mail.iap.BadCommandException: AU3 BAD User is authenticated but not connected.
|
|
|
|
|
// at com.sun.mail.iap.Protocol.handleResult(SourceFile:415)
|
|
|
|
|
// at com.sun.mail.imap.protocol.IMAPProtocol.select(SourceFile:1230)
|
|
|
|
|
// at com.sun.mail.imap.IMAPFolder.open(SourceFile:1034)
|
|
|
|
|
|
|
|
|
|
if (ex instanceof MessagingException &&
|
|
|
|
|
ex.getCause() instanceof BadCommandException &&
|
|
|
|
|
ex.getCause().getMessage() != null &&
|
|
|
|
|
ex.getCause().getMessage().contains("User is authenticated but not connected"))
|
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
if (ex instanceof IOException &&
|
|
|
|
|
ex.getCause() instanceof MessageRemovedException)
|
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
if (ex instanceof ConnectionException)
|
|
|
|
|
return null;
|
|
|
|
|
|
2019-12-09 07:37:08 +00:00
|
|
|
|
if (ex instanceof StoreClosedException ||
|
|
|
|
|
ex instanceof FolderClosedException || ex instanceof FolderClosedIOException)
|
2019-12-06 07:50:46 +00:00
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
if (ex instanceof IllegalStateException &&
|
|
|
|
|
("Not connected".equals(ex.getMessage()) ||
|
|
|
|
|
"This operation is not allowed on a closed folder".equals(ex.getMessage())))
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
|
if (BuildConfig.DEBUG)
|
|
|
|
|
sb.append(ex.toString());
|
|
|
|
|
else
|
|
|
|
|
sb.append(ex.getMessage() == null ? ex.getClass().getName() : ex.getMessage());
|
|
|
|
|
|
|
|
|
|
Throwable cause = ex.getCause();
|
|
|
|
|
while (cause != null) {
|
|
|
|
|
if (BuildConfig.DEBUG)
|
|
|
|
|
sb.append(separator).append(cause.toString());
|
|
|
|
|
else
|
|
|
|
|
sb.append(separator).append(cause.getMessage() == null ? cause.getClass().getName() : cause.getMessage());
|
|
|
|
|
cause = cause.getCause();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return sb.toString();
|
|
|
|
|
}
|
|
|
|
|
|
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);
|
2020-02-20 09:35:01 +00:00
|
|
|
|
|
|
|
|
|
File file = draft.getFile(context);
|
|
|
|
|
Helper.writeText(file, body);
|
2019-05-12 17:14:34 +00:00
|
|
|
|
db.message().setMessageContent(draft.id,
|
|
|
|
|
true,
|
2020-03-26 14:25:44 +00:00
|
|
|
|
HtmlHelper.getLanguage(context, body),
|
2019-05-12 17:14:34 +00:00
|
|
|
|
false,
|
2020-03-26 14:25:44 +00:00
|
|
|
|
HtmlHelper.getPreview(body),
|
2019-05-12 17:14:34 +00:00
|
|
|
|
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-06-19 10:23:17 +00:00
|
|
|
|
|
2019-12-09 18:44:27 +00:00
|
|
|
|
ServiceSynchronize.eval(context, "debuginfo");
|
2019-12-07 19:32:58 +00:00
|
|
|
|
|
2019-05-12 17:14:34 +00:00
|
|
|
|
return draft;
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-06 07:50:46 +00:00
|
|
|
|
static void unexpectedError(FragmentManager manager, Throwable ex) {
|
|
|
|
|
Log.e(ex);
|
|
|
|
|
|
|
|
|
|
Bundle args = new Bundle();
|
|
|
|
|
args.putSerializable("ex", ex);
|
|
|
|
|
|
|
|
|
|
FragmentDialogUnexpected fragment = new FragmentDialogUnexpected();
|
|
|
|
|
fragment.setArguments(args);
|
|
|
|
|
fragment.show(manager, "error:unexpected");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static class FragmentDialogUnexpected extends FragmentDialogBase {
|
|
|
|
|
@NonNull
|
|
|
|
|
@Override
|
|
|
|
|
public Dialog onCreateDialog(@Nullable Bundle savedInstanceState) {
|
|
|
|
|
final Throwable ex = (Throwable) getArguments().getSerializable("ex");
|
|
|
|
|
|
|
|
|
|
return new AlertDialog.Builder(getContext())
|
|
|
|
|
.setTitle(R.string.title_unexpected_error)
|
|
|
|
|
.setMessage(Log.formatThrowable(ex, false))
|
|
|
|
|
.setPositiveButton(android.R.string.cancel, null)
|
|
|
|
|
.setNeutralButton(R.string.title_report, new DialogInterface.OnClickListener() {
|
|
|
|
|
@Override
|
|
|
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
|
|
|
// Dialog will be dismissed
|
|
|
|
|
final Context context = getContext();
|
|
|
|
|
|
|
|
|
|
new SimpleTask<Long>() {
|
|
|
|
|
@Override
|
|
|
|
|
protected Long onExecute(Context context, Bundle args) throws Throwable {
|
|
|
|
|
return Log.getDebugInfo(context, R.string.title_crash_info_remark, ex, null).id;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected void onExecuted(Bundle args, Long id) {
|
|
|
|
|
context.startActivity(new Intent(context, ActivityCompose.class)
|
|
|
|
|
.putExtra("action", "edit")
|
|
|
|
|
.putExtra("id", id));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
|
|
|
|
protected void onException(Bundle args, Throwable ex) {
|
|
|
|
|
if (ex instanceof IllegalArgumentException)
|
|
|
|
|
ToastEx.makeText(context, ex.getMessage(), Toast.LENGTH_LONG).show();
|
|
|
|
|
else
|
|
|
|
|
ToastEx.makeText(context, ex.toString(), Toast.LENGTH_LONG).show();
|
|
|
|
|
}
|
|
|
|
|
}.execute(getContext(), getActivity(), new Bundle(), "error:unexpected");
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
.create();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-12 17:14:34 +00:00
|
|
|
|
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-12-24 09:53:42 +00:00
|
|
|
|
sb.append(String.format("Storage space: %s/%s\r\n",
|
|
|
|
|
Helper.humanReadableByteCount(Helper.getAvailableStorageSpace(), true),
|
|
|
|
|
Helper.humanReadableByteCount(Helper.getTotalStorageSpace(), true)));
|
2019-10-19 19:53:19 +00:00
|
|
|
|
|
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));
|
|
|
|
|
}
|
|
|
|
|
|
2019-05-29 06:51:20 +00:00
|
|
|
|
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
|
2019-06-25 17:58:50 +00:00
|
|
|
|
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-29 06:51:20 +00:00
|
|
|
|
|
2019-05-12 17:14:34 +00:00
|
|
|
|
sb.append("\r\n");
|
|
|
|
|
|
2020-03-26 07:40:26 +00:00
|
|
|
|
sb.append(new Date(Helper.getInstallTime(context))).append("\r\n");
|
|
|
|
|
sb.append(new Date()).append("\r\n");
|
2019-05-12 17:14:34 +00:00
|
|
|
|
|
|
|
|
|
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();
|
2020-04-03 06:59:22 +00:00
|
|
|
|
List<String> keys = new ArrayList<>(settings.keySet());
|
|
|
|
|
Collections.sort(keys);
|
|
|
|
|
for (String key : keys)
|
2019-05-12 17:14:34 +00:00
|
|
|
|
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");
|
|
|
|
|
|
2020-04-03 06:59:22 +00:00
|
|
|
|
List<EntityFolder> folders = db.folder().getFolders(account.id, false, false);
|
|
|
|
|
if (folders.size() > 0)
|
|
|
|
|
Collections.sort(folders, folders.get(0).getComparator(context));
|
|
|
|
|
for (EntityFolder folder : folders)
|
|
|
|
|
size += write(os,
|
|
|
|
|
folder.name + ":" + folder.type + ":" + folder.level +
|
|
|
|
|
" sync=" + folder.synchronize + "/" + folder.download +
|
|
|
|
|
" poll=" + folder.poll + ":" + folder.poll_factor +
|
|
|
|
|
" days=" + folder.sync_days + "/" + folder.keep_days + "/" + folder.initialize + "\r\n" +
|
|
|
|
|
" unified=" + folder.unified + "/" + folder.notify +
|
|
|
|
|
" hide=" + folder.hide + "/" + folder.collapsed + "/" + folder.subscribed + "\r\n" +
|
|
|
|
|
" read-only=" + folder.read_only +
|
|
|
|
|
" selectable=" + folder.selectable + "/" + folder.inferiors +
|
|
|
|
|
" auto-delete=" + folder.auto_delete + "\r\n" +
|
|
|
|
|
" state=" + folder.state + "/" + folder.total +
|
|
|
|
|
" error=" + folder.error +
|
|
|
|
|
" last_sync=" + (folder.last_sync == null ? "" : new Date(folder.last_sync)) + "\r\n");
|
|
|
|
|
|
2019-05-12 17:14:34 +00:00
|
|
|
|
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);
|
|
|
|
|
|
2019-10-22 06:55:27 +00:00
|
|
|
|
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");
|
|
|
|
|
}
|
2020-02-12 11:56:19 +00:00
|
|
|
|
|
|
|
|
|
size += write(os, "VPN active=" + ConnectionHelper.vpnActive(context) + "\r\n\r\n");
|
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 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())
|
2020-02-11 16:47:41 +00:00
|
|
|
|
size += write(os, String.format("%s %d %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,
|
2020-02-11 16:46:04 +00:00
|
|
|
|
op.tries,
|
2019-05-12 17:14:34 +00:00
|
|
|
|
op.args,
|
2020-02-11 16:47:41 +00:00
|
|
|
|
op.state,
|
2019-05-12 17:14:34 +00:00
|
|
|
|
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-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;
|
|
|
|
|
}
|
2019-09-14 08:50:47 +00:00
|
|
|
|
|
|
|
|
|
static boolean isXiaomi() {
|
|
|
|
|
return "Xiaomi".equalsIgnoreCase(Build.MANUFACTURER);
|
|
|
|
|
}
|
2018-12-24 12:27:45 +00:00
|
|
|
|
}
|