Apktool回编译APK变大原因--关于apktool.yml的可压缩文件配置

1.背景

为了实现apk字节码加密技术,我在CI上通过插入了脚本重新打包任务。然而,这改动导致apktool回编译apk包变大,文件经调研和分析发现:在apktool 2.0.3之后为了快速解压和打包,加入了反编译文件回编译不压缩机制。该配置文件位于apktool.yml文件中。

1-1 反编译产物
1-2 apktool.yml详情

几乎所有的二进制和非二进制文件打包时都未采用压缩,导致生成的apk变大(即使未签名依然比原apk大),最后在回编过程进行分析。

在反编译文件中查看yml文件,该文件记录了apk反编译信息(见apktool源码brut/androlib/meta/MetaInfo
compressionType:记录arsc的是否压缩
doNotCompress:记录不压缩的文件类型和路径

2.解法

由于最新版本的apktool回编采用了不压缩的策略,导致我们的重编译apk变大。
可以想到的有两种方法:
1.通过apktool高版本来释放1.apk(framworks apk见上次分享),低版本处理反编译和打包的问题。
2.通过脚本将apktool.yml文件进行修改和定制,可以使得我们的release包更小。

3.YAML格式

我们需要对yaml格式的文件进行解析,有一个支持yaml格式的著名开源库snakeyaml(用java实现)。
我采用jar的形式对这一部分代码(解析YAML和写入文件)进行实现。通过shell脚本插入该任务,从而达到了自动化过程。
snakeyaml的使用参考:
snakeyaml项目地址:https://bitbucket.org/asomov/snakeyaml
snakeyaml java使用:https://www.jianshu.com/p/d8136c913e52

4.CI上打包结果

通过对反编译脚本的优化使得回编译签名包和zipalign包显著减小。

原始apk 反编译签名apk 反编译zipalign-apk
23.90M 22.48M 22.48M
4-1 CI上打包产物

5.脚本实现

#定义out文件的路径
rebApkFile=$cache_dir/rebuildApp.apk
echo "rebuildTools> 修改yml打包的配置文件"
#定义jar的路径,用来提取和修改apktool.yml文件
yamlFileCovert_jar=./yamlFileCovert.jar
#执行yamlFileCovert_jar
java -jar $yamlFileCovert_jar ${apkDecodeDir}/apktool.yml ${apkDecodeDir}/apktool.yml
echo "rebuildTools> 开始回编译apk并替换目标文件..."

apktool b -o ${rebApkFile} ${apkDecodeDir}
echo "rebuildTools> 开始签名新apk..."

...

java部分:

public class YamlFileParse {

    public static void main(String[] args) {
        String fileName;
        String outFile;
        if (args != null && args.length >= 2) {
            fileName = args[0];
            outFile = args[1];
        } else {
            System.out.print("请输入yml路径!\n" +
                    "java -jar yamlFileCovert.jar [in] [out] ");

            return;
        }
        try {
            MetaInfo metaInfo = loadYml(fileName);
            saveYml(outFile, metaInfo);
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    private static void saveYml(String outFile, MetaInfo ret) throws IOException {
        FileOutputStream fos = new FileOutputStream(new File(outFile));
        OutputStreamWriter outputStreamWriter = new OutputStreamWriter(fos, StandardCharsets.UTF_8);
        Writer writer = new BufferedWriter(outputStreamWriter);
        save(ret, writer);
        outputStreamWriter.close();
    }

    private static MetaInfo loadYml(String fileName) throws IOException {
        FileInputStream yamlInput = new FileInputStream(new File(fileName));
        MetaInfo ret = new Yaml().loadAs(yamlInput, MetaInfo.class);
        yamlInput.close();
        ret.doNotCompress = null;
        return ret;
    }

    private static void save(MetaInfo info, Writer output) {
        DumperOptions options = new DumperOptions();
        options.setDefaultFlowStyle(DumperOptions.FlowStyle.BLOCK);
        new Yaml().dump(info, output);
    }
}

java的源码部分包括:yaml格式文件的读写和MetaInfo(用于apktool.yml序列化指定,原因见apktool.yml首行)类。

5-1 java工程的结构

以下是apktool.yml文件,由于文件中包含了MetaInfo类,所以必须指定yml序列化需要的类(这一部分拷贝了apktool源码中部分MetaInfo部分)。

!!brut.androlib.meta.MetaInfo
apkFileName: Rong360-release.apk
compressionType: false
doNotCompress: null
isFrameworkApk: false
packageInfo: {forcedPackageId: '127', renameManifestPackage: null}
sdkInfo: {minSdkVersion: '14', targetSdkVersion: '21'}
sharedLibrary: false
sparseResources: false
usesFramework:
  ids: [1]
  tag: null
version: 2.3.3
versionInfo: {versionCode: '316', versionName: 3.1.6}

运行java -jar $yamlFileCovert_jar ${apkDecodeDir}/apktool.yml对yml进行修改后,回编apk,从而产物的apk大小达到了预期。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,948评论 18 139
  • 学习的方式有很多种,Read the fucking source code 绝对不失为最佳的方式。除了阅读 An...
    亦枫阅读 3,581评论 2 18
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,958评论 6 342
  • 走近了才知道柴窑的魅力,巧夺天工用在这里显得生硬,这是大自然赐予的礼物,浴火重生的生命之杯。 暮春日归人 晚来 陶...
    日照汉唐茶文化阅读 424评论 0 0
  • 最近几天看了一本书,书中谈到一个案例:一个年轻活泼的姑娘总是把自己弄得病恹恹的,借此得到关心。大概是日渐增...
    翶五月阅读 343评论 0 0