宏观剖析Tinker整体玩法

最近根据基于Tinker1.9.14.7做了一套热修复框架,对tinker做了一些学习研究,结合自己之前framework经验,理解起来还比较快,产出8篇文章,内容牵扯到的android源码是基于Android Q的:
热修复框架 - 从Tinker 1.9.14.7开始
热修复框架 - TinkerApplication启动(一) - 初始化过程
热修复框架 - TinkerApplication启动(二) - 加载dex补丁过程
热修复框架 - TinkerApplication启动(三) - 加载资源补丁过程
热修复框架 - TinkerApplication启动(四) - 加载so补丁过程
热修复框架 - Tinker 安装流程分析
热修复框架 - Tinker patch合成流程
热修复框架 - Tinker disable逻辑梳理
从Tinker加载dex补丁看动态加载插件过程

本篇文章来做一个最后的总结,目的是梳理出整体脉络,对不管了是做过tinker热修复还是想了解的朋友,提供一点点小思路。

一、Tinker整体玩法

核心内容主要分4个部分:

  • 新旧包根据差分算法做出diff patch
  • 服务端管理不同基准包对应的diff patch,与客户端指定检验和下发规则。
  • 客户端获取diff patch与当前apk合并,生成修复包。
  • 加载修复包,替换基准包相关内容,到修复目的。

二、客户端通过Tinker处理patch包闭环

整体流程包括patch包校验、加载、合成环境准备、合成:


客户端通过Tinker处理patch包闭环
  • 2.1 校验、合成:TinkerApplication初始化过程:
    通过TinkerLoader.tryLoader对patch包进行校验,这里也包括tinker功能设置的校验等,如果patch包存在,且有效且tinker相关功能enable则尝试加载patch;

  • 2.2 合成环境准备:Tinker.install
    通过Tinker.install部署好Tinker合成patch包的整体环境:

    @Override
    public void onBaseContextAttached(Context base) {
        super.onBaseContextAttached(base);
        Log.d(TAG, "HotFixApplicationLike onBaseContextAttached");

        MultiDex.install(base);//使应用支持分包

        LoadReporter loadReporter = new DefaultLoadReporter(base);
        PatchReporter patchReporter = new DefaultPatchReporter(base);
        PatchListener patchListener = new DefaultPatchListener(base);
        AbstractPatch upgradePatchProcessor = new UpgradePatch();

        TinkerInstaller.install(this,
                loadReporter,//加载合成的包的报告类
                patchReporter,//打修复包过程中的报告类
                patchListener,//对修复包最开始的检查
                DefaultTinkerResultService.class, //patch包合成完成的后续操作服务
                upgradePatchProcessor);//生成一个新的patch合成包
    }

2.3 合成:TinkerInstaller.onReceiveUpgradePatch()
对服务端下发的patch包和当前基准包进行合成,生成合成包:tinker_classN.apk

对Tinker宏观玩法有个全局认识之后,再来看看tinker的局部玩法,总共3个点:

  • Dexdiff算法如何做patch dex;
  • 如何加载修复补丁(包括dex、resource、so);
  • 如何设计与服务端交互。

下面一个个来看。

三、dexDiff算法

DexDiff是微信自研的差分包算法。

粗略理解差分思路:

dex文件结构

将dex包的关键section读取封装为对象,新旧包对比时,每个section对应一个算法比较器:

各section计算diff的算法
from dodola tinker

以stringDataSection为例:

先新旧数据排序,然后以compareTo比较字符串内容,以二路归并的方式,整理出带del、add、replace标签的diff内容。然后重新计算index和offset。

最终将修改的内容重新写入新文件,生成patch包。

整体dexdiff流程

读取old dex和new dex文件包装为Dex,经过DexPathGenerator处理,dex不同的section对应不同diff algorithm算法处理器处理,算法经过对新旧数据排序,然后通过二路归并的方式,由compareTo进行内容对比,打出del、add、replace标签(这个算法我玩了下,确实有意思),将一个个item封装为PatchOPeration,加入集合PatchOperationList,这部分属于内容分别。然后计算出对应section的size即为:patchedSectionSize。然后通过DexPathGenerator执行executeAndSaveTo 方法,分别计算各区域的offset、收集各区域的PatchOperationList, 最终创建一个新的patch dex按dex格式写入如上内容。然后将dex 、assets和META-INF/ 打成apk。这就是diff出来的差分包。其中,assets中package描述包信息,其他三个文件分别记录dex、res、so的相关信息,会在tinker加载补丁的时候做验证用。

我这姑且班门弄斧地尝试总结了下dexdiff的大概过程,其中的细节非常多,也非常难,坑也非常多,典型的包括Android N混合编译、厂商OTA后因为补丁包过大造成编译卡顿问题等等,dexDiff是Tinker最难的技术点,微信自己人也说这条路是跪着走完的。

四、加载修复补丁简介

dex:

hook classLoader对应的dexPathList中的makeDexElements,将修复dex插入dexElements最前面,保证相同类加载修复dex中的,而其后的失效。

so:
hook classLoader对应的dexPathList中的makePathElements方法,注入so到nativeLibraryPathElements数组中。这部分跟dexElements类似。

res:
替换LoadedApk对应的mResDir指向补丁包,ResourcesImpl mAssets替换为新的AssetManager,并用新AssetManager调用其addAssetPath加载补丁包。

