shell script

Iniciado por EMPICAL, 20 Mayo 2003, 20:34 PM

0 Miembros y 2 Visitantes están viendo este tema.

EMPICAL

¿Que es un shell script?

Son basicamente programas escritos en lenguaje de shell muy parecidos a los programas de procesamiento por lotes de windows. Es natural que linux nos ofrezca esta capacidad gracias su sistema de entrada/salida que lo hace tan potente.

¿Para que nos sirve un shell script?

Para automatizar tareas que hacemos habitualmente ahorrando un tiempo considerable al usuario.

Ejemplo:

Para que comprendamos el funcionamiento y que es un shell script la mejor manera es que veamos un ejemplo de un sencillo script que por ejemplo mandaremos un mail con el contenido de un archivo de log a un usuario registrado en nuestro sistema.

Lo primero que debemos hacer es determinar que tipo de shell usaremos para el script y donde se encuentra en el sistema.

El nombre del destinatario y su mail es algo que tambien deveremos averiguar.

Pensaremos que esto sera como leer noticias y para ellos usaremos el comando inn (internetnews)

Antes de seguir hay que explicar que es el concepto de canalizacion en linux. En linux no solo se puede redireccionar la entrada STDIN y la salida STDOUT de unos archivos a otros, si no que, como linux concibe esta entrada y salida como flujos de datos, tambien existe el concepto de canalizacion de flujos directamente de un programa a otro. Gracias a las canalizaciones se puede usar directamente la salida STDOUT de un programa como la entrada STIN de otro.

Para introducir una canalizacion en linux hay que introduccir en la linea de comandos la barra vertical (|) entre los programas.

Usaremos entre otros el comando tail que sirve para mostrar la parte final del documento. Como se trata de un archivo de logs solo nos interesara enviarle los ultimos registros del archivo puesto que la mayor parte de los registros aumentan por el final.

Usaremos otros comando que no voy a explicar ya que esto es solo un ejemplo para que se vea el funcionamiento de un shell script.
Código (bash) [Seleccionar]

#!/bin/bash
echo Hola a todos!! | mail destinatario@ensistema.mail
grep "destinatario" /usr/local/httpd/logs/access_log | tail -l100 | lpr xterm -e trn &
exit

Guardaremos esto en un archivo llamado mail_log y le daremos permisos de ejecucion chmod 755 mail_log.

Ya solo nos quedara ejecutarlo ./mail_log.

Pues hasta aqui creo que queda claro que es un script, para que se usa y como hacer uno.

Cualquiera que desee añadir informacion a este tema puede hacerlo libremente intetare postear mas cosas sobre shell script ya que me parece un tema interesante y algo que cualquier usuario de linux deberia dominar en mayor o menor medida.

Aver si la gente se anima y creamos un buen tema sobre shell script para empezar aprender desde 0.

..sR. aDiKtO..

#1
Ya k E*M*P*I*C*A*L ha empezado este tema ( k me parece MUY interesante), voy a continuar yo, primero con como se construye un shell script:
  • Se crea un archivo de texto y se introduce en la primera linea el interprete k se va a utilizar para ejecutarlo, en el caso anterior:
#!/bin/bash
  • En las lineas siguientes se ponen las intrucciones, una por linea:
    Código (bash) [Seleccionar]
    echo Hola a todos!! | mail destinatario@ensistema.mail
    grep "destinatario" /usr/local/httpd/logs/access_log | tail -l100 | lpr xterm -e trn &
    exit

    o en la misma linea separadas por un punto y coma ( ; ), excepto cuando es background o nohup.
    Código (bash) [Seleccionar]
    echo Hola a todos!! | mail destinatario@ensistema.mail; grep "destinatario" /usr/local/httpd/logs/access_log | tail -l100 | lpr xterm -e trn &
    exit

  • Luego se tiene k dar permisos de ejecucion y ejecutarlo como ha dicho E*M*P*I*C*A*L. Tambien se puede ejecutar con un interprete de comandos distito al predefinido, por ejemplo con:
    Código (bash) [Seleccionar]
    sh mail_log
    ó
    ksh mail_log
    ó
    ...

    Con estas ordenes el script seria ejecutado por sh ó ksh ó ... cuando en verdad tenia k ser ejecutado por bash (primera linea del script)
