这篇文章上次修改于 1404 天前,可能其部分内容已经发生变化,如有疑问可询问作者。

title: 提取腾讯视频软字幕
date: 2021/03/13 23:30:00
updated: 2021/03/14 23:30:00
permalink: extract-tencent-video-subtitle/

toc: true


环境

背景和目标

腾讯视频有一个台词海报的功能,该功能可以将一句或者多句台词生成海报,用户可以分享给他人

这个功能出现很久了,使用时稍加注意就可以发现这样的情形:点击台词背景缩略图会变化,也就是说台词有对应视频时间位置

本质上来说,它就是软字幕

腾讯视频拓展海外,平台是WETV,在WETV则直接能从视频接口拿到字幕链接

腾讯国内的视频即使有台词海边这个功能,但视频内容仍然是硬字幕的

不过不重要,关键是如果想要这个台词海报(软字幕)该怎么办呢

这就是本文的目标了

分析

首先找一个有台词海报的视频,大江大河2第1集(通常来说电视剧才有台词海报功能)

台词海报入口

台词海报具体分享页面如下

台词海报页面

既然可以点点点,那不得直接用hookEvent.js安排

然而选择一句台词,并没有反应

选择台词

啊这...不要紧,试试前面的台词海报按钮

有了!定位到的类是

com.tencent.qqlive.ona.player.plugin.operate.helper.VodSwMoreOperateCaptionIconHelper

VodSwMoreOperateCaptionIconHelper

然而这种情况也不知道如何继续追踪...

咋办呢?这个台词内容显然会放到android.widget.TextView

当然是objection一把梭

android hooking watch class_method android.widget.TextView.setText "java.lang.CharSequence" --dump-args

显然!

android.widget.TextView.setText

然后得到一份调用栈

(agent) [135218] Called android.widget.TextView.setText(java.lang.CharSequence)
(agent) [135218] Backtrace:
        android.widget.TextView.setText(Native Method)
        com.tencent.qqlive.ona.share.caption.CaptionEditor.a(CaptionEditor.java:231)
        com.tencent.qqlive.ona.share.caption.CaptionEditor.addData(CaptionEditor.java:173)
        com.tencent.qqlive.ona.share.caption.CaptionEditActivity$2.onCaptionChoiceChange(CaptionEditActivity.java:172)
        com.tencent.qqlive.ona.share.caption.CaptionListAdapter.onLoadFinish(CaptionListAdapter.java:106)
        com.tencent.qqlive.ona.model.base.a$2$1.a(BaseModel.java:51)
        com.tencent.qqlive.ona.model.base.a$2$1.onNotify(BaseModel.java:48)
        com.tencent.qqlive.utils.v.a(ListenerMgr.java:85)
        com.tencent.qqlive.ona.model.base.a$2.run(BaseModel.java:48)
        android.os.Handler.handleCallback(Handler.java:873)
        android.os.Handler.dispatchMessage(Handler.java:99)
        android.os.Looper.loop(Looper.java:201)
        android.app.ActivityThread.main(ActivityThread.java:6882)
        java.lang.reflect.Method.invoke(Native Method)
        com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:547)
        com.android.internal.os.ZygoteInit.main(ZygoteInit.java:873)

(agent) [135218] Arguments android.widget.TextView.setText(当初我把我姐)

关注

com.tencent.qqlive.ona.model.base.a$2.run

既然调用了run

那么肯定要初始化的,那么hook初始化的调用栈

