Inoltra stderr e stdout a diversi comandi (non solo ai file)


11

Sto realizzando uno script di backup per ldap. Voglio che gli errori vadano a un file in / var / log e che l'output vada a un altro file nella cartella di backup. Attualmente sto reindirizzando a un file temporaneo e quindi inviando il file temporaneo al registro. Preferirei fare questo come 1 liner però ...

/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

Qualche idea su come reindirizzare stderr e stdout su diverse pipe per condensare questo comando in 1 riga? O c'è un modo migliore?


1
Dai un'occhiata a questa dimostrazione: stackoverflow.com/a/16283739/1765658 o questo altro esempio di significato: unix.stackexchange.com/a/84012/27653
F. Hauri

Risposte:


10

Come indicato da questa risposta a 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

Rendimenti correnti:

  • stderr.log 6

  • stdout.log 1


24

In Bash, è possibile utilizzare la sostituzione del processo per gestire i descrittori di file extra per te. Potresti trovare questo un po 'più ordinato rispetto al metodo di scambio del descrittore di file.

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

Il tuo comando potrebbe assomigliare a questo:

/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
Questa è la risposta corretta
Michael Martinez,

Potresti voler reindirizzare l'output su stderr se vuoi mantenere la catena invece di reindirizzare a un file, qualcosa del genere: sh f>> (sed -e "s / ^ / stdout: /") 2>> ( sed -e "s / ^ / stderr: /"> & 2)
James Moore,

Qual è il nome tecnico per la >(process)notazione?
jchook,

1
@jchook Uso il termine nella prima frase: "sostituzione del processo".
In pausa fino a ulteriore avviso.

1

Questo è il modo in cui stampo stdout e stderr per separare i file con i timestamp (reindirizzamento a ts dal pacchetto Debian moreutils):

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

PS se non hai ts, crea il tuo alias:

alias ts='while IFS= read -r line; do printf "%s %s\n" "$(date +%F\ %T)" "$line"; done'
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.