Continuemos con los redireccionamientos:
  • La tuberia ( pipe, o raya vertical '|' ), es un programa k redirige la salida de un programa a la entrada de otro, veamos un ejemplo:
    cat archivo.txt | grep adikto
    Esta orden hace:
    • cat archivo --> muestra el contenido de un archivo de texto.
    • grep adikto --> muestra por pantalla las lineas k tengan el patron, en este caso 'adikto'.
    • cat archivo.txt | grep adikto --> muestra por pantalla las lineas del archivo archivo.txt k tengan la palabra adikto.
    Como se ha podido comprobar, la salida de 'cat archivo.txt' se ha introducido como entrada a 'grep adikto' a traves de '|' y el resultado se ha mostrado por la salida estandar ( la pantalla).
  • Redireccion de salida ( '>' ), es redirigir la salida estandar a otro sitio, veamos las clases k hay:
    • '>' --> redireccion truncada, redirige la salida a otro sitio. Un ejemplo de esto seria:
    ls -l / > archivo.txt
    Con lo k el contenido del archivo.txt contendria la salida del 'ls -l /'
  • '>>' --> redireccion añadida, añade la salida a otro sitio. Por ejemplo:
    ls -l /boot >> archivo.txt
    El contenido del archivo archivo.txt contiene la salida del 'ls -l /' del ejemplo anterior mas el 'ls -l /boot' de este ejemplo.
    Si ahora hicieramos lo mismo k en el ejemplo anterior:
    ls -l / > archivo.txt
    El archivo archivo.txt SOLO contendria el 'ls -l /'.
  • '2>' --> redireccion de errores, redirige la salida del programa y los errores a otro sitio. Por ejemplo:
    cdrecord -adikto 2> archivo.txt
    La opcion '-adikto' no existe, y el error k tendria k salir va a parar al archivo archivo.txt.
  • '&>' --> redireccionamiento total, redirige TODO a otro sitio. Por ejemplo:
    cdrecord --help &> archivo.txt

      La ayuda de cdrecord ( --help ) va a parar a archivo.txt.
  • Redireccionamiento de entrada ( '<' ), la entrada de un programa es un archivo. Por ejemplo:
    grep adikto < archivo.txt

      La orden 'grep aditko' toma como entrada, el contenido de 'archivo.txt', (vease la similitud con las tuberias). Muestra las lineas k contiene archivo.txt con la palabra 'adikto'.

    Ahora le toca a otro continuar......xDDD

Rojodos

#2
----------   Uso de variables en shell-script:

Asignación: Nombre=valor
Acceso a su contenido: $Nombre

Podemos utilizar el comando read para leer variables desde teclado y el comando echo para visualizar su contenido (este comando también puede utilizarse para mostrar mensajes por pantalla).

Ejemplo:

Código (bash) [Seleccionar]
read nombre apellido1 apellido2
echo Los valores leidos por teclado son: $nombre $apellido1 $apellido2


----------   Variables especiales:

$0: Nombre del Shell-Script que se está ejecutando.
$n: Parámetro o argumento pasado al Shell-Script en la posición n, n=1,2,...
$#: Número de argumentos.
$*: Lista de todos los argumentos.
$$: PID del proceso que se está ejecutando.
$!: PID del último proceso ejecutado.
$?: Salida del último proceso ejecutado



Le toca a otro :D

..sR. aDiKtO..

#3
Ampliando un poco lo de las varibles, una varible puede tener como valor el resultado de un programa, por ejemplo:
variable=$(ls --help)
El unico problema es k lo guarda sin retorno de carro. Para ver el resultado, ya sabeis:
Código (bash) [Seleccionar]
echo $variable
Una cosa curiosa es k los bash scripts tienen un error con la orden 'echo', para solucionarlo simplemente hay k utilizar la opcion '-e' y poner entre comillas los datos a mostrar:
Código (bash) [Seleccionar]
echo -e "Asi ya no hay error"

Ademas se puede utilizar la barra invertida al final de una linea para indicar al echo k la frase continua, aparte el echo tiene opciones adicionales como salto de linea, .... (vease 'man echo' y 'info echo'), por ejemplo:
Código (bash) [Seleccionar]
echo -e " toda \
esta \
frase \
se \
escribe \
en la \
misma \
linea."