(agent) [989534] Called com.tencent.qqlive.ona.model.base.a$2.a$2(com.tencent.qqlive.ona.model.base.a, com.tencent.qqlive.ona.model.base.a, int, boolean, boolean)
(agent) [989534] Backtrace:
        com.tencent.qqlive.ona.model.base.a$2.<init>(Native Method)
        com.tencent.qqlive.ona.model.base.a.sendMessageToUI(BaseModel.java:44)
        com.tencent.qqlive.ona.model.base.a.sendMessageToUI(BaseModel.java:61)
        com.tencent.qqlive.ona.model.base.d.a(BasePreGetNextPageModel.java:301)
        com.tencent.qqlive.ona.model.base.d.a(BasePreGetNextPageModel.java:279)
        com.tencent.qqlive.ona.circle.c.i.a(FriendsScreenShotModel.java:225)
        com.tencent.qqlive.ona.model.base.d.onProtocolRequestFinish(BasePreGetNextPageModel.java:251)
        com.tencent.qqlive.route.ProtocolManager.onNetWorkFinish(ProtocolManager.java:405)
        com.tencent.qqlive.route.n.a(NetWorkTask.java:377)
        com.tencent.qqlive.route.n.a(NetWorkTask.java:314)
        com.tencent.qqlive.x.i$1.a(NetworkConfig.java:223)
        com.tencent.qqlive.x.i$1.a(NetworkConfig.java:105)
        com.tencent.qqlive.route.r.a(RouteConfig.java:234)
        com.tencent.qqlive.route.r.a(RouteConfig.java:220)
        com.tencent.qqlive.route.n.a(NetWorkTask.java:144)
        com.tencent.qqlive.route.n.a(NetWorkTask.java:122)
        com.tencent.qqlive.route.n.run(NetWorkTask.java:108)
        com.tencent.qqlive.modules.vb.threadservice.a.d.run(VBExecuteRunnable.java:48)
        java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1167)
        java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:641)
        java.lang.Thread.run(Thread.java:764)

(agent) [989534] Arguments com.tencent.qqlive.ona.model.base.a$2.a$2(com.tencent.qqlive.ona.circle.c.i@b9fc8cc, com.tencent.qqlive.ona.circle.c.i@b9fc8cc, (none), true, true)

这个时候观察到一个FriendsScreenShotModel

反编译代码中有一个CircleGetScreenShotsResponse类型

这样一看就有意思了

com.tencent.qqlive.ona.protocol.jce.CircleGetScreenShotsResponse.readFrom这里做更详细的hook

com.qq.taf.jce.JceInputStream的相关函数如下

private <K,V> java.util.Map<K, V> com.qq.taf.jce.JceInputStream.readMap(java.util.Map<K, V>,java.util.Map<K, V>,int,boolean)
private <T> T[] com.qq.taf.jce.JceInputStream.readArrayImpl(T,int,boolean)
private int com.qq.taf.jce.JceInputStream.peakHead(com.qq.taf.jce.JceInputStream$HeadData)
private void com.qq.taf.jce.JceInputStream.skip(int)
private void com.qq.taf.jce.JceInputStream.skipField()
private void com.qq.taf.jce.JceInputStream.skipField(byte)
public <K,V> java.util.HashMap<K, V> com.qq.taf.jce.JceInputStream.readMap(java.util.Map<K, V>,int,boolean)
public <T> T[] com.qq.taf.jce.JceInputStream.readArray(T[],int,boolean)
public <T> java.lang.Object com.qq.taf.jce.JceInputStream.read(T,int,boolean)
public <T> java.util.List<T> com.qq.taf.jce.JceInputStream.readArray(java.util.List<T>,int,boolean)
public boolean com.qq.taf.jce.JceInputStream.read(boolean,int,boolean)
public boolean com.qq.taf.jce.JceInputStream.skipToTag(int)
public boolean[] com.qq.taf.jce.JceInputStream.read(boolean[],int,boolean)
public byte com.qq.taf.jce.JceInputStream.read(byte,int,boolean)
public byte[] com.qq.taf.jce.JceInputStream.read(byte[],int,boolean)
public com.qq.taf.jce.JceStruct com.qq.taf.jce.JceInputStream.directRead(com.qq.taf.jce.JceStruct,int,boolean)
public com.qq.taf.jce.JceStruct com.qq.taf.jce.JceInputStream.read(com.qq.taf.jce.JceStruct,int,boolean)
public com.qq.taf.jce.JceStruct[] com.qq.taf.jce.JceInputStream.read(com.qq.taf.jce.JceStruct[],int,boolean)
public double com.qq.taf.jce.JceInputStream.read(double,int,boolean)
public double[] com.qq.taf.jce.JceInputStream.read(double[],int,boolean)
public float com.qq.taf.jce.JceInputStream.read(float,int,boolean)
public float[] com.qq.taf.jce.JceInputStream.read(float[],int,boolean)
public int com.qq.taf.jce.JceInputStream.read(int,int,boolean)
public int com.qq.taf.jce.JceInputStream.setServerEncoding(java.lang.String)
public int[] com.qq.taf.jce.JceInputStream.read(int[],int,boolean)
public java.lang.String com.qq.taf.jce.JceInputStream.read(java.lang.String,int,boolean)
public java.lang.String com.qq.taf.jce.JceInputStream.readByteString(java.lang.String,int,boolean)
public java.lang.String com.qq.taf.jce.JceInputStream.readString(int,boolean)
public java.lang.String[] com.qq.taf.jce.JceInputStream.read(java.lang.String[],int,boolean)
public java.nio.ByteBuffer com.qq.taf.jce.JceInputStream.getBs()
public java.util.List com.qq.taf.jce.JceInputStream.readList(int,boolean)
public java.util.Map<java.lang.String, java.lang.String> com.qq.taf.jce.JceInputStream.readStringMap(int,boolean)
public long com.qq.taf.jce.JceInputStream.read(long,int,boolean)
public long[] com.qq.taf.jce.JceInputStream.read(long[],int,boolean)
public short com.qq.taf.jce.JceInputStream.read(short,int,boolean)
public short[] com.qq.taf.jce.JceInputStream.read(short[],int,boolean)
public static int com.qq.taf.jce.JceInputStream.readHead(com.qq.taf.jce.JceInputStream$HeadData,java.nio.ByteBuffer)
public void com.qq.taf.jce.JceInputStream.readHead(com.qq.taf.jce.JceInputStream$HeadData)
public void com.qq.taf.jce.JceInputStream.skipToStructEnd()
public void com.qq.taf.jce.JceInputStream.warp(byte[])
public void com.qq.taf.jce.JceInputStream.wrap(byte[])

