小目标:使用Jenkins
一键构建,并自动上传到App Store
。html
fastlane
是为iOS
和Android
应用程序自动化测试部署和发布的最简单方法。🚀它处理全部繁琐的任务,如生成屏幕截图,处理代码签名以及发布应用程序。ios
fastlane
将以下的工具套件有机地结合起来,从管理证书到单元测试,从编译打包到上传发布,都能经过命令行轻松完成.该套件支持与 Jenkins
和 CocoaPods
,xctools
等其余第三方工具的集成,而且可以定义多个通道(lanes)以支持不一样的部署目标。git
配置当前设备环境,最新的fastlane(2.75.1)
须要2.1
以上的ruby
版本,正常的版本低点的,也是要求2.0
以上的。由于fastlane
工具是使用ruby
写的。版本太低的建议安装rvm
来升级ruby
。github
curl -L get.rvm.io | bash -s stable
source ~/.bashrc
source ~/.bash_profile
# 检测是否安装成功
rvm -v
复制代码
设置环境变量,fastlane
须要设置一些环境变量才能正确运行,若是当前的语言环境没有设置为UTF-8
,会致使构建和上传的时候出现问题。在~/.bashrc
, ~/.bash_profile
或者 ~/.zshrc
文件下添加以下内容:swift
export LC_ALL=en_US.UTF-8
export LANG=en_US.UTF-8
复制代码
安装Xcode
命令行工具 xcode-select --install
,若是已经安装会提示xcode-select: error: command line tools are already installed, use "Software Update" to install updates
。xcode
建立App ID
,证书
,在iTunes connect
建立一个用于测试的 app。ruby
安装fastlane。bash
建立一个测试demo。app
并将其scheme
设置为shared
,否则fastlane init
的时候会失败。curl
设置好签名配置文件。
为app添加icon
。
修改devices
为iPad
或者iPhone
。
cd 到项目目录下,对于ruby安装程序,使用命令 sudo fastlane init
。(swift使用fastlane init swift
,Swift安装仍在测试阶段。有关更多信息,请参阅Fastlane.swift文档。 )
会问你想使用fastlane
作什么?这里咱们输入3
,自动发布到Apple Store
。
执行过程当中会要求你输入Apple开发证书的Apple ID
,若是有多个Team,会让你选择team。
接着会问是否想用fastlane
来管理你的app metadata
。
输入y
,fastlane
则会下载现有的元数据和屏幕截图。若是咱们编辑了download下来的.txt
文件,在使用fastlane
上传app到iTunes connect
的时候也会将这些内容上传到iTunes connect
。
输入n
,不作任何操做,仍然能使用fastlane
上传app到App Store
。
若是最后出现fastlane release
,就表示init成功了。
此时项目目录下会多出一个fastlane
的文件夹。
若是Deliverfile
,screenshots
和metadata
目录没被建立,能够运行deliver init
来建立。
在Deliverfile
文件里,添加force true
,否则会在上传到iTunes connect
的时候会弹出一个Preview.html
网页。
在项目根目录下touch
一个Gemfile
文件,添加如下内容
source "https://rubygems.org"
gem "fastlane"
复制代码
执行以下命令:
# 安装bundler
sudo gem install bundler
# 更新 bundle,成功以后会生成一个版本控制的Gemfile.lock文件
[sudo] bundle update
复制代码
执行命令:bundle exec fastlane [lane_name]
,执行lane_name
脚本。这里的lane_name
是脚本的名称,咱们能够理解为函数名,若是咱们只执行bundle exec fastlane
命令,则会有一个让咱们选择的地方,选择须要执行的脚本。
会将项目名,ipa
存放的路径,app_identifier
等一系列信息打印出来。
ipa
和dYSM
文件都存放在项目根目录。
紧接着会自动上传metadata
和ipa
到iTunes Connect
。
最后会输出每一个脚本执行所消耗的时间(s)。
若是只是很简单的上传到iTunes connect
,上面的操做就能够知足。
若是咱们是多个target
或者须要配置一些ITC
上面的内容,则须要进一步的深刻。
metadata
是包含应用在ITC
上面的各类信息,可使用它配置咱们的ITC
,建议使用Deliverfile
。
屏幕截图数据。
存储App信息,好比Apple ID
,bundle ID
等信息。
交付文件。在这个文件里面能够设置iTunes connect
的全部配置项,例如:
release_notes
,此版本新增内容。copyright
,版权信息。submit_for_review
,上传完成后是否直接提交新版本进行审查。force
,跳过HTML报告文件验证。请在设置release_nores
、support_url
、private_url
等配置的时候,采用hash
的方式写,国家代码,例如:
release_notes(
# 中国
'zh-Hans' => ENV['RELEASE_NOTES'],
# 澳大利亚
'en-au' => ENV['RELEASE_NOTES_AU'],
# 美国
'en-us' => ENV['RELEASE_NOTES_US']
)
复制代码
自动化脚本配置文件。 是咱们脚本的入口,全部的事件驱动都是在这个文件来调度的。
default_platform(:ios)
platform :ios do
desc "demo upload_to_app_store"
lane : Archive_TargetA do |options|
scheme = options[:scheme]
date = Time.new.strftime("%Y%m%d-%h%M")
# export_method 支持 app-store, ad-hoc, package, enterprise, development
gym(
scheme: "#{scheme}",
output_name: "#{scheme}-#{date}.ipa",
clean: true,
export_method: 'app-store',
)
# upload_to_app_store
deliver # 当deliverfile为空的时候,同 upload_to_app_store 做用同样
end
end
复制代码
cd到项目根目录执行命令:bundle exec fastlane Archive_TargetA scheme:"CDDemo"
,后面的scheme
是带的参数。
若是咱们须要配置多个target进行打包的话,咱们可使用环境变量,来进行配置。假如咱们如今有两个target
,targetA
和 targetB
,则咱们须要建立两个.env
文件,例如.env.targetA
,.env.targetB
,放在Fastfile
文件同级目录下
在.env
文件里面咱们能够配置一些不一样的内容(非公共),好比app_identifier
,release_notes
等等。截图以下:
在Appfile
,Deliverfile
,Fastfile
等文件,咱们均可以直接使用.env
文件里面的内容。
Appfile
# Appfile
#The bundle identifier of your app
app_identifier ENV['APP_IDENTIFIER']
# Your Apple email address
apple_id ENV['APPLE_ID']
# Developer Portal Team ID
team_id ENV['TEAM_ID']
复制代码
Deliverfile,请在设置release_nores、support_url、private_url等配置的时候,采用hash的方式写。
# app_identifier
app_identifier ENV['APP_IDENTIFIER']
# 用户名,Apple ID电子邮件地址
username ENV['APPLE_ID']
# 团队ID
team_id ENV['TEAM_ID']
# 团队name
team_name ENV['TEAM_NAME']
# copyright
copyright ENV['COPYRIGHT']
# 关键字
keywords(
'zh-Hans' => ENV['KEYWORDS'],
)
# 新版本修改记录
release_notes(
# 中国
'zh-Hans' => ENV['RELEASE_NOTES'],
# 澳大利亚
'en-au' => ENV['RELEASE_NOTES_AU'],
# 美国
'en-us' => ENV['RELEASE_NOTES_US']
)
# 支持网址
support_url(
# 中国
'zh-Hans' => ENV['SUPPORT_URL'],
# 澳大利亚
'en-au' => ENV['SUPPORT_URL_AU'],
# 美国
'en-us' => ENV['SUPPORT_URL_US']
)
# 隐私政策网址 国家代码 https://www.cnblogs.com/Mien/archive/2008/08/22/1273950.html
privacy_url(
# 中国
'zh-Hans' => ENV['PRIVACY_URL'],
# 澳大利亚
'en-au' => ENV['PRIVACY_URL_AU'],
# 美国
'en-us' => ENV['PRIVACY_URL_US']
)
# 上传完成后提交新版本进行审查
submit_for_review false
# 跳过HTML报告文件验证
force true
# 启用iTC的分阶段发布功能 灰度发布
phased_release true
# 应用审核小组的联系信息 app 审核信息
app_review_information(
first_name: "xx",
last_name: "xx",
phone_number: "+86 18888888888",
email_address: "xxxx",
demo_user: "test1@test.com",
demo_password: "test123"
)
...
复制代码
Fastfile文件里面使用环境变量,跟上面略有不一样。在Fastfile里面,咱们须要告诉lane
要使用那个.env
文件,这时候咱们须要使用sh
脚本命令的形式来调用一个lane 后面跟上--env 环境变量文件名
,此时调用的lane
不能声明为private_lane
,调用方式以下:
lane :releaseDemo2 do
# 无参数
sh "fastlane Archive_TargetA --env TargetA"
# 有参数
sh 'fastlane Archive_TargetA type:\'哈哈哈哈\' --env TargetA'
end
外部直接调用(带参数)
bundle exec fastlane Archive_TargetA type:"haha" --env TargetA
复制代码
而后咱们在Archive_TargetA
lane 里面使用ENV['xx']
方式,读取出来的内容就是从.env.TargetA
文件读取出来的。同理,deliver
Action 对应的DeliverFile
文件里面的内容也是从.env.TargetA
文件读取出来的。
private_lane : Archive_TargetA do |options|
scheme = ENV['SCHEME'] # 这时候读取出来的'scheme'就是'TargetA',从'.env.TargetA'读取出来的
# export_method 支持 app-store, ad-hoc, package, enterprise, development
gym(
scheme: "#{scheme}",
output_name: "#{scheme}.ipa",
clean: true,
export_method: 'app-store',
)
deliver # 这时候deliverfile里面读取的内容就是从'.env.TargetA'文件读取的
end
复制代码
跟咱们本身写方法调用同样,例如:
desc "打包统一入口"
lane :Archive do |options|
# 若是咱们传入的参数'type'是targetA,那么咱们就执行Archive_TargetA 这个lane。。。
type = options[:type]
if type == "TargetA"
Archive_TargetA(options)
elsif type == "TargetB"
Archive_TargetB(options)
else
Archive_TargetA(options)
end
end
复制代码
fastlane
默认有 lane。
before_all
,就是在执行一次脚本以前首先执行的代码,咱们能够在这里面执行一些公共的东西,好比git_pull
,cocoapods
。
before_all do
# 检出到 Developer 分支
sh 'git checkout Developer'
git_pull
cocoapods(repo_update: true)
end
复制代码
after_all
, 成功结束以后,处理共有的后置逻辑。
before_each
,每次执行 lane 以前都会执行一次。
after_each
,每次执行 lane 以后都会执行一次。
error
,在执行上述状况任意环境报错都会停止并执行一次。
执行顺序 | 方法名 | 说明 |
---|---|---|
1 | before_all | 在执行 lane 以前只执行一次。 |
2 | before_each | 每次执行 lane 以前都会执行一次。 |
3 | lane | 自定义的任务。 |
4 | after_each | 每次执行 lane 以后都会执行一次。 |
5 | after_all | 在执行 lane 成功结束以后执行一次。 |
6 | error | 在执行上述状况任意环境报错都会停止并执行一次。 |
出现 Command timed out after 10 seconds on try 1 of 4, trying again...
,在fastlane
文件开头加上:
ENV["FASTLANE_XCODEBUILD_SETTINGS_TIMEOUT"] = "180"
ENV["FASTLANE_XCODE_LIST_TIMEOUT"] = "180"
复制代码
versioning
,用来修改build
版本号和version
版本号。Fastlane
内嵌的actionincrement_build_number
使用的是苹果提供的agvtool
,agvtool在更改Build的时候会改变全部target的版本号。这时若是你在一个工程里有多个产品的话,每次编译,全部的Build都要加1,最后就不知道高到哪里去了。fversioning
不只能够指定target增长Build,并且能够按照「语义化版本」规范增长Version,固然也能够直接设定Version。firim
,直接把AdHoc
或者InHouse
打包的ipa上传到fir.im
,供测试下载。fastlane add_plugin [name]
,须要到项目根目录下执行。
fastlane update_plugins
插件更新,同上,须要cd到项目根目录下。
private_lane
。private_lane
表示私有lane,使用bundle exec fastlane
命令,声明为 private_lane
的是否是显示出来的,使用脚本命令形式调用的时候不能设置成private_lane
。能够直接在lane
里面执行git命令,例如sh 'git checkout Developer'
,检出Developer
分支。
因为本人的水平有限,不免会有错误和疏漏,欢迎 issue 指正。若是你们在Fastlane
的使用上,有更好的案例,也欢迎交流和分享。
转载请注明出处!