Android多渠道打包

深藍旭發表於2020-10-07

使用 Android studio 自帶的打包工具通過 productFlavors 來打多渠道包,效率太低,每次只變更了一個渠道名稱,卻要重頭打包編譯,後來看到 美團多渠道方案,確實很方便,參考別人的程式碼整理了份指令碼,在此記錄一下; 系統: mac 10.12 python: 3.x

要求:

1. 只使用使用v1簽名方式:

P.S. 若使用v2簽名的,可以參考 這篇 ;

android{
    signingConfigs {
        release {
            keyAlias '******'
            keyPassword '******'
            storeFile file('***.jks')
            storePassword '******'
            v2SigningEnabled false // 禁用v2簽名
        }
    }
    buildTypes {
        release {
            signingConfig signingConfigs.release
            ......
        }
    }
}
複製程式碼

或者

generate_signed_apk

2. 第三方庫的渠道名稱可通過程式碼來設定

如我專案中使用的talkingdata,就可以先去查詢 META_INF/ 中的渠道檔案,若有則使用其定義的渠道名來初始化:

private void initTalkingData() {
    String channel = getChannel(this);
    String tdAppId = CommonUtils.getMetaValue(this, "TD_APP_ID");
    if (TextUtils.isEmpty(channel) || TextUtils.isEmpty(tdAppId)) {
        TCAgent.init(this);
    } else {
        TCAgent.init(this, tdAppId, channel);
    }
    TCAgent.setReportUncaughtExceptions(true);
}
複製程式碼

使用

/**
    * 獲取meta_data中指定的渠道號
    * 檔案為: META-INF/tdchannel_{channelName} 
    */
public static String getChannel(Context context) {
    ApplicationInfo appinfo = context.getApplicationInfo();
    String sourceDir = appinfo.sourceDir;
    String ret = "";
    ZipFile zipfile = null;
    try {
        zipfile = new ZipFile(sourceDir);
        Enumeration<?> entries = zipfile.entries();
        while (entries.hasMoreElements()) {
                ZipEntry entry = ((ZipEntry) entries.nextElement());
                String entryName = entry.getName();
            // 檔名與python指令碼定義的相匹配即可
            if (entryName.startsWith("META-INF/tdchannel")) {
                ret = entryName;
                break;
            }
        }
    } catch (Exception e) {
        e.printStackTrace();
    } finally {
        if (zipfile != null) {
            try {
                zipfile.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    String[] split = ret.split("_");
    if (split.length >= 2) {
        return ret.substring(split[0].length() + 1);
    } else {
        return "";
    }
}
複製程式碼

python指令碼

把下面的程式碼儲存為 multi_channel.py 檔案

#!/usr/bin/env python3
# -*- coding: utf-8 -*-

'''
根據美團多渠道打包方式,自動生成多渠道目錄
要求apk是使用V1加密方式打包的;
python3 multi_channel -s srcApkPath -v versionCode
python3 multi_channel -srcFile=srcApkPath --version=versionCode
'''
import os
import shutil
import zipfile
import time
import sys
import getopt

startTime = time.time()

prefixInfo = "release"
srcApk = "./channelApk.apk"
version = ""  # 版本號
channelFilePath = "./channel"  # 渠道配置檔案路徑,每行定義一個渠道

toolInfo = '''參考美團多渠道打包方案1的打包工具;
預設使用當前目錄下 "channel" 檔案中定義的渠道資訊,每行一個渠道名稱,可通過 -c 引數來指定渠道檔案路徑;
要求apk使用的是v1簽名方式,若使用v2則本工具無效;
python3 multi_channel -s srcApkPath -v 1.7 -p demo -c ./channel
-s --srcFile : 新增一個源apk,會依據該apk生成多渠道apk,並儲存於 "./channelApk/" 中;
-v --version : 可選, 給生成的apk名稱新增一個版本號,會自動新增字首 _v{version},如 demo_v1.7.apk;
-p --prefix  : 可選, 給生成的apk名稱新增一個字首資訊,預設為"release"
-c --channel : 定義要生成的渠道包資訊,每行定義一個渠道名稱,會依次生成對應的渠道包'''

opts, args = getopt.getopt(sys.argv[1:], "hs:v:p:c:", ["help", "srcFile=", "version=", "prefix=", "channel="])
for name, value in opts:
    if name in ("-s", "--srcFile"):  # 原始檔名稱
        srcApk = value
    elif name in ("-v", "--version"):  # 版本號
        version = "_v%s" % value
    elif name in ("-p", "--prefix"):  # apk名稱字首資訊
        prefixInfo = value
    elif name in ("-c", "--channel"):  # 多渠道配置檔案
        channelFilePath = value
    elif name in ("-h", "--help"):
        print(toolInfo)
        exit()

print("srcApk = %s , version = %s" % (srcApk, version))

isApkExist = os.path.exists(srcApk)
if not isApkExist or not os.path.isfile(srcApk):
    print("%s 源apk檔案不存在,請重試" % srcApk)
    exit()

if not os.path.exists(channelFilePath) or not os.path.isfile(channelFilePath):
    print("%s channel渠道檔案不存在或者不是有效的file,請檢查後重試" % channelFilePath)
    exit()

pkgPath = os.path.join(os.getcwd(), "channelApk")  # 生成的多渠道apk存放的目錄
print("生成的apk會被存放於 %s" % pkgPath)

isPathExist = os.path.exists(pkgPath)
isDir = os.path.isdir(pkgPath)
if not isPathExist or not isDir:
    os.makedirs(pkgPath)

f = open(channelFilePath, 'r', encoding='utf-8')
for line in f:
    channel_name = line.strip('\n')
    # print("當前正在生成渠道包: %s" % channel_name)
    channelPath = pkgPath + "/{prefix}_{channel}{version}.apk".format(prefix=prefixInfo, channel=channel_name,
                                                                      version=version)
    shutil.copy(srcApk, channelPath)
    zipped = zipfile.ZipFile(channelPath, 'a', zipfile.ZIP_DEFLATED)
    empty_channel_file = "META-INF/tdchannel_{channel}".format(channel=channel_name)
    # zipped.write("empty", empty_channel_file) # 使用這種方式需要在當前目錄下存在empty檔案
    zipped.writestr(empty_channel_file, data=channel_name)
diff = time.time() - startTime
print("耗時: %s" % diff)
複製程式碼

使用時需要定義多渠道版本檔案

channel.png

python3 multi_channel -s ./SrcApk.apk -v 1.0 -p demo -c ./channel
// 就會在 `./channelApk/` 中生成多渠道apk包,apk檔名類似: demo_baidu_v1.0.apk
// 若channel 檔案位於當前目錄下,則可省略  -c ./channel ,即:
python3 multi_channel -s ./SrcApk.apk -v 1.0 -p demo
複製程式碼

當然,也可以給 multi_channel.py 新增執行許可權,並將其所在目錄新增到PATH 環境變數中,就不需要使用 python3 命令來執行了;

chmod a+x multi_channel.py //新增可執行許可權

vi ~/.bash_profile
// 新增 multi_channel.py 檔案所在目錄到 PATH 
export PATH = ****:$PATH // ****表示python指令碼所在目錄路徑

source   ~/.bash_profile

multi_channel.py -s SrcApk.apk -c channel
複製程式碼

shell.png

相關文章