2018-08-01 16:41:10 +00:00
|
|
|
package com.topjohnwu.magisk.asyncs;
|
|
|
|
|
|
|
|
import android.app.Activity;
|
|
|
|
import android.app.ProgressDialog;
|
|
|
|
import android.os.AsyncTask;
|
|
|
|
import android.widget.Toast;
|
|
|
|
|
2018-12-02 20:28:18 +00:00
|
|
|
import com.topjohnwu.magisk.BuildConfig;
|
2018-08-01 16:41:10 +00:00
|
|
|
import com.topjohnwu.magisk.Const;
|
|
|
|
import com.topjohnwu.magisk.Data;
|
|
|
|
import com.topjohnwu.magisk.MagiskManager;
|
|
|
|
import com.topjohnwu.magisk.R;
|
|
|
|
import com.topjohnwu.magisk.utils.RootUtils;
|
|
|
|
import com.topjohnwu.magisk.utils.Utils;
|
|
|
|
import com.topjohnwu.superuser.ShellUtils;
|
|
|
|
import com.topjohnwu.superuser.io.SuFile;
|
|
|
|
import com.topjohnwu.superuser.io.SuFileOutputStream;
|
|
|
|
import com.topjohnwu.utils.JarMap;
|
2018-08-04 18:29:40 +00:00
|
|
|
import com.topjohnwu.utils.SignAPK;
|
2018-08-01 16:41:10 +00:00
|
|
|
|
2018-10-26 06:50:45 +00:00
|
|
|
import java.nio.ByteBuffer;
|
|
|
|
import java.nio.ByteOrder;
|
|
|
|
import java.nio.CharBuffer;
|
2018-08-01 16:41:10 +00:00
|
|
|
import java.security.SecureRandom;
|
|
|
|
import java.util.jar.JarEntry;
|
|
|
|
|
|
|
|
public class PatchAPK {
|
|
|
|
|
2018-10-26 06:50:45 +00:00
|
|
|
public static final String LOWERALPHA = "abcdefghijklmnopqrstuvwxyz";
|
|
|
|
public static final String UPPERALPHA = LOWERALPHA.toUpperCase();
|
|
|
|
public static final String ALPHA = LOWERALPHA + UPPERALPHA;
|
|
|
|
public static final String DIGITS = "0123456789";
|
|
|
|
public static final String ALPHANUM = ALPHA + DIGITS;
|
|
|
|
public static final String ALPHANUMDOTS = ALPHANUM + "............";
|
|
|
|
|
2018-08-01 16:41:10 +00:00
|
|
|
private static String genPackageName(String prefix, int length) {
|
|
|
|
StringBuilder builder = new StringBuilder(length);
|
|
|
|
builder.append(prefix);
|
|
|
|
length -= prefix.length();
|
|
|
|
SecureRandom random = new SecureRandom();
|
2018-10-26 06:50:45 +00:00
|
|
|
char next, prev = '.';
|
2018-08-01 16:41:10 +00:00
|
|
|
for (int i = 0; i < length; ++i) {
|
2018-10-26 06:50:45 +00:00
|
|
|
if (prev == '.' || i == length - 1) {
|
|
|
|
next = ALPHA.charAt(random.nextInt(ALPHA.length()));
|
2018-08-01 16:41:10 +00:00
|
|
|
} else {
|
2018-10-26 06:50:45 +00:00
|
|
|
next = ALPHANUMDOTS.charAt(random.nextInt(ALPHANUMDOTS.length()));
|
2018-08-01 16:41:10 +00:00
|
|
|
}
|
|
|
|
builder.append(next);
|
|
|
|
prev = next;
|
|
|
|
}
|
|
|
|
return builder.toString();
|
|
|
|
}
|
|
|
|
|
2018-10-26 06:50:45 +00:00
|
|
|
private static boolean findAndPatch(byte xml[], String a, String b) {
|
|
|
|
if (a.length() != b.length())
|
|
|
|
return false;
|
|
|
|
char[] from = a.toCharArray(), to = b.toCharArray();
|
|
|
|
CharBuffer buf = ByteBuffer.wrap(xml).order(ByteOrder.LITTLE_ENDIAN).asCharBuffer();
|
2018-08-01 16:41:10 +00:00
|
|
|
int offset = -1;
|
2018-10-26 06:50:45 +00:00
|
|
|
for (int i = 0; i < buf.length() - from.length; ++i) {
|
2018-08-01 16:41:10 +00:00
|
|
|
boolean match = true;
|
2018-10-26 06:50:45 +00:00
|
|
|
for (int j = 0; j < from.length; ++j) {
|
|
|
|
if (buf.get(i + j) != from[j]) {
|
2018-08-01 16:41:10 +00:00
|
|
|
match = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-10-26 06:50:45 +00:00
|
|
|
// Make sure it is null terminated
|
|
|
|
if (match && buf.get(i + from.length) == '\0') {
|
2018-08-01 16:41:10 +00:00
|
|
|
offset = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (offset < 0)
|
|
|
|
return false;
|
2018-10-26 06:50:45 +00:00
|
|
|
buf.position(offset);
|
|
|
|
buf.put(to);
|
2018-08-01 16:41:10 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static boolean patchAndHide() {
|
|
|
|
MagiskManager mm = Data.MM();
|
|
|
|
|
|
|
|
// Generate a new app with random package name
|
|
|
|
SuFile repack = new SuFile("/data/local/tmp/repack.apk");
|
2018-12-02 20:28:18 +00:00
|
|
|
String pkg = genPackageName("com.", BuildConfig.APPLICATION_ID.length());
|
2018-08-01 16:41:10 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
JarMap apk = new JarMap(mm.getPackageCodePath());
|
2018-12-02 20:28:18 +00:00
|
|
|
if (!patchPackageID(apk, BuildConfig.APPLICATION_ID, pkg))
|
2018-08-01 16:41:10 +00:00
|
|
|
return false;
|
2018-08-04 18:29:40 +00:00
|
|
|
SignAPK.sign(apk, new SuFileOutputStream(repack));
|
2018-08-01 16:41:10 +00:00
|
|
|
} catch (Exception e) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Install the application
|
|
|
|
if (!ShellUtils.fastCmdResult("pm install " + repack))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
repack.delete();
|
|
|
|
|
|
|
|
mm.mDB.setStrings(Const.Key.SU_MANAGER, pkg);
|
|
|
|
Data.exportPrefs();
|
2018-12-02 20:28:18 +00:00
|
|
|
RootUtils.rmAndLaunch(BuildConfig.APPLICATION_ID, pkg);
|
2018-08-01 16:41:10 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static boolean patchPackageID(JarMap apk, String from, String to) {
|
|
|
|
try {
|
|
|
|
JarEntry je = apk.getJarEntry(Const.ANDROID_MANIFEST);
|
|
|
|
byte xml[] = apk.getRawData(je);
|
|
|
|
|
2018-10-26 06:50:45 +00:00
|
|
|
if (!findAndPatch(xml, from, to) ||
|
|
|
|
!findAndPatch(xml, from + ".provider", to + ".provider"))
|
2018-08-01 16:41:10 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// Write in changes
|
|
|
|
apk.getOutputStream(je).write(xml);
|
|
|
|
} catch (Exception e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void hideManager(Activity activity) {
|
|
|
|
ProgressDialog dialog = ProgressDialog.show(activity,
|
2018-12-02 20:15:42 +00:00
|
|
|
activity.getString(R.string.hide_manager_title),
|
|
|
|
activity.getString(R.string.hide_manager_msg));
|
2018-08-01 16:41:10 +00:00
|
|
|
AsyncTask.THREAD_POOL_EXECUTOR.execute(() -> {
|
|
|
|
boolean b = patchAndHide();
|
|
|
|
Data.mainHandler.post(() -> {
|
|
|
|
dialog.cancel();
|
|
|
|
if (!b) {
|
|
|
|
Utils.toast(R.string.hide_manager_fail_toast, Toast.LENGTH_LONG);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|