Come utilizzare set e pipefail negli script Bash su Linux

Pubblicato: 2022-06-25
Terminale Linux sullo schermo di un laptop su sfondo blu.
fatmawati achmad zaenuri/Shutterstock.com

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.

Che cos'è Bash Shell e perché è così importante per Linux?
CORRELATI Cos'è Bash Shell e perché è così importante per Linux?

È 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 

Esecuzione di un semplice script senza errori.

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 

Esecuzione di uno script e generazione di una condizione di errore.

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 $? 

Verifica del codice di ritorno per l'ultimo script eseguito.

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 $? 

Terminare uno script in una condizione di errore e impostare correttamente il codice di ritorno.

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 $? 

I comandi integrati bash shell true e false.

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 $? 

Piping falso in vero.

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]}" 

Utilizzo di PIPESTATUS per visualizzare il codice di ritorno di tutti i programmi in una catena di pipe.

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 $? 

Esecuzione di uno script con un errore in una catena di pipe.

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 $? 

Esecuzione di uno script che intercetta gli errori nelle catene di pipe e imposta correttamente il codice di ritorno.

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 $? 

Esecuzione di uno script che non acquisisce variabili non inizializzate.

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.

Come lavorare con le variabili in Bash
CORRELATI Come lavorare con le variabili in Bash

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 $? 

Esecuzione di uno script che acquisisce variabili non inizializzate.

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 

Esecuzione di due script in cui le variabili non inizializzate vengono gestite internamente e l'opzione -u non si attiva.

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 

Esecuzione di uno script con linee di traccia -x scritte sul terminale.

Individuare i bug in questi banali script di esempio è facile. Quando inizi a scrivere script più coinvolti, queste opzioni dimostreranno il loro valore.