ShellScripting

bash manpage

http://linux-cd.com.ar/manuales/howtos/programacion-bash/Bash-Prog-Intro-COMO.html

2.1 Típico script `hola mundo'

              #!/bin/bash          
              echo Hola Mundo 

Este script tiene sólo dos líneas. La primera le indica al sistema qué programa usar para ejecutar el fichero.

La segunda línea es la única acción realizada por este script, que imprime 'Hola Mundo' en la terminal.

Si le sale algo como ./hello.sh: Comando desconocido., probablemente la primera línea, '#!/bin/bash', está mal. Ejecute whereis bash, o vea 'encontrando el bash' para saber cómo debe escribir esta línea.

2.2 Un script de copia de seguridad muy simple

            #!/bin/bash          
            tar -cZf /var/my-backup.tgz /home/yo/

En este script, en vez de imprimir un mensaje en la terminal, creamos un tar-ball del directorio home de un usuario. Esto NO pretende ser un script útil; más tarde se ofrece un script de copia de seguridad más útil.

5. Variables

Puede usar variables como en cualquier otro lenguaje de programación. No existen tipos de datos. Una variable de bash puede contener un número, un caracter o una cadena de caracteres.

No necesita declarar una variable. Se creará sólo con asignarle un valor a su referencia.

5.1 Ejemplo: ¡Hola Mundo! utilizando variables

                #!/bin/bash          
                CAD="¡Hola Mundo!"
                echo $CAD    

La segunda línea crea una variable llamada STR y le asigna la cadena "¡Hola Mundo!". Luego se recupera el VALOR de esta variable poniéndole un '$' al principio. Por favor, tenga en cuenta (¡inténtelo!) que si no usa el signo '$', la salida del programa será diferente, y probablemente no sea lo que usted quería.

5.2 Ejemplo: Un script de copia de seguridad muy simple (algo mejor)

               #!/bin/bash          
               OF=/var/mi-backup-$(date +m%d).tgz
               tar -cZf $OF /home/yo/

Este script introduce algo nuevo. Antes que nada, debería familiarizarse con la creación y asignación de variable de la línea 2. Fíjese en la expresión '$(date +m%d)'. Si ejecuta el script se dará cuenta de que ejecuta el comando que hay dentro de los paréntesis, capturando su salida.

Tenga en cuenta que en este script, el fichero de salida será distinto cada día, debido al formato pasado al comando date (+m%d). Puede cambiar esto especificando un formato diferente.

Algunos ejemplos más:

 echo ls

 echo $(ls)

5.3 Variables locales

Las variables locales pueden crearse utilizando la palabra clave local.

                    #!/bin/bash
                    HOLA=Hola
                    function hola {
                            local HOLA=Mundo
                            echo $HOLA
                    }
                    echo $HOLA
                    hola
                    echo $HOLA

Este ejemplo debería bastar para mostrarle el uso de una variable local.

6. Estructuras Condicionales

Las estructuras condicionales le permiten decidir si se realiza una acción o no; esta decisión se toma evaluando una expresión.

6.1 Pura teoría

Los condicionales tienen muchas formas. La más básica es: if expresión then sentencia donde 'sentencia' sólo se ejecuta si 'expresión' se evalúa como verdadera. '2<1' es una expresión que se evalúa falsa, mientras que '2>1' se evalúa verdadera.

Los condicionales tienen otras formas, como: if expresión then sentencia1 else sentencia2. Aquí 'sentencia1' se ejecuta si 'expresión' es verdadera. De otra manera se ejecuta 'sentencia2'.

Otra forma más de condicional es: if expresión1 then sentencia1 else if expresión2 then sentencia2 else sentencia3. En esta forma sólo se añade "ELSE IF 'expresión2' THEN 'sentencia2'", que hace que sentencia2 se ejecute si expresión2 se evalúa verdadera. El resto es como puede imaginarse (véanse las formas anteriores).

Unas palabras sobre la sintaxis:

La base de las construcciones 'if' es ésta:

 if [expresión];

 then

     código si 'expresión' es verdadera.

 fi

6.2 Ejemplo: Ejemplo básico de condicional if .. then

                #!/bin/bash
                if [ "petete" = "petete" ]; then
                   echo expresión evaluada como verdadera
                fi

El código que se ejecutará si la expresión entre corchetes es verdadera se encuentra entre la palabra 'then' y la palabra 'fi', que indica el final del código ejecutado condicionalmente.

6.3 Ejemplo: Ejemplo básico de condicional if .. then ... else

                #!/bin/bash     if [ "petete" = "petete" ]; then
                   echo expresión evaluada como verdadera
                else
                   echo expresión evaluada como falsa
                fi

6.4 Ejemplo: Condicionales con variables

                #!/bin/bash
                T1="petete"
                T2="peteto"
                if [ "$T1" = "$T2" ]; then
                    echo expresión evaluada como verdadera
                else
                    echo expresión evaluada como falsa
                fi

6.5 Ejemplo: comprobando si existe un fichero

un agradecimiento más a mike

                #!/bin/bash
                FILE=~/.basrc
                if [ -f $FILE ]; then
                    echo el fichero $FILE existe
                else
                    echo fichero no encontrado
                fi
                if [ 'test -f $FILE']

7. Los bucles for, while y until

En esta sección se encontrará con los bucles for, while y until.

El bucle for es distinto a los de otros lenguajes de programación. Básicamente, le permite iterar sobre una serie de `palabras' contenidas dentro de una cadena.

