开启辅助访问 切换到宽版

精易论坛

 找回密码
 注册

QQ登录

只需一步,快速开始

用微信号发送消息登录论坛

新人指南 邀请好友注册 - 我关注人的新帖 教你赚取精币 - 每日签到


求职/招聘- 论坛接单- 开发者大厅

论坛版规 总版规 - 建议/投诉 - 应聘版主 - 精华帖总集 积分说明 - 禁言标准 - 有奖举报

查看: 32066|回复: 14
收起左侧

[技术文章] APP脱壳,适用于360、TX等各大主流加密平台

[复制链接]
结帖率:0% (0/1)
发表于 2018-6-10 23:31:43 | 显示全部楼层 |阅读模式   广东省中山市
本文章介绍如何使用xposed框架编写插件来脱app加固的壳。
本文章仅供安卓逆向爱好者学习,切勿用于商业用途。

无意间了解一篇文章介绍了dumpdex,GitHub地址:https://github.com/a813630449/dumpDex

当时心情十分的激动,总算找到一个对我非常有用的项目了。

然后参考了下代码,思路大概是这样的:

一、过滤掉其他应用,只处理360加固、腾讯乐加固这些等。
二、hook Classloader的loadClass(String,boolean) (注意:loadclass(String)函数里最终也还是调用loadClass(String,boolean))
三、获取到hook loadcalss函数返回的Class对象
四、反射调用java.lang.Class的getDex方法,获取到对象
五、反射调用Dex对象的getBytes方法,取到字节集。
六、将字节集流写出到存储卡中。
看完思路后,迫不及待的去试了。
然而并没有想象中的那样,需要脱壳的应用卡爆了,直接报ANR(应用程序无响应)异常。
然后我回过头来,看着代码,不断地调试发现:
Classloader的loadClass(String,boolean)方法被调用了几千次!!!
其中我又调试输出这些class的名称
结果又发现,非常多与之无关的class!!!!
所以过滤下就行了,而且只执行一次就可以。
完整的代码贴下:
  1. package com.wrbug.dumpdex;

  2. import android.os.Environment;

  3. import com.example.admin.myapplication.BuildConfig;
  4. import com.example.admin.myapplication.MainActivity;

  5. import java.io.File;
  6. import java.lang.reflect.Field;
  7. import java.lang.reflect.Method;

  8. import de.robv.android.xposed.IXposedHookLoadPackage;
  9. import de.robv.android.xposed.XC_MethodHook;
  10. import de.robv.android.xposed.XposedBridge;
  11. import de.robv.android.xposed.XposedHelpers;
  12. import de.robv.android.xposed.callbacks.XC_LoadPackage;

  13. /**
  14. * XposedInit
  15. *
  16. * @author wrbug
  17. * @since 2018/3/20
  18. */
  19. public class XposedInit implements IXposedHookLoadPackage {

  20. private Method getBytesMethod;
  21. private Method getDexMethod;


  22. /**
  23. * 加固应用包含的包名,如果无法脱壳,请将application的包名,加到此数组
  24. * com.stub.StubApp 360加固
  25. * s.h.e.l.l.S 爱加密
  26. * com.secneo.apkwrapper.ApplicationWrapper 梆梆加固
  27. * com.tencent.StubShell.TxAppEntry 腾讯加固
  28. */
  29. private String[] packages = {"com.stub.StubApp", "s.h.e.l.l.S",
  30. "com.secneo.apkwrapper.ApplicationWrapper", "com.tencent.StubShell.TxAppEntry"};

  31. public static void log(String txt) {
  32. if (!BuildConfig.DEBUG) {
  33. return;
  34. }
  35. XposedBridge.log("dumpdex-> " + txt);
  36. }

  37. public static void log(Throwable t) {
  38. if (!BuildConfig.DEBUG) {
  39. return;
  40. }
  41. XposedBridge.log(t);
  42. }

  43. @Override
  44. public void handleLoadPackage(final XC_LoadPackage.LoadPackageParam lpparam) {
  45. Class<?> clazz = null;
  46. for (String aPackage : packages) {
  47. clazz = XposedHelpers.findClassIfExists(aPackage, lpparam.classLoader);
  48. if (clazz != null) {
  49. log("find class:" + aPackage);
  50. FileUtils.writeByteToFile("2".getBytes(), Environment.getExternalStorageDirectory().getPath()+"/1.dex");
  51. break;
  52. }
  53. }
  54. if (clazz == null) {
  55. return;
  56. }
  57. final String packageName = lpparam.packageName;
  58. XposedBridge.log(packageName);
  59. try {
  60. initDexMethod();
  61. } catch (Throwable t) {
  62. //Android版本不支持该插件
  63. log(t);
  64. return;
  65. }
  66. XposedHelpers.findAndHookMethod("java.lang.ClassLoader", lpparam.classLoader, "loadClass", String.class, boolean.class, new XC_MethodHook() {
  67. @Override
  68. protected void afterHookedMethod(MethodHookParam param) throws Throwable {
  69. Class c = (Class) param.getResult();

  70. if (c == null) {
  71. return;
  72. }
  73. Object object = getDexMethod.invoke(c);
  74. byte[] array = (byte[]) getBytesMethod.invoke(object);
  75. if (array == null) {
  76. return;
  77. }
  78. if (param.args[0].toString().startsWith("com.newapp")){
  79. saveData(packageName, array);
  80. log("dump dex :" + param.args[0]);
  81. }
  82. }
  83. });
  84. }
  85. public boolean isone=false;
  86. private void saveData(String packageName, byte[] array) {
  87. if (isone){
  88. return;
  89. }
  90. String path = Environment.getExternalStorageDirectory().getPath();
  91. File parent = new File(path);
  92. if (!parent.exists() || !parent.isDirectory()) {
  93. parent.mkdirs();
  94. }
  95. File file = new File(path, "source-" + array.length + ".dex");
  96. if (!file.exists()) {
  97. FileUtils.writeByteToFile(array, file.getAbsolutePath());
  98. //XposedHelpers.callStaticMethod(MainActivity.class,"setArray",byte[].class,array);

  99. log("dump dex :" + file.getAbsolutePath());
  100. isone=true;
  101. }
  102. }

  103. public void initDexMethod() throws ClassNotFoundException, NoSuchMethodException {
  104. Class dex = Class.forName("com.android.dex.Dex");
  105. this.getBytesMethod = dex.getDeclaredMethod("getBytes");
  106. this.getDexMethod = Class.forName("java.lang.Class").getDeclaredMethod("getDex");
  107. }
  108. }
