Introducción a la Programación en BASH

Programación en BASH – COMO de introducción
Mike G (mikkey) disponible en dynamo.com.ar
Traducido por Gabriel Rodríguez Alberich chewie@asef.us.es

jueves, 27 de julio de 2000, a las 09:36:18 ART

Este artículo pretende ayudarle a comenzar a programar shell scripts a
un nivel básico/intermedio. No pretende ser un documento avanzado (vea
el título). NO soy un experto ni un gurú de la programación en shell.
Decidí escribir esto porque aprenderé mucho con ello y puede serle
útil a otras personas. Cualquier aportación será apreciada, especial­
mente en forma de parche :)

______________________________________________________________________

Índice general

1. Introducción
1.1 Obteniendo la última versión
1.2 Requisitos
1.3 Usos de este documento

2. Scripts muy sencillos
2.1 Típico script `hola mundo’
2.2 Un script de copia de seguridad muy simple

3. Todo sobre redirección
3.1 Teoría y referencia rápida
3.2 Ejemplo: stdout a un fichero
3.3 Ejemplo: stderr a un fichero
3.4 Ejemplo: stdout a stderr
3.5 Ejemplo: stderr a stdout
3.6 Ejemplo: stderr y stdout a un fichero

4. Tuberías
4.1 Qué son y por qué querrá utilizarlas
4.2 Ejemplo: una tubería sencilla con sed
4.3 Ejemplo: una alternativa a ls -l *.txt

5. Variables
5.1 Ejemplo: ¡Hola Mundo! utilizando variables
5.2 Ejemplo: Un script de copia de seguridad muy simple (algo mejor)
5.3 Variables locales

6. Estructuras Condicionales
6.1 Pura teoría
6.2 Ejemplo: Ejemplo básico de condicional if .. then
6.3 Ejemplo: Ejemplo básico de condicional if .. then … else
6.4 Ejemplo: Condicionales con variables
6.5 Ejemplo: comprobando si existe un fichero

7. Los bucles for, while y until
7.1 Por ejemplo
7.2 for tipo-C
7.3 Ejemplo de while
7.4 Ejemplo de until

8. Funciones
8.1 Ejemplo de funciones
8.2 Ejemplo de funciones con parámetros

9. Interfaces de usuario
9.1 Utilizando select para hacer menús sencillos
9.2 Utilizando la línea de comandos

10. Miscelánea
10.1 Leyendo información del usuario
10.2 Evaluación aritmética
10.3 Encontrando el bash
10.4 Obteniendo el valor devuelto por un programa
10.5 Capurando la salida de un comando

11. Tablas
11.1 Operadores de comparación de cadenas
11.2 Ejemplo de comparación de cadenas
11.3 Operadores aritméticos
11.4 Operadores relacionales aritméticos
11.5 Comandos útiles

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)
12.3 Re-nombrador de ficheros
12.4 Re-nombrador de ficheros (sencillo)

13. Cuando algo va mal (depuración)
13.1 Maneras de llamar a BASH

14. Sobre el documento
14.1 (sin) Garantía
14.2 Traducciones
14.3 Agradecimientos
14.4 Historia
14.5 Más recursos

______________________________________________________________________

11..IInnttrroodduucccciióónn

11..11..OObbtteenniieennddoo llaa úúllttiimmaa vveerrssiióónn

http://www.linuxdoc.org/HOWTO/Bash-Prog-Intro-HOWTO.html

11..22..RReeqquuiissiittooss

Le será útil tener una cierta familiaridad con la línea de comandos de
GNU/Linux y con los conceptos básicos de la programación.Aunque esto
no es una introducción a la programación, explica (o al menos lo
intenta) muchos conceptos básicos.

11..33..UUssooss ddee eessttee ddooccuummeennttoo

Este documento intenta ser útil en las siguientes situaciones

·Si tiene alguna idea de programación y quiere empezar a programar
algunos shell scripts.

·Si tiene una idea vaga de programar en shell y quiere algún tipo de
referencia.

·Si quiere ver algunos scripts y comentarios para empezar a escribir
los suyos propios.

·Si está migrando desde DOS/Windows (o ya lo ha hecho) y quiere
hacer procesos «por lotes».

·Si es un completo novato y lee todo COMO disponible.

22..SSccrriippttss mmuuyy sseenncciillllooss

Este COMO tratará de darle algunos consejos sobre la programación de
shell scripts, basándose profundamente en ejemplos.

En esta sección encontrará varios scripts pequeños que
esperanzadamente le ayudarán a entender algunas técnicas.

22..11..TTííppiiccoo ssccrriipptt ``hhoollaa mmuunnddoo’’

#!/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 _._/_h_e_l_l_o_._s_h_: _C_o_m_a_n_d_o _d_e_s_c_o_n_o_c_i_d_o_., 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.

22..22..UUnn ssccrriipptt ddee ccooppiiaa ddee sseegguurriiddaadd mmuuyy ssiimmppllee

#!/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.

33..TTooddoo ssoobbrree rreeddiirreecccciióónn

33..11..TTeeoorrííaa yy rreeffeerreenncciiaa rrááppiiddaa

Existen 3 descriptores de ficheros: stdin, stdout y stderr
(std=estándar).

Básicamente, usted puede:

1. redirigir stdout a un fichero

2. redirigir stderr a un fichero

3. redirigir stdout a stderr

4. redirigir stderr a stdout

5. redirigir stderr y stdout a un fichero

6. redirigir stderr y stdout a stdout

7. redirigir stderr y stdout a stderr

El número 1 ‘representa’ a stdout, y 2 a stderr.

Una pequeña nota para ver todo esto: con el comando less puede
visualizar stdout (que permanecerá en el búfer) y stderr, que se
imprimirá en la pantalla, pero será borrado si intenta leer el búfer.

33..22..EEjjeemmpplloo:: ssttddoouutt aa uunn ffiicchheerroo

Esto hará que la salida de un programa se escriba en un fichero.

ls -l > ls-l.txt

En este caso, se creará un fichero llamado ‘ls-l.txt’ que contendrá lo
que se vería en la pantalla si escribiese el comando ‘ls -l’ y lo eje­
cutase.

33..33..EEjjeemmpplloo:: ssttddeerrrr aa uunn ffiicchheerroo

Esto hará que la salida stderr de un programa se escriba en un
fichero.

grep da * 2> errores-de-grep.txt

En este caso, se creará un fichero llamado ‘errores-de-grep.txt’ que
contendrá la parte stderr de la salida que daría el comando ‘grep da
*’.

33..44..EEjjeemmpplloo:: ssttddoouutt aa ssttddeerrrr

Esto hará que la salida stdout de un programa se escriba en el mismo
descriptor de fichero que stderr.

grep da * 1>&2

En este caso, la parte stdout del comando se envía a stderr; puede
observar eso de varias maneras.

33..55..EEjjeemmpplloo:: ssttddeerrrr aa ssttddoouutt

Esto hará que la salida stderr de un programa se escriba en el mismo
descriptor de fichero que stdout.

grep * 2>&1

En este caso, la parte stderr del comando se envía a stdout. Si hace
una tubería con less, verá que las líneas que normalmente ‘desapare­
cen’ (al ser escritas en stderr), ahora permanecen (porque están en el
stdout).

33..66..EEjjeemmpplloo:: ssttddeerrrr yy ssttddoouutt aa uunn ffiicchheerroo

Esto colocará toda la salida de un programa en un fichero. A veces,
esto es conveniente en las entradas del cron, si quiere que un comando
se ejecute en absoluto silencio.

rm -f $(find / -name core) &> /dev/null

Esto (pensando en la entrada del cron) eliminará todo archivo llamado
`core’ en cualquier directorio. Tenga en cuenta que tiene que estar
muy seguro de lo que hace un comando si le va a eliminar la salida.

