Programación en shell scripts

Anuncio
Aventurandose por los shell scripts
En esta anotación voy a intentar dar unas breves ideas para aquellos que quieran hacer sus primeros
pinitos en la programación de shell scripts.
La shell utilizada será bash pues además de ser la que se utiliza por defecto en GNU/Linux es también la
más potente, al menos de las que conozco, csh, ksh y bash.
Un shell script no es más que un fichero que contiene un conjunto de comandos y que puede ser ejecutado
desde un terminal.
Cuando se ejecuta un shell script realmente lo que se hace es lanzar una shell (bash, ksh, csh o la que sea)
que se encarga de interpretar y ejecutar las órdenas contenidas en el fichero del script.
Para ejecutar un script se puede hacer de varias maneras:
•
•
•
Dandole permisos de ejecución y lanzarlo escribiendo el path donde se encuentra y el nombre. Por
ejemplo si tenemos un script que se llama sx99 y que se encuentra en el directorio /opt/scripts para
lanzarlo (una vez dados los permisos de ejecución) tendrías que hacer:
/opt/scripts/sx99
Dandole permisos de ejecución y situarlo en alguno de los directorios de la variable PATH (por
ejemplo en /usr/local/bin). Es similar al anterior pero para lanzarlo no hay que poner la ruta de
donde se encuentra ya que es suficiente con poner su nombre
Lanzando explícitamente la shell y pasandoe por parámetro el script (no son necesarios permisos
de ejecución). En el ejemplo anterior:
bash /opt/scripts/sx99
La primera línea que se suele poner en un script (no es obligatoria) es una línea que le indica al sistema
cuál es el intérprete que debe utilizar para ejecutar nuestro script (si no se pone nada utilizará el shell por
defecto que como hay he mencionado suele ser bash).
#!/bin/bash
En este caso se está indicando que el intérprete será bash y que está ubicado (bash) en /bin.
Si por ejemplo nuestro script fuera para ksh y ksh se encontrara en /usr/bin deberiamos poner en la
primera línea:
#!/usr/bin/ksh
Cuando se quieren añadir comentarios, es decir líneas en el fichero de script que el intérprete debe ignorar
se preceden por el carácter #
# Lanzo el comando ls con el parámetro -l
ls -l
Si queremos asignar valores a variables lo haremos mediante el = pero teniendo cuidado de no dejar
ningun espacio entre el = y el nombre de la variable y el = y el valor que le damos.
# Así no
otravariable = 1
# Así sí
variable=1
Para acceder al valor contenido en otra variable utilizaremos el $ delante del nombre de la variable (es
una práctica recomendable utilizar el nombre de la variable encerrado entre {})
# Esto muestra por la salida estandar,
# en principio la pantalla, el valor de variable
echo $variable
# Esto tambien pero a mi me gusta mas que la anterior
echo ${variable}
Si después de ejecutar un comando queremos comprobar si ha ido mal podemos utilizar $?
inmediatamente después de ejecutar el comando. Si no es 0 es que algo ha ido mal.
# Esta ejecucion ira bien y por tanto mostrara un 0
ls -l
echo $?
# Esta ejecucion ira mal (siempre que no exista un fichero
# llamado dsafhjsdf) y mostrara un valor distinto de 0
ls -l dsafhjsdf
echo $?
Si queremos preguntar por alguna variable para hacer o no hacer algo utilizaremos la sentencia if
ls -l dsafhjsdf
retorno=$?
if [ ${retorno} -ne 0 ]; then
echo "Ha dado el error ${retorno} la ejecucion de ls -l dsafhjsdf"
else
echo "Ha ido bien la ejecucion de ls -l dsafhjsdf"
fi
Hay que darse cuenta que un if se cierra con un fi, que puede tener un else y que la condición va entre [].
Es necesario que después del [ y antes del ] debe existir al menos un espacio. El ; que va entre el ] y el
then no es necesario pero si no se pone la palabra then debe ir en la siguiente línea.
ls -l dsafhjsdf
retorno=$?
if [ ${retorno} -ne 0 ]
then
echo "Ha dado el error ${retorno} la ejecucion de ls -l dsafhjsdf"
else
echo "Ha ido bien la ejecucion de ls -l dsafhjsdf"
fi
Si lo que se van a comparar son números (como en el ejemplo), para comparar se utiliza lo siguiente:
•
•
•
•
•
•
-eq -> igual
-ne -> distinto
-lt -> menor
-le -> menor o igual
-gt -> mayor
-ge -> mayor o igual
Mientras que si lo que se van a comparar son textos lo que se utiliza es:
•
•
= -> igual
== -> igual (como el anterior aunque yo prefiero esta sintaxis que es la de C para diferenciarlo de
la asignacion)
•
!= -> distinto
En el caso de la comparación de textos conviene utilizar las variables encerradas entre “”
nombre="Pedro"
if [ "${nombre}" == "Antonio" ]; then
echo Es Antonio
else
echo No es Antonio, es ${nombre}
fi
Si una variable no está definida o está vacía se puede detectar utilizando la opción -z dentro del if:
if [ -z ${apellido} ]; then
echo La variable apellido no existe o no tiene valor
else
echo El apellido es ${apellido}
fi
En ocasiones querremos repetir algo, por ejemplo convertir de formato todas las imágenes jpg de un
directorio y pasarlas a png:
for i in *.jpg ; do
convert "${i}" "`echo $i | sed "s/.jpg$/.png/g`"
done
En este caso mediante *.jpg sacamos una lista de los archivos del directorio en el que se ejecute que
terminan en .jpg, la variable i va tomando en cada iteración del bucle el nombre de cada uno de estos
archivos.
Luego utilizamos la utilidad convert para pasar del jpg a png y para ello le pasamos como primer
parámetro el fichero jpg (${i}) encerrado además entre “” para evitar problemas con los espacios que
pueda tener el nombre y utilizamos sed para construir el segundo parámetro que tendrá el mismo nombre
que el fichero jpg pero cambiando jpg por png.
Además utilizamos los acentos graves (`) para que al ejecutar un determinado comando nos asigne su
salida a una variable o como en el ejemplo lo ponga en el segundo parámetro del convert. Si por ejemplo
hubieramos querido guardar el nombre del nuevo fichero png en otra variable deberiamos hacer:
nombre_del_png=`echo $i | sed "s/.jpg$/.png/g`
o también se podría hacer sin utilizar el acento con:
nombre_del_png=$(echo $i | sed "s/.jpg$/.png/g)
También nos puede interesar, hay gente para todo ;), sacar los n primeros números de la serie de
Fibonacci (esa que empieza por 1 1 2 3 5 8 13 21 34 … y que tiene algo que ver con la capacidad
reproductora de los conejos).
Para ello podemos utilizar el bucle while y algunos otros apaños:
max=10
n_2=1
n_1=1
echo 1 - ${n_2}
echo 2 - ${n_1}
i=3
while [ ${i} -le ${max} ]; do
n=$((${n_2} + ${n_1}))
n_2=${n_1}
n_1=${n}
echo ${i} - ${n}
i=$((${i}+1))
done
En este caso utilizamos while para montar un bucle que se ejecutará mientras que se cumpla la condición,
en el ejemplo lo hará mientras la variable i (que comienza siendo 3) sea menor o igual que la variable
max (10 porque no me apetece sacar más de 10 números de Fibonacci).
Además el ejemplo sirve para ver como se hacen operaciones aritméticas en bash
n=$((${n_2} + ${n_1}))
i=((${i}+1))
Si no utilizasemos los dobles paréntesis bash no sabría que querriamos hacer una suma y lo que haría es
concatenar los valores como si de una cadena de texto se tratase, n valdría 1+1 en la primera iteración (no
2) e i valdría 3+1 (no 4).
Descargar