Android 使用 Jenkins 参数化构建

这几天,闲来无事,就想鼓捣鼓捣测试搭建的 Jenkins

Jenkins 上原有的 Android 相关的 Job 有编译 debug 的、编译 release 的和多渠道 release 的,一共三个 Job,感觉一点也不极客范,遂想折腾一下

关于 Tomcat , Jenkins 等环境,编译工具链的安装这里就不介绍了,网上一大堆,都玩烂了,也没什么难度,下面进入正题

首先,想要把三个 Job 集合起来,肯定要用到参数化构建的,下面是我需要的参数

参数名 作用
Branch 指定编译哪一个分支
BuildType 编译类型 debug, release, 多渠道等
EmailList 接收编译结果的邮箱列表

首先 Branch 参数,我一开始使用的是手动填写的方式,但是感觉不爽,后来在网上看到了 Dynamic Choice Parameter ,它可以使用脚本生成动态的下拉列表框


Android 使用 Jenkins 参数化构建_第1张图片
branch

代码如下

def git_address = "[email protected]:xxxxx.git"
def Branches_build = ("git ls-remote -h " + git_address).execute()
Branches_build.text.readLines().collect {
    it.split()[1].replaceAll('refs/heads/', '')
}.unique()

这段代码的核心是 git ls-remote -h remote_address
ls-remote 的意思是:列出远程仓库中,所有可用的引用
-h / --heads 是用来只显示 refs/heads 的引用
命令运行效果如下

git ls-remote -h [email protected]:xxxx.git
e2e7cb5e614915627ec46188613aa0e68fff824d        refs/heads/dev
710ce06a02cca7e48e557dbc29d36d0791cb3c3a        refs/heads/master
18aacde7d9d7047d3e1581569fa901b2f8eb07ac        refs/heads/release/0.5.0_no_face_test
ae7ee8b75622dc69857462f7e5e9599f46afe90c        refs/heads/release/v0.2.0
1ed148ca137daca49663f760d72a72666cb7a761        refs/heads/release/v0.3.0
702c29dafdf5ac3fa9edb8ab2d4ee5803ae98553        refs/heads/release/v0.4.0
61d1466dd051e497722a03c2efbd77db9c853e3a        refs/heads/release/v0.5.0
6326873bdfc5819c3ca8ee756682d086d56614be        refs/heads/release/v0.5.3
6d2ea95393185f2d366a234aa0910d8e145d2bc2        refs/heads/release/v1.0.0
3f0ac34e9b334aae34030756c4a8c0470578ad0c        refs/heads/release/v1.0.1

it.split()[1].replaceAll('refs/heads/', '') 是把每一行分隔后,取第 1 个位置的字符串,然后把 refs/heads/ 替换为空字符串
最终效果如图


Android 使用 Jenkins 参数化构建_第2张图片
效果

第一个参数搞定,接下来是第二个 BuildType 这个就简单了,直接一个 Choice 即可搞定,第三个 EmailList 也很简单,一个 String Parameter,就不贴图了

在 配置 -> 源码管理 -> Branch Specifier (blank for 'any') 填入 ${Branch} 来指定分支

构建触发器 也不详细说了,我加上了 Poll SCM 每 15 分支检查一次,但是好像和参数化构建有冲突,上网搜这个,有人说,参数化构建和 Poll SCM 最好不要在一个 Job 里

构建环境 可以把 Set Build Name 勾上,这样就可以修改每个 Build 的名字了,在列表里看起来更直观,比如 #${BUILD_NUMBER}-${Branch}-${BuildType} ,就会在 Job 页显示形如 #25-dev-debug 的 Build Name,很直观

这里推荐一个插件 user build vars,它可以把开启构建的用户的相关信息写入到构建环境中
下面的表格是所有支持的参数

Variable Description
BUILD_USER Full name (first name + last name)
BUILD_USER_FIRST_NAME First name
BUILD_USER_LAST_NAME Last name
BUILD_USER_ID Jenkins user ID
BUILD_USER_EMAIL Email address

安装后,在 构建环境 中,把 Set jenkins user build variables 勾上,就可以使用上面的参数了
这个插件很有用,比如自动发送编译结果的邮件给开启编译任务的人

然后是构建,我没有使用 Gradle 插件来编译,而是自己写的脚本

