首页
社区
课程
招聘
[原创]初识Frida--Android逆向之Java层hook (二)
发表于: 2018-6-5 17:40 20497

[原创]初识Frida--Android逆向之Java层hook (二)

2018-6-5 17:40
20497

今天继续一个新的示例,同样采用CTF作为例子,难度稍微加大了一点,如果对Frida基本的使用还不是很了解,建议先看看之前的文章初识Frida--Android逆向之Java层hook (一)
博客同步:访问

文章涉及到的知识点:

示例下载:whyshouldIpay

下载apk后安装,一样还是先来看看是什么功能,这是一个比较简单的验证程序,简单的使用后,了解到PREMIUM CONETNT内容需要输入License验证后才能查看。那估计PREMIUM CONETNT按钮中的内容应该就是答案了吧。

使用jadx将apk反编译出来,分析,在AndroidManifest.xml中找到了启动的Activity是LauncherActivity

找到其中验证的主要代码verifyClick,分析如下:

verifyClick中可以知道生成激活秘钥的算法是MainActivity.xor

来到MainActivity中,查看该方法,看上去笔算起来还是比较麻烦。

接下来当程序被激活成功后,点击PREMIUM CONETNT按钮,会调用MainActivity中的方法,可以看到它将MAC,以及生成的Key发送到了MainActivity中。

MainActivityonCreate方法中,看到了最终答案生成的native方法stringFromJNI(key, mac)

好,现在源代码分析基本上能够理清楚了,大概的过程就是这样。

接下来重点就是要寻找hook点,经过刚才解题流程的分析,得出hook思路如下:

先来一个简单的示例,看看getMac()方法返回的的是什么,采用的方法是hook showPremium,这样就能通过点击PREMIUM CONETNT按钮直接得到getMac()的返回值。

JavaScript代码如下:

完整python代码如下:

运行看看结果:

接下来开始真正第一步的hook,将mac值与“LICENSEKEYOK"通过MainActivity.xor获取秘钥Key。那就直接hook getKey方法吧,这样可以自己来构造秘钥Key。
仔细分析,会发现在这一步中可能会遇到下面的问题:

怎么将javascript参数进行类型转换并传递到java语言中?其实方法很简单,既然java是强类型语言,那就根据它要求的类型传递对应参数即可,看看它参数的类型。

那么,在javascript代码中,先准备一个将字符串类型转换为byte[]类型的方法stringToBytes,再通过实例化MainActivity类的方式调用xor(),然后还需要一个将byte[]回转为String的方法,因为秘钥key是Sting类型的。

接下来,执行看看,能不能获取秘钥Key。
不知道怎么启动模拟器中的frida-server,以及端口转发,可以先看看初识Frida--Android逆向之Java层hook (一)
启动python脚本,在模拟器中直接点击PREMIUM CONTENT,即可看到执行结果。

前面2个步骤,可以说是已经完成90%了,接下来只需要在hook一个能够触发showPremium方法的即可。方法就随意了,这里采用hook verifyClick的方式,这样点击app上的VERIFY按钮,触发verifyClick方法去调用showPremium,进而获得最终答案。

启动脚本,点击app上的VERIFY按钮看看执行结果:

完整python代码:下载

通过上面的例子,可以学习在java层怎么使用frida实现:

 
 
 
 
 
 
 
public void verifyClick(View v) {
    //第一个验证,将输入的Licese通过网络验证,但这个肯定是通不过的,这是一个可能需要绕过的点。
        try {
            InputStream in = new URL("http://broken.license.server.com/query?license=" + ((EditText) findViewById(R.id.text_license)).getText().toString()).openConnection().getInputStream();
            StringBuilder responseBuilder = new StringBuilder();
            byte[] b = new byte[0];
            while (in.read(b) > 0) {
                responseBuilder.append(b);
            }
            String response = responseBuilder.toString();
     //网络验证需要服务器返回 "LICENSEKEYOK",才能进行下一步
            if (response.equals("LICENSEKEYOK")) {
      //当网络验证成功后,生成激活秘钥,并写入到preferences文件中
                String activatedKey = new String(MainActivity.xor(getMac().getBytes(), response.getBytes()));
                Editor editor = getApplicationContext().getSharedPreferences("preferences", 0).edit();
                editor.putString("KEY", activatedKey);
                editor.commit();
      //这样便成功激活
                new Builder(this).setTitle((CharSequence) "Activation successful").setMessage((CharSequence) "Activation successful").setIcon(17301543).show();
                return;
            }
            new Builder(this).setTitle((CharSequence) "Invalid license!").setMessage((CharSequence) "Invalid license!").setIcon(17301543).show();
        } catch (Exception e) {
            new Builder(this).setTitle((CharSequence) "Error occured").setMessage((CharSequence) "Server unreachable").setNeutralButton((CharSequence) "OK", null).setIcon(17301543).show();
        }
    }
 String activatedKey = new String(MainActivity.xor(getMac().getBytes(), response.getBytes()));