El bucle while ejecuta un trozo de códico si la expresión de control es verdadera, y sólo se para cuando es falsa (o se encuentra una interrupción explícita dentro del código en ejecución).

El bucle until es casi idéntico al bucle loop, excepto en que el código se ejecuta mientras la expresión de control se evalúe como falsa.

Si sospecha que while y until son demasiado parecidos, está en lo cierto.

7.1 Por ejemplo

            #!/bin/bash
            for i in $( ls ); do
                echo item: $i
            done

En la segunda línea declaramos i como la variable que recibirá los diferentes valores contenidos en $( ls ).

La tercera línea podría ser más larga o podría haber más líneas antes del done (4).

`done' (4) indica que el código que ha utilizado el valor de $i ha acabado e $i puede tomar el nuevo valor.

Este script no tiene mucho sentido, pero una manera más útil de usar el bucle for sería hacer que concordasen sólo ciertos ficheros en el ejemplo anterior.

7.2 for tipo-C

Fiesh sugirió añadir esta forma de bucle. Es un bucle for más parecido al for de C/perl...

            #!/bin/bash
            for i in `seq 1 10`;
            do
                    echo $i
            done    

7.3 Ejemplo de while

             #!/bin/bash 
             CONTADOR=0
             while [  $CONTADOR -lt 10 ]; do
                 echo El contador es $CONTADOR
                 let CONTADOR=CONTADOR+1 
             done

Este script 'emula' la conocida (C, Pascal, perl, etc) estructura `for'.

7.4 Ejemplo de until

             #!/bin/bash 
             CONTADOR=20
             until [  $CONTADOR -lt 10 ]; do
                 echo CONTADOR $CONTADOR
                 let CONTADOR-=1
             done

8. Funciones

Como en casi todo lenguaje de programación, puede utilizar funciones para agrupar trozos de código de una manera más lógica, o practicar el divino arte de la recursión.

Declarar una función es sólo cuestión de escribir function mi_func { mi_código }.

Llamar a la función es como llamar a otro programa, sólo hay que escribir su nombre.

8.1 Ejemplo de funciones

               #!/bin/bash 
               function salir {
                   exit
               }
               function hola {
                   echo ¡Hola!
               }
               hola
               salir
               echo petete 

Las líneas 2-4 contienen la función 'salir'. Las líneas 5-7 contienen la función 'hola'. Si no está completamente seguro de lo que hace este script, por favor, ¡pruébelo!.

Tenga en cuenta que una función no necesita que sea declarada en un orden específico.

Cuando ejecute el script se dará cuenta de que: primero se llama a la función 'hola', luego a la función 'quit', y el programa nunca llega a la línea 10.

8.2 Ejemplo de funciones con parámetros

                    #!/bin/bash 
                    function salir {
                       exit
                    }  
                    function e {
                        echo $1 
                    }  
                    e Hola
                    e Mundo
                    salir
                    echo petete 

Este script es casi idéntico al anterior. La diferencia principal es la función 'e'. Esta función imprime el primer argumento que recibe. Los argumentos, dentro de las funciones, son tratados de la misma manera que los argumentos suministrados al script.

9. Interfaces de usuario

9.1 Utilizando select para hacer menús sencillos

               #!/bin/bash
               OPCIONES="Hola Salir"
               select opt in $OPCIONES; do
                   if [ "$opt" = "Salir" ]; then
                    echo done
                    exit
                   elif [ "$opt" = "Hola" ]; then
                    echo Hola Mundo
                   else
                    clear
                    echo opción errónea
                   fi
               done

Si ejecuta este script verá que es el sueño de un programador para hacer menús basados en texto. Probablemente se dará cuenta de que es muy similar a la construcción 'for', sólo que en vez de iterar para cada 'palabra' en $OPCIONES, se lo pide al usuario.

