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

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

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

Gradle脚本

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

《Android持续集成:Jenkins+Gradle+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

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

二.源码管理

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

三.构建

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

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

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

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

    原文作者:Wang_Yi
    原文地址: https://www.jianshu.com/p/3fa0f4f0794e
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