feat: 反防抓包,强显再创按钮,不跟踪

This commit is contained in:
MoonLeeeaf 2024-06-08 18:04:27 +08:00
parent 5bc2e427df
commit 7469415b2a
3 changed files with 134 additions and 37 deletions

View File

@ -0,0 +1,35 @@
{
"allFiles": [
{
"file": "/storage/emulated/0/铃芽の文件/Projects/FuckMaoNemo/app/src/main/res/xml/config.xml",
"selection": {
"end": {
"column": 7,
"index": 1062,
"line": 30
},
"start": {
"column": 7,
"index": 1062,
"line": 30
}
}
},
{
"file": "/storage/emulated/0/铃芽の文件/Projects/FuckMaoNemo/app/src/main/java/io/github/moonleeeaf/fuckmaonemo/Hook.java",
"selection": {
"end": {
"column": 31,
"index": 5591,
"line": 138
},
"start": {
"column": 31,
"index": 5591,
"line": 138
}
}
}
],
"selectedFile": "/storage/emulated/0/铃芽の文件/Projects/FuckMaoNemo/app/src/main/res/xml/config.xml"
}

View File

@ -1,6 +1,7 @@
package io.github.moonleeeaf.fuckmaonemo; package io.github.moonleeeaf.fuckmaonemo;
import android.app.Activity; import android.app.Activity;
import android.app.Application; import android.app.Application;
import android.content.Context;
import android.os.Bundle; import android.os.Bundle;
import android.widget.Toast; import android.widget.Toast;
import de.robv.android.xposed.IXposedHookLoadPackage; import de.robv.android.xposed.IXposedHookLoadPackage;
@ -11,10 +12,16 @@ import de.robv.android.xposed.XposedBridge;
import de.robv.android.xposed.XposedHelpers; import de.robv.android.xposed.XposedHelpers;
import de.robv.android.xposed.callbacks.XC_LoadPackage; import de.robv.android.xposed.callbacks.XC_LoadPackage;
import java.lang.reflect.Method; import java.lang.reflect.Method;
import java.net.Proxy;
import java.util.HashMap;
import java.util.Map;
public class Hook implements IXposedHookLoadPackage { public class Hook implements IXposedHookLoadPackage {
private static boolean isHooked = false; private static boolean isHooked = false;
private XSharedPreferences xsp; private XSharedPreferences xsp;
private ClassLoader classLoader;
private int nohengheng;
private int aaaa;
@Override @Override
public void handleLoadPackage(XC_LoadPackage.LoadPackageParam param) throws Throwable { public void handleLoadPackage(XC_LoadPackage.LoadPackageParam param) throws Throwable {
@ -29,15 +36,15 @@ public class Hook implements IXposedHookLoadPackage {
protected void afterHookedMethod(MethodHookParam mParam) throws Throwable { protected void afterHookedMethod(MethodHookParam mParam) throws Throwable {
super.afterHookedMethod(mParam); super.afterHookedMethod(mParam);
Object initApp = XposedHelpers.getObjectField(mParam.thisObject, "mInitialApplication"); Object initApp = XposedHelpers.getObjectField(mParam.thisObject, "mInitialApplication");
hook(param, initApp.getClass().getClassLoader()); classLoader = initApp.getClass().getClassLoader();
hook(param);
} }
} }
); );
} }
} }
public static Method getMethod(Class clazz, String name, Class... args) throws NoSuchMethodException { public static Method getMethod(Class clazz, String name, Class<?>... args) throws NoSuchMethodException {
return clazz.getDeclaredMethod(name, args); return clazz.getDeclaredMethod(name, args);
} }
@ -45,18 +52,19 @@ public class Hook implements IXposedHookLoadPackage {
return (Application) XposedHelpers.callStaticMethod(Class.forName("android.app.ActivityThread"), "currentApplication"); return (Application) XposedHelpers.callStaticMethod(Class.forName("android.app.ActivityThread"), "currentApplication");
} }
public void hook(XC_LoadPackage.LoadPackageParam param, ClassLoader classLoader) throws Exception { public void hook(XC_LoadPackage.LoadPackageParam param) throws Exception {
if (isHooked) return; if (isHooked) return;
else isHooked = true; else isHooked = true;
int nohengheng = 0; nohengheng = 0;
aaaa = 0;
xsp = new XSharedPreferences("io.github.moonleeeaf.fuckmaonemo", "config"); xsp = new XSharedPreferences("io.github.moonleeeaf.fuckmaonemo", "config");
XposedBridge.log("[FuckMaoNemo] 注入中..."); XposedBridge.log("[FuckMaoNemo] 注入中...");
// 拦截40x码 // 拦截40x码
if (xsp.getBoolean("fuck_40x", false)) { load("fuck_40x", () -> {
XposedBridge.log("[FuckMaoNemo] Hook_拦截40x码"); XposedBridge.log("[FuckMaoNemo] Hook_拦截40x码");
XposedBridge.hookMethod( XposedBridge.hookMethod(
getMethod( getMethod(
@ -72,53 +80,95 @@ public class Hook implements IXposedHookLoadPackage {
if(code >= 400 && code <500) { if(code >= 400 && code <500) {
Object rawRes = XposedHelpers.getObjectField(res, "rawResponse"); Object rawRes = XposedHelpers.getObjectField(res, "rawResponse");
XposedHelpers.setIntField(rawRes, "code", 200); XposedHelpers.setIntField(rawRes, "code", 200);
XposedBridge.log("[FuckMaoNemo] 拦截响应 40x 码"); XposedBridge.log("[FuckMaoNemo] 拦截响应 " + code + "");
String t = ""; String t = "服务端返回响应码 " + code;
switch (code){
case 401:
t = "已拦截异常登出";
break;
case 422:
t = "已拦截封号页面替换资料卡";
break;
case 405:
t = "评论接口被禁止";
break;
default:
t = "未知拦截,响应码为 " + code;
}
Toast.makeText(getApplication(), "[FuckMaoNemo] " + t, Toast.LENGTH_SHORT).show(); Toast.makeText(getApplication(), "[FuckMaoNemo] " + t, Toast.LENGTH_SHORT).show();
} }
} }
} }
); );
nohengheng++; });
}
// 绕过防沉迷 // 绕过防沉迷
if (xsp.getBoolean("fuck_fcm", false)) { load("fuck_fcm", () -> {
XposedBridge.log("[FuckMaoNemo] Hook_绕过防沉迷"); XposedBridge.log("[FuckMaoNemo] Hook_绕过防沉迷");
XposedBridge.hookMethod( methodToVoid(getMethod(
getMethod(
XposedHelpers.findClass("com.codemao.nemo.activity.WorkDetailActivity", classLoader), XposedHelpers.findClass("com.codemao.nemo.activity.WorkDetailActivity", classLoader),
"checkAntiAddictionState", "checkAntiAddictionState",
null null
));
});
// 强制显示再创作按钮
load("force_show_rework", () -> {
XposedBridge.log("[FuckMaoNemo] Hook_强制显示再创作按钮");
XposedBridge.hookMethod(
getMethod(
XposedHelpers.findClass("com.codemao.creativecenter.utils.bcm.bean.CreativeWorkDetailInfo", classLoader),
"isFork_enable",
null
), ),
new XC_MethodReplacement() { new XC_MethodReplacement() {
@Override @Override
protected Object replaceHookedMethod(MethodHookParam arg0) throws Throwable { protected Object replaceHookedMethod(MethodHookParam arg0) throws Throwable {
XposedBridge.log("[FuckMaoNemo] 拦截防沉迷方法调用"); return true;
return null;
} }
} }
); );
nohengheng++; });
}
// 不追踪
load("no_records", () -> {
XposedBridge.log("[FuckMaoNemo] Hook_不追踪");
methodToVoid(getMethod(
XposedHelpers.findClass("cn.codemao.android.stat.CodeMaoStat", classLoader),
"recordEvent",
String.class,
Map.class
));
});
// 反防抓包
load("fuck_no_proxy", () -> {
XposedBridge.log("[FuckMaoNemo] Hook_反防抓包");
// TODO其实可以从 OkHttp 底层去Hook的
methodToVoid(
getMethod(
XposedHelpers.findClass("okhttp3.OkHttpClient$Builder", classLoader),
"proxy",
Proxy.class
)
);
});
XposedBridge.log("[FuckMaoNemo] 执行完毕"); XposedBridge.log("[FuckMaoNemo] 执行完毕");
Toast.makeText(getApplication(), "[FuckMaoNemo] 加载成功 (≧▽≦)\n" + nohengheng + " 个功能已加载", Toast.LENGTH_LONG).show(); Toast.makeText(getApplication(), "[FuckMaoNemo] 加载成功 (≧▽≦)\n" + nohengheng + " 个功能加载成功, " + aaaa + " 个失败", Toast.LENGTH_LONG).show();
}
public interface Callback {
public void onCallback() throws Exception;
}
public void methodToVoid(Method m) {
XposedBridge.hookMethod(m, new XC_MethodReplacement() {
@Override
protected Object replaceHookedMethod(MethodHookParam arg0) throws Throwable {
return null;
}
});
}
public void load(String pref, Callback cb) {
if (xsp.getBoolean(pref, false)) {
try {
cb.onCallback();
nohengheng++;
} catch (Exception e) {
XposedBridge.log(e);
aaaa++;
}
}
} }
} }

