将 stderr 和 stdout 传送到不同的命令(而不仅仅是文件)

将 stderr 和 stdout 传送到不同的命令(而不仅仅是文件)

我正在为 ldap 编写一个备份脚本。我希望错误转到 /var/log 中的一个文件,输出转到备份文件夹中的另一个文件。目前我正在重定向到一个临时文件,然后将临时文件发送到日志。不过我宁愿用一行代码来做这件事……

/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 和 stdout 重定向到不同的管道以将此命令压缩为一行,您有什么想法吗?或者有更好的方法吗?

答案1

正如本回答在Unix SE上:

我的WeirdCommand.sh

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

测试重定向.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

答案2

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

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" \
  )

答案3

这就是我如何将 stdout 和 stderr 打印到带有时间戳的单独文件中(从 Debian moreutils 包传输到 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'

相关内容