Potokuj stderr i stdout do różnych poleceń (nie tylko do plików)

11

Tworzę kopię zapasową skryptu dla ldap. Chcę, aby błędy trafiły do ​​pliku w / var / log, a dane wyjściowe do innego pliku w folderze kopii zapasowej. Obecnie przekierowuję do pliku tymczasowego, a następnie wysyłam plik tymczasowy do dziennika. Wolałbym to zrobić jako 1 liniowiec ...

/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

Jakieś pomysły na przekierowanie stderr i stdout do różnych rur w celu skondensowania tego polecenia w 1 linii? Czy jest jakiś lepszy sposób?

Robert
źródło
1
Spójrz na tę demonstrację: stackoverflow.com/a/16283739/1765658 lub inny przykładowy przykład: unix.stackexchange.com/a/84012/27653
F. Hauri

Odpowiedzi:

10

Jak wskazano w tej odpowiedzi w 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

Zyski z pracy:

  • stderr.log 6

  • stdout.log 1

Łowca jeleni
źródło
24

W Bash możesz użyć zastępowania procesów, aby zarządzać dodatkowymi deskryptorami plików. Może się to wydawać nieco ładniejsze niż metoda zamiany deskryptorów plików.

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

Twoje polecenie może wyglądać mniej więcej tak:

/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" \
  )
Wstrzymano do odwołania.
źródło
1
To jest poprawna odpowiedź.
Michael Martinez,
Możesz przekierować wyjście z powrotem do stderr, jeśli chcesz zachować łańcuch zamiast przekierowywać do pliku, coś takiego: sh f>> (sed -e "s / ^ / stdout: /") 2>> ( sed -e "s / ^ / stderr: /"> & 2)
James Moore,
Jaka jest nazwa techniczna >(process)notacji?
jchook
1
@ jchook Używam terminu w pierwszym zdaniu: „podstawienie procesu”.
Wstrzymano do odwołania.
1

Oto jak wypisuję stdout i stderr do oddzielnych plików ze znacznikami czasu (przesyłanie do ts z pakietu moreutils Debiana):

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

PS, jeśli nie masz ts, stwórz własny alias:

alias ts='while IFS= read -r line; do printf "%s %s\n" "$(date +%F\ %T)" "$line"; done'
Wewnętrzny błąd serwera
źródło