一次金融APP的解密历程

打印 上一主题 下一主题

主题 845|帖子 845|积分 2535

前言:

客户仅提供官网下载地址给我们测试。但是由于官网的版本不是最新的,APP会强制你升级。而升级后的APP,是进行加固后的,无法使用frida进行hook,注入进程。那同样也无法使用SSL Unpinning进行限制客户端校验证书。新版app使用查壳软件显示未加壳,但是查看源代码明显少了很多代码,且很多都是变量声明而已。
绕过更新:

我们要想能对APP渗透测试,一般都是需要抓包和解密的。首先使用burp进行抓包代理,官网版本的APP(以下统称旧版APP),是可以轻松抓到APP的包的(该条请求为检验APP最新版本的请求)。但是内容使用了加密,具体什么加密是不得而知。
[img=720,226.22437569884457]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331496.jpg[/img]
获取到请求密文:
  1. vVAK0jos5eT9gmQJaHOaYbqZ1mgXoBH3bee3MTF3G5wNRHRoPPOYokZLT4MQqaPDN%2BLeEYpIzzDJeErDHcDfhY8muosLfOaw35W3BuCxDNtuNFB86RumMBtOcQXT08qw
复制代码
 
响应包未json,urldecode后为:
  1. {"duration":"0ms","note":"","code":1,"resultDES":"UX/jHk6yqix2yxZIrf0rSIuOjCy6oGxjCPUfBL2avG+DWy/++NW16+YQHVFQ+Nj2w9VOWGcH4OxFtGxbR6K7I6pY0Q9hkP9gc0K0JLZ5O+PwOW72nzissCiLG+cHqadKHzkPOQDdBUuBoa4W1Jz7fQ=="}
复制代码
 
通过desStr和resultDES,一开始我猜测他为des加密,具体是不是,后续再说。
先进入APP,但是一进入APP就提示更新:
[img=500,737.993076923077]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331504.jpg[/img]
通过前言,我们知道是不能更新的。(当然不乏某些技术大佬也可以把新版APP搞定,我技术有限,感觉旧版的比较容易搞)。那我们就明确了目标,要先绕过更新校验。
对于不了解hook**和frida的同学,我这边推荐先去网上了解下,还有安装之类的,再来看此篇文章。
 
【----帮助网安学习,以下所有学习资料免费领!加vx:yj009991,备注 “博客园” 获取!】
 ① 网安学习成长路径思维导图
 ② 60+网安经典常用工具包
 ③ 100+SRC漏洞分析报告
 ④ 150+网安攻防实战技术电子书
 ⑤ 最权威CISSP 认证考试指南+题库
 ⑥ 超1800页CTF实战技巧手册
 ⑦ 最新网安大厂面试题合集(含答案)
 ⑧ APP客户端安全检测指南(安卓+IOS)
 
首先我们明确一下思路,要怎么绕过这个更新校验呢?
(1)直接反编译,修改APP的版本信息为99.99之类的;
(2)通过修改版本验证请求,使用http层面去绕过;
(3)使用hook,去重写更新函数,或者绕过更新函数;
第一点要app能支持反编译且不存在校验签名。第二点要能知道加密密文的密钥。所以我选择第三种:
通过jadx搜索更新,发现了两处,成功获取到源代码。
[img=720,219.71830985915494]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331520.jpg[/img]
类名分别为:com.xxxx.AppUpdate和com.xxxx.WelcomeActivity,通过代码审计可以看到,是先调用的WelcomeActivity,WelcomeActivity再去调用的AppUpdate:
[img=720,240.3453237410072]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331523.jpg[/img]
跟踪进入AppUpdate,调用的checkNativeAppVersion():
[img=720,313.0131004366812]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331525.jpg[/img]
通过上述代码,我们可以看到,这边就是用于判断是否升级的函数。
  1. public void onResponse(Call call, Response response) throws IOException {
  2.     try {
  3.         JSONObject jSONObject = new JSONObject(C.s2(new JSONObject(URLDecoder.decode(response.body().string(), DataUtil.UTF8)).getString("resultDES"), Config.WHITE_KEY, Config.IV.getBytes()));
  4.         if (jSONObject.optInt("code", -1) > 0) {
  5.             JSONObject optJSONObject = jSONObject.optJSONObject("object");
  6.             if (optJSONObject == null) {
  7.                 return;
  8.             }
  9.             if (WakedResultReceiver.CONTEXT_KEY.equals(optJSONObject.optString("isUpdate", ChatConfig.CARD_TYPE))) {
  10.                 nativeAppVersionInterface.updateApp(optJSONObject.optString("desc", "当前有新版本,是否需要更新"), optJSONObject.optString(ClientCookie.VERSION_ATTR, ""));
  11.             } else {
  12.                 nativeAppVersionInterface.noUpdateApp();
  13.             }
  14.         } else {
  15.             nativeAppVersionInterface.showError(jSONObject.optString("note"));
  16.         }
  17.     } catch (JSONException e) {
  18.         e.printStackTrace();
  19.         nativeAppVersionInterface.showError(e.getMessage());
  20.     }
  21. }
