Comment utiliser set et pipefail dans les scripts Bash sous Linux
Publié: 2022-06-25 Les commandes Linux set
et pipefail
dictent ce qui se passe lorsqu'un échec se produit dans un script Bash. Il y a plus à penser que cela devrait-il s'arrêter ou devrait-il continuer.
CONNEXION : Le guide du débutant pour les scripts Shell : les bases
Scripts bash et conditions d'erreur
Les scripts shell bash sont excellents. Ils sont rapides à écrire et n'ont pas besoin d'être compilés. Toute action répétitive ou en plusieurs étapes que vous devez effectuer peut être intégrée dans un script pratique. Et comme les scripts peuvent appeler n'importe quel utilitaire Linux standard, vous n'êtes pas limité aux capacités du langage shell lui-même.
Mais des problèmes peuvent survenir lorsque vous appelez un utilitaire ou un programme externe. S'il échoue, l'utilitaire externe se fermera et enverra un code de retour au shell, et il pourrait même imprimer un message d'erreur au terminal. Mais votre script continuera à être traité. Ce n'est peut-être pas ce que vous vouliez. Si une erreur se produit au début de l'exécution du script, cela peut aggraver les problèmes si le reste du script est autorisé à s'exécuter.
Vous pouvez vérifier le code de retour de chaque processus externe à mesure qu'il se termine, mais cela devient difficile lorsque les processus sont redirigés vers d'autres processus. Le code de retour proviendra du processus à la fin du tube, et non de celui du milieu qui a échoué. Bien sûr, des erreurs peuvent également se produire à l'intérieur de votre script, comme essayer d'accéder à une variable non initialisée.
Les commandes set
et pipefile
vous permettent de décider ce qui se passe lorsque des erreurs comme celles-ci se produisent. Ils vous permettent également de détecter les erreurs même lorsqu'elles se produisent au milieu d'une chaîne de tuyaux.
Voici comment les utiliser.
Démonstration du problème
Voici un script Bash trivial. Il renvoie deux lignes de texte au terminal. Vous pouvez exécuter ce script si vous copiez le texte dans un éditeur et l'enregistrez sous « script-1.sh ».
#!/bin/bash echo Cela arrivera en premier echo Cela arrivera en second
Pour le rendre exécutable, vous devrez utiliser chmod
:
chmod +x script-1.sh
Vous devrez exécuter cette commande sur chaque script si vous souhaitez les exécuter sur votre ordinateur. Exécutons le script :
./script-1.sh
Les deux lignes de texte sont envoyées à la fenêtre du terminal comme prévu.
Modifions légèrement le script. Nous demanderons à ls
de lister les détails d'un fichier qui n'existe pas. Cela échouera. Nous l'avons enregistré sous le nom de "script-2.sh" et l'avons rendu exécutable.
#!/bin/bash echo Cela arrivera en premier ls nom_fichier_imaginaire echo Cela arrivera en second
Lorsque nous exécutons ce script, nous voyons le message d'erreur de ls
.
./script-2.sh
Bien que la commande ls
ait échoué, le script a continué à s'exécuter. Et même s'il y a eu une erreur lors de l'exécution du script, le code de retour du script au shell est zéro, ce qui indique un succès. Nous pouvons vérifier cela en utilisant echo et le $?
variable qui contient le dernier code de retour envoyé au shell.
echo $?
Le zéro qui est signalé est le code de retour du deuxième écho du script. Il y a donc deux problèmes avec ce scénario. Le premier est que le script a eu une erreur mais qu'il a continué à s'exécuter. Cela peut entraîner d'autres problèmes si le reste du script attend ou dépend de l'action qui a échoué et qui a effectivement réussi. Et la seconde est que si un autre script ou processus a besoin de vérifier le succès ou l'échec de ce script, il obtiendra une fausse lecture.
L'option set -e
L'option set -e
(exit) provoque la fermeture d'un script si l'un des processus qu'il appelle génère un code de retour différent de zéro. Tout ce qui n'est pas nul est considéré comme un échec.
En ajoutant l'option set -e
au début du script, nous pouvons modifier son comportement. Il s'agit de "script-3.sh".
#!/bin/bash ensemble -e echo Cela arrivera en premier ls nom_fichier_imaginaire echo Cela arrivera en second
Si nous exécutons ce script, nous verrons l'effet de set -e
.
./script-3.sh
echo $?
Le script est arrêté et le code de retour envoyé au shell est une valeur non nulle.
Faire face aux défaillances dans les tuyaux
La tuyauterie ajoute plus de complexité au problème. Le code de retour qui sort d'une séquence de commandes canalisée est le code de retour de la dernière commande de la chaîne. S'il y a un échec avec une commande au milieu de la chaîne, nous revenons à la case départ. Ce code de retour est perdu et le script poursuivra le traitement.
Nous pouvons voir les effets des commandes de canalisation avec différents codes de retour à l'aide des fonctions intégrées true
et false
du shell. Ces deux commandes ne font que générer un code de retour de zéro ou un, respectivement.
vrai
echo $?
faux
echo $?
Si nous dirigeons false
vers true
- avec false
représentant un processus défaillant - nous obtenons le code de retour de true
égal à zéro.
faux | vrai
echo $?
Bash a une variable de tableau appelée PIPESTATUS
, et cela capture tous les codes de retour de chaque programme de la chaîne de tubes.
faux | vrai | faux | vrai
echo "${PIPESTATUS[0]} ${PIPESTATUS[1]} ${PIPESTATUS[2]} ${PIPESTATUS[3]}"
PIPESTATUS
ne contient que les codes de retour jusqu'à ce que le programme suivant s'exécute, et essayer de déterminer quel code de retour va avec quel programme peut devenir très désordonné très rapidement.
C'est là qu'interviennent set -o
(options) et pipefail
. C'est "script-4.sh". Cela essaiera de diriger le contenu d'un fichier qui n'existe pas dans wc
.
#!/bin/bash ensemble -e echo Cela arrivera en premier chat script-99.sh | wc -l echo Cela arrivera en second
Cela échoue, comme on pouvait s'y attendre.
./script-4.sh
echo $?
Le premier zéro est la sortie de wc
, nous indiquant qu'il n'a lu aucune ligne pour le fichier manquant. Le deuxième zéro est le code de retour de la deuxième commande echo
.
Nous allons ajouter le -o pipefail
, l'enregistrer sous "script-5.sh" et le rendre exécutable.
#!/bin/bash set -eo pipefail echo Cela arrivera en premier chat script-99.sh | wc -l echo Cela arrivera en second
Exécutons cela et vérifions le code de retour.
./script-5.sh
echo $?
Le script s'arrête et la deuxième commande echo
n'est pas exécutée. Le code de retour envoyé au shell est un, indiquant correctement un échec.
CONNEXION: Comment utiliser la commande Echo sous Linux
Attraper des variables non initialisées
Les variables non initialisées peuvent être difficiles à repérer dans un script réel. Si nous essayons d' echo
la valeur d'une variable non initialisée, echo
imprime simplement une ligne vide. Il ne génère pas de message d'erreur. Le reste du script continuera à s'exécuter.
Il s'agit de script-6.sh.
#!/bin/bash set -eo pipefail echo "$notset" echo "Une autre commande d'écho"
Nous allons l'exécuter et observer son comportement.
./script-6.sh
echo $?
Le script parcourt la variable non initialisée et continue de s'exécuter. Le code de retour est zéro. Essayer de trouver une erreur comme celle-ci dans un script très long et compliqué peut être très difficile.
Nous pouvons intercepter ce type d'erreur en utilisant l'option set -u
(unset). Nous allons l'ajouter à notre collection croissante d'options set en haut du script, l'enregistrer sous « script-7.sh » et le rendre exécutable.
#!/bin/bash set -eou pipefail echo "$notset" echo "Une autre commande d'écho"
Exécutons le script :
./script-7.sh
echo $?
La variable non initialisée est détectée, le script s'arrête et le code de retour est défini sur un.
L'option -u
(unset) est suffisamment intelligente pour ne pas être déclenchée par des situations où vous pouvez légitimement interagir avec une variable non initialisée.
Dans "script-8.sh", le script vérifie si la variable New_Var
est initialisée ou non. Vous ne voulez pas que le script s'arrête ici, dans un script du monde réel, vous effectuerez un traitement supplémentaire et gérerez la situation vous-même.
Notez que nous avons ajouté l'option -u
comme deuxième option dans l'instruction set. L'option -o pipefail
doit venir en dernier.
#!/bin/bash set -euo pipefail if [ -z "${New_Var :-}" ] ; alors echo "New_Var n'a pas de valeur assignée." Fi
Dans "script-9.sh", la variable non initialisée est testée et si elle n'est pas initialisée, une valeur par défaut est fournie à la place.
#!/bin/bash set -euo pipefail default_value=484 Valeur=${New_Var :-$default_value} echo "Nouvelle_Var=$Valeur"
Les scripts sont autorisés à s'exécuter jusqu'à leur achèvement.
./script-8.sh
./script-9.sh
Scellé à la hache
Une autre option pratique à utiliser est l'option set -x
(exécuter et imprimer). Lorsque vous écrivez des scripts, cela peut vous sauver la vie. il imprime les commandes et leurs paramètres au fur et à mesure de leur exécution.
Il vous donne une forme de trace d'exécution rapide et prête à l'emploi. Isoler les failles logiques et repérer les bogues devient beaucoup, beaucoup plus facile.
Nous allons ajouter l'option set -x à "script-8.sh", l'enregistrer sous "script-10.sh" et le rendre exécutable.
#!/bin/bash set -euxo pipefail if [ -z "${New_Var :-}" ] ; alors echo "New_Var n'a pas de valeur assignée." Fi
Exécutez-le pour voir les lignes de trace.
./script-10.sh
Repérer les bogues dans ces exemples de scripts triviaux est facile. Lorsque vous commencerez à écrire des scripts plus complexes, ces options feront leurs preuves.