Bash:如何为ssh命令实时过滤tee输出?

时间:2014-06-23 22:43:54

标签: linux bash ssh tee

我的目标是为ssh会话中使用的命令创建一个带时间戳的日志。

只要未过滤输出,Tee就会实时工作。当我tail -F test.log时,以下命令实时附加输出:

#!/bin/bash
ssh "$@" | tee -a test.log

但是,当我尝试通过this question中建议的方法修改tee的输出时,输出不再是实时的。例如:

#!/bin/bash
ssh "$@" | tee >(grep -e '.*\@.*\:.*\$.*' --color=never >> test.log)

奇怪的是,如果我用“yes”命令代替ssh命令,输出就会被实时过滤掉。

实时处理非常重要,因为我的脚本需要为每一行添加当前时间戳并尽可能多地删除输出。到目前为止,这是我的脚本:

#!/bin/bash
logfile=~/test.log
desc="sshlog ${@}"
tab="\t"
format_line() {
    while IFS= read -r line; do
        echo "$(date +"%Y-%m-%d %H:%M:%S %z")${tab}${desc}${tab}${line}"
    done
}
echo "[START]" | format_line >> $logfile
ssh "$@" | tee >(grep -e '.*\@.*\:.*\$.*' --color=never | format_line >> $logfile)
echo "[END]" | format_line >> $logfile

我该如何解决这个问题,为什么ssh命令对于tee的行为与yes命令不同?

1 个答案:

答案 0 :(得分:11)

问题很可能是grep正在缓冲其输出 - 选择大块输入,过滤它们并输出结果 - 因此它可以顺利处理yes的输出(因为{{1}快速生成大量输入以进行过滤和输出),而您的yes命令可能不会快速生成那么多输出。

ssh的许多版本提供了一种调整此缓冲的机制。既然你在Linux上,你可能会使用GNU Grep,它为此目的提供grep标志(参见the "Other Options" section of the GNU Grep Manual),这样输出一次只缓冲一行: / p>

--line-buffered