目 录CONTENT

文章目录

Pipeline 流水线指令:声明式和脚本式

Administrator
2024-02-28 / 0 评论 / 0 点赞 / 18 阅读 / 0 字

官方文档手册

https://www.jenkins.io/zh/doc/book/pipeline/jenkinsfile/

常用逻辑

在上一篇文章中,我们介绍了 Jenkins 2.x 实现流水线的两种语法,以及在实际工作中该如何选择脚本式语法或声明式语法。原文可查阅:「持续集成实践系列」Jenkins 2.x 搭建 CI 需要掌握的硬核要点(一)

在使用传统的 Jenkins Web 界面和项目时,比如自由风格类型的任务,我们对处理流程的控制能力是有限的。所采用的典型形式是任务链:任务完成后触发其他的任务。或者我们可能会包括构建后处理,不管任务成功完成与否,总是去做一些类似发送通知的事情。

除了这些基本的功能外,还可以添加条件性构建步骤插件,通过基于单个或者多个条件的构建步骤来定义更加复杂的流程。但即便如此,相比于我们编写程序时可以直接控制执行流程的方法,条件性构建步骤插件对流程的控制能力依然有限。

在本篇中,我们将聊一下,关于 Jenkins 流水线 DSL 语言所提供的用于控制流水线执行流程基本结构和一些常见技巧。

1. Pipeline 流水线指令常见结构

正如在系列第一篇文章中介绍到的,Jenkins DSL 采用的是 Groovy 脚本语言。这也意味着如果当你掌握了 Groovy 语言,可以按照需求在流水线中使用 Groovy 语言的结构和习惯用法,针对这一类使用者,通常会更倾向于用脚本式语法来实现流水线。但不管采用的是哪种语法,从流水线组成的角度来讲,都是由一些不同指令 + 步骤构建结构化代码块。

对于脚本式流水线,基本结构如下:

node('worker'){
    stage('阶段'){
        // DSL
    }
}

构建脚本式流水线常用的结构或者说代码块节点主要由nodestage两个组成。

而,声明式流水线基本结构构成环节相对要多一些,整理了一张图如下:
需要划一个重点:可以简单理解node是用于脚本式流水线,而agent则是用于声明式流水线。

Jenkins Pipeline 支持的指令 (常见):

指令名 说明 作用域
agent 指定流水线或特定阶段在哪里运行。 stage 或 pipeline
environment 设置环境变量 stage 或 pipeline
tools 自动下载并安装指定的工具,并将其加入到 PATH 变量中 stage 或 pipeline
input 暂停 pipeline,提示输入内容 stage
options 用来指定一些预定义选项 stage 或 pipeline
parallel 并行执行多个 step stage
parameters 允许执行 pipeline 前传入一些参数 pipeline
triggers 定义执行 pipeline 的触发器 pipeline
when 定义阶段执行的条件 stage
build 触发其他的 job steps

options Jenkins Pipeline 常见配置参数:

