Android持续集成:Jenkins+Gradle+360加固+多渠道打包

首先说下我们项目的对于打包的需求,这里只针对发布正式环境的包。
项目的代码放在Gitlab,需要打包的应用市场有十多个,apk都需要使用360加固,打包的工作由开发完成,然后将所有市场的apk文件压缩成一个zip文件发给市场的同事上线。最初的流程是由开发用AS打包后手动的进行加固,然后每次发布光打包-加固-修改apk文件名+发邮件这个流程都得花上半个小时以上。为了提高效率,所以我决定使用gradle+jenkins来完成这个任务。

其实这样的文章挺多的,但是别人的需求总是不太能完美的解决我的问题,所以我自己通过gradle写了个task来解决我的需求。

Gradle脚本

一. 在Project下新建一个目录reinforce,将360加固相关文件导入


channel这个目录是我自己创建的,里面保存了多渠道打包的配置模板

二. 修改Android Studio生成apk文件名

build.gradle中添加配置:

    android.applicationVariants.all { variant ->
            variant.outputs.all {
                outputFileName = "pccb-v" + defaultConfig.versionName + "-" +
                        variant.productFlavors[0].name + "-" + variant.buildType.name + ".apk"
            }
    }

pccb是我们项目名,生成的apk文件名pccb-v3.2.0-vivo-release.apk这种形式,后面从这个文件名中获取渠道和版本信息。

三. 创建gradle脚本文件app/pack-release.gradle

我创建了一个task packageRelease,这个task依赖assembleRelease,assembleRelease执行完成后会执行packageRelease的doLast方法。

packageRelease的执行流程:

1.从outputs/apk/xx/release中找出assembleRelease生成的所有apk
我这里有4个渠道,所以最终生成了4个apk文件。理论上来说我们在打多渠道包的时候,可以使用360加固的多渠道打包功能由一个包就可以生成N个渠道包,但是我这里有点特殊的是我们十多个渠道的app名字并不是一样的,总共有4个app名,每个对应几个渠道。360加固只能修改AndroidManifest.xml中meta-data标签中的值,所以我这里必须为每个app名生成一个apk文件,并且在reinforce/channel中创建了4个多渠道打包模板。

2. 创建一个保存加固后的apk目录:
根据版本号创建目录,build/outputs/release/pccb-x.x.x

3. 将4个原始的apk进行360加固,生成多个渠道的apk,自动签名
4. 删除加固后生成的temp.apk和jiagu_sign.apk结尾的文件,保留渠道名+_sign.apk结尾的文件

5. 根据需要修改保留的apk的文件名
6. 压缩pccb-x.x.x文件夹,生成pccb-x.x.x.zip

pack-release.gradle代码:

import java.util.zip.ZipEntry
import java.util.zip.ZipOutputStream

ext {
    BASE = "../reinforce/"
    JAR = BASE + "jiagu.jar"
    NAME = ""//360加固账号
    PASSWORD = ""//360加固密码
    KEY_PATH = "" //密钥路径
    KEY_PASSWORD = "" //密钥密码
    ALIAS = "" //密钥别名
    ALIAS_PASSWORD = "" //别名密码
    OUTPUT_PATH = "build/outputs/release/" //加固后所有apk的保存路径
    CHANNEL_CONFIG = BASE + "channel/"//保存渠道配置
}

class ApkFile {
    String channel
    File file
}

/**
 * 查找所有apk
 * @param buildType release 或者 debug
 * @return ArrayList <ApkFile>
 */
def findApkFiles(String buildType) {
    println "findApkFiles buildType: " + buildType

    File apkDir = new File("build/outputs/apk")
    File[] channelDirs = apkDir.listFiles()

    List<ApkFile> apkFiles = new ArrayList<>()
    for (int i = 0; i < channelDirs.length; i++) {
        File channelDir = channelDirs[i]
        ApkFile apkFile = new ApkFile()
        apkFile.channel = channelDir.name

        File[] files = new File(channelDir, "/" + buildType).listFiles()
        if (files == null || files.length == 0) {
            continue
        }
        File lastFile = files[files.length - 1]
        if (!lastFile.name.endsWith(".apk")) {
            continue
        }

        apkFile.file = lastFile
        apkFiles.add(apkFile)
    }

    return apkFiles
}

/**
 * 360加固
 * @param apk 加固的原始apk File
 * @param outputPath 输出目录
 * @param channel 原始渠道(baidu,yyb,...)
 */
def reinforce(apk, outputPath, channel) {
    println "reinforce apk:" + apk

    //jiagu.db中缓存了多渠道信息,如果不删除会合并到当前多渠道配置
    def db = new File(BASE + "jiagu.db")
    if (db.exists()) {
        if (!db.delete()) {
            throw new RuntimeException("delete jiagu.db failure!")
        }
    }

    exec {
        commandLine "powershell", "java -jar", JAR, "-login", NAME, PASSWORD
    }
    exec {
        commandLine "powershell", "java -jar", JAR, "-importsign", KEY_PATH, KEY_PASSWORD, ALIAS, ALIAS_PASSWORD
    }
    exec {
        commandLine "powershell", "java -jar", JAR, "-showsign"
    }
    exec {
        commandLine "powershell", "java -jar", JAR, "-importmulpkg", CHANNEL_CONFIG + "template_" + channel + ".txt"
    }
    exec {
        commandLine "powershell", "java -jar", JAR, "-showmulpkg"
    }
    exec {
        commandLine "powershell", "java -jar", JAR, "-jiagu", apk, outputPath, "-autosign", "-automulpkg"
    }
}