Esto puede ser util para organizar el bash script, al final os pondre un ejemplo k kedara todo mas claro ;)
                     
En los bash scripts se pueden utilizar colores en las letras y de fondo, simplemente con:
echo -e "\033[1;37m BLANCO \033[0m"
Con lo k mostrara la palabra BLANCO en color blanco ( k original k soy xDD). Veamos ahora k colores hay:
Negro --> 0;30m    
Gris oscuro --> 1;30m
Rojo --> 0;31m    
Rojo claro --> 1;31m
Verde --> 0;32m    
Verde claro --> 1;32m
Marrón --> 0;33m    
Amarillo --> 1;33m
Azul --> 0;34m    
Azul claro --> 1;34m
Púrpura --> 0;35m    
Púrpura claro --> 1;35m
Cyan --> 0;36m    
Cyan claro --> 1;36m
Gris claro --> 0;37m    
blanco --> 1;37m

Como habreis supuesto para k las palabras salgan de un color hay k escribir:
Código (bash) [Seleccionar]
\033[
Y acontinuacion el codigo del color k os he puesto un par de lineas mas arriba, y para volver al color por defecto:
Código (bash) [Seleccionar]
\033[0m

Tambien se puede cambiar el color de fondo, de la misma manera k antes pero con esta codificacion:
Código (bash) [Seleccionar]
Negro --> 40m  
Rojo --> 41m  
Verde --> 42m  
Marrón --> 43m  
Azul --> 44m  
Púrpura --> 45m    
Cyan --> 46m
Gris claro --> 47m


Un shell script k os puede ser util para recordar esto es:
Código (bash) [Seleccionar]
#!/bin/bash
 echo "  Sobre gris claro:        Sobre negro:"
 echo -e "\033[47m\033[1;37m  Blanco        \033[0m\
 1;37m \
 \033[40m\033[1;37m  Blanco        \033[0m"
 echo -e "\033[47m\033[37m  Gris Claro    \033[0m\
    37m \
 \033[40m\033[37m  Gris Claro    \033[0m"
 echo -e "\033[47m\033[1;30m  Gris          \033[0m\
  1;30m \
 \033[40m\033[1;30m  Gris          \033[0m"
 echo -e "\033[47m\033[30m  Negro         \033[0m\
    30m \
 \033[40m\033[30m  Negro         \033[0m"
 echo -e "\033[47m\033[31m  Rojo          \033[0m\
    31m \
 \033[40m\033[31m  Rojo          \033[0m"
 echo -e "\033[47m\033[1;31m  Rojo Claro    \033[0m\
  1;31m \
 \033[40m\033[1;31m  Rojo Claro    \033[0m"
 echo -e "\033[47m\033[32m  Verde         \033[0m\
    32m \
 \033[40m\033[32m  Verde         \033[0m"
 echo -e "\033[47m\033[1;32m  Verde Claro   \033[0m\
  1;32m \
 \033[40m\033[1;32m  Verde Claro   \033[0m"
 echo -e "\033[47m\033[33m  Marrón        \033[0m\
    33m \
 \033[40m\033[33m  Marrón        \033[0m"
 echo -e "\033[47m\033[1;33m  Amarillo      \033[0m\
  1;33m \
 \033[40m\033[1;33m  Amarillo      \033[0m"
 echo -e "\033[47m\033[34m  Azul          \033[0m\
    34m \
 \033[40m\033[34m  Azul          \033[0m"
 echo -e "\033[47m\033[1;34m  Azul Claro    \033[0m\
  1;34m \
 \033[40m\033[1;34m  Azul Claro    \033[0m"
 echo -e "\033[47m\033[35m  Púrpura       \033[0m\
    35m \
 \033[40m\033[35m  Púrpura       \033[0m"
 echo -e "\033[47m\033[1;35m  Rosa          \033[0m\
  1;35m \
 \033[40m\033[1;35m  Rosa          \033[0m"
 echo -e "\033[47m\033[36m  Cyan          \033[0m\
    36m \
 \033[40m\033[36m  Cyan          \033[0m"
 echo -e "\033[47m\033[1;36m  Cyan Claro    \033[0m\
  1;36m \
 \033[40m\033[1;36m  Cyan Claro    \033[0m"



Las variables son creadas y destruidas en el mismo script, si se kieren conservar o exportar a otra shell o script, se hace con:
export variable
A partir de ese momento la varible 'variable' k era local, pasa a ser global, y se puede utilizar en otras shells.

EMPICAL

#4
Ejecuciones condicionales while

Para utilizar las instrucciones condicionales es necesario crear una condicion que la instruccion pueda comprobar.

While

El comando while realiza acciones mientras se cumpla una condicion determinada.

Para entender mejor como funciona esta condicion vamos a poner un ejemplo:

Queremos mandar mails cada cierto tiempo a una serie de personas y se seguiran enviando mientras no recibamos contestacion. Una vez recibida la contestacion los mails dejaran de enviarse.

>set saludo=('/bin/mail | grep -cv No')
>wihle ($saludo==0)
?echo "Hola soy EMPICAL" | mail destinatario@caulquiera.mail
?echo "Hola soy EMPICAL" | mail destinatario2@caulquiera.mail
?sleep 60
?set saludo=('/bin/mail | grep -cv No')
?end

Hacemos uso del comando grep para contar las lineas (opcion -c) que no conitienen la palabra No y asigna el resultado a la variable saludo. Si se canaliza la salida de /bin/mail hacia grep y se escriben los cambios entre comillas, la shell ejecutara los cambios y devolvera la salida correspondiente, que es lo que se asigna a la variable.

Si fueramos malintencionados y variaramos algunas cosillas de este script podriamos llenar los buzones de correo de los destinatarios en poco tiempo, por que mientras la variable $saludo siga siendo igual a 0 el script seguira enviando mensajes cada 60 segundos a los destinatarios. Piensen que pasaria si modificamos sleep y le damos el valor 1, apenas daria tiempo a los destinatarios a contestar y tendrian su correo saturado y mientras no se reciba ningun mensaje en /bin/mail seguira mandando mensajes. Ya que hasta que no reciba un mail no habra una linea que no contenga No en mail.

Mientras se cumpla la condicion while ($saludo==0) mail seguira enviando mensajes a los destinatarios.

cokeing

#5
(X)Dialogs

Xdialog y dialog son dos utilidades clásicas que permiten mejorar tus scripts de shell con una interfaz de usuario gráfica.

INGRESAR (X)DIALOG...
con dialog y Xdialog puedes diseñar una aplicación gráfica con sólo escribir un breve shell script. Dialog es un programa basado puramente en terminal Xdialog es un programa X11.
Aquí encontrarás un ejemplo:
Puedes escribir (o bien copiar/pegar) lo siguiente en un shell (xterm, konsole,....):

Código (bash) [Seleccionar]

bash
Xdialog --yesno "Do you want to learn more about Xdialog?" 0 0;\
case $? in
0)
echo "Result: Yes chosen.";;
1)
echo "Result: No chosen.";;
255)
echo "ESC pressed.";;
esac


