Android逆向分析实例(三)-解密微信EnMicroMsg

562 阅读3分钟

首先将EnMicroMsg.db文件从模拟器(手机)复制到电脑上(手机或者模拟器要登陆微信):

**adb pull /data/data/com.tencent.mm/MicroMsg/8e1435ec4ddf157ca48ec73b4fc108ac/EnMicroMsg.db C:\Users\lxh\Desktop\WeiXin\EnMicroMsg**
#8e1435ec4ddf157ca48ec73b4fc108ac这个文件夹名称可能不一样,要注意一下。

然后可以这样想:微信在启动时,肯定会连接到这个数据库,连接后需要一个函数以及密码来打开数据库读取其中的内容,我们要做的就是找到这个打开数据库的函数,然后用frida hook 这个函数,打印出密码。

将微信.apk拖入jadx,打开腾讯微信官方api文档:tencent.github.io/wcdb/refere…

因为涉及到数据库的操作函数肯定带有database字段,所以我们直接在页面搜索"database“,经过搜索与分析后可以发现一个openOrCreateDatabase函数:(可以看到第一个参数file应该就是需要打开的数据库文件,第二个参数password就是打开数据库的密码)

于是在jadx中搜索**"openOrCreateDatabase"**,可以得到如下结果:

点进去:

可以看到这个y就是数据库文件名,而str2就是打开数据库的密码,这两个参数传给了openOrCreateDatabase函数,接下来查看**"openOrCreateDatabase"**函数的调用:

可以看到这个函数是个重载函数,其中有一个调用了openDatabase函数,查看其调用

这里的大概意思是:如果str3字符串以**"EnMicroMsg.db"结尾,也就是说如果数据库名为EnMicroMsg.db**,则调用openDatabase(str3, bytes,sQLiteCipherSpec, null, i, fVar, 8)打开这个数据库。接着查找该函数的声明:

到这里我们就可以直接用frida来hook这个函数了,并打印前两个参数(数据库名和密码):

hook.js:

import frida  
import sys    

jscode = """
    Java.perform(function(){  
        var utils = Java.use("com.tencent.wcdb.database.SQLiteDatabase"); // 类的加载路径
        
        utils.openDatabase.overload('java.lang.String', '[B', 'com.tencent.wcdb.database.SQLiteCipherSpec', 'com.tencent.wcdb.database.SQLiteDatabase$CursorFactory', 'int', 'com.tencent.wcdb.DatabaseErrorHandler', 'int').implementation = function(a,b,c,d,e,f,g){   
            console.log("Hook start......");
            var JavaString = Java.use("java.lang.String");
            var database = this.openDatabase(a,b,c,d,e,f,g);
            send(a);
            console.log(JavaString.$new(b));
            send("Hook ending......");
            return database;
        };
        
    });
"""


def on_message(message,data): #js中执行send函数后要回调的函数
    if message["type"] == "send":
        print("[*] {0}".format(message["payload"]))
    else:
        print(message)
    
process = frida.get_remote_device()
pid = process.spawn(['com.tencent.mm']) #spawn函数:进程启动的瞬间就会调用该函数
session = process.attach(pid)  # 加载进程号
script = session.create_script(jscode) #创建js脚本
script.on('message',on_message) #加载回调函数,也就是js中执行send函数规定要执行的python函数
script.load() #加载脚本
process.resume(pid)  # 重启app
sys.stdin.read()

执行代码可以得到如下结果:

这个95af940就是我EnMicroMsg.db数据库的密码,用sqlcipher.exe打开EnMIcroMsg.db,输入密码:

成功打开了数据库,然后可以在message表中看到所有的聊天记录。

3.总结

本次实例的难点就是找那个打开EnMicroMsg.db数据库的函数,以及js中重载函数部分代码的编写,只要函数找对了,找出密码也就近在咫尺了。