Skip to Content

Dupliquer automatiquement une base de données MySQL d’un serveur distant sur un serveur local.

But de ce document

Ce document contient quelques programmes pour sauvegarder et dupliquer automatiquement une base de données MySQL d’un site internet sur un serveur local.

Le but final étant de dupliquer un site sous Spip ou un Wikini ou tout autre programme en PHP+MySQL d’un serveur distant sur un serveur local sans aucune intervention manuelle.

Ce premier article donne une solution pour dupliquer la base de données. L’article suivant donnera une solution pour dupliquer tous les fichiers (.html, .php, .jpg,...) du site distant sur le site local.

Objectifs de cette sauvegarde

La duplication de la base de données doit pouvoir se réaliser sans aucune intervention, car le programmer sera placé dans une Crontab d’un poste sous Linux et sera donc lancé automatiquement tous les jours à heure fixe.

De plus, cette duplication doit fonctionner avec un simple accès FTP au serveur (Pas d’accès total de type SSH).

Sauvegarde de la base de données

Pour sauvegarder la base de données de Spip, il est possible d’utiliser la procédure manuelle en passant par l’interface utilisateur, (Sauvegarde dans un fichier XML), mais il n’est pas possible d’automatiser cette procédure.

Pour le Wikini, il n’existe pas de procédure de sauvegarde et il faut obligatoirement passer par un autre programme comme phpMyAdmin pour arriver à faire une sauvegarde de la base de données. Mais une nouvelle fois, il n’est pas possible d’automatiser la procédure et celle-ci est relativement longue, compliquée et dangereuse.

De plus, l’installation de PhpMyAdmin sur un serveur ouvre des failles de sécurité qui pourraient compromettre les données.

Programme pour faire un dump d’une table MySQL

Sur un serveur sans accès SSH, la seule solution à ma connaissance pour sauvegarder la base de données est de passer par un programme PHP, qui va faire un dump de la table dans un fichier texte. Ensuite, il faudra récupérer par FTP ce fichier texte pour l’utiliser pour créer la table dans la base de données locale.

Ce premier programme va permettre de sauvegarder dans un fichier texte le contenu d’une table de MySQL.

Il serait possible de créer un script permettant de sauvegarder toutes les tables de la base de données, mais ce script serait probablement trop long à s’exécuter et le serveur Web abandonnerait et enverrait une erreur avant d’avoir terminé. C’est pour cela que j’ai réalisé un script permettant de sauvegarder uniquement une table. Ce script sera ensuite appelé autant de fois que nécessaire pour sauvegarder toutes les tables.

Pour faire fonctionner ce programme, il faut que le serveur html puisse créer et accéder en écriture au fichier « sqldump.sql ». Si ce n’est pas le cas, il faut créer ce fichier manuellement et lui donner un accès complet.

Ensuite, il faut renseigner le nom du serveur, l’utilisateur, le mot de passe et la base de données à utiliser pour effectuer la sauvegarde.

Il est possible de préciser le nombre maximum de lignes à transférer (10000 par défaut)

Pour finir, il faut placer ce programme sur le serveur à sauvegarder.

Voici le programme PHP nommé « save_mysql_distant.php »

<?php

$Serveur    ="www.monserveur.com";  //Serveur contenant la base de données

$Utilisateur="MonUtilisateur";      //Utilisateur de connexion à la base de données

$MotDePasse ="MonMotDePasse";       //Mot de passe

$Base       ="MaBase";              //Nom de la base de données

$NbLigMax=10000; //Nombre maximum d'enregistrements à sauvegarder

$LeFichier="sqldump.sql"; //Nom du fichier contenant le dump de la table

$LaTable =$HTTP_GET_VARS['LaTable']; //** Nom de la table passé en paramètre

if ($LaTable=="") {

        echo "Le paramètre LaTable est obligatoire !";

        exit;

}

$Donnees=mysql_structure($Serveur,$Utilisateur,$MotDePasse,$Base,$LaTable,$NbLigMax);

                                         

$fp = fopen($LeFichier, "w+");

if(!$fp) {

        echo "Impossible d'écrire dans le fichier $LeFichier !";

        exit;

}

$NbCar=fputs($fp, $Donnees);

fclose($fp);

                                                                                                 

function mysql_structure($Serveur, $Utilisateur, $MotDePasse, $Base, $LaTable, $NbLigMax) {

        $ct=0;

        mysql_connect($Serveur, $Utilisateur, $MotDePasse)

                or die ("Impossible de se connecter à $Serveur sous $Utilisateur \n");

        mysql_select_db($Base);

        $Resultat = mysql_query("SHOW CREATE TABLE $LaTable");

        if ($Resultat) {

                $insertions = "";

                $tableau = mysql_fetch_array($Resultat);

                $tableau[1] .= ";";

                $dumpsql[] = str_replace("\n", "", $tableau[1]);

                $req_table = mysql_query("SELECT * FROM $LaTable");

                $nbr_champs = mysql_num_fields($req_table);

                $nb_lig = mysql_num_rows($req_table);

                while ($ligne = mysql_fetch_array($req_table) and $ct<$NbLigMax) {

                        $insertions .= "INSERT INTO $LaTable VALUES(";

                        for ($i=0; $i<=$nbr_champs-1; $i++) {

                                $insertions .= "'" . mysql_real_escape_string($ligne[$i]) . "', ";

                        }

                        $insertions = substr($insertions, 0, -2);

                        $insertions .= ");\n";

                        $ct++;

                }

                if ($insertions != "") $dumpsql[] = $insertions;

        }

        if ($NbLigMax<$nb_lig)

                echo "ATTENTION : $NbLigMax / $nb_lig lignes transférées pour ".trim($LaTable);

        else  echo "$LaTable -> ".($nb_lig/1)." lignes transférées.";

        $dumpsql[] = "\n ";

        return implode("\r", $dumpsql);

}

?>

Sécuriser l’accès au programme.

Ce programme ouvre une faille de sécurité sur le serveur, car il permet de sauvegarder n’importe quelle table et ensuite de télécharger le fichier « sqldump.sql ».

Il est donc fortement conseillé de placer ce programme dans un dossier sécurisé par un fichier « .htaccess ».

La sécurisation d’un dossier sur le serveur dépasse le cadre de ce mémo.

Exécuter ce programme

Pour exécuter ce programme, il suffit de saisir son adresse dans un navigateur sans oublier de mettre en paramètre le nom de la table à sauvegarder :

http://www.monserveur.com/save/save_mysql_distant.php?LaTable=MaTable

Pour exécuter ce programme en ligne de commande, il faut utiliser un navigateur qui fonctionne en ligne de commande comme « lynx ».

Si l’accès au programme n’est pas sécurisé, la commande est la suivante :

lynx -dump http://www.monserveur.com/save/save_mysql_distant.php?LaTable=MaTable

Si l’accès au programme est sécurisé, la commande est la suivante :

lynx -dump http://www.monserveur.com/save/save_mysql_distant.php?LaTable=MaTable -auth=LeLogin:LeMotDePasse        

Obtenir la liste des tables de la base de données

Le programme précédent, permet uniquement de sauvegarder une table. Donc, il est nécessaire de connaître la liste des tables pour pouvoir sauvegarder toute la base de données.

Le programme suivant qu’il faut placer également sur le serveur et de préférence dans un dossier sécurisé permet d’obtenir la liste des tables.

Il faut renseigner le nom du serveur, l’utilisateur, le mot de passe et la base de données à interroger.

Il est possible de préciser une liste de tables à exclure de la liste.

Voici le programme PHP nommé « save_mysql_liste_tables.php »

<?php

$Serveur    ="www.monserveur.com";  //Serveur contenant la base de données

$Utilisateur="MonUtilisateur";      //Utilisateur de connexion à la base de données

$MotDePasse ="MonMotDePasse";       //Mot de passe

$Base       ="MaBase";              //Nom de la base de données

//** Liste des tables à exclure de la sauvegarde ***************

$exclure_table=array(

        "spip_visites_articles",

        "spip_index_articles");

//*************************************************************

mysql_connect($Serveur, $Utilisateur, $MotDePasse)

        or die ("Impossible de se connecter à $Serveur sous $Utilisateur \n");

mysql_select_db($Base);

$tables = mysql_list_tables($Base);

while ($donnees = mysql_fetch_array($tables))  {

        $table = $donnees[0];

        if (in_array($table,$exclure_table)==FALSE) echo "$table
";

}

?>

Ce programme peut également se lancer en ligne de commande avec Lynx :

lynx -dump http://www.monserveur.com/save/save_mysql_liste_tables.php

Dupliquer la base de données distante sur un serveur locale.

