Передача stderr и stdout различным командам (не только файлам)


11

Я делаю резервный скрипт для ldap. Я хочу, чтобы ошибки шли в файл в / var / log, а вывод - в другой файл в папке резервной копии. В настоящее время я перенаправляю во временный файл и затем отправляю временный файл в журнал. Я бы предпочел сделать это как 1 лайнер, хотя ...

/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 строку? Или есть лучший способ?


1
Взгляните на эту демонстрацию: stackoverflow.com/a/16283739/1765658 или этот другой пример значения: unix.stackexchange.com/a/84012/27653
Ф. Хаури

Ответы:


10

Как указано в этом ответе на 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


24

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

1
Это правильный ответ.
Майкл Мартинес

Возможно, вы захотите перенаправить вывод обратно в stderr, если вы хотите сохранить цепочку вместо перенаправления в файл, что-то вроде этого: sh f>> (sed -e "s / ^ / stdout: /") 2>> ( sed -e "s / ^ / stderr: /"> & 2)
Джеймс Мур

Какое техническое название для >(process)обозначения?
Jchook

1
@jchook Я использую термин в первом предложении: «процесс замены».
Приостановлено до дальнейшего уведомления.

1

Вот как я печатаю stdout и stderr в отдельные файлы с временными метками (отправка по ts из пакета Debian moreutils):

(./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'
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.