Si utilizas dialog en lugar de Xdialog (borra la X en la segunda línea en el script que se mostró anteriormente) entonces obtendrás una aplicación que se basa en expresiones extrañas que se ejecuta sin el xterm y no abre una ventana separada. En algunos casos esto es más apropiado para un shell script ya que se ejecuta simplemente sin la ventana terminal. Esto es importante si quieres ejecutarlo en forma remota con diferentes hosts entre tu ordenador y el host remoto donde el IP routing no se encuentra disponible. En este caso dialog funcionará pero no podrás iniciar una aplicación X11 del tipo de Xdialog.


Lo que antecede es una aplicación de dialog/Xdialog que no tendrá realmente uso pero que muestra que fácil es programar un diálogo gráfico sencillo. Hay más cuadros de diálogo interesantes tales como calendario, menúes, administrador de archivos, barra progess, cuadro de texto, casilla de mensajes, cuadro de diálogo de contraseña, ... Si ejecutas:


dialog --help
o
Xdialog --help


verás un listado de los cuadros de diálogo disponibles. Xdialog ofrece algunos cuadros de diálogo más que dialog.  


COMO FUNCIONA
Los cuadros de diálogo se configuran en la línea de comandos.

dialog --yesno "text string" <height> <width>


Después de escribir dialog o Xdialog debes indicar el nombre del cuadro de diálogo que deseas, seguido de sus parámetros específicos.

El cuadro de diálogo yesno implica 3 parámetros. La altura <height> y el ancho <width> pueden establecerse en cero en cuyo caso la geometría del cuadro será ajustada automáticamente al tamaño del texto. El resultado se mostrará como estado de salida en la pantalla en la variable "$?". Si hay algo más que deba mostrarse como, por ejemplo, los nombres de las opciones seleccionadas, esto aparecerá como un error tipo (o standard). El error tipo (o standard) generalmente se muestra en la pantalla pero puede ser redireccionado con ">2".

Una solución muy simple pero eficaz.  

-----------------------------------------------------------------------------
Venga. que lo siga otro ;-)

2.6.0-r

#6
Hace tiempo que empece con este script pero lo tengo aparcado y apenas ya me pongo a hacer script de bash. Si alguien quiere mejorarlo es libre de hacerlo.
Código (bash) [Seleccionar]

#!/bin/bash
#bash script para la administracion de bases de datos mysql
#by ^Galactus^, estelaplateada@softhome.net
echo ".:EASYMySQL:. script para la administración de bases de datos MySQL"

#Comrpobando la existencia de gdialog en el sistema
if [ ! -f wihch dialog ]; then
echo "No se ha encontrado gdialog, necesitas gnome-utils"
exit 1
fi

       echo "..-:IDENTIFICANDOSE:-.."
       echo Nombre de usuario:
       read NAME
       echo Nombre del host:
       read HOST
       echo Password
       read PASS

#Función para crear bases de datos en mysql.
function create {
       echo "..-:CREAR UNA BASE DE DATOS NUEVA:-.."
       echo Nombre de la base de datos:
       read new_db
       touch newbase.sql
       date > easylog.sql
       echo "CREATE DATABASE $new_db;" >> newbase.sql
       echo "SHOW DATABASES;" >> newbase.sql
       echo "quit" >> newbase.sql
       cat newbase.sql >> easylog.sql
       dialog --title "Crear Una Base De Datos Nueva" --yesno "¿Desea crear la nueva db?" 200 100;
       if [ $? -eq 0 ]; then
       mysql -t -h $HOST  -u $NAME -p $PASS < newbase.sql
       rm newbase.sql
       else
       create
       rm newbase.sql
       fi

}
#Función para borrar bases de datos en mysql
function drop {
       echo "..-:BORRAR BASE DE DATOS:-.."
       echo "SHOW DATABASES;" > delbase.sql
       mysql -t -h $HOST -u $NAME -p $PASS < delbase.sql
       echo "Escriba el nombre de la base de datos que desea borrar:"
       read del_db
       echo "Esta seguro de querer borrar esta db '$del_db' [si/no]"
       read respuesta
       if [ $respuesta = si ]; then
       echo "DROP DATABASE $del_db;" > delbase.sql
       echo "SHOW DATABASES;" >> delbase.sql
       mysql -t -h $HOST -u $NAME -p $PASS < delbase.sql
       cat delbase.sql >> easylog.sql
       rm delbase.sql
       elif [ $respuesta = no ]; then
       drop
       else
       exit
       fi

}
#Función para crear tablas en db de mysql
function table {
       echo "..-:CREAR NUEVA TABLA EN DB:-.."
       echo "SHOW DATABASES;" > newtable.sql
       mysql -t -h $HOST -u $NAME < newtable.sql
       echo "Escriba el nombre de la base de datos en la que desea crear una tabla:"
       read db_use
       echo "USE $db_use;" > newtable.sql
       mysql -t -h $HOST -u $NAME < newtable.sql
       echo "Cambiando a.. '$db_use'"
       echo "Escriba el nombre de la tabla que desea crear en la db '$db_use':"        read db_table
       echo "CREATE TABLE $db_table;" > newtable.sql
       mysql -t -h $HOST -u $NAME < newtable.sql
       echo "Se creo la tabla '$db_table' en la db '$db_use'."
       date > easylog.sql
       cat newtable.sql >> easylog.sql
       rm newtable.sql
       }