其中:

  • LoadedApk是 APK文件信息封装对象。它在ActivityThread启动过程中被初始化,参与Apk加载过程。
  • Resource通过代理ResourceImpl来处理,ResourceImpl先尝试从缓存获取,没有缓存在通过AssetManager获取,AssetManager通过addAssetPath来加载apk资源,而具体实现是在native做的。

类关系如下图:

五、如何设计与服务端的交互

热修复patch在商业化项目中一般会通过服务端下发的方式给到app去合成,然后加载。这里就简单介绍下我做的一版:

上传:向服务端上传patch包策略文件,以及patch包。

下发
1.check下当前基准包是否有patch包,通过版本号、渠道号、包名等几个维度来唯一定位。
2.获取策略文件域名,拉取策略文件。
3.根据策略文件来判断拉取哪个patch,是做新修复还是回滚,回滚是回滚到哪个版本。
4.拉取对应的patch包,并对包做校验。
5.如果是做小需求发版,还会拉取引导图,在开机时设置引导页面。
6.客户端合成patch包。
7.设计重启策略,最暴力的是直接kill。

patch下发流程

六、如何排查Tinker问题

最后,再来简单介绍下客户端相关Tinker问题如何排查
前面介绍了:TinkerApplication在初始化的时候,会执行TinkerLoader.tryLoad,它主要做两部分事情,校验与加载修复包。

这个校验过程,如果报错,会通过如下方法设置错误码:

ShareIntentUtil.setIntentReturnCode(resultIntent, ShareConstants.ERROR_LOAD_DISABLE);

然后到TinkerApplication初始化完成后后执行:Tinker.install,它通过如下方法解析TinkerApplication启动过程中反馈的加载补丁结果:

tinkerLoadResult.parseTinkerResult(getContext(), intentResult);

然后通过tinker.getLoadReporter()返回对应的回调,如果客户端注册了LoadReporter,会收到对应的回调,可以在这做打印。当然tryLoad 设置错误码时就会有相应的打印,直接通过错误码去反推出现的问题,能瞬间缩小排除范围。

举例:
我在写demo时遇到的问题:

I/Tinker.TinkerLoadResult: parseTinkerResult loadCode:-3, process name:com.stan.tinkersdkdemo, main process:true, systemOTA:false, fingerPrint:Xiaomi/dipper/dipper:9/PKQ1.180729.001/9.10.122:user/test-keys, oatDir:null, useInterpretMode:false

loadCode -3 :对应ERROR_LOAD_PATCH_INFO_NOT_EXIST
看看是什么原因设置的这种状态码:

if (!patchInfoFile.exists()) {
   Log.w(TAG, "tryLoadPatchFiles:patch info not exist:" + patchInfoFile.getAbsolutePath());
   ShareIntentUtil.setIntentReturnCode(resultIntent, ShareConstants.ERROR_LOAD_PATCH_INFO_NOT_EXIST);
   return;
}

patch.info不存在, adb查看下果然是,patch.info生成是在合成patch的地方,tinker通过如下方法合成patch,合成过程会生成tinker文件夹,以及内部的相关文件

TinkerInstaller.onReceiveUpgradePatch(getApplicationContext(), patch);

那就debug下这个流程就好了,最终问题是对应patch合成的service没有在manifest注册,因为patch合成任务是在service中做的,所以问题解决。

我这里只是提供一种简单的思路。

附:tinker文件夹:

cepheus:/data/data/com.stan.tinkersdkdemo/tinker # ls -al
total 36
drwx------ 3 u0_a350 u0_a350 4096 2020-08-12 10:14 .
drwx------ 8 u0_a350 u0_a350 4096 2020-08-12 10:14 ..
-rw------- 1 u0_a350 u0_a350    0 2020-08-12 10:15 info.lock
drwx------ 4 u0_a350 u0_a350 4096 2020-08-12 10:14 patch-8b79c8cc
-rw------- 1 u0_a350 u0_a350  367 2020-08-12 10:15 patch.info // patch信息描述文件

cepheus:/data/data/com.stan.tinkersdkdemo/tinker/patch-8b79c8cc # ls -al
total 40
drwx------ 4 u0_a350 u0_a350 4096 2020-08-12 10:14 .
drwx------ 3 u0_a350 u0_a350 4096 2020-08-12 10:14 ..
drwx------ 3 u0_a350 u0_a350 4096 2020-08-12 10:14 dex
drwx------ 2 u0_a350 u0_a350 4096 2020-08-12 10:14 odex
-rw------- 1 u0_a350 u0_a350 3443 2020-08-12 10:14 patch-8b79c8cc.apk //合成前的diff patch包

cepheus:/data/data/com.stan.tinkersdkdemo/tinker/patch-8b79c8cc/dex # ls -al
total 2328
drwx------ 3 u0_a350 u0_a350    4096 2020-08-12 10:14 .
drwx------ 4 u0_a350 u0_a350    4096 2020-08-12 10:14 ..
drwxrwx--x 3 u0_a350 u0_a350    4096 2020-08-12 10:14 oat
-rw------- 1 u0_a350 u0_a350 2351677 2020-08-12 10:14 tinker_classN.apk //合成后的patch包

当然tinker绝对不止这么点东西,能力有限,只简单窥探了冰山一角,文中错误之处欢迎批评指正!

参考:
Tinker
微信Tinker的一切都在这里,包括源码(一)
Android热修复Tinker源码分析之DexDiff / DexPatch
tinker github
全面解析Android热修复原理
Tinker源码分析(一):TinkerApplication

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351