首先說下我們項(xiàng)目的對(duì)于打包的需求则吟,這里只針對(duì)發(fā)布正式環(huán)境的包槐臀。
項(xiàng)目的代碼放在Gitlab,需要打包的應(yīng)用市場有十多個(gè),apk都需要使用360加固氓仲,打包的工作由開發(fā)完成水慨,然后將所有市場的apk文件壓縮成一個(gè)zip文件發(fā)給市場的同事上線得糜。最初的流程是由開發(fā)用AS打包后手動(dòng)的進(jìn)行加固,然后每次發(fā)布光打包-加固-修改apk文件名+發(fā)郵件這個(gè)流程都得花上半個(gè)小時(shí)以上晰洒。為了提高效率朝抖,所以我決定使用gradle+jenkins來完成這個(gè)任務(wù)。
其實(shí)這樣的文章挺多的谍珊,但是別人的需求總是不太能完美的解決我的問題治宣,所以我自己通過gradle寫了個(gè)task來解決我的需求。
Gradle腳本
一. 在Project下新建一個(gè)目錄reinforce,將360加固相關(guān)文件導(dǎo)入
channel這個(gè)目錄是我自己創(chuàng)建的,里面保存了多渠道打包的配置模板
二. 修改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是我們項(xiàng)目名,生成的apk文件名pccb-v3.2.0-vivo-release.apk這種形式砌滞,后面從這個(gè)文件名中獲取渠道和版本信息侮邀。
三. 創(chuàng)建gradle腳本文件app/pack-release.gradle
我創(chuàng)建了一個(gè)task packageRelease,這個(gè)task依賴assembleRelease贝润,assembleRelease執(zhí)行完成后會(huì)執(zhí)行packageRelease的doLast方法绊茧。
packageRelease的執(zhí)行流程:
1.從outputs/apk/xx/release中找出assembleRelease生成的所有apk
我這里有4個(gè)渠道,所以最終生成了4個(gè)apk文件打掘。理論上來說我們在打多渠道包的時(shí)候华畏,可以使用360加固的多渠道打包功能由一個(gè)包就可以生成N個(gè)渠道包,但是我這里有點(diǎn)特殊的是我們十多個(gè)渠道的app名字并不是一樣的胧卤,總共有4個(gè)app名唯绍,每個(gè)對(duì)應(yīng)幾個(gè)渠道。360加固只能修改AndroidManifest.xml中meta-data標(biāo)簽中的值枝誊,所以我這里必須為每個(gè)app名生成一個(gè)apk文件况芒,并且在reinforce/channel中創(chuàng)建了4個(gè)多渠道打包模板。
2. 創(chuàng)建一個(gè)保存加固后的apk目錄:
根據(jù)版本號(hào)創(chuàng)建目錄叶撒,build/outputs/release/pccb-x.x.x
3. 將4個(gè)原始的apk進(jìn)行360加固绝骚,生成多個(gè)渠道的apk,自動(dòng)簽名
4. 刪除加固后生成的temp.apk和jiagu_sign.apk結(jié)尾的文件祠够,保留渠道名+_sign.apk結(jié)尾的文件
5. 根據(jù)需要修改保留的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 = ""http://360加固賬號(hào)
PASSWORD = ""http://360加固密碼
KEY_PATH = "" //密鑰路徑
KEY_PASSWORD = "" //密鑰密碼
ALIAS = "" //密鑰別名
ALIAS_PASSWORD = "" //別名密碼
OUTPUT_PATH = "build/outputs/release/" //加固后所有apk的保存路徑
CHANNEL_CONFIG = BASE + "channel/"http://保存渠道配置
}
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中緩存了多渠道信息,如果不刪除會(huì)合并到當(dāng)前多渠道配置
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"
}
}
/**
* 刪除一些臨時(shí)文件
* @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()
}
//構(gòu)建發(fā)布到生產(chǎn)環(huán)境的所有渠道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)
}
}
四. 應(yīng)用pack-release.gradle
在build.gradle頂部添加
apply from: 'pack-release.gradle'
jenkins配置
一. General
二.源碼管理
三.構(gòu)建
關(guān)于Root Build Script和Build File這里遇到了問題記錄下止剖,我前面腳本中的目錄下的是 ../reinforce/,然后我在Android Studio中執(zhí)行 gradlew packageRelease是沒有問題的落君,但是在jenkins一直找不到對(duì)應(yīng)的文件穿香。后來找到原因是因?yàn)槲以贏ndroid studio中是從app目錄開始構(gòu)建的所以沒有問題,但是jenkins中是從project目錄開始構(gòu)建所以根本找不到對(duì)應(yīng)的目錄绎速。最后通過設(shè)置 Root Build Script->app,Build File->build.gradle解決皮获。