吾爱破解 - 52pojie.cn

 找回密码
 注册[Register]

QQ登录

只需一步,快速开始

查看: 18208|回复: 113
收起左侧

[Android 原创] Android微信逆向--实现发朋友圈动态

    [复制链接]
luoyesiqiu 发表于 2020-3-20 23:28
本帖最后由 luoyesiqiu 于 2020-3-21 12:19 编辑

0x0 前言

最近一直在研究Windows逆向的东西,想着快要把Android给遗忘了。所以就想利用工作之余来研究Android相关的技术,来保持对Android热情。调用微信代码来发送朋友圈动态一直是自己想实现的东西,研究了一下,果然实现了,遂写下本文当作记录。本文主要分析发送纯文字朋友圈动态和发送图片朋友圈动态。

0x1 朋友圈动态类型分析

本文用到的工具如下:

  • PC
  • 一台可以调试微信进程的Android手机
  • 微信7.0.11
  • ddms(用于跟踪调用过程)
  • uiautomatorviewer(用于定位控件id)
  • jadx(用于对微信apk静态分析)
  • frida(用于hook微信,获得相关信息,发布朋友圈)

在分析代码之前,首先要定位到与之相近的地方,我们首先想到的肯定是发朋友圈动态那个界面,如何查看发朋友圈动态的界面是哪个Activity呢?很简单,先在手机上打开发表朋友圈动态的界面

截图.png

把手机连接电脑,打开USB调试,在PC的cmd窗口中执行以下命令:

adb shell dumpsys activity top

top.png

可以看到发朋友圈动态的Activity就是com.tencent.mm.plugin.sns.ui.SnsUploadUI

虽然找到了Activity,但还是不能高兴太早,想要通过Activity知道哪部分是发朋友圈的动态代码还是比较费力的。于是我们就想到从“发表”按钮入手,找出发表朋友圈动态的相关代码。点击“发表”按钮会发生什么?发表是一个动态的行为,我们可以通过跟踪点击“发表”按钮时的调用过程,来找到有用的信息。跟踪调用过程,可以使用ddms工具来完成。打开ddms,选中微信进程,在手机中打开发表朋友圈界面,然后在ddms中点击下图圈出的图标开始跟踪:
微信进程.png

将朋友圈动态发出,再点一次上图圈出的图标停止跟踪。ddms会生成跟踪结果,对于跟踪结果,怎么找到按钮事件相关的信息呢,学过Android的朋友就会想到onClick方法,那我们就在ddms的搜索结果中搜索这个名称:

onClick.png

成功的定位到了onClick的位置,但是比起这条onClick结果,更加令人引人注目的是它的上一条结果,因为它包含了我们刚才找到的Activity的类名:

onMemuItemClick.png

知道这个方法被调用,我们去看看com.tencent.mm.plugin.sns.ui.SnsUploadUI类里的OnMemuItemClick究竟是什么。

用jadx打开微信apk,定位到com.tencent.mm.plugin.sns.ui.SnsUploadUI类,在类中搜索onMemuItemClick,结果不多,看起来比较像的就是这个onMemuItemClick了:

onMenuItemClick函数.png

在onMemuItemClick方法中看到了:

String unused2 = SnsUploadUI.this.desc = SnsUploadUI.this.tQN.getText().toString();

这行代码有什么特别的呢,在我看来,有两个特别的地方:

  • desc是描述(description)的英文单词的缩写
  • this.desc被赋予this.tQN.getText().toString()

我们发朋友圈动态时候,是要写动态的描述的,所以这个desc可能就是发朋友圈动态的描述,如果是描述,我们就可以根据这个描述,顺藤摸瓜找到发朋友圈动态的地方。而且this.desc的值又来自于this.tQN.getText().toString(),即this.tQN很有可能就是我们填写动态描述的文本框。我们来看看this.tQN赋值的地方,它在onCreate方法被赋值:

文本框变量赋值.png

可以知道它的id是d41,那么d41是哪个控件?打开uiautomatorviewer,对发朋友圈界面截图分析,点击截图中的文本框,uiautomatorviewer右侧跳转到了相应的位置,果然,d41就是发动态时填写描述文本框的id

文本框id.png

好了,现在知道this.desc就是发表朋友圈动态时的描述,跟上他应该就可以找到发朋友圈动态的地方。继续往onMemuItemClick方法下部分析,可以看到this.desc被传入了SnsUploadUI.this.tQO.a方法:

看到发朋友圈函数.png

