Script
Script
Script
En informtica, un script es un guin o conjunto de instrucciones de un programa Es un guin o archivo de procesamiento por lotes (en ingls "script") es un conjunto de instrucciones, sentencias de control, variables y dems elementos de programacin generalmente almacenadas en un archivo de texto (pueden considerarse como un archivo de instrucciones o como un programa).
Los lenguajes de programacin interpretados o, incluso, las shell los van interpretando y ejecutando para realizar diversas tareas como combinar componentes, realizar pequeos programas, interactuar con el sistema operativo o con el usuario. Esto es difiere de los programas de lenguajes como C, C++, Java, Pithon, Bisual, Delphi, etc. En los cuales primero se escribe el cdigo del programa en una hoja de texto y luego se lo pasa por un interpretador y compilador para que nos de un determinado producto el cual seria el programa en si. (Si no se entendi avisen les explico con ejemplos) Pueden estar embebidos en otro lenguaje como es el caso de los scripts PHP en cdigo HTML En algunos textos se lo traduce como guiones, pero esta traduccin no ha logrado salir del mundo de las buenas intenciones. Con frecuencia, se distingue entre programas y scripts porque los primeros han de ser traducidos a lenguaje mquina (ceros y unos) antes de ser ejecutados, mientras que los scripts conservan su forma original, siendo interpretados comando por comando cada vez que son ejecutados. El trmino "guin" (o script) se tom del guin escrito de las artes escnicas, el cual es interpretado por una serie de actores/actrices (o, en nuestro caso, programas). En el sistema operativo DOS, se le conoce como "BATCH" (Grupo de instrucciones o una sola instruccin a cierta condicin dada), en los shell de linux se los conoce como scripts.
Podemos observar como en la primera ejecucin ls -la devuelve un valor 0 de terminacin correcta y en la segunda devuelve un error y n cdigo 1.
Ejecucin consecutiva
Podemos agrupar varias rdenes en una misma lnea de ordenes separndolas por ";" La agrupacin de rdenes separadas por ";" es til cuando tenemos que repetir una misma secuencia de rdenes varias veces. La ejecucin de cada una de las rdenes se realiza cuando ha concluido la anterior, e independiente de que el resultado haya sido correcto o no.
Esto nos puede resultar til para demorar la ejecucin de una o varias rdenes un determinado tiempo, por ejemplo
$ sleep 300 ; ps axu
Ejecucin condicional
Otra situacin algo ms elaborada que la anterior es ejecutar una orden condicionada a la terminacin correcta o no de una orden previa. Esta funcionalidad nos la proporcionan los operadores "&&" y "||".
Operador &&
El primer operador, "&& " separa dos rdenes de forma que la que tiene a la derecha slo se ejecuta cuando la de la izquierda termina correctamente, es decir
orden1 && orden2
orden2
Por ejemplo, queremos ejecutar la orden cat fichero slo si existe fichero; entonces tendremos que buscar una orden que termine con un error si no existe fichero, por ejemplo ls fichero y condicionar la ejecucin de cat fichero a esta:
$ ls fichero && cat fichero
Otro ejemplo, para compilar los controladores de dispositivos de linux e instalarlos, lo podemos hacer como:
make module && make modules_install
Operador ||
El segundo operador, "|| " tiene un comportamiento similar al anterior, separa dos rdenes de forma que la que tiene a la derecha slo se ejecuta cuando la de la izquierda termina incorrectamente, es decir
orden1 || orden2
orden2
Por ejemplo si no existe fichero queremos crearlo vaca y si existe no hacemos nada. Igual que en el ejemplo anterior, buscamos una orden que termine con un error si no existe fichero y condicionamos la ejecucin de la orden touch fichero al error de la orden previa:
$ ls fichero || touch fichero
Tambin, al igual que en el ejempo anterior podramos hacer que si el proceso make modules falla, se borraran todos los ficheros temporales que se crean:
make modules || rm -r *.o
Ejecucin simultnea
Otra posibilidad de ejecucin tambin posible es lanzar varios procesos simutneamente en segundo plano; basta escribir uno a continuacin de otro en la lnea de rdenes separados por "&". Este es el smbolo que se utiliza para indicar que el proceso se tiene que ejecutar en segundo plano, pero tambin acta como separador para la ejecucin de distintas rdenes. por ejemplo :
[pfabrega@port pfabrega]$ sleep 10 & sleep 20 & sleep 15 & [pfabrega@port pfabrega]$ ps axu USER PID %CPU %MEM VSZ RSS TTY STAT START TIME COMMAND root 1 0.1 0.2 1408 540 ? S 22:19 0:04 init [3] ... pfabrega 1263 0.3 0.2 1624 516 pts/4 S 23:24 0:00 sleep 10 pfabrega 1264 0.0 0.2 1624 516 pts/4 S 23:24 0:00 sleep 20 pfabrega 1265 0.0 0.2 1624 516 pts/4 S 23:24 0:00 sleep 15 pfabrega 1266 0.0 0.3 2732 848 pts/4 R 23:24 0:00 ps axu
Esta caracterstica puede ser interesante en diferentes situaciones: Quemos enviar una secuencia de rdene s a segundo plno
(mkdir copiaseg; cp -r ./original/* ./copiaseg; rm -r ./original~; rm -r ./original.tmp) &
Resultado de la ejecucin de una orden Es habitual necesitar almacenar el resultado de la ejecucin de una orden en una variable en lugar de que se dirija a la salida estndar o simplemente ejecutar como orden el resultado de otra orden.
Comillas invertidas `
Las comillas invertidas consideran una orden lo que tengan dentro y lo ejecutan devolviendo el resultado como lneas de texto a la shell. Por ejemplo:
$ A="ls /bin" $ `echo $A` arch consolechars ed igawk mount rpm tar ash cp egrep ipcalc mt rvi tcsh ash.static cpio ex kill mv rview touch awk csh false ln netstat sed true basename date fgrep loadkeys nice setserial umount bash dd gawk login nisdomainname sfxload uname bash2 df gawk-3.0.6 ls ping sh usleep bsh dmesg grep mail ps sleep vi cat dnsdomainname gtar mkdir pwd sort view chgrp doexec gunzip mknod red stty ypdomainname chmod domainname gzip mktemp rm su zcat chown echo hostname more rmdir sync Tambin podramos haber puesto: $ A="ls /bin" $ B=`$A` $ echo $B arch ash ash.static awk basename bash bash2 bsh cat chgrp chmod chown consolechars cp cpio csh date dd df dmesg dnsdomainname doexec domainname echo ed egrep ex false fgrep gawk gawk-3.0.6 grep gtar gunzip gzip hostname igawk ipcalc kill ln loadkeys login ls mail mkdir mknod mktemp more mount mt mv netstat nice nisdomainname ping ps pwd red rm rmdir rpm rvi rview sed setserial sfxload sh sleep sort stty su sync tar tcsh touch true umount uname usleep vi view ypdomainname zcat
El operador $()
La shell bash proporciona el operador $() similar a las comillas invertidas. Ejecuta como orden los que haya entre parntesis y devuelve su resultado. El mecanismo de funcionamiento es idntico, con la ventaja de poder anidar operadores.
Programas de shell
Adems de las anteriores posibilidades tambin se pueden agrupar una serie de rdenes en un fichero de texto que se ejecutarn consecutivamente siguiendo el flujo determinado por rdenes de control similares a cualquier lenguaje de programacin. Estos ficheros se conocen como scripts, guiones o simplemente programas de shell. A las rdenes agrupadas en ficheros tambin se le aplican todas las caractersticas descritas anteriormente. No olvidemos que para un sistema unix, una lnea leda de un fichero es idntica a una lnea leda desde el teclado, una serie de caracteres terminado por un carcter de retorno de carro. Cualquier forma de ejecucin que se pueda dar en la lnea de rdenes tambin se puede incluir en un fichero de texto, con lo que facilitamos su repeticin. Y si por ltimo aadimos las estructuras que controlan el flujo de ejecucin y ciertas condiciones lgicas, tenemos un perfecto lenguaje de programacin para administrar el sistema con toda facilidad. Un administrador que sabe cual es su trabajo cotidiano, realizar copias de seguridad, dar de alta o baja usuarios, comprobar que los servicios estn activos, analizar log de incidencias, configurar cortafuegos, lanzar o parar servicios, modificar configuraciones, etc., normalmente se crear sus script personalizados. Algunos los utilizar cuando sea necesario y para otros programar el sistema para que se ejecuten peridicamente. La programacin en shell es imprescindible para poder administrar un sistema Unix de forma cmoda y eficiente. Vemos un primer ejemplo:
#!/bin/bash echo Hola Mundo
Puestas estas dos lnea en un fichero de texto con permiso de ejecucin, al ejecutarlo escribira en pantalla "Hola Mundo". La primera lnea, como veremos con posterioridad, indica qu shell es la que interpreta el programa de shell.
subshell
Para la ejecucin de programas de shell, la shell se encarga de interpretar unas rdenes en unos casos o de lanzar el programa adecuado en otros casos. En general, cuando lanzamos la ejecucin de un conjunto de rdenes agrupadas en un programa de shell, se abre una nueva shell (subshell hija de la anterior) que es la encargada de interpretar las
rdenes del fichero. Una vez concluida la ejecucin esta subshell muere y volvemos a la shell inicial. Esto es importante tenerlo presente para saber el comportamiento de los programas. Por ejemplo, los cambios hechos en las variables de shell dentro de un programa no se conservan una vez concluida la ejecucin. Vamos a ilustrar este comportamiento con nuestro primer ejemplo de programa de shell: Creamos un programa en un fichero de texto, lo ejecutamos, comprobamos que se crea una nueva shell y que los cambios en las variables hechos dentro del programa no se mantienen una vez concluido. Editamos un fichero llamado "pruebashell" con el siguiente contenido:
echo "******** GUION ********" echo "el valor previo de VAR es ** $VAR **" VAR="valor asignado dentro del guion" echo "Ahora VAR vale ** $VAR **" ps echo "******** FIN DEL GUION ********"
Con este guion mostramos el valor previo de una variable llamada VAR, le asignamos un valor nuevo y tambin los mostramos. Para verificar que se lanza una nueva shell mostramos la lista de procesos con la orden ps. Una vez editado el fichero tendremos que asignarle el permiso de ejecucin
$ chmod u+x pruebashell
despus asignamos un una valor a la variable VAR para comprobar como cambia. Adems tendremos que exportarla par que la shell hija pueda heredarla:
$ export VAR="valor previo"
Ahora mostramos la lista de procesos para ver cuantas shell tenemos abiertas:
$ ps
o bien
$ ps |wc -l
$ echo $VAR
Podremos observar como aparece una shell ms. Si la variable VAR est exportada veremos como muestra el valor que asignamos antes de ejecutar el guion y como muestra el que le asignamos dentro del guion. Y al final, al mostrar la variable VAR, observamos como nos muestra el valor que tena antes de ejecutar el guion; el guion no ha modificado la variable. Este mismo comportamiento se puede aplicar a la orden cd. Veamos el siguiente ejemplo, un simple script que cambia de directorio. Editamos el fichero llamado "cambia" con el siguiente contenido:
echo "cambiando de directorio" cd /tmp echo "estamos en:" pwd
Es decir, el script simplemente cambia al directorio /tmp. Una vez editado le asignomos el permiso de ejecucin
$ chmod u+x cambia
ejecutamos el script
$ ./cambia
y observamos como estamos situados en el mismo directorio que antes de la ejecucin del guion. Por qu ocurre todo esto?, Porque todos los cambios se realizan en la subshell que ha interpretado el guion.
Para aadir un comentario a un programa de shell se utiliza el carcter #. Cuando la shell interprete el programa ignorar todo lo que haya desde este carcter hasta el final de la lnea. Por otro lado si nos vemos obligados a partir un lnea en dos o ms, tendremos que finalizar cada lnea de texto inconclusa con el carcter \. De esta forma la shell las ver todas ellas como si se tratara de una nica lnea.
Parmetros posicionales
En un programa de shell definen unas variables especiales, identificadas por nmeros, que toman los valores de los argumentos que se indican en la lnea de rdenes al ejecutarlo. Tras el nombre de un script se pueden aadir valores, cadenas de texto o nmeros separados por espacios, es decir, parmetros posicionales del programa de shell, a los que se puede acceder utilizando estas variables. La variable $0 contiene el parmetro 0 que es el nombre del programa de shell. Las variables $1, $2, $3, $4, ... hacen referencia a los argumentos primero, segundo, tercero, cuarto, ... que se le hayan pasado al programa en el momento de la llamada de ejecucin. Por ejemplo, si tenemos un programa de shell llamado parametros con el siguiente contenido:
echo $0 echo $1 echo $2 echo $3 al ejecutarlo $ parametros primero segundo tercero prametros primero segundo tercero
La sentencia shift
La sentencia shift efecta un desplazamiento de los parmetros posicionales hacia la izquierda un nmero especificado de posiciones. La sintaxis para la sentencia shift es:
$ shift n
donde n es el nmero de posiciones a desplazar. El valor predeterminado para n es 1. Hay que observar que al desplazar los parmetros hacia la izquierda de pierden los primeros valores, tantos como hayamos desplazado, al superponerse los que tiene a la derecha. Por ejemplo:
$ set uno dos tres cuatro $ echo $1 uno $ shift $ echo $1 dos $ shift $ echo $1 tres $ shift $ echo $1 cuatro
Operador {}
Hemos visto la forma de acceder a los diez primeros parmetros posicionales, pero para acceder a parmetros de ms de dos dgitos tendremos que usar una pareja { } para englobar el nmero.
$ echo ${10} $echo ${12}
El operador { } tambin se usa para delimitar el nombre de las variables si se quiere utilizarla incluida dentro de un texto sin separaciones:
$DIR=principal $ DIRUNO=directorio $ UNO=subdirectorio $ echo $DIRUNO directorio $ echo ${DIR}UNO
principalUNO
Variables predefinidas
Adems de las variables de shell propias del entorno, las definidas por el usuario y los parmetros posicionales en un shell existen otra serie de variables cuyo nombre est formado por un carcter especial, precedido por el habitual smbolo $.
Variable $*
* La variable $* contiene una cadena de caracteres con todos los parmetros posicionales de la shell activa excepto el nombre del programa de shell. Cuando se utiliza entre comillas dobles se expande a una sola cadena y cada uno de los componentes est separado de los otros por el valor del carcter separador del sistema indicado en la variable IFS. Es decir si IFS tiene un valor "s" entonces "$*" es equivalente a "$1s$2s...". Si IFS no est definida, los parmetros se separan por espacios en blanco. Si IFS est definida pero tiene un contenido nulo los parmetros se unen sin separacin.
Variable $@
@ La variable $@ contiene una cadena de caracteres con todos los parmetros posicionales de la shell activa excepto el nombre del programa de shell. La diferencia con $* se produce cuando se expande entre comillas dobles; $@ entre comillas dobles se expande en tantas cadenas de caracteres como parmetros posicionales haya. Es decir "$@" equivale a "$1" "$2" ...
Variable $#
# Contiene el nmero de parmetros posicionales excluido el nombre del probrama de shell. Se suele utilizar en un guion de shell para verificar que el nmero de argumentos es el correcto.
Variable $?
? Contiene el estado de ejecucin de la ltima orden, 1 para una terminacin con error o 0 para una terminacin correcta. Se utiliza de forma interna por los operadores || y && que vimos con anterioridad, se utilizar por la orden test que vermos ms adelante y tambin la podremos usar explcitamente.
Variable $$
$ contiene el PID de la shell. En un subshell obtenida por una ejecucin con (), se expande al PID de la shell actual, no al de la subshell. Se puede utilizar para crear ficheros con nombre nico, por ejemplo $$.tmp, para datos temporales.
Variable $!
! Contiene el PID de la orden ms recientemente ejecutada en segundo plano. Esta variable no puede ayudar a controlar desde un guion de shell los diferentes procesos que hayamos lanzado en segundo plano. Ejemplos Vemos algunos ejemplos a continuacin:
$ $ 5 $ a $ $ 3 $ a set a b c d e echo $# echo $* b c d e set "a echo $# echo $* b c d
b" c d
al ejecutarlo
$ ./ejvar1 PID TTY TIME CMD 930 pts/3 00:00:00 bash 1011 pts/3 00:00:00 bash 1012 pts/3 00:00:00 ps el PID es 1011
y vemos como muestra el PID de la shell que ejecuta el script. Como el PID del prodceso es nico en el sistema, este valor puede utilizarse para construir nombres de ficheros temporales nicos para un proceso. Estos ficheros normalmente se suelen situar en el directorio temporal /tmp. Por ejemplo:
miproctemp=/tmp/miproc.$$ . . . . rm -f $miproctemp
Esta expresin devuelve el contenido de variable si est definida y tiene un valor no nulo. Por ejemplo si la variable resultado inicialmente no esta definida:
$ echo ${resultado} $ echo "E1 resultado es: {resultado:-0}" E1 resultado es: O $ resultado=1 $ echo "E1 resultado es: ${resultado:-0}" E1 resultado es: 1
$ {variable: +valorpredeterminado} Por ejemplo: $ resultado=10 $ echo ${resultado:+5} 5 $ resultado= $ echo ${resultado:+30} $
Si el contenido de variable es no nulo, esta expresin devuelve dicho valor. Si el valor es nulo o la variable no est definida entonces el valor de la expresin es valorpredeterminado, el cual ser tambin asignado a la variable variable. Veamos un ejemplo en el que se supone que la variable resultado no est definida:
$ echo ${resultado} $ echo "El resultado es: ${resultado:=0}" E1 resultado es: 0 $ echo ${resultado} 0
Por ejemplo:
$ res=${resultado:? "variable no vlida''} resultado variable no vlida
En el caso de que la variable resultado no est definida o contenga un valor nulo, se mostrar el mensaje especificado en pantalla, y si esta instruccin se ejecuta desde un programa de shell, ste finalizar.
Variable no definida
Anlogo al caso anterior para el caso de que la variable no est definida. La expresin para ello es:
${variable?mensaje}
Extrae una subcadena de variable, partiendo de inicio y de tamao indicado por longitud. Si se omite longitud toma hasta el fin de la cadena original.
$ A=abcdef $ echo ${A:3:2} de $ echo ${A:1:4} bcde $ echo ${A:2} cdef
Corta texto de variable si variable comienza por texto. Si variable no comienza por texto variable se usa inalterada. El siguiente ejemplo muestra el mecanismo de funcionamiento:
$ A=abcdef $ echo ${A#ab} cdef $ echo ${A#$B} cdef $ B=abc $ echo ${A#$B} def $ echo ${A#cd} abcdef
Corta texto de variable si variable termina por texto. Si variable no termina por texto variable se usa inalterada. Vemos un ejemplo:
$ PS1=$ $PS1="$ " $ A=abcdef $ echo ${A%def} abc $ B=cdef $ echo ${A%$B} ab
Sustituye texto1 por texto2 en variable. En la primera forma, slo se reemplaza la primera aparicin. La segunda forma hace que se sustituyan todas las apariciones de texto1 por texto2.
$ A=abcdef $ echo ${A/abc/x} xdef $ echo ${A/de/x} abcxf
Evaluacin aritmtica
En habitual tener que efectuar evaluaciones de expresiones aritmticas enteras durante la ejecucin de un script de shell; por ejemplo para tener contadores o acumuladores o en otros casos. Hasta ahora habamos visto que esto lo podamos hacer con expr, pero hay otra forma ms cmoda: let La sintaxis de let es la siguiente:
let variable=expresin aritmtica
por ejemplo
let A=A+1
En algunas shell incluso podremos omitir la palabra let, aunque por motivos de compatibilidad esto no es aconsejable. Para evaluar expresiones reales, es decir con coma decimal, tendremos que usar otros mecanismos y utilidades que pueda proporcionar el sistema. En linux disponemos de la orden bc.
Por ejemplo, si queremos que sea la shell bash la que interprete nuestro script tendramos que comenzarlo por
#!/bin/bash
En ciertas ocasiones es interesante forzar que sea una shell concreta la que interprete el script, por ejemplo si el script es simple podemos seleccoinar una shell que ocupe pocos recursos como sh. Si por el contrario el script hace uso de caractersticas avanzadas puede que nos interese seleccionar bash como shell.
Despus de esto, el programa quedar esperando hasta que se le proporcione una cadena de caracteres terminada por un salto de lnea. El valor que se le asigna a variable es esta cadena (sin el salto de lnea final). La instruccin read tambin puede leer simultneamente varias variables:
read varl var2 var3 var4
La cadena suministrada por el usuario empieza por el primer carcter tecleado hasta el salto de lnea final. Esta lnea se supone dividida en campos por el separador de campos definido por la variable de shell IFS (Internal Field Separator) que de forma predeterminada es una secuencia de espacios y tabuladores. El primer campo tecleado por el usuario ser asignado a var1, el segundo a var2, etc. Si el nmero de campos es mayor que el de variables, entonces la ltima variable contiene los campos que sobran. Si por el contrario el nmero de campos es mayor que el de variables las variables que sobran tendrn un valor nulo. La segunda caracterstica es la posibilidad incluir un mensaje informativo previo a la lectura. Si en la primera variable de esta instruccin aparece un carcter "?". todo lo que quede hasta el final de este primer argumento de read, se considerar el mensaje que se quiere enviar a la salida estndar. Ejemplo:
read nombre?"Nombre y dos apellidos? " apl ap2
Para construir la expresin disponemos una serie de facilidades proporcionadas por la shell. Estas expresiones evalan una determinada condicin y devuelven una condicin que puede ser verdadera o falsa. El valor de la condicin actualiza la variable $? con los valores cero o uno para los resultados verdadero o falso. Pasamos a describir a continuacin estas condiciones, y tenemos que tener en cuenta que es importante respetar todos los espacios que aqu aparecen. -f fichero existe el fichero y es un fichero regular -r fichero existe el fichero y es de lectura -w fichero existe el fichero y es de escritura -x fichero existe el fichero y es ejecutable -h fichero existe el fichero y es un enlace simblico. -d fichero existe el fichero y es un directorio -p fichero existe el fichero y es una tubera con nombre -c fichero existe el fichero y es un dispositivo de carcter -b fichero existe el fichero y es un dispositivo de bloques -u fichero existe el fichero y est puesto el bit SUID -g fichero existe el fichero y est puesto el bit SGID -s fichero existe el fichero y su longitud es mayor que O -z s1 la longitud de la cadena s1 es cero -n s1 la longitud de la cadena s1 es distinta de cero s1=s2 la cadena s1 y la s2 son iguales s1!=s2 la cadena s1 y la s2 son distintas
n1 -eq n2 los enteros n1 y n2 son iguales. n1 -ne n2 los enteros n1 y n2 no son iguales. n1 -gt n2 n1 es estrictamente mayor que n2. n1 -ge n2 n1 es mayor o igual que n2. nl -lt n2 n1 es menor estricto que n2. nl -le n2 n1 es menor o igual que n2. Estas expresiones las podemos combinar con: ! operador unario de negacin -a operador AND binario -o operador OR binario Ejemplos:
$ $ $ $ $ $ test echo test echo test echo -f /etc/profile $? -f /etc/profile -a -w /etc/profile $? 0 -lt 0 -o -n "No nula'' $?
La orden test se puede sustituir por unos corchetes abiertos y cerrados. Es necesario dejar espacios antes y despus de los corchetes; este suele ser unos de los errores ms frecuentes.
Estructura de control
La programacin en shell dispone de las sentencias de control del flujo de instrucciones necesarias para poder controlar perfectamente la ejecucin de las rdenes necesrias.
Sentencia if
La shell dispone de la sentencia if de bifurcacin del flujo de ejecucin de un programa similar a cualquier otro lenguaje de programacin. La forma ms simple de esta sentencia es:
if lista_rdenes then lista_rdenes fi
fi, que es if alrevs, indica donde termina el if. En la parte reservada para la condicin de la sentencia if, aparece una lista de rdenes separados por ";". Cada uno de estos mandatos es ejecutado en el orden en el que aparecen. La condicin para evaluar por if tomar el valor de salida del ltimo mandato ejecutado. Si la ltima orden ha terminado correctamente, sin condicin de error, se ejecutar la lista de rdenes que hay tras then. Si esta rden ha fallado debido a un error, la ejecucin contina tras el if. Como condicin se puede poner cualquier mandato que interese, pero lo ms habitual es utilizar diferentes formas de la orden test. Por ejemplo:
if [ -f mifichero ] then echo "mifichero existe" fi Pero tambin podemos poner: if grep body index.html then echo "he encontrado la cadena body en index.html" fi
Como en cualquier lenguaje de programacin tambin podemos definir las acciones que se tieenen que ejecutar en el caso de que la condicin resulte falsa:
if lista_rdenes then lista_rdenes else lista_rdenes fi
Por ejemplo:
if [ -f "$1" ] then pr $1 else echo "$1 no es un fichero regular" fi
Cuando queremos comprobar una condicin cuando entramos en el else, es decir, si tenemos else if es posible utilizar elif. Vemos un ejemplo:
if [ -f "$1" ] then cat $1 elif [ -d "$1" ] then ls $1/* else echo "$1 no es ni fichero ni directorio" fi
Sentencia while
La sentencia while tiene la siguiente sintaxis:
while lista_rdenes do lista rdenes done
La lista de rdenes que se especifican en el interior del bucle while se ejecutar mientras que lista_rdenes devuelva un valor verdadero, lo que significa que la ltima orden de esta lista termina correctamente. Vemos un ejemplo:
I=0 while [ ${resp:=s} = s ] do I=\`{ }expr $I + 1\`{ } echo $I read resp?"Quiere usted continuar(s/n)? " done
Sentencia until
La sentencia until similar a while, es otro bucle que se ejecutar hasta que se cumpla la condicin, es decir, hasta que la lista de rdenes termina correctamente. Su formato es el siguiente:
until lista_rdenes do lista rdenes done
Sentencia for
La sentencia for repite una serie de rdenes a la vez que una variable de control va tomando los sucesivos valores indicado por una lista de cadenas de texto. Para cada iteracin la variable de control toma el valor de uno de los elementos de la lista. La sintaxis de for es la siguientes
for variabl in lista do lista mandatos done
lista es una serie de cadenas de texto separadas por espacios y tabuladores. En cada iteracin del bucle la variable de control variable toma el valor del siguiente campo y se ejecuta la secuencia de mandatos lista_mandatos. Ejemplo:
for i in $* do echo $i done
Sentencia case
La sentencia case proporciona un if mltiple similar a la sentencia switch de C. El formato bsico de esta sentencia es el siguiente:
case variable in patrn1) lista_rdenes1 ;; patrn2) lista_rdenes2 ;; ... patrnN) lista_rdenesN;; esac
La shell comprueba si variable coincide con alguno de los patrones especificados. La comprobacin se realiza en orden, es decir empezando por patrn1 terminando por
patrnN. En el momento en que se detecte que la cadena cumple algn patrn, se ejecutar la secuencia de mandatos correspondiente hasta llegar a ";;". Estos dos puntos y comas fuerzan a salir de la sentencia case y a continuar por la siguiente sentencia despus de esac (esac es case alrevs). Las reglas para componer patrones son las mismas que para formar nombres de ficheros, as por ejemplo, el carcter "*" es cumplido por cualquier cadena, por lo que suele colocarse este patrn en el ltimo lugar, actuando como accin predeterminada para el caso de que no se cumpla ninguna de las anteriores. Ejemplo:
case "$1" in start) echo ;; stop) echo ;; status) echo ;; restart) echo ;; *) echo exit esac
-n "Ha seleccionado start " -n "Ha seleccionado stop " -n "Ha seleccionado stop " -n " Ha seleccionado restart " "No es una opcin vlida" 1
En cadenaopciones se sitan las opciones vlidas del programa. Cada letra en esta cadena significa una opcin vlida. El carcter ":" despus de una letra indica que esa opcin lleva un argumento asociado o grupo de argumentos separados por una secuencia de espacios y tabuladores. Esta orden se combina con la sentencia while para iterar por cada opcin que aparezca en la lnea de rdenes en la llamada al programa. En variable se almacena el valor de cada una de las opciones en las sucesivas llamadas. En la variable OPTIND se guarda el ndice del siguiente argumento que se va a procesar. En cada llamada a un programa de shell esta variable se inicializa a 1. Adems: Cuando a una opcin le acompaa un argumento, getopts lo sita en la variable OPTARG. Si en la lnea de mandatos aparece una opcin no vlida entonces asignar "?" a variable. Veamos un ejemplo:
while getopts ab:cd opcion do case $opcion in a) echo "Opcin a" ;; b) echo "Opcin b con argumento $OPTARG" ;; c) echo "Opcin c" ;; d) echo "Opcion d" ?) echo "Uso: $0 -acd -b opcion " esac
Observamos como hemos desplazados todos los argumentos con shift para enerlos disponibles con los parmetros posicionales y descartando las opciones previamente analizadas.
los expande siguiendo las normas de expansin de la shell, separndolos por espacios y trata de ejecutar la cadena resultante como si fuera cualquier orden. Esta instruccin se debera utilizar cuando: Pretendemos examinar el resultado de una expansin realizada por la shell. Para encontrar el valor de una variable cuyo nombre es el valor de otra variable. Ejecutar una lnea que se ha ledo o compuesto internamente en el programa de shell.
Funciones
Ciertas shell como bash permiten la declaracin de funciones para agrupar bloques cdigo como en un lenguaje de programacin convencional. La forma de declarar un funcin es
function mi_funcion { cdigo de la funcin }
Para realizar la llamada a la funcin slo tenemos que usar su nombre. Adems las funciones pueden tener argumentos en su llamada. No es necesario declarar los parmetros en la declaracin de la funcin, basta usar las variables $1, $2, etc. dentro de la definicin de las instrucciones y sern los parametros en su orden correspondiente. Para llamar a una funcin con argumentos no se usan los habituales parntesis. Ejemplos:
#!/bin/bash
function terminar { exit 0 } function saludo { echo Hola Mundo! } saludo terminar
Otro ejemplo:
#!/bin/bash function terminar { exit 0 } function saludo { echo $1 } saludo Hola saludo Mundo terminar
En este ltimo ejemplo podemos observar el uso de una funcin con argumentos, tanto en la delaracin como en la llamada.
Por ejemplo, hacemos un script que mueva todos los ficheros pasado como argumento al directorio ./papelera:
for i in $* do if [ -f $i ] then mv $i ./papelera fi done
En este caso estamos redirigiendo la entrada estndar de la orden read, que es el teclado, por un fichero. Al igual que en el caso del teclado, la lectura se realizar hasta que se encuentre un salto de lnea. Observamos como la redireccin se realiza tras el final de la sentencia while. Otra forma posible para hacer esto mismo sera:
Este mtodo difiere ligeramente del anterior, ya que al utilizar una tubera creamos una nueva shell con lo cual puede ocurrir que no se conserven ciertos valores de las variables de shell.
Ahora usamos sed para sustituir cualquier secuencia de espacios en blanco ([ ][ ]*) por un separador ";":
$ /sbin/ifconfig |sed "s/[ ][ ]*/;/g" eth0;Link;encap:Ethernet;HWaddr;00:90:F5:08:37:E4; ;inet;addr:192.168.1.5;Bcast:192.168.1.255;Mask:255.255.255.0 ;UP;BROADCAST;MULTICAST;MTU:1500;Metric:1 ;RX;packets:0;errors:0;dropped:0;overruns:0;frame:0 ;TX;packets:5241;errors:0;dropped:0;overruns:0;carrier:0 ;collisions:0;txqueuelen:100; ;Interrupt:10;Base;address:0x3200;
Ahora podramos cortar de forma exacta el campo que nos interese, por ejemplo:
$ /sbin/ifconfig | sed "s/[ ][ ]*/:/g" | grep inet | cut -f4 -d: 192.168.1.5 127.0.0.1
Vemos paso a paso la anterior orden compuesta: Primero ejecutamos la orden ifconfig Sustituimos los espacios en blanco por ":" Buscamos la lnea que contenga la palabra inet Cortamos el campo 4 usando ":" como separador.
Explica que realizara cada una de las siguientes rdenes ejecutadas en secuencia
$ $ $ $ A=\$B B=ls echo $A eval $A
Asignar el limo parmetro posicional a la variable ULT Realizar un programa que escriba los 20 primeros nmeros enteros. Realizar un programa que numere las lneas de un fichero Realizar un programa que tomando como base el contenido de un directorio escriba cada elemento contenido indicando si es fichero o directorio. Realizar un programa que muestre todos los ficheros ejecutables del directorio activo. Modificar el programa anterior para que indique el tipo de cada elemento contenido en el directorio activo: fichero, directorio, ejecutable,... Ejercicios resueltos sobre ficheros y directorios Guion de shell que genere un fichero llamado listaetc que contenga los ficheros con permiso de lectura que haya en el directorio /etc:
for F in /etc/* do if [ -f $F -a -r $F ] then echo $F >> listaetc fi done
Hacer un guion de shell que, partiendo del fichero generado en el ejercicio anterior, muestre todos los ficheros del directorio /etc que contengan la palagra "procmail":
while read LINEA do if grep procmail $L >/dev/null 2>&1 then echo $L fi done <listaetc
Hacer un guion de shell que cuente cuantos ficheros y cuantos directorios hay en el directorio pasado como argumento:
DI=0 FI=0 for I in $1/* do if [ -f $I ] then let FI=FI+1 fi if [ -d $I ] then let DI=DI+1
fi done
Hacer un guion de shell que compruebe si existe el directorio pasado como argumento dentro del directorio activo. En caso de que exista, que diga si no est vaco.
if [ -d $1 ] then echo "$1 existe" N=$(ls | wc -l) if [ $N -gt 0 ] then echo "S1 no est vacio, contiene $N ficheros no ocultos" fi fi
Hacer un guion de shell que copie todos los ficheros del directorio actual en un directorio llamado csg. Si el directorio no existe el guion lo debe de crear.
if [ ! -d csg ] then mkdir csg fi cp * csg
Hacer un script que muestre el fichero del directorio activo con ms lneas:
NLIN=0 for I in * do if [ -f $I ] then N=$(wc -l $I) if [ $N -gt $NLIN ] then NOMBRE=$I NLIN=$N fi fi done echo "$NOMBRE tiene $NLIN lineas"
-type f)
-type d)
Introduccin
Todos los scripts estn diseados para ejecutarse en el shell bash, pero tampoco debe suponer demasiadadificultad utilzarlos en otra shell de tipo Bourne. En estos ejemplos adems, de una forma prctica, vamos a recordar, repasar e introducir ciertos conceptos tericos. De todas formas, para aprovechar estos ejemplos y ganar tiempo y conocimientos, deberas leer el captulo de programacin en shell. Ni que decir tiene que es necesario conocer todas las rdenes de Unix que vamos a utilizar en los ejemplos propuestos.
He intentado secuenciar los script por temas y por dificultad aunque esto no es siempre posible.
Parmetros posicionales y variables internas Script que muestra los tres primeros parmetros pasado en la lnea de rdenes
Los parmetros que se pasan a un script los podemos encontrar en las variables $1, $2, ... Tambin resulta til $0 que contiene el nombre del script. En este caso bastara:
echo $1 $2 $3
o
echo $(EN)
Script que escribe todos los parmetros pasados en una lnea y separados por el signo +
Si vemos el ejemplo anterior podemos deducir que si ponemos IFS=+ ahora el separador de campos ser el que nos interesa:
IFS=+ echo $*
condiciones (test) Script que comprueba que hay dos parmetros posicionales
En este script tenemis que tomar una decisin dependiendo del contenido de la variable $# como vimos en un ejemplo anterior.
if [ $# -eq 2 ] then echo "Hay dos parmetros" fi
Todo lo que pongamos tras "then" y hasta el (else) "fi" se ejecurar cuando la condicin sea cierta. Este ejmplo resulta particularmente til para construir scripts que deban tener un nmero exacto de parmetros.
Destacamos aqu:
if [ $1 == $2 ]
Script que escribe IGUALES si los dos parmetros posicionales pasado son iguales y DISTINTOS en otro caso
Ahora empezamos con las comparacines para poder tomar una decisin dentro del script. Segn el enunciado vamos a suponer que el script tiene slo dos argumentos; despus vamos a comparar los argumentos y mostrar lo que se pide:
if [ $# -eq 2 ] then echo "Uso: $0 arg1 arg2" exit fi if [ $1 == $2 ] then echo "IGUALES" else echo "DISTINTOS" fi
Analizamos el programa:
En primer lugar, en general vemos que un if puede tener o no un "else" asociado, como en cualquier lenguaje. Y con ms detalle, vemos las lnea ms significativas: Mostramos un mensaje sobre como se usa el script. Observamos que $0 es el nombre del script.
echo "Uso: $0 arg1 arg2"
Da por terminado el script, puesto que al no tener los argumentos necesarios no debera continuar su ejecucin.
exit
Script que dice Es un fichero si el argumento pasado es un fichero regular. En otro caso debe escribir No es un fichero.
El objetivo de este script es hacer prcticas con las condiciones que proporciona bash para realizar comprobaciones del sistema de ficheros, comprobar ficheros, directorios, permisos, etc. En primer lugar deberemos verficar el nmero de argumentos como veamos en el ejemplo anterior y postriormente comprobamos si el argumento se corresponde con un fichero:
if [ $# -eq 1 ] then echo "Uso: $0 fichero" exit fi if [ -f $1 ] then echo "$1 es un fichero" else echo "$1 NO es un fichero" fi
La opcin "-f" de la orden test o "[ ]" devuelve "Verdadero" si el valor que indicamos se corresponde con un fichero regular. Aconsejo un repaso a las distintas condiciones para comprobar otros tipos de ficheros como pipes, dispositivos o enlaces. Si tenemos ficheros con nombres que contengan espacios y otro metacarcter, el argumento debera estar entre comillas dobles:
if [ $# -eq 1 ] then echo "Uso: $0 fichero" exit fi if [ -f "$1" ]
Script que dice Es un directorio si el argumento pasado es un directorio. En otro caso debe escribir No es un directorio.
Este script es idntico al anterior salvo que tenemos que modificar la condicin para comprobar directorios:
if [ $# -eq 1 ] then echo "Uso: $0 fichero" exit fi if [ -d $1 ] then echo "$1 es un directorio" else echo "$1 NO es un directorio" fi
Script que dice Es un enlace si el argumento pasado es un enlace simblico. En otro caso debe escribir No es un enlace simblico.
Este ejemplo tambin es idntico a los anteriores cambiando la condicin:
if [ $# -eq 1 ] then echo "Uso: $0 fichero" exit fi if [ -h $1 ] then echo "$1 es un enlace simblico" else
Script que escribe las caractersticas del fichero o directorio, pasado como argumento: tipo y permisos.
En este ejemplo vamos a ver un resumen de distintas condiciones que podemos consulatr al sistema de ficheros. Adems, vamos a ver como podemos concatener cadenas de caracteres.
if [ $# -eq 1 ] then echo "Uso: $0 fichero" exit fi FICHERO="$0: " if [ -f $1 ] then FICHERO="$FICHERO fichero" else
if [ -d $1 ]
then FICHERO="$FICHERO directorio" else
Como novedad de este script destacamos un "if" anidado, un "if" dentro de otro.
fi
if [ -d $1 ] then echo "$1 directorio"
fi
if [ -c $1 ] then echo "$1 dispositivo carcter"
fi
if [ -b $1 ] then echo "$1 dispositivo bloque"
fi
if [ -p $1 ] then echo "$1 tubera con nombre"
fi
Bucles
Hasta ahora todos los script se ejecutaban de una sola pasada, ahora vamos a ver como podemos repetir un bloque de instrucciones un determinado nmero de veces.
pero no se trata de eso, sino de comprender los bucles en shell. En primer lugar vamos a hacer este ejemplo con un bucle "for" que va tomando sucesivamento los valores que indicamos:
for N in 1 2 3 4 5 do echo $N done
Para hacer este script con un bucle while tendremos que utilizar un contador para saber cuantas veces lo hemos repetido.
CONTADOR=0 while [ CONTADOR -le 5 ] do let CONTADOR=CONTADOR+1 echo $CONTADOR done
Cuando estemos diseando un script el sentido comn nos debe orientar sobre qu tipo de bucle debemos elegir.
Script que suma todos los nmeros pasados como argumentos y escriba la operacin y el resultado: p.e. 1 + 2 +3=6
Ante todo hay quetener claro el ejemplo anterior. Para este ejemplo tambin vamos a usar la variable $* y el separador de campos, que en nuestro caso nos interesa que sea "+" para que lo incluya entre cada uno de los argumentos simulando la operacin:
ACUMULADOR=0 for N in $(EN) do let ACUMULADOR=ACUMULADOR+N done
IFS=+
echo "$*=$ACUMULADOR"
Script que escribe el tipo de todos los ficheros pasados como argumento.
En este ejercicio lo que pretendemos es ver como podemos procesar mltiples argumentos de idntica forma, es decir, nosotros al ejecutar el script lo que hacemos es poner una lista de ficheros y para cada uno de ellos realizar las mismas comprobaciones. Por lo dems las comprobaciones seran idnticas a las que vimos en un ejericicio anterior por lo que, para no extendernos innecesariamente, vamos a poner slo si es un fichero regular o directorio:
for F IN $(EN) do if [ -f $F ] then echo "es un fichero" fi if [ -d $F ] then echo "es un directorio" fi done
Script que renombra todos los ficheros cuyo nombre sea de la forma texto.html y en otro que sea de la forma texto.xhml.
Este script slo cambia el nombre a los ficheros del directorio activo
for fichero in *.html do nombre=$(bsasename $fichero .html) mv $fichero $nombre.xhtml done
done
Una forma de procesar todos los ficheros de una rama del rbol de directorios es 'for fichero in $(find . -name "*.html" -type f)', donde "fichero" va tomando la ruta relativa de cada uno de los ficheros encontrados. Este ejemplo puede servir como base para un script que realice ciertas copias de seguridad.
Lectura de ficheros
En muchos casos vamos a necesitar leer un fichero lnea a lnea desde un script, y como el cdigo siempre va a ser muy parecido lo recordamos para los siguientes ejemplos. Este cdigo lee el fichero /ruta/datos lnea a lnea y la muestra en pantalla:
while read LINEA do echo $LINEA done < /ruta/datos
let NUM=NUM+1
done < /etc/passwd
Script que saca una lista de todos los usuarios que no tienen un directorio con su nombre en /home.
while read LINEA do $usuario=$(echo $LINEA|cut -f1 -d:) if ! [ -f /home/usuario ] then echo $usuario fi done </etc/passwd
Comentarios y recordatorio: el operador $() de bash ejecuta una orden y devuelve su salida estndar. Tambin se pueden usar comillas invertidas (`) para ejecutar una orden y obtener el resultado. Cada lnea la leemos del fichero /etc/passwd cut -f1 -d: cortara el campo uno ( -f 1) obtenido por un separador : de los valores de la entrada estndar, que es el resutl . Como el primer campo del fichero /etc/passwd es el nombre de usuario, entonces estaramos obteniendo este nombre.
Caso general
La base de datos de usuarios no tiene por qu estar completa en el fichero /etc/passwd, puede estar sobre otros soportes como NIS, LDAP, Winbind, db, ... En consecuencia la forma de obtener la lista real de usuarios es ejecutar la orden "getent passwd", as como para obtener la lista de grupos ejecutamos "getent group". En este caso, nuestro anterior script podra quedar como:
for LINEA in $(getent passwd) do $usuario=$(echo $LINEA|cut -f1 -d:) if ! [ -f /home/usuario ] then echo $usuario fi done
Corolario
Esta forma de obtener una parte de una cadena para almacenarla en una variable va a ser de uso frecuente en el desarrollo de scripts.
Suponemos que tenemos un fichero llamado cantidades en el que cada lnea est formada por cantidades separadas por un espacio y tenemos que hacer un guion que sume cada lnea.
while read LINEA do TL=0 for CIFRA in $LINEA do let TL=TL+CIFRA done echo $TL done <fichero_cifras.txt
Comentarios Cada lnea ser de la forma "11 22 33 44" lo que permite utilzar un for para ir procesando individualmente cada uno de los componentes. Muy parecido a ajercicios anteriores done utilizbamos $(EN). Si quisiramos calcular tambin el total del fichero haramos; TT=0
while read LINEA do TL=0 for CIFRA in $LINEA do let TL=TL+CIFRA
echo $TT
Hacer un guion que separe las frases del fichero pasado como argumento y las guarde en otro fichero nuevo.
Este script es tan simple que se puede hacer en una sola lnea. En realidad separar las frases consiste en aadir un retorno de carrlo tras cada fin de frase, que es un punto. Evidentemente esto lo vamos a hacer utilzando expresiones regulares.
perl -p -e "s/\./.\n/g" $1 >$1.separado.txt
Comentarios: El primer punto corresponde la a expresin regular y por tanto necesitamos protegerlo para que no se interprete como un carcter cualquiera, que es su significado como expresin regular. El segundo punto no es necesario protegerlo porque no forma parte de la expresin regular, forma parte del texto sustiutivo, un punto y un retorno de carro.
Guion que sustituye la palabra "á" por "" en los ficheros pasados como argumento.
for F in $(EN)
do perl -p -e "s/á//g" $F >$F.mod mv $F.mod $F done Comentarios No podemos redirigir al mismo fichero que estamos editando, tenemos que utilzar un fichero intermedio.
Guion que transforma un fichero cuyas lnea tengan el formato nombre:apellidos:edad en el formato INSERT INTO TABLA VALUES ("nombre",apellidos",edad);
while read LINEA do NOMBRE=$(echo $LINEA|cut -f1 -d:) APELLIDOS=$(echo $LINEA|cut -f2 -d:) EDAD=$(echo $LINEA|cut -f3 -d:) echo "INSERT INTO tabla VALUES ('$NOMBRE','$APELLIDOS','$EDAD');">fichero.sql done<fichero.dat
Prcticas con comandos Unix Guion que diga si el usuario cuyo nombre pasamos como argumento est o no conectado.
if who|grep $1 >/dev/null 2>&1 then echo $1 conectado else echo $1 NO conectado fi
Recordatorio: Podemos usar la ejecucin de una orden como una condicin para el if, de forma que si la orden termina correctamente toma el valor verdadero y falso en otro caso. Cuando un orden termina fija la variable $? con el cdigo de terminacin que es la misma que utiliza if para tomar la decisin.
Guion que diga qu usuarios de la lista que pasamos como argumento estn o no conectados.
for usuario in $(EN)
Este ejercicio es una generalizacin del anterior. Aqu podemos observar una forma de ampliar un script usando la variable $(EN) que contiene la lista de argumentos. Ahora queremos comprobar que el usuario realmente existe:
for usuario in $(EN)
do if getent passwd| grep $usuario >/dev/null 2>&1 if who|grep $usuario >/dev/null 2>&1
then echo $usuario conectado else echo $usuario NO conectado fi done
Slo queda por mencionar que hemos redirigido las salida estndar y de errores a /dev/null, primero redirigimos la salida estndar (>/dev/null) y luego redirigimos la salida de errores a la salida estndar (2>&1). Si la bsqueda slo nos interesa sobre /etc/passwd la condicin quedara como: