1
0
Bifurcation 0
miroir de https://github.com/PAPAMICA/Wiki-Tech.io.git synchronisé 2024-09-15 12:15:18 +02:00
Wiki-Tech.io/Scripting/Bash.html

529 lignes
40 Kio
HTML
Brut Vue normale Historique

2021-05-17 10:41:24 +02:00
<!--
title: Scripting - Bash
description: Les bases de l'automatisation !
published: true
2021-05-17 10:58:35 +02:00
date: 2021-05-17T08:58:33.704Z
2021-05-17 10:41:24 +02:00
tags:
editor: ckeditor
dateCreated: 2021-05-17T08:41:21.766Z
-->
<h1 style="text-align:justify;">Shell</h1>
<p style="text-align:justify;">Sous Linux, un shell est l’interpréteur de commandes qui fait office d’interface entre l'utilisateur et le système d’exploitation. Il s’agit d’interpréteurs, cela implique que chaque commande saisie par l’utilisateur et vérifiée puis exécutée. Nous avons déjà parlé des différents types de shell existants (csh, tcsh, sh, bash, ksh, dash, etc.). Nous travaillerons plus avec le bash (Bourne Again SHell).</p>
<p style="text-align:justify;"><br>Pour rappel, un shell possède deux principaux aspects :</p>
<ul>
<li style="text-align:justify;">Un aspect environnement de travail</li>
<li style="text-align:justify;">Un aspect langage de programmation</li>
</ul>
<h1 style="text-align:justify;">L’environnement</h1>
<p style="text-align:justify;">L’environnement de travail d’un Shell doit être agréable et puissant (rappel CLI GUI), bash permet entre autres choses de :</p>
<ul>
<li style="text-align:justify;">Rappeler des commandes précédentes (historique)</li>
<li style="text-align:justify;">Modifier en ligne du texte de la commande courante (bi, emacs, nano)</li>
<li style="text-align:justify;">Gestion des travaux lancés en arrières-plan (jobs)</li>
<li style="text-align:justify;">Initialisation adéquate des variables de configuration (chaîne d’appel de l’interpréteur, chemins de recherche par défaut)</li>
</ul>
<p style="text-align:justify;">Pour illustrer ça, voyons que le Shell permet d’exécuter une commande en mode interactif ou bien par l’intermédiaire de fichiers de commandes (scripts). En mode interactif, bash affiche à l’écran une chaîne d’appel (appelée prompt ou invite) qui se termine par défaut par le caractère # pour l’administrateur (root), et par le caractère $ pour les autres utilisateurs. Sous Windows le prompt est souvent le nom du chemin où l’on se trouve, suivis de &gt; , par exemple : “<strong>C:\Windows\System32&gt;</strong></p>
<h1 style="text-align:justify;">Un langage de programmation</h1>
<p style="text-align:justify;">Les shells ne sont pas uniquement des interpréteurs de commandes mais de véritables langages de programmation, un shell comme le bash intègre :</p>
<ul>
<li style="text-align:justify;">les notions de variables</li>
<li style="text-align:justify;">les opérateurs arithmétiques</li>
<li style="text-align:justify;">les structures de contrôle</li>
<li style="text-align:justify;">les fonctions</li>
<li style="text-align:justify;">etc.</li>
</ul>
<p style="text-align:justify;">Par exemple :</p>
<pre><code class="language-plaintext">$ mavariable=5 #=&gt; affectation de la valeur 5 dans la variable qui a pour nom “mavariable”
$ echo $((mavariable +3)) #=&gt; affiche la valeur de l’expression mavariable
8</code></pre>
<h2 style="text-align:justify;">Avantages et inconvénients des shells</h2>
<p style="text-align:justify;">L’étude d’un shell comme le bash dispose de plusieurs avantages :</p>
<ul>
<li style="text-align:justify;">langage interprété, il est facile de trouver les erreurs et de les traiter</li>
<li style="text-align:justify;">modification rapide sans besoin de recompiler</li>
<li style="text-align:justify;">langage orienté chaîne de caractères (pas de pointeurs), moins de risque d’avoir des erreurs d’adressage.</li>
<li style="text-align:justify;">prototypage rapide d’application, il est facile de composer des programmes avec les commandes existantes et l’utilisation des tubes et substitution dans l’environnement Unix</li>
<li style="text-align:justify;">langage “glu”: on peut connecter des composants écrits dans des langages différents. Ils doivent juste respecter certains standards :<ul>
<li style="text-align:justify;">lire sur l’entrée standard</li>
<li style="text-align:justify;">accepter des arguments et options éventuels,</li>
<li style="text-align:justify;">écrire ses résultats sur la sortie standard</li>
<li style="text-align:justify;">écrire les messages d’erreurs sur la sortie dédiée au erreur.</li>
</ul>
</li>
</ul>
<p style="text-align:justify;">Voyons maintenant certains inconvénients des shells :</p>
<ul>
<li style="text-align:justify;">La syntaxe est “ésotérique” et d’accès difficile pour les débutants</li>
<li style="text-align:justify;">Suivant le contexte, l’ajout ou l’oubli d’un espace peut provoquer des erreurs de syntaxe ou de traitement</li>
<li style="text-align:justify;">Plusieurs syntaxes pour implanter la même fonctionnalité (principalement à cause de la compatibilité ascendante avec le Bourne Shell</li>
<li style="text-align:justify;">Le sens de certains caractères spéciaux, comme les parenthèses, change avec le contexte, elle peuvent définir :<ul>
<li style="text-align:justify;">une liste de commandes</li>
<li style="text-align:justify;">une définition de fonction</li>
<li style="text-align:justify;">imposer un ordre d’évaluation</li>
<li style="text-align:justify;">une expression arithmétique</li>
</ul>
</li>
</ul>
<p style="text-align:justify;">Si vous voulez connaître le shell qui tourne actuellement, utilisez la commande ps, et si vous voulez connaître la version du bash :<br><code>bash -- version</code></p>
<h1 style="text-align:justify;">Variables</h1>
<p style="text-align:justify;">Ils existent différents types de variables utilisables dans le shell. Elles sont identifiées par un nom (suite de lettres, chiffres, caractères espace<br>ou souligné ne commençant pas par un chiffre. la casse est gérée). On peut classer 3 groupes de variables :</p>
<ul>
<li style="text-align:justify;">utilisateurs (ex: a, valeur)</li>
<li style="text-align:justify;">prédéfinies par le shell (ex; PS1, PATH, REPLY, IFS, HOME)</li>
<li style="text-align:justify;">prédéfinies pour les commandes unix (ex: TERM)</li>
</ul>
<p style="text-align:justify;">Pour affecter une variable, on peut utiliser l’opérateur = ou bien la commande interne read (pour demander une saisie utilisateur)</p>
<h2 style="text-align:justify;">Les variables locales :</h2>
<p style="text-align:justify;">Elles ne sont disponibles que dans l’instance du shell dans lesquelles elles ont été créées. (Elles ne sont pas utilisées par les commandes dans ce shell). Par défaut une variable est créée en tant que variable locale, on utilise couramment des lettres minuscules pour nommer ses variables locales.<br>Exemple : <code>mavariablelocale = 1</code></p>
<h2 style="text-align:justify;">Les constantes :</h2>
<p style="text-align:justify;">Une constante est une variable en lecture seul d’une certaine manière, elle n’a pas pour but d’être modifié dans le programme (d’où son nom). Pour créer une constante, vous pouvez utiliser la commande declare -r.<br>Exemple: <code>declare -r pi=3.14159</code></p>
<h2>Les variables d’environnement :</h2>
2021-05-17 10:58:35 +02:00
<p style="text-align:justify;">Les variables d’environnement existent dans le shell pour lequel elles sont créées, mais aussi pour toutes les commandes qui sont utilisées dans ce shell. On utilise couramment des majuscules pour nommer ses variables d’environnement.<br><strong>Exemple1 : Transformer une variable</strong></p>
<pre><code class="language-plaintext">ENVVAR=10 #Création d’une variable locale
export ENVVAR #Transforme la variable locale en variable d’environnement</code></pre>
<p style="text-align:justify;"><strong>Exemple 2: Créer une variable d’environnement</strong></p>
<pre><code class="language-plaintext">export ENVVAR2 = 11 # Première solution
declare -x ENVVAR3 = 12 # Deuxième solution
typeset -x ENVVAR4 = 13 # Troisième solution</code></pre>
<h2 style="text-align:justify;">Commandes utiles pour les variables :</h2>
2021-05-17 10:41:24 +02:00
<ul>
2021-05-17 10:58:35 +02:00
<li style="text-align:justify;"><code>echo</code> : Vous pouvez utiliser la commande echo si vous souhaitez connaître le contenu d’une variable.</li>
2021-05-17 10:41:24 +02:00
</ul>
<p style="text-align:justify;">Exemple : <code>echo $PATH</code> permettra d’afficher le contenu de la variable d’environnement PATH qui contient les chemins de fichier de<br>commande dans le shell.</p>
2021-05-17 10:58:35 +02:00
<pre><code class="language-plaintext">$ set param1 param2
$ echo $1
param1
$ set --
$ echo $1
$ #on a perdu les valeurs</code></pre>
2021-05-17 10:41:24 +02:00
<p style="text-align:justify;">Pour connaître le nombre de variables de position, il existe une variable spéciale $#</p>
<ul>
2021-05-17 10:58:35 +02:00
<li style="text-align:justify;"><code>shift</code> : permet de décaler les variables de position (sans toucher au $0)</li>
2021-05-17 10:41:24 +02:00
</ul>
2021-05-17 10:58:35 +02:00
<pre><code class="language-plaintext">$ set a b c d e f g h i j # param 1 2 3 4 5 6 7 8 9
$ echo $1 $2 $#
a b 9
$ shift 2 # variable deviennent c d e f g h i j
$ echo $1 $2 $#
c d 7</code></pre>
2021-05-17 10:41:24 +02:00
<p style="text-align:justify;">L’utilisation du shift sans argument équivaut à faire un <code>shift 1</code></p>
<ul>
2021-05-17 10:58:35 +02:00
<li style="text-align:justify;"><code>unset</code> : permet de supprimer une variable</li>
2021-05-17 10:41:24 +02:00
</ul>
2021-05-17 10:58:35 +02:00
<pre><code class="language-plaintext">$ set myvar=1
$ echo $myvar
1
$ unset myvar
$ echo $myvar
$</code></pre>
<p>&nbsp;</p>
<h1 style="text-align:justify;">Mon premier bash</h1>
<pre><code class="language-plaintext">#!/bin/bash
echo “Nom du programme : $0”
echo “Nombre d’arguments : $#”
echo “Source : $1”
echo “Destination $2”
cp $1 $2</code></pre>
<pre><code class="language-plaintext">$ chmod u+x monpremierbash.sh
$ monpremierbash.sh /etc/passwd /root/copiepasswd
Nom du programme : ./monpremierbash.sh
Nb d’arguments : 2
Source : /etc/passwd
Destination : /root/copiepasswd</code></pre>
<pre><code class="language-plaintext">$ set un deux trois quatre
$ echo $* # affiche tous les arguments
un deux trois quatre
$ echo $@ # affiche tous les arguments
un deux trois quatre
$ set un “deux trois” quatre # testons avec 3 paramètres et des guillemets
$ set “$*” # équivalent à set “un deux trois quatre”
$ echo $#
1
$ echo $1
un deux trois quatre
$ set un “deux trois” quatre # testons $@ avec 3 paramètres et des guillemets
$ set “$@” # équivalent à set “bonjour” “deux trois” “quatre”
$ echo $#
3
$ echo $2
deux trois</code></pre>
<p>Si dans un bash on souhaite supprimer les ambiguïtés d’interprétation des paramètres de position, on utilise le ${paramètre}, comme dans l’exemple suivant.</p>
<p><strong>Exemple :</strong></p>
<pre><code class="language-plaintext">$ x=bon
$ x1=jour
$ echo $x1
jour
$ echo ${x}1
bon1
$ set un deux trois quatre cinq six sept huit neuf dix onze douze
$ echo $11
un1
$ echo ${11}
onze</code></pre>
<h1 style="text-align:justify;">Indirections</h1>
<p style="text-align:justify;">Bash offre la possibilité d’obtenir la valeur d’une variable v1 dont le nom est contenu “<code>v1</code>” dans une autre variable <code>mavar</code>. Il suffit pour cela d’utiliser la syntaxe de substitution : <code>${!mavar}</code>.</p>
<p style="text-align:justify;"><strong>Exemple :</strong></p>
<pre><code class="language-plaintext">$ var=v1
$ v1=un
$ echo ${!var}
un</code></pre>
<p style="text-align:justify;">Ce mécanisme, appelé indirection, permet d’accéder de manière indirecte et par conséquent de façon plus souple, à la valeur d’un deuxième objet. Voyons un autre exemple d’utilisation :</p>
<p style="text-align:justify;"><strong>Exemple d’un fichier indir :</strong></p>
<pre><code class="language-plaintext">#!/bin/bash
agePierre=10
ageJean=20
read -p “Quel âge (Pierre ou Jean) voulez-vous connaître ? “ prenom
rep=age$prenom #construction du nom de la variable
echo ${!rep}
$ indir
Quel âge (Pierre ou Jean) voulez-vous connaître ? Pierre
10
$ indir
Quel âge (Pierre ou Jean) voulez-vous connaître ? Jean
20</code></pre>
2021-05-17 10:41:24 +02:00
<p style="text-align:justify;">Ce mécanisme s’applique également aux deux autres types de paramètres : les paramètres de position et les paramètres spéciaux ($1, $2, , ...)</p>
2021-05-17 10:58:35 +02:00
<h1 style="text-align:justify;">Résultats, Code de retour et opérateur sur les code de retour</h1>
<p style="text-align:justify;">Il ne faut pas confondre le résultat d’une commande et son code de retour : le résultat correspond à ce qui est écrit sur sa sortie standard; le code de retour indique uniquement si l’exécution de la commande s’est bien effectuée ou non. Parfois, on est intéressé uniquement par le code de retour d’une commande et non par les résultats qu’elle produit sur la sortie standard ou la sortie d’erreur.</p>
<p style="text-align:justify;"><strong>Exemple :</strong></p>
<pre><code class="language-plaintext">$ grep toto pass &gt; /dev/null 2&gt;&amp;1 #=&gt; ou bien : grep toto pass &amp;&gt;/dev/null
$ echo $?
1 #=&gt; on en déduit que la chaîne toto n’est pas présente dans pass</code></pre>
<p style="text-align:justify;">Les opérateurs <code>&amp;&amp;</code> et <code>||</code> autorisent l’exécution conditionnelle d’une commande cmd suivant la valeur qu’a pris le code de retour de la dernière commande précédemment exécutée.</p>
<p style="text-align:justify;"><strong>Exemple pour &amp;&amp; :</strong></p>
<pre><code class="language-plaintext">$ grep toto pass &gt; /dev/null 2&gt;&amp;1 #=&gt; ou bien : grep toto pass &amp;&gt;/dev/null
$ echo $?
1 #=&gt; on en déduit que la chaîne toto n’est pas présente dans pass</code></pre>
<p style="text-align:justify;">La chaîne de caractères daemon est présente dans le fichier <code>pass</code>, le code de retour renvoyé par l’exécution de <code>grep</code> est <code>0</code>; par conséquent, la commande <code>echo</code> est exécutée.</p>
<p style="text-align:justify;"><strong>Exemple pour || :</strong></p>
<pre><code class="language-plaintext">$ ls pass tutu
ls : impossible d’accéder à tutu: Aucun fichier ou dossier de ce type pass
$ rm tutu || echo tutu non effacé
rm : impossible de supprimer tutu: Aucun fichier ou dossier de ce type
tutu non effacé</code></pre>
<p>Le fichier tutu n’existant pas, la commande <code>rm tutu</code> affiche un message d’erreur et produit un code de retour différent de <code>0</code>; la commande interne <code>echo</code> est exécutée.</p>
<p><strong>Exemple combiné || et || :</strong></p>
<pre><code class="language-plaintext">$ ls pass || ls tutu || echo fin aussi
pass</code></pre>
<p style="text-align:justify;">Le code de retour <code>ls pass</code> est égal à <code>0</code> car <code>pass</code> existe, la commande<code> ls tutu</code> ne sera pas exécutée. D’autre part le code de retour de l’ensemble <code>ls pass || ls tutu</code> est le code de retour de la dernière commande exécutée, c’est-à-dire <code>0</code> (<code>ls pass</code>). Donc <code>echo</code> fini aussi n’est pas exécutée.</p>
<p style="text-align:justify;"><strong>Exemple combiné &amp;&amp; et || :</strong></p>
<pre><code class="language-plaintext">$ ls pass || ls tutu || echo suite et &amp;&amp; echo fin
pass
fin</code></pre>
<p>La commande <code>ls pass</code> a un code de retour égal à 0, donc la commande <code>ls tutu</code> ne sera pas exécutée; le code de retour de l’ensemble <code>ls pass || ls tutu</code> sera donc égal à <code>0</code>. La commande <code>echo</code> suite et n’est pas exécutée donc le code de retour de l’ensemble reste <code>0</code> <code>echo fin</code> sera donc exécutée.</p>
2021-05-17 10:41:24 +02:00
<h4 style="text-align:justify;">Boucles et structure de contrôle</h4>
<h5 style="text-align:justify;">case, structure de choix multiple</h5>
<p style="text-align:justify;">Syntaxe :</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>1</p>
<p>&nbsp;</p>
<p>case mot in</p>
<p>2</p>
<p>&nbsp;</p>
<p>[modele [ | modele] ...) suite de commandes ;; ] ...</p>
<p>3</p>
<p>&nbsp;</p>
<p>esac</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p style="text-align:justify;">Le shell va étudier la valeur de mot puis la comparer séquentiellement à chaque modèle. Quand un modèle correspond à mot, la suite de commandes associée est exécutée, terminant l’exécution de la commande interne case. Les mots case et esac sont des mots-clés; ils doivent être le premier mot d’une commande. suite de commandes doit se terminer par 2 caractères ; collés de manière à ce qu’il n’y ait pas d’ambiguïté avec l’enchaînement séquentiel de commande cmd1; cmd2; etc. Quand au modèle , il peut-être construit à l’aide de caractères et expressions génériques de bash (*, . , [], etc.). Dans ce contexte le symbole | signifiera OU. Pour indiquer le cas par défaut (si aucun des autres ne survient) on utilisera le modèle *. il doit être placé en dernier modèle. Le code de retour de la commande composée case est égal à 0, si aucun modèle n’a pu correspondre à la valeur de mot. Sinon, c’est celui de la dernière commande exécutée de suite de commandes.</p>
<p style="text-align:justify;">Exemple 1: Programme shell oui affichant OUI si l’utilisateur a saisi le caractère o ou O</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>1</p>
<p>&nbsp;</p>
<p>#!/bin/bash</p>
<p>2</p>
<p>&nbsp;</p>
<p>read -p “Entrez votre réponse : “ rep</p>
<p>3</p>
<p>&nbsp;</p>
<p>case $rep in</p>
<p>4</p>
<p>&nbsp;</p>
<p>o|O ) echo OUI ;;</p>
<p>5</p>
<p>&nbsp;</p>
<p>* ) echo Indefini</p>
<p>6</p>
<p>&nbsp;</p>
<p>esac</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p style="text-align:justify;">Exemple 2 : Programme shell nombre prenant une chaîne de caractères en argument, et qui affiche cette chaîne si elle est constituée d’une suite de chiffres. ([:digit:] , [:upper:], [:lower:], [:alnum:]</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>1</p>
<p>&nbsp;</p>
<p>#!/bin/bash</p>
<p>2</p>
<p>&nbsp;</p>
<p># on autorise l’utilisation des expressions générique</p>
<p>3</p>
<p>&nbsp;</p>
<p>shopt -s extglob</p>
<p>4</p>
<p>&nbsp;</p>
<p>case $1 in</p>
<p>5</p>
<p>&nbsp;</p>
<p>+([[:digit:]]) ) echo $1 est une suite de chiffres ;;</p>
<p>6</p>
<p>&nbsp;</p>
<p>esac</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p style="text-align:justify;">Exemple 3 : Si l’on souhaite ignorer la casse on peut modifier le flash de shopt</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>1</p>
<p>&nbsp;</p>
<p>#!/bin/bash</p>
<p>2</p>
<p>&nbsp;</p>
<p>read -p “Entrez un mot : “ mot</p>
<p>3</p>
<p>&nbsp;</p>
<p>shopt -s nocasematch</p>
<p>4</p>
<p>&nbsp;</p>
<p>case $mot in</p>
<p>5</p>
<p>&nbsp;</p>
<p>oui ) echo Vous avez écrit oui” ;;</p>
<p>6</p>
<p>&nbsp;</p>
<p>* ) echo “$mot n’est pas le mot oui” ;;</p>
<p>7</p>
<p>&nbsp;</p>
<p>esac</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<h5 style="text-align:justify;">While</h5>
<p style="text-align:justify;">La commande interne while correspond à l’itération faire - tant que’ présente dans de nombreux langage de programmation.<br>Syntaxe :</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>1</p>
<p>&nbsp;</p>
<p>while suite_cmd1</p>
<p>2</p>
<p>&nbsp;</p>
<p>do</p>
<p>3</p>
<p>&nbsp;</p>
<p>suite_cmd2</p>
<p>4</p>
<p>&nbsp;</p>
<p>done</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p style="text-align:justify;">La suite de commandes suite_cmd1 est exécutée, si son code de retour est égal à 0, alors la suite de commande suite_cmd2 est exécutée, puis suite_cmd1 est re-exécutée. Si son code de retour est différent de 0, la suite se termine. L’originalité de cette méthode est que le test ne porte pas sur une condition booléenne, mais sur le code de retour issu de l’exécution d’une suite de commandes. Une commande while, comme toutes commandes internes, peut être écrite directement sur la ligne de commande.<br>Exemple :</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>1</p>
<p>&nbsp;</p>
<p>$ while who | grep root&gt; /dev/null</p>
<p>2</p>
<p>&nbsp;</p>
<p>&gt; do</p>
<p>3</p>
<p>&nbsp;</p>
<p>&gt; echo “Utilisateur root est connecté”</p>
<p>4</p>
<p>&nbsp;</p>
<p>&gt; sleep 5</p>
<p>5</p>
<p>&nbsp;</p>
<p>&gt; done</p>
<p>6</p>
<p>&nbsp;</p>
<p>Utilisateur root est connecté</p>
<p>7</p>
<p>&nbsp;</p>
<p>Utilisateur root est connecté</p>
<p>8</p>
<p>&nbsp;</p>
<p>Utilisateur root est connecté</p>
<p>9</p>
<p>&nbsp;</p>
<p>^C</p>
<p>10</p>
<p>&nbsp;</p>
<p>$</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p style="text-align:justify;">Commande interne while est :<br>La commande interne : associée à une itération while compose rapidement un serveur (démon) rudimentaire.<br>Exemple :</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>1</p>
<p>&nbsp;</p>
<p>$ while : #=&gt; Boucle infinie</p>
<p>2</p>
<p>&nbsp;</p>
<p>&gt; do</p>
<p>3</p>
<p>&nbsp;</p>
<p>&gt; who | cut -d’ -f1 &gt; fic #=&gt; Traitement à effectuer</p>
<p>4</p>
<p>&nbsp;</p>
<p>&gt; sleep 300 #=&gt; Temporiser</p>
<p>5</p>
<p>&nbsp;</p>
<p>&gt; done &amp;</p>
<p>6</p>
<p>&nbsp;</p>
<p>[1] 1123 #=&gt; pour arrêter l’exécution kill -15 1123</p>
<p>7</p>
<p>&nbsp;</p>
<p>$</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p style="text-align:justify;">On peut parfois utilisée la commande while pour lire un fichier texte. La lecture se fait alors ligne par ligne. Pour cela, il suffit de :</p>
<ul>
<li style="text-align:justify;">placer la commande read dans suite_cmd1</li>
<li style="text-align:justify;">de placer les commandes de traitement de la ligne courante dans suite_cmd2</li>
<li style="text-align:justify;">de rediriger l’entrée standard de la commande while avec le fichier lire</li>
</ul>
<p style="text-align:justify;">Syntaxe :</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>1</p>
<p>&nbsp;</p>
<p>while read [var1 ...]</p>
<p>2</p>
<p>&nbsp;</p>
<p>do</p>
<p>3</p>
<p>&nbsp;</p>
<p>Commandes de traitements</p>
<p>4</p>
<p>&nbsp;</p>
<p>done &lt; fichier à lire</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p style="text-align:justify;">Exemple :</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>1</p>
<p>&nbsp;</p>
<p>#!/bin/bash</p>
<p>2</p>
<p>&nbsp;</p>
<p>who &gt; tmp</p>
<p>3</p>
<p>&nbsp;</p>
<p>while read nom divers</p>
<p>4</p>
<p>&nbsp;</p>
<p>do</p>
<p>5</p>
<p>&nbsp;</p>
<p>echo $nom</p>
<p>6</p>
<p>&nbsp;</p>
<p>done &lt; tmp</p>
<p>7</p>
<p>&nbsp;</p>
<p>rm tmp</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p style="text-align:justify;">Notez que l’utilisation du while pour lire un fichier n’est pas très performante. On préférera en général utiliser une suite de filtre pour obtenir les résultats voulus (cut, awk, ...)</p>
<h4 style="text-align:justify;">Modificateur de chaîne</h4>
<h5 style="text-align:justify;">Échappement</h5>
<p style="text-align:justify;">Différents caractères particuliers servent en shell pour effectuer ses propres traitements ($ pour la substitution, &gt; pour la redirection, * en joker). Pour utiliser ces caractères particuliers en tant que simple caractère, il faut les échapper en les précédant du caractère \<br>Exemple :</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>1</p>
<p>&nbsp;</p>
<p>$ ls</p>
<p>2</p>
<p>&nbsp;</p>
<p>tata toto</p>
<p>3</p>
<p>&nbsp;</p>
<p>$ echo *</p>
<p>4</p>
<p>&nbsp;</p>
<p>tata toto</p>
<p>5</p>
<p>&nbsp;</p>
<p>$ echo \*</p>
<p>6</p>
<p>&nbsp;</p>
<p>*</p>
<p>7</p>
<p>&nbsp;</p>
<p>$ echo \\</p>
<p>8</p>
<p>&nbsp;</p>
<p>\</p>
<p>9</p>
<p>&nbsp;</p>
<p>$</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p style="text-align:justify;">Autre particularité, le caractère \ peut aussi échapper les retours à la ligne. On peut donc aller à la ligne sans exécuter la commande.<br>Comme nous l’avons déjà vu, les caractères “ et permettre une protection partielle, ou total () d’une chaîne de caractères.<br>Exemple :</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<p>1</p>
<p>&nbsp;</p>
<p>$ echo “&lt; * $PWD &gt;</p>
<p>2</p>
<p>&nbsp;</p>
<p>&lt; * /root &gt;</p>
<p>3</p>
<p>&nbsp;</p>
<p>$</p>
<p>4</p>
<p>&nbsp;</p>
<p>$ echo “\”$PS2\””</p>
<p>5</p>
<p>&nbsp;</p>
<p>&gt;</p>
<p>6</p>
<p>&nbsp;</p>
<p>$ echo &lt; * $PWD “ &gt;</p>
<p>7</p>
<p>&nbsp;</p>
<p>&lt; * $PWD “ &gt;</p>
<p>8</p>
<p>&nbsp;</p>
<p>$ echo c’est lundi</p>
<p>9</p>
<p>&nbsp;</p>
<p>&gt;</p>
<p>10</p>
<p>&nbsp;</p>
<p>&gt; </p>
<p>11</p>
<p>&nbsp;</p>
<p>cest lundi</p>
<p>12</p>
<p>&nbsp;</p>
<p>$ echo c\’est lundi</p>
<p>13</p>
<p>&nbsp;</p>
<p>c’est lundi</p>
<p>14</p>
<p>&nbsp;</p>
<p>$ echo “c’est lundi”</p>
<p>15</p>
<p>&nbsp;</p>
<p>c’est lundi</p>
<p>&nbsp;</p>
<p>&nbsp;</p>
<h5 style="text-align:justify;">Chaîne de caractères longueur</h5>
<p style="text-align:justify;">Syntaxe : <code>${#paramètre}</code><br>Cette syntaxe permet d’obtenir la longueur d’une chaîne de caractères.<br>Exemple :</p><p>&nbsp;</p><p>&nbsp;</p><p>&nbsp;</p><p>1</p><p>&nbsp;</p><p>$ echo $PWD</p><p>2</p><p>&nbsp;</p><p>/root</p><p>3</p><p>&nbsp;</p><p>$ echo ${#PWD}</p><p>4</p><p>&nbsp;</p><p>5</p><p>5</p><p>&nbsp;</p><p>$ set “Bonjour à tous”</p><p>6</p><p>&nbsp;</p><p>$ echo ${#1}</p><p>7</p><p>&nbsp;</p><p>14</p><p>8</p><p>&nbsp;</p><p>$ ls &gt; /dev/null</p><p>9</p><p>&nbsp;</p><p>$</p><p>10</p><p>&nbsp;</p><p>$ echo ${#?}</p><p>11</p><p>&nbsp;</p><p>1 #=&gt; la longueur du code de retour (0) est de 1 caractère</p><p>&nbsp;</p><p>&nbsp;</p><h5 style="text-align:justify;">Chaîne de caractères modificateur</h5><p style="text-align:justify;">On peut modifier les chaîne de caractères directement :<br>Syntaxe : ${paramètre#modèle} pour supprimer la plus <strong>courte</strong> sous-chaîne à gauche<br>Exemple :</p><p>&nbsp;</p><p>&nbsp;</p><p>&nbsp;</p><p>1</p><p>&nbsp;</p><p>$ echo $PWD</p><p>2</p><p>&nbsp;</p><p>/home/christophe</p><p>3</p><p>&nbsp;</p><p>$ echo ${PWD#*/}</p><p>4</p><p>&nbsp;</p><p>home/christophe #=&gt; le premier caractère / a été supprimé</p><p>5</p><p>&nbsp;</p><p>$</p><p>6</p><p>&nbsp;</p><p>$ set “25b75b”</p><p>7</p><p>&nbsp;</p><p>$</p><p>8</p><p>&nbsp;</p><p>$ echo ${1#*b}</p><p>9</p><p>&nbsp;</p><p>75b #=&gt; Suppression de la sous-chaîne 25b</p><p>&nbsp;</p><p>&nbsp;</p><p style="text-align:justify;">Syntaxe: <code>${paramètre##modèle}</code> pour supprimer la plus <strong>longue</strong> sous-chaîne à gauche<br>Exemple :</p><p>&nbsp;</p><p>&nbsp;</p><p>&nbsp;</p><p>1</p><p>&nbsp;</p><p>$ echo $PWD</p><p>2</p><p>&nbsp;</p><p>/home/christophe</p><p>3</p><p>&nbsp;</p><p>$ echo ${PWD##*/}</p><p>4</p><p>&nbsp;</p><p>christophe #=&gt; suppression jusqu’au dernier caractère /</p><p>5</p><p>&nbsp;</p><p>$</p><p>6</p><p>&nbsp;</p><p>$ set “25b75b”</p><p>7</p><p>&nbsp;</p><p>$</p><p>8</p><p>&nbsp;</p><p>$ echo ${1##*b}</p><p>9</p><p>&nbsp;</p><p>b</p><p>&nbsp;</p><p>&nbsp;</p><p style="text-align:justify;">Pour la suppression par la droite, c’est la même chose en utilise le caractère % comme contrôle<br>Syntaxe :<br><code>${paramètre%modèle}</code> pour supprimer la plus courte sous-chaîne à droite<br><code>${paramètre%%modèle}</code> pour supprimer la plus longue sous-chaîne à droite</p><p style="text-align:justify;">On peut extraire une sous-chaîne également :<br>Syntaxe:<br><code>${paramètre:ind}</code> : extrait la valeur de paramètre de la sous-chaîne débutant à l’indice ind.<br><code>${paramètre:ind:nb}</code> : extrait nb caractères à partir de l’indice ind<br>Exemple :</p><p>&nbsp;</p><p>&nbsp;</p><p>&nbsp;</p><p>1</p><p>&nbsp;</p><p>$ lettres=”abcdefghijklmnopqrstuvwxyz”</p><p>2</p><p>&nbsp;</p><p>$</p><p>3</p><p>&nbsp;</p><p>$ echo {$lettre:20}</p><p>4</p><p>&nbsp;</p><p>uvwxyz</p><p>5</p><p>&nbsp;</p><p>$ echo {$lettre:3:4}</p><p>6</p><p>&nbsp;</p><p>defg</p><p>7</p><p>&nbsp;</p><p>$</p><p>&nbsp;</p><p>&nbsp;</p><p style="text-align:justify;">Remplacement dans une sous-chaîne<br>Syntaxe:<br><code>${paramètre/mod/ch}</code><br>bash recherchera dans paramètre la plus longue sous-chaîne satisfaisant le modèle mod puis remplacera cette sous-chaîne par<br>la chaîne ch. Seule la première sous-chaîne trouvée sera remplacée. mod peut être des caractères ou expressions génériques.<br><code>${paramètre//mod/ch}</code> :<br>Pour replacer toutes les occurrences et pas seulement la première<br><code>${paramètre/mod/}</code> :<br><code>${paramètre//mod/}</code> :<br>Supprime au lieu de remplacer<br>Exemple :</p><p>&nbsp;</p><p>&nbsp;</p><p>&nbsp;</p><p>1</p><p>&nbsp;</p><p>$ v=totito</p><p>2</p><p>&nbsp;</p><p>$ echo {$v/to/lo}</p><p>3</p><p>&nbsp;</p><p>lotito</p><p>4</p><p>&nbsp;</p><p>$ echo {$v//to/lo}</p><p>5</p><p>&nbsp;</p><p>lotilo</p><p>&nbsp;</p><p>&nbsp;</p><h4 style="text-align:justify;">Structure de contrôle for et if</h4><h5 style="text-align:justify;">Itérati