复制代码
 
当JSONObject.optInt("code", -1) > 0时,是会去进行升级的,否则则执行nativeAppVersionInterface.noUpdateApp()。
这边分析完后,其实我们就可以写js进行hook操作了。
我们的hook思路可以这样设置了:
重写checkNativeAppVersion函数,执行执行nativeAppVersionInterface.noUpdateApp()。
Ps:因为我一开始直接重写了checkNativeAppVersion,只执行了console.log(“enter checkNativeAppVersion”),没有对APP进行启动,这样就会直接卡死在启动页。
附上js代码:
  1. if(Java.available){
  2.     console.log('success');
  3.         Java.perform(function(){
  4.         var appUpdate = Java.use("com.xxxx.AppUpdate");
  5.         appUpdate.checkNativeAppVersion.implementation = function(a,b,c,d,e,f){
  6.             console.log("enter AppUpdate");//判断是否进入该hook函数,进入会执行该命令
  7.             f.noUpdateApp();//直接执行不需要更新函数,APP会自动进入
  8.         }
  9.         });
  10. }
复制代码
使用命令:frida -U -l .\xxx.js -f 包名 --no-pause
[img=720,290.67285382830624]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331528.jpg[/img]
成功进入:
[img=500,520.0573065902579]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331747.jpg[/img]
解密:

已经成功进入该APP,但是如果想成功进行渗透测试的话,还需要能解开APP的加密。通过des字段,初步判断为des加密,再回头看看刚刚更新的那个请求,是有用c.s2()函数进行操作的,大概率s2就是解密函数。
  1. JSONObject jSONObject = new JSONObject(C.s2(new JSONObject(URLDecoder.decode(response.body().string(), DataUtil.UTF8)).getString("resultDES"), Config.WHITE_KEY, Config.IV.getBytes()));
复制代码
 
可以看到s2的三个参数,即前面响应包中的json字段里面的resultDES参数,然后其次是Config.WHITE_KEY, Config.IV两个参数,其中Config.IV是以字节数组的形式进行传参的。通过跳转可以看到配置文件的参数。
[img=720,45.03092783505155]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331767.jpg[/img]
然后呢,因为获取到密钥和偏移量iv,这样的话des就可以解了。但是问题是解不开。后续的思路就是如果可以直接hook这两个加解密函数的话,是不是就可以不用管他的加解密了。
[img=720,381.53846153846155]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331772.jpg[/img]
s1和s2函数不在java层,那我们就需要hook native层的代码。Hook so文件。首先我们先把安装包后缀apk改成zip,然后解压。就可以找到wkb-1.2.2.so的文件了。(路径为lib/arm64-v8a/wkb-1.2.2.so,前面的arm64根据自己测试机的CPU架构进行选择。)直接用ida打开,在导出函数里面搜索des:
[img=720,379.15064818953954]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331779.jpg[/img]
里面有很多des的相关函数。可使用以下js进行hook导出函数:
  1. if(Java.available){
  2.     console.log('success');
  3.         Java.perform(function(){
  4.         var point = Module.findExportByName("libwkb-1.2.2.so","desDecryptByteArray");
  5.         Interceptor.attach(point,{
  6.             onEnter: function(args){
  7.                 console.log("Hook start");
  8.                 console.log("args[0]=" + args[0]); //打印我们java层第一个传入的参数
  9.                 console.log("args[1]=" + args[1]); //打印我们java层传入的第二个参数
  10.             },
  11.             onLeave: function(retval){ //onLeave: function(retval)是该函数执行结束要执行的代码,其中retval参数即是返回值
  12.                 console.log("return:" + retval); //打印返回值
  13.             }
  14.         });
  15.         });
  16. }
复制代码
但是这边很奇怪的是,通过函数findExportByName找到的地址都是为null,一开始以为是还没加载到so文件,但是后续进入APP后还是一样为null。(有知道的大佬可以说下)
[img=700,66.75257731958763]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331795.jpg[/img]
这就比较蛋疼了,得手动计算地址。首先先获取so文件的地址,看能不能获取到,若不行,则表示未加载so文件。
  1. var soAddr = Module.findBaseAddress("libwkb-1.2.2.so");<br>console.log("soAddr:" + soAddr);
复制代码

有地址出来,说明so文件是存在的,可以正常调用。那么这边就要去计算函数偏移量。之前在网上看到别人的一个公式:
函数地址=so**初始地址+函数偏移量+1**
但是我后面尝试了好几个,好像不同手机不同的计算方法,也可能我操作的有问题。我这边的函数地址就是:
函数地址=so**初始地址+函数偏移量**
不用加一。我自己是用这个方法测试计算的:找到一个导出函数可以被查询到的,比如我这边使用的就是JNI_OnLoad函数:
[img=720,102.64462809917356]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331805.jpg[/img]

获取JNI_OnLoad的地址为0x79d5d7883c,然后使用这个地址减去so的地址:
0x79d5d7883c − 0x79d5d67000 = 1183c
差值刚好为JNI_OnLoad的偏移量,所以我这边就不用再进行加一操作了。
这样我们就可以成功hook任意函数了。通过我一个个尝试发现,以下函数一个都没调用过:
[img=720,325.3998532648569]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331835.jpg[/img]
然后呢,我查找了s2函数的用例,发现被decodeSm4的函数调用过。
[img=720,151.38140747176368]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331857.jpg[/img]
我就尝试了一下,hook了sm4EncryptByteArr:
[img=720,281.37931034482756]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331862.jpg[/img]
[img=720,64.97480531378837]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331864.jpg[/img]
附上js:
  1. var soAddr = Module.findBaseAddress("libwkb-1.2.2.so");
  2. var point = soAddr.add(0x136f0);
  3. Interceptor.attach(point,{
  4.             onEnter: function(args){
  5.                 console.log("Hook start");
  6.                 console.log("args[0]=" + args[0]); //打印我们java层第一个传入的参数
  7.                 console.log("args[2]=" + Java.vm.getEnv().getStringUtfChars(args[2], null).readCString()); //打印我们java层传入的第三个参数
  8.                 console.log("args[3]=" + Java.vm.getEnv().getStringUtfChars(args[3], null).readCString()); //打印我们java层传入的第四个参数
  9.             },
  10.             onLeave: function(retval){ //onLeave: function(retval)是该函数执行结束要执行的代码,其中retval参数即是返回值
  11.                 console.log("return:" + Java.vm.getEnv().getStringUtfChars(retval, null).readCString()); //打印返回值
  12.                 // retval.replace(0); //替换返回值为0
  13.                 // return retval;
  14.             }
  15.         });
复制代码
Ps:通过ida里面的参数,我们可以看到第二个参数为类,我们就没给他打印出来。
我人傻了,一开始的des字眼和偏移量这些都符合des的加密方式,误导了我好久,一直往des方向去找。
尾声:
其实很早我就已经解密成功了,直接通过java层,刚刚发现调用s2的decodeSm4函数,直接hook那边即可成功获取请求和响应的明文:
[img=720,150.57602490918526]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331158.jpg[/img]
但是若通过js去操作修改数值,实在太麻烦了,要获取密钥和加密方式,通过脚本自动去加解密,所以我才会去hook native层,获取到密钥。因为上述密钥Config.WHITE_KEY,其实是还有一层加密的,通过hook decodeWhiteKey函数的返回值,成功获取了密钥。
[img=720,329.51824817518246]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331735.jpg[/img]
[img=720,91.29411764705883]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331834.jpg[/img]
其实后续我也尝试去修改版本号绕过,但是事实证明,代码存在验签:
[img=500,397.37430555555557]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331855.jpg[/img]
可以看到,把版本号修改为99.9.99,成功绕过了更新检测,但是他还存在一个盗版验签检测:
[img=500,1111.842105263158]https://m-1254331109.cos.ap-guangzhou.myqcloud.com/202211111331859.jpg[/img]
验签代码一样需要用hook去绕过。所以前面说的方法一也是行不通的。然后我又突发奇想,有没有可能他密文里面就包含版本信息,那如果我使用99.9.99的版本,抓取密文,然后再安装旧版APP,在他去请求版本更新时,替换密文,是不是可以绕过呢?经过尝试,结果是:可以。他的版本校验就是在服务端,这种方法也可以绕过。
总结:

不用轻易相信别人留下的信息,还是得根据自己的分析得出结论。其实后续我一直在想为什么那个字段是des呢,感觉之前是des加密,后续金融行业都进行了国密改造,然后字段并未更改,导致这种现象,当然只是猜测。至此,已完成对这APP的抓包和加解密。
更多靶场实验练习、网安学习资料,请点击这里>>
 

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

麻花痒

金牌会员
这个人很懒什么都没写!

标签云

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