Другая возможность, с некоторыми bash
ароматами, состоит в том, чтобы включить опцию pipefail
:
pipefail
Если установлено, возвращаемое значение конвейера
значение последней (самой правой)
Команда для выхода с ненулевым
статус или ноль, если все команды в
успешный выход из трубопровода. это
опция отключена по умолчанию.
set -o pipefail
...
command 2>&1 | tee -a file.txt || echo "Command (or tee?) failed with status $?"
Сказано, что единственный способ достижения функциональности PIPESTATUS
переносимым образом (например, так, чтобы он также работал с POSIX sh
) немного запутан, то есть требуется временной файл передать статус выхода канала обратно в родительский процесс оболочки:
{ command 2>&1 ; echo $? >"/tmp/~pipestatus.$$" ; } | tee -a file.txt
if [ "`cat \"/tmp/~pipestatus.$$\"`" -ne 0 ] ; then
...
fi
или, инкапсулируя для повторного использования:
log2file() {
LOGFILE="$1" ; shift
{ "$@" 2>&1 ; echo $? >"/tmp/~pipestatus.$$" ; } | tee -a "$LOGFILE"
MYPIPESTATUS="`cat \"/tmp/~pipestatus.$$\"`"
rm -f "/tmp/~pipestatus.$$"
return $MYPIPESTATUS
}
log2file file.txt command param1 "param 2" || echo "Command failed with status $?"
или, возможно, более обобщенно:
save_pipe_status() {
STATUS_ID="$1" ; shift
"$@"
echo $? >"/tmp/~pipestatus.$$.$STATUS_ID"
}
get_pipe_status() {
STATUS_ID="$1" ; shift
return `cat "/tmp/~pipestatus.$$.$STATUS_ID"`
}
save_pipe_status my_command_id ./command param1 "param 2" | tee -a file.txt
get_pipe_status my_command_id || echo "Command failed with status $?"
...
rm -f "/tmp/~pipestatus.$$."* # do this in a trap handler, too, to be really clean