9.2 Utilizando la línea de comandos

              #!/bin/bash        
              if [ -z "$1" ]; then 
                  echo uso: $0 directorio
                  exit
              fi
              SRCD=$1
              TGTD="/var/backups/"
              OF=home-$(date +m%d).tgz
              tar -cZf $TGTD$OF $SRCD

Lo que hace este script debería estar claro para usted. La expresión del primer condicional comprueba si el programa ha recibido algún argumento ($1) y sale si no lo ha recibido, mostrándole al usuario un pequeño mensaje de uso. El resto del script debería estar claro.

10. Miscelánea

10.1 Leyendo información del usuario

En muchas ocasiones, puede querer solicitar al usuario alguna información, y existen varias maneras para hacer esto. Ésta es una de ellas:

                    #!/bin/bash
                    echo Por favor, introduzca su nombre
                    read NOMBRE
                    echo "¡Hola $NOMBRE!"

Como variante, se pueden obtener múltiples valores con read. Este ejemplo debería clarificarlo.

                    #!/bin/bash
                    echo Por favor, introduzca su nombre y primer apellido
                    read NO AP 
                    echo "¡Hola $AP, $NO!"

10.2 Evaluación aritmética

Pruebe esto en la línea de comandos (o en una shell):

echo 1 + 1

Si esperaba ver '2', quedará desilusionado. ¿Qué hacer si quiere que BASH evalúe unos números? La solución es ésta:

echo $((1+1))

Esto producirá una salida más 'lógica'. Esto se hace para evaluar una expresión aritmética. También puede hacerlo de esta manera:

echo 1+1

Si necesita usar fracciones, u otras matemáticas, puede utilizar bc para evaluar expresiones aritméticas.

Si ejecuta "echo 3/4" en la línea de comandos, devolverá 0, porque bash sólo utiliza enteros en sus respuestas. Si ejecuta "echo 3/4|bc -l", devolverá 0.75.

10.3 Encontrando el bash

De un mensaje de mike (vea los agradecimientos):

siempre usas #!/bin/bash .. a lo mejor quieres dar un ejemplo

de cómo saber dónde encontrar el bash.

 `locate bash' es preferible, pero no todas las máquinas

tienen locate.

 `find ./ -name bash' desde el directorio raíz funcionará,

normalmente.

Sitios donde poder buscar:

 ls -l /bin/bash

 ls -l /sbin/bash

 ls -l /usr/local/bin/bash

 ls -l /usr/bin/bash

 ls -l /usr/sbin/bash

 ls -l /usr/local/sbin/bash

(no se me ocurre ningún otro directorio... lo he encontrado

la mayoría de estos sitios en sistemas diferentes).

También puedes probar 'which bash'.

10.4 Obteniendo el valor devuelto por un programa

En bash, el valor de retorno de un programa se guarda en una variable especial llamada $?.

Esto ilustra cómo capturar el valor de retorno de un programa. Supongo que el directorio dada no existe. (Esto también es sugerencia de Mike).

            #!/bin/bash
            cd /dada &> /dev/null
            echo rv: $?
            cd $(pwd) &> /dev/null
            echo rv: $?

10.5 Capurando la salida de un comando

Este pequeño script muestra todas las tablas de todas las bases de datos (suponiendo que tenga MySQL instalado). Considere también cambiar el comando 'mysql' para que use un nombre de usuario y clave válidos.

            #!/bin/bash
            DBS=`mysql -uroot  -e"show databases"`
            for b in $DBS ;
            do
                    mysql -uroot -e"show tables from $b"
            done

11. Tablas

11.1 Operadores de comparación de cadenas

 s1 = s2

    s1 coincide con s2

 s1 != s2

    s1 no coincide con s2

 s1 < s2

    s1 es alfabéticamente anterior a s2, con el locale actual

 s1 > s2

    s1 es alfabéticamente posterior a s2, con el locale actual

 -n s1

    s1 no es nulo (contiene uno o más caracteres)

 -z s1

    s1 es nulo 

11.2 Ejemplo de comparación de cadenas

Comparando dos cadenas

            #!/bin/bash
            S1='cadena'
            S2='Cadena'
            if [ $S1!=$S2 ];
            then
                    echo "S1('$S1') no es igual a S2('$S2')"
            fi
            if [ $S1=$S1 ];
            then
                    echo "S1('$S1') es igual a S1('$S1')"
            fi

Cito aquí el consejo de un correo enviado por Andreas Beck, referido al uso de if [ $1 = $2 ].

Esto no es buena idea, porque si $S1 o $S2 son vacíos, aparecerá un parse error. Es mejor: x$1=x$2 or "$1"="$2"

11.3 Operadores aritméticos

 + (adición)

 - (sustracción)

 * (producto)

 / (división)

 % (módulo)

11.4 Operadores relacionales aritméticos

 -lt (<)

 -gt (>)

 -le (<=)

 -ge (>=)

 -eq (==)

 -ne (!=)

Los programadores de C tan sólo tienen que corresponder el operador con su paréntesis.

11.5 Comandos útiles

Esta sección ha sido reescrita por Kees (véanse agradecimientos)

Algunos de estos comandos contienen lenguajes de programación completos. Sólo se explicarán las bases de estos comandos. Para una descripción más detallada, eche un vistazo a las páginas man de cada uno.

sed (editor de flujo)

Sed es un editor no interactivo. En vez de alterar un fichero moviendo el cursor por la pantalla, se utiliza una serie de instrucciones de edición de sed, y el nombre del fichero a editar. También se puede describir a sed como un filtro. Miremos algunos ejemplos:

            $sed 's/a_sustituir/sustituto/g' /tmp/petete

Sed sustituye la cadena 'a_sustituir' por la cadena 'sustituto', leyendo del fichero /tmp/petete. El resultado se envía a stdout (normalmente la consola), pero se puede añadir '> captura' al final de la línea de arriba para que sed envíe la salida al fichero 'capture'.

            $sed 12, 18d /tmp/petete

Sed muestra todas las líneas de /tmp/petete excepto la 12 y la 18. El fichero original no queda alterado por este comando.

awk (manipulación de bases de datos, extracción y proceso de texto)

Existen muchas implementaciones del lenguaje de programacin AWK (los intérpretes más conocidos son gawk de GNU, y el 'nuevo awk' mawk). El principio es sencillo: AWK busca un patrón, y por cada patrón de búsqueda que coincida, se realiza una acción.

Si tenemos un fichero /tmp/petete con las siguientes líneas:

"prueba123

prueba

pprruueebbaa"

y ejecutamos:

            $awk '/prueba/ {print}' /tmp/petete

test123

test

