AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 507352
Accepted
T.M
T.M
Asked: 2019-03-20 23:58:12 +0800 CST2019-03-20 23:58:12 +0800 CST 2019-03-20 23:58:12 +0800 CST

使用 eval 获取第一个管道命令的返回码?

  • 772

我希望执行不同的命令并在之后检查返回代码,然后再转到脚本中的下一步。同时,我还希望使用 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

returncode 是 0 应该是 > 0

我希望 returncode 变量具有已执行命令的实际返回(在本例中为 ls -l 命令。

我已经看到有一个解决方案使用文件将命令的输出写入其中,然后从中读取返回代码(此处),但我正在寻找更优雅的解决方案。

bash eval
  • 3 3 个回答
  • 2978 Views

3 个回答

  • Voted
  1. Kusalananda
    2019-03-21T00:46:10+08:002019-03-21T00:46:10+08:00

    ls -l在这种特殊情况下,直接执行并对其退出状态采取行动会更容易:

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

    在bashshell 中,您还可以调查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"
    
    • 4
  2. Freddy
    2019-03-21T00:49:22+08:002019-03-21T00:49:22+08:00

    您可以使用将命令作为参数并将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"
    
    • 0
  3. Best Answer
    T.M
    2019-03-29T01:17:16+08:002019-03-29T01:17:16+08:00

    经过一些额外的测试,我发现这个代码开关被很好地包装并返回了执行的命令返回代码。@Freddy 发布的代码已接近完成。返回码在函数内部导出,但不在函数外部导出。

    shopt -s lastpipe 的使用取自此页面:Bash FAQ entry #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};
    
    • 0

相关问题

  • 通过命令的标准输出以编程方式导出环境变量[重复]

  • 从文本文件传递变量的奇怪问题

  • 虽然行读取保持转义空间?

  • `tee` 和 `bash` 进程替换顺序

  • 运行一个非常慢的脚本直到它成功

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve