我需要在bash脚本中运行一个命令,并在运行时间过长时终止它。我还需要捕获变量的所有输出。如果命令首先完成,我需要释放/终止看门狗进程(例如睡眠),因为我可能会运行这样的命令列表。
不幸的是,"超时"命令对我来说不可用,否则我可以做这样的事情:
output=`timeout -s 9 $TIMEOUT my-command`
并且检查退出代码124以查看是否存在超时。
因此,我选择的解决方案是由@Dmitry解决类似的问题:
( my_command ) & pid=$!
( sleep $TIMEOUT && kill -HUP $pid ) 2>/dev/null & watcher=$!
wait $pid 2>/dev/null && pkill -HUP -P $watcher
不幸的是,以下内容没有捕获$output的任何内容:
( output=`my_command` ) & pid=$!
我可以将输出转储到一个文件中,然后像这样加载到变量中,但我宁愿不使用文件:
( `my_command >$outfile` ) & pid=$!
...
output=`cat $outfile`
rm -f $outfile
我的问题是:有更好的方法吗?理想情况下,在不使用文件的情况下将stderr捕获到另一个变量?
幸运的是,$()
表示法允许多个命令,所以您可以这样做:
output=$(
( my_command ) & pid=$!
( sleep $TIMEOUT && kill -HUP $pid ) 2>/dev/null & watcher=$!
wait $pid 2>/dev/null && pkill -HUP -P $watcher
)
您还可以使用常规的()
对命令进行分组,然后重定向它们的所有输出。可以使用2>&1
将stderr重定向到stdout,因此您最终得到的是:
output=$(
(
( my_command ) & pid=$!
( sleep $TIMEOUT && kill -HUP $pid ) 2>/dev/null & watcher=$!
wait $pid 2>/dev/null && pkill -HUP -P $watcher
) 2>&1
)