echo ${BUILD_NUMBER}
echo ${Branch}
echo ${BUILD_USER_EMAIL}
echo ${BUILD_USER}
source ~/.bash_profile

# 拷贝后的要用来存档的目录
output_app_dir="$WORKSPACE/app/build/jenkins"

# 编译工具输出的目录
output_apk_dir="$WORKSPACE/app/build/outputs"

build_type="${BuildType}"

# 上传蒲公英的 apk 路径
upload_apk_path=""

# 下面判断编译类型,来执行不同的 task
if [ "$build_type" == "debug" ]; then
  gradle clean assembleDebug --stacktrace
  mkdir "$output_app_dir"
  cp $output_apk_dir/apk/app-debug.apk $output_app_dir/debug_${BUILD_NUMBER}.apk
  upload_apk_path=$output_app_dir/debug_${BUILD_NUMBER}.apk
elif [ "$build_type" == "release" ]; then
  gradle clean assembleRelease --stacktrace
  mkdir "$output_app_dir"
  cp $output_apk_dir/apk/app-release.apk $output_app_dir/release_${BUILD_NUMBER}.apk
  cp $output_apk_dir/mapping/release/mapping.txt $output_app_dir/mapping_${BUILD_NUMBER}.txt
  upload_apk_path=$output_app_dir/release_${BUILD_NUMBER}.apk
else
  # 多渠道打包用的是 美团的 walle
  #gradle clean assembleReleaseChannels --stacktrace
  # -DBUGLY_ENABLED=true 是向 Gradle 中传递一个参数,用来开启 bugly 的 mapping 上传任务
  gradle clean -DBUGLY_ENABLED=true assembleReleaseChannels --stacktrace
  mkdir "$output_app_dir"
  cp -r $output_apk_dir/channels $output_app_dir/channels_${BUILD_NUMBER}
  cp $output_apk_dir/mapping/release/mapping.txt $output_app_dir/mapping_${BUILD_NUMBER}.txt
  # 找到含有指定渠道名的 apk 文件的路径用来上传蒲公英
  upload_apk_path=`find "$output_app_dir/channels_${BUILD_NUMBER}" -name "*-xxxx-*.apk"`
fi

#上传到蒲公英
echo "++++++++++++++upload to pgyer+++++++++++++"
#蒲公英上的User Key
uKey="xxxxxx"     
#蒲公英上的API Key
apiKey="xxxxx"    
#要上传的apk文件路径
APK_PATH="${upload_apk_path}"
#执行上传至蒲公英的命令
if [ -f "$APK_PATH" ]; then
  curl -F "file=@${APK_PATH}" -F "uKey=${uKey}" -F "_api_key=${apiKey}" http://www.pgyer.com/apiv1/app/upload
fi

BUGLY_ENABLED 的值是用来控制 bugly 开启和关闭状态的

bugly {
    appId = 'xxxxx' // 注册时分配的App ID
    appKey = 'xxxxxx' // 注册时分配的App Key
    // 获取 BUGLY_ENABLED 的值
    def isEnabled = System.getProperty("BUGLY_ENABLED", "false")
    System.out.println("BUGLY_ENABLED " + isEnabled)
    boolean enable = Boolean.parseBoolean(isEnabled)
    execute = enable
    upload = enable
    uploadMapping = enable
    uploadSymbol = enable
}

美团 walle 多渠道打包配置

walle {
    // 指定渠道包的输出路径
    apkOutputFolder = new File("${project.buildDir}/outputs/channels")
    // 尝试获取 Jenkins 的 BUILD_NUMBER 的值
    def buildNum = System.getenv("BUILD_NUMBER") as Integer ?: '${buildTime}'
    // 定制渠道包的APK的文件名称
    apkFileNameFormat = 'XXXXX-${channel}-${buildType}-v${versionName}-' + buildNum + '.apk'
    // 渠道配置文件
    channelFile = new File("${project.getProjectDir()}/channel")
}

构建后,用于存档的文件

app/build/jenkins/*.txt,app/build/jenkins/channels*/,app/build/jenkins/*.apk

最后是构建完成后的自动发送邮件,网上也有许多我这里不赘述了
我发送的是 ${BUILD_USER_EMAIL},${EmailList}

完!

你可能感兴趣的:(Android 使用 Jenkins 参数化构建)