Jenkins实践指南-09-pipeline 扩展

5. pipeline 扩展

    如果在大量使用pipelin后,会发现Jenkins内置的功能并不能满足我们的需求,这时就需要pipeline 扩展。

5.1 pipeline中使用函数

    pipeline本质上就是一个Groovy脚本。因此,也可以在pipeline中定义函数,这样就可以使用Groovy的特性了。示例如下所示:

// 定义一个函数
def hello(name="Kevin"){
    return "Hello,${name}"
}

pipeline{
    agent any
    options{
        timestamps()
    }
    stages{
        stage("pipeline extend demo"){
            steps{
                // 调用函数
                echo "${hello('Surpass')}"
            }
        }
    }
}

    另外也可以将变量定义在environment中,如下所示:

// 定义一个函数
def hello(name="Kevin"){
    return "Hello,${name}"
}

pipeline{
    agent any
    options{
        timestamps()
    }
    environment{
        // 调用函数
        HELLO=hello("Surpass")
    }
    stages{
        stage("pipeline extend demo"){
            steps{
                echo "${env.HELLO}"
            }
        }
    }
}

    运行结果如下所示:

0501 pipeline中定义函数.png

5.2 使用共享库

    如果在pipeline中定义的函数具有通用性,那很可能另外一个pipeline也会使用,这个时候该怎么办呢?针对这一问题,Jenkins提供了共享库(Shared Library)功能,从而可以将重复代码定义一个独立的代码仓库中,其他的pipeline通过摘取加载使用,达到共享功能

5.2.1 目录结构

    共享库存储库的目录结构如下:

|--src
|    └──org
|        └──foo
|            └──Bar.groovy # org.foo.Bar 类
|--vars
|    └──foo.groovy  # 全局变量 foo
|    └──foo.txt     # 全局变更 foo 帮助文档
|--resources
|    └──org
|        └──foo
|            └──bar.json #

    以上各目录详细解释如下所示:

  • src: 该目录是一个标准的Java源码结构。目录中的类称这为库类,在执行pipeline时,将被添加到classpath路径中,文件格式为.groovy

在pipeline中使用时@Library("surpass-shared-library") _,其中-代表一次性静态加载src目录下所有代码到classpath中

  • vars:该目录用于存放pipeline可以直接使用的全局变量

    假设现在有一个文件名var/log.groovy,其文件内容如下所示:

// vars/logger.groovy

def info(message){
    println "message is ${message}"
}

    在pipeline中,我们可以这样进行调用

  logger.info "Hello,Surpass"

    其注意事项如下所示:

  • 1.每个groovy文件应该是一个Groovy标识符,遵循camelCased命名规范

  • 2.存在相应的使用帮助文档,可以HTML、Markdown等,但却需要以.txt做为扩展名

  • resources:该目录允许从外部库中使用该步骤libraryResource来加载相关联的非groovy文件。目前暂内部库不支持该功能

以上参考来自于:https://www.jenkins.io/doc/book/pipeline/shared-libraries/

5.2.2 创建共享库

    按共享库目录分别创建文件,如下所示:

// src/org/surpass/SharedLibrarySample.groovy

package org.surpass

class SharedLibrarySample{
   String name
   String location

   SharedLibrarySample(String name,String location){
        this.name=name
        this.location=location
   }

   def greet(){
      return "Hello ${this.name},welcome to ${this.location}"
   }
}

// var/logger.groovy
def info(message){
    println "[INFO] ${message}"
}

def error(message){
    println "[ERROR] ${message}"
}

def warning(message){
    println "[WARNNING] ${message}"
}

    将以上代码推送至代码仓库中,在Jenkins中按以下进行配置:

Manage Jenkins -> Configure System -> Global Pipeline Libraries
0502 共享库配置.png

    主要配置说明如下所示:

  • Name:共享库的唯一标识,在pipeline中会使用到
  • Default version:默认版本,可以是branchname、tag等
  • Load implicitly:隐式加载。若勾选,将自动加载全局共享库,在pipeline中不需要显式引用,就可以直接使用。
  • Allow default version to be overridden:若勾选,则表示允许默认被 pipeline中的配置覆盖
  • Include @Library changes in job recent changes:若勾选,则共享库的最近变更信息将会打印在构建日志中
  • Cache fetched versions on controller for quick retrieval:若勾选,则表示允许从当前节点缓存中加载共享库
  • Retrieval method:获取共享库代码的方法

