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 / 问题 / 439035
Accepted
Timothy Swan
Timothy Swan
Asked: 2018-04-21 14:31:05 +0800 CST2018-04-21 14:31:05 +0800 CST 2018-04-21 14:31:05 +0800 CST

捕获函数中的所有命令

  • 772

我正在使用一组脚本,其函数被视为只读。这些函数不仅仅是一个命令列表,例如,可以有循环和更改目录,甚至可以调用其他函数:

func() {
    cd folder/
    run command1
    mkdir folder2/ ; cd folder2/
    run command2
}

有一会儿,假装我可以更改脚本,向您展示我想要的内容,可能看起来像这样:

func() {
    cd folder/
    string[0]="command1" ; run command1 |& tee out0.log ; result[0]="$?" ; finished_command_number 0
    mkdir folder2/ ; cd folder2/
    string[1]="command2" ; run command2 |& tee out1.log ; result[1]="$?" ; finished_command_number 1
}

因此,对于可以通过管道传输但不能用于cd循环的命令,我想存储一个字符串,存储标准输出(stderr),存储退出代码,然后运行另一个命令。但是,我无法添加它,它必须在我的脚本中完成,该脚本使用 func() 调用脚本。

为了仅获得 post 命令调用功能,我尝试复制该函数并使用 a 从我的脚本中运行它trap foo debug,但这似乎并没有传播到函数中。我不认为逐行复制会起作用,因为cdandloops不能真正被隔离,因为它们是控制语句而不是 subshel​​l 命令。

如果字符串可以只是命令本身的函数,那仍然很有用。

bash shell-script
  • 2 2 个回答
  • 1478 Views

2 个回答

  • Voted
  1. Best Answer
    Timothy Swan
    2018-05-02T14:15:12+08:002018-05-02T14:15:12+08:00
    #!/bin/bash
    # test.sh
    post() {
            echo "post [$BASH_COMMAND] [$?]"
            echo "== $RANDOM =="
    }
    set -o functrace
    trap post debug
    func() {
            . check.sh
            tryme |& tee out.txt
    }
    func
    

    输出可以通过随机标记的行进行过滤。我应该进一步测试它以查看它与多个进程的工作情况如何,但它似乎与短期命令一起工作得很好。请注意退出代码滞后一个命令,因为显然首先调用了调试。

    #!/bin/bash
    # check.sh
    tryme() {
            echo "one"
            echo "two"
            mkdir -p hello
            cd hello/
            echo "three"
            false
            echo "four"
    }
    

    ===

    $ bash test.sh 
    post [func] [0]
    == 22542 ==
    post [func] [0]
    == 10758 ==
    post [. check.sh] [0]
    == 9115 ==
    post [tryme 2>&1] [0]
    == 11979 ==
    post [tee out.txt] [0]
    == 17814 ==
    post [tryme 2>&1] [0]
    == 22838 ==
    post [echo "one"] [0]
    == 5251 ==
    one
    post [echo "two"] [0]
    == 18036 ==
    two
    post [mkdir -p hello] [0]
    == 4247 ==
    post [cd hello/] [0]
    == 21611 ==
    post [echo "three"] [0]
    == 24685 ==
    three
    post [false] [0]
    == 8557 ==
    post [echo "four"] [1]
    == 7565 ==
    four
    
    • 2
  2. DopeGhoti
    2018-04-21T14:50:02+08:002018-04-21T14:50:02+08:00

    如果我正确阅读了您的问题,那么您正在寻找一种方法:

    • 独立于函数调用的输出捕获 stdout 和 stderr
    • 保留当前工作目录,防止在函数执行过程中对其进行任何更改

    这可以相当简单地完成:

    #!/bin/bash
    stdout=/path/to/output.std
    stderr=/path/to/output.err
    
    somefunc() {
        : Do things
    }
    
    ( somefunc arg1 arg2 ) >> $stdout 2>> $stderr
    

    在子 shell 中运行该函数将允许它执行与更改工作目录相关的任何操作,并正常输出到 stdout 和 stderr。当 subshel​​l 退出时,脚本的工作目录不变,所有的 stdout 和 stderr 输出都被捕获并重定向到您的日志文件中。

    添加一个(全局或其他)数组来捕获命令的退出代码是微不足道的:

    declare -a exitcodes
    ( exit 0 )
    exitcodes+=( $? )
    ( exit 2 )
    exitcodes+=( $? )
    ( exit 0 )
    exitcodes+=( $? )
    
    echo ${exitcodes[@]}
    
    • 1

相关问题

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

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

  • MySQL Select with function IN () with bash array

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

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

Sidebar

Stats

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

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

    • 4 个回答
  • Marko Smith

    ssh 无法协商:“找不到匹配的密码”,正在拒绝 cbc

    • 4 个回答
  • Marko Smith

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

    • 5 个回答
  • Marko Smith

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

    • 3 个回答
  • Marko Smith

    如何卸载内核模块“nvidia-drm”?

    • 13 个回答
  • 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
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Wong Jia Hau ssh-add 返回:“连接代理时出错:没有这样的文件或目录” 2018-08-24 23:28:13 +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
  • Martin Hope
    Bagas Sanjaya 为什么 Linux 使用 LF 作为换行符? 2017-12-20 05:48:21 +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