#menú de opciones
opcion="Crear_db Borrar_db Crear_Tabla Salir"
select opt in $opcion; do
       if [ "$opt" = "Crear_db" ]; then
       touch easylog.sql
       create
       elif [ "$opt" = "Borrar_db" ]; then
       touch delbase.sql
       drop
       elif [ "$opt" = "Crear_Tabla" ]; then
       touch newtable.sql
       table
       elif [ "$opt" = "Salir" ]; then
       echo GOOD BYE
       exit
       else
       echo seleccione una opción correcta
       fi


Tengo algunas funciones mas para mysql echas en bash, si me da la vena las pondre por aqui.

2.6.0-r

#7
un lapsus, no me habia dado cuenta que puse las mismas funciones que el script anterior.

Os pongo este, que es un menu grafico que empece a hacer para mi script de mysql. Como el anterior tampoco esta terminado.

Código (bash) [Seleccionar]

#!/bin/bash
#bash script para la administracion de bases de datos mysql
echo ".:EASYMySQL:. script para la administración de bases de datos MySQL"
#source llama al archivo que contiene las funciones
source functions

dialog --backtitle "EasyMySQL" --title "Login" --clear --ok-label "Aceptar"\
      --inputmenu "Para poder usar EasyMySQL debe ingresar los datos de usuario y conexion" 20 50 10 \
"Nombre de usuario:"    "$user" \
"Nombre del host:"      "$host" \
"Password:"             "$pass" \

#menú de opciones
echo "[1] Crear Base De Datos"
echo "[2] Borrar Base De Datos"
echo "[3] Crear Tabla"
echo "[4] Salir"
echo
read -p "Seleccione una opcion: " OPCION
case $OPCION in
1) create;;
2) drop;;
3) table;;
4) exit;;
esac

..sR. aDiKtO..

#8
Yo tambien quiero poner un pequeño script que hice hace un par de dias:
Código (bash) [Seleccionar]
#!/bin/sh
#
############################################################################
# "adk.sh v0.1" hecho por el ..sR. aDiKtO.. <adikto@elhacker.net>
#
# Este script borra todas las lineas de todos los archivos
# que esten en /var que contengan tu ip, conservando la fecha
# de antes de la modificacion .
#
# Este script se distribuye segun la licencia GPL v.2 o posteriores y
# no tiene garantias de ningun tipo. Puede obtener una copia de la
# licencia GPL o ponerse en contacto con la Free Software
# Foundation en http://www.gnu.org
############################################################################
#
# Variable que contiene tu ip
IP="127.0.0.1"

############################################################################
# Funcion encargada de limpiar todos los logs
############################################################################
function main()
{
mkdir -p /tmp/.adk &>/dev/null
for i in `find /var 2>/dev/null`
do
 linea=$(cat $i 2>/dev/null | grep $IP )
 if [ "$linea" != "" ]
 then
     ls -l $i > /tmp/.adk/fecha 2>/dev/null
     aux=$(awk '{ print $6 $7 }' /tmp/.adk/fecha 2>/dev/null)
     TIEMPO=$(echo $aux | tr "-:" "\000" 2>/dev/null)
     sed "/$IP/d" $i > /tmp/.adk/datos 2>/dev/null
     cat /tmp/.adk/datos > $i 2>/dev/null
     touch -t $TIEMPO $i 2>/dev/null
     echo -e "IP limpiada del archivo $i"
 fi
done

rm -rf /tmp/.adk &>/dev/null
}

############################################################################
#Funcion principal
############################################################################
clear
echo -e "\n    \033[40m\033[1;37m  adk.sh v0.1  \033[0m\n"
if [ $GROUPS != 0 ]
then
 echo -e "ERROR: Necesitas tener privilegios de root para poder ejecutar este script"
 exit
fi
echo -e "\n Empecemos a borrar tu ip de los log de este sistema.\n"
main
echo -e "\n Ordenador Limpiado Completamente"

SALU2

2.6.0-r

Wow! casi, casi lo que estaba buscando... Funcionaria en un SunOS?

Solaris creo que no usa bash ¿no?