使用Jenkins搭建持续集成服务

Jenkins Service

Posted by Alpaca on March 1, 2016

使用Jenkins搭建持续集成服务

1. 什么是持续集成

持续集成(Continuous Integration, 简称 CI) 是软件工程中的一种实践, 用于将开发人员不同阶段的工作成果集成起来, 通常一天之中会进行多次. 持续集成最初在 极限编程(Extreme Programming) 中提出, 主要用于执行自动化测试. 目前持续集成的概念已经逐渐独立出来, 并扩展为构建服务器(Build Server), 质量控制(Quality Control) 和持续交付(Continuous Delivery) 等多种形式和实践. 详细可以参见Wikipedia.

下面我们就看一下如何搭建Jenkins(一个基于Java的持续集成工具) 并用于执行自动化测试. 我们要达到这样的效果: 在向位于BitBucket的项目 push 代码时, Jenkins 自动获取最新的代码并执行测试, 并将测试结果通过Email或其他方式通知我们.

另外, 本文也会介绍几个持续集成服务商, 可以免去自己安装和维护的工作. 详见持续集成服务商.

2. 安装和配置Jenkins

下面介绍如何在Ubuntu系统上安装和配置Jenkins. 如果你对Ubuntu系统不熟悉, 可以参考 Linux服务器的初步配置流程, 那里有如何配置 sudo 等信息.

2.1 安装Jenkins

详细的安装步骤可以参考官方文档, 主要步骤如下:

1
2
3
wget -q -O - http://pkg.jenkins-ci.org/debian/jenkins-ci.org.key|sudo apt-key add -
sudo sh -c'echo deb http://pkg.jenkins-ci.org/debian binary/ > /etc/apt/sources.list.d/jenkins.list'sudo apt-get update
sudo apt-get install jenkins

这样就会自动安装Jenkins和它的依赖(如OpenJDK), 并自动在 8080 端口启动Jenkins. 如果8080端口已被占用则会启动失败, 可以关闭占用8080端口的程序, 或者更改Jenkins的配置使用其他端口(见后文).

为了执行项目自动化测试而需要的软件和工具可以在这里一并安装和配置, 例如我的项目需要 Rvm, Ruby 和PostgreSQL.

2.2 Jenkins命令

以下是用于管理Jenkins服务的常用命令:

  • 查看Jenkins是否正在运行:sudo service jenkins status
  • 运行Jenkins:sudo service jenkins start
  • 停止Jenkins:sudo service jenkins stop
  • 重启Jenkins:sudo service jenkins restart

2.3 Jenkins运行选项

Jenkins默认使用的配置文件位于/etc/default/jenkins, 在这里可以更改Jenkins的运行选项. 例如, 如果要修改内存大小和运行端口, 可以使用sudo vi /etc/default/jenkins 打开这个文件然后修改这两个选项:

1
2
JAVA_ARGS="-Xmx512m"
HTTP_PORT=9080

修改配置文件后记得重启Jenkins:sudo service jenkins restart

2.4 Jenkins安全配置