参数名 说明 例子
buildDiscarder 保留最近历史构建记录的数量 buildDiscarder(logRotator(numToKeepStr: '10')
timestamps 添加时间戳到控制台输出 timestamps()
disableConcurrentBuilds 阻止 Jenkins 并发执行同一个流水线 disableConcurrentBuilds()
retry pipeline 发生失败后重试次数 retry(4)
timeout pipeline 运行超时时间 timeout(time:1, unit: 'HOURS')

示例:

pipeline{
    agent any
    options{
        buildDiscarder(logRotator(numToKeepStr: '10')
        timestamps()
        retry(3)
        timeout(time:1, unit: 'HOURS')
    }
    stages{
        stage('demo'){
            steps{
                sh 'echo hello'
            }
        }
    }
}

更多 pipeline 指令,可参见官方介绍:

https://www.jenkins.io/doc/book/pipeline/syntax/#

0.agent

下述仅挑几个常用的,用于流水线流程控制选项的指令项,介绍一些常用技巧。 Agent 表示整个流水线或特定阶段中的步骤和命令执行的位置,该部分必须在 pipeline 块的顶层被定义,也可以在 stage 中再次定义,但是 stage 级别是可选的。

any

在任何可用的代理上执行流水线,配置语法

pipeline {  
  agent any  
}

none

表示该 Pipeline 脚本没有全局的 agent 配置。当顶层的 agent 配置为 none 时, 每个 stage 部分都需要包含它自己的 agent。配置语法

pipeline {  
  agent none  
  stages {  
    stage('Stage For Build'){  
      agent any  
    }  
  }  
}

label

以节点标签形式选择某个具体的节点执行 Pipeline 命令,例如:agent { label 'my-defined-label' }。节点需要提前配置标签。

pipeline {  
  agent none  
    stages {  
      stage('Stage For Build'){  
        agent { label 'role-master' }  
        steps {  
          echo "role-master"  
        }  
      }  
    }  
}

node

和 label 配置类似,只不过是可以添加一些额外的配置,比如 customWorkspace(设置默认工作目录)

pipeline {  
  agent none  
    stages {  
      stage('Stage For Build'){  
        agent {  
          node {  
            label 'role-master'  
            customWorkspace "/tmp/zhangzhuo/data"  
          }  
        }  
        steps {  
          sh "echo role-master > 1.txt"  
        }  
      }  
    }  
}

dockerfile

使用从源码中包含的 Dockerfile 所构建的容器执行流水线或 stage。此时对应的 agent 写法如下

agent {  
   dockerfile {  
     filename 'Dockerfile.build'  //dockerfile文件名称  
     dir 'build'                  //执行构建镜像的工作目录  
     label 'role-master'          //执行的node节点,标签选择  
     additionalBuildArgs '--build-arg version=1.0.2' //构建参数  
   }  
}

docker

相当于 dockerfile,可以直接使用 docker 字段指定外部镜像即可,可以省去构建的时间。比如使用 maven 镜像进行打包,同时可以指定 args

agent{  
  docker{  
    image '192.168.10.15/kubernetes/alpine:latest'   //镜像地址  
    label 'role-master' //执行的节点,标签选择  
    args '-v /tmp:/tmp'      //启动镜像的参数  
  }  
}

kubernetes

需要部署 kubernetes 相关的插件,官方文档:

https://github.com/jenkinsci/kubernetes-plugin/

Jenkins 也支持使用 Kubernetes 创建 Slave,也就是常说的动态 Slave。配置示例如下

  • cloud: Configure Clouds 的名称,指定到其中一个 k8s

  • slaveConnectTimeout: 连接超时时间

  • yaml: pod 定义文件,jnlp 容器的配置必须有配置无需改变,其余 containerd 根据自己情况指定

  • workspaceVolume:持久化 jenkins 的工作目录。

  • persistentVolumeClaimWorkspaceVolume:挂载已有 pvc。

workspaceVolume persistentVolumeClaimWorkspaceVolume(claimName: "jenkins-agent", mountPath: "/", readOnly: "false")
  • nfsWorkspaceVolume:挂载 nfs 服务器目录
workspaceVolume nfsWorkspaceVolume(serverAddress: "192.168.10.254", serverPath: "/nfs", readOnly: "false")
  • dynamicPVC:动态申请 pvc,任务执行结束后删除
workspaceVolume dynamicPVC(storageClassName: "nfs-client", requestsSize: "1Gi", accessModes: "ReadWriteMany")
  • emptyDirWorkspaceVolume:临时目录,任务执行结束后会随着 pod 删除被删除,主要功能多个任务 container 共享 jenkins 工作目录。
workspaceVolume emptyDirWorkspaceVolume()
  • hostPathWorkspaceVolume:挂载 node 节点本机目录,注意挂载本机目录注意权限问题,可以先创建设置 777 权限,否则默认 kubelet 创建的目录权限为 755 默认其他用户没有写权限,执行流水线会报错。
workspaceVolume hostPathWorkspaceVolume(hostPath: "/opt/workspace", readOnly: false)

示例

agent {  
  kubernetes {  
      cloud 'kubernetes'  
      slaveConnectTimeout 1200  
      workspaceVolume emptyDirWorkspaceVolume()  
      yaml '''  
kind: Pod  
metadata:  
  name: jenkins-agent  
spec:  
  containers:  
  - args: [\'$(JENKINS_SECRET)\', \'$(JENKINS_NAME)\']  
    image: '192.168.10.15/kubernetes/jnlp:alpine'  
    name: jnlp  
    imagePullPolicy: IfNotPresent  
  - command:  
      - "cat"  
    image: "192.168.10.15/kubernetes/alpine:latest"  
    imagePullPolicy: "IfNotPresent"  
    name: "date"  
    tty: true  
  restartPolicy: Never  
'''  
  }  
}

agent 的配置示例

kubernetes 示例

Docker+K8s+Jenkins 主流技术全解视频资料【干货免费分享】

pipeline {  
  agent {  
    kubernetes {  
      cloud 'kubernetes'  
      slaveConnectTimeout 1200  
      workspaceVolume emptyDirWorkspaceVolume()  
      yaml '''  
kind: Pod  
metadata:  
  name: jenkins-agent  
spec:  
  containers:  
  - args: [\'$(JENKINS_SECRET)\', \'$(JENKINS_NAME)\']  
    image: '192.168.10.15/kubernetes/jnlp:alpine'  
    name: jnlp  
    imagePullPolicy: IfNotPresent  
  - command:  
      - "cat"  
    image: "192.168.10.15/kubernetes/alpine:latest"  
    imagePullPolicy: "IfNotPresent"  
    name: "date"  
    tty: true  
  - command:  
      - "cat"  
    image: "192.168.10.15/kubernetes/kubectl:apline"  
    imagePullPolicy: "IfNotPresent"  
    name: "kubectl"  
    tty: true  
  restartPolicy: Never  
'''  
    }  
  }  
  environment {  
    MY_KUBECONFIG = credentials('kubernetes-cluster')  
  }  
  stages {  
    stage('Data') {  
      steps {  
        container(name: 'date') {  
          sh """  
            date  
          """  
        }  
      }  
    }  
    stage('echo') {  
      steps {  
        container(name: 'date') {  
          sh """  
            echo 'k8s is pod'  
          """  
        }  
      }  
    }  
    stage('kubectl') {  
      steps {  
        container(name: 'kubectl') {  
          sh """  
            kubectl get pod -A  --kubeconfig $MY_KUBECONFIG  
          """  
        }  
      }  
    }  
  }  
}

docker 的示例

pipeline {  
  agent none  
  stages {  
    stage('Example Build') {  
      agent { docker 'maven:3-alpine' }  
      steps {  
        echo 'Hello, Maven'  
        sh 'mvn --version'  
      }  
    }  
    stage('Example Test') {  
      agent { docker 'openjdk:8-jre' }  
      steps {  
        echo 'Hello, JDK'  
        sh 'java -version'  
      }  
    }  
  }  
}

2. 超时(Timeout)

这个 timeout 步骤允许限制等待某个行为发生时脚本所花费的时间。其语法相当简单。示例如下:

timeout(time:60,unit:'SECONDS'){
    //该代码块中的过程被设置为超时
}

默认的时间单位是 min。如果发生超时,该步骤就会抛出一个异常。如果异常没有被处理,将导致整个流水线过程被中止。

通常推荐的做法是,在使用timeout对任何造成流水线暂停的步骤(如一个 input 步骤)进行封装,这样做的结果是,即使出现差错导致在限定的时间内没有得到期望的输入,流水线也会继续执行。

示例如下:

node{
    def response
    stage('input'){
        timeout(time:10,unit:'SECONDS'){
            response = input message :'Please Input User'
            parameters:[string(defaultValue:'mikezhou',description:'Enter UserId:',name:'userid')]
        }
        echo "Username = " + response
    }
}

在这种情况下,Jenkins 将会给用户 10s 做出反应,如果时间到了,Jenkins 会抛出一个异常来中止流水线。

如果实际在设计流水线时,当超时发生时,并不想中止流水线向下执行,可以引入try...catch代码块来封装 timeout。

如下代码块所示:

node{
    def response
    stage('input'){
      try {
        timeout(time:10,unit:'SECONDS'){
            response = input message :'Please Input User'
            parameters:[string(defaultValue:'mikezhou',description:'Enter UserId:',name:'userid')]
         }
       }
       catch(err){
            response = 'user1'
      }
    }
}

需要注意的是,在处理异常的时候,可以在捕获异常处设置为期望的默认值。

3. 重试(retry)

这个 retry 闭包将代码封底装为一个步骤,当代码中有异常发生时,该步骤可以重试 n 次。其语法如下:

retry(n){
  //代码过程
}

如果达到重试的限制并且发生了一个异常,那么整个过程将会被中止(除非异常被处理,如使用try...catch代码块)

retry(2){
    try {
       def result=build job: "test_job"
       echo result
      }
    catch(err){
        if(!err.getMessage().contains("UNSTABLE"))
        throw err
    }
}

4. 等待直到(waitUntil)

引入waitUntil步骤,会导致整个过程一直等待某件事发生,通常这里的 “某件事” 指的是可以返回 true 的闭包。

如果代码过程永不返回 true 的话,这个步骤将会无期限地等待下去而不会结束。所以一般常见的做法,会结合 timeout 步骤来封装 waitUntil 步骤。

例如,使用 waitUntil 代码块来等待一个标记文件出现:

timeout(time:15,unit:'SECONDS'){
    waitUntil{
        def ret = sh returnStatus:true,script:'test -e /home/jenkins2/marker.txt'
        return (ret==0)
    }
}

再举一个例子,假如我们要等待一个 Docker 容器运行起来,以便我们可以在流水线中通过 REST API 调用获取一些数据。在这种情况下,如果这个 URL 还不可用,就会得到一个异常。为了保证异常被抛出的时候进程不会立即退出,我们可以使用try...catch代码块来捕获异常并且返回 false。

timeout(time:150,unit:'SECONDS'){
    waitUntil{
        try{
            sh "docker exec ${containerid} curl --silent http://127.0.0.1:8080/api/v1/registry >/test/output/url.txt"
            return true
        }
        catch(err)
            return false
    }
}

Stash 暂存:实现跨节点文件共享

在 Jenkins 的 DSL 中,stashunstash函数允许在流水线的节点间和阶段间保存或获取文件。

基本用法格式:

stash name:"<name>" [includes:"<pattern>" excludes:"<pattern>"]
unstash "<name>"

我们通过名称或模式来指定一个被包括或被排除的文件的集合。给这些文件的暂存处命名,以便后面通过这个名称使用这些文件。

提到 stash,很多读者可能会把Jenkins stashGit stash功能弄混,需要说明一下,Jenkins stashGit stash功能是不同的。Git stash函数是为了暂存一个工作目录的内容,缓存那些还没有提交到本地代码仓库的代码。而Jenkins stash函数是为了暂存文件,以便在节点间共享。

例如,master 节点和 node 节点,实现跨主机共享文件:

pipeline{
    agent none
    stages{
        stage('stash'){
            agent { label "master" }
            steps{
                writeFile file: "test.txt", text: "$BUILD_NUMBER"
                stash name: "test", includes: "test.txt"
            }
        }
        stage('unstash'){
            agent { label "node" }
            steps{
                script{
                    unstash("test")
                    def content = readFile("test.txt")
                    echo "${content}"
                }
            }
        }
    }
}

1.Environment

Environment 主要用于在流水线中配置的一些环境变量,根据配置的位置决定环境变量的作用域。可以定义在 pipeline 中作为全局变量,也可以配置在 stage 中作为该 stage 的环境变量。该指令支持一个特殊的方法 credentials(),该方法可用于在 Jenkins 环境中通过标识符访问预定义的凭证。对于类型为 Secret Text 的凭证,credentials()可以将该 Secret 中的文本内容赋值给环境变量。对于类型为标准的账号密码型的凭证,指定的环境变量为 username 和 password,并且也会定义两个额外的环境变量,分别为MYVARNAME_USRMYVARNAME_PSW

基本变量使用

//示例  
pipeline {  
  agent any  
  environment {   //全局变量,会在所有stage中生效  
    NAME= 'zhangzhuo'  
  }  
  stages {  
    stage('env1') {  
      environment { //定义在stage中的变量只会在当前stage生效,其他的stage不会生效  
        HARBOR = 'https://192.168.10.15'  
      }  
      steps {  
        sh "env"  
      }  
    }  
    stage('env2') {  
      steps {  
        sh "env"  
      }  
    }  
  }  
}

使用变量引用 secret 的凭证

//这里使用k8s的kubeconfig文件示例
pipeline {
  agent any
  environment {
    KUBECONFIG = credentials('kubernetes-cluster')
  }
  stages {
    stage('env') {
      steps {
        sh "env"  //默认情况下输出的变量内容会被加密
      }
    }
  }
}```

使用变量引用类型为标准的账号密码型的凭证

这里使用 HARBOR 变量进行演示,默认情况下账号密码型的凭证会自动创建 3 个变量

  • HARBOR_USR:会把凭证中 username 值赋值给这个变量

  • HARBOR_PSW:会把凭证中 password 值赋值给这个变量

  • HARBOR:默认情况下赋值的值为usernamme:password

//这里使用k8s的kubeconfig文件示例  
pipeline {  
  agent any  
  environment {  
    HARBOR = credentials('harbor-account')  
  }  
  stages {  
    stage('env') {  
      steps {  
        sh "env"  
      }  
    }  
  }  
}

2.Options

Jenkins 流水线支持很多内置指令,比如 retry 可以对失败的步骤进行重复执行 n 次,可以根据不同的指令实现不同的效果。比较常用的指令如下:

  • buildDiscarder :保留多少个流水线的构建记录

  • disableConcurrentBuilds:禁止流水线并行执行,防止并行流水线同时访问共享资源导致流水线失败。

  • disableResume :如果控制器重启,禁止流水线自动恢复。

  • newContainerPerStage:agent 为 docker 或 dockerfile 时,每个阶段将在同一个节点的新容器中运行,而不是所有的阶段都在同一个容器中运行。

  • quietPeriod:流水线静默期,也就是触发流水线后等待一会在执行。

  • retry:流水线失败后重试次数。

  • timeout:设置流水线的超时时间,超过流水线时间,job 会自动终止。如果不加 unit 参数默认为 1 分。

  • timestamps:为控制台输出时间戳。

定义在 pipeline 中

pipeline {  
  agent any  
  options {  
    timeout(time: 1, unit: 'HOURS')  //超时时间1小时,如果不加unit参数默认为1分  
    timestamps()                     //所有输出每行都会打印时间戳  
    buildDiscarder(logRotator(numToKeepStr: '3')) //保留三个历史构建版本  
    quietPeriod(10)  //注意手动触发的构建不生效  
    retry(3)    //流水线失败后重试次数  
  }  
  stages {  
    stage('env1') {  
      steps {  
        sh "env"  
        sleep 2  
      }  
    }  
    stage('env2') {  
      steps {  
        sh "env"  
      }  
    }  
  }  
}

定义在 stage 中

Option 除了写在 Pipeline 顶层,还可以写在 stage 中,但是写在 stage 中的 option 仅支持 retry、 timeout、timestamps,或者是和 stage 相关的声明式选项,比如 skipDefaultCheckout。处于 stage 级别的 options 写法如下

pipeline {  
  agent any  
  stages {  
    stage('env1') {  
      options {   //定义在这里这对这个stage生效  
        timeout(time: 2, unit: 'SECONDS') //超时时间2秒  
        timestamps()                     //所有输出每行都会打印时间戳  
        retry(3)    //流水线失败后重试次数  
      }  
      steps {  
        sh "env && sleep 2"  
      }  
    }  
    stage('env2') {  
      steps {  
        sh "env"  
      }  
    }  
  }  
}

3.Parameters

Parameters 提供了一个用户在触发流水线时应该提供的参数列表,这些用户指定参数的值可以通过 params 对象提供给流水线的 step(步骤)。只能定义在 pipeline 顶层。

目前支持的参数类型如下

  • string:字符串类型的参数。

  • text:文本型参数,一般用于定义多行文本内容的变量。

  • booleanParam:布尔型参数。

  • choice:选择型参数,一般用于给定几个可选的值,然后选择其中一个进行赋值。

  • password:密码型变量,一般用于定义敏感型变量,在 Jenkins 控制台会输出为*。

插件 Parameters

  • imageTag:镜像 tag,需要安装 Image Tag Parameter 插件后使用

  • gitParameter:获取 git 仓库分支,需要 Git Parameter 插件后使用

示例

pipeline {  
  agent any  
  parameters {  
    string(name: 'DEPLOY_ENV', defaultValue:  'staging', description: '1')   //执行构建时需要手动配置字符串类型参数,之后赋值给变量  
    text(name:  'DEPLOY_TEXT', defaultValue: 'One\nTwo\nThree\n', description: '2')  //执行构建时需要提供文本参数,之后赋值给变量  
    booleanParam(name: 'DEBUG_BUILD',  defaultValue: true, description: '3')   //布尔型参数  
    choice(name: 'CHOICES', choices: ['one', 'two', 'three'], description: '4')  //选择形式列表参数  
    password(name: 'PASSWORD', defaultValue: 'SECRET', description: 'A  secret password')  //密码类型参数,会进行加密  
    imageTag(name: 'DOCKER_IMAGE', description: '', image: 'kubernetes/kubectl', filter: '.*', defaultTag: '', registry: 'https://192.168.10.15', credentialId: 'harbor-account', tagOrder: 'NATURAL')   //获取镜像名称与tag  
    gitParameter(branch: '', branchFilter: 'origin/(.*)', defaultValue: '', description: 'Branch for build and deploy', name: 'BRANCH', quickFilterEnabled: false, selectedValue: 'NONE', sortMode: 'NONE',  tagFilter: '*', type: 'PT_BRANCH')  
  }  //获取git仓库分支列表,必须有git引用  
  stages {  
    stage('env1') {  
      steps {  
        sh "env"  
      }  
    }  
    stage('git') {  
      steps {  
        git branch: "$BRANCH", credentialsId: 'gitlab-key', url: 'git@192.168.10.14:root/env.git'   //使用gitParameter,必须有这个  
      }  
    }  
  }  
}

4.Triggers

在 Pipeline 中可以用 triggers 实现自动触发流水线执行任务,可以通过 Webhook、Cron、 pollSCM 和 upstream 等方式触发流水线。

Cron

定时构建假如某个流水线构建的时间比较长,或者某个流水线需要定期在某个时间段执行构建,可以 使用 cron 配置触发器,比如周一到周五每隔四个小时执行一次

注意:H 的意思不是 HOURS 的意思,而是 Hash 的缩写。主要为了解决多个流水线在同一时间同时运行带来的系统负载压力。

pipeline {  
  agent any  
  triggers {  
    cron('H */4 * * 1-5')   //周一到周五每隔四个小时执行一次  
    cron('H/12 * * * *')   //每隔12分钟执行一次  
    cron('H * * * *')   //每隔1小时执行一次  
  }  
  stages {  
    stage('Example') {  
      steps {  
        echo 'Hello World'  
      }  
    }  
  }  
}

Upstream

Upstream 可以根据上游 job 的执行结果决定是否触发该流水线。比如当 job1 或 job2 执行成功时触发该流水线

Docker+K8s+Jenkins 主流技术全解视频资料【干货免费分享】

目前支持的状态有 SUCCESS、UNSTABLE、FAILURE、NOT_BUILT、ABORTED 等。

pipeline {  
  agent any  
  triggers {  
    upstream(upstreamProjects: 'env', threshold: hudson.model.Result.SUCCESS)  //当env构建成功时构建这个流水线  
  }  
  stages {  
    stage('Example') {  
      steps {  
        echo 'Hello World'  
      }  
    }  
  }  
}

5.Input

Input 字段可以实现在流水线中进行交互式操作,比如选择要部署的环境、是否继续执行某个阶段等。

配置 Input 支持以下选项

  • message:必选,需要用户进行 input 的提示信息,比如:“是否发布到生产环境?”;

  • id:可选,input 的标识符,默认为 stage 的名称;

  • ok:可选,确认按钮的显示信息,比如:“确定”、“允许”;

  • submitter:可选,允许提交 input 操作的用户或组的名称,如果为空,任何登录用户均可提交 input;

  • parameters:提供一个参数列表供 input 使用。

假如需要配置一个提示消息为“还继续么”、确认按钮为“继续”、提供一个 PERSON 的变量的参数,并且只能由登录用户为 alice 和 bob 提交的 input 流水线

pipeline {  
  agent any  
  stages {  
    stage('Example') {  
      input {  
        message "还继续么?"  
        ok "继续"  
        submitter "alice,bob"  
        parameters {  
          string(name: 'PERSON', defaultValue: 'Mr Jenkins', description: 'Who should I say hello to?')  
        }  
      }  
      steps {  
        echo "Hello, ${PERSON}, nice to meet you."  
      }  
    }  
  }  
}

6.when

When 指令允许流水线根据给定的条件决定是否应该执行该 stage,when 指令必须包含至少 一个条件。如果 when 包含多个条件,所有的子条件必须都返回 True,stage 才能执行。

When 也可以结合 not、allOf、anyOf 语法达到更灵活的条件匹配。

目前比较常用的内置条件如下

  • branch:当正在构建的分支与给定的分支匹配时,执行这个 stage。注意,branch 只适用于多分支流水线

  • changelog:匹配提交的 changeLog 决定是否构建,例如:when { changelog '.*^\\[DEPENDENCY\\] .+$' }

  • environment:当指定的环境变量和给定的变量匹配时,执行这个 stage,例如:when { environment name: 'DEPLOY_TO', value: 'production' }

  • equals:当期望值和实际值相同时,执行这个 stage,例如:when { equals expected: 2, actual: currentBuild.number };

  • expression:当指定的 Groovy 表达式评估为 True,执行这个 stage,例如:when { expression { return params.DEBUG_BUILD } };

  • tag:如果 TAG_NAME 的值和给定的条件匹配,执行这个 stage,例如:when { tag "release-" };

  • not:当嵌套条件出现错误时,执行这个 stage,必须包含一个条件,例如:when { not { branch 'master' } };

  • allOf:当所有的嵌套条件都正确时,执行这个 stage,必须包含至少一个条件,例如:when { allOf { branch 'master'; environment name: 'DEPLOY_TO', value: 'production' } };

  • anyOf:当至少有一个嵌套条件为 True 时,执行这个 stage,例如:when { anyOf { branch 'master'; branch 'staging' } }。

示例:当分支为 main 时执行 Example Deploy 步骤

pipeline {  
  agent any  
  stages {  
    stage('Example Build') {  
      steps {  
        echo 'Hello World'  
      }  
    }  
    stage('Example Deploy') {  
      when {  
        branch 'main' //多分支流水线,分支为才会执行。  
      }  
      steps {  
        echo 'Deploying'  
      }  
    }  
  }  
}

也可以同时配置多个条件,比如分支是 production,而且 DEPLOY_TO 变量的值为 main 时,才执行 Example Deploy

pipeline {  
  agent any  
  environment {  
    DEPLOY_TO = "main"  
  }  
  stages {  
    stage('Example Deploy') {  
      when {  
        branch 'main'  
        environment name: 'DEPLOY_TO', value: 'main'  
      }  
      steps {  
        echo 'Deploying'  
      }  
    }  
  }  
}

也可以使用 anyOf 进行匹配其中一个条件即可,比如分支为 main 或 DEPLOY_TO 为 main 或 master 时执行 Deploy

pipeline {  
  agent any  
  stages {  
    stage('Example Deploy') {  
      when {  
        anyOf {  
          branch 'main'  
          environment name: 'DEPLOY_TO', value: 'main'  
          environment name: 'DEPLOY_TO', value: 'master'  
        }  
      }  
      steps {  
        echo 'Deploying'  
      }  
    }  
  }  
}

也可以使用 expression 进行正则匹配,比如当 BRANCH_NAME 为 main 或 master,并且 DEPLOY_TO 为 master 或 main 时才会执行 Example Deploy

pipeline {  
  agent any  
  stages {  
    stage('Example Deploy') {  
      when {  
        expression { BRANCH_NAME ==~ /(main|master)/ }  
        anyOf {  
          environment name: 'DEPLOY_TO', value: 'main'  
          environment name: 'DEPLOY_TO', value: 'master'  
        }  
      }  
      steps {  
        echo 'Deploying'  
      }  
    }  
  }  
}

默认情况下,如果定义了某个 stage 的 agent,在进入该 stage 的 agent 后,该 stage 的 when 条件才会被评估,但是可以通过一些选项更改此选项。比如在进入 stage 的 agent 前评估 when, 可以使用 beforeAgent,当 when 为 true 时才进行该 stage

目前支持的前置条件如下

  • beforeAgent:如果 beforeAgent 为 true,则会先评估 when 条件。在 when 条件为 true 时,才会进入该 stage

  • beforeInput:如果 beforeInput 为 true,则会先评估 when 条件。在 when 条件为 true 时,才会进入到 input 阶段;

  • beforeOptions:如果 beforeInput 为 true,则会先评估 when 条件。在 when 条件为 true 时,才会进入到 options 阶段;

beforeOptions 优先级大于 beforeInput 大于 beforeAgent

示例

pipeline {  
  agent none  
  stages {  
    stage('Example Build') {  
      steps {  
        echo 'Hello World'  
      }  
    }  
    stage('Example Deploy') {  
      when {  
        beforeAgent true  
        branch 'main'  
      }  
      steps {  
        echo 'Deploying'  
      }  
    }  
  }  
}

Parallel

在声明式流水线中可以使用 Parallel 字段,即可很方便的实现并发构建,比如对分支 A、B、 C 进行并行处理

pipeline {  
  agent any  
  stages {  
    stage('Non-Parallel Stage') {  
      steps {  
        echo 'This stage will be executed first.'  
      }  
    }  
    stage('Parallel Stage') {  
      failFast true         //表示其中只要有一个分支构建执行失败,就直接推出不等待其他分支构建  
      parallel {  
        stage('Branch A') {  
          steps {  
            echo "On Branch A"  
          }  
        }  
        stage('Branch B') {  
          steps {  
            echo "On Branch B"  
          }  
        }  
        stage('Branch C') {  
          stages {  
            stage('Nested 1') {  
              steps {  
                echo "In stage Nested 1 within Branch C"  
              }  
            }  
            stage('Nested 2') {  
              steps {  
               echo "In stage Nested 2 within Branch C"  
              }  
            }  
          }  
        }  
      }  
    }  
  }  
}

steps

Steps 部分在给定的 stage 指令中执行的一个或多个步骤,比如在 steps 定义执行一条 shell 命令

//Jenkinsfile (Declarative Pipeline)  
pipeline {  
  agent any  
  stages {  
    stage('Example') {  
      steps {  
        echo 'Hello World'  
      }  
    }  
  }  
}

或者是使用 sh 字段执行多条指令

//Jenkinsfile (Declarative Pipeline)  
pipeline {  
  agent any  
  stages {  
    stage('Example') {  
      steps {  
        sh """  
           echo 'Hello World1'  
           echo 'Hello World2'  
        """  
      }  
    }  
  }  
}

post

  1. Post 一般用于流水线结束后的进一步处理,比如错误通知等。Post 可以针对流水线不同的结果做出不同的处理,就像开发程序的错误处理,比如 Python 语言的 try catch。

  2. Post 可以定义在 Pipeline 或 stage 中,目前支持以下条件

  3. always:无论 Pipeline 或 stage 的完成状态如何,都允许运行该 post 中定义的指令;

  4. changed:只有当前 Pipeline 或 stage 的完成状态与它之前的运行不同时,才允许在该 post 部分运行该步骤;

  5. fixed:当本次 Pipeline 或 stage 成功,且上一次构建是失败或不稳定时,允许运行该 post 中定义的指令;

  6. regression:当本次 Pipeline 或 stage 的状态为失败、不稳定或终止,且上一次构建的 状态为成功时,允许运行该 post 中定义的指令;

  7. failure:只有当前 Pipeline 或 stage 的完成状态为失败(failure),才允许在 post 部分运行该步骤,通常这时在 Web 界面中显示为红色

  8. success:当前状态为成功(success),执行 post 步骤,通常在 Web 界面中显示为蓝色 或绿色

  9. unstable:当前状态为不稳定(unstable),执行 post 步骤,通常由于测试失败或代码 违规等造成,在 Web 界面中显示为黄色

  10. aborted:当前状态为终止(aborted),执行该 post 步骤,通常由于流水线被手动终止触发,这时在 Web 界面中显示为灰色;

  11. unsuccessful:当前状态不是 success 时,执行该 post 步骤;

  12. cleanup:无论 pipeline 或 stage 的完成状态如何,都允许运行该 post 中定义的指令。和 always 的区别在于,cleanup 会在其它执行之后执行。

示例

一般情况下 post 部分放在流水线的底部,比如本实例,无论 stage 的完成状态如何,都会输出一条 I will always say Hello again!信息

//Jenkinsfile (Declarative Pipeline)  
pipeline {  
  agent any  
  stages {  
    stage('Example1') {  
      steps {  
        echo 'Hello World1'  
      }  
    }  
    stage('Example2') {  
      steps {  
        echo 'Hello World2'  
      }  
    }  
  }  
  post {  
    always {  
      echo 'I will always say Hello again!'  
    }  
  }  
}

也可以将 post 写在 stage,下面示例表示 Example1 执行失败执行 post。

//Jenkinsfile (Declarative Pipeline)  
pipeline {  
  agent any  
  stages {  
    stage('Example1') {  
      steps {  
        sh 'ip a'  
      }  
      post {  
        failure {  
          echo 'I will always say Hello again!'  
        }  
      }  
    }  
  }  
}

环境变量

1.静态变量

Jenkins 有许多内置变量可以直接在 Jenkinsfile 中使用,可以通过 JENKINS_URL/pipeline/syntax/globals#env 获取完整列表。目前比较常用的环境变量如下

  • BUILD_ID:当前构建的 ID,与 Jenkins 版本 1.597+中的 BUILD_NUMBER 完全相同

  • BUILD_NUMBER:当前构建的 ID,和 BUILD_ID 一致

  • BUILD_TAG:用来标识构建的版本号,格式为:jenkins-${JOB_NAME}-${BUILD_NUMBER}, 可以对产物进行命名,比如生产的 jar 包名字、镜像的 TAG 等;

  • BUILD_URL:本次构建的完整 URL,比如:

    http://buildserver/jenkins/job/MyJobName/17/%EF%BC%9B

  • JOB_NAME:本次构建的项目名称

  • NODE_NAME:当前构建节点的名称;

  • JENKINS_URL:Jenkins 完整的 URL,需要在 SystemConfiguration 设置;

  • WORKSPACE:执行构建的工作目录。

示例如果一个流水线名称为print_env,第 2 次构建,各个变量的值。

BUILD_ID:2BUILD_NUMBER:2BUILD_TAG:jenkins-print_env-2BUILD_URL:http://192.168.10.16:8080/job/print_env/2/JOB_NAME:print_envNODE_NAME:built-inJENKINS_URL:http://192.168.10.16:8080/WORKSPACE:/bitnami/jenkins/home/workspace/print_env

上述变量会保存在一个 Map 中,可以使用 env.BUILD_ID 或 env.JENKINS_URL 引用某个内置变量

pipeline {  
  agent any  
  stages {  
    stage('print env') {  
      parallel {  
        stage('BUILD_ID') {  
          steps {  
            echo "$env.BUILD_ID"  
          }  
        }  
        stage('BUILD_NUMBER') {  
          steps {  
            echo "$env.BUILD_NUMBER"  
          }  
        }  
        stage('BUILD_TAG') {  
          steps {  
            echo "$env.BUILD_TAG"  
          }  
        }  
      }  
    }  
  }  
}

2.动态变量

动态变量是根据某个指令的结果进行动态赋值,变量的值根据指令的执行结果而不同。如下所示

  • returnStdout:将命令的执行结果赋值给变量,比如下述的命令返回的是 clang,此时 CC 的值为“clang”。

  • returnStatus:将命令的执行状态赋值给变量,比如下述命令的执行状态为 1,此时 EXIT_STATUS 的值为 1。

//Jenkinsfile (Declarative Pipeline)  
pipeline {  
  agent any  
  environment {  
    // 使用 returnStdout  
    CC = """${sh(  
         returnStdout: true,  
         script: 'echo -n "clang"'   //如果使用shell命令的echo赋值变量最好加-n取消换行  
         )}"""  
    // 使用 returnStatus  
    EXIT_STATUS = """${sh(  
         returnStatus: true,  
         script: 'exit 1'  
         )}"""  
  }  
  stages {  
    stage('Example') {  
      environment {  
        DEBUG_FLAGS = '-g'  
      }  
      steps {  
        sh 'printenv'  
      }  
    }  
  }  
}

3.2 凭证管理

Jenkins 的声明式流水线语法有一个 credentials()函数,它支持 secret text(加密文本)、username 和 password(用户名和密码)以及 secret file(加密文件)等。接下来看一下一些常用的凭证处理方法。

1.加密文本

本实例演示将两个 Secret 文本凭证分配给单独的环境变量来访问 Amazon Web 服务,需要 提前创建这两个文件的 credentials(实践的章节会有演示),Jenkinsfile 文件的内容如下

//Jenkinsfile (Declarative Pipeline)  
pipeline {  
  agent any  
  environment {  
    AWS_ACCESS_KEY_ID = credentials('txt1')  
    AWS_SECRET_ACCESS_KEY = credentials('txt2')  
  }  
  stages {  
    stage('Example stage 1') {  
      steps {  
        echo "$AWS_ACCESS_KEY_ID"  
      }  
    }  
    stage('Example stage 2') {  
      steps {  
        echo "$AWS_SECRET_ACCESS_KEY"  
      }  
    }  
  }  
}

2.用户名密码

本示例用来演示 credentials 账号密码的使用,比如使用一个公用账户访问 Bitbucket、GitLab、 Harbor 等。假设已经配置完成了用户名密码形式的 credentials,凭证 ID 为 harbor-account

//Jenkinsfile (Declarative Pipeline)  
pipeline {  
  agent any  
  environment {  
    BITBUCKET_COMMON_CREDS = credentials('harbor-account')  
  }  
  stages {  
    stage('printenv') {  
      steps {  
        sh "env"  
      }  
    }  
}

上述的配置会自动生成 3 个环境变量

  • BITBUCKET_COMMON_CREDS:包含一个以冒号分隔的用户名和密码,格式为 username:password

  • BITBUCKET_COMMON_CREDS_USR:仅包含用户名的附加变量

  • BITBUCKET_COMMON_CREDS_PSW:仅包含密码的附加变量。

3.加密文件

需要加密保存的文件,也可以使用 credential,比如链接到 Kubernetes 集群的 kubeconfig 文件等。

假如已经配置好了一个 kubeconfig 文件,此时可以在 Pipeline 中引用该文件

//Jenkinsfile (Declarative Pipeline)  
pipeline {  
  agent {  
    kubernetes {  
      cloud 'kubernetes'  
      slaveConnectTimeout 1200  
      workspaceVolume emptyDirWorkspaceVolume()  
      yaml '''  
kind: Pod  
metadata:  
  name: jenkins-agent  
spec:  
  containers:  
  - args: [\'$(JENKINS_SECRET)\', \'$(JENKINS_NAME)\']  
    image: '192.168.10.15/kubernetes/jnlp:alpine'  
    name: jnlp  
    imagePullPolicy: IfNotPresent  
  - command:  
      - "cat"  
    image: "192.168.10.15/kubernetes/kubectl:apline"  
    imagePullPolicy: "IfNotPresent"  
    name: "kubectl"  
    tty: true  
  restartPolicy: Never  
'''  
    }  
  }  
  environment {  
    MY_KUBECONFIG = credentials('kubernetes-cluster')  
  }  
  stages {  
    stage('kubectl') {  
      steps {  
        container(name: 'kubectl') {  
          sh """  
            kubectl get pod -A  --kubeconfig $MY_KUBECONFIG  
          """  
        }  
      }  
    }  
  }  
}

0
  1. 支付宝打赏

    qrcode alipay
  2. 微信打赏

    qrcode weixin

评论区