/**
 * 删除一些临时文件
 * @param outputDir apk保存目录
 */
def filterApk(File outputDir) {
    println "*************** filter apk ***************"

    File[] files = outputDir.listFiles()
    for (int i = 0; i < files.length; i++) {
        File file = files[i]
        String fileName = file.getName()

        if (fileName.endsWith("jiagu_sign.apk") || fileName.endsWith("temp.apk")
                || !fileName.endsWith("_sign.apk")) {
            file.delete()
        }
    }
}

/**
 * 修改所有apk文件名
 * @param outputDir apk保存目录
 */
def renameApk(File outputDir) {
    println "*************** rename apk ***************"

    File[] files = outputDir.listFiles()
    for (int i = 0; i < files.length; i++) {
        File file = files[i]
        String fileName = file.getName()

        String[] prefixArr = fileName.split("-")
        String[] suffixArr = fileName.split("_")

        String rename = prefixArr[0] + "-" + prefixArr[1] +
                "-" + (i + 1) + "-" + suffixArr[suffixArr.length - 2] + ".apk"
        file.renameTo(file.getParent() + "/" + rename)

        println "rename apk: " + fileName + " --> " + rename
    }
}

/**
 * zip压缩apk保存目录,生成 build/outputs/release/pccb-x.x.x.zip
 * @param outputDir apk保存目录
 */
def compressDir(File outputDir) {
    println "*************** compress apk output dir ***************"

    File zipFile = new File(outputDir.getParent() + "/" + outputDir.getName() + ".zip")
    if (zipFile.exists()) {
        zipFile.delete()
    }

    ZipOutputStream zos = new ZipOutputStream(new FileOutputStream(zipFile))
    File[] files = outputDir.listFiles()
    for (int i = 0; i < files.length; i++) {
        File file = files[i]

        byte[] bf = new byte[8192]
        FileInputStream fis = new FileInputStream(file)
        zos.putNextEntry(new ZipEntry(file.getName()))

        int len
        while ((len = fis.read(bf)) > 0) {
            zos.write(bf, 0, len)
        }
        zos.flush()
        fis.close()
    }

    zos.close()
}

//构建发布到生产环境的所有渠道apk,生成压缩文件 pccb-x.x.x.zip
task packageRelease {
    dependsOn("assembleRelease")

    doLast {
        List<ApkFile> apkFiles = findApkFiles("release")
        if (apkFiles.size() == 0) {
            throw new RuntimeException("no apk files has found!")
        }

        String[] nameSlice = apkFiles.get(0).file.name.split("-")
        File outputDir = new File(OUTPUT_PATH + nameSlice[0] + "-" + nameSlice[1])
        if (outputDir.exists()) {
            if (!outputDir.delete()) {
                throw new RuntimeException("delete outputDir failure!")
            }
        } 

        if (!outputDir.mkdirs()) {
            throw new RuntimeException("make outputDir failure!")
        }

        for (int i = 0; i < apkFiles.size(); i++) {
            ApkFile apkFile = apkFiles.get(i)
            reinforce(apkFile.file, outputDir.getPath(), apkFile.channel)
        }

        filterApk(outputDir)
        renameApk(outputDir)
        compressDir(outputDir)
    }
}

四. 应用pack-release.gradle

在build.gradle顶部添加

apply from: 'pack-release.gradle'

jenkins配置

一. General

二.源码管理

三.构建

关于Root Build Script和Build File这里遇到了问题记录下,我前面脚本中的目录下的是 ../reinforce/,然后我在Android Studio中执行 gradlew packageRelease是没有问题的,但是在jenkins一直找不到对应的文件。后来找到原因是因为我在Android studio中是从app目录开始构建的所以没有问题,但是jenkins中是从project目录开始构建所以根本找不到对应的目录。最后通过设置 Root Build Script->app,Build File->build.gradle解决。

四,构建后操作-归档文件

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,522评论 25 707
  • 最近在项目中遇到需要实现 Apk 多渠道、定制化打包, Google 、百度查找了一些资料,成功实现了上述功能,在...
    看一季残花落幕阅读 2,483评论 1 8
  • 关于作者: 李涛,腾讯Android工程师,14年加入腾讯SNG增值产品部,期间主要负责手Q动漫、企鹅电竞等项目的...
    稻草人_3e17阅读 3,595评论 0 10
  • 今晚的作业写到10:00终于全部完成。我说:“快洗洗睡吧。”你却答道:“妈妈我再看一会作文书,补充一下我的...
    1614268fa8a5阅读 123评论 0 0
  • 一,太棒了,好开心一起和儿子互动游玩 我很开心美天奇迹醒来,开心心悦无比 我很开心太傻不断显化奇迹和我 二,感恩宇...
    磁场爱生活阅读 225评论 0 0