结合反编译代码,最终的操作都是落在了java.nio.ByteBuffer

那在readFrom时直接那一份二进制内容出来吧!

尝试了很多(省略N多过程),然后发现这个点没有找到数据

还是要在最开始找到数据的最近的地方追踪

com.tencent.qqlive.ona.share.caption.CaptionEditor进行追踪

  • com.tencent.qqlive.ona.share.caption.CaptionEditor.a
  • com.tencent.qqlive.ona.share.caption.CaptionEditor.addData

可以发现在点击选择其他字幕的时候,会调用addData

(agent) [ouzhno7bvfi] Called com.tencent.qqlive.ona.share.caption.CaptionEditor.addData(int, com.tencent.qqlive.ona.protocol.jce.CaptionInfo)
(agent) [ouzhno7bvfi] Arguments com.tencent.qqlive.ona.share.caption.CaptionEditor.addData(1, "<instance: com.tencent.qqlive.ona.protocol.jce.CaptionInfo>")

然后找一个com.tencent.qqlive.ona.protocol.jce.CaptionInfo实例看看

那现在应该继续追踪com.tencent.qqlive.ona.protocol.jce.CaptionInfo

调用栈如下

(agent) [o1dgw84u3vm] Called com.tencent.qqlive.ona.protocol.jce.CaptionInfo.CaptionInfo()
(agent) [o1dgw84u3vm] Backtrace:
        com.tencent.qqlive.ona.protocol.jce.CaptionInfo.<init>(Native Method)
        java.lang.Class.newInstance(Native Method)
        com.qq.taf.jce.JceInputStream.read(JceInputStream.java:933)
        com.qq.taf.jce.JceInputStream.read(JceInputStream.java:978)
        com.qq.taf.jce.JceInputStream.readArrayImpl(JceInputStream.java:839)
        com.qq.taf.jce.JceInputStream.readArray(JceInputStream.java:819)
        com.qq.taf.jce.JceInputStream.read(JceInputStream.java:976)
        com.tencent.qqlive.ona.protocol.jce.GetCaptionResponse.readFrom(GetCaptionResponse.java:54)
        com.tencent.qqlive.route.ProtocolPackage.unPackageJceResponse(ProtocolPackage.java:156)
        com.tencent.qqlive.route.n.a(NetWorkTask.java:312)
        com.tencent.qqlive.x.i$1.a(NetworkConfig.java:223)
        com.tencent.qqlive.x.i$1.a(NetworkConfig.java:105)
        com.tencent.qqlive.route.r.a(RouteConfig.java:234)
        com.tencent.qqlive.route.r.a(RouteConfig.java:220)
        com.tencent.qqlive.route.n.a(NetWorkTask.java:144)
        com.tencent.qqlive.route.n.a(NetWorkTask.java:122)
        com.tencent.qqlive.route.n.run(NetWorkTask.java:108)
        com.tencent.qqlive.modules.vb.threadservice.a.d.run(VBExecuteRunnable.java:48)
        java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1167)
        java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:641)
        java.lang.Thread.run(Thread.java:764)