复制代码

注意几点:

1、开发请先搭建xposed环境
2、如果无法脱壳,请将该应用的加密壳的application添加到packages数组中
3、过滤那将com.newapp替换成你需要脱壳的app package

评分

参与人数 2好评 +2 精币 +4 收起 理由
羽傲天 + 1 + 2 好评,脱壳容易修复难~~~~
王导演 + 1 + 2 感谢分享,很给力!~

查看全部评分

本帖被以下淘专辑推荐:

发表于 2020-5-13 20:29:50 | 显示全部楼层   山东省潍坊市
788888888888888888888888888888888
回复 支持 反对

使用道具 举报

发表于 2020-4-22 08:13:04 | 显示全部楼层   湖南省长沙市
回复 支持 反对

使用道具 举报

发表于 2020-4-12 16:24:32 | 显示全部楼层   贵州省遵义市
老哥,最新的软件无法使用了吧
回复 支持 反对

使用道具 举报

签到天数: 1 天

发表于 2020-4-9 23:26:36 | 显示全部楼层   湖北省黄冈市
wrbug的作品 可以
回复 支持 反对

使用道具 举报

发表于 2020-4-1 23:28:04 | 显示全部楼层   广东省揭阳市
学到了666666666
回复 支持 反对

使用道具 举报

结帖率:100% (4/4)

签到天数: 12 天

发表于 2020-1-14 10:33:09 | 显示全部楼层   四川省成都市
脱壳时 APP直接停止运行 这个怎么样处理?
回复 支持 反对

使用道具 举报

发表于 2019-10-24 18:21:15 | 显示全部楼层   广东省深圳市
看一下好不用好
回复 支持 反对

使用道具 举报

发表于 2019-5-23 04:56:39 | 显示全部楼层   湖北省武汉市
感谢分享,很给力
回复 支持 反对

使用道具 举报

发表于 2019-4-28 09:44:20 | 显示全部楼层   北京市北京市
学习了,谢谢分享
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 注册

本版积分规则 致发广告者

发布主题 收藏帖子 返回列表

sitemap| 易语言源码| 易语言教程| 易语言论坛| 诚聘英才| 易语言模块| 手机版| 广告投放| 精易论坛
拒绝任何人以任何形式在本论坛发表与中华人民共和国法律相抵触的言论,本站内容均为会员发表,并不代表精易立场!
论坛帖子内容仅用于技术交流学习和研究的目的,严禁用于非法目的,否则造成一切后果自负!如帖子内容侵害到你的权益,请联系我们!
防范网络诈骗,远离网络犯罪 违法和不良信息举报电话0663-3422125,QQ: 800073686,邮箱:800073686@b.qq.com
Powered by Discuz! X3.4 揭阳市揭东区精易科技有限公司 ( 粤ICP备12094385号-1) 粤公网安备 44522102000125 增值电信业务经营许可证 粤B2-20192173

快速回复 返回顶部 返回列表