View File

@ -8,7 +8,7 @@
<Preference <Preference
android:title="作者" android:title="作者"
android:summary="编程猫 @满月叶\nGitHub @MoonLeeeaf" /> android:summary="编程猫 @满月叶\nGitHub @MoonLeeeaf\n喵呜~" />
</PreferenceCategory> </PreferenceCategory>
@ -19,6 +19,16 @@
android:title="阻止服务端响应 40x 码" android:title="阻止服务端响应 40x 码"
android:summary="启用此开关后,将可以:\n拦截 401 码强制登出账号\n强制浏览已封禁账号的资料卡\n评论接口检测\n其他作用" /> android:summary="启用此开关后,将可以:\n拦截 401 码强制登出账号\n强制浏览已封禁账号的资料卡\n评论接口检测\n其他作用" />
<SwitchPreference
android:key="no_records"
android:title="不要追踪我"
android:summary="启用后,模块将会拦截 CodeMaoStat.recordEvent(str, map) 调用(比如点击创作按钮都会发送一个 collect 请求),风险未知" />
<SwitchPreference
android:key="fuck_no_proxy"
android:title="反防抓包"
android:summary="效果不太明显,目前只能发现账号相关的有反防抓包" />
</PreferenceCategory> </PreferenceCategory>
<PreferenceCategory android:title="作品"> <PreferenceCategory android:title="作品">
@ -28,11 +38,13 @@
android:title="绕过防沉迷" android:title="绕过防沉迷"
android:summary="滞空防沉迷检测方法的调用实现无伤速通破解防沉迷,无需 Player 链接" /> android:summary="滞空防沉迷检测方法的调用实现无伤速通破解防沉迷,无需 Player 链接" />
</PreferenceCategory>
<PreferenceCategory android:title="已弃用">
<SwitchPreference <SwitchPreference
android:key="force_rework" android:key="force_show_rework"
android:title="强制显示再创作" android:title="强制显示再创作图标"
android:enabled="false" android:summary="使作品的 再创作 按钮永远显示,即使作品并未设置为开放源代码(显示了也没啥用,服务端给你拦截了:作品未开源)" />
android:summary="使作品的 再创作 按钮永远显示即使作品并未设置为开放源代码未制作QWQ" />
</PreferenceCategory> </PreferenceCategory>
</PreferenceScreen> </PreferenceScreen>