Update snet extension

This commit is contained in:
topjohnwu 2018-08-22 11:50:21 +08:00
parent e42b608444
commit d2c196896d
15 changed files with 150 additions and 257 deletions

View File

@ -79,7 +79,7 @@
<!-- Hardcode GMS version --> <!-- Hardcode GMS version -->
<meta-data <meta-data
android:name="com.google.android.gms.version" android:name="com.google.android.gms.version"
android:value="7095000" /> android:value="12451000" />
</application> </application>

View File

@ -34,7 +34,6 @@ public class Const {
// Versions // Versions
public static final int UPDATE_SERVICE_VER = 1; public static final int UPDATE_SERVICE_VER = 1;
public static final int SNET_VER = 10;
public static int MIN_MODULE_VER() { public static int MIN_MODULE_VER() {
return Data.magiskVersionCode >= MAGISK_VER.REMOVE_LEGACY_LINK ? 1500 : 1400; return Data.magiskVersionCode >= MAGISK_VER.REMOVE_LEGACY_LINK ? 1500 : 1400;
@ -76,7 +75,6 @@ public class Const {
public static class Url { public static class Url {
public static final String STABLE_URL = "https://raw.githubusercontent.com/topjohnwu/magisk_files/master/stable.json"; public static final String STABLE_URL = "https://raw.githubusercontent.com/topjohnwu/magisk_files/master/stable.json";
public static final String BETA_URL = "https://raw.githubusercontent.com/topjohnwu/magisk_files/master/beta.json"; public static final String BETA_URL = "https://raw.githubusercontent.com/topjohnwu/magisk_files/master/beta.json";
public static final String SNET_URL = "https://github.com/topjohnwu/magisk_files/raw/a300521162587da23e45010797bfd8c9a03594f6/snet.apk";
public static final String REPO_URL = "https://api.github.com/users/Magisk-Modules-Repo/repos?per_page=100&sort=pushed&page=%d"; public static final String REPO_URL = "https://api.github.com/users/Magisk-Modules-Repo/repos?per_page=100&sort=pushed&page=%d";
public static final String FILE_URL = "https://raw.githubusercontent.com/Magisk-Modules-Repo/%s/master/%s"; public static final String FILE_URL = "https://raw.githubusercontent.com/Magisk-Modules-Repo/%s/master/%s";
public static final String ZIP_URL = "https://github.com/Magisk-Modules-Repo/%s/archive/master.zip"; public static final String ZIP_URL = "https://github.com/Magisk-Modules-Repo/%s/archive/master.zip";

View File

@ -39,6 +39,8 @@ public class Data {
public static String managerLink; public static String managerLink;
public static String managerNoteLink; public static String managerNoteLink;
public static String uninstallerLink; public static String uninstallerLink;
public static int snetVersionCode;
public static String snetLink;
// Install flags // Install flags
public static boolean keepVerity = false; public static boolean keepVerity = false;

View File

@ -226,11 +226,9 @@ public class MagiskFragment extends BaseFragment
boolean hasNetwork = Download.checkNetworkStatus(mm); boolean hasNetwork = Download.checkNetworkStatus(mm);
boolean hasRoot = Shell.rootAccess(); boolean hasRoot = Shell.rootAccess();
boolean hasGms = hasGms();
boolean isUpToDate = Data.magiskVersionCode > Const.MAGISK_VER.UNIFIED; boolean isUpToDate = Data.magiskVersionCode > Const.MAGISK_VER.UNIFIED;
magiskUpdate.setVisibility(hasNetwork ? View.VISIBLE : View.GONE); magiskUpdate.setVisibility(hasNetwork ? View.VISIBLE : View.GONE);
safetyNetCard.setVisibility(hasNetwork && hasGms ? View.VISIBLE : View.GONE);
installOptionCard.setVisibility(hasNetwork ? View.VISIBLE : View.GONE); installOptionCard.setVisibility(hasNetwork ? View.VISIBLE : View.GONE);
uninstallButton.setVisibility(isUpToDate && hasRoot ? View.VISIBLE : View.GONE); uninstallButton.setVisibility(isUpToDate && hasRoot ? View.VISIBLE : View.GONE);
coreOnlyNotice.setVisibility(mm.prefs.getBoolean(Const.Key.COREONLY, false) ? View.VISIBLE : View.GONE); coreOnlyNotice.setVisibility(mm.prefs.getBoolean(Const.Key.COREONLY, false) ? View.VISIBLE : View.GONE);
@ -254,6 +252,8 @@ public class MagiskFragment extends BaseFragment
private void updateCheckUI() { private void updateCheckUI() {
int image, color; int image, color;
safetyNetCard.setVisibility(hasGms() ? View.VISIBLE : View.GONE);
if (Data.remoteMagiskVersionCode < 0) { if (Data.remoteMagiskVersionCode < 0) {
color = colorNeutral; color = colorNeutral;
image = R.drawable.ic_help; image = R.drawable.ic_help;
@ -312,12 +312,6 @@ public class MagiskFragment extends BaseFragment
} else { } else {
@StringRes int resid; @StringRes int resid;
switch (response) { switch (response) {
case ISafetyNetHelper.CAUSE_SERVICE_DISCONNECTED:
resid = R.string.safetyNet_network_loss;
break;
case ISafetyNetHelper.CAUSE_NETWORK_LOST:
resid = R.string.safetyNet_service_disconnected;
break;
case ISafetyNetHelper.RESPONSE_ERR: case ISafetyNetHelper.RESPONSE_ERR:
resid = R.string.safetyNet_res_invalid; resid = R.string.safetyNet_res_invalid;
break; break;

View File

@ -2,7 +2,6 @@ package com.topjohnwu.magisk.asyncs;
import android.app.Activity; import android.app.Activity;
import com.topjohnwu.magisk.Const;
import com.topjohnwu.magisk.Data; import com.topjohnwu.magisk.Data;
import com.topjohnwu.magisk.utils.ISafetyNetHelper; import com.topjohnwu.magisk.utils.ISafetyNetHelper;
import com.topjohnwu.magisk.utils.Topic; import com.topjohnwu.magisk.utils.Topic;
@ -20,7 +19,7 @@ import java.net.HttpURLConnection;
import dalvik.system.DexClassLoader; import dalvik.system.DexClassLoader;
public class CheckSafetyNet extends ParallelTask<Void, Void, Exception> { public class CheckSafetyNet extends ParallelTask<Void, Void, Void> {
public static final File dexPath = public static final File dexPath =
new File(Data.MM().getFilesDir().getParent() + "/snet", "snet.apk"); new File(Data.MM().getFilesDir().getParent() + "/snet", "snet.apk");
@ -33,7 +32,7 @@ public class CheckSafetyNet extends ParallelTask<Void, Void, Exception> {
private void dlSnet() throws Exception { private void dlSnet() throws Exception {
Shell.sh("rm -rf " + dexPath.getParent()).exec(); Shell.sh("rm -rf " + dexPath.getParent()).exec();
dexPath.getParentFile().mkdir(); dexPath.getParentFile().mkdir();
HttpURLConnection conn = WebService.mustRequest(Const.Url.SNET_URL, null); HttpURLConnection conn = WebService.mustRequest(Data.snetLink, null);
try ( try (
OutputStream out = new BufferedOutputStream(new FileOutputStream(dexPath)); OutputStream out = new BufferedOutputStream(new FileOutputStream(dexPath));
InputStream in = new BufferedInputStream(conn.getInputStream())) { InputStream in = new BufferedInputStream(conn.getInputStream())) {
@ -52,13 +51,13 @@ public class CheckSafetyNet extends ParallelTask<Void, Void, Exception> {
.invoke(null, ISafetyNetHelper.class, dexPath.getPath(), getActivity(), .invoke(null, ISafetyNetHelper.class, dexPath.getPath(), getActivity(),
(ISafetyNetHelper.Callback) code -> (ISafetyNetHelper.Callback) code ->
Topic.publish(false, Topic.SNET_CHECK_DONE, code)); Topic.publish(false, Topic.SNET_CHECK_DONE, code));
if (helper.getVersion() != Const.SNET_VER) { if (helper.getVersion() < Data.snetVersionCode) {
throw new Exception(); throw new Exception();
} }
} }
@Override @Override
protected Exception doInBackground(Void... voids) { protected Void doInBackground(Void... voids) {
try { try {
try { try {
dyload(); dyload();
@ -67,21 +66,12 @@ public class CheckSafetyNet extends ParallelTask<Void, Void, Exception> {
dlSnet(); dlSnet();
dyload(); dyload();
} }
} catch (Exception e) { // Run attestation
return e;
}
return null;
}
@Override
protected void onPostExecute(Exception e) {
if (e == null) {
helper.attest(); helper.attest();
} else { } catch (Exception e) {
e.printStackTrace(); e.printStackTrace();
Topic.publish(false, Topic.SNET_CHECK_DONE, -1); Topic.publish(false, Topic.SNET_CHECK_DONE, -1);
} }
super.onPostExecute(e); return null;
} }
} }

View File

@ -78,6 +78,10 @@ public class CheckUpdates {
JSONObject uninstaller = getJson(json, "uninstaller"); JSONObject uninstaller = getJson(json, "uninstaller");
Data.uninstallerLink = getString(uninstaller, "link", null); Data.uninstallerLink = getString(uninstaller, "link", null);
JSONObject snet = getJson(json, "snet");
Data.snetVersionCode = getInt(snet, "versionCode", -1);
Data.snetLink = getString(snet, "link", null);
} }
public static void check(Runnable cb) { public static void check(Runnable cb) {

View File

@ -4,10 +4,8 @@ import android.support.annotation.Keep;
public interface ISafetyNetHelper { public interface ISafetyNetHelper {
int CAUSE_SERVICE_DISCONNECTED = 0x01; int RESPONSE_ERR = 0x01;
int CAUSE_NETWORK_LOST = 0x02; int CONNECTION_FAIL = 0x02;
int RESPONSE_ERR = 0x04;
int CONNECTION_FAIL = 0x08;
int BASIC_PASS = 0x10; int BASIC_PASS = 0x10;
int CTS_PASS = 0x20; int CTS_PASS = 0x20;

View File

@ -154,6 +154,7 @@
android:layout_marginEnd="5dp" android:layout_marginEnd="5dp"
android:layout_marginStart="5dp" android:layout_marginStart="5dp"
android:layout_marginTop="4dp" android:layout_marginTop="4dp"
android:visibility="gone"
app:cardCornerRadius="@dimen/card_corner_radius" app:cardCornerRadius="@dimen/card_corner_radius"
app:cardElevation="@dimen/card_elevation"> app:cardElevation="@dimen/card_elevation">

View File

@ -25,7 +25,7 @@ allprojects {
ext { ext {
compileSdkVersion = 28 compileSdkVersion = 28
buildToolsVersion = "28.0.0" buildToolsVersion = "28.0.2"
supportLibVersion = "27.1.1" supportLibVersion = "27.1.1"
} }

View File

@ -8,23 +8,20 @@ android {
applicationId "com.topjohnwu.snet" applicationId "com.topjohnwu.snet"
minSdkVersion 14 minSdkVersion 14
targetSdkVersion rootProject.ext.compileSdkVersion targetSdkVersion rootProject.ext.compileSdkVersion
versionCode 1 versionCode 11
versionName "1.0" versionName "snet"
} }
buildTypes { buildTypes {
release { release {
minifyEnabled true minifyEnabled true
shrinkResources true shrinkResources true
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
} }
} }
} }
dependencies { dependencies {
implementation fileTree(dir: 'libs', include: ['*.jar']) implementation fileTree(dir: 'libs', include: ['*.jar'])
/* The oldest version */ implementation 'com.google.android.gms:play-services-safetynet:15.0.1'
implementation('com.google.android.gms:play-services-safetynet:7.0.0') {
exclude module: 'support-v4'
}
} }

View File

@ -20,7 +20,4 @@
# hide the original source file name. # hide the original source file name.
#-renamesourcefileattribute SourceFile #-renamesourcefileattribute SourceFile
-ignorewarnings
-keep class com.topjohnwu.snet.Snet { *; } -keep class com.topjohnwu.snet.Snet { *; }
-dontwarn java.lang.invoke**
-dontwarn com.google.android.gms.common.GooglePlayServicesUtil**

View File

@ -1,148 +0,0 @@
package com.topjohnwu.snet;
import android.app.Activity;
import android.app.AlertDialog;
import android.app.Dialog;
import android.content.Context;
import android.content.ContextWrapper;
import android.content.DialogInterface;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.content.res.AssetManager;
import android.content.res.Resources;
import android.util.Log;
import com.google.android.gms.common.GooglePlayServicesUtil;
import com.google.android.gms.internal.zzlu;
/* Decompiled and modified from GooglePlayServiceUtil.class */
public class ModdedGPSUtil {
private static final String TAG = "GooglePlayServicesUtil";
static String dexPath;
static Dialog getErrorDialog(int errCode, final Activity activity, final int requestCode) {
SwapResContext ctx = new SwapResContext(activity, dexPath);
Resources res = ctx.getResources();
if (zzlu.zzQ(ctx) && errCode == 2) {
errCode = 42;
}
AlertDialog.Builder builder = new AlertDialog.Builder(activity);
builder.setMessage(GooglePlayServicesUtil.zze(ctx, errCode));
String btnMsg = GooglePlayServicesUtil.zzf(ctx, errCode);
if (btnMsg != null) {
final Intent intent = GooglePlayServicesUtil.zzan(errCode);
builder.setPositiveButton(btnMsg, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int i) {
PackageManager pm = activity.getPackageManager();
if (intent != null && intent.resolveActivity(pm) != null)
activity.startActivityForResult(intent, requestCode);
dialog.dismiss();
}
});
}
switch(errCode) {
case 0:
return null;
case 1:
return builder.setTitle(res.getString(com.google.android.gms.R.string.common_google_play_services_install_title)).create();
case 2:
return builder.setTitle(res.getString(com.google.android.gms.R.string.common_google_play_services_update_title)).create();
case 3:
return builder.setTitle(res.getString(com.google.android.gms.R.string.common_google_play_services_enable_title)).create();
case 4:
case 6:
return builder.create();
case 5:
Log.e(TAG, "An invalid account was specified when connecting. Please provide a valid account.");
return builder.setTitle(res.getString(com.google.android.gms.R.string.common_google_play_services_invalid_account_title)).create();
case 7:
Log.e(TAG, "Network error occurred. Please retry request later.");
return builder.setTitle(res.getString(com.google.android.gms.R.string.common_google_play_services_network_error_title)).create();
case 8:
Log.e(TAG, "Internal error occurred. Please see logs for detailed information");
return builder.create();
case 9:
Log.e(TAG, "Google Play services is invalid. Cannot recover.");
return builder.setTitle(res.getString(com.google.android.gms.R.string.common_google_play_services_unsupported_title)).create();
case 10:
Log.e(TAG, "Developer error occurred. Please see logs for detailed information");
return builder.create();
case 11:
Log.e(TAG, "The application is not licensed to the user.");
return builder.create();
case 12:
case 13:
case 14:
case 15:
case 18:
case 19:
case 20:
case 21:
case 22:
case 23:
case 24:
case 25:
case 26:
case 27:
case 28:
case 29:
case 30:
case 31:
case 32:
case 33:
case 34:
case 35:
case 36:
case 37:
case 38:
case 39:
case 40:
case 41:
default:
Log.e(TAG, "Unexpected error code " + errCode);
return builder.create();
case 16:
Log.e(TAG, "One of the API components you attempted to connect to is not available.");
return builder.create();
case 17:
Log.e(TAG, "The specified account could not be signed in.");
return builder.setTitle(res.getString(com.google.android.gms.R.string.common_google_play_services_sign_in_failed_title)).create();
case 42:
return builder.setTitle(res.getString(com.google.android.gms.R.string.common_android_wear_update_title)).create();
}
}
public static class SwapResContext extends ContextWrapper {
private AssetManager asset;
private Resources resources;
public SwapResContext(Context base, String apk) {
super(base);
try {
asset = AssetManager.class.newInstance();
AssetManager.class.getMethod("addAssetPath", String.class).invoke(asset, apk);
} catch (Exception e) {
e.printStackTrace();
}
Resources res = base.getResources();
resources = new Resources(asset, res.getDisplayMetrics(), res.getConfiguration());
}
@Override
public Resources getResources() {
return resources;
}
@Override
public AssetManager getAssets() {
return asset;
}
}
}

View File

@ -1,15 +1,24 @@
package com.topjohnwu.snet; package com.topjohnwu.snet;
import android.app.Activity; import android.app.Activity;
import android.os.Bundle; import android.app.AlertDialog;
import android.app.Dialog;
import android.content.Context;
import android.content.Intent;
import android.support.annotation.NonNull;
import android.util.Base64; import android.util.Base64;
import android.util.Log;
import com.google.android.gms.common.ConnectionResult; import com.google.android.gms.common.ConnectionResult;
import com.google.android.gms.common.GooglePlayServicesUtil; import com.google.android.gms.common.GoogleApiAvailability;
import com.google.android.gms.common.api.GoogleApiClient; import com.google.android.gms.common.api.ApiException;
import com.google.android.gms.common.api.ResultCallback; import com.google.android.gms.common.internal.ConnectionErrorMessages;
import com.google.android.gms.common.internal.DialogRedirect;
import com.google.android.gms.safetynet.SafetyNet; import com.google.android.gms.safetynet.SafetyNet;
import com.google.android.gms.safetynet.SafetyNetApi; import com.google.android.gms.safetynet.SafetyNetApi;
import com.google.android.gms.safetynet.SafetyNetClient;
import com.google.android.gms.tasks.OnFailureListener;
import com.google.android.gms.tasks.OnSuccessListener;
import org.json.JSONException; import org.json.JSONException;
import org.json.JSONObject; import org.json.JSONObject;
@ -18,54 +27,29 @@ import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method; import java.lang.reflect.Method;
import java.security.SecureRandom; import java.security.SecureRandom;
public class SafetyNetHelper implements InvocationHandler, GoogleApiClient.ConnectionCallbacks, public class SafetyNetHelper implements InvocationHandler,
GoogleApiClient.OnConnectionFailedListener, ResultCallback<SafetyNetApi.AttestationResult> { OnSuccessListener<SafetyNetApi.AttestationResponse>, OnFailureListener {
public static final int CAUSE_SERVICE_DISCONNECTED = 0x01; private static final int RESPONSE_ERR = 0x01;
public static final int CAUSE_NETWORK_LOST = 0x02; private static final int CONNECTION_FAIL = 0x02;
public static final int RESPONSE_ERR = 0x04; private static final int BASIC_PASS = 0x10;
public static final int CONNECTION_FAIL = 0x08; private static final int CTS_PASS = 0x20;
public static final int BASIC_PASS = 0x10; private static final GoogleApiAvailability API_AVAIL = GoogleApiAvailability.getInstance();
public static final int CTS_PASS = 0x20; private static final SecureRandom RANDOM = new SecureRandom();
private static final String TAG = "SNET";
public static final int SNET_EXT_VER = 10; /* Insert the magic API key here :) */
private static final String API_KEY = "";
private GoogleApiClient mGoogleApiClient; private final Activity mActivity;
private Activity mActivity; private final Object callback;
private Object callback;
SafetyNetHelper(Activity activity, Object cb) { SafetyNetHelper(Activity activity, Object cb) {
mActivity = activity; mActivity = activity;
callback = cb; callback = cb;
} }
/* Override ISafetyNetHelper.getVersion */
private int getVersion() {
return SNET_EXT_VER;
}
/* Override ISafetyNetHelper.attest */
private void attest() {
// Connect Google Service
mGoogleApiClient = new GoogleApiClient.Builder(mActivity)
.addApi(SafetyNet.API)
.addOnConnectionFailedListener(this)
.addConnectionCallbacks(this)
.build();
mGoogleApiClient.connect();
}
@Override
public Object invoke(Object o, Method method, Object[] args) {
if (method.getName().equals("attest")) {
attest();
} else if (method.getName().equals("getVersion")) {
return getVersion();
}
return null;
}
private void invokeCallback(int code) { private void invokeCallback(int code) {
Class<?> clazz = callback.getClass(); Class<?> clazz = callback.getClass();
try { try {
@ -73,34 +57,50 @@ public class SafetyNetHelper implements InvocationHandler, GoogleApiClient.Conne
} catch (Exception ignored) {} } catch (Exception ignored) {}
} }
@Override /* Override ISafetyNetHelper.getVersion */
public void onConnectionSuspended(int i) { private int getVersion() {
invokeCallback(i); return BuildConfig.VERSION_CODE;
} }
@Override /* Override ISafetyNetHelper.attest */
public void onConnectionFailed(ConnectionResult result) { private void attest() {
if (GooglePlayServicesUtil.isUserRecoverableError(result.getErrorCode())) int code = API_AVAIL.isGooglePlayServicesAvailable(mActivity);
ModdedGPSUtil.getErrorDialog(result.getErrorCode(), mActivity, 0).show(); if (code != ConnectionResult.SUCCESS) {
invokeCallback(CONNECTION_FAIL); if (API_AVAIL.isUserResolvableError(code))
} getErrorDialog(code, 0).show();
invokeCallback(CONNECTION_FAIL);
@Override return;
public void onConnected(Bundle bundle) { }
// Create nonce // Create nonce
byte[] nonce = new byte[24]; byte[] nonce = new byte[24];
new SecureRandom().nextBytes(nonce); RANDOM.nextBytes(nonce);
// Call SafetyNet SafetyNetClient client = SafetyNet.getClient(mActivity.getBaseContext());
SafetyNet.SafetyNetApi.attest(mGoogleApiClient, nonce).setResultCallback(this); client.attest(nonce, API_KEY).addOnSuccessListener(this).addOnFailureListener(this);
}
private Dialog getErrorDialog(int errorCode, int requestCode) {
AlertDialog.Builder builder = new AlertDialog.Builder(mActivity);
Context swapCtx = new SwapResContext(mActivity, Snet.dexPath);
Intent intent = API_AVAIL.getErrorResolutionIntent(swapCtx, errorCode, "d");
builder.setMessage(ConnectionErrorMessages.getErrorMessage(swapCtx, errorCode));
builder.setPositiveButton(
ConnectionErrorMessages.getErrorDialogButtonMessage(swapCtx, errorCode),
DialogRedirect.getInstance(mActivity, intent, requestCode));
String title;
if ((title = ConnectionErrorMessages.getErrorTitle(swapCtx, errorCode)) != null) {
builder.setTitle(title);
}
return builder.create();
} }
@Override @Override
public void onResult(SafetyNetApi.AttestationResult result) { public void onSuccess(SafetyNetApi.AttestationResponse result) {
int code = 0; int code = 0;
try { try {
if (!result.getStatus().isSuccess())
throw new JSONException("");
String jsonStr = new String(Base64.decode( String jsonStr = new String(Base64.decode(
result.getJwsResult().split("\\.")[1], Base64.DEFAULT)); result.getJwsResult().split("\\.")[1], Base64.DEFAULT));
JSONObject json = new JSONObject(jsonStr); JSONObject json = new JSONObject(jsonStr);
@ -110,10 +110,34 @@ public class SafetyNetHelper implements InvocationHandler, GoogleApiClient.Conne
code = RESPONSE_ERR; code = RESPONSE_ERR;
} }
// Disconnect
mGoogleApiClient.disconnect();
// Return results // Return results
invokeCallback(code); invokeCallback(code);
} }
@Override
public void onFailure(@NonNull Exception e) {
if (e instanceof ApiException) {
int errCode = ((ApiException) e).getStatusCode();
if (API_AVAIL.isUserResolvableError(errCode))
getErrorDialog(errCode, 0).show();
else
Log.e(TAG, "Cannot resolve: " + e.getMessage());
} else {
Log.e(TAG, "Unknown: " + e.getMessage());
}
invokeCallback(CONNECTION_FAIL);
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) {
switch (method.getName()) {
case "attest":
attest();
return null;
case "getVersion":
return getVersion();
default:
return null;
}
}
} }

View File

@ -5,9 +5,11 @@ import android.app.Activity;
import java.lang.reflect.Proxy; import java.lang.reflect.Proxy;
public class Snet { public class Snet {
public static Object newHelper(Class<?> clazz, String dexPath, Activity activity, Object cb) { static String dexPath;
ModdedGPSUtil.dexPath = dexPath;
public static Object newHelper(Class<?> interfaceClass, String dexPath, Activity activity, Object cb) {
Snet.dexPath = dexPath;
return Proxy.newProxyInstance(SafetyNetHelper.class.getClassLoader(), return Proxy.newProxyInstance(SafetyNetHelper.class.getClassLoader(),
new Class[] { clazz }, new SafetyNetHelper(activity, cb)); new Class[] { interfaceClass }, new SafetyNetHelper(activity, cb));
} }
} }

View File

@ -0,0 +1,34 @@
package com.topjohnwu.snet;
import android.content.Context;
import android.content.ContextWrapper;
import android.content.res.AssetManager;
import android.content.res.Resources;
public class SwapResContext extends ContextWrapper {
private AssetManager asset;
private Resources resources;
public SwapResContext(Context base, String apk) {
super(base);
try {
asset = AssetManager.class.newInstance();
AssetManager.class.getMethod("addAssetPath", String.class).invoke(asset, apk);
} catch (Exception e) {
e.printStackTrace();
}
Resources res = base.getResources();
resources = new Resources(asset, res.getDisplayMetrics(), res.getConfiguration());
}
@Override
public Resources getResources() {
return resources;
}
@Override
public AssetManager getAssets() {
return asset;
}
}