SnsUploadUI.this.tQO.a方法定义在接口com.tencent.mm.plugin.sns.ui.z中:

z

z

z_interface

z_interface

知道它定义在哪个接口并不能解决问题,毕竟接口没有实质性代码,要找还得找接口的的实现类,在com.tencent.mm.plugin.sns.ui.SnsUploadUI类中寻找this.tQO在哪里会被赋值。最终,我们在com.tencent.mm.plugin.sns.ui.SnsUploadUI类的ag方法中看到了许多给this.tQO赋值的地方:

tQO赋值的地方.png

由此,可见this.tQO被赋予什么值是根据this.tMY来决定的,this.tMY是一个int类型的数据,那我们hook com.tencent.mm.plugin.sns.ui.SnsUploadUI类的ag方法就可以知道this.tMY是什么值。在这里,我用frida来hook,frida的javascript部分代码如下:

var SnsUploadUI= Java.use('com.tencent.mm.plugin.sns.ui.SnsUploadUI');
var ag = SnsUploadUI.ag.overload("android.os.Bundle");
//get sns type
ag.implementation=function(bundle){
    var ret = ag.call(this,bundle);
    send("sns type = " + this.tMY.value);
    return ret;
}

hook之后,每当我们在手机上打开发布朋友圈动态的界面,ag方法被调用,控制台就会输出相应的数字。经过我的测试,这个数字是发表朋友圈动态的类型。朋友圈类型和其对应类如下:

  • 0 带图片的动态,对应:com.tencent.mm.plugin.sns.ui.ai
  • 9 纯文字动态,对应:com.tencent.mm.plugin.sns.ui.ae

这些类都直接或间接的实现了上面讲到的com.tencent.mm.plugin.sns.ui.z接口。这样一来,就知道this.tQO会根据朋友圈的动态类型进行初始化,那么,上面的SnsUploadUI.this.tQO.a方法很有可能就是发朋友圈动态的方法。接下来,我们根据不同的朋友圈动态所对应的类来分别分析

0x2 文字动态分析

文字动态分析起来应该比图片动态来说简单一些,我们就先来分析它。上面讲到,这类动态对应类是com.tencent.mm.plugin.sns.ui.ae,这个类里我们主要看a方法,在看a方法之前,先看它传入什么参数,为了看清楚,这就要回看上文onMenuItemClick方法调用a方法的地方:

public final boolean onMenuItemClick(MenuItem menuItem) {
    String unused2 = SnsUploadUI.this.desc = SnsUploadUI.this.tQN.getText().toString();
    int pasterLen = SnsUploadUI.this.tQN.getPasterLen();
    int privated = SnsUploadUI.this.tKm.getPrivated();
    int syncFlag2 = SnsUploadUI.this.tKm.getSyncFlag();
    ......
    PInt pInt = new PInt();
    if (SnsUploadUI.this.tQO instanceof a) {
        Bundle bundle = new Bundle();
        bundle.putInt("param_is_privated", privated);
        bundle.putString("param_description", SnsUploadUI.this.desc);
        bundle.putStringArrayList("param_with_list", new ArrayList(SnsUploadUI.this.uij.getAtList()));
        bundle.putInt("param_paste_len", pasterLen);
        try {
            bundle.putByteArray("param_localtion", SnsUploadUI.this.uik.getLocation().toByteArray());
        } catch (IOException e2) {
            ab.printErrStackTrace("MicroMsg.SnsUploadUI", e2, "parse location error", new Object[0]);
        }
        bundle.putBoolean("param_is_black_group", SnsUploadUI.this.tQS);
        bundle.putStringArrayList("param_group_user", SnsUploadUI.this.tQR);
        bundle.putInt("param_contact_tag_count", SnsUploadUI.this.tOk);
        bundle.putInt("param_temp_user_count", SnsUploadUI.this.tOl);
        pInt.value = ((a) SnsUploadUI.this.tQO).getContentType();
        z unused4 = SnsUploadUI.this.tQO;
    } else {
        SnsUploadUI.this.tQO.a(privated, syncFlag2, SnsUploadUI.this.tKm.getTwitterAccessToken(), SnsUploadUI.this.desc, SnsUploadUI.this.uij.getAtList(), SnsUploadUI.this.uik.getLocation(), (LinkedList<Long>) null, pasterLen, SnsUploadUI.this.tQS, SnsUploadUI.this.tQR, pInt, SnsUploadUI.this.tOj, SnsUploadUI.this.tOk, SnsUploadUI.this.tOl);
    }
}

