Скрипт bash: как сохранить возвращаемое значение первой команды в конвейере?

Bash: я хочу запустить команду и передать результаты через какой-то фильтр, но если команда не удалась, я хочу вернуть значение ошибки команды, а не скучное возвращаемое значение фильтра:

Например:

if !(cool_command | output_filter); then handle_the_error; fi

Или же:

set -e
cool_command | output_filter

В любом случае это возвращаемое значение cool_command что меня волнует - условие "если" в первом случае или выход из сценария во втором случае.

Есть ли какая-то чистая идиома для этого?

3 ответа

Использовать PIPESTATUS встроенная переменная.

От man bash:

PIPESTATUS

Переменная массива (см. Массивы ниже), содержащая список значений состояния выхода из процессов в самом последнем выполняемом конвейере переднего плана (который может содержать только одну команду).

Следующий скрипт использует fifo для фильтрации вывода в отдельном процессе. Это имеет следующие преимущества перед другими ответами. Во-первых, это не специфично для Bash. В частности, это не зависит от PIPESTATUS. Во-вторых, вывод не останавливается до тех пор, пока команда не будет выполнена.

$ cat >test_filter.sh <<EOF
#!/bin/sh

cmd()
{  
    echo $1
    echo $2 >&2
    return $3
}

filter()
{  
    while read line
    do     
            echo "... $line"
    done
}

tmpdir=$(mktemp -d)
fifo="$tmpdir"/out
mkfifo "$fifo"

filter <"$fifo" &
pid=$!
cmd a b 10 >"$fifo" 2>&1
ret=$?
wait $pid
echo exit code: $ret
rm -f "$fifo"
rmdir "$tmpdir"
EOF
$ sh ./test_filter.sh
... a
... b
exit code: 10

Если вам не нужно отображать вывод ошибки команды, вы можете сделать что-то вроде

if ! echo | mysql $dbcreds mysql; then
    error "Could not connect to MySQL. Did you forget to add '--db-user=' or '--db-password='?"
    die "Check your credentials or ensure server is running with /etc/init.d/mysqld status"
fi

В этом примере error и die являются определенными функциями. в другом месте в сценарии. $dbcreds также определен, хотя он построен из параметров командной строки. Если команда не сгенерировала ошибку, ничего не возвращается. Если произойдет ошибка, текст будет возвращен этой конкретной командой.

Поправь меня, если я ошибаюсь, но у меня складывается впечатление, что ты действительно хочешь сделать что-то более замысловатое, чем

[ `id -u` -eq '0' ] || die "Must be run as root!"

где вы фактически получаете идентификатор пользователя перед оператором if, а затем выполняете тест. Делая это таким образом, вы можете отобразить результат, если вы выберете. Это было бы

UID=`id -u`
if [ $UID -eq '0' ]; then
    echo "User is root"
else
    echo "User is not root"
    exit 1 ##set an exit code higher than 0 if you're exiting because of an error
fi
Другие вопросы по тегам