Come utilizzare set e pipefail negli script Bash su Linux
Pubblicato: 2022-06-25 I comandi Linux set
e pipefail
determinano cosa succede quando si verifica un errore in uno script Bash. C'è altro a cui pensare che dovrebbe fermarsi o dovrebbe andare avanti.
CORRELATI: La guida per principianti allo scripting di shell: le basi
Script Bash e condizioni di errore
Gli script della shell Bash sono fantastici. Sono veloci da scrivere e non hanno bisogno di essere compilati. Qualsiasi azione ripetitiva o in più fasi che devi eseguire può essere racchiusa in un comodo script. E poiché gli script possono chiamare qualsiasi utilità standard di Linux, non sei limitato alle capacità del linguaggio della shell stesso.
Ma possono sorgere problemi quando si chiama un'utilità o un programma esterno. Se fallisce, l'utilità esterna si chiuderà e invierà un codice di ritorno alla shell e potrebbe persino stampare un messaggio di errore sul terminale. Ma il tuo script continuerà l'elaborazione. Forse non è quello che volevi. Se si verifica un errore all'inizio dell'esecuzione dello script, potrebbero verificarsi problemi peggiori se il resto dello script può essere eseguito.
È possibile controllare il codice di ritorno da ogni processo esterno mentre viene completato, ma diventa difficile quando i processi vengono reindirizzati ad altri processi. Il codice di ritorno proverrà dal processo alla fine della pipe, non da quello nel mezzo che ha avuto esito negativo. Naturalmente, possono verificarsi anche errori all'interno dello script, come il tentativo di accedere a una variabile non inizializzata.
I comandi set
e pipefile
ti consentono di decidere cosa succede quando si verificano errori come questi. Consentono inoltre di rilevare gli errori anche quando si verificano nel mezzo di una catena di tubi.
Ecco come usarli.
Dimostrare il problema
Ecco una banale sceneggiatura di Bash. Fa eco a due righe di testo al terminale. Puoi eseguire questo script se copi il testo in un editor e lo salvi come "script-1.sh".
#!/bin/bash eco Questo accadrà prima echo Questo accadrà in secondo luogo
Per renderlo eseguibile dovrai usare chmod
:
chmod +x script-1.sh
Dovrai eseguire quel comando su ogni script se vuoi eseguirli sul tuo computer. Eseguiamo lo script:
./script-1.sh
Le due righe di testo vengono inviate alla finestra del terminale come previsto.
Modifichiamo leggermente lo script. Chiederemo a ls
di elencare i dettagli di un file che non esiste. Questo fallirà. Lo abbiamo salvato come "script-2.sh" e lo abbiamo reso eseguibile.
#!/bin/bash eco Questo accadrà prima ls nome-file-immaginario echo Questo accadrà in secondo luogo
Quando eseguiamo questo script, vediamo il messaggio di errore da ls
.
./script-2.sh
Sebbene il comando ls
non sia riuscito, lo script ha continuato a essere eseguito. E anche se si è verificato un errore durante l'esecuzione dello script, il codice di ritorno dallo script alla shell è zero, il che indica il successo. Possiamo verificarlo usando echo e $?
variabile che contiene l'ultimo codice di ritorno inviato alla shell.
eco $?
Lo zero che viene segnalato è il codice di ritorno dalla seconda eco nello script. Quindi ci sono due problemi con questo scenario. Il primo è che lo script ha avuto un errore ma ha continuato a funzionare. Ciò può portare ad altri problemi se il resto dello script si aspetta o dipende dall'azione che non è riuscita in realtà è riuscita. E il secondo è che se un altro script o processo deve verificare il successo o il fallimento di questo script, otterrà una lettura falsa.
L'opzione set -e
L'opzione set -e
(exit) fa uscire uno script se uno qualsiasi dei processi che chiama genera un codice di ritorno diverso da zero. Qualsiasi cosa diversa da zero è considerata un fallimento.
Aggiungendo l'opzione set -e
all'inizio dello script, possiamo cambiarne il comportamento. Questo è "script-3.sh".
#!/bin/bash impostare -e eco Questo accadrà prima ls nome-file-immaginario echo Questo accadrà in secondo luogo
Se eseguiamo questo script vedremo l'effetto di set -e
.
./script-3.sh
eco $?
Lo script viene interrotto e il codice di ritorno inviato alla shell è un valore diverso da zero.
Gestire i guasti nei tubi
Le tubazioni aggiungono più complessità al problema. Il codice di ritorno che esce da una sequenza di comandi convogliata è il codice di ritorno dell'ultimo comando della catena. Se c'è un errore con un comando nel mezzo della catena, torniamo al punto di partenza. Il codice di ritorno viene perso e lo script continuerà l'elaborazione.
Possiamo vedere gli effetti dei comandi di piping con codici di ritorno diversi usando le shell integrate true
e false
. Questi due comandi non fanno altro che generare un codice di ritorno rispettivamente di zero o uno.
VERO
eco $?
falso
eco $?
Se convogliamo false
in true
, con false
che rappresentano un processo in errore, otteniamo il codice di ritorno di true
pari a zero.
falso | VERO
eco $?
Bash ha una variabile array chiamata PIPESTATUS
, che acquisisce tutti i codici di ritorno da ciascun programma nella catena di pipe.
falso | vero | falso | VERO
echo "${PIPESTATUS[0]} ${PIPESTATUS[1]} ${PIPESTATUS[2]} ${PIPESTATUS[3]}"
PIPESTATUS
conserva i codici di ritorno solo fino all'esecuzione del programma successivo e il tentativo di determinare quale codice di ritorno va con quale programma può diventare molto disordinato molto rapidamente.
È qui che entrano in set -o
(opzioni) e pipefail
. Questo è "script-4.sh". Questo proverà a reindirizzare il contenuto di un file che non esiste in wc
.
#!/bin/bash impostare -e eco Questo accadrà prima gatto script-99.sh | wc -l echo Questo accadrà in secondo luogo
Questo fallisce, come ci aspetteremmo.
./script-4.sh
eco $?
Il primo zero è l'output di wc
, che ci dice che non ha letto alcuna riga per il file mancante. Il secondo zero è il codice di ritorno dal secondo comando echo
.
Aggiungeremo -o pipefail
, lo salveremo come "script-5.sh" e lo renderemo eseguibile.
#!/bin/bash set -eo pipefail eco Questo accadrà prima gatto script-99.sh | wc -l echo Questo accadrà in secondo luogo
Eseguiamolo e controlliamo il codice di ritorno.
./script-5.sh
eco $?
Lo script si interrompe e il secondo comando echo
non viene eseguito. Il codice di ritorno inviato alla shell è uno, indicando correttamente un errore.
CORRELATI: Come utilizzare il comando Echo su Linux
Cattura di variabili non inizializzate
Le variabili non inizializzate possono essere difficili da individuare in uno script del mondo reale. Se proviamo a fare echo
al valore di una variabile non inizializzata, echo
stampa semplicemente una riga vuota. Non genera un messaggio di errore. Il resto dello script continuerà a essere eseguito.
Questo è script-6.sh.
#!/bin/bash set -eo pipefail echo "$notset" echo "Un altro comando eco"
Lo eseguiremo e osserveremo il suo comportamento.
./script-6.sh
eco $?
Le istruzioni di script sulla variabile non inizializzata e continua a essere eseguita. Il codice di ritorno è zero. Cercare di trovare un errore come questo in uno script molto lungo e complicato può essere molto difficile.
Possiamo intercettare questo tipo di errore usando l'opzione set -u
(unset). Lo aggiungeremo alla nostra crescente raccolta di opzioni impostate nella parte superiore dello script, lo salveremo come "script-7.sh" e lo renderemo eseguibile.
#!/bin/bash set -eou pipefail echo "$notset" echo "Un altro comando eco"
Eseguiamo lo script:
./script-7.sh
eco $?
La variabile non inizializzata viene rilevata, lo script si interrompe e il codice di ritorno viene impostato su uno.
L'opzione -u
(unset) è abbastanza intelligente da non essere attivata da situazioni in cui è possibile interagire legittimamente con una variabile non inizializzata.
In “script-8.sh”, lo script controlla se la variabile New_Var
è inizializzata o meno. Non vuoi che lo script si fermi qui, in uno script del mondo reale eseguirai ulteriori elaborazioni e affronterai la situazione da solo.
Nota che abbiamo aggiunto l'opzione -u
come seconda opzione nell'istruzione set. L'opzione -o pipefail
deve essere l'ultima.
#!/bin/bash set -eu pipefail if [ -z "${Nuova_Varia:-}"]; poi echo "New_Var non ha alcun valore assegnato." fi
In "script-9.sh", viene verificata la variabile non inizializzata e, se non è inizializzata, viene invece fornito un valore predefinito.
#!/bin/bash set -eu pipefail valore_predefinito=484 Valore=${Nuova_Var:-$valore_predefinito} echo "Nuova_Var=$Valore"
Gli script possono essere eseguiti fino al loro completamento.
./script-8.sh
./script-9.sh
Sigillato con ascia
Un'altra opzione utile da usare è l'opzione set -x
(esegui e stampa). Quando stai scrivendo script, questo può essere un vero toccasana. stampa i comandi ei relativi parametri man mano che vengono eseguiti.
Ti dà una forma di traccia di esecuzione rapida e "pronta". Isolare i difetti logici e individuare i bug diventa molto, molto più semplice.
Aggiungeremo l'opzione set -x a "script-8.sh", la salveremo come "script-10.sh" e la renderemo eseguibile.
#!/bin/bash set -euxo pipefail if [ -z "${Nuova_Var:-}"]; poi echo "New_Var non ha alcun valore assegnato." fi
Eseguilo per vedere le linee di traccia.
./script-10.sh
Individuare i bug in questi banali script di esempio è facile. Quando inizi a scrivere script più coinvolti, queste opzioni dimostreranno il loro valore.