赤兔项目持续集成系统-Jenkins CI部署¶
当前(2013-10-16)赤兔项目主要针对:android, iOS和Web浏览器三个平台发布客户端。本文主要介绍客户端的持续集成。
项目文件结构说明¶
http://devhome/repos/rhsrc/trunk/src/Client2下的代码目录结构为:
Client2/
Assets/
Editor/
Bundles/
Library/
ProjectSettings/
StreamingAssets/
Temp_Data/
Tex/
StreamingIOS/
Tools/
bin/
etc/
...
WebPlayer/
其中:
- “Assets”目录存放着所有源代码和原始资源
- “Assets/Editor”目录存放着命令命令编译脚本“CommandCompiler.cs”
- “Bundles”用于存放Web版客户端编译后资源 (可以在编译代码中调整)
- “StreamingAssets”用于存放Android版客户端编译后资源(可以编译代码中修改调整)
- “StreamingIOS”用于存放iOS客户端编译后资源(可以在编译代码中修改调整)
- “Tools存放命令行编译工具
- “WebPlayer”基本废弃
资源的链接¶
当前项目的设定中,将编译后的资源根据不同的平台分别存放在Bundles, StreamingAssets, StreamingIOS这三个目录。 而根据Unity3D的特殊目录要求,资源应该保存在Assets/StreamingAssets中,所以在编译输出二进制包和xcode项目文件时,需要将对应编译后的资源文件连接至此。如:当编译iOS的xcode项目文件时,需要将Assets/StreamingIOS目录链接到Assets/StreamingAssets;编译Android客户端时,需要将StreamingAssets链接到Assets/StreamingAssets等。
部分贴图资源仅在目录StreamingAssets/Tex下维护,所以当编译Web客户端和iOS客户端时需要将此两个目录拷贝到Bundles或StreamingIOS
由于Bundles,StreamingAssets和StreamingIOS每次SVN更新可能有新的资源加入,也可能有资源被移除,所以在每次编译前需要将子目录都清空,进行SVN更新时将这些目录排除。
软件需求¶
开发及编译工具:
- 服务端使用gcc和automake进行编译
- Unity3D (>=4.1.2f)
- Android SDK (API Level >= 10)
- Xcode (>=4.6)
- Subversion客户端
- Unity3D CacheServer
Jenkins CI系统
- jenkins (>=1.534)
- jenkins插件:Jenkins Unity3d plugin, Jenkins SSH Slaves plugin,Log Parser Plugin, SSH Credentials Plugin, Token Macro Plugin,Xcode integration, IPMessager plugin等。
- jre(Java Runtime Enviroment. >=1.7)
上述开发工具和软件在内网服务器上均有保存。(\\server\SOFTHOME\开发相关\CI)所有运行jenkins的系统(包括slave)都必需安装java运行时环境(jre),其它软件根据不同平台角色需要安装即可。注意jenkins和相应插件必要时应进行升级更新(出现莫名其妙的问题,请检查jenkins官方网站的更新升级日志)
配置Master-Slave¶
由于不同的版本需要在不同平台上编译,所以jenkins需要采用Master-Slave形式将编译工作分配到不同的平台执行。当前使用Windows平台作为Master,Mac OS和Linux作为Slave。(当前使用过程中发现,使用JNLP方法连接slave时,随着时间的推进,java会战胜大量内存,而且经常断开。可能原因是编译是产生了大量日志>100M)
Master和Slave都需要有足够的磁盘空间存放赤兔客户端源代码,编译中间文件和输出文件。[1]
安装及配置Master¶
安装jenkins Master¶
在作为Master的主机上安装jre
设置环境变量JENKINS_HOME,JENKINS_HOME所指向的目录应该有足够的空间(每个job >= 1GB)
从命令行启动jenkins [2]
java -jar Your_Path/jenkins.war --httpPort=80 # 如果是在linux上运行,可以使用daemon模式 # Windows可以在进行主界面后安装成系统服务 java -jar Your_Path/jenkins.war --daemon --httpPort=80
当jenkins 界面后可以对系统进行第一步配置:
- 从浏览器打开jenksin后,进入插件管理(系统管理 -> 管理插件 -> 高级 -> 上传插件)。安装必备的一些插件:Jenins Unity3d plugin, Jenkins SSH Slaves plugin,Log Parser Plugin, SSH Credentials Plugin, Token Macro Plugin,Xcode integration, IPMessager plugin等。
- 指定Unity3D安装位置(系统管理 -> 系统设置 -> Unity3D -> Unity3D安装)。即使不在master执行编译任务,也需要设定一个名字,因为向其它节点添加工具路径时需要此值。
- 指定Xcode安装信息(系统管理 -> 系统设置 -> Xcode Builder)
- 指定jenkins使用的Subversion客户端版本(系统管理 -> 系统设置 -> Subversion ->Subversion Workspace Version)
- 指定Unity3D日志分析工具的规则集(系统管理 -> 系统设置 -> Console OutputParsing)
- 添加SSH登陆凭证(系统管理 -> Manage Credentials)。在这里添加需要通过SSH登陆的主机的用户名和密码。
- 添加“标签”
安装插件¶
安装一些jenkins插件,可以更好的进行持续集成:
- Environment Injector Plugin 可以输出预定义变量以供构建时使用,可以大大方便对job参数的修改。如当前Unity3D项目,Unity3D命令行编译时需要ProjectPath,将编译资源和输出二进制包分开,就需要输入两次,当需要进行修改时,极易出现遗漏,导致构建失败。
- Build Pipeline Plugin
- IPMessage 便于通知用户,不足之处是中文显示错误。
配置Slave¶
根据不同平台和版本的需求,需要将不同的工作分发给Slave来执行。如:iPhone/iPad 版客户端只能在MacOS上进行编译。
以配置MacOS Slave为例:
- 在主机上安装jre,Unity3D等必需软件
- 在浏览器打开Master界面,进行系统管理 -> 管理节点 -> 新建节点。
- 设定节点名,
- 类型选择Dumb Slave。
- 然后进一步设定节点信息。其中重点关注的是:
- 可同时执行文件数“# of excutors”
- Slave上的工作目录“Remote FS root”。此目录所在分区必须有足够磁盘空间[4]
- 设定一个标签,便于工作分类。“Labels”
- 连接Slave的方法“Launch method”。对于MacOS(类Unix系统)使用SSH连接是最为便利的;对于Windows则使用Java Web Start比较方便,当然也可以使用cygwin或其它ssh工具连接Windows。接着指定slave的IP,选择已经保存的凭证(Credentials)
- 设定要使用的工具路径(Unity3D)
- 设定slave上的工作目录,并确认此目录存在且ssh用户具有读写权限。
- 回到系统管理 -> 管理节点就可以看到节点连接状态。如果连接失败,点击相应的节点名,查看日志,修正问题
- 用于编译android客户端的主机,还需要安装Android SDK,并且需要打开Unity3D指定路径 1. 用Unity3D打开客户端项目,指定Android SDK位置。(Editor -> Preference -> External Tools -> Android SDK Location) 2. 如果有配置CacheServer,同样在Unity3D中指定CacheServer(Editor -> Preference -> Cache Server)
- 编译iPhone/iPad客户端,只能在Mac OS平台上进行,步骤也相对复杂,需要导入prov, 证书,证书需要解密,需要编译为IPA文件等 1. 新建一个本地用户(xcode),所有CI相关操作都通过此用户完成 2. 导入Apple开发者证书并确认证书有效 3. 导入mobileprovision文件,并确认有效[5]
在使用master/slave进行构建前,最好手动Checkout一份代码;然后用Unity3D打开,手动完成一次完整的编译,并确保成功。对于iPhone/iPad客户端,需要先由Unity3D输出一个Xcode项目代码;用Xcode打开,并编译一个APP,确认整个流程正常。
slave节点连接正常后就可以创建任务并向其分发。
创建工作任务¶
当Master配置好后,如果满足编译环境要求即可进行编译;当连接上Slave后,即可将的工作分发给slave执行,对于不同的任务,可以通过标签来加以限定。 手动确认slave上的编译工作可以正常完成后,可以打开jenkins的界面创建新的工作。 为了便于进行持续集成,所以通常使用Unity3D的命令行编译来完成编译。[6]
Android¶
确认作为slave的主机是否已经连上master。
确认jenkins已经安装了插件Jenkins Unity3d plugin, Xcode integration
“新Job -> 设定任务名称 -> 类型选择自由风格或拷贝已存在任务”,进入任务详细设定
为任务指定一个标签(创建Slave时设定的),以限制任务的运行节点(Restrict where this project can be run),这样就可以将不同的任务分发到不同的主机了。如iOS客户端分发到MacOS编译,而Android客户端分发给Windows编译。
源码管理可以选择Subversion,但由于当前代码冲突的问题没有很好解决,所以使用的是None。在构建步骤中增加构建步骤通过命令来更新SVN。
增加构建步骤执行自动构建和发布客户端。
更新SVN:svn up –force –accept tf
使用Unity3D命令行编译,命令行编译参数为:
# 编译资源 -batchmode -projectPath $UNITY3D_PROJECT_DIR -executeMethod CommandCompiler.CompileResource -quit # 编译xcode项目 -batchmode -quit -projectPath $UNITY3D_PROJECT_DIR -executeMethod CommandCompiler.PerformBuild "" "onwind" "yourpath" "iPhone"
客户端编译完成后,可以通过sftp, ftp等方法将客户端发布:
::echo off set GAME_HOST=10.1.0.190 set GAME_PATH=/var/www/html/install/branch/android set GAME_VERSION=0.00.02 set SCP=D:\CI\soft\PSCP.EXE set SSH_USER=root set SSH_PASSWD=setupthepassword set iFILE=E:\output\android.apk :: BUILD_NUMBER and BUILD_ID come from jenkins CI System set oFILE=E:\output\fhsgCommon_%GAME_VERSION%_%BUILD_NUMBER%_%BUILD_ID%.apk move %iFILE% %oFILE% %SCP% -batch -pw %SSH_PASSWD% %oFILE% %SSH_USER%@%GAME_HOST%:%GAME_PATH%
另外可以使用Public-over-ssh/ftp/samba等插件来进行发布客户端。
添加构建后动作-进行日志分析和通知相关人员构建结果
- Log Parser Plugin可以添加规则来分析构建日志。
- IPMessager Plugin可以通过IPMessage(飞秋)即时通知
- 还有一些其它通知插件
iOS/iPhone/iPad¶
确认MacOS Slave已经连接上Master,确认jenkins已经安装了插件Jenkins Unity3d plugin, Xcode integration
类似Android任务创建一个新的任务
通过“restrict where this project can be run 参数将此任务限制在MacOS上运行
源码管理同样选用“None”,通过命令行来控制源码的更新
添加构建步骤:
更新源码
cd ${WORKSPACE} svn up --force --accept theirs-full Assets ProjectSettings StreamingAssets\Tex Tools StreamingAssets\hud.cfg StreamingAssets\gameconfig.cfg
Unity3D命令行编译输出Xcode项目代码,命令行编译参数为:
# 编译资源 -batchmode -projectPath $UNITY3D_PROJECT_DIR -executeMethod CommandCompiler.CompileResource -quit # 编译xcode项目 -batchmode -quit -projectPath $UNITY3D_PROJECT_DIR -executeMethod CommandCompiler.PerformBuild "" "onwind" "yourpath" "iPhone"
Xcode编译iOS APP程序,需要注意下面的参数设定:
- Clean before build
- Xcode Project Directory
- Build output directory
- Build IPA?
- Unlock Keychain?
- Keychain path (${HOME}/Library/Keychains/login.keychain)
- Keychain password (帐号登陆密码)
Nightly Build¶
针对每一个平台(客户端)建立一个每日构建的任务,以保证每天提交的代码是可以通过编译,每日构建任务与上面任务的建立方法几乎完全一样,唯一的差别在于源码控制。每日构建任务需要每次使用完全干净的代码(与SVN服务器上的代码一致)。所以建议使用jenkins内置的版本控制工具来管理源码,每次执行构建之前都从SVN服务器上checkout一份新的源码,或者更新源码之前对当前代码进行svn revert。基本步骤:
- 按照通用方法建立任务。
- 在任务配置页中,Source Code Management部分选择使用subversion。填写好相关选项,如:Repository URL, Credentials,最重要的是:Check-out Strategy 项,默认为:”Use ‘svn update’ as much as possible“,需要更改为选项:”Use ‘svn update’ as much as possible, with ‘svn revert’ before update“以保证更新代码前revert所有本地更改。
- 构建步骤按相应平台(客户端)设置即可。
每日构建的关键是:每次执行构建时使用一份全新,干净的源码。然后自动部署到测试环境中,由测试人员进行测试。而日常由于各种情况需要不定期进行编译,此时执行其它非每日构建的任务完成构建。
建议步骤¶
由于目录结构的原因,当前编译生成的资源目录也在SVN的管理下,所以可能存在最终资源的混乱,建议在正式自动集成前执行下面操作:
此步骤为必需步骤:将“Bundles, StreamingAssets, StreamingIOS”根据不同平台需求,建立一个符号链接至“Assets/StreamingAssets”。(可使用脚本SetupAPK.bat和SetupIOS.sh完成此操作)
对于不同的平台,SVN更新时将其它平台的资源文件排除不更新,以减少更新时间。如Windows平台输出Web版本客户端时,使用TortoiseSVN从SVN选择性的checkout与当前平台相关的目录;更新时只更新必需目录。
:: 必须在jenkins的构建步骤中添加 svn up --force --accept theirs-full Assets ProjectSettings Tools StreamingAssets/Tex StreamingAssets/gameconfig.cfg StreamingAssets/hud.cfg
对于其它平台每第一次运行时,执行类似的命令排除无关目录
利用插件Environment Injector Plugin对环境(变量)进行统一管理
针对每个平台建立一个由jenkins内置版本控制工具(SVN)管理源码的工作,实现每日构建确保每天提交的代码可以正常工作。
构建步骤¶
Unity3D的编译过程可以通过代码自定制,相当比较灵活。当前编译程序位于Assets/Editor目录下,BuildBundle.cs是用于GUI界面编译的菜单选项;CommandCompiler.cs是命令编译代码。资源和程序的编译功能由程序维护。CommandCompiler.cs中需要根据不同平台,不同渠道分别进行一些设定。对于iPhone/iPad可能来需要向生成的xcode项目添加一些额外的SDK文件,此时需要利用Unity3D提供的Post Process BuildPlayer功能。[7]
Unity3D的命令行编译¶
Unity3D支持命令行编译,常用命令行参数选项有:[9]
- -batchmode 启用命令模式
- -projectPath 指定项目路径。Unix环境可以使用环境变量$HOME
- -executeMethod 指定执行编译的类与其方法
- -quit 完成自动退出。没有此选项,即使编译完成也不会返回
- -buildTarget 激活相应的平台。由于SVN上只保存一份代码(即只针对一个平台),所有在编译时需要将工程切换到指定平台。
一个标准的命令行编译命令如:(与平台无关)
unity3d -batchmode -projectPath $HOME/jenkins/workspace/android.trunk.rh.onwind.cn
PostprocessBuildPlayer¶
Unity3D程序编译Player完成后会执行Editor目录下PostprocessBuildPlayer程序(任意可执行代码)进行相关操作。[8]在当前项目中由于不同渠道提供的SDK千奇百怪,有时需要使用此功能向xcode项目中添加文件。当前项目中的PostprocessBuildPlayer是使用python所写,用于向xcode项目添加91的SDK。
Unity调用PostprocessBuildPlayer时会向其传递7个参数:
#!/usr/bin/perl
my $installPath = $ARGV[0];
# The type of player built:
# "dashboard", "standaloneWin32", "standaloneOSXIntel", "standaloneOSXPPC", "standaloneOSXUniversal", "webplayer"
my $target = $ARGV[1];
# What optimizations are applied. At the moment either "" or "strip" when Strip debug symbols is selected.
my $optimization = $ARGV[2];
# The name of the company set in the project settings
my $companyName = $ARGV[3];
# The name of the product set in the project settings
my $productName = $ARGV[4];
# The default screen width of the player.
my $width = $ARGV[5];
# The default screen height of the player
my $height = $ARGV[6];
print ("\n*** Building at '$installPath' with target: $target \n");
在三国中我们使用python来实现PostprocessBuildPlayer向Xcode项目中添加文件:
#!/usr/bin/env python
# -*- coding: utf-8 -*-
"""Unity3D PostprocessBuildPlayer
用于向Unity3D导出的xcode项目中添加额外的文件。
由于unity3D会输出各种不同的客户端,并接入不同的平台(使用不同的SDK),由此知道\
会添加各种不同的文件,而Unity3D再编译完成后均会调用\ ``PostprocessBuildPlayer``\
。为了可以正确的添加相应的SDK信息,需要在\ ``PostprocessBuildPlayer``\ 中根据\
Unity3D传递的参数来进行判断。Unity3D会向\ ``PostprocessBuildPlayer``\ 传递一些\
参数,其中\ ``sys.argv[1]``\ 为安装路径(即导入xcode的路径);\ ``sys.argv[2]``\
为BuildTarget(即:Andriod, iPhone ...)。
**在编译时将不同版本,不同运营商的客户端输出到不同的路径,上面两个参数就可以用\
于确定相应添加什么文件。**
Author: Liu Hui
Date: Sat Feb 8 14:07:41 CST 2014
"""
import sys
from mod_pbxproj import XcodeProject
# 请根据实际情况修改\ ``xcode_path, agent``\ 的值
# ``agent``\ 的值与命令行编译脚本\ ``CommandCompiler.cs``\ 中的output相对应
xcode_path = '/Users/xcode/xcode_project'
agent = 'dj91'
xcode_project_path = '%s/%s' % (xcode_path, agent)
if sys.argv[2].lower() != 'iphone':
sys.exit(0)
elif sys.argv[2].lower() == 'iphone' and sys.argv[1] == xcode_project_path:
pass
# 请根据实际情况修改或增加SDK路径:\ ``DJ_PATH``\ 的值
DJ_PATH = '/Users/xcode/GameSDK/DJGameSDK1'
system_framework_path = '/Applications/Xcode.app/Contents/Developer/Platforms/iPhoneOS.platform/Developer/SDKs/iPhoneOS6.1.sdk/System/Library/Frameworks'
pbxproj = '%s/Unity-iPhone.xcodeproj/project.pbxproj' % xcode_project_path
# 添加第三方SDK信息
dj_resource = '%s/Resources' % DJ_PATH
dj_frm = '%s/DJGame.framework' % DJ_PATH
dj_unity3d_frm = '%s/DJGameForUnity3D.framework' % DJ_PATH
# 添加依赖系统框架
messageui = '%s/MessageUI.framework' % system_framework_path
coretext = '%s/CoreText.framework' % system_framework_path
project = XcodeProject.Load(pbxproj)
project.add_folder(dj_resource)
project.add_folder(dj_frm)
project.add_folder(dj_unity3d_frm)
project.add_folder(messageui)
project.add_folder(coretext)
if project.modified:
project.backup()
project.save()
基本构建步骤¶
当前由于需要在资源编译完成后生成gameconfig.cfg和资源的版本信息文件version.cfg,所以当前将整个编译过程分为四部分
- 从SVN上更新完代码后,清理上次编译生成的资源
- 调用Tools/bin/datamaker.py生成数据文件,并检查当前工程的资源读取方式是否正确
- 编译资源
- 调用Tools/bin/fileversion.py生成gameconfig.cfg和version.cfg
- 编译二进制文件(Android系统为apk包,iPhone/iPad为xcode项目文件)
- 进一步编译使用xcodebuild和xcrun编译xcode项目,生成ipa文件
常见问题¶
资源,特效,贴图丢失
导致这些问题的原因大多是因为资源的meta文件丢失或混乱所造成的。
- gameconfig.cfg文件是否更新正常
- 在Unity3D中运行游戏,运行到故障场景时,查看相应的资源加载情况,找到丢失了什么资源,然后去检查相应的meta文件是否存在,与prefeb目录中一致。找到不一致的原因。
- 也可能是某次更新时,资源的meta文件丢失,编译时Unity3D会自动为没有meta文件的资源创建一个新的meta文件;而后来丢失的meta文件被补充至SVN服务器,当再次更新时,SVN服务器上的meta文件将不会被下载,就会导致meta文件混乱而找不到资源。
项目属性的设定
对于不同的版本的客户端,其输出参数不尽相同。在代码中可以通过Unity3D中的PlayerSettings类进行设定;在图形界面可以通过菜单File -> Build Settings -> Player Settings打开选项卡进行设定。当前已通过代码的方式指定(“Assets/Editor/CommandCompiler.cs”)。
SVN更新时冲突的解决
说明¶
[1] | Master的临时文件夹所在分区也应该有足够磁盘空间,否则master将不能执行job并离线。 |
[2] | 如果主机上运行着其它服务占用了80, 8080等端口,可以通过命令行参数调整jenkins侦听的端口。 |
[3] | jenkins界面语言与你的浏览器默认语言一致。即浏览器默认英文则为英文界面,默认为中文则为中文界面。 |
[4] | 相关阀值由系统管理 -> 管理节点 -> 设置处指定 |
[5] | 证书可以通过查看钥匙串确认是否有效;mobileprovision需要打开Xcode查看 |
[6] | file:///C:/Program%20Files/Unity/Editor/Data/Documentation/Documentation/Manual/CommandLineArguments.html |
[7] | file:///C:/Program%20Files/Unity/Editor/Data/Documentation/Documentation/Manual/BuildPlayerPipeline.html |
[8] | file:///C:/Program%20Files/Unity/Editor/Data/Documentation/Documentation/Manual/BuildPlayerPipeline.html |
[9] | http://www.unity3d.com |