是否可以从管道中的sh DSL命令捕获stdout


92

例如:

var output=sh "echo foo";
echo "output=$output";

我会得到:

output=0

因此,显然我得到的是退出代码,而不是标准输出。是否有可能将stdout捕获到管道变量中,这样我就可以得到: output=foo 作为结果?

Answers:


227

现在,该sh步骤通过提供参数来支持返回stdoutreturnStdout

// These should all be performed at the point where you've
// checked out your sources on the slave. A 'git' executable
// must be available.
// Most typical, if you're not cloning into a sub directory
gitCommit = sh(returnStdout: true, script: 'git rev-parse HEAD').trim()
// short SHA, possibly better for chat notifications, etc.
shortCommit = gitCommit.take(6)

请参阅此示例


10
请注意.trim()此答案的一部分,否则,您可能会在行尾得到换行符
Will Munn

2
附加--shortrev-parse即可直接获得简短的哈希值
莱昂

不知道是什么导致失败,但我也必须将输出转换为字符串,例如gitCommit = sh(returnStdout: true, script: 'git rev-parse HEAD').toString().trim()
Balkrishna

嗨,“。take(6)”代表什么?
Vano

1
@Vano引用Groovy方法take(),在这种情况下,它将获取前6个字符。docs.groovy-lang.org/docs/groovy-2.3.2/html/api/org/codehaus/…–
ahillman3

47

注意:链接的詹金斯问题已解决。

正如JENKINS-26133中提到的那样,不可能将shell输出作为变量。解决方法建议使用临时文件中的写命令。因此,您的示例将如下所示:

sh "echo foo > result";
def output=readFile('result').trim()
echo "output=$output";

21
对于新手,请参见下面的答案stackoverflow.com/a/38912813/345845,这是因为通过将新returnStdout参数传递给sh步骤,使此操作变得更加容易。
Baptiste Mathus

2
“不可能将shell输出作为变量输出”-不正确。这是一个hack,正确的答案是returnStdout。
格雷厄姆

4
这其实是一个很好的答案的唯一情况是,如果你需要双方stdoutexit statusshell命令。其他时候,使用returnStdout参数。
西蒙·佛斯伯格

4

试试这个:

def get_git_sha(git_dir='') {
    dir(git_dir) {
        return sh(returnStdout: true, script: 'git rev-parse HEAD').trim()
    }
}

node(BUILD_NODE) {
    ...
    repo_SHA = get_git_sha('src/FooBar.git')
    echo repo_SHA
    ...
}

经过测试:

  • 詹金斯版 2.19.1
  • 管道2.4

3

您也可以尝试使用此功能来捕获StdErr StdOut并返回代码。

def runShell(String command){
    def responseCode = sh returnStatus: true, script: "${command} &> tmp.txt" 
    def output =  readFile(file: "tmp.txt")

    if (responseCode != 0){
      println "[ERROR] ${output}"
      throw new Exception("${output}")
    }else{
      return "${output}"
    }
}

注意:

&>name means 1>name 2>name -- redirect stdout and stderr to the file name

1

一个简短的版本是:

echo sh(script: 'ls -al', returnStdout: true).result


0

我遇到了同样的问题,并尝试了几乎所有发现的问题,之后才知道我尝试了错误的方法。我在步骤块中尝试过它,但它必须在环境块中。

        stage('Release') {
                    environment {
                            my_var = sh(script: "/bin/bash ${assign_version} || ls ", , returnStdout: true).trim()
                                }
                    steps {                                 
                            println my_var
                            }
                }
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.