在多个工作中重复使用jenkins管道的各个阶段

ade*_*ady 9 jenkins jenkins-pipeline

我的团队正在转向Jenkins 2,我正在使用管道插件,以便我们的构建可以存在于我们的存储库中.因为在我们公司中获得分配的存储库有很多开销,所以我们有一个存储库,其中包含许多子项目和子模块.

我想要的是为每个子模块单独构建和报告Junit/checkstyle/etc报告,以及每个子项目的最终"构建和部署"步骤.

我目前的计划是为每个子模块创建单独的作业,以便他们获得自己的junit/checkstyle/etc报告页面.然后有一个多作业项目来为子项目编排子模块构建.由于所有子项目都是简单的jar构建,我想将大量逻辑放在一个公共文件中,让我们在子项目的根目录中调用它为JenkinsfileForJars.回购结构是

  • 子项目
    • JenkinsfileForJars.groovy
    • 子moduleA
      • Jenkinsfile
    • 子moduleB
      • Jenkinsfile

我的Jenkinsfile包含

def submoduleName = "submoduleA"
def pipeline
node {

    pipeline = load("${env.WORKSPACE}/subproject/JenkinsfileForJars.groovy")

}
pipeline.build()
pipeline.results()
Run Code Online (Sandbox Code Playgroud)

我的JenkinsfileForJars包含

def build() {

    stage('Build') {
        // Run the maven build
        dir("subproject") {
            sh "./gradlew ${submoduleName}:build"
        }

    }
}
def results() {

    stage('Results') {
        dir("subproject/${submoduleName}") {
            junit 'build/test-results/TEST-*.xml'
            archive 'build/libs/*.jar'
            publishHTML([allowMissing: false, alwaysLinkToLastBuild: false, keepAll: false, reportDir: 'build/reports/cobertura/', reportFiles: 'frame-summary.html', reportName: 'Cobertura Report'])
            publishHTML([allowMissing: false, alwaysLinkToLastBuild: false, keepAll: false, reportDir: 'build/reports/findbugs/', reportFiles: 'main.html', reportName: 'Fidbugs Report'])
            publishHTML([allowMissing: false, alwaysLinkToLastBuild: false, keepAll: false, reportDir: 'build/reports/pmd/', reportFiles: 'main.html', reportName: 'PMD Report'])
            step([$class: 'CheckStylePublisher', pattern: 'build/reports/checkstyle/main.xml', unstableTotalAll: '200', usePreviousBuildAsReference: true])
        }
    }

}

return this;
Run Code Online (Sandbox Code Playgroud)

当我运行上面的Jenkinsfile时,我收到以下错误:

Running on master in /var/lib/jenkins/workspace/jobA
[Pipeline] {
[Pipeline] load
[Pipeline] { (/var/lib/jenkins/workspace/jobA/subproject/JenkinsfileForJars.groovy)
[Pipeline] }
[Pipeline] // load
[Pipeline] }
[Pipeline] // node
[Pipeline] End of Pipeline
java.lang.NullPointerException: Cannot invoke method build() on null object
Run Code Online (Sandbox Code Playgroud)

据我所知,我正在关注加载手动脚本的文档中显示的内容以及为加载的脚本提供示例.我不明白为什么我的脚本在load命令后为null.

如何让我的Jenkinsfile加载JenkinsfileForJars.groovy?

小智 4

该问题与 Blake Mitchell 在上面的评论中提到的 SCM 结帐有关。由于您是从子模块加载常规函数,因此如果您只想在主模块上保留裸露的存储库,则需要首先检出子模块,最好在构建代理/从属模块上检出。

def pipeline
node( 'myAgentLabel' ) {
    stage ( 'checkout SCM' ) {
        checkout([
            $class: 'GitSCM'
            ,branches: scm.branches
            ,extensions: scm.extensions 
                + [[ $class: 'SubmoduleOption', disableSubmodules: false, parentCredentials: true, recursiveSubmodules: true, reference: '', trackingSubmodules: false]]
            ,doGenerateSubmoduleConfigurations: false
            ,userRemoteConfigs: scm.userRemoteConfigs
        ])
        pipeline = load( "${env.WORKSPACE}/path/to/submodule/myGroovyFunctions.grooovy" )
    }
    pipeline.build()
}
Run Code Online (Sandbox Code Playgroud)

请注意,在结帐示例中,对 scm.* 属性的访问也需要由 Jenkins 中的管理员列入白名单(进程内脚本批准)