Backups incrementales vía FTP en Linux
Recientemente he decidido pagar un servidor compartido para hacer las copias de seguridad de mis documentos, imágenes y demás. Es un servidor realmente barato pero con la pega que no tengo acceso vía SSH, lo que me ha impedido utilizar aplicaciones como rsync o similares.
Después de buscar un rato la única solución viable que he encontrado ha sido lftp, ya que es el único modo que he encontrado de hacer copias incrementales vía FTP.
lftp tiene un método llamado “mirror” que es el que nos permitirá hacer esto. Un ejemplo sencillo de su uso sería…
lftp -c "set ftp:list-options -a; open ftp://usuarioftp:passwordftp@hostftp; lcd /directorio/donde/copiar; cd /directorio/local/a/copiar; mirror --reverse \ --delete; close -a;"
Como también quería hacer backups de MySQL he decidido entretenerme un rato y hacer mi primer script de bash para hacer copias de mysql y directorios del sistema periódicamente (cada dos días a través de un cron).
Para utilizar el script necesitaréis tener instalado en vuestro servidor linux los paquetes lftp y mailx (o hairloom-mailx); este último servirá para enviarnos un e-mail en caso de error en la subida de nuestros backups.
#!/bin/bash
# :: General Backup Setup ::
BACKUP=/tmp/backup.$$
NOW=$(date +"%Y%m%d")
EMAILID="tu_correo@ejemplo.com"
DOMYSQL=1 # 0 para no hacer copia de seguridad de mysql
DOSYSTEM=1 # 0 para no hacer copia de seguridad del sistema
DEBUG=0 # 1 activa un poco de debug (pero solo un poco xD)
# :: File System Backup Setup ::
FOLDERS=( '/home/usuario/Documentos' '/home/usuario/Imágenes' )
# FTP destination folder
SFOLDER="directorio_destino_donde_guardar_la_copia"
# :: MySQL Backup Setup ::
MUSER="usuarioSQL"
MPASS="pass"
MHOST="localhost"
# MySQL FTP destination folder
MFOLDER="directorio_destino_donde_guardar_los_sql"
# :: FTP server Setup ::
# Root FTP folder
RCD="/directorio_raiz_del_servidor"
FUSER="usuario"
FPASS="contraseña"
FHOST="servidor.dominio"
FOPTIONS="set ftp:list-options -a;set ftp:ssl-allow no" #;set net:limit-total-rate 61440"
# end config
# :: Binaries ::
MYSQL="$(which mysql)"
MYSQLDUMP="$(which mysqldump)"
GZIP="$(which gzip)"
FTP="$(which lftp)"
debug() {
if [ $DEBUG == 1 ]; then
echo $1
fi
}
if [ $DOMYSQL == 1 ]; then
# Create backup folder
debug "Creating backup folder ${BACKUP}"
[ ! -d $BACKUP ] && mkdir -p "${BACKUP}" || :
# Start MySQL backup
debug "Showing databases..."
DBS="$($MYSQL -u $MUSER -h $MHOST -p$MPASS -Bse 'show databases')"
for db in $DBS
do
if [ $db != "information_schema" ]; then
debug "Creating $db.sql.gz backup file"
FILE=$BACKUP/$db.sql.gz
$MYSQLDUMP -u $MUSER -h $MHOST -p$MPASS $db | $GZIP -9 > $FILE
chmod 755 $FILE
fi
done
# Start FTP backup using lftp
debug "Starting FTP transaction for MySQL backup files..."
$FTP -c "$FOPTIONS;
open ftp://$FUSER:$FPASS@$FHOST;
lcd $BACKUP;
cd $RCD/$MFOLDER;
mkdir ${NOW};
cd ${NOW};
mirror --reverse \
--delete;
close -a;"
# Find out if ftp mysql backup failed or not
if [ "$?" != "0" ]; then
debug "FTP upload failed"
T=/tmp/backup.fail
echo "Date: $(date)">$T
echo "Hostname: $(hostname)" >>$T
echo "Backup failed" >>$T
mail -s "MYSQL BACKUP FAILED" "$EMAILID" <$T
rm -f $T
fi
# Delete files
debug "Removing files..."
if [ $DEBUG ]; then
rm -frv $BACKUP
else
rm -fr $BACKUP
fi
fi
if [ $DOSYSTEM == 1 ]; then
# Start System Backup
# Get number of folders
FELEM=${#FOLDERS[@]}
for (( i=0;i<$FELEM;i++)); do
THISFOLDER=${FOLDERS[${i}]}
if [ -d $THISFOLDER ]; then
debug "$THISFOLDER exists"
REMOTEFOLDER=( $(echo $THISFOLDER | tr "/" " ") )
REMOTEFOLDER=${REMOTEFOLDER[${#REMOTEFOLDER[@]}-1]}
debug "Using $REMOTEFOLDER as remote folder name"
debug "Starting FTP transaction from $THISFOLDER to $RCD/$SFOLDER/$REMOTEFOLDER"
# FTP transfer
$FTP -c "$FOPTIONS;
open ftp://$FUSER:$FPASS@$FHOST;
lcd $THISFOLDER;
cd $RCD/$SFOLDER;
mkdir $REMOTEFOLDER;
cd $REMOTEFOLDER;
mirror --reverse \
--delete;
close -a;"
# Find out if ftp system backup failed
if [ "$?" != "0" ]; then
debug "File system backup failed"
T=/tmp/backup.fail
echo "Date: $(date)">$T
echo "Hostname: $(hostname)" >>$T
echo "Backup failed on dir $THISFOLDER" >>$T
mail -s "SYSTEM BACKUP FAILED" "$EMAILID" <$T
rm -f $T
fi
fi
done
fi
Como he dicho, es el primer script de bash que he hecho en la vida (a parte de alguno otro realmente tonto..) así que si me proponéis ideas para mejorarlo o encontráis algún error os agradecería que me lo comentarais 


Acabo de arreglar un fallico de nada y de paso he añadido un poquitín de debug
Ei, genial tu!
L’he copiat tal qual i sembla que funciona a les 1000 meravelles.
Gràcies!
Me n’alegro nano! La veritat és que a mi em va de perles! De fet vaig descobrir fa poc que en realitat sí tinc accés SSH als servidors (tenia que habilitar-ho..) i tot i així prefereixo deixar això abans d’enmerdar-me a fer un script amb rsync (entre d’altres coses per que hi ha cops que no puc accedir via SSH.. hostings compartits de merda…)
Gracias, fenómeno!