evalを使用して最初のパイプラインコマンドの戻りコードを取得しますか?

evalを使用して最初のパイプラインコマンドの戻りコードを取得しますか?

別のコマンドを実行してスクリプトの後続のステップを続行する前に、戻りコードを確認したいと思います。同時に、teeコマンドを使用して実行されたコマンドの出力をファイルに記録したいと思います。例:

#set non-existing folder
local_path="~/njn"
log_path_file="test.log"
cmd="ls -l ${local_path} | tee -a ${log_path_file}";
eval ${cmd}

returncode=$?
echo "execution result: ${returncode}"  | tee -a ${log_path_file};

if [ ${returncode} -eq 0 ]; then
  echo "success"  | tee -a ${log_path_file}
else
  echo "not success"  | tee -a ${log_path_file}
fi

戻りコードは0です。 0より大きくなければなりません。

returncode変数が実行されたコマンド(この場合はls -lコマンド)の実際の戻り値を持つことを望みます。

ファイルを使用してコマンド出力を作成し、ファイルから戻りコードを読み取るソリューションがあります(ここ)、しかし私はもう少しエレガントな解決策を探しています。

答え1

ls -lこの特別なケースでは、終了ステータスを実行してアクションを実行する方が簡単です。

if ls -l "$local_path"; then
    echo 'success'
else
    printf 'failure (code %d)\n' "$?"
fi | tee -a "$log_path_file"

bashシェルでは、配列の値を参照することもできますPIPESTATUS

$ false | true | true | false | false
$ printf '%s\n' "${PIPESTATUS[@]}"
1
0
0
1
1

あなたの場合:

ls -l "$local_path" | tee -a "$log_path_file"

ls_status=${PIPESTATUS[0]}
if [ "$ls_status" -eq 0 ]; then
    echo 'success'
else
    printf 'failure (code %d)\n' "$ls_status"
fi | tee -a "$log_path_file"

答え2

コマンドを引数として使用し、tee出力(stdoutとstderr)をログファイルに送信する関数を使用できます。

local_path="~/njn"
log_path_file="test.log"

function log_cmd ()
{
        {
                "$@"
                returncode=$?
                if [ "$returncode" -eq 0 ]; then
                  echo "[successfully executed \"$@\"]"
                else
                  echo "[failed to execute \"$@\", exit code: ${returncode}]"
                fi
                return $returncode

        } 2>&1 | tee -a "$log_path_file"
        # return exit code of first command in pipeline
        return ${PIPESTATUS[0]}
}

log_cmd ls -l "$local_path"
log_cmd echo "hello world"

答え3

いくつかの追加テストの後、このコードスイッチが見事に仕上げられ、実行されたコマンドリターンコードを返すことがわかりました。 @Freddyが投稿したコードはほぼ完成しました。戻りコードは関数の内部にエクスポートされますが、関数の外部にはエクスポートされません。

shopt -s lastpipe の使用はこのページから取られました:Bash FAQ 項目 #24: 「ループに変数を設定しました。なぜループが終了した後に突然消えるのですか?それともパイプを介してデータを読み取ることができないのですか?」

これが最終作業ソリューションです。

#!/bin/bash

log_path_file="./logs/test.log"
exe_cmd()
{
    echo "`date +%Y-%m-%d---%r` [Info]: Command to execute: $@"  | tee -a ${log_path_file};
    echo ""  | tee -a ${log_path_file};
    echo ""  | tee -a ${log_path_file};

    set +m
    shopt -s lastpipe

    cmdResult=0
    {
            "$@"
            returncode=$?
            # save result code
            cmdResult=${returncode}

            if [ "$returncode" -eq 0 ]; then
              echo "`date +%Y-%m-%d---%r` [Info]: successfully executed \"$@\""
            else
              echo "`date +%Y-%m-%d---%r` [Info]: failed to execute \"$@\", exit code: ${returncode}"
            fi
    } 2>&1 | tee -a "$log_path_file"

    echo "`date +%Y-%m-%d---%r` [Info]: cmdResult result ${#cmdResult[@]}"

    return ${#cmdResult[@]};
}

cmd="scp some_user@$some_host:some_path/* a_local_path/sub_path";
exe_cmd ${cmd}
returncode=$?
echo "`date +%Y-%m-%d---%r` [Info]: scp execution result: ${returncode}"  | tee -a ${log_path_file};

関連情報