Bash script

De Cacauet Wiki
Salta a la navegació Salta a la cerca

Intro

Els diversos Unix's han tingut sempre una potent interfície de comandes. La shell és la interfície que ens permet "dialogar" amb el nucli del sistema operatiu passant-li comandes.

Hi ha diversos llenguatges de shell script, entre ells el CSH (C-shell), i el KSH (Korn-shell), però sembla que el què s'ha imposat definitivament és el BASH o Bourne-Again SHell.

És important dominar algun dels editors de text per consola. El més típic ara és el nano, tot i que convé també conèixer vi ja que sol tenir syntax highlight (vim o vi improved) i perquè per raons històriques sempre està present i ens pot treure d'un apuro.

Referències:


Arxius i permisos

Els scripts son arxius de text amb instruccions llegibles per la nostra shell. Es comenten amb un # tot i que la 1a línia és una excepció (comença per #!) i indica quin serà l'intèrpret que l'executarà. Edita hola.sh des del teu editor de text favorit:

#!/bin/bash
# això és un comentari
echo "hola!"  # això ja és una instrucció

...i abans de començar a executar-lo cal donar permisos d'execució a l'arxiu:

$ chmod +x hola.sh

...i ara sí que ja el podem executar:

$ ./hola.sh


Entorn

Les comandes de la shell s'executen sense incloure tota la ruta mercès a la variable d'entorn $PATH

$ echo $PATH

...i podrem veure els directoris (separats per :) en els que la shell buscarà una comanda quan la teclegem. Les que (quasi) segur que hi han de ser son:

/bin:/sbin:/usr/bin:/usr/local/bin:/usr/local/sbin

En aquestes carpetes hi ha les comandes més habituals que utilitzem com ls a /bin o ifconfig a /sbin (a sbin hi ha les de Superusuari, d'aquí la 's' al davant).

Si algun dia volem saber on hi ha una comanda que estem utilitzant, podem esbrinar-ho amb:

$ which ls

...i ens dirà on està el binari que s'executarà amb aquesta comanda (que es buscarà als diferents directoris $PATH, en l'ordre que estan situats).

Quan obrim una shell el sistema operatiu crea unes variables d'entorn. Les podem veure fent:

$ env

Directori ~/bin

Un truc força útil és crear la carpeta bin al nostre home directory. Aquesta es posarà automàticament al $PATH (fes logout i login després de crear la carpeta). Aquest és el lloc on posar scripts nostres i que no calgui que estiguin disponibles per altres usuaris.

$ mkdir ~/bin

...fes logout i login i comprova el $PATH


Variables

Les variables porten un $ al davant (similar a Perl i PHP), com p.ex. $PATH. En qualsevol moment i des de la pròpia shell (sense crear un script) podem crear-ne:

$ aaa=22
$ sss='els strings millor entre cometes per evitar errors amb els espais'
$ echo $aaa
22

OJU!: és important que el signe "=" estigui enganxat, és a dir sense espais, al nom de la variable i al valor.

Recordem que podem veure totes les variables d'entorn amb:

$ env

Busca la nostra variable 'aaa' a la llista, veuràs que no hi és. Si ho vols fer com els pros, filtra la sortida amb grep:

$ env | grep aaa

Si volem que les nostres variables estiguin disponibles per altres programes, aplicacions, scripts, etc. caldrà exportar-les:

$ export aaa=22

Si ara filtrem la llista de variables d'entorn, veurem que sí que hi és:

$ env | grep aaa

Assignant a una variable el resultat d'una comanda

Si volem que una variable tingui com a valor el resultat d'una comanda, tenim 2 formes de fer-ho. Una és amb amb $() i l'altra utilitzant l'accent obert:

$ avui=$(date)

o bé utilitzant l'accent obert:

$ avui=`date`
$ echo $avui
dt feb 16 15:57:09 CET 2016


Expressions aritmètiques

Per efectuar càlculs matemàtics ens caldrà utilitzar la següent sintaxi:

$((expressió))

Per exemple:

$ echo $((11+5))
16

