Перенаправить stdout в команду-A в случае успеха, иначе stderr в команду-b. При этом избегая использования временных файлов

Перенаправить stdout в команду-A в случае успеха, иначе stderr в команду-b. При этом избегая использования временных файлов

У меня есть команда, которая пишет в stdout, когда она счастлива, и в stderr, когда нет. Я хочу проверить статус выхода команды и сделать что-нибудь либо со stdout, либо со stderr.

if ! command >/tmp/stdout 2>/tmp/stderr; then
  // do something with /tmp/stderr
  exit 1
fi

// do something else with /tmp/stdout

Конечно, можно иметь только один файл вместо двух, но я хотел бы избавиться от всех файлов, а не только от одного.

Есть ли способ избежать временных файлов? Я пробовал с mkfifo и пользовательскими файловыми дескрипторами, но у меня не получилось это сделать.

решение1

Вы можете использовать синтаксис подстановки команд $(..)в ifусловии и посмотреть, успешно ли выполнена команда или нет, и при этом сохранить результат команды как в переменной, так stdoutи stderrв переменной.

if ! var=$(cmd 2>&1); then
    printf 'process stderr contents from $var'
fi

Вы можете увидеть, как это работает, если я смоделирую cmdзапуск в виде простого скрипта, который делает это

# cat temp.sh    
echo foo >&2
exit 1

и если я запущу скрипт как

if ! var=$(bash tmp.sh 2>&1); then
    printf '%s\n' 'process stderr contents from $var'
    printf '%s\n' "$var"
fi

Таким же образом вы можете заставить его работать для захвата, stdoutесли подстановка команды прошла успешно, что будет в elseпредложении в приведенном выше примере. В любом случае, манипулирование содержимым "$var"(убедитесь, что кавычки включены) гарантирует, что вы обрабатываете результат так, как если бы он был сохранен в файле.

Вы можете пойти дальше и процитировать синтаксис подстановки команднетпозвольте оболочке выполнить разбиение результатов на слова. Например, сделав это так, как показано ниже. Это может быть не нужно для простых случаев, таких как тот, который я показал здесь, но для случаев, когда результаты содержат некоторые специальные метасимволы оболочки

if ! var="$(bash tmp.sh 2>&1)"; then

Примечание: название вопроса изменено с момента публикации первой версии ответа.

решение2

вы можете использовать код ниже

if [[ $? == 0 ]]
then
command 1>stdout.txt
else
echo "exit code is not successfull"
command 2> stderr.txt
fi

Связанный контент