Vous n'êtes pas identifié(e).
L'icône rouge permet de télécharger chaque page du wiki visitée au format PDF et la grise au format ODT →
Ceci est une ancienne révision du document !
Voici une étude du Bash qui prend comme entrée les différents caractères.
Plus précisément, j'ai suivi la page man bash et j'ai tenté d'en expliquer et d'en illustrer le détail par des exemples.
Cette étude est développée sur huit wiki (voir la liste ci-dessous), elle se veut à la fois progressive et synthétique.
Si vous êtes débutant, suivez la progression, si vous cherchez un simple rappel, consultez les tableaux récapitulatifs de chacune des parties.
Hypathie
Enfin, les liens en gras renvoient aux pages du wiki relatives au shell, terminal, chemin relatif et absolu, scripts, alias, etc.
Dans ce tuto :
Yep ! C'est parti !
Le shell (« coquille » en anglais) est également appelé interpréteur de commandes.
C'est une couche logicielle qui fournit une l'interface utilisateur à système d'exploitation. Il correspond à la couche la plus externe de ce dernier. Il se présente sous la forme d'une interface en ligne de commande accessible depuis la console ou un terminal. L'utilisateur lance des commandes sous forme d'une entrée texte exécutée ensuite par le shell.
Quand on installe Debian Stable, le shell utilisateur par défaut est le shell Bash2).
La commande env
permet de lister les variables d'environnement du contexte qui l'exécute.
env
Le retour est copieux ! Mais on trouve dans la liste :
SHELL=/bin/bash
Ce qui signifie que le nom de l'interpréteur de commande est bash.
Plus directement, on peut faire :
echo $SHELL
/bin/bash
Pour connaître la version de votre shell Bash, tapez :
bash --version
GNU bash, version 3.2.39(1)-release (i486-pc-linux-gnu) Copyright (C) 2007 Free Software Foundation, Inc.
Avant d'aller plus loin :
Distinguons maintenant les commandes internes et les commandes externes au shell bash, commandes simples et les commandes composées.
Une commande interne est une commande dont le code est implémenté au sein même du shell. Les commandes sont intégrées, soit pour des raisons de performances (l'appel d'une telle commande ne crée pas de processus fils du shell courant); soit parce qu'une commande intégrée se sert des variables internes du shell.
Cela signifie que lorsqu'on change de shell courant (par exemple bash, dash, zsh ou C-shell3), on ne dispose plus des mêmes commandes internes.
Néanmoins, les commandes courantes qui sont essentielles à l'utilisateur, se retrouvent sous les différents shell des distributions Linux (le tronc commun standardisé respectant en général la norme POSIX).
help
help nom_commande
Une commande externe est une commande dont le code se trouve dans un fichier exécutable séparé.
whereis nom_commande
which nom_commande
Le shell crée un processus pour exécuter une commande externe. Parmi les commandes externes que l'on trouve dans un système, il y a les exécutables ELF (ex. ls, mkdir, vi, sleep) et les fichiers de scripts (dont par exemple les scripts shell).
La localisation du code d'une commande externe doit être connu du shell pour qu'il puisse exécuter cette commande. A cette fin, bash utilise la valeur de sa variable prédéfinie PATH.
type cd
cd est une primitive du shell
type cp
cp est /bin/cp
type sleep
sleep est /bin/sleep
/bin/commande
signifie donc que c'est une commande externe.
type ls
ls est un alias vers « ls --color=auto »
whereis ls
ls: /bin/ls /usr/share/man/man1/ls.1.gz
which ls
/bin/ls
ls
est donc l'alias de la commande externe /bin/ls
Les commandes simples peuvent être des commandes internes ou des commandes externes.
cat, chgrp, chmod, chown, cp, date, dd, df, dmesg, echo, ed, export, false, kill, ln, login, ls, mkdir, mknod, more, mount, mv, ps, pwd, rm, rmdir, sed, setserial, sh, stty, su, sync, true, umount, uname.
(Consultez la documentation pour plus d'information sur chacune d'entre elles.)
Les commandes composées peuvent toutes être considérées comme des commandes internes, en tant qu'elles sont des structures de contrôle.
case ... esac ; if ... fi ; for ... done ; select ... done ; until ... done ; while ... done ; {...} ; ( ... ) ; ((...)) ; [ ... ] ; [[ ]]
Les simples quotes : ' délimitent une chaîne de caractères.
Même si cette chaîne contient des commandes ou des variables shell, celles-ci ne seront pas interprétées. Par exemple :
variable='secret' echo 'Mon mot de passe est $variable.'
Mon mot de passe est $variable.
Les doubles quotes : “ délimitent une chaîne de caractères, mais les noms de variable sont interprétés par le shell. Par exemple :
variable="secret" echo "Mon mot de passe est $variable."
Mon mot de passe est secret.
Ceci est utile pour générer des messages dynamiques au sein d'un script.
echo coucou tout le monde
echo "coucou tout le monde"
Bash considère que les anti-quotes (`) délimitent une commande à exécuter.
Les noms de variable et les commandes entre ` sont donc interprétés, et remplacées par la sortie de ces commandes. Autrement dit, les anti-quotes7) remplacent de manière itérative un argument par une commande, comme le fait la commande xargs.
Dossier.txt
contenant les fichiers dossier1
; dossier2
; dossier3
.cd /tmp mkdir Dossier.txt cd Dossier.txt/ touch dossier1 touch dossier2 touch dossier3
ls
dossier1 dossier2 dossier3
rm `ls`
ls
touch dossier1 touch dossier2 touch dossier3 ls
dossier1 dossier2 dossier3
ls | xargs rm
ls
echo `ls`
Cette commande affiche le contenu du répertoire courant à l'écran. Elle est (presque) strictement équivalente à ls.
Certains caractères spéciaux sont appelés méta-caractères ; soit parce qu'ils servent à effectuer des recherches sur les mots ; soient parce qu'ils servent dans les expressions rationnelles; soit encore parce qu'ils représentent symboliquement quelque chose, un fichier, la valeur d'une variable ; finalement parce que ces caractères représentes symboliquement quelque chose;
Voici un tableau qui regroupe les méta-caractères :
Communs à différents shell | ||
---|---|---|
stricts | ? (pour un caractère) |
|
* (plusieurs, ou aucun, caractères) | ||
brackets | [ ] (plage de caractères [12] ou union [1-3]) |
Man glob et man bash appellent ces caractères, non pas des métacaractères mais caractères génériques8).
En définitive, dans la suite j'emploierai le terme “métacaractère” comme synonyme de globs simples et bracket.
Pour un rappel des principaux caractères des expressions rationnelles voir : l'index des regex bash.
# Dans les accolades précédées de $ : ${ } on peut trouver les caractères : :- := : :: :+ :? @ * ## %% % # Ne pas confondre avec joker ou bracket vus ci-dessus
C'est le sujet du tuto : les caractères de transformation de paramètres
Détails et exercices, l'essentiel est là : métacaractères, ou globs, ou encore patterns.
On sait que ls -A
permet de lister tous les fichiers, le retour est alors un peu trop copieux
De même ls -a *
est très prolixe.
On peut alors faire :
ls -d .*
⇒ On obtiendra alors tous les fichiers et dossiers cachés.
Pour affiner la recherche on faire pour n'avoir que ceux dont le nom commence par un c :
ls -d .c*
Retour (exemple) :
.cache .config
Il y a un piège !
.txt
.\ls .* # et son équivalent : ls .?*
⇒ va afficher du répertoire courant (.) : tous les fichiers cachés (normaux) et tous les fichiers cachés de type-répertoire avec les noms des fichiers cachés contenus dans ces répertoires ;
mais aussi du répertoire parent (..) : tous les fichiers cachés normaux et les fichiers cachés de type-répertoire, avec les noms des fichiers cachés contenus dans ces répertoires !
ls .[!.]*
⇒ liste du répertoire courant : tous les fichiers cachés normaux et les fichiers cachés de type-répertoire, avec les fichiers cachés de ces répertoires.
ls -d .[!.]* # et son équivalent : ls -d .??*
⇒ liste du répertoire courant : tous les noms de fichiers normaux cachés et le nom de tous les fichiers cachés de type-répertoire (sans leurs contenus cette fois !)
cd /chemin/vers/FICHIER-cach.essai
ls -la
total 8 drwxr-xr-x 2 hypathie hypathie 4096 mai 26 09:29 . drwxr-xr-x 34 hypathie hypathie 4096 mai 26 09:28 .. -rw-r--r-- 1 hypathie hypathie 0 mai 26 09:29 fichier1 -rw-r--r-- 1 hypathie hypathie 0 mai 26 09:29 .fichier1 -rw-r--r-- 1 hypathie hypathie 0 mai 26 09:29 fichier2 -rw-r--r-- 1 hypathie hypathie 0 mai 26 09:29 .fichier2 -rw-r--r-- 1 hypathie hypathie 0 mai 26 09:29 fichier3 -rw-r--r-- 1 hypathie hypathie 0 mai 26 09:29 .fichier3
rm -i \.f*
rm : supprimer fichier vide « .fichier1 » ? #répondre yes rm : supprimer fichier vide « .fichier2 » ? #répondre yes rm : supprimer fichier vide « .fichier3 » ? #répondre yes
ls -la
total 8 drwxr-xr-x 2 hypathie hypathie 4096 mai 26 10:03 . drwxr-xr-x 34 hypathie hypathie 4096 mai 26 09:28 .. -rw-r--r-- 1 hypathie hypathie 0 mai 26 09:29 fichier1 -rw-r--r-- 1 hypathie hypathie 0 mai 26 09:29 fichier2 -rw-r--r-- 1 hypathie hypathie 0 mai 26 09:29 fichier3
(Observez le prompt : user@nom-machine:~/REPERTOIRE-PARENT$
)
rm \.f*
(Il suffit de mettre par accident un espace après le slash : rm / home/user-à-supprimer
et c'est la catastrophe !)
SI VOUS VOUS TROMPEZ VOUS POUVEZ EFFACER ENTIÈREMENT VOTRE SYSTÈME !
Évitez :
rm -rf /
Petits curieux, petites curieuses, sachez qu'en user comme en root, la plupart des systèmes possèdent une sécurité (pas forcément sur tous), et le retour de la commande ci-dessus est :
rm: il est dangereux d'opérer récursivement sur "/" rm: utiliser --no-preserve-root pour inhiber cette mesure de sûreté
DE MÊME LA COMMANDE :
rm /*
RETOUR :
rm: impossible de supprimer "/bin": est un dossier
LA COMMANDE FATALE À TOUS LES COUPS EST :
rm -rf /*
Après sont exécutions le système est effacé, vos données aussi, et vous êtes triste
méta-caractère Un caractère qui, non protégé, sépare les mots. Un de ceux-ci : | & ; ( ) < > espace tabulation
Un mot est une séquence de caractères considérée comme une unité élémentaire par le shell. On parle également de token (jeton).
|| && ; & ;; ( ) | <retour-chariot>
|| && ; <retour-chariot>
C'est un ” OU “ logique qui apparaît dans deux cas :
1) avec ;; dans le contexte de la commande “case” ;
case $variable-name in pattern1|pattern2|pattern3 # ^ ^ # ou ou command1 ... commandN *) esac
2) dans le contexte d'utilisation des globs étendus et des expressions rationnelles
voir : bash-vii-globs-etendus-regex
Par exemple :
ls ~/Test/!(*jpg|*bmp)
name (){ commands return $TRUE } name
Ou encore dans les substitutions de commande $( ).
Enfin il apparaît une paire de parenthèses dans le cas des globs étendus (voir lien ci-dessus) ;
et dans les expressions rationnelle (voir : bash-vii-globs-etendus-regex
La double paire de parenthèse sert à faire des calculs (voir : page-man-bash-iv-symboles-dans-les-calculs-mathematiques.
À voir : enchainer-plusieurs-commandes
< > > | << >> <& >&
> >> < << >& |
Il s'agit bien du pipe cette fois, et étant une redirection un peu différente, on le trouve souvent explicité à part.
<&- <&-
Permettent la fermeture de l'entrée standard et de la sortie standard.
Un script est la rédaction dans un fichier texte d'un ensemble de commandes et d'expressions régulières (caractères utilisés symboliquement) orientant les instructions données aux commandes.
ls /bin
affiche la liste des commandes externes essentielles utilisées par le système pendant le démarrage, mais utilisables par tous les utilisateurs.ls /sbin
affiche la liste des commandes externes essentielles utilisées par le système pendant le démarrage, et souvent réservées à l'administrateur (rootls /usr/bin /usr/sbin
affiche la liste des commandes externes secondaires, non utilisées pendant le démarrage du système, et respectement utilisables par tous ou plutôt réservées à l'administrateur