[培训]内核驱动高级班,冲击BAT一流互联网大厂工作,每周日13:00-18:00直播授课

最后于 2018-6-6 10:54 被ghostmazeW编辑 ,原因: 修复图片
收藏
免费 9
支持
分享
最新回复 (25)
雪    币: 289
活跃值: (77)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
2
很好很详细,谢谢分享
2018-6-5 19:43
0
雪    币: 233
活跃值: (1346)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
3
好牛逼的帖子,不过不会js
2018-6-6 11:11
0
雪    币: 930
活跃值: (731)
能力值: ( LV7,RANK:110 )
在线值:
发帖
回帖
粉丝
4
zhuangbx 很好很详细,谢谢分享
谢谢
2018-6-6 11:24
0
雪    币: 930
活跃值: (731)
能力值: ( LV7,RANK:110 )
在线值:
发帖
回帖
粉丝
5
安卓qq 好牛逼的帖子,不过不会js[em_5]
没事,抽点时间简单学习js一下就OK了
2018-6-6 11:24
0
雪    币: 233
活跃值: (1346)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
6
不错,大佬还有后续么
2018-6-7 23:15
0
雪    币: 930
活跃值: (731)
能力值: ( LV7,RANK:110 )
在线值:
发帖
回帖
粉丝
7
安卓qq 不错,大佬还有后续么[em_6]
当然有啊!工作太忙了,我一定会抽时间给大家分享的
2018-6-8 13:21
0
雪    币: 905
活跃值: (1082)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
8
感谢楼主分享,未来还会出Native层的吗
2018-6-11 22:39
0
雪    币: 1256
活跃值: (41)
能力值: ( LV4,RANK:50 )
在线值:
发帖
回帖
粉丝
9
兄弟对DexClassLoader动态加载的class有什么好的hook方法吗?
frida作者只是简单讲了一下思路  https://github.com/frida/frida/issues/150#issuecomment-238328067
另一个提问者说搞定了,但没分享方案  https://github.com/frida/frida-java/issues/65
2018-6-12 18:36
0
雪    币: 930
活跃值: (731)
能力值: ( LV7,RANK:110 )
在线值:
发帖
回帖
粉丝
10
wx_La0s 感谢楼主分享,未来还会出Native层的吗
会的
2018-6-28 14:46
0
雪    币: 7129
活跃值: (4392)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
11
感谢分享, 期待native的
最后于 2018-7-2 18:35 被MaYil编辑 ,原因:
2018-7-2 18:34
0
雪    币: 930
活跃值: (731)
能力值: ( LV7,RANK:110 )
在线值:
发帖
回帖
粉丝
12
jackyspy 兄弟对DexClassLoader动态加载的class有什么好的hook方法吗? frida作者只是简单讲了一下思路 https://github.com/frida/frida/issues/15 ...
第三篇文章,我会详细给大家演示
2018-7-3 16:16
0
雪    币: 1
活跃值: (51)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
13
牛逼  期待多出一点教程
2018-7-27 15:40
0
雪    币: 222
活跃值: (10)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
14
楼主帖子写的仔细,很不错,能学到很多东西。感觉我这种小白还是从正向和其他简单逆向教程慢慢学起好了。
2018-8-3 17:37
1
雪    币: 608
活跃值: (176)
能力值: ( LV4,RANK:50 )
在线值:
发帖
回帖
粉丝
15

感谢作者分享,写的很详细,针对byte[]转string,string转byte[] 传参这个问题上,这里有更简单快捷的实现方式,便于不擅于js的童鞋参考一下。

import frida,sys

def on_message(message,data):
    if message['type'] == 'send':
        print("[*] {0}".format(message['payload']))
    else:
        print(message)

