在一个shell脚本,我如何回显所有shell命令调用和展开任何变量名?
例如,给定以下一行:
ls $DIRNAME
我希望脚本运行该命令并显示以下内容
ls /full/path/to/some/dir
目的是保存所有调用的shell命令及其参数的日志。是否有更好的方法来生成这样的日志?
在一个shell脚本,我如何回显所有shell命令调用和展开任何变量名?
例如,给定以下一行:
ls $DIRNAME
我希望脚本运行该命令并显示以下内容
ls /full/path/to/some/dir
目的是保存所有调用的shell命令及其参数的日志。是否有更好的方法来生成这样的日志?
当前回答
为了允许复合命令被回显,我使用eval加上Soth的exe函数来回显和运行命令。这对于管道命令非常有用,否则这些管道命令只显示不显示或只显示管道命令的初始部分。
没有eval:
exe() { echo "\$ $@" ; "$@" ; }
exe ls -F | grep *.txt
输出:
$
file.txt
eval:
exe() { echo "\$ $@" ; "$@" ; }
exe eval 'ls -F | grep *.txt'
的输出
$ exe eval 'ls -F | grep *.txt'
file.txt
其他回答
另一种选择是在你的脚本顶部放置"-x",而不是在命令行上:
$ cat ./server
#!/bin/bash -x
ssh user@server
$ ./server
+ ssh user@server
user@server's password: ^C
$
对于zsh,使用echo
setopt VERBOSE
为了调试,
setopt XTRACE
Set -x或Set -o xtrace展开变量并在行前打印一个小的+号。
Set -v或Set -o verbose不会在打印之前展开变量。
使用set +x和set +v关闭上述设置。
在脚本的第一行,可以输入#!/bin/sh -x(或-v)与后面脚本中的set -x(或-v)具有相同的效果。
上面的代码也适用于/bin/sh。
关于set属性和调试,请参阅bash-hackers的wiki。
$ cat shl
#!/bin/bash
DIR=/tmp/so
ls $DIR
$ bash -x shl
+ DIR=/tmp/so
+ ls /tmp/so
$
为了允许复合命令被回显,我使用eval加上Soth的exe函数来回显和运行命令。这对于管道命令非常有用,否则这些管道命令只显示不显示或只显示管道命令的初始部分。
没有eval:
exe() { echo "\$ $@" ; "$@" ; }
exe ls -F | grep *.txt
输出:
$
file.txt
eval:
exe() { echo "\$ $@" ; "$@" ; }
exe eval 'ls -F | grep *.txt'
的输出
$ exe eval 'ls -F | grep *.txt'
file.txt
Set -x会得到你想要的。
下面是一个示例shell脚本来演示:
#!/bin/bash
set -x #echo on
ls $PWD
这将展开所有变量,并在输出命令之前打印完整的命令。
输出:
+ ls /home/user/
file1.txt file2.txt