(agent) [o1dgw84u3vm] Return Value: "(none)"

根据反编译代码可以看到,在readFrom后就会得到字幕

编写hook脚本如下

function HookCaptionInfo(){
    Java.perform(function(){
        let CaptionInfo = Java.use("com.tencent.qqlive.ona.protocol.jce.CaptionInfo");
        let ByteBuffer = Java.use("java.nio.ByteBuffer");
        CaptionInfo.readFrom.overload("com.qq.taf.jce.JceInputStream").implementation = function(inputstream){
            let field_bs = inputstream.class.getDeclaredField("bs");
            field_bs.setAccessible(true);
            let bs = inputstream.bs.value;
            if (flag == false){
                // get raw data from buffer
                flag = true;
                console.log(inputstream);
                let position = bs.position();
                let length = bs.capacity();
                console.log('info', position, length);
                // get a copy of bs
                let copy = ByteBuffer.allocate(bs.capacity());
                bs.position(0)
                copy.put(bs);
                bs.position(position)
                copy.flip();
                // transfer copy to buffer
                let buffer = Java.array('byte', new Uint8Array(length));
                copy.get(buffer);
                send({"buffer":length}, new Uint8Array(buffer));
            }
            let ret = this.readFrom(inputstream);
            return ret;
        }
    })
}
var flag = false
setImmediate(HookCaptionInfo)

然后编写python脚本接收数据,保存到本地之后是这个样子的(第37集)

但是如何解析成更加正常一点的结果呢

继续追踪就是com.qq.taf.jce.JceInputStream

这里面有很多操作,解析它自定义结构体的

尝试还原了一部分,但是发现工作量太大...暂时搁置

继续追踪请求来源,再往前看一个调用栈

也就是com.tencent.qqlive.route.ProtocolPackage.unPackageJceResponse

嗯哼,请求就是这里在发起了

可以看到有一个buildPostData方法,追踪它

这样一来就可以确认com.tencent.qqlive.route.jce.RequestCommand.writeTo(com.qq.taf.jce.JceOutputStream)

就是需要关注的,依葫芦画瓢,把请求的内容拿下来

可以看到熟悉的参数,红线的参数就是视频的cid和vid了

其他部分含有大量的环境信息,已水印

  • IMEI
  • 手机型号
  • APP安装时间
  • APP版本
  • session

这里不用过于关心这些参数怎么来,我的目的是寻找更加上层的位置

达成通过视频cid+vid直接获取返回

追踪com.tencent.qqlive.route.jce.RequestCommand.writeTo

(agent) [890vo80ecxr] Called com.tencent.qqlive.route.jce.RequestCommand.writeTo(com.qq.taf.jce.JceOutputStream)
(agent) [890vo80ecxr] Backtrace:
        com.tencent.qqlive.route.jce.RequestCommand.writeTo(Native Method)
        com.tencent.qqlive.route.ProtocolPackage.jceStructToUTF8Byte(ProtocolPackage.java:177)
        com.tencent.qqlive.route.ProtocolPackage.buildPostData(ProtocolPackage.java:73)
        com.tencent.qqlive.route.t.a(UnifiedProtocolUtils.java:47)
        com.tencent.qqlive.route.n.f(NetWorkTask.java:194)
        com.tencent.qqlive.route.n.run(NetWorkTask.java:97)
        com.tencent.qqlive.modules.vb.threadservice.a.d.run(VBExecuteRunnable.java:48)
        java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1167)
        java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:641)
        java.lang.Thread.run(Thread.java:764)