El patrón que busca AWK es 'prueba' y la acción que realiza cuando encuentra una línea en /tmp/petete con la cadena 'prueba' es `print'.

            $awk '/prueba/ {i=i+1} END {print i}' /tmp/petete

3

Cuando se utilizan muchos patrones, se puede reemplazar el texto entre comillas por '-f fichero.awk', y poner todos los patrones y acciones en 'fichero.awk'.

grep (impresión de líneas que coinciden con un patrón de búsqueda)

Ya hemos visto ejemplos del comando grep en los capítulos anteriores, que muestra las líneas que concuerdan con un patrón. Pero grep puede hacer más que eso.

            $grep "busca esto" /var/log/messages -c

12

Se ha encontrado 12 veces la cadena "busca esto" en el fichero /var/log/messages.

[vale, este ejemplo es falso, el fichero /var/log/messages está alterado :-)]

wc (cuenta líneas, palabras y bytes)

En el siguiente ejemplo, vemos que la salida no es lo que esperábamos. El fichero petete utilizado en este ejemplo contiene el texto siguiente:

"programación en bash como de introducción"

            $wc --words --lines --bytes /tmp/petete

2 5 41 /tmp/petete

Wc no tiene en cuenta el orden de los parámetros. Wc siempre los imprime en un orden estándar, que es, como se puede ver: líneas, palabras, bytes y fichero.

sort (ordena líneas de ficheros de texto)

Esta vez, el fichero petete contiene el texto siguiente:

"b c a"

            $sort /tmp/petete

Esto es lo que muestra la salida:

a b c

Los comandos no deberían ser tan fáciles :-)

bc (un lenguaje de programación de cálculos matemáticos)

Bc acepta cálculos desde la línea de comandos (entrada desde un fichero, pero no desde una redirección o una tubería), y también desde una interfaz de usuario. La siguiente demostración expone algunos de los comandos. Note que ejecuto bc con el parámetro -q para evitar el mensaje de bienvenida.

       $bc -q

1 == 5

0

0.05 == 0.05

1

5 != 5

0

2 ^ 8

256

sqrt(9)

3

while (i != 9) {

i = i + 1;

print i

}

123456789

quit

tput (inicializa una terminal o consulta la base de datos de terminfo)

Una pequeña demostración de las capacidades de tput:

            $tput cup 10 4

La línea de comandos aparece en (y10,x4).

            $tput reset

Limpia la pantalla y la línea de comandos aparece en (y1,x1). Observe que (y0,x0) es la esquina superior izquierda.

            $tput cols

80

Muestra el número de caracteres que caben en la dirección x.

Es muy recomendable familiarizarse con estos programas (al menos). Hay montones de programillas que le permitirán hacer virguerías en la línea de comandos.

[algunos ejemplos están copiados de las páginas man o los PUFs]

12. Más scripts

12.1 Aplicando un comando a todos los ficheros de un directorio.

12.2 Ejemplo: Un script de copia de seguridad muy simple (algo mejor)

                #!/bin/bash          
                ORIG="/home/"
                DEST="/var/copias_de_seguridad/"
                FICH=home-$(date +m%d).tgz
                tar -cZf $DEST$FICH $ORIG

12.3 Re-nombrador de ficheros

                 #!/bin/sh
                 # renom: renombra múltiples ficheros de acuerdo con ciertas
                 # reglas
                 # escrito por Felix Hudson  Enero - 2000

                 # primero comprueba los distintos 'modos' que tiene este
                 # programa
                 # si la primera ($1) condición coincide, se ejecuta esa parte
                 # del programa y acaba

                 # comprueba la condición de prefijo
                 if [ $1 = p ]; then

                 # ahora nos libramos de la variable de modo ($1) y ponemos $2
                 # de prefijo
                   prefijo=$2 ; shift ; shift

                 # una rápida comprobación para ver si se especificó algún
                 # fichero
                 # si no, hay cosas mejores que hacer que renombrar ficheros
                 # inexistentes!!
                   if [$1 = ]; then
                      echo "no se especificaron ficheros"
                      exit 0
                   fi

                 # este bucle for itera a lo largo de todos los ficheros que
                 # le hemos especificado al programa
                 # renombra cada uno de ellos
                   for fichero in $*
                     do
                     mv ${fichero} $prefijo$fichero
                   done

                 # ahora salimos del programa
                   exit 0
                 fi

                 # comprueba si es un renombramiento con sufijo
                 # el resto es casi idéntico a la parte anterior
                 # lea los comentarios anteriores
                 if [ $1 = s ]; then
                   sufijo=$2 ; shift ; shift

                    if [$1 = ]; then
                     echo "no se especificaron ficheros"
                    exit 0
                    fi

                  for fichero in $*
                   do
                    mv ${fichero} $fichero$sufijo
                  done

                  exit 0
                 fi

                 # comprueba si es una sustitución
                 if [ $1 = r ]; then

                   shift

                 # he incluído esto para no dañar ningún fichero si el
                 # usuario no especifica que se haga nada
                 # tan sólo una medida de seguridad             
                   if [ $# -lt 3 ] ; then
                     echo "uso: renom r [expresión] [sustituto] ficheros... "
                     exit 0
                   fi

                 # elimina el resto de información
                   VIEJO=$1 ; NUEVO=$2 ; shift ; shift

                 # este bucle for itera a lo largo de todos los ficheros que
                 # le hemos especificado al programa
                 # renombra cada fichero utilizando el programa 'sed'
                 # es un sencillo programa desde la línea de comandos que
                 # analiza la entrada estándar y sustituye una expresión por
                 # una cadena dada
                 # aquí le pasamos el nombre del fichero (como entrada
                 # estándar)
                   for fichero in $*
                   do
                     nuevo=`echo ${fichero} | sed s/${VIEJO}/${NUEVO}/g`
                     mv ${fichero} $nuevo
                   done
                 exit 0
                 fi

                 # si se llega a esta parte es que no se le pasó nada
                 # apropiado al programa, por lo que le decimos al usuario
                 # cómo hacerlo
                 echo "uso:"
                 echo " renom p [prefijo] ficheros.."
                 echo " renom s [sufijo] ficheros.."
                 echo " renom r [expresión] [sustituto] ficheros.."
                 exit 0

                 # hecho!

12.4 Re-nombrador de ficheros (sencillo)

         #!/bin/bash
         # renombra.sh
         # renombrador de ficheros básico

         criterio=$1
         expresion=$2
         sustituto=$3

         for i in $( ls *$criterio* ); 
         do
             orig=$i
             dest=$(echo $i | sed -e "s/$expresion/$sustituto/")
             mv $orig $dest
         done

13. Cuando algo va mal (depuración)

13.1 Maneras de llamar a BASH

Una buena idea es poner esto en la primera línea:

              #!/bin/bash -x

Esto producirá información interesante.

Última modificación de la página el 2007-12
Powered by PmWiki