这就是a方法调用的地方,根据这段代码和编写hook a方法的代码来推出它的参数。hook代码如下:

var ae = Java.use('com.tencent.mm.plugin.sns.ui.ae');
var ae_a = ae.a.overload("int","int","org.b.d.i","java.lang.String","java.util.List","com.tencent.mm.protocal.protobuf.bdi","java.util.LinkedList","int","boolean","java.util.List","com.tencent.mm.pointers.PInt","java.lang.String","int","int");
ae_a.implementation = function(isPrivate,syncFlag2,twitterAccessToken,desc,atList,location,list1,pasterLen,bool1,list2,pint1,str1,num1,num2){
    var ret = ae_a.call(this,isPrivate,syncFlag2,twitterAccessToken,desc,atList,location,list1,pasterLen,bool1,list2,pint1,str1,num1,num2);
    console.log("************Basic Info************");
    console.log("isPrivate = " + isPrivate);
    console.log("syncFlag2 = " + syncFlag2);
    console.log("twitterAccessToken = " + twitterAccessToken);
    console.log("desc = " + "'" + desc + "'");
    if(atList.size()>0){
        for(var i=0;i<atList.size();i++){
            console.log("atList[" + i + "] = " + atList.get(0));
        }
    }
    if(location != null){

        if(location.yRD.value != null){
            console.log("location.yRD = " + location.yRD.value);
        }

        if(location.yRE.value != null){
            console.log("location.yRE = " + location.yRE.value);
        }

    }
    console.log("list1 = " + list1);
    console.log("pasterLen = " + pasterLen);
    console.log("bool1 = " + bool1);
    if(list2 != null){
        console.log("list2 = " + list2.size());
    }
    else{
        console.log("list2 = " + list2);
    }
    console.log("pint1 = " + pint1.value.value);
    console.log("str1 = " + str1);
    console.log("num1 = " + num1);
    console.log("num1 = " + num1);

    return ret;
}//ae.a

hook成功后,发一条纯文字的朋友圈动态,打印出:

发布带图片的朋友圈的基本信息.png

所以,可得:

- privated(int):动态是否私密:0公开,1私密
- desc(String):朋友圈的文本
- AtList(List<String>):艾特人的wxid
- Location(com.tencent.mm.protocal.protobuf.bdi):定位信息

好多参数我们不知道是什么,不过问题不大,那些我们需要的参数已经能搞懂了。懂得a方法的参数,那能否尝试直接调用它?先来看一下com.tencent.mm.plugin.sns.ui.ae类的构造函数能否调用:

ae构造函数.png

构造函数有Activity类型的参数,Activity类型的参数是很难构造的,所以放弃构造com.tencent.mm.plugin.sns.ui.ae类来调用a方法。那我们直接去看a方法,看能不能找到有用的东西。由于是文字动态,所以我们着重关注传入的文本,即com.tencent.mm.plugin.sns.ui.SnsUploadUI类的desc成员,在a方法中它是第4个参数:

a方法第4个参数.png

可见this.desc在a方法中它是str,而且在a方法中,str只有一处引用:

ae的a方法的str的引用.png

str传给了ayVar.adk()方法,找一下ayVar来自哪里,它在a方法里初始化,而且初始化方式很简单:

ay类的初始化.png

只传入一个数字就能初始化,我们初始化ay类的时候不用深究这个数字是什么,和它传入一样的值即可。在a方法的尾部,还看到一个引人注目的commit方法:

ae的a方法调用commit.png

猜测这就是发布朋友圈的方法,写个简单的frida脚本来验证一下:

if(Java.available)
{
    Java.perform(function(){
        var ay_class = Java.use("com.tencent.mm.plugin.sns.model.ay");
        var desc = "To be, or not to be, that is a question.";
        var ayInstance = ay_class.$new(2);
        ayInstance.adk(desc);
        ayInstance.commit();
    });
}

文字动态的内容是:To be, or not to be, that is a question.。果不其然,脚本运行后,文字动态发表成功了

文字朋友圈发送成功.jpg

经过对com.tencent.mm.plugin.sns.ui.ae的a方法的分析,我们可以知道,a方法主要传入一些发朋友圈动态所需要的通用的数据,比如动态是否私密,动态的文字描述,艾特的人,定位信息等,这些信息在其他类型的朋友圈动态中也会用得到。我们还知道发文字动态只需要文字描述就能发表成功。

0x3 带图片的朋友圈动态分析

