Ich verwende derzeit das folgende Setup, um die Ausgabe mehrerer Befehle umzuleiten:
echo "Some normal commands"
(
echo "Error: something happened"
echo "Warning: this incident will be logged"
) >> logfile
echo "More normal commands"
Dies ist ziemlich nützlich und funktioniert auch mit Rohren.
Ist das der beste Weg, dies zu tun? Gibt es eine Alternative, die ich in Betracht ziehen sollte?
bash
shell-script
io-redirection
wchargin
quelle
quelle
:)
Antworten:
Die Alternative besteht darin, Klammern anstelle von Klammern zu verwenden. Diese Änderung führt die Befehle in der aktuellen Shell aus, nicht in einer Subshell
Ref: https://www.gnu.org/software/bash/manual/bashref.html#Command-Grouping
Dies ist besonders relevant, wenn Sie Variablen innerhalb der Gruppe ändern:
quelle
{ }
aber nicht( )
.)(echo msg1; echo msg2)
- aber bei geschweiften Klammern muss es{ echo msg1; echo msg2;}
ein Leerzeichen nach dem{
und ein Semikolon (;
) oder ein kaufmännisches Und (&
) vor dem sein}
.Glenns Antwort ist gut - die Unterscheidung zwischen
( ... )
und{ ... }
ist wichtig.Eine Strategie, die ich häufig für die Fehlerausgabe verwende, wie z. B. Ihre Frage, ist der
tee
Befehl. Sie könnten so etwas tun:Der
tee
Befehl sendet die Ausgabe an zwei Stellen.-a
Die Option "hängt" die Ausgabe an die benannte Datei an, und der Befehl leitet die Eingabe auch an stdout weiter. Das>&2
am Ende der Zeile stehendetee
stdout leitet es an stderr weiter, was möglicherweise anders gehandhabt wird (dh in einem Cron-Job).Ein weiterer Tipp, den ich häufig in Shell-Skripten verwende, besteht darin, das Verhalten der Debug- oder ausführlichen Ausgabe zu ändern, je nachdem, ob das Skript auf einem Terminal ausgeführt wird oder über eine
-v
Option verfügt. Zum Beispiel:Skripte können mit etwas Allgemeinem wie diesem oben beginnen, wobei die Ausgabe von Verbose und Debug im gesamten Skript verteilt ist. Es ist nur eine Möglichkeit, dies zu tun - es gibt viele, und verschiedene Leute werden alle ihre eigene Art haben, mit diesen Dingen umzugehen, besonders wenn sie schon eine Weile da sind. :) :)
Eine weitere Option besteht darin, Ihre Ausgabe mit einem "Handler" zu verarbeiten - einer Shell-Funktion, die möglicherweise intelligentere Aufgaben ausführt. Zum Beispiel:
(Beachten Sie, dass dies
${var^^}
nur Bash ist.)Dadurch wird eine Shell-Funktion erstellt, die möglicherweise die
syslog
Funktionen Ihres Systems verwendet (mit demlogger
Befehl) to send things to system logs. The
logme (). Die Funktion kann entweder mit Optionen verwendet werden, die einzelne Zeilen von Protokolldaten generieren, oder mit mehreren Eingabezeilen, die auf stdin verarbeitet werden. Spielen Sie damit, wenn dies der Fall ist scheint ansprechend.Beachten Sie, dass dies ein Beispiel ist und wahrscheinlich nicht wörtlich kopiert werden sollte, es sei denn, Sie verstehen es und wissen, dass es genau das tut, was Sie brauchen. Eine bessere Idee ist es, die Konzepte hier zu übernehmen und sie selbst in Ihren eigenen Skripten zu implementieren.
quelle
function log () { cat >> $logfile }
, die im Grunde eine einfachere Version von Ihrer istlogme
.ts(1)
; Verwendung :{ printf '%s\n' "Warning text"; printf '%s\n' "This event will be logged"; } | ts '[%Y-%m-%d %T]' | tee -a "$logfile" >&2
.ts(1)
ist nicht auf den von mir verwendeten Systemen installiert - FreeBSD, OSX und eine alte Ubuntu-Box. Können Sie uns sagen, was es bietet?sponge(1)
(erst nach dem Schließen von stdin in eine Datei schreiben, damit Sie nicht durch Umleitungsomething < foo | sponge foo
überladen könnenfoo
) undvipe(1)
(einen Texteditor in eine Pipe einfügen).Ein geeigneterer Weg, dies zu tun, ist
{ command; }
eher mit als(command)
. Der Grund dafür ist, dass beim Gruppieren von Befehlen mit()
einer Subshell diese Befehle ausgeführt werden und die Variablen, die während dieses Blocks initialisiert werden, anderen Abschnitten des Skripts nicht zur Verfügung stehen.Wenn wir stattdessen
{}
für die Befehlsgruppierung verwenden, werden die Befehle in derselben Shell ausgeführt, sodass die Variablen anderen Abschnitten des Skripts zur Verfügung stehen.Wenn dieser Abschnitt ausgeführt
$var
wird, behält die Variable hier ihren Wert bei, wo dies im anderen Fall nicht der Fall ist.quelle
{ command; }
.