É possível capturar o stdout do comando sh DSL no pipeline

94

Por exemplo:

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

Eu vou pegar:

output=0

Então, aparentemente eu recebo o código de saída em vez do stdout. É possível capturar o stdout em uma variável de pipeline, de modo que eu pudesse obter: output=foo como meu resultado?

Jesse S
fonte

Respostas:

230

Agora , a shetapa oferece suporte ao retorno de stdout , fornecendo o parâmetro returnStdout.

// 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)

Veja este exemplo .

iloahz
fonte
10
observe a .trim()parte desta resposta, caso contrário, você pode obter um caractere de nova linha no final da linha
Will Munn
2
anexar --shorta rev-parsepode obter diretamente um hash curto
Leon,
não tenho certeza do que está causando a falha, mas eu tive que converter a saída para string também assimgitCommit = sh(returnStdout: true, script: 'git rev-parse HEAD').toString().trim()
Balkrishna
oi, o que '.take (6)' significa?
Vano
1
@Vano que se refere ao método take () do Groovy, que obterá os primeiros 6 caracteres neste caso. docs.groovy-lang.org/docs/groovy-2.3.2/html/api/org/codehaus/…
ahillman3
46

Observação: o problema do Jenkins vinculado já foi resolvido.

Conforme mencionado em JENKINS-26133 , não foi possível obter a saída do shell como uma variável. Como solução alternativa, sugerimos o uso de escrita-leitura de arquivo temporário. Então, seu exemplo seria parecido com:

sh "echo foo > result";
def output=readFile('result').trim()
echo "output=$output";
Andrey Prokopiev
fonte
21
Para os novatos, consulte a resposta stackoverflow.com/a/38912813/345845 abaixo; isso ficou mais fácil com o novo returnStdoutparâmetro passado para a shetapa.
Baptiste Mathus
2
"não é possível obter a saída do shell como uma variável" - não é verdade. Este é um hack, a resposta correta é returnStdout.
Graham
4
A única vez que esta é realmente uma boa resposta é se você precisa tanto o stdouteo exit statusdo comando shell. Outras vezes, use o returnStdoutparâmetro.
Simon Forsberg
4

Experimente isto:

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
    ...
}

Testado em:

  • Jenkins ver. 2.19.1
  • Pipeline 2.4
METAJIJI
fonte
3

Você também pode tentar usar essas funções para capturar StdErr StdOut e o código de retorno.

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}"
    }
}

Aviso prévio:

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

Uma versão curta seria:

echo sh(script: 'ls -al', returnStdout: true).result
A.Hab
fonte
0

Eu tive o mesmo problema e tentei quase tudo que encontrei depois que descobri que estava tentando no bloco errado. Eu estava tentando no bloco de etapas, mas precisa estar no bloco de ambiente.

        stage('Release') {
                    environment {
                            my_var = sh(script: "/bin/bash ${assign_version} || ls ", , returnStdout: true).trim()
                                }
                    steps {                                 
                            println my_var
                            }
                }
Nusrat ul Hassan
fonte