带图片的的动态和文字动态差不多,只是多加了图片的参数,我们在分析此类动态时多关注图片在哪传入即可。带图片的动态对应的类是com.tencent.mm.plugin.sns.ui.ai,有了上面的经验,我们直接去看它的a方法(ai类有许多a方法,注意这里说的a方法参数和com.tencent.mm.plugin.sns.ui.z接口里的a方法参数一致)。在a方法的开头,看到利用迭代器去遍历一个列表,遍历过程中组装com.tencent.mm.plugin.sns.data.j类的数据,然后把j类放入链表linkedList2中:

ai_a循环.png
在组装数据的时候,我们看到j类构造时传入一个字符串和数字,而这个字符串对应j类的path字段,这可能就是图片的路径:

j类.png

那么我们猜测j类就是存储朋友圈的动态图片信息的类,上面提到j类被放入链表linkedList2中,那么来看linkedList2被哪里引用了

ai图片路径被打印出来.png

看到醒目的字符串:commit pic size,这应该是日志要打印的字符串,现在基本上可以确定j类就是存储要发表的图片的信息的类了,那么linkedList2就是存储所有将要发表的图片信息,继续往下寻找linkedList2还被哪里引用了

linklist2传入.png

可以看到linkedList2传入两个地方,一处传入a方法:

ai_a设置图片1.png

另一处传入com.tencent.mm.plugin.sns.ui.ai$a类构造函数:

ai类的a内部类.png

linkedList2传入a类后,又赋值给成员变量tPF,这个tPF成员变量只在a类的dU方法中被引用

ai_a类的dU方法.png

而dU方法在哪里调用呢?在a类的父类:com.tencent.mm.plugin.sns.model.h中,我们看到dU方法在u方法被调用:

a_dU方法被调用.png

而u方法在ai类的a方法中调用(可以回看前面的图)。分析到这,上面的linkedList2传出去之后都终有所属了,即最终都传入了com.tencent.mm.plugin.sns.model.ay类ey方法。知道图片往哪传了,就写段frida代码调用试试吧

if(Java.available)
{
    Java.perform(function(){
        var ay_class = Java.use("com.tencent.mm.plugin.sns.model.ay");
        var j_class = Java.use("com.tencent.mm.plugin.sns.data.j")
        var desc = "To be, or not to be, that is a question.";
        var likedList_class = Java.use("java.util.LinkedList");
        var linkedListInstance = likedList_class.$new();
        var ayInstance = ay_class.$new(1);
        var jInstance1 = j_class.$new("/storage/emulated/0/test1.jpg",2);
        var jInstance2 = j_class.$new("/storage/emulated/0/test2.jpg",2);
        var jInstance3 = j_class.$new("/storage/emulated/0/test3.jpg",2);

        linkedListInstance.add(jInstance1);
        linkedListInstance.add(jInstance2);
        linkedListInstance.add(jInstance3);
        ayInstance.ey(linkedListInstance);
        ayInstance.adk(desc);
        ayInstance.commit();
    });
}

上面的代码在发送文本动态代码的基础上初始化三个j类,分别传入三个本地图片路径,再将三个类实例添加到链表,再将链表传入ay类的ey方法,最后调用ay类的commit方法将动态发送出去,代码运行,发现带图片的朋友圈动态发表成功:

图片朋友圈发送成功.jpg
本文结束


linklist2传入.png
j类.png
a方法第4个参数.png
调用tQO.a.png
发送链接图片构成.png
发送链接传入数据.png
发链接的初始化判断.png
tQO赋值的地方.png
ay类的初始化.png
ai图片路径被打印出来.png
ai类的a内部类.png
ai_a循环.png
ai_a设置图片1.png
ai_a类的dU方法.png
ae的a方法的str的引用.png
ae的a方法调用commit.png
ae构造函数.png

du

du

免费评分