接下来在浏览器打开host:port(例如 http://localhost:8080) 就可以访问到Jenkins的页面了, 其中host是你的服务器IP或域名,port是Jenkins的运行端口(默认是8080). 你也可以配合Nignx或Apache来使用80端口, 以及设置SSL等.

第一次运行Jenkins时要做的特别重要的一件事就是配置安全选项, 也就是访问和修改Jenkins的权限. 默认的设置是所有人拥有所有的权限. 我们来改成需要使用用户名和密码登录.

首先创建一个用户. 从左边的菜单区依次进入Manage Jenkins,Manage Users,Create User, 然后填入自己的登录信息.

之后回到首页, 依次进入Manage Jenkins,Configure Global Security, 勾上Enable securityJenkins’ own user database, 去掉Allow users to sign up 前面的勾.

接下来, 如果你的Jenkins是内部使用, 不想公开给大众, 可以使用Matrix-based security, 在User/group to add:后面输入刚才创建的用户名并点Add, 然后在出现的那一行勾上 Administer(Anonymous那一行都不要勾上).

而如果你想允许没登录的人查看Jenkins(只读权限), 可以使用Logged-in users can do anything.

以上是典型的安全配置, 你也可以根据自己的情况决定如何配置.

3. Jenkins Job 示例: 自动化测试

由于我们的示例需要使用 Git 和 Bitbucket, 先做一些必要的安装和配置.

3.1 安装和配置 Git 及 ssh key

首先安装Git:sudo apt-get install git

然后切换到jenkins用户, 下面的命令都要以这个用户的身份执行:

1
sudo su jenkins

jenkins用户配置 Git 用户名和email (根据你的情况进行修改):

1
git config --global user.name"Jenkins"git config --global user.email"[email protected]"

然后用ssh-keygen命名生成 ssh key. 可以一路回车, 默认生成的 ssh public key 位于~/.ssh/id_rsa.pub. 详细可以参见SSH原理与运用(一):远程登录.

然后运行一次ssh [email protected]并输入yes, 这样BitBucket就会被自动加到 known_hosts里了.

然后需要把 ssh public key 加到BitBucket项目的Deployment keys, 这样jenkins 用户才有权限 clone 项目代码. 执行cat ~/.ssh/id_rsa.pub并复制输出的 public key 内容 (注意不要复制多余的空行), 然后打开BitBucket项目管理页面的Deployment keys页面, 粘贴到Key文本框里,Label建议填写jenkins ci, 然后点击Add key.

最后, 需要给Jenkins安装Git插件. 在Jenkins页面依次选择Manage Jenkins,Manage Plugins, Available, 找到并勾上GIT plugin. 由于我的项目是使用了Rvm, 所以我也会勾上 Rvm. 然后点击页面最下方的Install without restart.

3.2 创建Job

下面我们就来创建用于执行自动化测试的Jenkins Job.

在Jenkins页面点击New Item, 在Item name输入Job名称, 例如my-project-rspec, 点击OK.

Source Code Management区域勾选Git,Repository URL填写项目在BitBucket的地址, 例如[email protected]:user/repo.git. 在Branches to build后面填写想要对哪个分支持续集成, 我们留空来针对所有分支.

Build Triggers区域勾选Poll SCM(ScheduleIgnore post-commit hooks留空).

如果要使用 Rvm 插件, 在Build Environment区域勾选Run the build in a RVM-managed environment, 并在Implementation后面填写你在 Rvm 中使用的 Ruby 版本, 例如2.0.0-p451.

Build区域依次选择Add build step,Execute shell, 然后在Command 右边的文本框输入执行持续集成的脚本, 例如为我的项目执行自动化测试:

1
2
bundle installexportRAILS_ENV=testbundleexecrake db:schema:load
bundleexecrspec spec

最后, 点击Save, 回到刚创建好的Job的页面, 点击Build Now, 查看Build的Console Output, 如果遇到错误就做相应的修改, 之后再Build Now直到成功.

3.3 配置自动触发

现在我们已经可以手动触发持续集成来运行测试了, 下面配置在 push 代码时自动触发.

进入BitBucket项目管理页面的Hooks页面, 在Select a hook…选择POST 并点击Add hook, 然后在URL填写JENKINS_URL/git/notifyCommit?url=GIT_REPO, 其中JENKINS_URL是你的Jenkins地址,GIT_REPO是在上一步填写的Git仓库地址. 例如:

1
http://localhost:8080/git/[email protected]:user/repo.git

然后点Save. 以后当有代码push到BitBucket项目时, Jenkins就会自动触发执行测试.

3.4 配置通知

最后, 我们需要配置通知以便在集成失败时得到通知. 如果你使用HipChatSlack 那么建议你直接使用他们接收通知, 只需安装对应的插件和简单的配置即可.

下面我们介绍如何让Jenkins发送Email通知. 首先需要配置SMTP. 依次进入Manage Jenkins, Configure System, 在E-mail Notification区域进行设置. 以Gmail为例: SMTP server输入smtp.gmail.com, 点击Advanced…, 勾选Use SMTP Authentication, User NamePassword输入 Gmail 邮箱地址和密码, 勾选Use SSL, SMTP Port填写465. 然后可以勾选Test configuration by sending test e-mail 并填写自己的Email地址再点Test configuraton进行测试, 如果可以收到测试邮件则说明配置成功. 然后点击页面底部的Save.

接下来进入Job页面, 点击Configure进入Job的配置页面, 在最下面的Add post-build action 选择E-mail Notification, 然后在Recipients输入接收通知邮件的邮箱地址, 多个地址之间用英文空格隔开 (建议使用邮件列表地址).

4. 持续集成服务商

以上就是Jenkins持续集成服务器的基本配置和使用. 除了某些情况下必须像这样自己搭建持续集成服务器外, 其实可以考虑一些SaaS 类型的持续集成服务, 例如Travis CI,CircleCI,CodeshipDrone. 此类的服务对于开源项目基本都是免费的. 对于私有项目的收费可能也会比自己搭建和维护持续集成服务器的成本更低, 值得考虑.

以下是我对这几个持续集成服务的初步印象:

*Travis CI: 在GitHub项目中非常流行,收费版的起步价($129)比较高. *CircleCI: 只支持GitHub项目. 听说速度很快. *Codeship: 网站界面很漂亮. 我在使用中遇到两个不常见的问题: 在用git push -f 覆盖了之前失败的一次commit时, Codeship会由于找不到之前的commit而无法继续执行; 另外那个触发自动执行的hook URL只在刚开始创建项目时可以看到, 之后在设置页面找不到, 这样如果在BitBucket/GitHub临时删除了hook就找不回来了. 这两个问题貌似只能通过删除并重建项目来解决.

*Drone: 基于Docker. 我在使用中发现的一个问题是, 每次执行都会从零开始构建环境, 例如我的Ruby项目, 每次执行bundle install都重新安装所有依赖导致用时太长, 暂时没有研究有无解决方法.

最后, 关于自己搭建持续集成服务, 除了Jenkins还有其他类似的工具, 例如GitLab CI, ThoughtWorks Go,Drone(开源版) 等, 可以参考Wikipedia开源中国社区上的列表.

文章使用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议