Bash scripting: differenze tra le versioni
(espansione di stringa e modifiche minori) |
|||
Riga 1: | Riga 1: | ||
{{Versioni_compatibili}} | {{Versioni_compatibili}} | ||
==Introduzione== | ==Introduzione== | ||
Questa non può essere una guida completa, ma intende fornire un elenco di costrutti per lo scripting Bash eleganti, curiosi e/o poco noti. Si cercherà anche di far luce sui suoi comportamenti più distintivi, partendo dai più facili da fraintendere. | |||
Per l'uso interattivo si rimanda invece a [[Bash tips]]. | Per l'uso interattivo si rimanda invece a [[Bash tips]]. | ||
Riga 7: | Riga 7: | ||
== Variabili == | == Variabili == | ||
In Bash ogni variabile | In Bash ogni variabile di default è trattata come una stringa. E il suo contenuto si accede con <code>${nome}</code> oppure con la forma abbreviata <code>$nome</code>. | ||
=== Nomi di variabili === | === Nomi di variabili === | ||
Un nome di variabile ammette soltanto caratteri alfabetici (maiuscoli e minuscoli), l'underscore ('_') e numeri (non in prima posizione). | |||
La forma abbreviata assume che il nome della variabile sia composto da tutti i caratteri validi incontrati. Per esempio <code>"$nome$cognome"</code> (due variabili concatenate) è equivalente a <code>"${nome}${cognome}"</code>, ma <code>"$nome_$cognome"</code> non lo è a <code>"${nome}_${cognome}"</code> perché <code>nome_</code> (con underscore finale) sarebbe un nome valido. Con la concatenazione di variabili è preferibile accedere alle variabili con le graffe, o in alternativa delimitarle una a una dalle virgolette (<code>"$nome"_"$cognome"</code>). | |||
=== Assegnazioni === | === Assegnazioni === | ||
Nelle assegnazioni non si deve usare il <code>'''$'''</code> davanti | Nelle assegnazioni non si deve usare il <code>'''$'''</code> davanti alla variabile, salvo che per accedere al contenuto di altre variabili: | ||
<pre> | <pre> | ||
var=stringa # assegno un valore (una stringa senza spazi e caratteri speciali) | |||
var="stringa con spazi" # assegno una stringa con spazi (con caratteri speciali preceduti da '\') | |||
var='stringa senza apici' # assegno una stringa contenente spazi e caratteri speciali (ma non apici) | |||
var=$var2 # assegno un'altra variabile | |||
var=${var2} # equivalente a sopra | |||
var="$var2" # equivalente a sopra (non serve quotare nelle assegnazioni) | |||
var='$var2' # si assegna letteralmente $var2 (e non il suo contenuto) | |||
var="\$var2" # come sopra, perché $ è preceduto da \ | |||
var="${var1} testo ${var2}_${var3}" # assegno una concatenazione di variabili e stringhe | |||
</pre> | </pre> | ||
I caratteri speciali (<code>"</code>, <code>`</code>, <code>$</code> e <code>\</code>) nelle stringhe quotate (tra virgolette) devono essere preceduti dal carattere di escape <code>\</code>, mentre | I caratteri speciali (<code>"</code>, <code>`</code>, <code>$</code> e <code>\</code>) nelle stringhe quotate (tra virgolette) devono essere preceduti dal carattere di escape <code>\</code>, mentre quelle tra apici possono contenere tutti i caratteri ma non l'apice, come si vedrà più avanti. | ||
=== Espansione di una variabile === | === Espansione di una variabile === | ||
Con l'unica eccezione dell'assegnazione, quando si accede al contenuto di una variabile senza quotarla, questa può essere trasformata in più di una singola stringa ('''esplosione''') | Con l'unica eccezione dell'assegnazione, quando si accede al contenuto di una variabile senza quotarla, questa può essere trasformata in più di una singola stringa ('''esplosione''') in base agli spazi (e tabulazioni e "a capo") contenuti, e perfino in "niente" se è vuota ("niente" proprio come se non presente nel codice). | ||
Entrambi i comportamenti non sono intuitivi e costituiscono una comune sorgente di errori. Se si vuole sempre considerare il contenuto della variabile come una singola stringa, è necessario accederla quotata (tra virgolette), ossia con <code>"$variabile"</code> oppure <code>"${variabile}"</code>. | Entrambi i comportamenti non sono intuitivi e costituiscono una comune sorgente di errori. Se si vuole sempre considerare il contenuto della variabile come una singola stringa, è necessario accederla quotata (tra virgolette), ossia con <code>"$variabile"</code> oppure <code>"${variabile}"</code>. | ||
Riga 53: | Riga 46: | ||
</pre> | </pre> | ||
Ciò è ancora più importante quando si passa la variabile a un comando, specie se questo agisce su un file indicato dalla variabile, il cui contenuto in presenza di spazi (comuni per i nomi di file degli utenti) potrebbe venir trattato come una lista di file. | |||
Esempio di codice che crea un backup di un file indicato da una variabile, se il suo nome non termina già con estensione .bak. Si notino le virgolette attorno alla variabili, omesse solo nelle assegnazioni (senza spazi), per garantire che ogni variabile venga espansa in | Esempio di codice che crea un backup di un file indicato da una variabile, se il suo nome non termina già con estensione .bak. Si notino le virgolette attorno alla variabili, omesse solo nelle assegnazioni (senza spazi), per garantire che ogni variabile venga espansa in un solo argomento: | ||
<pre> if [ "$file" = "${file%.bak}" ]; then | <pre> if [ "$file" = "${file%.bak}" ]; then | ||
oldfile=$file | oldfile=$file | ||
Riga 63: | Riga 56: | ||
</pre> | </pre> | ||
D'altra parte accedere una variabile senza quotarla permette di assegnare tutte le opzioni da passare a un comando, se sono stringhe senza spazi, alla variabile per poi accederle in una volta sola: | D'altra parte accedere una variabile senza quotarla permette di assegnare tutte le opzioni da passare a un comando, se sono stringhe senza spazi e caratteri speciali, alla variabile per poi accederle in una volta sola: | ||
<pre>ARGUMENTS="" | <pre>ARGUMENTS="--arg1 --arg2 ..." | ||
... | ... | ||
comando $ARGUMENTS | comando $ARGUMENTS | ||
</pre> | </pre> | ||
Si noti | Si noti che usando <code>"$ARGUMENTS"</code> (quotata) per una variabile contenente la stringa vuota, il comando leggerebbe lo stesso un argomento e potrebbe fallire. | ||
===Variabili speciali=== | ===Variabili speciali=== | ||
; <code>$?</code> : contiene il valore di uscita dell'ultimo comando o funzione. Il comando ha successo se restituisce zero, qualsiasi altro valore indica invece un codice di errore; | ; <code>$?</code> : contiene il valore di uscita dell'ultimo comando o funzione. Il comando ha successo se restituisce zero, qualsiasi altro valore indica invece un codice di errore; | ||
; <code>$0</code> : contiene il nome usato per lanciare lo script; | ; <code>$0</code> : contiene il nome usato per lanciare lo script; | ||
Riga 182: | Riga 166: | ||
Una stringa non racchiusa tra apici o tra virgolette ha i seguenti caratteri speciali: <code>'</code>, <code>"</code>, <code>`</code>, <code>$</code>, <code>\</code>, <code>{</code>, <code>[</code>, <code>*</code>, <code>?</code>, <code>~</code>. Se non preceduti dal carattere di escape <code>\</code> possono, in base ai caratteri immediatamente successivi, essere espansi. Questa sezione non è esaustiva, ma consiglia qualche semplice accorgimento, in particolare riguardo l'uso di apici e virgolette per ridurre le necessità dell'escape. | Una stringa non racchiusa tra apici o tra virgolette ha i seguenti caratteri speciali: <code>'</code>, <code>"</code>, <code>`</code>, <code>$</code>, <code>\</code>, <code>{</code>, <code>[</code>, <code>*</code>, <code>?</code>, <code>~</code>. Se non preceduti dal carattere di escape <code>\</code> possono, in base ai caratteri immediatamente successivi, essere espansi. Questa sezione non è esaustiva, ma consiglia qualche semplice accorgimento, in particolare riguardo l'uso di apici e virgolette per ridurre le necessità dell'escape. | ||
Inoltre gli spazi (comprendendo tabulazioni e a capo) non quotati vengono compressi: | Inoltre gli spazi (comprendendo tabulazioni e a capo) non quotati (con apici o virgolette) e non preceduti dal carattere di escape <code>\</code> vengono compressi: | ||
<pre> | <pre> | ||
echo parola1 parola2 # stampa parola1 parola2 con un singolo spazio | echo parola1 parola2 # stampa parola1 parola2 con un singolo spazio | ||
echo "parola1 parola2" # stampa mantenendo gli spazi tra le due parole | echo "parola1 parola2" # stampa mantenendo gli spazi tra le due parole | ||
# con una variabile | |||
var="parola1 parola2" # assegno la stringa alla variabile | |||
echo $var # la stampo con un singolo spazio tra le parole | |||
echo "$var" # la stampo così com'è scritta | |||
</pre> | </pre> | ||
Riga 200: | Riga 188: | ||
righe" | righe" | ||
</pre> | </pre> | ||
===Espansione dei caratteri di escape=== | |||
Sintassi: <code>$'stringa'</code> | |||
Se non quotata e la stringa non è una variabile, ne espande i caratteri di escape (utilizzabili anche con ''printf'' ed ''echo -e''): | |||
* '''\n''', nuova riga; | |||
* '''\b''', backspace (cancella un carattere); | |||
* '''\r''', carriage return (ritorna a inizio riga); | |||
* '''\t''', tabulazione; | |||
* '''\nnn''', carattere ASCII in base 8; | |||
* ecc... | |||
===Racchiudere tra apici=== | ===Racchiudere tra apici=== | ||
Riga 214: | Riga 214: | ||
<pre> | <pre> | ||
echo "$PATH" # espande la variabile PATH e ne stampa il contenuto. | echo "$PATH" # espande la variabile PATH e ne stampa il contenuto. | ||
echo "\"\" \\" # è equivalente a '"" \', si noti l'uso dei \ con | echo "\$HOME" # stampa letteralmente $HOME, senza espanderla. È equivalente a '$HOME'. | ||
echo "\"\" \\" # è equivalente a '"" \', si noti l'uso dei \ con i caratteri speciali. | |||
echo "`ls ..`" # Esegue il comando "ls .." e ne stampa l'output. È equivalente a "$(ls ..)". | echo "`ls ..`" # Esegue il comando "ls .." e ne stampa l'output. È equivalente a "$(ls ..)". | ||
echo "~ * .[a-z]*" # non effettua le espansioni di tilda e percorso, ma stampa letteralmente. | echo "~ * .[a-z]*" # non effettua le espansioni di tilda e percorso, ma stampa letteralmente. | ||
</pre> | </pre> | ||
Riga 251: | Riga 251: | ||
===Output con a capo finali=== | ===Output con a capo finali=== | ||
Si noti che l'espansione di comando, come anche in '''sh''' (''POSIX'') non espande il comando a tutto l'output prodotto, ma omette sempre gli "a capo" finali. Se da una parte è utile nella maggior parte delle situazioni, può talvolta avere effetti collaterali difficili da prevedere. Si consideri per esempio: | Si noti che l'espansione di comando, come anche in '''sh''' (''POSIX''), non espande il comando a tutto l'output prodotto, ma omette sempre gli "a capo" finali. Se da una parte è utile nella maggior parte delle situazioni, può talvolta avere effetti collaterali difficili da prevedere. Si consideri per esempio: | ||
<pre> | <pre> | ||
echo ciao > file_prova | echo ciao > file_prova # scrive ciao (più un "a capo") nel file_prova | ||
testo=$(cat file_prova) | testo=$(cat file_prova) # associa il contenuto (senza "a capo") a $testo | ||
echo " | echo "Bytes: ${#testo}" # NON è la dimensione esatta del file! | ||
echo "$testo" | echo "$testo" # stampa il contenuto di $testo (più un "a capo") | ||
echo "$testo" | # invia la stringa a cmp per un confronto | |||
cmp - file_prova # nessun errore! | |||
echo $? # stampa exit status 0 | |||
</pre> | </pre> | ||
Riga 309: | Riga 312: | ||
** <code>./*."${estensione}"</code> espande dopo aver espanso la variabile (contrariamente a ~), che può anche essere quotata. | ** <code>./*."${estensione}"</code> espande dopo aver espanso la variabile (contrariamente a ~), che può anche essere quotata. | ||
È importante | È importante sapere che, se nessun file combacia con un dato pattern, allora l'espansione '''non''' viene effettuata e i caratteri mantengono il loro valore letterale. Inoltre <code>*</code> e <code>?</code> sono caratteri validi per un nome di file, e un confronto sull'espansione ottenuta non sarebbe sempre risolutivo. | ||
L'esistenza di file ottenuti da tali espansioni va pertanto sempre controllata, per esempio con il costrutto <code>[ -e "$file" ]</code>: | L'esistenza di file ottenuti da tali espansioni va pertanto sempre controllata, per esempio con il costrutto <code>[ -e "$file" ]</code>: | ||
Riga 334: | Riga 337: | ||
Sintassi: <code>prefisso{x..y[..z]}suffisso</code> | Sintassi: <code>prefisso{x..y[..z]}suffisso</code> | ||
L'espansione avviene | L'espansione avviene per tutte le stringhe nell'intervallo compreso da "prefisso'''x'''suffisso" fino a "prefisso'''y'''suffisso", con incrementi di 1 (o '''z''' se specificato). Le stringhe prefisso e suffisso possono essere omesse, o essere variabili (anche quotate, purché le graffe non lo siano), mentre x e y (e z, se presente) devono essere determinati valori: | ||
* <code>{x..y}</code> dove x e y sono due interi (non possono essere variabili); | * <code>{x..y}</code> dove x e y sono due interi (non possono essere variabili); | ||
* <code>{x..y..z}</code> dove x, y e z sono tre interi (non possono essere variabili); | * <code>{x..y..z}</code> dove x, y e z sono tre interi (non possono essere variabili); | ||
Riga 414: | Riga 417: | ||
===Catturare l'exit status=== | ===Catturare l'exit status=== | ||
Per catturare lo stato d'uscita di un comando appena eseguito è sufficiente espandere la variabile speciale <code>$?</code>, come già visto. Tuttavia in caso di fallimento del comando, il controllo effettuato via <code>$?</code> avverrebbe soltanto '''dopo''' un blocco con errore (si veda la parte sul debug). | |||
Per catturare | |||
Per evitare che un blocco abbia un exit status diverso da zero, si possono usare le concatenazioni (oppure un <code>if</code>): | Per evitare che un blocco abbia un exit status diverso da zero, si possono usare le concatenazioni (oppure un <code>if</code>): | ||
Riga 441: | Riga 442: | ||
var=$(printf 'X\000X') # SBAGLIATO: $var contiene XX | var=$(printf 'X\000X') # SBAGLIATO: $var contiene XX | ||
var="$(printf 'X\000X')" # SBAGLIATO: equivalente a sopra | var="$(printf 'X\000X')" # SBAGLIATO: equivalente a sopra | ||
var=$'\000' # SBAGLIATO anche così! | |||
</pre> | </pre> | ||
Riga 503: | Riga 505: | ||
|Verificata_da= | |Verificata_da= | ||
:[[Utente:S3v|S3v]] (versione in Bash tips) | :[[Utente:S3v|S3v]] (versione in Bash tips) | ||
:[[Utente:HAL 9000|HAL 9000]] | :[[Utente:HAL 9000|HAL 9000]] 20:25, 5 lug 2014 (CEST) | ||
|Estesa_da= | |Estesa_da= | ||
:[[Utente:S3v|S3v]] (versione in Bash tips) | :[[Utente:S3v|S3v]] (versione in Bash tips) |
Versione delle 18:25, 5 lug 2014
Versioni Compatibili Tutte le versioni supportate di Debian |
Introduzione
Questa non può essere una guida completa, ma intende fornire un elenco di costrutti per lo scripting Bash eleganti, curiosi e/o poco noti. Si cercherà anche di far luce sui suoi comportamenti più distintivi, partendo dai più facili da fraintendere.
Per l'uso interattivo si rimanda invece a Bash tips.
Variabili
In Bash ogni variabile di default è trattata come una stringa. E il suo contenuto si accede con ${nome}
oppure con la forma abbreviata $nome
.
Nomi di variabili
Un nome di variabile ammette soltanto caratteri alfabetici (maiuscoli e minuscoli), l'underscore ('_') e numeri (non in prima posizione).
La forma abbreviata assume che il nome della variabile sia composto da tutti i caratteri validi incontrati. Per esempio "$nome$cognome"
(due variabili concatenate) è equivalente a "${nome}${cognome}"
, ma "$nome_$cognome"
non lo è a "${nome}_${cognome}"
perché nome_
(con underscore finale) sarebbe un nome valido. Con la concatenazione di variabili è preferibile accedere alle variabili con le graffe, o in alternativa delimitarle una a una dalle virgolette ("$nome"_"$cognome"
).
Assegnazioni
Nelle assegnazioni non si deve usare il $
davanti alla variabile, salvo che per accedere al contenuto di altre variabili:
var=stringa # assegno un valore (una stringa senza spazi e caratteri speciali) var="stringa con spazi" # assegno una stringa con spazi (con caratteri speciali preceduti da '\') var='stringa senza apici' # assegno una stringa contenente spazi e caratteri speciali (ma non apici) var=$var2 # assegno un'altra variabile var=${var2} # equivalente a sopra var="$var2" # equivalente a sopra (non serve quotare nelle assegnazioni) var='$var2' # si assegna letteralmente $var2 (e non il suo contenuto) var="\$var2" # come sopra, perché $ è preceduto da \ var="${var1} testo ${var2}_${var3}" # assegno una concatenazione di variabili e stringhe
I caratteri speciali ("
, `
, $
e \
) nelle stringhe quotate (tra virgolette) devono essere preceduti dal carattere di escape \
, mentre quelle tra apici possono contenere tutti i caratteri ma non l'apice, come si vedrà più avanti.
Espansione di una variabile
Con l'unica eccezione dell'assegnazione, quando si accede al contenuto di una variabile senza quotarla, questa può essere trasformata in più di una singola stringa (esplosione) in base agli spazi (e tabulazioni e "a capo") contenuti, e perfino in "niente" se è vuota ("niente" proprio come se non presente nel codice).
Entrambi i comportamenti non sono intuitivi e costituiscono una comune sorgente di errori. Se si vuole sempre considerare il contenuto della variabile come una singola stringa, è necessario accederla quotata (tra virgolette), ossia con "$variabile"
oppure "${variabile}"
.
Si considerino per esempio i seguenti confronti (usati in genere con if
, while
, ...):
[ $var = $var2 ] # SBAGLIATO! (se una delle due è vuota) [ "$var" = "$var2" ] # corretto [ -n $var ] # SBAGLIATO (se var è vuota) [ -n "$var" ] # corretto
Ciò è ancora più importante quando si passa la variabile a un comando, specie se questo agisce su un file indicato dalla variabile, il cui contenuto in presenza di spazi (comuni per i nomi di file degli utenti) potrebbe venir trattato come una lista di file.
Esempio di codice che crea un backup di un file indicato da una variabile, se il suo nome non termina già con estensione .bak. Si notino le virgolette attorno alla variabili, omesse solo nelle assegnazioni (senza spazi), per garantire che ogni variabile venga espansa in un solo argomento:
if [ "$file" = "${file%.bak}" ]; then oldfile=$file file=${file}.bak cp -- "$oldfile" "$file" fi
D'altra parte accedere una variabile senza quotarla permette di assegnare tutte le opzioni da passare a un comando, se sono stringhe senza spazi e caratteri speciali, alla variabile per poi accederle in una volta sola:
ARGUMENTS="--arg1 --arg2 ..." ... comando $ARGUMENTS
Si noti che usando "$ARGUMENTS"
(quotata) per una variabile contenente la stringa vuota, il comando leggerebbe lo stesso un argomento e potrebbe fallire.
Variabili speciali
$?
- contiene il valore di uscita dell'ultimo comando o funzione. Il comando ha successo se restituisce zero, qualsiasi altro valore indica invece un codice di errore;
$0
- contiene il nome usato per lanciare lo script;
$#
- contiene il numero di argomenti passati allo script (o a una funzione, all'interno di una funzione);
$1, $2, ...
- contengono, se presenti, i parametri passati allo script (o a una funzione);
$@
- contiene la lista di tutti i parametri passati allo script corrente o a una funzione. Ogni parametro viene opportunamente quotato, se questa variabile è quotata, e questo ne permette l'utilizzo nei cicli for per processare (ad esempio) una lista di nomi di file che possono contenere anche spazi. L'uso di questa variabile è quindi in genere preferito rispetto a
$*
che ha la stessa funzione ma, se quotata, non quota i vari parametri ma l'intera stringa;
Esempio:
for file in "$@"; do # Fare quello che si vuole con $file echo "$file" # ... done
$$
- PID del processo corrente;
$!
- PID dell'ultimo job in background.
Esempio:
comando & # lancio un comando in background pid=$! # ottengo il PID del comando ... # eseguo altre operazioni wait $pid # attendo la terminazione del comando status=$? # catturo il suo exit status
Manipolazione delle stringhe
Nelle shell *nix, storicamente, la manipolazione delle stringhe viene fatto attraverso programmi esterni alla shell come sed, awk e perl. Questi programmi vengono ancora usati quando si vuole mantenere la compatibilità con la shell sh
(POSIX), tuttavia imparare anche il solo sed (il più semplice dei tre) non è cosa immediata.
Se si usa Bash non è necessario usare nessun programma esterno, ma basta imparare i tre operatori fondamentali ed alcuni concetti di base, per poter fare tutte le manipolazioni più comuni direttamente sulle variabili.
Si assegna una stringa a una variabile e accedendola tramite la forma con le graffe, si può ricorrere a un modificatore che manipola la stringa (senza modificare il contenuto della variabile), ad esempio:
VAR="stringa-di-esempio" echo ${VAR#stringa-}
ritorna il contenuto della variable VAR senza il prefisso "stringa-". VAR non viene modificata, salvo una nuova assegnazione:
VAR=${VAR#stringa-}
ora il prefisso "stringa-" è stato eliminato anche dalla variabile VAR.
I modificatori sono molti, ma possono essere facilmente ricordati se si imparano i tre fondamentali:
#
- sottrae dall'inizio della stringa (minimale)
%
- sottrae dalla fine della stringa (minimale)
/
- sostituisce una sottostringa con un'altra (solo la prima volta che viene incontrata)
Questi operatori sono minimali, questo vuol dire che se si usano le espressioni regolari per indicare la sottostringa (da eliminare o sostituire) verrà individuata in caso di ambiguità la sottostringa più piccola (o solo la prima nel caso della sostituzione).
Per ottenere gli operatori massimali basta raddoppiare il simbolo:
##
- sottrae dall'inizio della stringa (massimale)
%%
- sottrae dalla fine della stringa (massimale)
//
- sostituisce una sottostringa con un'altra (tutte le volte che viene incontrata)
Gli operatori massimali cercano di individuare la sottostringa più grande che corrisponde all'espressione regolare (nel caso del modificatore // tutte le sottostringhe vengono sostituite). Gli operatori di questo tipo vengono comunemente chiamati anche greedy (ingordi).
Per una spiegazione dettagliata di tutti i modificatori e anche di altri modi di manipolare le stringhe in Bash (ad esempio expr
) vedere:
Esempi: manipolazione delle stringhe
VAR="questa.sarebbe.una.stringa.di.esempio" # Risultato: echo ${VAR#*.} # --> sarebbe.una.stringa.di.esempio echo ${VAR##*.} # --> esempio echo ${VAR%.*} # --> questa.sarebbe.una.stringa.di echo ${VAR%%.*} # --> questa echo ${VAR/st/ST} # --> queSTa.sarebbe.una.stringa.di.esempio echo ${VAR//st/ST} # --> queSTa.sarebbe.una.STringa.di.esempio
Esempio: alternativa a basename
Quando in uno script ci si deve riferire al nome dello script stesso è usuale utilizzare il comando (esterno a bash) basename
. Tuttavia, tramite i modificatori del paragrafo precedente, Bash stessa è in grado di fornire questa funzionalità. Basta usare l'espressione ${0##*/}
.
usage () { echo "usage: ${0##*/} " exit 1 }
Caratteri di escape, apici e virgolette
Alcuni caratteri hanno un valore speciale per la shell, per consentirne le espansioni (di variabile, parametro, comando, percorso, ecc...). Di conseguenza se si intende scrivere un carattere speciale senza espanderlo, è necessario comunicarlo alla shell facendolo precedere da un carattere di escape '\
' oppure racchiudendolo tra apici o virgolette (a seconda dell'espansione da disattivare).
Una stringa non racchiusa tra apici o tra virgolette ha i seguenti caratteri speciali: '
, "
, `
, $
, \
, {
, [
, *
, ?
, ~
. Se non preceduti dal carattere di escape \
possono, in base ai caratteri immediatamente successivi, essere espansi. Questa sezione non è esaustiva, ma consiglia qualche semplice accorgimento, in particolare riguardo l'uso di apici e virgolette per ridurre le necessità dell'escape.
Inoltre gli spazi (comprendendo tabulazioni e a capo) non quotati (con apici o virgolette) e non preceduti dal carattere di escape \
vengono compressi:
echo parola1 parola2 # stampa parola1 parola2 con un singolo spazio echo "parola1 parola2" # stampa mantenendo gli spazi tra le due parole # con una variabile var="parola1 parola2" # assegno la stringa alla variabile echo $var # la stampo con un singolo spazio tra le parole echo "$var" # la stampo così com'è scritta
Il carattere di escape \
prima di un "a capo", anche se quotato (tra virgolette), ha un significato speciale che consente di scrivere un comando su più righe, trattando ogni riga preceduta da \
come la continuazione del comando precedente:
# stampa tutto su una riga echo "testo su \ più \ righe" # stampa su più righe echo "testo su più righe"
Espansione dei caratteri di escape
Sintassi: $'stringa'
Se non quotata e la stringa non è una variabile, ne espande i caratteri di escape (utilizzabili anche con printf ed echo -e):
- \n, nuova riga;
- \b, backspace (cancella un carattere);
- \r, carriage return (ritorna a inizio riga);
- \t, tabulazione;
- \nnn, carattere ASCII in base 8;
- ecc...
Racchiudere tra apici
Racchiudendo tra apici (apostrofi) una stringa, si riducono i caratteri speciali a uno soltanto, ossia lo stesso apice, rappresentando la stringa per il suo solo valore letterale e impedendo tutte le espansioni. Lo svantaggio è che non esiste un carattere di escape, e che quindi un apice non può essere racchiuso tra apici in alcun modo. Per esempio:
echo '$PATH "" \ `ls ..` \$HOME ~ * .[a-z]*' # stampa la stringa tra apici, così com'è scritta
echo 'L'\''albero di... ' # stampa "L'albero di..." (l'accento non può essere racchiuso tra apici e va preceduto da \)
Quotare (racchiudere tra virgolette)
Racchiudere tra virgolette (ossia quotare) ogni stringa è in genere raccomandabile e la scelta consigliata, così da ridurre drasticamente il numero di caratteri speciali, permettendo allo stesso tempo l'espansione delle variabili e dei comandi. I soli caratteri speciali sono $
, `
(ma non l'apice), "
e \
, che devono essere preceduti dal carattere di escape \
. Le espansioni di percorso e tilda (*, ?, [...], ~, ... ) non sono possibili; mentre lo sono quelle di variabile e parametro (attraverso un $
senza \
), e quelle di comando (attraverso un `
o un $(
senza \
).
Riprendendo l'esempio e considerandolo per parti:
echo "$PATH" # espande la variabile PATH e ne stampa il contenuto. echo "\$HOME" # stampa letteralmente $HOME, senza espanderla. È equivalente a '$HOME'. echo "\"\" \\" # è equivalente a '"" \', si noti l'uso dei \ con i caratteri speciali. echo "`ls ..`" # Esegue il comando "ls .." e ne stampa l'output. È equivalente a "$(ls ..)". echo "~ * .[a-z]*" # non effettua le espansioni di tilda e percorso, ma stampa letteralmente.
Espansione di comando
Consiste nel trasformare l'output di un comando qualsiasi (interno della shell, esterno, una funzione e anche forme composte) in argomenti per un altro comando, oppure nel valore da assegnare a una variabile. Si effettua racchiudendo un comando tra $(...)
, oppure tra due apici gravi `...`
(su tastiera con layout italiano: Alt Gr + '
):
$(comando)
oppure, meno leggibile e sconsigliata:
`comando`
L'output del comando consiste in zero, una o più stringhe, in base agli spazi presenti nell'output prodotto e in maniera analoga all'espansione a cui sono soggette le variabili. Per trasformare l'output di un comando in una singola stringa è necessario che l'espansione di comando sia quotata, con l'eccezione dell'assegnazione a una variabile.
Esempi di assegnazione:
# assegna alla variabile $oggi la data in formato YYYY_MM_DD oggi=$(date +%F) # senza virgolette oggi="$(date +%F)" # equivalente a sopra (non servono nelle assegnazioni) # associo a testo il contenuto del file indicato da $file testo=$(cat $file) # SBAGLIATO! (se la variabile $file contiene spazi o caratteri speciali) testo=$(cat "$file") # le virgolette attorno alla variabile sono necessarie testo="$(cat "$file")" # equivalente a sopra
Esecuzione dei comandi e passaggio come argomento del loro output:
# stampa stati echo "Login name: $(logname); Name: $(whoami); UID: $(id -ur); EUID: $(id -u); Groups: $(groups)" echo "OS: Debian GNU/Linux $(cat /etc/debian_version) ($(lsb_release -sc))" # uguale a $(lsb_release -sd) echo "Kernel: $(uname) $(uname -r) ($(uname -v))"
Output con a capo finali
Si noti che l'espansione di comando, come anche in sh (POSIX), non espande il comando a tutto l'output prodotto, ma omette sempre gli "a capo" finali. Se da una parte è utile nella maggior parte delle situazioni, può talvolta avere effetti collaterali difficili da prevedere. Si consideri per esempio:
echo ciao > file_prova # scrive ciao (più un "a capo") nel file_prova testo=$(cat file_prova) # associa il contenuto (senza "a capo") a $testo echo "Bytes: ${#testo}" # NON è la dimensione esatta del file! echo "$testo" # stampa il contenuto di $testo (più un "a capo") echo "$testo" | # invia la stringa a cmp per un confronto cmp - file_prova # nessun errore! echo $? # stampa exit status 0
Un altro esempio, vogliamo associare il carattere "a capo" a una variabile:
nuova_riga=" " # funziona, ma è brutto esteticamente e non permette l'indentazione nuova_riga=$'\n' # funziona (con Bash) ed è il modo consigliato # a titolo esemplicativo con l'espansione di comando nuova_riga=$(echo) # SBAGLIATO, $nuova_riga è vuota nuova_riga="$(echo)" # SBAGLIATO, come sopra nuova_riga=$(printf "\n") # SBAGLIATO, $nuova_riga è sempre vuota # una possibile soluzione... nuova_riga=$(printf "\nX") # $nuova_riga contiene "a capo" seguito da X nuova_riga=${nuova_riga%X} # $nuova_riga contiene "a capo" (la X è rimossa) # Attenzione che il carattere aggiunto dev'essere nell'output del comando nuova_riga=$(printf "\n")X # SBAGLIATO, $nuova_riga contiene solo X
Espansione di tilda e percorso
File Si ricordi per le espansioni che su ambienti UNIX e Unix-like per file si può intendere sia un file regolare, ma anche una directory, un link simbolico, una pipe, un socket, un device, ecc... |
Non racchiudere tra virgolette e apici serve a permettere le espansioni di percorso e tilda. È sempre consigliabile racchiudere tutto il resto tra virgolette, per non permettere espansioni accidentali. In particolare si noti che un'espansione di percorso (ma non di tilda) può avvenire anche in base al contenuto di una variabile, se questa non è quotata.
Per esempio:
var=* # assegno * alla variabile var="*" # è equivalente a sopra echo "$var" # stampa letteralmente * echo $var # stampa la lista di tutti i file non nascosti # nella directory corrente, oppure * se è vuota var=~ # assegno ~ alla variabile echo "$var" # stampa letteralmente ~ echo $var # equivalente a sopra, nessuna espansione
Le espansioni più comuni sono:
~
(tilda) espande alla home (equivalente alla variabile $HOME, che però può essere quotata);~utente
espande alla home di un dato utente, ma la stringa utente non può essere quotata né essere una variabile da espandere;
?
pattern per un singolo carattere di un file;nomefile.???
combacia con tutti i file con nome "nomefile" e terminanti con tre caratteri qualsiasi per estensione;
*
pattern per tutti i possibili caratteri di un file (di default tranne quelli inizianti con .):*
da solo espande a tutti i file non nascosti nella directory corrente. È sempre buona norma far precedere l'asterisco da un ./, che indica la cartella corrente, se è il primo carattere del pattern, per impedire espansioni di file inizianti con "-", che potrebbero essere visti come opzioni da alcuni comandi;./*
equivalente a*
ma più sicuro; di seguito si aggiunge il prefisso della directory corrente a tutte le espansioni inizianti con*
;./*.txt
espande a tutti i file con estensione .txt;./*/
espande a tutte le directory non nascoste;prefisso*suffisso
espande a tutti i file con un dato prefisso e suffisso;./*."${estensione}"
espande dopo aver espanso la variabile (contrariamente a ~), che può anche essere quotata.
È importante sapere che, se nessun file combacia con un dato pattern, allora l'espansione non viene effettuata e i caratteri mantengono il loro valore letterale. Inoltre *
e ?
sono caratteri validi per un nome di file, e un confronto sull'espansione ottenuta non sarebbe sempre risolutivo.
L'esistenza di file ottenuti da tali espansioni va pertanto sempre controllata, per esempio con il costrutto [ -e "$file" ]
:
for file in ./*; do if [ -e "$file" ]; then ... fi done
Esempio: cambiare l'estensione ai file
Rinomina tutti i file *.txt
della directory corrente in *.log
:
$ for f in ./*.txt; do if [ -e "$f" ]; then mv -- "$f" "${f/%txt/log}"; fi; done
Espansione di parentesi (graffa)
Se i caratteri {
e }
non sono quotati, e non sono preceduti dal carattere di escape \
, possono essere espansi con due diverse sintassi.
Con indici di intervallo
Sintassi: prefisso{x..y[..z]}suffisso
L'espansione avviene per tutte le stringhe nell'intervallo compreso da "prefissoxsuffisso" fino a "prefissoysuffisso", con incrementi di 1 (o z se specificato). Le stringhe prefisso e suffisso possono essere omesse, o essere variabili (anche quotate, purché le graffe non lo siano), mentre x e y (e z, se presente) devono essere determinati valori:
{x..y}
dove x e y sono due interi (non possono essere variabili);{x..y..z}
dove x, y e z sono tre interi (non possono essere variabili);{a..b}
dove a e b sono due caratteri (non possono essere variabili);{a..b..z}
dove a e b sono due caratteri, e z è un intero (non possono essere variabili).
Per esempio:
# crea un file temporaneo, associa il percorso a $tmp_file tmp_file=$(tempfile) # crea altri dieci file temporanei (.0, .1, .., .9) con lo stesso nome touch -- "$tmp_file".{0..9}
Con lista di stringhe
Sintassi: prefisso{stringa1,stringa2,...}suffisso
L'espansione avviene per tutte le stringhe nella lista, racchiudendole tra il prefisso e il suffisso dati, se presenti. Il prefisso, il suffisso e tutte le stringhe possono essere variabili, anche quotate, purché non siano quotate le graffe e le virgole interne.
Questa espansione è effettuata prima di tutte le altre, e il risultato dell'espansione se non quotato può quindi subire ulteriori espansioni. Per esempio per effettuare un'operazione sui file nella cartella corrente che hanno una data estensione, si può scrivere:
for file in ./*.{odt,abw,txt,rtf,doc}; do if [ -e "$file" ]; then ... fi done
e la prima riga è equivalente a:
for file in ./*.odt ./*.abw ./*.txt ./*.rtf ./*.doc; do
Concatenazione e redirezione
&&
- operatore logico AND, il secondo comando verrà eseguito solo se il primo avrà esito positivo
;
- separatore di comandi, il secondo comando verrà eseguito in ogni caso (in uno script è equivalente a un "a capo" e questa è la concatenazione di default)
$ cd ... && echo done $ cd ... ; echo done
Nel caso specifico di un'applicazione che resta in esecuzione il secondo comando non verrà eseguito finché il
primo non sarà terminato, in questo caso si usa &
e saranno eseguiti entrambi in quanto il primo comando viene mandato in background; prendiamo xterm come esempio:
$ xterm && xterm -rv $ xterm ; xterm -rv $ xterm & xterm -rv
|
- pipe, passa l'output del comando che la precede come input del comando che la segue
$ ls -A1 | less
||
- operatore logico OR, restituisce esito positivo se almeno una delle condizioni di verifica valutate è vera
$ if [ -f ~/.bashrc ] || [ -d ~/.config ]; then echo 'w00t!'; else echo 'no such file or directory'; fi $ if [ -f ~/.bashrc ] || [ -d ~/.configs ]; then echo 'w00t!'; else echo 'no such file or directory'; fi $ if [ -f ~/.bash ] || [ -d ~/.configs ]; then echo 'w00t!'; else echo 'no such file or directory'; fi
Dirige output di comando su file:
$ man xterm > xterm.txt
Dirige output di errore su file:
$ xterm 2> xterm-errors.log
Entrambi su file diversi:
$ xterm > xterm_out.log 2> xterm_err.log
Entrambi sullo stesso file:
$ xterm &> xterm.log
Entrambi sullo stesso file, alternativa:
$ xterm > xterm.log 2>&1
Appende nuovo testo a quello già presente in un file:
$ man uxterm >> xterm.txt
Cancella contenuto di un file:
$ :> xterm.txt
Catturare l'exit status
Per catturare lo stato d'uscita di un comando appena eseguito è sufficiente espandere la variabile speciale $?
, come già visto. Tuttavia in caso di fallimento del comando, il controllo effettuato via $?
avverrebbe soltanto dopo un blocco con errore (si veda la parte sul debug).
Per evitare che un blocco abbia un exit status diverso da zero, si possono usare le concatenazioni (oppure un if
):
comando && status=0 || # se corretto status=$? # se sbagliato
I comandi precedenti costituiscono un unico blocco, sempre corretto, mentre invece:
comando # questo comando è anche un blocco status=$? comando; status=$? # sono sempre due blocchi distinti
Output dei comandi e carattere ASCII n. 0
Una variabile non può contenere il carattere ASCII n. 0, che è usato per indicare la fine della stringa. Non potendo gestire direttamente il carattere ASCII n. 0, questo non può essere presente nell'espansione di un comando, ma dev'essere lasciato a comandi esterni mediante l'uso di una o più pipe.
Infatti non esiste un modo di contenere il carattere ASCII n. 0 in nessuna posizione:
var=$(printf '\000') # SBAGLIATO: $var è vuota var=$(printf '\000X') # SBAGLIATO: $var contiene solo X var=$(printf 'X\000X') # SBAGLIATO: $var contiene XX var="$(printf 'X\000X')" # SBAGLIATO: equivalente a sopra var=$'\000' # SBAGLIATO anche così!
L'utilità di questo carattere si deve in particolare al fatto che nemmeno i file possono averlo nel proprio nome, mentre invece permettono caratteri jolly (*, ?, ...) come abbiamo già visto con le espansioni di percorso, e potrebbero contenere perfino il carattere "a capo".
L'espansione di percorso funziona normalmente, anche in presenza di "a capo", ma potrebbero sorgere problemi sfruttando l'espansione di comando. Per esempio con il comando esterno find
, utilizzato per effettuare ricerche in modo ricorsivo, e che di default restituisce tutti i file trovati che soddisfano determinate condizioni stampandoli uno per riga, assumendo implicitamente che non contengano il carattere "a capo".
Uno script a titolo esemplificativo:
# creo una directory e un file con "a capo" nel nome mkdir ./prova touch ./prova/"file contenente"$'\n'"a capo nel nome" # SBAGLIATO! (per via del nome del file particolare) num_file=$(find ./prova -type f | # stampo i file regolari nella directory, uno per riga wc -l) # conto il numero di righe echo "La directory prova contiene ${num_file} file" # restituisce 2 invece di 1 # forma corretta num_file=$(find ./prova -type f -print0 | # stampo il carattere ASCII n. 0 dopo ogni file tr -dc '\000' | # rimuovo tutti i caratteri ASCII diversi dal n. 0 wc -c) # conto il numero di caratteri echo "La directory prova contiene ${num_file} file" # restituisce 1 # pulizia rm -- ./prova/file*nome # elimino il file rmdir -- ./prova # elimino la directory (se vuota)
Un abbinamento comune al comando esterno GNU find
(dotato dell'opzione -print0) è il comando esterno GNU xargs
(dotato dell'opzione -0):
find /percorso -opzione1 ... -opzioneN -print0 | # trova file che soddisfano le condizioni date xargs -0 comando [ argomenti ... ] # li passa come argomenti a un comando esterno
Si leggano i rispettivi manuali per maggiori informazioni. find
ha inoltre la possibilità di eseguire altri comandi esterni sui file trovati direttamente con le opzioni -exec ed -execdir, ma la sintassi è più complessa e non supporta più di un processo per volta, come invece xargs
.
Debug integrato
Bash, proprio come dash, ha delle opzioni che ne consentono il debug.
Invocando uno script con -n
è possibile effettuare un primitivo controllo di sintassi. Non vengono controllati comandi inesistenti e nemmeno le espansioni, ma può essere utile per verificare che i blocchi sono stati chiusi correttamente prima di eseguire lo script:
$ bash -n script.sh
Altre opzioni utili, che possono essere impiegate anche congiuntamente durante l'esecuzione:
-x
stampa ogni comando prima di eseguirlo;-v
stampa l'intero blocco di codice che è stato letto (solo la prima volta);-u
interrompe l'esecuzione dello script se si accede a una variabile che non è mai stata assegnata;-e
interrompe lo script in caso un blocco ritorni un errore (se il valore di ritorno di un comando non è controllato da unif
,while
,until
o dalla concatenazione di comandi con||
).
Le opzioni possono essere anche accorpate:
$ bash -euvx script.sh
Link
Link ad altre risorse su GNU Bash:
- Bash Referece Manual: manuale ufficiale
- Advanced Bash-Scripting Guide: la Bibbia dello bash scripting.
Guida scritta da: ~ The_Noise (versione in Bash tips) | Debianized 60% |
Estesa da: | |
Verificata da: | |
Verificare ed estendere la guida | Cos'è una guida Debianized |