44..TTuubbeerrííaass

Esta sección explica de una manera muy sencilla y práctica cómo
utilizar tuberías, y por qué querría utilizarlas.

44..11..QQuuéé ssoonn yy ppoorr qquuéé qquueerrrráá uuttiilliizzaarrllaass

Las tuberías le permiten utilizar (muy sencillo, insisto) la salida de
un programa como la entrada de otro.

44..22..EEjjeemmpplloo:: uunnaa ttuubbeerrííaa sseenncciillllaa ccoonn sseedd

Ésta es una manera muy sencilla de utilizar tuberías.

ls -l | sed -e «s/[aeio]/u/g»

En este caso, ocurre lo siguiente: primero se ejecuta el comando ls
-l, y luego su salida, en vez de imprimirse en la pantalla, se envía
(entuba) al programa sed, que imprime su salida correspondiente.

44..33..EEjjeemmpplloo:: uunnaa aalltteerrnnaattiivvaa aa llss --ll **..ttxxtt

Probablemente ésta es una manera más difícil de hacer un ls -l *.txt,
pero se muestra para ilustrar el funcionamiento de las tuberías, no
para resolver ese dilema.

ls -l | grep «\.txt$»

En este caso, la salida del programa ls -l se envía al programa grep,
que imprimirá las líneas que concuerden con la regex (expresión regu­
lar) «\.txt$».

55..VVaarriiaabblleess

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.

55..11..EEjjeemmpplloo:: ¡¡HHoollaa MMuunnddoo!! uuttiilliizzaannddoo vvaarriiaabblleess

#!/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.

55..22..EEjjeemmpplloo:: UUnn ssccrriipptt ddee ccooppiiaa ddee sseegguurriiddaadd mmuuyy ssiimmppllee ((aallggoo mmeejjoorr))

#!/bin/bash
OF=/var/mi-backup-$(date +%Y%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 +%Y%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 (+%Y%m%d).Puede
cambiar esto especificando un formato diferente.

Algunos ejemplos más:

echo ls

echo $(ls)

55..33..VVaarriiaabblleess llooccaalleess

Las variables locales pueden crearse utilizando la palabra clave
_l_o_c_a_l.

#!/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.

66..EEssttrruuccttuurraass CCoonnddiicciioonnaalleess

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

66..11..PPuurraa tteeoorrííaa

Los condicionales tienen muchas formas. La más básica es: iiff _e_x_p_r_e_s_i_ó_n
tthheenn _s_e_n_t_e_n_c_i_a 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: iiff _e_x_p_r_e_s_i_ó_n tthheenn
_s_e_n_t_e_n_c_i_a_1 eellssee _s_e_n_t_e_n_c_i_a_
2.Aquí ‘sentencia1’ se ejecuta si
‘expresión’ es verdadera. De otra manera se ejecuta ‘sentencia2’.

Otra forma más de condicional es: iiff _e_x_p_r_e_s_i_ó_n_1 tthheenn _s_e_n_t_e_n_c_i_a_1 eellssee
iiff _e_x_p_r_e_s_i_ó_n_2 tthheenn _s_e_n_t_e_n_c_i_a_2 eellssee _s_e_n_t_e_n_c_i_a_
3.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

66..22..EEjjeemmpplloo:: EEjjeemmpplloo bbáássiiccoo ddee ccoonnddiicciioonnaall iiff .... tthheenn

#!/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.

66..33..EEjjeemmpplloo:: EEjjeemmpplloo bbáássiiccoo ddee ccoonnddiicciioonnaall iiff .... tthheenn ...... eellssee

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

66..44..EEjjeemmpplloo:: CCoonnddiicciioonnaalleess ccoonn vvaarriiaabblleess

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

66..55..EEjjeemmpplloo:: ccoommpprroobbaannddoo ssii eexxiissttee uunn ffiicchheerroo

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’]

77..LLooss bbuucclleess ffoorr,, wwhhiillee yy uunnttiill

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

El bucle ffoorr 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 wwhhiillee 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 uunnttiill 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.

77..11..PPoorr eejjeemmpplloo

#!/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.

77..22..ffoorr ttiippoo--CC

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

77..33..EEjjeemmpplloo ddee wwhhiillee

#!/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’.

77..44..EEjjeemmpplloo ddee uunnttiill

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

88..FFuunncciioonneess

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.

88..11..EEjjeemmpplloo ddee ffuunncciioonneess

#!/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.

88..22..EEjjeemmpplloo ddee ffuunncciioonneess ccoonn ppaarráámmeettrrooss

#!/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.

99..IInntteerrffaacceess ddee uussuuaarriioo

99..11..UUttiilliizzaannddoo sseelleecctt ppaarraa hhaacceerr mmeennúúss sseenncciillllooss

#!/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.

99..22..UUttiilliizzaannddoo llaa llíínneeaa ddee ccoommaannddooss

#!/bin/bash
if [ -z «$1″ ]; then
echo uso: $0 directorio
exit
fi
SRCD=$1
TGTD=»/var/backups/»
OF=home-$(date +%Y%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.

1100..MMiisscceelláánneeaa

1100..11..LLeeyyeennddoo iinnffoorrmmaacciióónn ddeell uussuuaarriioo

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!»

1100..22..EEvvaalluuaacciióónn aarriittmmééttiiccaa

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.

1100..33..EEnnccoonnttrraannddoo eell bbaasshh

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’.

1100..44..OObbtteenniieennddoo eell vvaalloorr ddeevvuueellttoo ppoorr uunn pprrooggrraammaa

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 _d_a_d_a no existe. (Esto también es sugerencia de
Mike).

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

1100..55..CCaappuurraannddoo llaa ssaalliiddaa ddee uunn ccoommaannddoo

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

1111..TTaabbllaass

1111..11..OOppeerraaddoorreess ddee ccoommppaarraacciióónn ddee ccaaddeennaass

ss11 == ss22
s1 coincide con s2

ss11 !!== ss22
s1 no coincide con s2

ss11 << ss22
s1 es alfabéticamente anterior a s2, con el _l_o_c_a_l_e actual

ss11 >> ss22
s1 es alfabéticamente posterior a s2, con el _l_o_c_a_l_e actual

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

--zz ss11
s1 es nulo

1111..22..EEjjeemmpplloo ddee ccoommppaarraacciióónn ddee ccaaddeennaass

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 _i_f _[ _$_1 _= _$_2 _].

Esto no es buena idea, porque si $S1 o $S2 son vacíos, aparecerá un
_p_a_r_s_e _e_r_r_o_r. Es mejor: x$1=x$2 or «$1″=»$2″

1111..33..OOppeerraaddoorreess aarriittmmééttiiccooss

+ (adición)

– (sustracción)

* (producto)

/ (división)

% (módulo)

1111..44..OOppeerraaddoorreess rreellaacciioonnaalleess aarriittmmééttiiccooss

-lt (<)

-gt (>)

-le (<=)

-ge (>=)

-eq (==)

-ne (!=)

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

1111..55..CCoommaannddooss úúttiilleess

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.

sseedd (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.

aawwkk (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:

_»_p_r_u_e_b_a_1_2_3

_p_r_u_e_b_a

_p_p_r_r_u_u_e_e_b_b_a_a_»

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’.

ggrreepp (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 :-)]

wwcc (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:

_»_p_r_o_g_r_a_m_a_c_i_ó_n _e_n _b_a_s_h
_c_o_m_o _d_e _i_n_t_r_o_d_u_c_c_i_ó_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.

ssoorrtt (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 :-)

bbcc (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 queejecuto bc con el parámetro -q para evitar el
mensaje de bienvenida.

$bc -q

_1 _=_= _5

_0

_0_._0_5 _=_= _0_._0_5

_1

_5 _!_= _5

_0

_2 _^ _8

_2_5_6

_s_q_r_t_(_9_)

_3

_w_h_i_l_e _(_i _!_= _9_) _{

_i _= _i _+ _1_;

_p_r_i_n_t _i

_}

_1_2_3_4_5_6_7_8_9

_q_u_i_t

ttppuutt (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

_8_0

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]

1122..MMááss ssccrriippttss

1122..11..AApplliiccaannddoo uunn ccoommaannddoo aa ttooddooss llooss ffiicchheerrooss ddee uunn ddiirreeccttoorriioo..

1122..22..EEjjeemmpplloo:: UUnn ssccrriipptt ddee ccooppiiaa ddee sseegguurriiddaadd mmuuyy ssiimmppllee ((aallggoo
mmeejjoorr))

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

1122..33..RRee--nnoommbbrraaddoorr ddee ffiicchheerrooss

#!/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!

1122..44.. RRee--nnoommbbrraaddoorr ddee ffiicchheerrooss ((sseenncciilllloo))

#!/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

1133..CCuuaannddoo aallggoo vvaa mmaall ((ddeeppuurraacciióónn))

1133..11..MMaanneerraass ddee llllaammaarr aa BBAASSHH

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

#!/bin/bash -x

Esto producirá información interesante.

1144..SSoobbrree eell ddooccuummeennttoo

Siéntase libre para hacer sugerencias/correcciones, o lo que crea que
sea interesante que aparezca en este documento. Intentaré actualizarlo
tan pronto como me sea posible.

1144..11..((ssiinn)) GGaarraannttííaa

Este documento no lleva garantía de ningún tipo.

1144..22..TTrraadduucccciioonneess

Italiano: por William Ghelfi (wizzy está en tiscalinet.it).
http://web.tiscalinet.it/penguin_rules

Francés: por Laurent Martelli ¿?

Coreano: Minseok Park http://kldp.org

Corean: Chun Hye Jin Desconocido

Spanish: Gabriel Rodríguez Alberich http://www.insflug.org

Supongo que habrá más traducciones, pero no tengo información sobre
ellas. Si las tiene, por favor, envíemelas para que actualice esta
sección.

1144..33..AAggrraaddeecciimmiieennttooss

·A la gente que ha traducido este documento a otras lenguas (sección
anterior).

·A Nathan Hurst por enviar montones de correcciones.

·A Jon Abbott por enviar comentarios sobre la evaluación de
expresiones aritméticas.

·A Felix Hudson por escribir el script _r_e_n_o_m

·A Kees van den Broek (por enviar tantas correcciones y reescribir
la sección de comandos útiles)

·Mike (pink) hizo algunas sugerencias sobre la localización del bash
y la comprobación de los ficheros

·Fiesh hizo una buena sugerencia sobre la sección de bucles.

·Lion sugirió mencionar un error común (./hello.sh: Comando no
encontrado.)

·Andreas Beck hizo varias correcciones y comentarios.

1144..44..HHiissttoorriiaa

Añadidas nuevas traducciones y correcciones menores.

Añadida la sección de comandos útiles reescrita por Kess.

Incorporadas más correcciones y sugerencias.

Añadidos ejemplos sobre la comparación de cadenas.

v0.8 abandono del versionamiento. Supongo que con la fecha es
suficiente.

v0.7 Más correcciones y algunas secciones TO-DO escritas.

v0.6 Correcciones menores.

v0.5 Añadida la sección de redireccionamiento.

v0.4 desaparición de su sitio debido a mi ex-jefe. Este documento
tiene un nuevo sitio en: http://www.linuxdoc.org.

Anteriores: no me acuerdo y no he usado rcs ni cvs :(

1144..55..MMááss rreeccuurrssooss

Introducción a bash (bajo BE)
http://org.laol.net/lamug/beforever/bashtut.htm

Los comentarios están cerrados.