Testé sur Debian Lenny et Debian Squeeze – MySQL 5.0 et 5.1
La réplication MySQL permet de maintenir en temps réel une copie d’une ou plusieurs bases d’un serveur maître vers un serveur esclave. Ce n’est pas à proprement parler une technique de sauvegarde, mysqldump est plus adapté pour ça, mais c’est très efficace pour implémenter un système à tolérance de panne. La mise en oeuvre n’est pas très compliquée mais un peu laborieuse et demande un peu de rigueur dans la planification des étapes.
Les tâches à réaliser seront les suivantes :
sur le serveur maître
Créer un utilisateur MySQL, ici ce sera replicuser avec le privilège « REPLICATION SLAVE », et le droit de connexion depuis le serveur esclave.
CREATE USER 'replicuser'@'serveur.esclave' IDENTIFIED BY '<mot_de_passe>'; GRANT REPLICATION SLAVE ON * . * TO 'replicuser'@'serveur.esclave' IDENTIFIED BY '<mot_de_passe>' ;
Configurer l’enregistrement des logs binaires dans le fichier /etc/mysql/my.cnf
[mysqld] # Un nombre entre 1 et 2^32-1, unique dans la communauté de réplication server-id = 123456 # Fichier des logs binaires log_bin = /var/log/mysql/mysql-bin.log # Fichier des logs d'erreur (facultatif) log_error = /var/log/mysql/mysql-bin.err # Liste des bases à répliquer binlog_do_db = base1, base2 # Si on veut répliquer toutes les bases, on ignorera binlog_do_db # et on exclura les bases système avec bin_log_ignore_db : #binlog_ignore_db = mysql, information_schema # Durée de rétention des logs expire_logs_days = 10 # Taille maxi des logs max_binlog_size = 100M
redémarrer le serveur MySQL :
/etc/init.d/mysql restart
sur le serveur esclave
Donner un server-id dans le fichier /etc/mysql/my.cnf
[mysqld] # Un nombre entre 1 et 2^32-1, unique dans la communauté de réplication server-id = 456789
redémarrer le serveur MySQL :
/etc/init.d/mysql restart
Les deux serveurs sont maintenant configurés, il ne reste plus qu’à initialiser les bases sur le serveur esclave et démarrer la réplication. Les étapes détaillées sont les suivantes :
sur le serveur maître
Verrouiller les tables en écriture :
FLUSH TABLES WITH READ LOCK;
Noter les valeurs « File » et « Position » de l’enregistrement des logs binaires avec la commande « SHOW MASTER STATUS ». Ces valeurs serviront à positionner les paramètres « MASTER_LOG_FILE » et « MASTER_LOG_POS » sur l’esclave :
SHOW MASTER STATUS; +------------------+----------+--------------------------+------------------+ | File | Position | Binlog_Do_DB | Binlog_Ignore_DB | +------------------+----------+--------------------------+------------------+ | mysql-bin.000029 | 106 | base1, base2 | | +------------------+----------+--------------------------+------------------+ 1 row in set (0.00 sec)
sur le serveur esclave
Recopier les bases du serveur maître avec la commande mysqldump. On suppose ici que le serveur esclave puisse faire une connexion MySQL en root sur le serveur maître :
mysqldump --host serveur.maitre --user root --password=mot.de.passe base1 | mysql --user root --password=mot.de.passe base1 mysqldump --host serveur.maitre --user root --password=mot.de.passe base2 | mysql --user root --password=mot.de.passe base2
sur le serveur maître
Déverrouiller les tables :
UNLOCK TABLES;
sur le serveur esclave
Stopper les threads esclaves :
STOP SLAVE;
Appliquer les paramètres de réplication. Les paramètres « MASTER_LOG_FILE » et « MASTER_LOG_POS » ont été récupérés précédemment sur le maître :
CHANGE MASTER TO \ MASTER_HOST='serveur.maitre', \ MASTER_USER='replicuser', \ MASTER_PASSWORD='mot_de_passe', \ MASTER_LOG_FILE='log_binaire_courante', \ MASTER_LOG_POS=Position_log_binaire;
Lancer la réplication :
START SLAVE;
On peut vérifier l’état de la réplication avec la commande MySQL « SHOW PROCESSLIST »
sur le serveur maître
SHOW PROCESSLIST\G *************************** 1. row *************************** Id: 27037 User: replicuser Host: serveur.maitre:53723 db: NULL Command: Binlog Dump Time: 1172 State: Has sent all binlog to slave; waiting for binlog to be updated Info: NULL
sur le serveur esclave
SHOW PROCESSLIST\G *************************** 1. row *************************** Id: 1 User: system user Host: db: NULL Command: Connect Time: 155886 State: Waiting for master to send event Info: NULL *************************** 2. row *************************** Id: 2 User: system user Host: db: NULL Command: Connect Time: 1625 State: Has read all relay log; waiting for the slave I/O thread to update it Info: NULL
L’initialisation de la réplication étant assez fastidieuse, un petit script simplifiera la vie. En voici un plutôt basique, à configurer avec les valeurs de réplication réelles, et à exécuter sur le serveur esclave. Les tests sont sommaires, je ne donne aucune garantie sur le bon fonctionnement du script dans votre environnement!
#!/bin/bash # -------------------------------------- # Template initialisation de la réplication MySQL # A exécuter sur le serveur esclave # François Grange 2012 # -------------------------------------- # Inits # User MySQL esclave avec droits root MYSQL_LUSER="root" MYSQL_LPASSWD="" # Saisie interactive # Serveur maitre MYSQL_RHOST="adresse.du.serveur.maitre" # User MySQL maitre avec droits root MYSQL_RUSER="root" MYSQL_RPASSWD="" # Saisie interactive # User MySQL de réplication MYSQL_REPL_USER="user_replication" MYSQL_REPL_PASSWD="mot_de_passe" # PorSSH du serveur maitre PSSH=22 # Bases à synchroniser dbl="base1 base2" echo "+++$(date) Debut $0" # Saisie du mot de passe root local while [ ${#MYSQL_LPASSWD} -eq 0 ]; do read -r -s -p "Mot de passe $MYSQL_LUSER MySQL local : " MYSQL_LPASSWD done # Test de connexion à la base mysql locale sql="USE mysql;" req=`mysql -u $MYSQL_LUSER -p$MYSQL_LPASSWD -e "$sql"` if [ $? -ne 0 ]; then echo -e "\n(E) $(date) Impossible de se connecter à la base locale mysql avec l'utilisateur $MYSQL_LUSER. Abandon du traitement" exit 1 else echo -e "\n(I) $(date) Connexion réussie à la base locale mysql." fi # Saisie du mot de passe root distant while [ ${#MYSQL_RPASSWD} -eq 0 ]; do read -r -s -p "Mot de passe $MYSQL_RUSER MySQL $MYSQL_RHOST : " MYSQL_RPASSWD done # Test de connexion à la base mysql distante sql="USE mysql;" req=`mysql --host $MYSQL_RHOST -u $MYSQL_RUSER -p$MYSQL_RPASSWD -e "$sql"` if [ $? -ne 0 ]; then echo -e "\n(E) $(date) Impossible de se connecter à la base mysql de $MYSQL_RHOST avec l'utilisateur $MYSQL_RUSER. Abandon du traitement" exit 1 else echo -e "\n(I) $(date) Connexion réussie à la base mysql de $MYSQL_RHOST." fi # Verrouillage tables maitre sql="FLUSH TABLES WITH READ LOCK;" req=`mysql --host $MYSQL_RHOST -u $MYSQL_RUSER -p$MYSQL_RPASSWD -e "$sql"` if [ $? -ne 0 ]; then echo -e "(E) $(date) Impossible de verrouiller les tables du serveur maitre. Abandon du traitement" exit 1 fi # Récupération de l'état du maitre sql="SHOW MASTER STATUS;" req=($(mysql --host $MYSQL_RHOST -u $MYSQL_RUSER -p$MYSQL_RPASSWD -e "$sql" -B -s)) if [ $? -ne 0 ]; then echo -e "(E) $(date) Echec de récupération de l'état du serveur maitre. Abandon du traitement" exit 1 fi MYSQL_LOGFILE=${req[0]} MYSQL_OFFSET=${req[1]} echo "Fichier de log courant sur $MYSQL_RHOST : $MYSQL_LOGFILE" echo "Offset courant sur $MYSQL_RHOST : $MYSQL_OFFSET" # Dump individuel des bases for db in $dbl do echo "$(date) Dump base $db en cours..." mysqldump --host $MYSQL_RHOST --user $MYSQL_RUSER --password=$MYSQL_RPASSWD $db | mysql --user $MYSQL_LUSER --password=$MYSQL_LPASSWD $db if [ $? -gt 0 ]; then echo "$(date) Erreur Dump base $db sur $MYSQL_RHOST" else echo "$(date) Base $db synchronisée" fi done # Déverrouillage tables maitre sql="UNLOCK TABLES;" req=`mysql --host $MYSQL_RHOST -u $MYSQL_RUSER -p$MYSQL_RPASSWD -e "$sql"` if [ $? -ne 0 ]; then echo -e "(E) $(date) Impossible de déverrouiller les tables du serveur maitre." fi # Arret des threads esclaves sql="STOP SLAVE;" req=`mysql -u $MYSQL_LUSER -p$MYSQL_LPASSWD -e "$sql"` if [ $? -ne 0 ]; then echo -e "(E) $(date) Impossible de mettre fin à l'esclavage :-(" exit 1 fi # Application des paramètres de réplication sur l'esclave sql="CHANGE MASTER TO \ MASTER_HOST='$MYSQL_RHOST', \ MASTER_USER='$MYSQL_REPL_USER', \ MASTER_PASSWORD='$MYSQL_REPL_PASSWD', \ MASTER_LOG_FILE='$MYSQL_LOGFILE', \ MASTER_LOG_POS=$MYSQL_OFFSET;" req=`mysql -u $MYSQL_LUSER -p$MYSQL_LPASSWD -e "$sql"` echo $req if [ $? -ne 0 ]; then echo -e "(E) $(date) Impossible d'appliquer les paramètres de réplication." exit 1 fi # Démarrage de la réplication sql="START SLAVE;" req=`mysql -u $MYSQL_LUSER -p$MYSQL_LPASSWD -e "$sql"` if [ $? -ne 0 ]; then echo -e "(E) $(date) Impossible de démarrer la réplication." fi
Lien
La doc officielle