仅在主节点上签出和运行 SCM 管道

时间:2021-03-08 21:44:22

标签: jenkins groovy continuous-integration jenkins-pipeline

我编写了一个通用管道,它接受多个参数,以便将预定义的 GitHub 存储库中的版本部署到特定节点。我想在 GitHub 上的 Jenkinsfile 上托管此管道,因此我将作业配置为使用“来自 SCM 的管道脚本”。事实是 - 当我尝试构建作业时 - Jenkinsfile 在每个节点上都被检出。是否可以仅在主节点上签出和执行 Jenkinsfile 并按预期运行管道?

编辑:正如我之前所说,管道工作得很好,并且按照预期将作业设置为使用管道脚本。 问题是当我尝试将其更改为“来自 SCM 的管道脚本”时,Jenkinsfile 会在每个代理上被检出,这是一个问题,因为我没有在任何代理上安装 git除了大师。我希望 Jenkinsfile 仅在主代理上签出并按预期执行。仅供参考以下管道:

def agents = "$AGENTS".toString()
def agentLabel = "${ println 'Agents: ' + agents; return agents; }"

pipeline {
    agent none

    stages {
        stage('Prep') {
            steps {
                script {
                    if (agents == null || agents == "") {
                        println "Skipping build"
                        skipBuild = true
                    }
                                    
                    if (!skipBuild) {
                        println "Agents set for this build: " + agents
                    }
                }
            }
        }
    
        stage('Powershell deploy script checkout') {
            agent { label 'master' }
        
            when {
                expression {
                    !skipBuild
                }
            }
        
            steps {
                git url: 'https://github.com/owner/repo.git', credentialsId: 'git-credentials', branch: 'main'
                stash includes: 'deploy-script.ps1', name: 'deploy-script'
            }
        }
    
        stage('Deploy') {
            agent { label agentLabel }
        
            when {
                expression {
                    !skipBuild
                }
            }
        
            steps {
                unstash 'deploy-script'
            
                script {
                    println "Execute powershell deploy script on agents set for deploy"
                }
            }
        }
    }
}

4 个答案:

答案 0 :(得分:0)

我认为你要求的是不可能的。

现在:

您的 Jenkinsfile 在您的 jenkins 配置中,并被发送给您的每个代理。您的代理不需要 git

SCM 的管道脚本:

由于您使用 git,SCM = git。所以你是说:我的管道需要从 git 存储库中获取。您正在声明 Deploy 步骤在 agent { label agentLabel } 上运行,因此该步骤应该在除 master 之外的另一个代理上运行。 您如何想象代理可以获取 Jenkinsfile 的内容以知道要做什么,但不使用 git

詹金斯会发生什么?

  • 您的主代理被触发,需要构建
  • 主代理使用 git 检出 Jenkinsfile(因为它是一个来自 SCM 的管道脚本
  • jenkins 读取 Jenkinsfile 并查看必须完成的操作。
  • 对于准备阶段,我不太确定没有代理会发生什么,我想这会在主代理上运行。
  • Powershell 部署脚本检出 被标记为在主代理上运行,因此它在主代理上运行(注意 Jenkinsfile 将使用 git 再检出两次:
    • 在开始阶段之前,因为 jenkins 需要知道要执行什么
    • 再结帐一次,因为您指定了 git url: 'https://github.com/owner/repo.git'...
  • Deploy 阶段被标记为在 agentLabel 上运行,因此 jenkins 尝试在该代理上检查您的 Jenkinsfile(使用 git)...

答案 1 :(得分:0)

你可以使用Scripted Pipeline来做到这一点,它基本上应该是这样的

node('master') {
    checkout scm
    stash includes: 'deploy-script.ps1', name: 'deploy-script'
}

def stepsForParallel = [:]
env.AGENTS.split(' ').each { agent ->
    stepsForParallel["deploy ${agent}"] = { ->

    node(agent) {
        unstash 'deploy-script'
    }
}
parallel stepsForParallel

答案 2 :(得分:0)

我认为 skipDefaultCheckout 正是您要找的:

pipeline {
    options {
        skipDefaultCheckout true
    }

    stages {
        stage('Prep') {
                steps {
                    script {
                            ........................
                    }
                }       
        }
    }
}

查看文档:

skipDefaultCheckout 默认情况下,在代理指令中跳过从源代码管理中检出代码。

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

答案 3 :(得分:-1)

您可以在 here 部分找到有关 jenkins 代理的所有信息。 很快:您可以按名称或标签呼叫任何代理。

pipeline {
  agent {
    label 'master'
  }
}

如果它不适合你,那么你需要在主节点上设置任何标签并通过标签调用它

pipeline {
  agent {
    label 'master_label_here'
  }
}
相关问题