jscode = """
Java.perform(function(){
    function stringToBytes(str){
    var javaString = Java.use('java.lang.String');
    var bytes = [];
    bytes = javaString.$new(str).getBytes();
    return bytes;
    }

    function bytesToString(bytes){
    var javaString = Java.use('java.lang.String');
    return javaString.$new(bytes);
    }    

    var LauncherActivity = Java.use('de.fraunhofer.sit.premiumapp.LauncherActivity');
    var activeKey_str="";
    LauncherActivity.getKey.implementation = function(v){
    return activeKey_str;
    }
    LauncherActivity.showPremium.implementation = function(v){
    send("Hook start...");
    var key  = this.getKey();
    var mac = this.getMac();
    send("key:"+key);
    send("mac:"+mac);
    var MainActivity = Java.use('de.fraunhofer.sit.premiumapp.MainActivity');
    var activeKey = MainActivity.xor(stringToBytes(mac),stringToBytes("LICENSEKEYOK"));
    activeKey_str = bytesToString(activeKey);    
    send("activeKey:" + activeKey_str);
    this.showPremium(v);
    }
}
);
"""

dev = frida.get_usb_device()
session = dev.attach('de.fraunhofer.sit.premiumapp')
script = session.create_script(jscode)
script.on('message',on_message)
script.load()
sys.stdin.read()
2018-11-16 13:52
6
雪    币: 930
活跃值: (731)
能力值: ( LV7,RANK:110 )
在线值:
发帖
回帖
粉丝
16
kimoh 感谢作者分享,写的很详细,针对byte[]转string,string转byte[] 传参这个问题上,这里有更简单快捷的实现方式,便于不擅于js的童鞋参考一下。 ```python import ...
赞!!!很好啊!
2018-11-16 14:16
1
雪    币: 880
活跃值: (212)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
17
很赞 学习ing
2018-11-16 15:38
1
雪    币: 48
活跃值: (3515)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
18
学习了,非常感谢,请问什么时候出点native的呢
最后于 2018-11-22 23:13 被Harpe编辑 ,原因:
2018-11-22 23:13
0
雪    币: 284
活跃值: (250)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
19
学习啦
2019-10-29 08:41
0
雪    币: 284
活跃值: (250)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
20
希望楼主以后继续写出这一类的文章
2019-10-29 08:42
0
雪    币: 2468
活跃值: (5118)
能力值: ( LV3,RANK:30 )
在线值:
发帖
回帖
粉丝
21
感谢楼主,学到很多,以下是一些疑惑:
1、str.charCodeAt(i)得到的是unicode编码,所以stringToBytes得到的是unicode编码串,而 byteToString(arr)函数看样子像是把utf-8编码串转成字符串,因而两个函数其实是不能互逆的,即把一个字符串先用stringToBytes再byteToString(arr),得到的不再是原来的字符串。
2、var store = _arr[i].toString(2).slice(7 - bytesLength); 中bytesLength应该是utf-8编码中第一个字节1的个数,所以感觉slice里面好像应该是7 - bytesLength才对。
编码知识参考了这个 http://www.ruanyifeng.com/blog/2007/10/ascii_unicode_and_utf-8.html
最后于 2019-11-27 04:34 被KerryS编辑 ,原因:
2019-11-27 04:31
0
雪    币:
能力值: ( LV1,RANK:0 )
在线值:
发帖
回帖
粉丝
22
给大佬点赞
2020-3-5 19:57
0
雪    币: 257
活跃值: (105)
能力值: ( LV8,RANK:130 )
在线值:
发帖
回帖
粉丝
23
看到了why:)
在得到密钥后,也可以直接 hook LauncherActivity 类的verifyClick方法
    Java.perform(function(){
                var hook_Activity = Java.use('de.fraunhofer.sit.premiumapp.LauncherActivity');
               hook_Activity.verifyClick.implementation = function(){
                       var editor = this.getApplicationContext().getSharedPreferences("preferences", 0).edit();
                        editor.putString("KEY", "|{yu~iu{iq|yyu~");
                       editor.commit();
               }
    });
最后于 2020-3-7 02:00 被舵手编辑 ,原因:
2020-3-7 01:37
0
雪    币: 7088
活跃值: (2915)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
24
很详细,很赞。支持楼主。
2020-3-29 11:23
0
雪    币: 155
活跃值: (386)
能力值: ( LV2,RANK:10 )
在线值:
发帖
回帖
粉丝
25
很详细,谢谢楼主
2020-4-22 17:00
0
游客
登录 | 注册 方可回帖
返回
//