参与人数 50吾爱币 +51 热心值 +38 收起 理由
6603547 + 1 + 1 谢谢@Thanks!
帝都小爬虫 + 1 + 1 我很赞同!
cheng941112 + 1 谢谢@Thanks!
小学生啊 + 1 + 1 感谢,学到了如何用studio进行跟踪。你那好用的jadx能分享下吗?
小飞X + 1 + 1 热心回复!
huaze + 1 + 1 用心讨论,共获提升!
saoker + 1 + 1 用心讨论,共获提升!
wusheren + 1 用心讨论,共获提升!
wlsk888 + 1 + 1 用心讨论,共获提升!
walker6899 + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
abrac + 1 + 1 热心回复!
paragon + 1 + 1 我很赞同!
yiha21 + 1 学会这个, 可以任意在朋友圈发视屏了
生有涯知无涯 + 1 我很赞同!
ShenBohemian + 1 谢谢@Thanks!
武装人员 + 1 + 1 我很赞同!
anyangmvp + 2 + 1 帅的一塌糊涂
litz52001 + 1 谢谢@Thanks!
52PG + 1 + 1 热心回复!
jiabode520 + 1 用心讨论,共获提升!
zxcscm + 1 + 1 谢谢@Thanks!
tony2526 + 1 + 1 我很赞同!
asq56747277 + 1 + 1 我很赞同!
alect + 1 + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
安静1994 + 1 + 1 我很赞同!
newpowersky + 1 + 1 收下我的膝盖!!!!!!!!
笙若 + 2 + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
wowater + 1 + 1 欢迎分析讨论交流,吾爱破解论坛有你更精彩!
还是小白azk + 1 用心讨论,共获提升!
heimianbao + 1 我很赞同!
qwe315333 + 1 + 1 我很赞同!
多凉凉 + 1 + 1 谢谢@Thanks!
kelvar + 1 + 1 我很赞同!
shashenzi + 1 + 1 请问,怎样在朋友圈发一段超长的高清视频?
yy303231 + 1 + 1 我很赞同!
在无世界 + 1 用心讨论,共获提升!
宝塔镇河妖 + 1 + 1 我很赞同!
pojie_nightmare + 1 + 1 鼓励转贴优秀软件安全工具和文档!
LjeA + 1 热心回复!
5omggx + 1 + 1 用心讨论,共获提升!
ke307 + 1 + 1 谢谢@Thanks!
乡熊 + 1 + 1 热心回复!
GenW + 3 + 1 感谢发布原创作品,吾爱破解论坛因你更精彩!
九江入海 + 1 欢迎分析讨论交流,吾爱破解论坛有你更精彩!
yawan18 + 1 + 1 我很赞同!
smile5 + 1 谢谢@Thanks!
裸奔会感冒 + 1 + 1 用心讨论,共获提升!
雨儿 + 1 + 1 热心回复!
莫忘初 + 1 + 1 我很赞同!
15126819695 + 1 我很赞同!

查看全部评分

本帖被以下淘专辑推荐:

发帖前要善用论坛搜索功能,那里可能会有你要找的答案或者已经有人发布过相同内容了,请勿重复发帖。

雨儿 发表于 2020-3-21 01:18
感谢楼主学到了学到了
5omggx 发表于 2020-3-21 11:34
luoyesiqiu 发表于 2020-3-21 10:26
都没听说过,这分析不了啊

这类文件存在于安卓机sdcard/tencent/micromsg/一串16进制数/images2/../../文件夹内,根据网上的说法是苹果手机传的原图就会生成jpg_hevc文件,猜测应该是heif图片格式的变体,但无法用heif转码工具转码。其文件头为wxgf。
貌似微信收到后会在同目录下转码生成一个jpg,不知其内部具体是如何转码的
heikis 发表于 2020-3-20 23:43
freeflys 发表于 2020-3-21 00:12
研究精神令人钦佩~
yywapj 发表于 2020-3-21 00:43
赞个,万物可逆啊,赞
446917139 发表于 2020-3-21 01:37
赞一个 很有用
Hmily 发表于 2020-3-21 01:54
图片盗链不能显示,传本地吧。
 楼主| luoyesiqiu 发表于 2020-3-21 04:25
Hmily 发表于 2020-3-21 01:54
图片盗链不能显示,传本地吧。

图片太多了,传起来好麻烦,要是能在发帖时能把外链图拉到本地就好了

点评

盗链是无法拉回的,看这个图片用discuz的方式贴进去吧https://www.52pojie.cn/misc.php?mod=faq&action=faq&id=29&messageid=36  详情 回复 发表于 2020-3-21 09:19
haidao123 发表于 2020-3-21 05:37
这个可以有
hansensindy 发表于 2020-3-21 06:47
学习一下
您需要登录后才可以回帖 登录 | 注册[Register]

本版积分规则

返回列表

RSS订阅|小黑屋|处罚记录|联系我们|吾爱破解 - LCG - LSG ( 京ICP备16042023号 | 京公网安备 11010502030087号 )

GMT+8, 2025-1-10 11:44

Powered by Discuz!

Copyright © 2001-2020, Tencent Cloud.

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