tuteurs.ens.fr/unix/shell/test.tml

425 lines
11 KiB
Text
Raw Normal View History

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE html
PUBLIC "-//ENS/Tuteurs//DTD TML 1//EN"
"tuteurs://DTD/tml.dtd">
<html>
<head>
<title>Tests</title>
</head>
<body>
<h1>Tests et calcul arithmétique</h1>
<h2>Qu'est-ce qu'un test ?</h2>
<p>
<strong>Un test est une opération dont le but est d'évaluer la valeur
d'une expression</strong>. Cette expression peut être simplement
l'<em>existence</em> de quelque chose (par exemple, d'un fichier, ou
bien d'une variable), ou ce peut être une <em>proposition</em>.</p>
<p>Concrètement, cela veut dire qu'<strong>un programme n'est pas limité
à la possibilité de donner des ordres : il peut aussi poser des
questions</strong>, et agir comme vous le décidez en fonction des
réponses apportées aux questions.</p>
<p>Posons par exemple le principe suivant : « Si Dieu n'existe
pas, alors tout est permis. » Nous allons ensuite tester si Dieu
existe, et s'il n'existe pas, nous saurons que tout est permis. Autre
exemple : posons que « si le train passe sous le tunnel avant
que j'aie compté jusqu'à dix, alors Manek est vivant. » Je vais
donc tester si le train passe sous le tunnel avant que j'aie compté
jusqu'à dix, et si c'est le cas, alors c'est que Manek est vivant. C'est
aussi simple que cela.</p>
<h3>Une condition</h3>
<p>Le shell propose deux principales façons de réaliser un test ;
ces deux méthodes sont équivalentes :</p>
<ul>
<li> <code>test <em>expression</em></code></li>
<li> <code>[ <em>expression</em> ]</code></li>
</ul>
<p>
Les deux commandes suivantes reviennent donc au même :</p>
<pre>
<span class="prompt">clipper ~ $</span> <strong>test -f foo</strong> || echo &quot;Le fichier foo n'existe pas.&quot;
Le fichier foo n'existe pas.
<span class="prompt">clipper ~ $</span> <strong>[ -f foo ]</strong> || echo &quot;Le fichier foo n'existe pas.&quot;
Le fichier foo n'existe pas.
</pre>
<h3>Code de retour d'un test</h3>
<p>Un test renvoie un <em>code de retour</em>. <strong>Un code de retour
est un nombre (0 ou autre), qui correspond à une réponse de
type « vrai » ou « faux »</strong>. C'est ce code de
retour qui permet la manipulation des tests dans des structures de
contrôle comme <code>if</code>, etc. </p>
2005-09-07 23:19:24 +02:00
<p>
Le code de retour 0 correspond à la réponse
« vrai ». Pour répondre « faux », le programme
répond... autre chose (ce peut être 1, 2, -1 ou autre).
2005-09-07 23:19:24 +02:00
</p>
<p>Par exemple, un pilote de Formule 1 utilise le programme
suivant :</p>
<pre>
while [[ &quot;$couleur_du_feu&quot; != &quot;vert&quot; ]]
do attendre
done
démarrer_en_trombe
</pre>
<h2>Les opérateurs</h2>
<h3>Opérateurs logiques</h3>
<h4>Non : l'opérateur « <code>!</code> »</h4>
<p><strong>L'opérateur logique « non » inverse le code de
retour d'une commande</strong>, c'est-à-dire renvoie vrai si elle
renvoie faux, et vice versa. </p>
<p>On utilise cet opérateur en précédant une condition d'un point
d'exclamation (« ! »). </p>
<h5>Comparaison de plusieurs combinaisons d'opérateurs</h5>
<p>Pour illustrer l'usage de cet opérateur, voici quatre cas de figure
différents :</p>
<pre>
# Premier cas
[ <strong>-f</strong> foo ] <strong>&amp;&amp;</strong> echo &quot;Le fichier foo existe.&quot;
</pre>
<p class="continue">Dans l'exemple précédent, le shell teste si le
fichier <code>foo</code> existe. Comme il n'existe pas, le code de
retour de ce test est 0. Or, l'opérateur « &amp;&amp; »
n'exécute ce qui suit que si le code de retour est 1. Comme ce
n'est pas le cas, les commandes ne sont pas exécutées.</p>
<pre>
# Deuxième cas
[ <strong>-f</strong> foo ] <strong>||</strong> echo &quot;Le fichier foo n'existe pas.&quot;
Le fichier foo n'existe pas.
</pre>
<p class="continue">Dans cet exemple, l'opérateur n'est plus
« &amp;&amp; » mais « || ». Les commandes ne
s'exécutent que si le code de retour vaut 0 ; comme c'est le cas,
elles sont exécutées.</p>
<pre>
# Troisième cas
2005-09-08 01:00:03 +02:00
[ <strong>! -f</strong> foo ] <strong>&amp;&amp;</strong> echo &quot;Le fichier foo n'existe pas.&quot;
Le fichier foo n'existe pas.
</pre>
<p class="continue">Ici, l'opérateur est de nouveau
« &amp;&amp; » ; mais contrairement aux deux exemples
précédents, le test n'est plus <code>[ -f foo ]</code>
mais <code>[ ! -f foo ]</code>. Par conséquent, le
code de retour est 1, et les commandes sont exécutées.</p>
<pre>
# Quatrième cas
2005-09-08 01:00:03 +02:00
[ <strong>! -f</strong> foo ] <strong>||</strong> echo &quot;Le fichier foo existe.&quot;
</pre>
<p class="continue">Voici la dernière combinaison possible. Le code de
retour est 1, mais il fallait 0 pour que les commandes soient
exécutées. </p>
<p>Ces quatre exemples correspondent aux énoncés suivants :</p>
<ul>
<li> <em>s'il est vrai que <code>foo</code> existe</em>, alors il faut
écrire : « Le fichier foo existe. ».</li>
<li> <em>s'il n'est pas vrai que <code>foo</code> existe</em>, alors il
faut écrire : « Le fichier foo n'existe pas. ».</li>
<li> <em>s'il est vrai que <code>foo</code> n'existe pas</em>, alors il
faut écrire : « Le fichier foo n'existe pas. ».</li>
<li> <em>s'il n'est pas vrai que <code>foo</code> n'existe pas</em>,
alors il faut écrire : « Le fichier foo existe. ».</li>
</ul>
<h5>Reformulation avec <code>if</code></h5>
<p>Dans un script, outre la formulation précédente, on pourra
écrire :</p>
<pre>
# Premier cas
if [ -f foo ]
then echo &quot;Le fichier foo existe.&quot;
else continue
fi
# Deuxième cas
if [ -f foo ]
then continue
else echo &quot;Le fichier foo n'existe pas.&quot;
fi
# Troisième cas
if [ ! -f foo ]
then echo &quot;Le fichier foo n'existe pas.&quot;
else continue
fi
# Quatrième cas
if [ ! -f foo ]
then continue
else echo &quot;Le fichier foo existe.&quot;
fi
</pre>
<h4>Et : l'opérateur « <code>-a</code> »</h4>
<p>L'opérateur « et » renvoie 1 (vrai) si et seulement si
les différentes conditions sont toutes réalisées ; si au moins
l'une d'entre elles ne l'est pas, le code de retour est 0
(faux). On note cet opérateur en insérant « -a » entre les
différentes conditions. Exemples : </p>
<pre>
touch foo # donc foo existe
rm bar # donc bar n'existe pas
# [ -f foo ] = vrai si le fichier foo existe
# [ ! -f bar ] = vrai si bar n'existe pas
# à n'exécuter que si foo existe ET que bar n'existe pas.
[ -f foo <strong>-a</strong> ! -f bar ] &amp;&amp;
mv foo bar
</pre>
<p class="continue">Autres formulations possibles :
</p>
<pre>
test -f foo -a ! -f bar
[ -f foo ] -a [ ! -f bar ]
[[ -f foo &amp;&amp; ! -f bar ]]
</pre>
<p><strong>Si vous débutez, vous n'êtes pas tenu de retenir par
c&oelig;ur toutes les combinaisons possibles. Sachez simplement les
reconnaître</strong> si vous les lisez quelque part ; et pour vos
propres scripts, vous il vous suffit de savoir bien manipuler la syntaxe
qui vous paraît la plus lisible.</p>
<h4>Ou : l'opérateur « <code>-o</code> »</h4>
<p>Pour réaliser la condition de l'opérateur « ou », il suffit
qu'une seule des conditions qu'il rassemble soit vraie : </p>
<ul>
<li> si <em>toutes</em> les conditions sont rassemblées, la condition
d'ensemble l'est aussi ;</li>
<li> si <em>une partie seulement</em> de ces conditions est rassemblée,
la condition d'ensemble l'est aussi ;</li>
<li> si <em>aucune</em> des conditions incluses n'est remplie, la
condition d'ensemble ne l'est pas non plus.</li>
</ul>
<p>Exemple :</p>
<pre>
if [[ &quot;$fichier&quot; == &quot;fichier_interdit&quot; -o ! -f &quot;$fichier&quot; ]]
then echo &quot;Je ne veux pas lire $fichier ou bien il n'existe pas.&quot;
fi
</pre>
<ul>
<li> si <code>$fichier</code> vaut
<code>&quot;fichier_interdit&quot;</code>, il n'est pas lu ;</li>
<li> si <code>$fichier</code> n'existe pas, il n'est pas lu ;</li>
<li> si <code>$fichier</code> vaut
<code>&quot;fichier_interdit&quot;</code> et qu'en plus il n'existe pas,
il n'est pas lu ;</li>
<li> si <code>$fichier</code> ne vaut pas
<code>&quot;fichier_interdit&quot;</code> et qu'il existe bien, il est
lu.</li> </ul>
<h3>Opérateurs arithmétiques</h3>
<p>
Le shell permet d'opérer des calculs arithmétiques, même s'il est moins
puissant que d'autres langages (Perl, Scheme, C, etc.) pour cela.
</p>
<p>
Les opérateurs sont les suivants :
</p>
<ul>
<li> <code>-eq</code> (<i>equal</i>) : « égal à » (signe
« = ») ;</li>
<li> <code>-ne</code> (<i>not equal</i>) : « différent
de » (signe « &ne; ») ;</li>
<li> <code>-gt</code> (<i>greater than</i>) : « strictement
supérieur à » (signe « &gt; ») ;</li>
<li> <code>-lt</code> (<i>lesser than</i>) : « strictement
inférieur à » (signe « &lt; ») ;</li>
<li> <code>-ge</code> (<i>greater or equal</i>) : « supérieur
ou égal à » (signe « &ge; ») ;</li>
<li> <code>-le</code> (<i>lesser or equal</i>) : « inférieur
ou égal à » (signe « &le; ») ;</li>
</ul>
<p>
On utilise ces opérateurs entre deux nombres ou variables
numériques. Par exemple :
</p>
<pre>
#!/bin/sh
if test 2 -lt 3
then echo &quot;C'est normal.&quot;
fi
if test 2 -gt 3
then echo &quot;C'est absurde.&quot;
fi
petit=2
grand=3
if test $petit -ne 3
then echo &quot;C'est normal.&quot;
fi
if test 2 -eq $grand
then echo &quot;C'est absurde.&quot;
fi
</pre>
<p class="continue">
Si vous exécutez ce programme, vous obtenez :
</p>
<pre>
C'est normal.
C'est normal.
</pre>
<h3>Opérateurs sur les fichiers</h3>
<p>
Une grande partie de la puissance du shell se déploie dans sa faculté de
manipuler des fichiers.
</p>
<p>
Les principaux opérateurs disponibles sont :
</p>
<ul>
<li> nature du fichier :
<ul>
<li> <code>-e</code> (<i>exists</i>) : vérifie l'existence d'un
fichier ;</li>
<li> <code>-f</code> (<i>file</i>) : vérifie l'existence d'un
fichier, et le fait qu'il s'agisse bien d'un fichier au sens
strict ;</li>
<li> <code>-d</code> (<i>directory</i>) : vérifie l'existence d'un
répertoire ;</li>
<li> <code>-L</code> (<i>link</i>) : vérifie si le fichier est un
lien symbolique ;</li>
</ul></li>
<li>attributs du fichier :
<ul>
<li> <code>-s</code> (<i>size</i>) : vérifie qu'un fichier n'est
pas vide ;</li>
</ul></li>
<li>droits sur le fichier :
<ul>
<li> <code>-r</code> (<i>readable</i>) : vérifie si un fichier peut
être lu ;</li>
<li> <code>-w</code> (<i>writable</i>) : vérifie si un fichier peut
être écrit ou modifié ;</li>
<li> <code>-x</code> (<i>writable</i>) : vérifie si un fichier peut
être exécuté ;</li>
</ul></li>
<li>comparaison de fichiers :
<ul>
<li> <code>-nt</code> (<i>newer than</i>) : vérifie si un fichier
est plus récent qu'un autre ;</li>
<li> <code>-ot</code> (<i>older than</i>) : vérifie si un fichier
est plus ancien qu'un autre.</li>
</ul>
</li>
</ul>
<p>Exemple :</p>
<pre>
#!/bin/sh
if test -e ~/.emacs
then echo &quot;~/.emacs existe.&quot;
else echo &quot;~/.emacs n'existe pas.&quot;
fi
if test -d ~/.emacs
then echo &quot;~/.emacs est un répertoire.&quot;
else echo &quot;~/.emacs n'est pas un répertoire.&quot;
fi
if test -f ~/.emacs
then echo &quot;~/.emacs est un fichier.&quot;
else echo &quot;~/.emacs n'est pas un fichier.&quot;
fi
if test ~/.vimrc -nt ~/.emacs
then &quot;~/.vimrc est plus récent que ~/.emacs.&quot;
fi
</pre>
<div class="metainformation">
Auteur : Baptiste Mélès.
<date value="from git" />
</div>
</body>
</html>