Czy można przechwycić stdout z polecenia sh DSL w potoku

92

Na przykład:

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

Wezmę:

output=0

Najwyraźniej otrzymuję kod zakończenia, a nie standardowe wyjście. Czy jest możliwe przechwycenie stdout do zmiennej potoku, tak że mogę uzyskać: output=foo jako mój wynik?

Jesse S
źródło

Odpowiedzi:

227

Teraz , gdy shkrok podpory powrocie stdout poprzez dostarczanie parametru 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)

Zobacz ten przykład .

iloahz
źródło
10
zanotuj .trim()część tej odpowiedzi, w przeciwnym razie możesz otrzymać znak nowej linii na końcu linii
Will Munn
2
append --shortto rev-parsemoże bezpośrednio uzyskać krótki hash
Leon
nie jestem pewien, co powoduje awarię, ale musiałem przekonwertować wyjście na ciąg również w ten sposóbgitCommit = sh(returnStdout: true, script: 'git rev-parse HEAD').toString().trim()
Balkrishna
cześć, co oznacza „.take (6)”?
Vano
1
@Vano, która odnosi się do metody take () Groovy, która w tym przypadku otrzyma pierwsze 6 znaków. docs.groovy-lang.org/docs/groovy-2.3.2/html/api/org/codehaus/...
ahillman3
47

Uwaga: powiązany problem z Jenkinsem został już rozwiązany.

Jak wspomniano w JENKINS-26133 , nie można było uzyskać danych wyjściowych powłoki jako zmiennej. Jako obejście sugerowano użycie zapisu-odczytu z pliku tymczasowego. Twój przykład wyglądałby więc następująco:

sh "echo foo > result";
def output=readFile('result').trim()
echo "output=$output";
Andrey Prokopiev
źródło
21
Dla nowicjuszy zapoznaj się z odpowiedzią stackoverflow.com/a/38912813/345845 poniżej, ponieważ zostało to ułatwione dzięki nowemu returnStdoutparametrowi przekazanemu do shkroku.
Baptiste Mathus
2
„nie można uzyskać wyniku powłoki jako zmiennej” - nieprawda. To jest hack, poprawna odpowiedź to returnStdout.
Graham,
4
Właściwie jedyną dobrą odpowiedzią jest to, że potrzebujesz zarówno polecenia, jakstdout i exit statuspolecenia powłoki. W innych przypadkach użyj returnStdoutparametru.
Simon Forsberg
4

Spróbuj tego:

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

Przetestowano na:

  • Jenkins ver. 2.19.1
  • Rurociąg 2.4
METAJIJI
źródło
3

Możesz spróbować użyć tej funkcji również do przechwycenia StdErr StdOut i kodu powrotu.

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

Ogłoszenie:

&>name means 1>name 2>name -- redirect stdout and stderr to the file name
GalloCedrone
źródło
1

Krótka wersja to:

echo sh(script: 'ls -al', returnStdout: true).result
A.Hab
źródło
0

Miałem ten sam problem i próbowałem prawie wszystkiego, a potem znalazłem, gdy dowiedziałem się, że próbuję go w niewłaściwym bloku. Próbowałem tego w bloku kroków, podczas gdy musi być w bloku środowiska.

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