Совместимость скриптов: Сохранить $? для использования позже

Я хотел бы написать небольшую часть скрипта, который сохраняет статус ошибки, выполняет другой код и устанавливает статус ошибки в исходный статус ошибки. В Bash это выглядит так:

<< some command >>; _EXIT=$?;( <<other code here>> ;exit $_EXIT) 

Но мне нужен код, который будет работать независимо от того, выполняется ли он под bash, zsh, csh или tcsh. Я не знаю, какая оболочка будет использоваться заранее, потому что она определяется пользователем. Пользователь также решает << некоторая команда >>.

  • Вывести результат печати непосредственно в bash без использования if
  • Awk - сопоставление по 2 столбцам для разных строк
  • Как запустить последующие оболочки с правами root?
  • Отображение ~ для $ HOME в приглашении zsh
  • Где найти ссылку на форматирование printf?
  • Сбой сценария: Ошибка синтаксиса: «(« неожиданно
  • Можно с уверенностью предположить, что << other code >> будет работать во всех оболочках, но небезопасно предполагать, что вы можете записать файл (поэтому его включение в файл не будет работать).

    Задний план

    GNU Parallel выполняет команды, заданные пользователем в оболочке, принятой пользователем. Когда команда завершена, GNU Parallel выполняет некоторую очистку. В то же время GNU Parallel должна запоминать значение выхода команды, заданной пользователем. Код, выполняемый перед приведенным выше фрагментом, является заданной пользователем командой. << другой код >> – это код очистки.

  • Как использовать вставленные значения с помощью getopts
  • Вывод аргумента в сценарии bash?
  • Это опечатка в разделе перенаправления руководства Bash?
  • Выполнять поиск по двум спискам
  • Вывод цветного скрипта только при вызове из интерактивной оболочки
  • удалите ^ M из solaris perl одним лайнером в том же файле
  • 2 Solutions collect form web for “Совместимость скриптов: Сохранить $? для использования позже”

    Этот ответ работает в bash-подобных оболочках и csh-подобных оболочках, и для него не требуется perl:

     sh -c '<< other code >>; [ "$0" != "" ] && exit "$0"; exit "$1"' "$?" "$status" 

    или вы можете добавить параметр фиктивного / пэда:

     sh -c '<< other code >>; [ "$1" != "" ] && exit "$1"; exit "$2"' foo "$?" "$status" 

    и тест можно изменить на [ ! -z "$0" ] [ ! -z "$0" ] (или "$1" ). Я не знаю, будет ли это работать в zsh или рыбе. Каждая оболочка, с которой я знаком, будет оценивать $? и $status когда команда разобрана, и передайте их команде sh -c качестве параметров. Оболочка sh присваивает их двум последним позиционным параметрам. Когда «другой код» заканчивается, оболочка проверяет параметр позиционирования, который получил свое значение от $? чтобы определить, является ли он нулевым или ненулевым, и выйдите с соответствующим значением позиционного параметра.

    Это не сработает, если окружающая оболочка использует нечто, отличное от $? или $status чтобы сохранить статус выхода. Он также потерпит неудачу, если оболочка разрешит $? (не нуль), если это не статус выхода, или $? является недопустимой последовательностью символов, которая заставляет всю команду отклоняться как синтаксическую ошибку.

    Использовать perl:

     perl -e '$a=shift; `<< other code >>`; exit $a' $? # Fails in csh 

    Редактировать Это работает и в csh (и zsh, tcsh, sh, pdksh, ksh93 – хотя и не рыба):

     perl -e '$a=shift; `<< other code >>`; $a=~s/h// and exit $a; exit shift' "$?h" "$status" 
    Linux и Unix - лучшая ОС в мире.