(agent) [890vo80ecxr] Arguments com.tencent.qqlive.route.jce.RequestCommand.writeTo("<instance: com.qq.taf.jce.JceOutputStream>")

com.tencent.qqlive.route.t.a方法如下

com.tencent.qqlive.route.t.a

这里显然还不够顶层,仍然是在操作基础数据

com.tencent.qqlive.route.n.f方法中有一个packageRequest,一并追踪

com.tencent.qqlive.route.n.f

可以发现packageRequest有参数是com.qq.taf.jce.JceStruct

实现类是com.tencent.qqlive.ona.protocol.jce.WatchRecordUploadV1Request

还是JceStruct,那么依然不够顶层

再往上看就是com.tencent.qqlive.route.n.run了,那么关注一下它的初始化

得到一份调用栈

(agent) [oo4t2ej42c] Called com.tencent.qqlive.route.n.n(com.tencent.qqlive.route.jce.ServerInfo, int, int)
(agent) [oo4t2ej42c] Backtrace:
        com.tencent.qqlive.route.n.<init>(Native Method)
        com.tencent.qqlive.route.ProtocolManager.a(ProtocolManager.java:380)
        com.tencent.qqlive.route.ProtocolManager.a(ProtocolManager.java:338)
        com.tencent.qqlive.route.ProtocolManager.sendRequest(ProtocolManager.java:304)
        com.tencent.qqlive.route.ProtocolManager.sendRequest(ProtocolManager.java:268)
        com.tencent.qqlive.route.ProtocolManager.sendRequest(ProtocolManager.java:218)
        com.tencent.qqlive.route.ProtocolManager.sendRequest(ProtocolManager.java:120)
        com.tencent.qqlive.ona.share.caption.GetCaptionModel.loadData(GetCaptionModel.java:59)
        com.tencent.qqlive.ona.share.caption.CaptionListAdapter.loadData(CaptionListAdapter.java:64)
        com.tencent.qqlive.ona.share.caption.CaptionEditActivity.f(CaptionEditActivity.java:427)
        com.tencent.qqlive.ona.share.caption.CaptionEditActivity.onCreate(CaptionEditActivity.java:121)
        android.app.Activity.performCreate(Activity.java:7232)
        android.app.Activity.performCreate(Activity.java:7221)
        android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1272)
        com.tencent.qqlive.ac.a.a$b.callActivityOnCreate(ActivityLifeCycleMonitor.java:104)
        android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2971)
        android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3126)
        android.app.servertransaction.LaunchActivityItem.execute(LaunchActivityItem.java:78)
        android.app.servertransaction.TransactionExecutor.executeCallbacks(TransactionExecutor.java:108)
        android.app.servertransaction.TransactionExecutor.execute(TransactionExecutor.java:68)
        android.app.ActivityThread$H.handleMessage(ActivityThread.java:1846)
        android.os.Handler.dispatchMessage(Handler.java:106)
        android.os.Looper.loop(Looper.java:201)
        android.app.ActivityThread.main(ActivityThread.java:6882)
        java.lang.reflect.Method.invoke(Native Method)
        com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:547)
        com.android.internal.os.ZygoteInit.main(ZygoteInit.java:873)

(agent) [oo4t2ej42c] Arguments com.tencent.qqlive.route.n.n("<instance: com.tencent.qqlive.route.jce.ServerInfo>", 103, 3)

可以看到ProtocolManagersendRequest多次重载,显然不需要关心它

看看com.tencent.qqlive.ona.share.caption.CaptionListAdapter.loadData

看来是合适的地方了,验证一下

很好!

要不要再往前一个栈看呢?我的答案是不需要了

因为可以看到com.tencent.qqlive.ona.share.caption.CaptionListAdapteronCreate方法

显然这会涉及到界面了,那要主动构造调用就比较麻烦

现在仔细看看com.tencent.qqlive.ona.share.caption.CaptionListAdapter

可以看到初始化后就调用了loadData方法

那么主动构造参数调用就比较简单了

vid和cid好说,有两个数字是怎么来的呢?

