Jenkins Pipeline (简称Pipeline)是支持Jenkins实现和集成持续交付流水线的一套插件。【摘自:https://jenkins.io/doc/pipeline/tour/hello-world/】
本来Jenkins的界面比较原始,加入了Pipeline,这种表现方式不错,很直观,而且每一个构建的步骤都可以查看日志。
本文题目写的是Pipeline与Docker,但是Pipeline和Docker之间没必然联系。
- 用原本的方式可以部署Docker容器承载的应用
- 用Pipeline可以部署非Docker容器承载的应用
- 用Pipeline可以部署Docker容器承载的应用
总之就是个排列组合关系,具体怎么做,按照你的需求来就好。
一、如何创建Pipeline工程?
- 新建工程
- 填写一个合法的名称
- 选择流水线(先来个简单的)
- 确定
- 配置
进入配置界面,构建触发器之类都可以先不管,请先关注流水线。
Pipeline script 使用代码方式描述构建,很酷!
可以试一下右边的Sample,保存就可以了。
点击立即构建就可以手动触发构建。
构建成功,因为脚本太简单了,出现了这个提示:"This Pipeline has run successfully, but does not define any stages. Please use the stage step to define some stages in this Pipeline."
下面我们做些更复杂的事情。
二、Jenkinsfile
又要引出一个概念了:Pipeline as Code(可以看一下这个网址https://jenkins.io/doc/book/pipeline-as-code/)
Pipeline as Code 描述了一组功能,允许Jenkins用户使用代码定义流水线作业流程,并在源代码存储库中存储和版本化。
要在源代码存储库中存储和版本化,就要以文件的形式保存。这个文件要存储在Repository的根目录,文件的名字默认Jenkinsfile,不建议改成别的,毕竟也算是一条规约。
Jenkinsfile的内容写什么呢?
Jenkinsfile支持两种方式的脚本:Declarative Pipeline 和 Scripted Pipeline。
脚本语法参考:https://jenkins.io/doc/book/pipeline/syntax/
Declarative Pipeline 声明式Pipeline
分为pipeline、stage、step三个级别。
stage是Pipeline的一部分,用于定义整个Pipeline的一个概念上不同的子集,例如:“Build”,“Test”和“Deploy”,许多插件使用它来可视化或呈现Jenkins管道状态/进度。(参考:https://jenkins.io/doc/book/glossary/#stage)
step是一项任务;从根本上讲,step告诉Jenkins Pipeline或项目内部要做什么。(参考:https://jenkins.io/doc/book/glossary/#step)
下面举个例子:
pipeline {
agent any
// 定义了参数输入
parameters {
string(name: 'PERSON', defaultValue: 'Mr Jenkins', description: 'Who should I say hello to?')
}
stages {
stage('Example') {
steps {
echo "Hello ${params.PERSON}"
}
}
stage('Example2') {
steps {
echo "Nice to meet you, ${params.PERSON}"
}
}
}
}
看执行结果
Scripted Pipeline 脚本式Pipeline
使用groovy语法的脚本,更加简单明了,程序员最爱。
分为node、stage两级,stage中使用脚本代码进行步骤操作。
例子如下:
node {
/* 定义构建参数 */
properties ([parameters([string(defaultValue: '', description: '请输入要发布的代码标签', name: 'GIT_TAG'),string(defaultValue: '', description: '请输入要发布的版本', name:'VERSION'), string(defaultValue: '', description: '请输入更新内容', name:'RELEASE_NOTE')])])
if (params.GIT_TAG == '' || params.VERSION == '') {
error '参数输入错误'
}
stage('Build') {
checkout([$class: 'GitSCM', branches: [[name: "refs/tags/${params.GIT_TAG}"]], doGenerateSubmoduleConfigurations: false, extensions: [], submoduleCfg: [], userRemoteConfigs: [[credentialsId: 'xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx', url: '[email protected]:xxxx/example.git']]])
dir ('example') {
/* 构建镜像 */
def customImage = docker.build("example-group/example:${params.VERSION}")
/* hub.xxxx.cn是你的Docker Registry */
docker.withRegistry('https://hub.xxxx.cn/', 'docker-registry') {
/* Push the container to the custom Registry */
customImage.push()
customImage.push('latest')
}
}
}
stage('Deploy-Testing-Env') {
if (currentBuild.result == null || currentBuild.result == 'SUCCESS') {
/* 如果Jenkins服务器和部署的服务器不是同一台,可以使用ansible或者其他远程的方式调用部署 */
/* 强制删除原来的容器 */
sh 'ansible ciserver -a "docker rm -f example"'
/* 从Registry拉取镜像 */
sh 'ansible ciserver -a "docker pull hub.xxxx.cn/example-group/example:latest"'
/* 运行容器 */
sh 'ansible ciserver -a "docker run -d --name example hub.xxxx.cn/example-group/example:latest"'
/* 发送邮件 */
mail bcc: '', body: "XXXX已部署\n地址:http://xxx.xxx.xxx/xxx\n\n代码标签:${params.GIT_TAG}\n更新内容:\n${params.RELEASE_NOTE}\n\n----------\n本邮件为自动发送\nBy Jenkins", cc: '', from: '[email protected]', replyTo: '', subject: "【Jenkins】测试环境已更新${params.VERSION}", to: '[email protected]'
}
}
stage('Test') {
sh 'echo "Run automated test"'
}
}
例子比较简单,还没有做异常的处理。
不过我把Pipeline同Docker的联动都写进去了。
我理解Pipeline同Docker的联动,无非就是两点:
- 在Build的时候构建镜像,push到Registry。
- 在Deploy的时候拉取镜像,部署。
先写到这里。
更多的例子可以看这:https://github.com/jenkinsci/pipeline-examples