diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml index 89b43a786f..c31dbf7674 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -67,13 +67,36 @@ android:name=".ui.MainActivity" android:exported="true" android:screenOrientation="portrait" - tools:ignore="DiscouragedApi,LockedOrientationActivity"> + tools:ignore="DiscouragedApi,LockedOrientationActivity" + android:theme="@style/Theme.HyperCeiler.SystemBarBackgrounds"> + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/java/com/sevtinge/hyperceiler/module/app/Demo.java b/app/src/main/java/com/sevtinge/hyperceiler/module/app/Demo.java index 1b79470c65..07bfb08cf8 100644 --- a/app/src/main/java/com/sevtinge/hyperceiler/module/app/Demo.java +++ b/app/src/main/java/com/sevtinge/hyperceiler/module/app/Demo.java @@ -1,11 +1,13 @@ package com.sevtinge.hyperceiler.module.app; import com.sevtinge.hyperceiler.module.base.BaseModule; +import com.sevtinge.hyperceiler.module.hook.demo.CrashDemo; import com.sevtinge.hyperceiler.module.hook.demo.ToastTest; public class Demo extends BaseModule { @Override public void handleLoadPackage() { initHook(new ToastTest(), true); + initHook(new CrashDemo(), true); } } diff --git a/app/src/main/java/com/sevtinge/hyperceiler/module/base/BaseModule.java b/app/src/main/java/com/sevtinge/hyperceiler/module/base/BaseModule.java index faeca5ec06..e9f037ffb7 100644 --- a/app/src/main/java/com/sevtinge/hyperceiler/module/base/BaseModule.java +++ b/app/src/main/java/com/sevtinge/hyperceiler/module/base/BaseModule.java @@ -22,11 +22,17 @@ import com.sevtinge.hyperceiler.XposedInit; import com.sevtinge.hyperceiler.module.base.dexkit.DexKit; import com.sevtinge.hyperceiler.module.base.dexkit.InitDexKit; +import com.sevtinge.hyperceiler.safe.CrashData; import com.sevtinge.hyperceiler.utils.ContextUtils; +import com.sevtinge.hyperceiler.utils.PropUtils; import com.sevtinge.hyperceiler.utils.api.ProjectApi; import com.sevtinge.hyperceiler.utils.log.XposedLogUtils; import com.sevtinge.hyperceiler.utils.prefs.PrefsMap; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; + import de.robv.android.xposed.callbacks.XC_LoadPackage.LoadPackageParam; public abstract class BaseModule implements IXposedHook { @@ -34,8 +40,14 @@ public abstract class BaseModule implements IXposedHook { public LoadPackageParam mLoadPackageParam = null; public String TAG = getClass().getSimpleName(); public final PrefsMap mPrefsMap = XposedInit.mPrefsMap; + private static HashMap swappedMap = CrashData.swappedData(); public void init(LoadPackageParam lpparam) { + if (swappedMap.isEmpty()) swappedMap = CrashData.swappedData(); + if (needIntercept(lpparam.packageName)) { + XposedLogUtils.logI(TAG, "进入安全模式: " + lpparam.packageName); + return; + } EzXHelper.initHandleLoadPackage(lpparam); EzXHelper.setLogTag(TAG); EzXHelper.setToastTag(TAG); @@ -73,6 +85,26 @@ public void init(LoadPackageParam lpparam) { } } + private boolean needIntercept(String pkg) { + ArrayList report = getReportCrashProp(); + for (String s : report) { + String mPkg = swappedMap.get(s); + if (mPkg != null) { + return mPkg.equals(pkg); + } + } + return false; + } + + private ArrayList getReportCrashProp() { + String data = PropUtils.getProp("persist.hyperceiler.crash.report", ""); + if (data.isEmpty()) { + return new ArrayList<>(); + } + String[] sp = data.split(","); + return new ArrayList<>(Arrays.asList(sp)); + } + @Override public void initZygote() { } diff --git a/app/src/main/java/com/sevtinge/hyperceiler/module/base/BaseXposedInit.java b/app/src/main/java/com/sevtinge/hyperceiler/module/base/BaseXposedInit.java index df7b0aeab4..f81962d812 100644 --- a/app/src/main/java/com/sevtinge/hyperceiler/module/base/BaseXposedInit.java +++ b/app/src/main/java/com/sevtinge/hyperceiler/module/base/BaseXposedInit.java @@ -18,6 +18,7 @@ */ package com.sevtinge.hyperceiler.module.base; +import static com.sevtinge.hyperceiler.callback.ITAG.TAG; import static com.sevtinge.hyperceiler.utils.Helpers.getPackageVersionCode; import static com.sevtinge.hyperceiler.utils.Helpers.getPackageVersionName; import static com.sevtinge.hyperceiler.utils.devicesdk.SystemSDKKt.getAndroidVersion; @@ -26,6 +27,7 @@ import static com.sevtinge.hyperceiler.utils.devicesdk.SystemSDKKt.isMoreAndroidVersion; import static com.sevtinge.hyperceiler.utils.devicesdk.SystemSDKKt.isMoreHyperOSVersion; import static com.sevtinge.hyperceiler.utils.log.LogManager.logLevelDesc; +import static com.sevtinge.hyperceiler.utils.log.XposedLogUtils.logE; import static com.sevtinge.hyperceiler.utils.log.XposedLogUtils.logI; import androidx.annotation.CallSuper; @@ -82,8 +84,8 @@ import com.sevtinge.hyperceiler.module.app.VoiceAssist; import com.sevtinge.hyperceiler.module.app.Weather; import com.sevtinge.hyperceiler.module.base.tool.ResourcesTool; +import com.sevtinge.hyperceiler.safe.CrashHook; import com.sevtinge.hyperceiler.utils.api.ProjectApi; -import com.sevtinge.hyperceiler.utils.log.AndroidLogUtils; import com.sevtinge.hyperceiler.utils.log.XposedLogUtils; import com.sevtinge.hyperceiler.utils.prefs.PrefsMap; import com.sevtinge.hyperceiler.utils.prefs.PrefsUtils; @@ -190,7 +192,7 @@ public void setXSharedPrefs() { mPrefsMap.putAll(allPrefs); } } catch (Throwable t) { - AndroidLogUtils.logD("setXSharedPrefs", t); + XposedLogUtils.logE("setXSharedPrefs", t); } } } @@ -208,12 +210,12 @@ public void init(LoadPackageParam lpparam) { mSystemFramework.init(lpparam); mVariousSystemApps.init(lpparam); } - // try { - // new CrashHook(lpparam); - // logI(TAG.TAG, "Success Hook Crash"); - // } catch (Exception e) { - // logE(TAG.TAG, "Hook Crash E: " + e); - // } + try { + new CrashHook(lpparam); + logI(TAG, "Success Hook Crash"); + } catch (Exception e) { + logE(TAG, "Hook Crash E: " + e); + } } case "com.android.systemui" -> { if (isSystemUIModuleEnable() && isMoreAndroidVersion(33)) { diff --git a/app/src/main/java/com/sevtinge/hyperceiler/module/hook/demo/CrashDemo.java b/app/src/main/java/com/sevtinge/hyperceiler/module/hook/demo/CrashDemo.java new file mode 100644 index 0000000000..27d507c7b1 --- /dev/null +++ b/app/src/main/java/com/sevtinge/hyperceiler/module/hook/demo/CrashDemo.java @@ -0,0 +1,21 @@ +package com.sevtinge.hyperceiler.module.hook.demo; + +import com.sevtinge.hyperceiler.module.base.BaseHook; + +import de.robv.android.xposed.XposedHelpers; + +public class CrashDemo extends BaseHook { + @Override + public void init() throws NoSuchMethodException { + XposedHelpers.findAndHookMethod("com.hchen.demo.MainActivity", lpparam.classLoader, + "crash", int.class, new MethodHook() { + @Override + protected void before(MethodHookParam param) throws Throwable { + int o = (int) param.args[0]; + param.args[0] = 0; + logE(TAG, "int: " + o); + } + } + ); + } +} diff --git a/app/src/main/java/com/sevtinge/hyperceiler/safe/CrashData.java b/app/src/main/java/com/sevtinge/hyperceiler/safe/CrashData.java new file mode 100644 index 0000000000..2a2d6cfba9 --- /dev/null +++ b/app/src/main/java/com/sevtinge/hyperceiler/safe/CrashData.java @@ -0,0 +1,208 @@ +package com.sevtinge.hyperceiler.safe; + +import static com.sevtinge.hyperceiler.utils.log.XposedLogUtils.logE; + +import com.sevtinge.hyperceiler.callback.ITAG; + +import org.json.JSONArray; +import org.json.JSONException; +import org.json.JSONObject; + +import java.util.ArrayList; +import java.util.HashMap; +import java.util.Map; + +public class CrashData { + private static final String TAG = ITAG.TAG + ": CrashRecord"; + private static final HashMap scopeMap = new HashMap<>(); + private static final HashMap swappedMap = new HashMap<>(); + + /** + * 把模块当前使用的全部作用域加入 Map。 + * + * @return Map + * @noinspection SameReturnValue + */ + public static HashMap scopeData() { + if (scopeMap.isEmpty()) { + scopeMap.put("android", "android"); + scopeMap.put("com.android.browser", "browser"); + scopeMap.put("com.android.camera", "camera"); + scopeMap.put("com.android.calendar", "calendar"); + scopeMap.put("com.android.externalstorage", "external"); + scopeMap.put("com.android.fileexplorer", "file"); + scopeMap.put("com.android.incallui", "incallui"); + scopeMap.put("com.android.mms", "mms"); + scopeMap.put("com.android.nfc", "nfc"); + scopeMap.put("com.android.phone", "phone"); + scopeMap.put("com.android.providers.downloads", "download"); + scopeMap.put("com.android.providers.downloads.ui", "ui"); + scopeMap.put("com.android.systemui", "systemui"); + scopeMap.put("com.android.settings", "settings"); + scopeMap.put("com.android.thememanager", "theme"); + scopeMap.put("com.android.updater", "updater"); + scopeMap.put("com.lbe.security.miui", "lbe"); + scopeMap.put("com.milink.service", "milink"); + scopeMap.put("com.miui.aod", "aod"); + scopeMap.put("com.miui.backup", "backup"); + scopeMap.put("com.miui.cloudservice", "cloud"); + scopeMap.put("com.miui.contentextension", "content"); + scopeMap.put("com.miui.creation", "creation"); + scopeMap.put("com.miui.gallery", "gallery"); + scopeMap.put("com.miui.guardprovider", "guard"); + scopeMap.put("com.miui.home", "home"); + scopeMap.put("com.miui.huanji", "huan"); + scopeMap.put("com.miui.mediaeditor", "media"); + scopeMap.put("com.miui.mishare.connectivity", "share"); + scopeMap.put("com.miui.misound", "sound"); + scopeMap.put("com.miui.miwallpaper", "wallpaper"); + scopeMap.put("com.miui.notes", "notes"); + scopeMap.put("com.miui.packageinstaller", "install"); + scopeMap.put("com.miui.personalassistant", "personal"); + scopeMap.put("com.miui.powerkeeper", "power"); + scopeMap.put("com.miui.screenrecorder", "recorder"); + scopeMap.put("com.miui.screenshot", "shot"); + scopeMap.put("com.miui.securityadd", "add"); + scopeMap.put("com.miui.securitycenter", "center"); + scopeMap.put("com.miui.tsmclient", "tsmclient"); + scopeMap.put("com.miui.voiceassist", "voice"); + scopeMap.put("com.miui.weather2", "weather"); + scopeMap.put("com.xiaomi.aiasst.vision", "vision"); + scopeMap.put("com.xiaomi.barrage", "barrage"); + scopeMap.put("com.xiaomi.joyose", "joyose"); + scopeMap.put("com.xiaomi.market", "market"); + scopeMap.put("com.xiaomi.mirror", "mirror"); + scopeMap.put("com.xiaomi.misettings", "misettings"); + scopeMap.put("com.xiaomi.mtb", "mtb"); + scopeMap.put("com.xiaomi.scanner", "scanner"); + scopeMap.put("com.xiaomi.trustservice", "trust"); + scopeMap.put("com.hchen.demo", "demo"); + return scopeMap; + } + return scopeMap; + } + + /** + * 交换 Map 中 Key 和 Value 位置。 + * + * @return 交换后的 Map + * @noinspection SameReturnValue + */ + public static HashMap swappedData() { + if (scopeMap.isEmpty()) scopeData(); + if (!swappedMap.isEmpty()) return swappedMap; + for (Map.Entry entry : scopeMap.entrySet()) { + swappedMap.put(entry.getValue(), entry.getKey()); + } + return swappedMap; + } +} + +/** + * 崩溃记录数据库 + */ +class CrashRecord { + public static final String TAG = ITAG.TAG + ": CrashRecord"; + public String label; + public String pkg; + public long time; + public int count; + + public CrashRecord(String l, String p, long t, int c) { + label = l; + pkg = p; + time = t; + count = c; + } + + public CrashRecord(String p, int c) { + pkg = p; + count = c; + } + + public JSONObject toJSON() { + JSONObject jsonObject = new JSONObject(); + try { + jsonObject.put("l", label); + jsonObject.put("p", pkg); + jsonObject.put("t", time); + jsonObject.put("c", count); + return jsonObject; + } catch (JSONException e) { + logE(TAG, "Failed to convert JSON!" + e); + } + return jsonObject; + } + + public JSONObject toJSONSmall() { + JSONObject jsonObject = new JSONObject(); + try { + jsonObject.put("p", pkg); + jsonObject.put("c", count); + return jsonObject; + } catch (JSONException e) { + logE(TAG, "Failed to convert JSON!" + e); + } + return jsonObject; + } + + public static String getLabel(JSONObject jsonObject) { + try { + return jsonObject.getString("l"); + } catch (JSONException e) { + logE(TAG, "Failed to get name!" + e); + } + return "null"; + } + + public static String getPkg(JSONObject jsonObject) { + try { + return jsonObject.getString("p"); + } catch (JSONException e) { + logE(TAG, "Failed to get package name!" + e); + } + return "null"; + } + + public static long getTime(JSONObject jsonObject) { + try { + return jsonObject.getLong("t"); + } catch (JSONException e) { + logE(TAG, "Failed to get timestamp!" + e); + } + return -1L; + } + + public static int getCount(JSONObject jsonObject) { + try { + return jsonObject.getInt("c"); + } catch (JSONException e) { + logE(TAG, "Failed to get the number of times!" + e); + } + return -1; + } + + public static JSONObject putCount(JSONObject jsonObject, int count) { + try { + return jsonObject.put("c", count); + } catch (JSONException e) { + logE(TAG, "Failed to set the number of times!" + e); + } + return null; + } + + public static ArrayList toArray(String json) { + try { + ArrayList list = new ArrayList<>(); + JSONArray jsonArray = new JSONArray(json); + for (int i = 0; i < jsonArray.length(); i++) { + JSONObject obj = jsonArray.getJSONObject(i); + list.add(obj); + } + return list; + } catch (Exception e) { + logE(TAG, "Failed to convert Array!" + e); + } + return new ArrayList<>(); + } +} diff --git a/app/src/main/java/com/sevtinge/hyperceiler/safe/CrashHook.java b/app/src/main/java/com/sevtinge/hyperceiler/safe/CrashHook.java new file mode 100644 index 0000000000..cffe4d41b8 --- /dev/null +++ b/app/src/main/java/com/sevtinge/hyperceiler/safe/CrashHook.java @@ -0,0 +1,282 @@ +package com.sevtinge.hyperceiler.safe; + +import android.app.ActivityOptions; +import android.app.ApplicationErrorReport; +import android.content.Context; +import android.content.Intent; +import android.content.pm.ApplicationInfo; +import android.content.pm.PackageManager; +import android.provider.Settings; + +import com.sevtinge.hyperceiler.callback.ITAG; +import com.sevtinge.hyperceiler.module.base.tool.HookTool; +import com.sevtinge.hyperceiler.utils.api.ProjectApi; + +import org.json.JSONObject; + +import java.lang.reflect.Method; +import java.util.ArrayList; +import java.util.HashMap; + +import de.robv.android.xposed.XposedHelpers; +import de.robv.android.xposed.callbacks.XC_LoadPackage; + +/** + * 推荐使用的方法,直接 Hook 系统, + * 可能误报,但是很稳定。 + */ +public class CrashHook extends HookTool { + private static final String TAG = ITAG.TAG + ": CrashHook"; + private static HashMap scopeMap = new HashMap<>(); + + public CrashHook(XC_LoadPackage.LoadPackageParam loadPackageParam) throws Exception { + backgroundActivity(loadPackageParam.classLoader); + init(loadPackageParam.classLoader); + scopeMap = CrashData.scopeData(); + } + + public void init(ClassLoader classLoader) throws Exception { + Class appError = findClassIfExists("com.android.server.am.AppErrors", classLoader); + if (appError == null) { + throw new ClassNotFoundException("No such 'com.android.server.am.AppErrors' classLoader: " + classLoader); + } + Method hookError = null; + for (Method error : appError.getDeclaredMethods()) { + if ("handleAppCrashInActivityController".equals(error.getName())) + if (error.getReturnType().equals(boolean.class)) { + hookError = error; + break; + } + } + if (hookError == null) { + throw new NoSuchMethodException("No such Method: handleAppCrashInActivityController, ClassLoader: " + classLoader); + } + + hookMethod(hookError, new MethodHook() { + @Override + protected void after(MethodHookParam param) { + Context mContext = (Context) XposedHelpers.getObjectField(param.thisObject, "mContext"); + Object proc = param.args[0]; + ApplicationErrorReport.CrashInfo crashInfo = (ApplicationErrorReport.CrashInfo) param.args[1]; + String shortMsg = (String) param.args[2]; + String longMsg = (String) param.args[3]; + String stackTrace = (String) param.args[4]; + long timeMillis = (long) param.args[5]; + int callingPid = (int) param.args[6]; + int callingUid = (int) param.args[7]; + logE(TAG, "context: " + mContext + " pkg: " + mContext.getPackageName() + " proc: " + proc + " crash: " + crashInfo + " short: " + shortMsg + + " long: " + longMsg + " stack: " + stackTrace + " time: " + timeMillis + " pid: " + callingPid + " uid: " + callingUid); + recordCrash(mContext, proc, crashInfo, shortMsg, longMsg, stackTrace, timeMillis, callingPid, callingUid); + } + } + ); + } + + private void backgroundActivity(ClassLoader classLoader) { + // 允许哈皮露后台启动界面 + try { + findAndHookMethod("com.android.server.wm.ActivityStarter", classLoader, "shouldAbortBackgroundActivityStart", + int.class, int.class, String.class, int.class, int.class, + "com.android.server.wm.WindowProcessController", "com.android.server.am.PendingIntentRecord", + boolean.class, Intent.class, ActivityOptions.class, + new MethodHook() { + @Override + protected void before(MethodHookParam param) { + String pkg = (String) param.args[2]; + if (pkg == null) return; + if (ProjectApi.mAppModulePkg.equals(pkg)) { + param.setResult(false); + } + } + } + ); + } catch (Throwable e) { + findAndHookMethod("com.android.server.wm.BackgroundActivityStartController", classLoader, "checkBackgroundActivityStart", + int.class, int.class, String.class, int.class, int.class, + "com.android.server.wm.WindowProcessController", "com.android.server.am.PendingIntentRecord", + "android.app.BackgroundStartPrivileges", Intent.class, ActivityOptions.class, + new MethodHook() { + @Override + protected void before(MethodHookParam param) { + String pkg = (String) param.args[2]; + if (pkg == null) return; + if (ProjectApi.mAppModulePkg.equals(pkg)) { + param.setResult(1); + } + } + } + ); + } + hookAllMethods("com.android.server.wm.ActivityStarterImpl", classLoader, + "isAllowedStartActivity", + new MethodHook() { + @Override + protected void before(MethodHookParam param) { + int count = -1; + for (Object clz : param.args) { + count = count + 1; + if (clz instanceof String) { + break; + } + } + String pkg = (String) param.args[count]; + if (pkg == null) return; + if (ProjectApi.mAppModulePkg.equals(pkg)) { + param.setResult(true); + } + } + } + ); + } + + private void recordCrash(Context mContext, Object proc, ApplicationErrorReport.CrashInfo crashInfo, + String shortMsg, String longMsg, String stackTrace, long timeMillis, int pid, int uid) { + PackageManager pm = mContext.getPackageManager(); + ApplicationInfo info = (ApplicationInfo) XposedHelpers.getObjectField(proc, "info"); + String pkg = info.packageName; // 包名 + String label = info.loadLabel(pm).toString(); // 应用名 + if (scopeMap.isEmpty()) scopeMap = CrashData.scopeData(); + if (scopeMap.get(pkg) == null) { + return; // 不属于作用域范围 + } + ArrayList arrayList = new ArrayList<>(); + arrayList.add(new CrashRecord(label, pkg, timeMillis, 0).toJSON()); // 添加本次崩溃记录 + ArrayList data = getCrashRecord(mContext); + if (!data.isEmpty()) { + boolean isNewCrash = false; // 此条崩溃记录是否是全新的 + boolean needReplace = false; // 是否需要更新记录 + boolean isReport = false; // 是否需要报告崩溃 + boolean needRemoveList = false; // 是否更新列表 + ArrayList removeList = new ArrayList<>(); + int remove = -1; + JSONObject add = null; + for (int i = 0; i < data.size(); i++) { + remove = remove + 1; + JSONObject jsonObject = data.get(i); // 读取 + long mTime = CrashRecord.getTime(jsonObject); + if ((timeMillis - mTime) > 60000) { + removeList.add(remove); + needRemoveList = true; + if (removeList.size() == data.size()) { + break; + } + continue; + } + if (compare(jsonObject, label, pkg)) { // 如果全部匹配代表已经在数据库中 + isNewCrash = false; // 不需要再添加 + if ((timeMillis - mTime) < 10240) { + needReplace = true; + int count = CrashRecord.getCount(jsonObject); + if (count >= 2) { // 崩溃报告临界值 + ArrayList report = getReportCrash(mContext); // 获取已经存在的报告记录 + ArrayList newReport = new ArrayList<>(); + if (!report.isEmpty()) { + boolean isNewReport = false; // 是否是新的报告 + for (JSONObject j : report) { + if (compare(j, label, pkg)) { + isNewReport = false; // 不是新的直接跳过 + break; + } + isNewReport = true; + } + if (isNewReport) newReport.add(jsonObject); // 是新的则添加 + } else { + newReport.add(jsonObject); + } + if (!newReport.isEmpty()) { // 非空则报告 + try { + report.addAll(newReport); + reportCrash(mContext, report); + reportCrashByIntent(mContext, longMsg, stackTrace, newReport.get(0)); + } catch (Throwable e) { + logE(TAG, "Report crash failed!" + e); + } + // logE(TAG, "new: " + report); + } + } else { + add = CrashRecord.putCount(jsonObject, count + 1); // 崩溃次数不足则累加 + } + } else { + needReplace = true; // 超出时间则直接清楚本条记录 + } + break; + } else { + if (isReport) break; // 是否已经报告过了 + isNewCrash = true; + ArrayList report = getReportCrash(mContext); + if (!report.isEmpty()) { + for (JSONObject c : report) { + if (compare(c, label, pkg)) { + isReport = true; // 已经报告过了 + break; + } + } + } + } + } + if (needRemoveList) { + for (int i = removeList.size() - 1; i >= 0; i--) { + int id = removeList.get(i); + data.remove(id); + } + } + if (needReplace) { + if (!(removeList.contains(remove) && needRemoveList)) + data.remove(remove); + if (add != null) data.add(add); + } + if (isNewCrash) { + if (!isReport) + data.addAll(arrayList); + } + // logE(TAG, "data: " + data); + setCrashRecord(mContext, data); + } else { + // logE(TAG, "arr: " + arrayList); + setCrashRecord(mContext, arrayList); + } + } + + private boolean compare(JSONObject object, String label, String pkg) { + String mLabel = CrashRecord.getLabel(object); + String mPkg = CrashRecord.getPkg(object); + return mLabel.equals(label) && mPkg.equals(pkg); + } + + private void reportCrashByIntent(Context context, String longMsg, String stackTrace, JSONObject report) { + if (scopeMap.isEmpty()) scopeMap = CrashData.scopeData(); + String pkg = CrashRecord.getPkg(report); + String abbr = scopeMap.get(pkg); + Intent intent = new Intent(); + intent.setAction("com.sevtinge.hyperceiler.crash.Service"); + intent.setPackage("com.sevtinge.hyperceiler"); + intent.putExtra("key_longMsg", longMsg); + intent.putExtra("key_stackTrace", stackTrace); + intent.putExtra("key_report", abbr); + context.startService(intent); + } + + private void reportCrash(Context mContext, ArrayList data) { + Settings.System.putString(mContext.getContentResolver(), "hyperceiler_crash_report", data.toString()); + } + + private ArrayList getReportCrash(Context context) { + String data = Settings.System.getString(context.getContentResolver(), "hyperceiler_crash_report"); + if (data == null || data.isEmpty() || data.equals("[]")) { + return new ArrayList<>(); + } + return CrashRecord.toArray(data); + } + + private void setCrashRecord(Context mContext, ArrayList data) { + Settings.System.putString(mContext.getContentResolver(), "hyperceiler_crash_record_data", data.toString()); + } + + private ArrayList getCrashRecord(Context mContext) { + String data = Settings.System.getString(mContext.getContentResolver(), "hyperceiler_crash_record_data"); + if (data == null || data.isEmpty() || data.equals("[]")) { + return new ArrayList<>(); + } + return CrashRecord.toArray(data); + } +} diff --git a/app/src/main/java/com/sevtinge/hyperceiler/safe/CrashReportActivity.java b/app/src/main/java/com/sevtinge/hyperceiler/safe/CrashReportActivity.java new file mode 100644 index 0000000000..676a6b4920 --- /dev/null +++ b/app/src/main/java/com/sevtinge/hyperceiler/safe/CrashReportActivity.java @@ -0,0 +1,84 @@ +package com.sevtinge.hyperceiler.safe; + +import android.annotation.SuppressLint; +import android.content.Intent; +import android.graphics.Paint; +import android.os.Bundle; +import android.view.LayoutInflater; +import android.view.View; +import android.widget.TextView; + +import androidx.annotation.Nullable; + +import com.sevtinge.hyperceiler.R; +import com.sevtinge.hyperceiler.utils.DialogHelper; +import com.sevtinge.hyperceiler.utils.shell.ShellInit; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.HashMap; + +import moralnorm.appcompat.app.AlertDialog; +import moralnorm.appcompat.app.AppCompatActivity; + +public class CrashReportActivity extends AppCompatActivity { + + TextView mMessageTv; + TextView mCrashRecordTv; + + private static HashMap swappedMap = CrashData.swappedData(); + + @SuppressLint("SetTextI18n") + @Override + public void onCreate(@Nullable Bundle bundle) { + super.onCreate(bundle); + ShellInit.init(); + if (swappedMap.isEmpty()) swappedMap = CrashData.swappedData(); + setContentView(R.layout.activity_crash_dialog); + Intent intent = getIntent(); + String code = intent.getStringExtra("key_report"); + String stackTrace = intent.getStringExtra("key_stackTrace"); + String longMsg = intent.getStringExtra("key_longMsg"); + String pkg = getReportCrashPkg(code); + View view = LayoutInflater.from(this).inflate(R.layout.crash_report_dialog, null); + mMessageTv = view.findViewById(R.id.tv_message); + mMessageTv.setText("作用域: " + "\n\"" + pkg + "\"\n已进入安全模式,点击确定解除,点击取消稍后处理。"); + mCrashRecordTv = view.findViewById(R.id.tv_record); + mCrashRecordTv.getPaint().setFlags(Paint.UNDERLINE_TEXT_FLAG | Paint.ANTI_ALIAS_FLAG);// 下划线并加清晰 + mCrashRecordTv.getPaint().setAntiAlias(true);// 抗锯齿 + mCrashRecordTv.setOnClickListener(v -> { + new AlertDialog.Builder(v.getContext()) + .setCancelable(false) + .setTitle("异常记录") + .setMessage("异常信息: \n" + longMsg + "\n堆栈跟踪: \n" + stackTrace) + .setHapticFeedbackEnabled(true) + .setPositiveButton(android.R.string.ok, (dialog, which) -> dialog.dismiss()) + .show(); + // Toast.makeText(this, "查看异常记录", Toast.LENGTH_SHORT).show(); + }); + DialogHelper.showCrashReportDialog(this, view); + } + + private String getReportCrashPkg(String data) { + if (data == null) return null; + String[] sp = data.split(","); + ArrayList report = new ArrayList<>(Arrays.asList(sp)); + StringBuilder string = null; + for (String s : report) { + String mPkg = swappedMap.get(s); + if (mPkg != null) { + if (string == null) string = new StringBuilder(mPkg); + else + string.append("\n").append(mPkg); + } + } + if (string == null) return null; + return string.toString(); + } + + @Override + protected void onDestroy() { + ShellInit.destroy(); + super.onDestroy(); + } +} diff --git a/app/src/main/java/com/sevtinge/hyperceiler/safe/CrashService.java b/app/src/main/java/com/sevtinge/hyperceiler/safe/CrashService.java new file mode 100644 index 0000000000..32635e49df --- /dev/null +++ b/app/src/main/java/com/sevtinge/hyperceiler/safe/CrashService.java @@ -0,0 +1,50 @@ +package com.sevtinge.hyperceiler.safe; + +import android.app.Service; +import android.content.Intent; +import android.os.IBinder; + +import androidx.annotation.Nullable; + +import com.sevtinge.hyperceiler.callback.ITAG; +import com.sevtinge.hyperceiler.utils.log.AndroidLogUtils; +import com.sevtinge.hyperceiler.utils.shell.ShellInit; + +public class CrashService extends Service { + @Nullable + @Override + public IBinder onBind(Intent intent) { + return null; + } + + @Override + public void onCreate() { + super.onCreate(); + ShellInit.init(); + AndroidLogUtils.logI(ITAG.TAG, "service create"); + } + + @Override + public int onStartCommand(Intent intent, int flags, int startId) { + AndroidLogUtils.logI(ITAG.TAG, "service onStartCommand"); + String report = intent.getStringExtra("key_report"); + String stackTrace = intent.getStringExtra("key_stackTrace"); + String longMsg = intent.getStringExtra("key_longMsg"); + ShellInit.getShell().run("setprop persist.hyperceiler.crash.report " + "\"" + report + "\"").sync(); + Intent intent1 = new Intent(this, CrashReportActivity.class); + intent1.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_SINGLE_TOP); + intent1.putExtra("key_report", report); + intent1.putExtra("key_stackTrace", stackTrace); + intent1.putExtra("key_longMsg", longMsg); + startActivity(intent1); + stopSelf(); + return super.onStartCommand(intent, flags, startId); + } + + @Override + public void onDestroy() { + AndroidLogUtils.logI(ITAG.TAG, "service onDestroy"); + ShellInit.destroy(); + super.onDestroy(); + } +} diff --git a/app/src/main/java/com/sevtinge/hyperceiler/ui/MainActivity.java b/app/src/main/java/com/sevtinge/hyperceiler/ui/MainActivity.java index 67d896dca7..05488a1f06 100644 --- a/app/src/main/java/com/sevtinge/hyperceiler/ui/MainActivity.java +++ b/app/src/main/java/com/sevtinge/hyperceiler/ui/MainActivity.java @@ -29,6 +29,7 @@ import com.sevtinge.hyperceiler.R; import com.sevtinge.hyperceiler.callback.IResult; import com.sevtinge.hyperceiler.prefs.PreferenceHeader; +import com.sevtinge.hyperceiler.safe.CrashData; import com.sevtinge.hyperceiler.ui.base.NavigationActivity; import com.sevtinge.hyperceiler.utils.BackupUtils; import com.sevtinge.hyperceiler.utils.Helpers; @@ -40,12 +41,17 @@ import com.sevtinge.hyperceiler.utils.search.SearchHelper; import com.sevtinge.hyperceiler.utils.shell.ShellInit; +import java.util.ArrayList; +import java.util.Arrays; + import moralnorm.appcompat.app.AlertDialog; public class MainActivity extends NavigationActivity implements IResult { private Handler handler; private Context context; + private ArrayList appCrash = new ArrayList<>(); + @Override public void onCreate(Bundle savedInstanceState) { SharedPreferences mPrefs = PrefsUtils.mSharedPreferences; @@ -62,7 +68,20 @@ public void onCreate(Bundle savedInstanceState) { ShellInit.init(this); PropUtils.setProp("persist.hyperceiler.log.level", (ProjectApi.isRelease() ? def : ProjectApi.isCanary() ? (def == 0 ? 3 : 4) : def)); - // test(); + if (haveCrashReport()) { + new AlertDialog.Builder(this) + .setTitle("提示") + .setMessage("存在处于安全模式的作用域: \n" + appCrash.toString() + " \n请选择是否解除?") + .setHapticFeedbackEnabled(true) + .setCancelable(false) + .setPositiveButton(android.R.string.ok, (dialog, which) -> { + ShellInit.getShell().run("setprop persist.hyperceiler.crash.report \"\"").sync(); + ShellInit.getShell().run("settings put system hyperceiler_crash_report \"[]\"").sync(); + dialog.dismiss(); + }) + .setNegativeButton(android.R.string.cancel, (dialog, which) -> dialog.dismiss()) + .show(); + } } @Override @@ -76,6 +95,31 @@ public void error(String reason) { .show()); } + private boolean haveCrashReport() { + return !needIntercept().isEmpty(); + } + + private ArrayList needIntercept() { + ArrayList report = getReportCrashProp(); + for (String s : report) { + String mPkg = CrashData.swappedData().get(s); + if (mPkg != null) { + appCrash.add(mPkg); + } + } + return appCrash; + } + + private ArrayList getReportCrashProp() { + String data = PropUtils.getProp("persist.hyperceiler.crash.report", ""); + if (data.isEmpty()) { + return new ArrayList<>(); + } + String[] sp = data.split(","); + return new ArrayList<>(Arrays.asList(sp)); + } + + @Override protected void onDestroy() { ShellInit.destroy(); diff --git a/app/src/main/java/com/sevtinge/hyperceiler/utils/DialogHelper.java b/app/src/main/java/com/sevtinge/hyperceiler/utils/DialogHelper.java index a6f7b9b43d..0df18a9ec9 100644 --- a/app/src/main/java/com/sevtinge/hyperceiler/utils/DialogHelper.java +++ b/app/src/main/java/com/sevtinge/hyperceiler/utils/DialogHelper.java @@ -1,28 +1,31 @@ /* - * This file is part of HyperCeiler. + * This file is part of HyperCeiler. - * HyperCeiler is free software: you can redistribute it and/or modify - * it under the terms of the GNU Affero General Public License as - * published by the Free Software Foundation, either version 3 of the - * License. + * HyperCeiler is free software: you can redistribute it and/or modify + * it under the terms of the GNU Affero General Public License as + * published by the Free Software Foundation, either version 3 of the + * License. - * This program 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 Affero General Public License for more details. + * This program 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 Affero General Public License for more details. - * You should have received a copy of the GNU Affero General Public License - * along with this program. If not, see . + * You should have received a copy of the GNU Affero General Public License + * along with this program. If not, see . - * Copyright (C) 2023-2024 HyperCeiler Contributions -*/ + * Copyright (C) 2023-2024 HyperCeiler Contributions + */ package com.sevtinge.hyperceiler.utils; import android.app.Activity; import android.content.Context; import android.content.DialogInterface; +import android.view.View; import com.sevtinge.hyperceiler.R; +import com.sevtinge.hyperceiler.utils.shell.ShellExec; +import com.sevtinge.hyperceiler.utils.shell.ShellInit; import com.sevtinge.hyperceiler.view.RestartAlertDialog; import moralnorm.appcompat.app.AlertDialog; @@ -35,44 +38,59 @@ public static void showDialog(Activity activity, String title, String message) { public static void showDialog(Activity activity, String title, String message, DialogInterface.OnClickListener onClickListener) { new AlertDialog.Builder(activity) - .setTitle(title) - .setMessage(message) - .setPositiveButton(android.R.string.ok, onClickListener) - .setNegativeButton(android.R.string.cancel, (dialog, which) -> dialog.dismiss()) - .show(); + .setTitle(title) + .setMessage(message) + .setPositiveButton(android.R.string.ok, onClickListener) + .setNegativeButton(android.R.string.cancel, (dialog, which) -> dialog.dismiss()) + .show(); } public static void showDialog(Activity activity, int title, int message, DialogInterface.OnClickListener onClickListener) { new AlertDialog.Builder(activity) - .setTitle(title) - .setMessage(message) - .setPositiveButton(android.R.string.ok, onClickListener) - .setNegativeButton(android.R.string.cancel, (dialog, which) -> dialog.dismiss()) - .show(); + .setTitle(title) + .setMessage(message) + .setPositiveButton(android.R.string.ok, onClickListener) + .setNegativeButton(android.R.string.cancel, (dialog, which) -> dialog.dismiss()) + .show(); } public static void showPositiveButtonDialog(Activity activity, String title, String message, DialogInterface.OnClickListener onClickListener) { new AlertDialog.Builder(activity) - .setTitle(title) - .setMessage(message) - .setPositiveButton(android.R.string.ok, onClickListener) - .show(); + .setTitle(title) + .setMessage(message) + .setPositiveButton(android.R.string.ok, onClickListener) + .show(); } public static void showXposedActivateDialog(Context context) { new AlertDialog.Builder(context) - .setCancelable(false) - .setTitle(R.string.tip) - .setMessage(R.string.hook_failed) - .setHapticFeedbackEnabled(true) - .setPositiveButton(R.string.exit, (dialogInterface, i) -> System.exit(0)) - .setNegativeButton(R.string.ignore, null) - .show(); + .setCancelable(false) + .setTitle(R.string.tip) + .setMessage(R.string.hook_failed) + .setHapticFeedbackEnabled(true) + .setPositiveButton(R.string.exit, (dialogInterface, i) -> System.exit(0)) + .setNegativeButton(R.string.ignore, null) + .show(); } public static void showRestartDialog(Context context) { new RestartAlertDialog(context).show(); } + public static void showCrashReportDialog(Activity activity, View view) { + new AlertDialog.Builder(activity) + .setCancelable(false) + .setTitle("警告") + .setView(view) + .setHapticFeedbackEnabled(true) + .setPositiveButton(android.R.string.ok, (dialog, which) -> { + ShellExec shellExec = ShellInit.getShell(); + shellExec.run("setprop persist.hyperceiler.crash.report \"\"").sync(); + shellExec.run("settings put system hyperceiler_crash_report \"[]\"").sync(); + activity.finish(); + }) + .setNegativeButton(android.R.string.cancel, (dialog, which) -> activity.finish()) + .show(); + } } diff --git a/app/src/main/java/com/sevtinge/hyperceiler/utils/shell/ShellUtils.java b/app/src/main/java/com/sevtinge/hyperceiler/utils/shell/ShellUtils.java index 1eb591b7a9..53b8593a3d 100644 --- a/app/src/main/java/com/sevtinge/hyperceiler/utils/shell/ShellUtils.java +++ b/app/src/main/java/com/sevtinge/hyperceiler/utils/shell/ShellUtils.java @@ -215,7 +215,7 @@ public static CommandResult execCommand(String[] commands, boolean isRoot, boole } } return new CommandResult(result, successMsg == null ? null : successMsg.toString(), errorMsg == null ? null - : errorMsg.toString()); + : errorMsg.toString()); } /** diff --git a/app/src/main/res/layout/activity_crash_dialog.xml b/app/src/main/res/layout/activity_crash_dialog.xml new file mode 100644 index 0000000000..98f2dc5b2d --- /dev/null +++ b/app/src/main/res/layout/activity_crash_dialog.xml @@ -0,0 +1,4 @@ + + \ No newline at end of file diff --git a/app/src/main/res/layout/crash_report_dialog.xml b/app/src/main/res/layout/crash_report_dialog.xml new file mode 100644 index 0000000000..4a0d1ae864 --- /dev/null +++ b/app/src/main/res/layout/crash_report_dialog.xml @@ -0,0 +1,24 @@ + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/res/values-night/themes.xml b/app/src/main/res/values-night/themes.xml index 888370e7cb..60cedd3634 100644 --- a/app/src/main/res/values-night/themes.xml +++ b/app/src/main/res/values-night/themes.xml @@ -7,4 +7,18 @@ true true + + + + \ No newline at end of file diff --git a/app/src/main/res/values/colors.xml b/app/src/main/res/values/colors.xml index d1a837d81b..a17c57b90c 100644 --- a/app/src/main/res/values/colors.xml +++ b/app/src/main/res/values/colors.xml @@ -10,6 +10,7 @@ #1A000000 #FF000000 #FFFFFFFF + #50000000 #80000000 #ccffffff diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index da1b1d973a..d2415e28ab 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -1,6 +1,7 @@ HyperCeiler + NFC VPN GPS diff --git a/app/src/main/res/values/themes.xml b/app/src/main/res/values/themes.xml index b904c555e7..aa5883fb1e 100644 --- a/app/src/main/res/values/themes.xml +++ b/app/src/main/res/values/themes.xml @@ -1,4 +1,12 @@ + + + - - + + + + + +