结合CaptionListAdapter稍作分析就能知道

  • 第一个是点击台词海报按钮时观看视频的位置(秒)
  • 第二个是视频总时长(毫秒)

具体解析也不用自己动手了,直接通过GetCaptionModel的属性去把结果取出来,然后转换成为需要的格式

...

com.tencent.qqlive.route.k.a进行overload可以得到解密后的结果

com.tencent.qqlive.ona.protocol.jce.GetCaptionResponse进行overload可以得到解析好的结果

...

本来这里有很多分析过程,但是没有记录,直接上代码了

下面的hook代码作用是截获点击台词海报按钮时,服务器返回的数据

这里需要复制一份解密(?)后的数据,需要处理一下

flag的作用是只取一次即可,因为一个CaptionInfo对应单条台词,这里会多次进入

function HookCaptionInfo(){
    Java.perform(function(){
        let CaptionInfo = Java.use("com.tencent.qqlive.ona.protocol.jce.CaptionInfo");
        let ByteBuffer = Java.use("java.nio.ByteBuffer");
        CaptionInfo.readFrom.overload("com.qq.taf.jce.JceInputStream").implementation = function(inputstream){
            let field_bs = inputstream.class.getDeclaredField("bs");
            field_bs.setAccessible(true);
            let bs = inputstream.bs.value;
            if (flag == false){
                // get raw data from buffer
                flag = true;
                console.log(inputstream);
                let position = bs.position();
                let length = bs.capacity();
                console.log('info', position, length);
                // get a copy of bs
                let copy = ByteBuffer.allocate(bs.capacity());
                bs.position(0)
                copy.put(bs);
                bs.position(position)
                copy.flip();
                // transfer copy to buffer
                let buffer = Java.array('byte', new Uint8Array(length));
                copy.get(buffer);
                send({"buffer":length}, new Uint8Array(buffer));
            }
            let ret = this.readFrom(inputstream);
            return ret;
        }
    })
}
var flag = false
setImmediate(HookCaptionInfo)

然后经过N久的分析、跳转、寻找、定位(太乱了以至于没有记录)...

终于找到了合适的hook点,可以达成主动调用,获取任意视频的台词海报内容(字幕,其实是腾讯自己OCR的,有一些错误)

不过不完全是主动调用,请求是主动调用,获取返回结果是通过GetCaptionResponse.readFrom处拦截的

下面是代码,用到了三方库,需要使用frida-compile进行编译,然后才能用于注入

const Koa = require('koa');
const Router = require('koa-router');

const server = new Koa();
const router = new Router();

router.get('/subtitle/:vid', (ctx, next) => {
    ctx.body = GetSubtitle(ctx.params.vid, '');
    ctx.type = 'application/json; charset=utf-8';
});

server.use(router.routes()).use(router.allowedMethods()).listen(16666);

function GetSubtitle(vid, cid) {
    var resp = {"name": "", "subtitles": new Array()};
    Java.perform(function () {
        function CaptionListToSRT(CaptionList) {
            for (let i = 0; i < CaptionList.size(); i++) {
                let info = Java.cast(CaptionList.get(i), CaptionInfo);
                let item = {
                    "number": info.captionId.value,
                    "time": info.captionTime.value,
                    "caption": info.caption.value,
                    "status": info.status.value,
                    "duration": info.captionDispearTime.value,
                }
                resp.subtitles.push(item);
            }
        }
        let CaptionInfo = Java.use("com.tencent.qqlive.ona.protocol.jce.CaptionInfo");
        let GetCaptionModel = Java.use("com.tencent.qqlive.ona.share.caption.GetCaptionModel");
        let GetCaptionResponse = Java.use("com.tencent.qqlive.ona.protocol.jce.GetCaptionResponse");
        GetCaptionResponse.readFrom.overload('com.qq.taf.jce.JceInputStream').implementation = function (a) {
            let ret = this.readFrom(a);
            resp.name = this.title.value;
            CaptionListToSRT(this.captionList.value);
            return ret
        }
        let model = GetCaptionModel.$new();
        model.loadData("vid=" + vid, 0, 0, cid, vid);
        Thread.sleep(0.8);
    })
    return resp;
}

编译建议配合frida-agent-example食用

最终效果图