Backups incrementales vía FTP en Linux

Reading time ~7 minutes

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…

1
2
3
4
5
6
7
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.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
#!/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 :)