Я делаю резервный скрипт для 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 строку? Или есть способ лучше?
В 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" \
)
На что указывает это ответ в 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
Вот как я печатаю stdout и stderr для разделения файлов с метками времени (подключение к ts из пакета Debian moreutils):
(./my_little_script.pl | ts %F\ %T > out.log) 2>&1 | ts > err.log
P.S. если у вас нет ts, создайте собственный псевдоним:
alias ts='while IFS= read -r line; do printf "%s %s\n" "$(date +%F\ %T)" "$line"; done'