Можно ли захватить stdout из команды sh DSL в конвейере

92

Например:

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

Я получу:

output=0

Итак, по-видимому, я получаю код выхода, а не стандартный вывод. Можно ли записать стандартный вывод в переменную конвейера, чтобы я мог получить: в output=foo качестве результата?

Джесси С
источник

Ответы:

227

Теперь , как shшаг поддерживает возвращение стандартного вывода путем подачи параметра 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)

См. Этот пример .

Iloahz
источник
10
обратите внимание на .trim()часть этого ответа, иначе вы можете получить символ новой строки в конце строки
Уилл Манн
2
append --shortto rev-parseможет напрямую получить короткий хэш
Леон
не уверена , что причина неудачи , но я должен был преобразовать вывод в строку , также как этоgitCommit = sh(returnStdout: true, script: 'git rev-parse HEAD').toString().trim()
Балкришна
привет, что означает '.take (6)'?
Vano
1
@Vano, который ссылается на Groovy-метод take (), который в этом случае получит первые 6 символов. docs.groovy-lang.org/docs/groovy-2.3.2/html/api/org/codehaus/…
ahillman3
47

Примечание. Связанная проблема с Jenkins с тех пор решена.

Как упоминалось в JENKINS-26133, невозможно было получить вывод оболочки в виде переменной. В качестве обходного пути предлагается использовать запись-чтение из временного файла. Итак, ваш пример выглядел бы так:

sh "echo foo > result";
def output=readFile('result').trim()
echo "output=$output";
Андрей Прокопьев
источник
21
Для новичков см. Ответ stackoverflow.com/a/38912813/345845 ниже, с тех пор это стало проще с новым returnStdoutпараметром, переданным на shшаг.
Baptiste Mathus
2
«невозможно получить вывод оболочки как переменную» - неверно. Это взлом, правильный ответ - returnStdout.
Грэм
4
Единственный раз , когда это на самом деле хороший ответ, если вам нужно какstdout и exit statusиз командной оболочки. В других случаях используйте 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
    ...
}

Проверено на:

  • Jenkins ver. 2.19.1
  • Трубопровод 2.4
METAJIJI
источник
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
GalloCedrone
источник
1

Краткая версия:

echo sh(script: 'ls -al', returnStdout: true).result
A.Hab
источник
0

У меня была та же проблема, и я перепробовал почти все, что нашел, когда узнал, что пробую не в том блоке. Я пробовал это в блоке шагов, тогда как он должен быть в блоке среды.

        stage('Release') {
                    environment {
                            my_var = sh(script: "/bin/bash ${assign_version} || ls ", , returnStdout: true).trim()
                                }
                    steps {                                 
                            println my_var
                            }
                }
Нусрат-уль-Хасан
источник