Jenkins iOS自动打包流程
背景
在开发中我们经常需要打测试包,然后上传至蒲公英等三方平台,这其中需要经历的操作为:
- Git仓库拉取代码
- 设置项目的打包环境
- 利用 xcode 进行打包
- 上传至蒲公英等三方平台
每一次打包,上面的过程必不可少,而且都是手工的,本篇文章我们将采用CD(Continuous Delivery)持续交付和CI(Continuous Integration)持续集成来进行自动化打包一键操作,解放双手,拒绝手动的重复低效率劳动。
本篇文章讲解如何使用 Jenkins + shell脚本 以及 Jenkins + fastlane 自动打包发布至蒲公英等三方平台。
CI/CD(持续集成、持续交付/持续部署)
CI/CD 是一种通过在应用开发阶段引入自动化来频繁向客户交付应用的方法。
CI/CD 的核心概念是持续集成、持续交付和持续部署。作为一个面向开发和运营团队的解决方案,CI/CD 主要针对在集成新代码时所引发的问题。
具体而言,CI/CD 可让持续自动化和持续监控贯穿于应用的整个生命周期(从集成和测试阶段,到交付和部署)。这些关联的事务通常被统称为“CI/CD 管道”,由开发和运维团队以敏捷方式协同支持。
CI(Continuous Integration)是指持续集成,它属于开发人员的自动化流程。成功的 CI 意味着应用代码的新更改会定期构建、测试并合并到共享存储库中。该解决方案可以解决在一次开发中有太多应用分支,从而导致相互冲突的问题。
-
CD(Continuous Delivery/Deployment)是指持续交付或持续部署
持续交付通常是指开发人员对应用的更改会自动进行错误测试并上传到存储库(如 GitHub 或容器注册表),然后由运维团队将其部署到实时生产环境中。这旨在解决开发和运维团队之间可见性及沟通较差的问题。因此,持续交付的目的就是确保尽可能减少部署新代码时所需的工作量。
持续部署(另一种“CD”)指的是自动将开发人员的更改从存储库发布到生产环境,以供客户使用。它主要为了解决因手动流程降低应用交付速度,从而使运维团队超负荷的问题。持续部署以持续交付的优势为根基,实现了管道后续阶段的自动化。
详细流程
自动化打包流程
graph LR
1[本地代码] --提交--> 2[远程仓库] --> 3[Jenkins自动打包] --> 4[部署发布到蒲公英/Testflight] --> 5[结果邮件/钉钉/QQ通知]
Jenkins概述
Jenkins 是一款流行的开源持续集成(Continuous Integration)工具,广泛用于项目开发,具有自动化构建、测试和部署等功能。
安装Jenkins环境
使用 Homebrew 软件包管理器安装Jenkins
没有Homebrew的请先安装Homebrew,网上资料很多,这里不再赘述。
安装命令
- 安装最新的LTS版本:brew install jenkins-lts
- 安装特定的LTS版本:brew install jenkins-lts@YOUR_VERSION
- 启动 Jenkins 服务:brew services start jenkins-lts
- 重新启动 Jenkins 服务:brew services restart jenkins-lts
- 更新 Jenkins 版本:brew upgrade jenkins-lts
安装步骤
- 启动 Jenkins 服务后,浏览 http://localhost:8080 ,该页面需要确认是管理员安装,让我们输入密码,根据提示获取管理员密码,点击继续。
- 选择安装推荐的插件,等待插件安装完毕。
- 插件安装完成后会自动跳转到配置管理员账户页,配置完成点击保存并完成注册。
- 浏览器输入 http://localhost:8080 打开Jenkins,登录即可。
Jenkins 配置-iOS
安装相关插件
构建xcode项目需要安装的插件
- Git Parameter:分支管理
在Jenkins首页->系统管理->插件管理搜索相关插件进行安装即可。
环境变量配置
选择系统管理 -> 系统配置 -> 全局属性 -> 勾选环境变量选项
键:PATH
值:在终端中输入echo $PATH
将输出内容复制填写。
新建任务
创建任务
在Jenkins首页->新建任务中输入任务名称,选择构建一个自由风格的软件项目,点击确定。
任务相关配置
选择General -> 参数化构建过程 -> Git参数
填写名称,选择类型为分支类型,默认origin/master
选择源码管理 -> Git,输入仓库地址,添加账号密码。
选择构建 -> 增加构建步骤 -> 执行shell
添加打包脚本,保存。
Fastlane打包
1、简介
Fastlane是用Ruby语言编写的一套自动化工具集和框架,每一个工具实际都对应一个Ruby脚本,用来执行某一个特定的任务,而Fastlane核心框架则允许使用者通过类似配置文件的形式,将不同的工具有机而灵活的结合在一起,从而形成一个个完整的自动化流程。
2、原理
- Fastlane命令执行的底层并不是自己实现的,而是调用其他的插件或者工具执行的。
- 核心一:打包命令,Fastlane中的
gym
工具只是xcodebuild工具的一个封装。因此安装Fastlane的步骤里有安装IDE的指令集。 - 核心二:iTC(即iTunesConnect 苹果的账号证书应用管理平台)命令。苹果除了提供图形化可操作的网页之外,还提供相对应的一整套底层 API 给开发者使用,Fastlane的底层操作是封装的这套API。
我们iOS开发者,感觉最繁琐的事就是打包上架了,打包过程不仅繁琐还特别耗费时间。那么有没有工具能将我们解放出来呢?有,答案就是Fastlane
。Fastlane
是移动端App开发的脚本工具。
使用Fastlane
自动打包上传的基本步骤如下:
1、安装Fastlane
-> 2、Fastlane
初始化 -> 3、配置证书和描述文件 -> 4、一键上传
Fastlane
的安装有多种途径,其中之一就是使用sudo gem install fastlane
命令安装,其他途径这里不再赘述,也是一行命令的事。可以用命令
fastlane --version
来查看是否安装成功在项目中初始化
fastlane
使用命令fastlane init
,根据提示一步步进行即可。初始化成功之后,在项目根目录下会多几个文件,
Appfile
苹果账号信息、Fastfile
填写打包脚本的地方(重点、重点、重点)。若使用蒲公英上传,需安装蒲公英插件,在终端输入
fastlane add_plugin pgyer
命令,即可安装蒲公英的 fastlane 插件。
Fastlane
打包脚本示例(使用Ruby
语言)
default_platform(:iOS)
#蒲公英api_key和user_key
api_key = "your api_key"
user_key = "your user_key"
configuration = "Release"#Debug_Production_Server
platform :iOS do
desc "🚀🚀🚀🚀🚀打包并上传蒲公英🚀🚀🚀🚀"
lane :pgybeta do
#指定项目的scheme名称
scheme = "TuWanApp"
#更新描述
update_desc = "更新描述"
gym(
#输出的ipa名称
output_name:"#{scheme}_#{Time.new.strftime("%Y%m%d%H%M")}",
# 是否清空以前的编译信息 true:是
clean:true,
scheme:scheme,
# 指定打包方式,Release 或者 Debug
configuration:"#{configuration}",
# 指定打包所使用的输出方式,目前支持app-store, package, ad-hoc, enterprise, development
export_method:"ad-hoc",
# 指定输出文件夹
output_directory:"./build/pgy",
)
pgyer(api_key: "#{api_key}", user_key: "#{user_key}", update_description: "#{update_desc}")
end
end
Jenkins + Fastlane配置
Jenkins shell脚本
#!/bin/bash
export LANG=en_US.UTF-8
export LANGUAGE=en_US.UTF-8
export LC_ALL=en_US.UTF-8
set -e
#输出错误
function echo_error() {
echo "❌❌❌❌❌❌❌❌❌❌ $1 ❌❌❌❌❌❌❌❌❌❌"
}
#输出信息
function echo_info() {
echo "==================== $1 ===================="
}
#输出成功
function echo_success() {
echo "✅✅✅✅✅✅✅✅✅✅ $1 ✅✅✅✅✅✅✅✅✅✅"
}
#json解析
function parseJson() {
local json=$1
local key=$2
if [ "$key" == "code" ]; then
echo "$json" | sed -E 's/,/\n/g' | grep -E "${key}" | sed -E 's/:/\n/g' | sed -E '1d' | sed -E 's/"//g'
else
echo "$json" | tr "\n" " " | sed -E 's/.*"'"${key}"'" *: *"([^"]*)".*/\1/g' | sed -E 's/\\\//\//g'
fi
}
#检测IPA包是否上传成功
function checkIPA() {
echo_info "检测应用是否发布完成,并获取发布应用的信息"
result=$(curl -D - --form-string "_api_key=$1" --form-string "buildKey=$2" "https://www.pgyer.com/apiv2/app/buildInfo")
#解析数据,生成下载地址
resultCode=$(parseJson "$result" "code")
resultMessage=$(parseJson "$result" "message")
resultBuildKey=$(parseJson "$result" "buildKey")
buildShortcutUrl=$(parseJson "$result" "buildShortcutUrl")
buildQRCodeURL=$(parseJson "$result" "buildQRCodeURL")
if [ "$resultCode" == 1246 ] || [ "$resultCode" == 1247 ]; then
echo_info "code: $resultCode"
echo_error "message: $resultMessage"
sleep 5
checkIPA $1 $2
elif [ -n "$resultBuildKey" ]; then
echo_success "上传成功"
echo_info "二维码地址=> ${buildQRCodeURL} <=二维码地址"
echo_info "app下载地址=> https://www.pgyer.com/${buildShortcutUrl} <=app下载地址"
else
echo_info "code: $resultCode"
echo_error "message: $resultMessage"
exit $resultCode
fi
}
#打包
function packaging() {
echo_info "开始打包"
#参数校验 $# 添加到Shell的参数个数
if [ $# != "1" ] || ([ $1 != "debug" ] && [ $1 != "release" ] && [ $1 != "appleStore" ] && [ $1 != "appleStore_beta" ]); then
echo_error '请指定打包类型:debug, release, appleStore, appleStore_beta'
exit 1
fi
#定义一些符变量
export v_env=$1
export v_project_name="TuWanApp"
export pgy_ipa_path="./build/pgy"
export pgy_api_key="蒲公英API Key"
export pgy_install_pwd="123456"
echo_info "安装pod依赖"
#安装pod依赖
#rm -rf './Pods/Local Podspecs'
#pod update
pod install --repo-update
#构建
echo_info "开始构建"
#fastlane打包
if ([ -n "${version}" ] && [ -n "${build}" ]); then
fastlane ${v_env} v:${version} b:${build}
elif [ -n "${version}" ]; then
fastlane ${v_env} v:${version}
else
fastlane ${v_env}
fi
#上传二进制包到蒲公英
if ([ $1 != "appleStore" ] && [ $1 != "appleStore_beta" ]); then
echo_info "上传二进制包到蒲公英"
tokenResp=$(curl -D - --form-string "_api_key=${pgy_api_key}" --form-string 'buildType=ios' --form-string 'buildInstallType=2' --form-string "buildPassword=${pgy_install_pwd}" --form-string "buildUpdateDescription=${updateContent}" "https://www.pgyer.com/apiv2/app/getCOSToken")
echo_info "获取预上传 url 和相关的签名参数=${tokenResp}"
echo_info "解析预上传数据"
code=$(parseJson "$tokenResp" "code")
message=$(parseJson "$tokenResp" "message")
key=$(parseJson "$tokenResp" "key")
signature=$(parseJson "$tokenResp" "signature")
xcossecuritytoken=$(parseJson "$tokenResp" "x-cos-security-token")
endpoint=$(parseJson "$tokenResp" "endpoint")
echo_info "预上传数据解析结果"
echo_info "code:$code"
echo_info "message:$message"
echo_info "key:$key"
echo_info "signature:$signature"
echo_info "xcossecuritytoken:$xcossecuritytoken"
echo_info "endpoint:$endpoint"
if [ "$code" != 0 ]; then
echo_error "$message"
exit $code
else
echo_info "开始上传IPA文件"
cd ${pgy_ipa_path}
pgy_ipa_name=`ls -lt | grep .ipa | head -n 1 | awk '{print $9}'`
echo_info "IPA文件名:${pgy_ipa_name}"
resp=$(curl -D - --form-string "key=${key}" --form-string "signature=${signature}" --form-string "x-cos-security-token=${xcossecuritytoken}" -F "file=@${pgy_ipa_name}" ${endpoint})
echo_info "pgy_resp=$resp"
sleep 5
checkIPA ${pgy_api_key} ${key}
fi
fi
}
export FASTLANE_XCODEBUILD_SETTINGS_TIMEOUT=120
#执行打包
packaging ${BuildType}
上传符号表到bugly
if ([ "${BuildType}" == "appleStore" ] || [ "${BuildType}" == "appleStore_beta" ]); then
cd /Users/tuwanmini/.jenkins/workspace/TuwanApp/build/appStore
rm -rf dsym_tmp
mkdir -p dsym_tmp
filename=`ls -lt | grep .zip | head -n 1 | awk '{print $9}'`
version=$(echo ${filename} | awk -F "_" '{print $2}')
unzip ${filename} -d dsym_tmp
cd /Users/tuwanmini/Desktop/buglyqq-upload-symbol/
java -jar buglyqq-upload-symbol.jar -appid xxxxxxxxx -appkey xxxxxxxxx -bundleid xxxxxxxxx -version ${version} -platform IOS -inputSymbol /Users/tuwanmini/.jenkins/workspace/TuwanApp/build/appStore/dsym_tmp
else
cd /Users/tuwanmini/.jenkins/workspace/TuwanApp/build/pgy
rm -rf dsym_tmp
mkdir -p dsym_tmp
filename=`ls -lt | grep .zip | head -n 1 | awk '{print $9}'`
version=$(echo ${filename} | awk -F "_" '{print $2}')
unzip ${filename} -d dsym_tmp
cd /Users/tuwanmini/Desktop/buglyqq-upload-symbol/
java -jar buglyqq-upload-symbol.jar -appid xxxxxxxxx -appkey xxxxxxxxx -bundleid xxxxxxxxx -version ${version} -platform IOS -inputSymbol /Users/tuwanmini/.jenkins/workspace/TuwanApp/build/pgy/dsym_tmp
fi
发送钉钉消息
注:使用非text消息类型,需在消息内容中添加"@xxx"的文本内容,@功能才能生效
#!/bin/bash
# 定义钉钉机器人 webhook 地址和 access_token
webhook_url="你的钉钉机器人webhook"
appName="你的应用名称"
downloadShortUrl="你的蒲公英下载地址短连接"
installPSW="你的蒲公英安装密码"
atUserMobiles="@你要艾特的钉钉用户手机号"
# 定义消息内容
pgyContent="### ${appName} \
\n版本号: ${version} (${build})\n \
\n更新内容:\n \
\n${updateContent}\n \
\n下载地址: https://www.pgyer.com/${downloadShortUrl}\n \
\n安装密码: ${installPSW}\n \
\n![扫码下载](https://www.pgyer.com/app/qrcode/${downloadShortUrl})\n \
\n${atUserMobiles}"
appStoreContent="### ${appName}提审包上传成功 \
\n版本号:${version}(${build})\n \
\n${atUserMobiles}"
releaseBateContent="### ${appName}队长试用包上传成功 \
\n版本号:${version}(${build})\n \
\n${atUserMobiles}"
if [ "${BuildType}" == "appleStore" ]; then
# 发送 HTTP 请求,向钉钉机器人发送报警消息
curl ${webhook_url} \
-H 'Content-Type: application/json' \
-d "{\"at\": {\"atMobiles\": [${atUserMobile}],\"isAtAll\": false},\"msgtype\": \"markdown\",\"markdown\": {\"title\": \"${appName}\", \"text\": \"${appStoreContent}\"}}"
elif [ "${BuildType}" == "appleStore_beta" ]; then
# 发送 HTTP 请求,向钉钉机器人发送报警消息
curl ${webhook_url} \
-H 'Content-Type: application/json' \
-d "{\"at\": {\"atMobiles\": [${atUserMobile}],\"isAtAll\": false},\"msgtype\": \"markdown\",\"markdown\": {\"title\": \"${appName}\", \"text\": \"${releaseBateContent}\"}}"
else
# 发送 HTTP 请求,向钉钉机器人发送报警消息
curl ${webhook_url} \
-H 'Content-Type: application/json' \
-d "{\"at\": {\"atMobiles\": [${atUserMobile}],\"isAtAll\": false},\"msgtype\": \"markdown\",\"markdown\": {\"title\": \"${appName}\", \"text\": \"${pgyContent}\"}}"
fi
构建打包任务
选择已创建的任务
选择Build with Parameters -> 分支 -> 开始构建。
查看控制台输出。
补充说明
找不到/ExportOptions/XXXExportOptions.plist文件
- AdHocExportOptions.plist,AppStoreExportOptios.plist,EnterpriseExportOptions.plist,DevelopmentExportOptions.plist文件可通过Xcode -> Product -> Archive打包生成的ExportOptions.plist文件添加相应前缀重命名即可获得。
- 在Jenkins -> workspace -> 项目根目录下新建名为ExportOptions的文件夹,将相应plist文件拷贝到改文件夹中即可。
证书及描述文件
在Xcode中配置完成即可。
大文件拉取git-lfs报错
运行
brew install git-lfs
即可安装若不再需要大文件支持,可前往项目根目录 -> .git -> hooks -> 移除post-checkout文件即可。
拉取代码出错
cd到Jenkins -> workspace目录下,手动通过git clone将工程clone到本地,重启Jenkins,再次执行打包任务。
局域网使用IP访问jenkins
使用brew安装jenkins会避免很多其他安装方式产生的用户权限问题,但是会将httpListenAddress默认设置为127.0.0.1,这样我们虽然可以在本地用localhost:8080访问,但是本机和局域网均无法用ip访问。
解决办法为修改两个路径下的plist配置:
- ~/Library/LaunchAgents/homebrew.mxcl.jenkins.plist
- /usr/local/opt/jenkins/homebrew.mxcl.jenkins.plist
第二个路径找不到的话,cmd + shift + G 输入/opt/homebrew/opt/jenkins,找到homebrew.mxcl.jenkins.plist
将上面两个plist中的httpListenAddress后的ip地址,修改为本机IP或者0.0.0.0,修改完成后,重启Jenkins,接下来就可以使用IP访问了。
fastlane版本升级注意事项
- 需要修改项目中Gemfile文件中的fastlane版本为当前要升级的版本;
- cd到项目目录下,执行bundle install,重新安装您的 gem 依赖;
- 再次运行 bundle update fastlane ;
- 重启电脑,环境变量生效;
- 若Jenkins中的fastlane版本仍然没有升级到最新,可以使用Jenkins新建任务的方式,在新建任务中添加构建shell脚本,执行如下代码:
echo 你的电脑密码 | sudo -S bundle update fastlane
fastlane --version
- 若版本仍然没有更新,请前往ruby安装目录下,卸载fastlane其他旧版本,只留新版本即可。