我在Jenkinsfile (Groovy)上有类似的东西,我想在一个变量中记录标准输出和退出码,以便以后使用这些信息。
sh "ls -l"
我该如何做到这一点呢,特别是当您似乎无法在Jenkinsfile中真正运行任何类型的groovy代码时?
我在Jenkinsfile (Groovy)上有类似的东西,我想在一个变量中记录标准输出和退出码,以便以后使用这些信息。
sh "ls -l"
我该如何做到这一点呢,特别是当您似乎无法在Jenkinsfile中真正运行任何类型的groovy代码时?
当前回答
如果你想获取stdout并知道命令是否成功,只需使用returnStdout并将其包装在异常处理程序中:
脚本化的管道
try {
// Fails with non-zero exit if dir1 does not exist
def dir1 = sh(script:'ls -la dir1', returnStdout:true).trim()
} catch (Exception ex) {
println("Unable to read dir1: ${ex}")
}
输出:
[Pipeline] sh
[Test-Pipeline] Running shell script
+ ls -la dir1
ls: cannot access dir1: No such file or directory
[Pipeline] echo
unable to read dir1: hudson.AbortException: script returned exit code 2
不幸的是,哈德逊。AbortException缺少任何用于获取退出状态的有用方法,因此如果需要实际值,则需要从消息中解析它(哎呀!)
与Javadoc https://javadoc.jenkins-ci.org/hudson/AbortException.html相反,当捕获到此异常时,构建不会失败。当它没有被抓住时,它就失败了!
更新: 如果您还想从shell命令得到STDERR输出,不幸的是Jenkins不能正确地支持这种常见的用例。一张2017年的JENKINS-44930票陷入了固执己见的乒乓状态,同时在解决方案方面没有取得任何进展-请考虑添加您的upvote。
至于现在的解决方案,可能有几个可能的方法:
a)将STDERR重定向到STDOUT 2>&1 -但这取决于你从主输出中解析出来,如果命令失败,你不会得到输出-因为你在异常处理程序中。
b)将STDERR重定向到一个临时文件(你之前准备的名称)2>文件名(但记得之后清理文件)-即。主代码变成:
def stderrfile = 'stderr.out'
try {
def dir1 = sh(script:"ls -la dir1 2>${stderrfile}", returnStdout:true).trim()
} catch (Exception ex) {
def errmsg = readFile(stderrfile)
println("Unable to read dir1: ${ex} - ${errmsg}")
}
c)反过来,设置returnStatus=true,不使用异常处理程序,始终捕获输出到文件中,即:
def outfile = 'stdout.out'
def status = sh(script:"ls -la dir1 >${outfile} 2>&1", returnStatus:true)
def output = readFile(outfile).trim()
if (status == 0) {
// output is directory listing from stdout
} else {
// output is error message from stderr
}
注意:上面的代码是Unix/ linux特定的- Windows需要完全不同的shell命令。
其他回答
最新版本的管道sh步骤允许您执行以下操作;
// Git committer email
GIT_COMMIT_EMAIL = sh (
script: 'git --no-pager show -s --format=\'%ae\'',
returnStdout: true
).trim()
echo "Git committer email: ${GIT_COMMIT_EMAIL}"
另一个特性是returnStatus选项。
// Test commit message for flags
BUILD_FULL = sh (
script: "git log -1 --pretty=%B | grep '\\[jenkins-full]'",
returnStatus: true
) == 0
echo "Build full flag: ${BUILD_FULL}"
这些选项是根据此问题添加的。
sh命令请参见官方文档。
对于声明式管道(见注释),你需要将代码包装到脚本步骤:
script {
GIT_COMMIT_EMAIL = sh (
script: 'git --no-pager show -s --format=\'%ae\'',
returnStdout: true
).trim()
echo "Git committer email: ${GIT_COMMIT_EMAIL}"
}
当前的Pipeline版本本机支持returnStdout和returnStatus,这使得从sh/bat步骤中获得输出或状态成为可能。
一个例子:
def ret = sh(script: 'uname', returnStdout: true)
println ret
正式文件。
如果你想获取stdout并知道命令是否成功,只需使用returnStdout并将其包装在异常处理程序中:
脚本化的管道
try {
// Fails with non-zero exit if dir1 does not exist
def dir1 = sh(script:'ls -la dir1', returnStdout:true).trim()
} catch (Exception ex) {
println("Unable to read dir1: ${ex}")
}
输出:
[Pipeline] sh
[Test-Pipeline] Running shell script
+ ls -la dir1
ls: cannot access dir1: No such file or directory
[Pipeline] echo
unable to read dir1: hudson.AbortException: script returned exit code 2
不幸的是,哈德逊。AbortException缺少任何用于获取退出状态的有用方法,因此如果需要实际值,则需要从消息中解析它(哎呀!)
与Javadoc https://javadoc.jenkins-ci.org/hudson/AbortException.html相反,当捕获到此异常时,构建不会失败。当它没有被抓住时,它就失败了!
更新: 如果您还想从shell命令得到STDERR输出,不幸的是Jenkins不能正确地支持这种常见的用例。一张2017年的JENKINS-44930票陷入了固执己见的乒乓状态,同时在解决方案方面没有取得任何进展-请考虑添加您的upvote。
至于现在的解决方案,可能有几个可能的方法:
a)将STDERR重定向到STDOUT 2>&1 -但这取决于你从主输出中解析出来,如果命令失败,你不会得到输出-因为你在异常处理程序中。
b)将STDERR重定向到一个临时文件(你之前准备的名称)2>文件名(但记得之后清理文件)-即。主代码变成:
def stderrfile = 'stderr.out'
try {
def dir1 = sh(script:"ls -la dir1 2>${stderrfile}", returnStdout:true).trim()
} catch (Exception ex) {
def errmsg = readFile(stderrfile)
println("Unable to read dir1: ${ex} - ${errmsg}")
}
c)反过来,设置returnStatus=true,不使用异常处理程序,始终捕获输出到文件中,即:
def outfile = 'stdout.out'
def status = sh(script:"ls -la dir1 >${outfile} 2>&1", returnStatus:true)
def output = readFile(outfile).trim()
if (status == 0) {
// output is directory listing from stdout
} else {
// output is error message from stderr
}
注意:上面的代码是Unix/ linux特定的- Windows需要完全不同的shell命令。
这是一个例子,我相信这是有意义的!
node('master'){
stage('stage1'){
def commit = sh (returnStdout: true, script: '''echo hi
echo bye | grep -o "e"
date
echo lol''').split()
echo "${commit[-1]} "
}
}
简单的回答是:
sh "ls -l > commandResult"
result = readFile('commandResult').trim()
我认为存在一个特性请求能够得到sh步骤的结果,但据我所知,目前没有其他选择。
编辑:詹金斯- 26133
EDIT2:不太确定从哪个版本开始,但是sh/bat步骤现在可以返回std输出,简单地说:
def output = sh returnStdout: true, script: 'ls -l'