5.2.3 使用共享库

    使用共享库全局变量的示例如下所示:

@Library("surpass-shared-library") _
pipeline{
    agent any
    options{
        timestamps()
    }
    stages{
        stage("use shared library demo"){
            steps{
                script{
                    logger.error "use shared library demo"
                }
            }
        }
    }
}

    运行结果如下所示:

0503 共享库运行全局变量示例.png

    使用共享库中类库的示例如下所示:

@Library("surpass-shared-library@master") _
pipeline{
    agent any
    options{
        timestamps()
    }
    stages{
        stage("use shared library demo"){
            steps{
                script{
                    def sharedLibraySample=new org.surpass.SharedLibrarySample("Surpass","Shanghai")
                    def ret=sharedLibraySample.greet()
                    println "result is ${ret}"
                }
            }
        }
    }
}

    运行结果如下所示:

0504 共享库运行类库.png

    若要使用共享库,需要在pipeline的最上方,使用@Library指定共享库,以上示例中surpass-shared-library为Jenkins中配置的共享库唯一标识符。在引入共享库后,就可以在pipeline中直接使用vars目录中的logger中的函数了。

    通过以上步骤,我们可以总结出,使用共享库的基本步骤为:

  • 按照共享给定的源码目录结构,创建实现自身逻辑的代码
  • 将共享库代码上传至代码仓库中用以托管
  • 在Jenkins中配置共享库信息,用于获取共享库代码
  • 在pipeline中使用@Library引用共享库

    使用@Library注解可以指定共享库在代码仓库中的版本,示例如下所求:

@Library("surpass-shared-library@<version>") _

    其中version可以是以下任意一种

  • branch:分支,如@Library("surpass-shared-library@master") _
  • tag:标签,如@Library("surpass-shared-library@release-v3.5.0") _
  • commit id: 提交的commit id,如@Library("surpass-shared-library@908d5d889bfaa0b5221b66c260a7c004114f0750") _

    在一个共享库无法满足要求进,Jenkins也支持添加多个共享库,使用方法如下所求:

@Library(["surpass-shared-library-a","surpass-shared-library-2","surpass-shared-library-3"]) _

若多个共享库存在相同的函数时,则优先使用先定义的共享库函数

5.2.4 使用共享库实现pipeline 模板

    在pipeline 1.2 版本后,可以在共享库定义pipeline。通过该特性,我们可以定义pipeline的模板。示例如下所求:

// vars/pipelineTemplate.groovy

def call(String opType){
   if (opType == "build"){
      pipeline{
        agent any
        options{
          timestamps()
        }
        parameters{
           string(name:"build",
                  defaultValue:"generate pipelin template by build",
                  description:"build demo"
                )
        }
        stages{
          stage("build demo"){
            steps{
              echo "generatePipelineTemplate demo to build"
              echo "params is ${params.build}"
            }
          }
        }
      }
   } else if (opType == "test"){
      pipeline{
        agent any
        options{
          timestamps()
        }
        parameters{
           string(name:"test",
                  defaultValue:"generate pipelin template by test",
                  description:"test demo"
                 )
        }
        stages{
          stage("test demo"){
            steps{
              echo "generatePipelineTemplate demo to test"
              echo "params is ${params.test}"
            }
          }
        }
      }
    } else if (opType == "deploy"){
      pipeline{
        agent any
        options{
          timestamps()
        }
        parameters{
           string(name:"deploy",
                  defaultValue:"generate pipelin template by deploy",
                  description:"deploy demo"
                 )
        }
        stages{
          stage("deploy demo"){
            steps{
              echo "generatePipelineTemplate demo to deploy"
              echo "params is ${params.deploy}"
            }
          }
        }
      }
    }
}

    在使用Jenkins时,Jenkinsfile仅有两行,是不是特别简洁,如下所示:

@Library("surpass-shared-library@master") _
pipelineTemplate("test")

    最终运行日志如下所示:

0505 使用共享库生成pipeline 模板.png

    生成的pipeline项目示意图如下所示:

0506 使用共享库生成pipeline项目示意图.png

当项目结构都是非常标准化的时候 ,利用pipeline模板可以批量生成统一的模板,同时也降低了维护pipeline的的成本。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270

推荐阅读更多精彩内容