$ awk -v f=<(cmdmayfail) -e 'BEGIN { r = getline < f ; print r }'
-bash: cmdmayfail: command not found
0
Im obigen Beispiel mit der unbenannten Pipe awk
ist der Fehler der unbenannten Pipe nicht bekannt.
$ awk -v f=<(cmdmayfail || echo "control sequence" ) -e 'BEGIN { r = getline < f ; print r }'
Um awk
auf diesen Fehler aufmerksam zu machen, könnte ich den obigen Code verwenden, indem ich eine Steuersequenz und einige Fehlerinformationen sende.
Da file
bereits viele Dateitypen bekannt sind, gibt es eine sinnvolle Steuersequenz, die für diese Anwendung verwendet werden kann, sodass awk
die Fehlersteuersequenz nicht als von einer legitimen Datei stammend missbraucht wird? Danke.
Antwort1
Handelt es sich um einen Standard-Unix-Befehl, sollte es ausreichen, commandmayfail
der Steuersequenz Ihren eigenen komplexen Text voranzustellen (beispielsweise ), um den Unterschied verständlich zu machen.__ERROR__CMDFAIL__:
awk
Wenn Sie jedoch auch Ihre eigene und/oder proprietäre Software einbinden, ist es schwierig, Ihnen eine allgemeine Zeichenfolge zu geben. Es ist möglich (wenn auch unwahrscheinlich), dass einer Ihrer proprietären Befehle eine solche Zeichenfolge verwendet. Sie sollten sich den allgemeinen Aufbau der Fehlermeldungen ansehen und eine Zeichenfolge erstellen, die wahrscheinlich nicht verwendet wird.
Wenn es commandmayfail
sich file
, wie die Frage nahelegt, um handelt, kann es ausreichend sein, eine Zeichenfolge ohne zu verwenden :
.
Antwort2
Wenn die Ausgabe cmdmayfail
relativ klein ist und der Befehl selbst unabhängig von anderen Teilen Ihres Codes beendet wird, können Sie seine Ausgabe in einer Variablen speichern und den Beendigungsstatus als allererste Zeile übergeben. Der Code in <()
würde dann etwa so aussehen:
out="$(cmdmayfail)"; printf '%s\n' "$?" "$out"
Sie awk
sollten getline<f
den Beendigungsstatus abrufen. Anschließend getline<f
wird die tatsächliche Ausgabe von gelesen cmdmayfail
.
Einschränkungen:
- Variablen in Bash können keine Nullzeichen speichern.
$()
entfernt alle nachfolgenden Zeilenumbruchzeichen undprintf
fügt dann genau eines hinzu. Ein umständlicher Trick, um dies zu vermeiden:out="$(cmdmayfail; s="$?"; printf X; exit "$s")"; printf '%s\n%s' "$?" "${out%X}"
cmdmayfail
Die Tatsache, dass Sie die Ausgabe im Block verarbeiten, BEGIN
kann darauf hinweisen, cmdmayfail
dass Sie mit einer vorzeitigen Beendigung rechnen. Vielleicht reicht diese Lösung aus.
Im Allgemeinen cmdmayfail
kann es sogar „endlos“ laufen (also bis Sie es beenden) und Sie möchten möglicherweise seine Ausgabe lesen, während Sie den („endlosen“) Standard-DIN von verarbeiten awk
. In diesem Fall funktioniert die obige Lösung nicht.
Sie können jeder Zeile der Ausgabe von cmdmayfail
eine feste Statuszeile voranstellen (z. B. OK
) und schließlich eine Zeile mit dem Beendigungsstatus von hinzufügen cmdmayfail
. Der Code in <()
würde etwa so aussehen:
cmdmayfail | sed 's/^/OK\n/'; printf '%s\n' "${PIPESTATUS[0]}"
Beispiel:
$ (printf '%s\n' foo "bar baz"; exit 7) | sed 's/^/OK\n/'; printf '%s\n' "${PIPESTATUS[0]}"
OK
foo
OK
bar baz
7
Dann awk
sollte Ihr Code getline<f
prüfen, ob es so ist OK
. Wenn ja, getline<f
ist die nächste Zeile (wieder) cmdmayfail
mit Sicherheit von . Führen Sie eine Schleife durch, um alle Zeilen zu analysieren, bis kein „ OK
wenn Sie es erwarten“ erscheint. Dann ist es der Beendigungsstatus.
Dies funktioniert einwandfrei, es sei denn, cmdmayfail
es entsteht einunvollständige Zeile. Beispiel:
$ (printf 'foo\nincomplete line'; exit 22) | sed 's/^/OK\n/'; printf '%s\n' "${PIPESTATUS[0]}"
Abhängig von der Implementierung von sed
kann das Tool
- die unvollständige Zeile überhaupt ignorieren oder
- verarbeiten und das fehlende Newline-Zeichen hinzufügen, oder
- verarbeiten Sie es so wie es ist.
Tatsächlich werden Sie
- einen Teil der Ausgabe verpassen oder
- nicht wissen, dass die Zeile unvollständig war, oder
- Holen Sie sich die Zeile mit dem daran angehängten Beendigungsstatus.
Im Falle von (3) printf '\n%s\n' "${PIPESTATUS[0]}"
kann Folgendes helfen. Es wird eine zusätzliche leere Zeile generiert, wenn die letzte Zeile von abgeschlossen ist; auf diese Weise kann cmdmayfail
Ihr Code dies erkennen.awk
Stellen Sie sich einen Fall vor, in dem cmdmayfail
mitten in der Zeile zwangsweise beendet wurde. Dann möchten Sie die unvollständige Zeile möglicherweise nicht analysieren. Das Problem ist: Um zu wissen, awk
ob die Zeilenform cmdmayfail
vollständig war oder nicht, müssen Sie die nächste (Status-)Zeile testen. Die Implementierung einer nützlichen Logik hierfür awk
kann zumindest unbequem sein.
Es ist gut, eine unvollständige Zeile so schnell wie möglich zu erkennen,read
in Bash kann dies tun. Der Nachteil ist, read
dass es langsam ist (und denken Sie daran, dass Bash-Variablen keine Nullzeichen speichern können). Beispiellösung:
# define this helper function in the main shell
encerr () { ( eval "$@" ) | (while IFS= read -r line; do printf 'C\n%s\n' "$line"; done; [ -n "$line" ] && printf 'I\n%s\n' "$line") ; printf 'E\n%s\n' "${PIPESTATUS[0]}"; }
# this part you want to put in <()
encerr cmdmayfail
Dann müssen Sie das benutzerdefinierte Protokoll darin dekodieren awk
. Die Zeilen kommen paarweise. (Sehen Sie sich die Beispiele weiter unten an, um das Protokoll intuitiver zu verstehen.)
- Lesen Sie die erste Zeile eines Paars (
getline<f
). - Speichern Sie die erste Zeile in einer Variablen (
first=$0
). - Lesen Sie die zweite Zeile eines Paars (
getline<f
). - Analysieren Sie die erste Zeile (
$first
).- Wenn dies
C
der Fall ist, ist die zweite (aktuelle$0
) eine vollständige Zeile voncmdmayfail
, die Sie analysieren können. - Wenn dies der
I
Fall ist, ist die zweite Zeile eine unvollständige Zeile auscmdmayfail
. Sie können sie analysieren, müssen es aber nicht. Erwarten SieE
das nächste Paar. - Wenn dies der
E
Fall ist, ist der zweite der Beendigungsstatus voncmdmayfail
. Sie sollten keine weiteren Paare erwarten.
- Wenn dies
- Schleife.
Beachten Sie, dass ich eval "$@"
innerhalb der Funktion verwendet habe. Was Sie danach schreiben, encerr
wird zum zweiten Mal ausgewertet, daher möchten Sie normalerweise etwas wie
encerr 'cmd1 -opt foo'
oder
encerr "cmd1 -opt foo"
oder auch
encerr 'cmd1 -opt foo | cmd2'
Dies ist im Grunde das Formular, das Sie zum Ausführen von Remote-Befehlen verwenden ssh
. Vergleichen Sie:
ssh a@b 'cmd1 -opt foo | cmd2'
Oder Sie können die Funktion folgendermaßen erstellen:
encerr () { "$@" | …
und nenne es so:
encerr cmd1 -opt foo
Vergleichen mit sudo
:
sudo cmd1 -opt foo
Beispiele (unter Verwendung der Originalfunktion mit eval
):
Erfolg mit leerer Ausgabe
$ encerr true E 0
Fehler mit leerer Ausgabe
$ # I'm redirecting stderr so "command not found" doesn't obfuscate the example $ encerr nonexisting-command-foo1231234 2>/dev/null E 127
Erfolg nach kompletten Zeilen
$ encerr 'date; sleep 1; date' C Mon Sep 30 09:07:40 CEST 2019 C Mon Sep 30 09:07:41 CEST 2019 E 0
Ausfall nach kompletten Zeilen
$ encerr 'printf "foo\nbar\n"; false' C foo C bar E 1
Erfolg nach einer unvollständigen Zeile
$ encerr 'printf "foo bar\n89 baz"' C foo bar I 89 baz E 0
Fehler nach einer unvollständigen Zeile
$ encerr 'printf "\nThe first line was empty and this one was interru"; exit 33' C I The first line was empty and this one was interru E 33