将stderr和stdoutpipe理到不同的命令(不仅仅是文件)

我正在为ldap制作备份脚本。 我希望将错误转移到/ var / log中的文件,并将输出转到备份文件夹中的另一个文件。 目前我正在redirect到临时文件,然后将临时文件发送到日志。 我宁愿做一个class轮,虽然…

/usr/bin/ldapsearch -x -LLL -b "dc=contoso,dc=com" "(objectclass=*)" -h ldap.server -v 2>>/tmp/ldaptmp.err | gzip -c > /mnt/backups/ldap/`date +\%Y\%m\%d`.ldif.gz || logger -t ldapbackup -p local6.err error exit $? cat /tmp/ldaptmp.err | grep -v "ldap_initialize( ldap://ldap.server )" | grep -v "filter: (objectclass=\*)" | grep -v "requesting: All userApplication attributes" >$ERR_LOG rm -f /tmp/ldaptmp.err 

有关如何将stderr和stdoutredirect到不同pipe道以将此命令压缩为1行的任何想法? 还是有更好的办法?

正如Unix SE上的这个答案所示:

MyWeirdCommand.sh

 #!/bin/bash echo "1 2 3" echo "4 5 6" >&2 

testRedirection.sh:

 #!/bin/bash (./MyWeirdCommand.sh | cut -f1 >stdout.log) 3>&1 1>&2 2>&3 | cut -f3 >stderr.log 

运行收益率:

  • stderr.log 6

  • stdout.log 1

在Bash中,您可以使用stream程replace为您pipe理额外的文件描述符。 您可能会发现这比文件描述符交换方法看起来更整洁一些。

 command > >(process_stdout) 2> >(process_stderr) 

你的命令可能看起来像这样:

 /usr/bin/ldapsearch -x -LLL -b "dc=contoso,dc=com" "(objectclass=*)" -h ldap.server -v \ > >( \ gzip -c > /mnt/backups/ldap/$(date '+%Y%m%d').ldif.gz || logger -t ldapbackup -p local6.err error exit $? ) \ 2> >( \ grep -Ev "ldap_initialize( ldap://ldap.server )|filter: (objectclass=\*)|requesting: All userApplication attributes" > "$err_log" \ ) 

这是我如何打印stdout和stderr来分隔带有时间戳的文件(从Debian的moreutils包pipe道到ts):

 (./my_little_script.pl | ts %F\ %T > out.log) 2>&1 | ts > err.log 

PS如果你没有ts,请自己别名:

 alias ts='while IFS= read -r line; do printf "%s %s\n" "$(date +%F\ %T)" "$line"; done'