标签:
最近付费购买了Travis CI,Travis CI的收费模式很有意思,不是按项目或者用户,而是按工作进程收费,比如初级版本是$129/月,总共提供2个工作进程。在项目不多的情况下,除了用于跑单元测试外,不免想利用的更充分一些,因此抽空搭建了一套基于Travis CI的Android自动发布工作流。
未自动化前安卓开发总是避免不了这样的工作流程:
实现了这套自动化发布后,工作流程被简化成:
git tag
对代码打一个内测版的tag,在tag的描述中对写当前完成的功能Tag提交后Travis CI会自动编译代码,生成APK文件并分发到Github和fir.im,Github和fir.im中会保持Tag的描述信息,分发完成后会有邮件通知所有参与测试的人员。而作为开发人员,只需要专注于对代码打好一个Tag就可以了。
整个流程看似做了不少工作,其实体现在Travis CI只有数行指令而已,以下逐一讲解:
Travis CI应该可以算是目前最好用的持续集成服务之一了,如果代码库是基于Github的话,可以很简单的开启。由于本文涉及到了很多Travis CI的基础概念,建议首先对Travis CI的自定义构建一节有所了解。
很早前在介绍PHP项目的持续集成时也写过如何在PHP项目中使用Travis CI。 对于安卓项目来说步骤几乎一致:
首先准备一个.travis.yml
文件放在安卓项目根目录下,.travis.yml
中记录了Travis CI所需的基础信息:
language: android
sudo: false
android:
components:
- build-tools-23.0.1
- android-23
- extra-android-m2repository
- extra-android-support
script:
- "./gradlew assembleRelease"
无需读文档就可以通过上面的配置大概知道,我们要运行的是一个安卓项目,安卓SDK版本为23,项目所用的BuildTools版本为23.0.1,为编译这个项目我们还引入了一些必须的组件,如Support Library(extra-android-support)、Android Support Repository(extra-android-m2repository)等。
当Travis CI准备好我们所需要的环境后,将自动运行yml文件script
部分所设置的指令,上例中运行的是./gradlew assembleRelease
,运行成功的话会在项目的主模块下生成build/outputs/apk/app-release.apk
。
最后进入Travis CI主页,使用有项目Admin权限的Github帐号直接登录。选择要开启Travis CI的项目,将右边的开关设为On即可。
Travis CI目前有2个网站:如果是开源项目,直接进入travis-ci.org即可,如果是私有付费项目,则需要进入travis-ci.com,2个网站除了域名外所有的界面及操作几乎一模一样。
配置中还有一行sudo: false
,是为了开启基于容器的Travis CI任务,让编译效率更高。
安卓项目发布需要证书文件和若干密码,但无论是开源项目还是私有项目,任何时候都不应该将原始证书或密码放入代码库(原则上来讲证书和密码也不应该交于开发人员,而应该只能通过发布服务器进行编译)。Travis CI为此提供了2种解决方案,一种是对敏感信息、密码、证书等进行对称加密,在CI构建环境时解密,另一种是将密码等通过Travis CI的控制台(即网站)设置为构建时的环境变量。
由于前者会在Travis控制台生成一对环境变量,所以我的做法是尽量选择后者,但由于Travis控制台无法上传文件,因此涉及到文件加密的部分,则只能选择前者。
说了这么多,首先还是需要先对编译脚本进行改造,如果不考虑安全问题,项目的build.gradle
文件可能会是这样:
android {
signingConfigs {
releaseConfig {
storeFile file("../keys/evandroid.jks")
storePassword "123456"
keyAlias "evandroid_alias"
keyPassword "654321"
}
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile(‘proguard-android.txt‘), ‘proguard-rules.pro‘
signingConfig signingConfigs.releaseConfig
}
}
}
而我们最终要的效果,还是希望一份编译脚本既可以用于开发环境,也可以在CI环境下使用,在Travis CI中,可以通过点击项目名称 -> Settings -> Environment Variables中设置环境变量,比如我们可以针对上面的配置,分别设置KEYSTORE_PASS
、ALIAS_NAME
、ALIAS_PASS
三个环境变量,在Travis CI环境下可以通过System.getenv()
获得这些环境变量。
本地开发环境中,我的做法是将这几个变量加到gradle.properties
文件中,这样就可以在build.gradle
内直接使用了。下面是开发环境的gradle.properties
KEYSTORE_PASS=123456
ALIAS_NAME=evandroid_alias
ALIAS_PASS=654321
这样一来build.gradle
就变成了
releaseConfig {
storeFile file("../keys/evandroid.jks")
storePassword project.hasProperty("KEYSTORE_PASS") ? KEYSTORE_PASS : System.getenv("KEYSTORE_PASS")
keyAlias project.hasProperty("ALIAS_NAME") ? ALIAS_NAME : System.getenv("ALIAS_NAME")
keyPassword project.hasProperty("ALIAS_PASS") ? ALIAS_PASS : System.getenv("ALIAS_PASS")
}
接下来处理证书文件,为了方便文件加密等功能,Travis CI提供了一个基于ruby的CLI命令行工具,可以直接使用gem安装
gem install travis
安装后进入安卓项目根目录,尝试对证书文件加密:
travis encrypt-file keys/evandroid.jks --add
如果首次运行,travis会提示需要登录,运行travis login --org
并输入Github用户名密码即可。(付费版则为travis login --pro
)
travis encrypt-file
指令会做几件事情:
encrypted_e41864bb9dab_key
,encrypted_e41864bb9dab_iv
openssl
对文件进行加密,上例中会项目根目录生成evandroid.jks.enc
文件.travis.yml
中自动生成Travis CI环境下解密文件的配置,上例运行后可以看到.travis.yml
中多了几行:before_install:
- openssl aes-256-cbc -K $encrypted_e41864bb9dab_key -iv $encrypted_e41864bb9dab_i -in keys/evandroid.jks.enc -out keys/evandroid.jks -d
Travis CI默认在项目根目录下运行,因此注意根据实际需求调整enc文件的路径。
最后别忘了在.gitignore
中忽略keys/evandroid.jks
以及gradle.properties
并在代码库中将其删除。
Travis CI的script
部分运行成功后,可以通过配置文件进入到发布阶段。下面是一个Travis CI发布的示例:
deploy:
provider: releases
user: "GITHUB USERNAME"
password: "GITHUB PASSWORD"
file: app/build/outputs/apk/app-release.apk
skip_cleanup: true
on:
tags: true
这个例子中配置了这样一些内容:
provider
:发布目标为Github Release,除了Github外,Travis CI还支持发布到AWS、Google App Engine等数十种providerfile
: 发布文件,输入文件路径即可skip_cleanup
: 默认情况下Travis CI在完成编译后会清除所有生成的文件,因此需要将skip_cleanup
设置为true来忽略此操作。on
: 发布的时机,这里配置为tags: true
,即只在有tag
的情况下才发布。虽然这样就能完成自动发布,但是直接暴露了Github密码是我们更加不能接受的。更好的做法是在Github -> settings -> Personal access tokens 生成一个只能访问当前项目并只有读取权限的Github Access Token,并通过Travis CI将Access Token加密。听起来有点繁琐,好在Travis CLI中已经可以通过一行指令做好这一切:
travis setup release
根据提示填写上述配置项目的信息后,Travis CLI会自动在.travis.yml
文件中生成好所有的配置项:
deploy:
provider: releases
api_key:
secure: XXX
file: app/build/outputs/apk/app-release.apk
skip_cleanup: true
on:
tags: true
all_branches: true
其中api_key
下的secure
就是加密后的Access Token。
在运行travis setup release
时有可能遇到
Invalid scheme format: git@github.com for a full error report, run travis report
这样的报错,看起来是Travis CLI还不支持通过密钥访问Github,因此可以将项目的源临时切换为http形式,运行成功后再切换回来:
git remote set-url origin https://github.com/AlloVince/evandroid.git
git remote set-url origin git@github.com:AlloVince/evandroid.git
在实际部署过程中,发现发布到Github Release比较坑的点是
git push
git push --tags
往往会同时生成2个Travis CI任务,但是在Travis网页中默认界面只能看到最后跑的一个任务,而未打Tag的任务又会报
Skipping a deployment with the releases provider because this is not a tagged commit
这曾让我一度以为自己的脚本哪里写错了,但是又找不到错误原因……
自动发布到Github对于开发人员已经足够,但是考虑到项目实际需要以及国情,还是有必要选择一个国内的App分发服务,fir.im、蒲公英都是不错的选择,不但允许游客下载,还提供了二维码等更适合对接手机的功能,国内下载速度也很快。由于fir.im提供了比较方便的CLI工具,因此本文以fir.im为例,在.travis.yml
中添加以下几行:
before_install:
- gem install fir-cli
after_deploy:
- fir p app/build/outputs/apk/app-release.apk -T $FIR_TOKEN -c "`git cat-file tag $TRAVIS_TAG`"
即在环境构建阶段安装fir-cli,在发布成功后通过fir命令行工具将apk上传到fir。
其中$FIR_TOKEN
可以在fir.im的用户->API Token中找到,然后在Travis CI控制台中创建环境变量FIR_TOKEN
并粘贴即可。
这里有个小技巧,如果我们仅仅上传APK文件到fir.im,看到链接的测试人员其实并不知道这次发布所包含的变动,因此通过git cat-file tag $TRAVIS_TAG
将当前发布tag所包含的附加信息一同上传了。其中$TRAVIS_TAG
变量是Travis CI每次运行自动附带的环境变量,还有很多其他的Travis环境变量供我们玩出更多花样。
虽然Travis CI也有通知功能,但不能定制模板,通知内容也仅仅为提示CI运行的结果,显然更适合开发人员。我们还是希望最终能以更友好的方式通知团队成员,同时考虑到邮件送达率,可以优先选择如Submail、SendCloud等国内邮件发送服务。
这里以Submail为例,首先需要在Submail内创建邮件模板,比如我们可以创建这样一封触发式邮件模板:
Hi 亲
@var(TRAVIS_REPO_SLUG)新版本@var(TRAVIS_TAG)已经发布了,功能更新:
@var(TAG_DESCRIPTION)
去下载:
http://fir.im/w13s
创建后可以得到邮件模板id,根据Submail手册,将模板中所需要的变量置入,最终可以使用一行Curl指令发送一封邮件:
after_deploy:- curl -d "appid=10948&to=allo.vince@gmail.com&subject=[自动通知] 安卓新版本$TRAVIS_TAG发布&project=u2c0r2&signature=$SUBMAIL_SIGN&vars={\"TRAVIS_REPO_SLUG\":\"$TRAVIS_REPO_SLUG\",\"TRAVIS_TAG\":\"$TRAVIS_TAG\",\"TAG_DESCRIPTION\":\"$(git cat-file tag $TRAVIS_TAG | awk 1 ORS=‘<br>‘)\"}" https://api.submail.cn/mail/xsend.json
其中Submail用到的认证凭据signature同样是通过Travis CI控制台配置的。
最终完成的示例项目在此。其实所有的yml文件配置不到30行,就能省去繁琐的日常工作,何乐而不为呢。最后回顾一下自动化后的日常工作:
提交代码:
git add .
git commit -m "这里是注释"
git push origin
打Tag
git tag -a v0.0.1-alpha.1 -m "这里是Tag注释,说清楚这个版本的主要改动,也可以省略-m参数直接写长文本"
git push origin --tags
如果发现打错了tag,可以删除本地及远程tag
git tag -d v0.0.1-alpha.1 git push origin --delete tag v0.0.1-alpha.1
问啊APP,程序员答题神器,解决你所有的技术难题, http://t.cn/R4vE2d7 下载注册送5元 快去下载注册吧!
QQ群290551701 聚集很多互联网精英,技术总监,架构师,项目经理!开源技术研究,欢迎业内人士,大牛及新手有志于从事IT行业人员进入!
基于Travis CI搭建Android自动打包发布工作流(支持Github Release及fir.im)
标签:
原文地址:http://www.cnblogs.com/fengliucaizi/p/5047489.html