Ce programme va donc utiliser les deux programmes précédents pour obtenir la liste des tables et sauvegarder chaque table individuellement.

Chaque table sauvegardée sera ensuite téléchargée en local via un accès FTP et sera intégrée dans la base de données du serveur local.

Ce programme contrairement aux deux précédents, sera lancé en local et non pas sur un serveur distant, ce qui permet de le lancer directement en tant que script et non pas à travers un navigateur.

ATTENTION : Ce programme commence par supprimer la table locale avant de la recréer à partir des données du serveur distant.

Voici le programme PHP nommé « save_mysql_local.php »

#!/usr/bin/php

<?php

$Serveur    ="www.monserveur.com";  //Serveur contenant la base de données et le serveur ftp

$Utilisateur="MonUtilisateur";      //Utilisateur de connexion à la base de données

$MotDePasse ="MonMotDePasse";       //Mot de passe

$Base       ="MaBase";              //Nom de la base de données

$ServeurDistant    ="www.monserveur.com"; //Serveur distant contenant la base de données

$LoginDistant      ="MonUtilisateur1";    //Utilisateur de connexion à la base de données

$MotDePasseDistant ="MonMotDePasse1";     //Mot de passe

$ServeurLocal    = 'localhost';           //Serveur local contenant la base de données

$LoginLocal      = 'MonUtilisateur2';     //Utilisateur de connexion à la base de données

$MotDePasseLocal = 'MonMotDePasse2';      //Mot de passe

$BaseLocal       = 'MaBase';              //Nom de la base de données

$MotDePasseSecurite="MonMotDePasse3";     //Mot de passe pour l'accès au site sécurisé

chdir("/tmp"); //Dossier de téléchargement du fichier "sqldump.sql"

//** Récupération de la liste des tables en lancant un script distant *******************

$LesTables=`lynx -dump http://www.coagul.org/save-site/save_mysql_liste_tables.php -auth=$LoginDistant:$MotDePasseSecurite`;

$TabTables=explode("\n",$LesTables);

echo count($TabTables)." tables à transférer... \n";

$TabTables[]="Vide"; //** Permet de vider le fichier sqldump.sql

for ($i=0;$i

        //** Lancement du script distant pour sauvegarder la table indiquée *************

        $table=trim($TabTables[$i]);

        if ($table!="") {

                //echo "Sauvegarde de la table $table en cours... \n";

                $resultat=`lynx -dump http://www.coagul.org/save-site/save_mysql_distant.php?LaTable=$table -auth=$LoginDistant:$MotDePasseSecurite`;

                echo trim($resultat)."\n";

       

                //** Téléchargement via ftp du fichier sauvegardé *******************************

                $NomFichier="sqldump.sql";

                $Connexion=ftp_connect($ServeurDistant)

                        or die ("Serveur $ServeurDistant non disponible ! \n");

                ftp_login($Connexion, $LoginDistant, $MotDePasseDistant)

                        or die ("Utilisateur $LoginDistant inconnu ou mot de passe non valide ! \n");

                ftp_chdir($Connexion, "html/save-site");

                ftp_get ($Connexion, $NomFichier, $NomFichier, FTP_BINARY);

                $ftpclose=ftp_close ($Connexion);

               

                //** Connexion à la base de données locale **************************************

                mysql_connect($ServeurLocal, $LoginLocal, $MotDePasseLocal)

                        or dir ("Connexion impossible à $ServeurLocal ! \n");

                mysql_select_db($BaseLocal)

                        or die ("Impossible de sélectionner la base $BaseLocal ! \n");

               

                //Suprresion de la table local et chargement des nouvelles données **************       

                mysql_query ("DROP TABLE $table");

                `cat "$NomFichier" | mysql -u $LoginLocal -p$MotDePasseLocal $BaseLocal`;

        }

}

?>

Lancement automatique du programme

Pour exécuter le programme précédent, il suffit de le lancer en ligne de commande :

$ ./save_mysql_local.php

Pour exécuter ce programme automatiquement à heure fixe, il faut le placer dans un fichier crontab.

Pour modifier le fichier crontab d’un utilisateur, il faut saisir la commande suivante :

$ crontab -e

Par exemple pour lancer le programme tous les jours à 22:30, il faut écrire la ligne suivante dans le crontab :

30 22 * * * /LeCheminDuProgramme/save_mysql_local.php

Historique des modifications

Version Date Commentaire
0.1 04/05/05 Création par Tony GALMICHE