OJU: en principi només ens permet aritmètica sencera (integers). No podem utilitzar nombres en coma flotant (floats). Si volem efectuar operacions matemàtiques en coma flotant haurem d'utilitzar l'aplicació bc (Basic Calculator). Feu-li un cop d'ull aquí a la utilització de bc.

Si algun dels elements no és un element interpretable es prendrà com un zero '0' (per exemple, variables no definides o strings no transformables a integer).


Arguments

Dintre d'un script podem accedir als arguments que ens passi l'usuari amb les variables $1, $2, etc.

Per exemple, podem crear un script args.sh com aquest:

echo El primer argument és $1
echo El segon és $2
echo el nom del script és $0

I l'executem amb:

$ ./args.sh hola que tal


Arrays

Un array és un conjunt de valors arranjats consecutivament. Pel nostre cas ho veurem com una variable que pot contenir diversos valors a dins, indexats amb una posició.

Assignació:

PARAULES=( un array ple de coses )

O el que seria el mateix:

PARAULES[0]="un"
PARAULES[1]="array"
PARAULES[2]="ple"
...

Utilització. És una mica tricky, AL TANTO SON CLAUS {} !!:

echo "element 3 = ${PARAULES[2]}"

La cosa es complica quan fem expressions aritmètiques:

NOMBRES=( 3 23 5 100 )
echo $(( ${NOMBRES[1]} + ${NOMBRES[3]} ))
# Ens donarà el resultat "123"

De fet, bash no és un llenguatge molt adequat per aquest tipus d'operacions. Però poder, es poden fer.

Per iterar en un array podem fer:

for var in ${PARAULES[@]}
do
    # printf ens permet escriure sense salt de línia (a diferència del echo que sí el posa)
    printf $var 
done


Bucles, llistes i altres gaites

És important recalcar que els bucles (loops) de bash script solen estar basats en llistes, enlloc de índexs. Com que habitualment els programadors ens posem nerviosos amb aquest canvi, us poso també com fer un bucle basat en índex (perquè quedi constància, ja que a la pràctica no es fa servir gaire ;)

Bucle indexat

Típicament:

for a in {5..10}
do
    echo $a
done

Es poden posar bucles inversos {10..5} i també amb salts {10..5..2}.

També es pot fer a l'estil C:

for (( i=0; i<10; i++ ))
do
    echo $a
done

Bucle de llista (a.k.a. "for each")

Aquest és el realment útil. Hi ha moltes maneres de fer-los, però solen seguir aquesta forma:

for elem in elements
do
    ...
done

El típic exemple és un llistat d'arxius d'alguna carpeta, per exemple, /var. En aquest cas posem els noms d'arxius a la variable $arxius (tal i com ho solem fer amb la comanda ls) i després podrem iterar pels seus elements:

arxius=/var/*.jpg  # triem només els arxius .jpg (no n'hi ha cap, però)
arxius=/var/l*     # triem només els arxius que comencen per "l"
for elem in $arxius
do
    echo arxiu = $elem
done

while...do

Segur que també coneixeu aquest. Per aquest, però, cal tenir clar els condicionals (IF). Segueix llegint més avall.

while...do funciona amb condició d'entrada:

while [ condició ]
do
    ...
done

Existeix la variant until...do. Aquesta simplement inverteix la condició (negació).


Condicionals

...

if...then

...

switch...case

...


stdin, stdout i redireccions a arxius

Aquests conceptes correponen als standard streams de entrada/sortida definits per la comunicació dels processos executats en una computadora.

  • stdin o entrada estàndard: sol ser el teclat
  • stdout o sortida estàndard: sol ser la pantalla
  • stderr: sol ser també la pantalla, però està en un canal separat ja que ens dona avantatges de gestió.

stdout

Podem redirigir stdout d'un programa cap a un arxiu amb ">" , per exemple (OJU, esborrarem tots els continguts del fitxer destí):

$ ls -la > directori.txt

i podrem veure els continguts del directori fent:

$ cat directori.txt

Podem AFEGIR dades a l'arxiu amb ">>":

$ echo "xin pom!" >> directori.txt

Comprovem amb:

$ cat directori.txt

stderr

Però si cometem un error en la comanda, el missatge d'error ens apareix a stderr, que continua sent la pantalla:

$ ls -la aaa > directori.txt
ls: no s’ha pogut accedir a aaa: El fitxer o directori no existeix

Si volem redirigir stderr ho podem fer amb "2>", per exemple:

$ ls -la aaa > directori.txt 2> errors.txt

...ara el missatge d'error no ens apareix, però el trobarem dins de l'arxiu "errors.txt".

stdin

L'entrada estàndard sol ser el teclat, però també podem redirigir-hi els continguts d'un arxiu. Això es pot fer de 2 maneres

  1. Amb la "tuberia" pipe "|"
    $ cat directori.txt | grep txt
  2. Amb "<":
    $ grep txt < directori.txt

llegir un arxiu línia a línia

Un truc important en els scripts és llegir un arxiu línia a línia per poder processar-lo i aplicar canvis en ell. Sol fer-se a l'estil de les redireccions de stdin, concretament podem llistar l'arxiu /etc/passwd:

while read LINE
do
    # processem LINE d'alguna manera
    echo $LINE
done < /etc/passwd

Tal com hem vist, el mateix es pot aconseguir amb el pipe:

echo /etc/passwd | while read LINE
do
    # processem LINE d'alguna manera
    echo $LINE
done


pipes

...


Exercicis

Exercicis de scripts:

  1. Crea un script que agafi tots els arxius .JPG indicats al directori que li passem com a argument, i els canvii d'extensió a .PNG
  2. Investiga la llibreria ImageMagick i utilitza la comanda convert per transformar les imatges de la carpeta que passem al script a una resolució fixa de 800x600 pixels.
  3. Fes el joc del penjat en Bash. Parteix d'una paraula fixa i demanem a l'usuari que ens entri una lletra. Hauríem d'iterar per les lletres del string de la paraula i descobrir si la lletra entrada per l'usuari figura en ella. Et pot convenir el truc següent, transforma un string en una llista de lletres (i és més fàcil iterar):
    $ echo "paraula" | fold -w1
  4. Millora el joc del penjat afegint un diccionari de paraules en un arxiu de text (una paraula per línia), fent que el programa trii una de les paraules aleatòriament.
  5. Fes un script que llegeixi el directori passat com a argument i llisti els arxius però no les carpetes.
  6. Amplia l'exercici anterior fent que llisti els arxius recursivament, és a dir, que entri a les subcarpetes mostrant els continguts.

Exercicis de backups:

  1. Crea un script per comprimir els continguts de la carpeta que es passa com a argument. Ha de demanar interactivament si utilitza compressió gzip o bz2.
  2. Crea un script que faci un backup de la carpeta /var/www i el guardi a /root/backups comprimit en .tgz . Fes que es guardi al directori de destí sense sobreescriure els arxius existents. Per exemple, si tens l'arxiu backup1.tgz , el següent arxiu s'ha de guardar com a backup2.tgz
  3. Crea un script que faci un backup de les bases de dades MySQL amb mysqldump i les comprimeixi amb bzip2.
  4. Instal·la el script de l'exercici nº2 (backup /var/www) al CRON i fes que s'executi cada minut. Comprova que es realitza correctament.

Exercicis de configuració de serveis:

  1. Fes un script que llegeixi un arxiu passat per argument (serà un arxiu de conf) i que el tregui per stdout sense comentaris.
    Potser et convé llegir una mica sobre expressions regulars.
  2. Crea un script que configuri un servidor Samba PDC. Ha de generar l'arxiu de configuració smb.conf d'acord amb les indicacions de l'article de Samba. De forma interactiva, l'script ha de demanar les dades necessàries: nom del domini, nom NetBIOS, si es publiquen el profiles separats dels homes, o junts, etc.
  3. Crea un script que configuri una màquina Linux Ubuntu LTS (en principi 14.04) i la entri en un domini Samba (com a client) d'acord amb el què s'explica en aquest article.
  4. Fes un script que configuri una màquina Linux amb la següent configuració:
    • Una doble xarxa (a) NAT i (b) interna amb 192.168.33.1
    • El servei DNSMASQ per fer de DNS
    • El servidor DHCP del DNSMASQ per servir IPs en el rang 192.168.33.100-200, i utilitzant la pròpia màquina de gateway i DNS.
  5. ...