MySQL高可用架构之MHA 原理与实践

时间:2023-12-30 18:27:02

MHA简介

关于MHA

MHA(Master HA)是一款开源的MySQL的高可用程序,它为MySQL主从复制架构提供了automating master failover 功能。MHA在监控到master节点故障时,会提升其中拥有最新数据的slave节点成为新的master节点,在此期间,MHA会通过与其它从节点获取额外信息来避免一致性方面的问题。MHA还提供了master节点的在线切换功能,即按需切换master/slave节点。
相较于其它HA软件,MHA的目的在于维持MySQL Replication中Master库的高可用性,其最大特点是可以修复多个Slave之间的差异日志,最终使所有Slave保持数据一致,然后从中选择一个充当新的Master,并将其它Slave指向它。

MHA角色部署

MHA 服务有两种角色,MHA Manager(管理节点)和MHA Node(数据节点):
MHA Manager:通常单独部署在一*立的机器上或者直接部署在其中一台slave上(不建议后者),管理多个master/slave集群,每个master/slave集群称作一个application;其作用有二:
(1)master自动切换及故障转移命令运行
(2)其他的帮助脚本运行:手动切换master;master/slave状态检测
MHA node:运行在每台MySQL服务器上(master/slave/manager),它通过监控具备解析和清理logs功能的脚本来加快故障转移。其作用有:
(1)复制主节点的binlog数据
(2)对比从节点的中继日志文件
(3)无需停止从节点的SQL线程,定时删除中继日志

目前MHA主要支持一主多从的架构,要搭建MHA,要求一个复制集群中必须最少有三台数据库服务器,一主二从,即一台充当master,一台充当备用master,另外一台充当从库,因为至少需要三台服务器,出于机器成本的考虑,淘宝也在该基础上进行了改造,目前淘宝TMHA已经支持一主一从。

我们自己使用其实也可以使用1主1从,但是master主机宕机后无法切换,以及无法补全binlog。master的mysqld进程crash后,还是可以切换成功,以及补全binlog的。

官方介绍:https://code.google.com/p/mysql-master-ha/

下图展示了如何通过MHA Manager管理多组主从复制。可以将MHA工作原理总结为如下:

MySQL高可用架构之MHA 原理与实践

(1)从宕机崩溃的master保存二进制日志事件(binlog events);

(2)识别含有最新更新的slave;

(3)应用差异的中继日志(relay log)到其他的slave;

(4)应用从master保存的二进制日志事件(binlog events);

(5)提升一个slave为新的master;

(6)使其他的slave连接新的master进行复制;

MHA组件

(1)、 Manager工具:
– masterha_check_ssh : 检查MHA的SSH配置。
– masterha_check_repl : 检查MySQL复制。
– masterha_manager : 启动MHA。
– masterha_check_status : 检测当前MHA运行状态。
– masterha_master_monitor : 监测master是否宕机。
– masterha_master_switch : 控制故障转移(自动或手动)。
– masterha_conf_host : 添加或删除配置的server信息。
(2)、 Node工具(这些工具通常由MHAManager的脚本触发,无需人手操作)。
– save_binary_logs : 保存和复制master的二进制日志。
– apply_diff_relay_logs : 识别差异的中继日志事件并应用于其它slave。
– filter_mysqlbinlog : 去除不必要的ROLLBACK事件(MHA已不再使用这个工具)。
– purge_relay_logs : 清除中继日志(不会阻塞SQL线程)。
(3)、自定义扩展:
-secondary_check_script:通过多条网络路由检测master的可用性;
-master_ip_failover_script:更新application使用的masterip; (需要修改)
-shutdown_script:强制关闭master节点;
-report_script:发送报告;
-init_conf_load_script:加载初始配置参数;
-master_ip_online_change:更新master节点ip地址;(需要修改)

MHA环境准备

OS:CentOS 6.8

MySQL :5.7.18

MHA 软件包: MHA 0.57

角色            ip地址                 主机名                server_id               "类型 "
Master       10.180.2.163       MHA-M1            13306                     "写入 "
S1              10.180.2.164      MHA-S1              23306                     "读 "  (其实可以一起部署监控,一组MHA 可以多个监控节点)
S2              10.180.2.165      MHA-S2              33306                     "读 ","监控复制组 " (监控一般不能部署到master 节点,防止Master 宕机不能切换)

安装MHA Node 包

(1)在所有节点安装MHA node所需的perl模块(DBD:mysql),并下载MHA 软件包

yum install perl-DBD-MySQL -y   (可能需要epel源)
https://mega.nz/#F!G4oRjARB!SWzFS59bUv9VrKwdAeIGVw (MHA0.57)

(2)在所有的节点安装mha node(包括Manager 节点):

tar xf mha4mysql-node-0.57.tar.gz
cd mha4mysql-node-0.57
perl Makefile.PL
make && make install

安装完成将产生文件如下:

[root@MHA-S1 bin]# ll
total
-r-xr-xr-x root root Aug : apply_diff_relay_logs
-r-xr-xr-x root root Aug : filter_mysqlbinlog
lrwxrwxrwx root root Aug : mysql -> /usr/local/mysql/bin/mysql
lrwxrwxrwx root root Aug : mysqlbinlog -> /usr/local/mysql/bin/mysqlbinlog
-r-xr-xr-x root root Aug : purge_relay_logs
-rwxr-xr-x root root Aug : purge_relay.sh
-r-xr-xr-x root root Aug : save_binary_logs
[root@MHA-S1 bin]# pwd
/usr/local/bin

增加系统环境变量:

echo "export PATH=\$PATH:/usr/local/bin" >> /etc/profile
source ~/.bash_profile

安装MHA Manager 包

tar xf mha4mysql-node-0.57.tar.gz
cd mha4mysql-node-0.57
perl Makefile.PL
make && make install

安装完成后会在/usr/local/bin目录下面生成以下脚本文件

[root@MHA-S2 bin]# pwd
/usr/local/bin
[root@MHA-S2 bin]# ll
total
-r-xr-xr-x root root Aug : apply_diff_relay_logs
-r-xr-xr-x root root Aug : filter_mysqlbinlog
-rwxr-xr-x root root Aug : manager.sh
-r-xr-xr-x root root Aug : masterha_check_repl
-r-xr-xr-x root root Aug : masterha_check_ssh
-r-xr-xr-x root root Aug : masterha_check_status
-r-xr-xr-x root root Aug : masterha_conf_host
-r-xr-xr-x root root Aug : masterha_manager
-r-xr-xr-x root root Aug : masterha_master_monitor
-r-xr-xr-x root root Aug : masterha_master_switch
-r-xr-xr-x root root Aug : masterha_secondary_check
-r-xr-xr-x root root Aug : masterha_stop
-rwxr-xr-x root root Aug : master_ip_failover
-rwxr-xr-x root root Aug : master_ip_failover.old
-rwxr-xr-x root root Aug : master_ip_online_change
-rwxr-xr-x root root Aug : master_ip_online_change.old
lrwxrwxrwx root root Aug : mysql -> /usr/local/mysql/bin/mysql
lrwxrwxrwx root root Aug : mysqlbinlog -> /usr/local/mysql/bin/mysqlbinlog
-rw------- root root Aug : nohup.out
-rwxr-xr-x root root Aug : power_manager
-r-xr-xr-x root root Aug : purge_relay_logs
-rwxr-xr-x root root Aug : purge_relay.sh
-r-xr-xr-x root root Aug : save_binary_logs
-rwxr-xr-x root root Aug : send_report

复制相关脚本到/usr/local/bin目录(软件包解压缩后就有了,不是必须,因为这些脚本不完整,需要自己修改,这是软件开发着留给我们自己发挥的,如果开启下面的任何一个脚本对应的参数,而对应这里的脚本又没有修改,则会抛错,自己被坑的很惨)

[root@MHA-S2 scripts]# ll
total 32
-rwxr-xr-x 1 root root 3443 Jan 8 2012 master_ip_failover #自动切换时vip管理的脚本,不是必须,如果我们使用keepalived的,我们可以自己编写脚本完成对vip的管理,比如监控mysql,如果mysql异常,我们停止keepalived就行,这样vip就会自动漂移
-rwxr-xr-x 1 root root 9186 Jan 8 2012 master_ip_online_change #在线切换时vip的管理,不是必须,同样可以可以自行编写简单的shell完成
-rwxr-xr-x 1 root root 11867 Jan 8 2012 power_manager #故障发生后关闭主机的脚本,不是必须
-rwxr-xr-x 1 root root 1360 Jan 8 2012 send_report #因故障切换后发送报警的脚本,不是必须,可自行编写简单的shell完成。
[root@MHA-S2 scripts]# cp * /usr/local/bin/

配置SSH登录无密码验证

ssh-keygen
ssh-copy-id root@xxx (XXX 请包括自己,要不然后面check-ssh那步要杯具的)

搭建主从复制环境

详解之前双主复制环境搭建文档

保证两台Slave都搭建成功

Slave_IO_Running: Yes
Slave_SQL_Running: Yes
两台slave服务器设置read_only(从库对外提供读服务,只所以没有写进配置文件,是因为随时slave会提升为master
root@localhost:mysql3306.sock [(none)]>set global read_only=

创建监控用户(在master上执行)

grant all privileges on *.* to root@'%' identified by '';
flush  privileges;

至此,复制搭建完毕,后面配置MHA

MHA环境配置

(1) 创建MHA 工作目录

mkdir -p /etc/mha

修改app1.cnf配置文件,修改后的文件内容如下:

[root@MHA-S2 ~]# more /etc/mha/app1.cnf
[server default]
manager_log=/var/log/masterha/app1/manager.log
manager_workdir=/var/log/masterha/app1
master_binlog_dir=/data/mysql//logs
master_ip_failover_script=/usr/local/bin/master_ip_failover
master_ip_online_change_script=/usr/local/bin/master_ip_online_change
password=
ping_interval=
remote_workdir=/tmp
repl_password=
repl_user=repl
report_script=/usr/local/bin/send_report
secondary_check_script=/usr/local/bin/masterha_secondary_check -s MHA-S1 -s MHA-S2
ssh_user=root
user=root [server1]
hostname=MHA-M1
port= [server2]
candidate_master=1 //设置为候选master,如果设置该参数以后,发生主从切换以后将会将此从库提升为主库,即使这个主库不是集群中事件最新的slave
check_repl_delay=0 //默认情况下如果一个slave落后master 100M的relay logs的话,MHA将不会选择该slave作为一个新的master,因为对于这个slave的恢复需要花费很长时间,通过设置check_repl_delay=0,MHA触发切换在选择一个新的master的时候将会忽略复制延时,
这个参数对于设置了candidate_master=1的主机非常有用,因为这个候选主在切换的过程中一定是新的master
hostname=MHA-S1 
port=
[server3]
hostname=MHA-S2
port=

(2)设置relay log的清除方式(在每个slave节点上):

'set global relay_log_purge=0'

注意:

MHA在发生切换的过程中,从库的恢复过程中依赖于relay log的相关信息,所以这里要将relay log的自动清除设置为OFF,采用手动清除relay log的方式。在默认情况下,从服务器上的中继日志会在SQL线程执行完毕后被自动删除。但是在MHA环境中,这些中继日志在恢复其他从服务器时可能会被用到,因此需要禁用中继日志的自动删除功能。定期清除中继日志需要考虑到复制延时的问题。在ext3的文件系统下,删除大的文件需要一定的时间,会导致严重的复制延时。为了避免复制延时,需要暂时为中继日志创建硬链接,因为在linux系统中通过硬链接删除大文件速度会很快。(在mysql数据库中,删除大表时,通常也采用建立硬链接的方式)

MHA节点中包含了pure_relay_logs命令工具,它可以为中继日志创建硬链接,执行SET GLOBAL relay_log_purge=1,等待几秒钟以便SQL线程切换到新的中继日志,再执行SET GLOBAL relay_log_purge=0。

pure_relay_logs脚本参数如下所示:

--user mysql                      用户名
--password mysql 密码
--port 端口号
--workdir 指定创建relay log的硬链接的位置,默认是/var/tmp,由于系统不同分区创建硬链接文件会失败,故需要执行硬链接具体位置,成功执行脚本后,硬链接的中继日志文件被删除
--disable_relay_log_purge 默认情况下,如果relay_log_purge=,脚本会什么都不清理,自动退出,通过设定这个参数,当relay_log_purge=1的情况下会将relay_log_purge设置为0。清理relay log之后,最后将参数设置为OFF。

(3)设置定期清理relay脚本(例如每天一次,所有服务器)

[root@MHA-S2 bin]# more purge_relay.sh
#!/bin/bash
user=root
passwd=
port=
log_dir='/etc/mha/log'
work_dir='/data'
purge='/usr/local/bin/purge_relay_logs' if [ ! -d $log_dir ]
then
mkdir $log_dir -p
fi $purge --user=$user --password=$passwd --disable_relay_log_purge --port=$port --workdir=$work_dir >> $log_dir/purge_relay_logs.log >&1
添加到crontab
[root@MHA-S2 bin]# crontab -l
* * * /bin/bash /root/purge_relay_log.sh

可以手工执行以下是否会报错

检查SSH配置

[root@MHA-S2 bin]# masterha_check_ssh --conf=/etc/mha/app1.cnf
Mon Aug :: - [warning] Global configuration file /etc/masterha_default.cnf not found. Skipping.
Mon Aug :: - [info] Reading application default configuration from /etc/mha/app1.cnf..
Mon Aug :: - [info] Reading server configuration from /etc/mha/app1.cnf..
Mon Aug :: - [info] Starting SSH connection tests..
Mon Aug :: - [debug]
Mon Aug :: - [debug] Connecting via SSH from root@MHA-M1(10.180.2.163:) to root@MHA-S1(10.180.2.164:)..
Mon Aug :: - [debug] ok.
Mon Aug :: - [debug] Connecting via SSH from root@MHA-M1(10.180.2.163:) to root@MHA-S2(10.180.2.165:)..
Mon Aug :: - [debug] ok.
Mon Aug :: - [debug]
Mon Aug :: - [debug] Connecting via SSH from root@MHA-S1(10.180.2.164:) to root@MHA-M1(10.180.2.163:)..
Mon Aug :: - [debug] ok.
Mon Aug :: - [debug] Connecting via SSH from root@MHA-S1(10.180.2.164:) to root@MHA-S2(10.180.2.165:)..
Mon Aug :: - [debug] ok.
Mon Aug :: - [debug]
Mon Aug :: - [debug] Connecting via SSH from root@MHA-S2(10.180.2.165:) to root@MHA-M1(10.180.2.163:)..
Mon Aug :: - [debug] ok.
Mon Aug :: - [debug] Connecting via SSH from root@MHA-S2(10.180.2.165:) to root@MHA-S1(10.180.2.164:)..
Mon Aug :: - [debug] ok.
Mon Aug :: - [info] All SSH connection tests passed successfully.

检查整个复制环境状况

发现有报错,

Tue Aug   ::  - [info] Checking master_ip_failover_script status:
Tue Aug :: - [info] /usr/local/bin/master_ip_failover --command=status --ssh_user=root --orig_master_host=MHA-M1 --orig_master_ip=10.180.2.163 --orig_master_port=
Bareword "FIXME_xxx" not allowed while "strict subs" in use at /usr/local/bin/master_ip_failover line .
Execution of /usr/local/bin/master_ip_failover aborted due to compilation errors.
Tue Aug :: - [error][/usr/local/share/perl5/MHA/MasterMonitor.pm, ln229] Failed to get master_ip_failover_script status with return code 255:0.
Tue Aug :: - [error][/usr/local/share/perl5/MHA/MasterMonitor.pm, ln427] Error happened on checking configurations. at /usr/local/bin/masterha_check_repl line
Tue Aug :: - [error][/usr/local/share/perl5/MHA/MasterMonitor.pm, ln525] Error happened on monitoring servers.
Tue Aug :: - [info] Got exit code (Not master dead).

原来Failover两种方式:一种是虚拟IP地址,一种是全局配置文件。MHA并没有限定使用哪一种方式,而是让用户自己选择,虚拟IP地址的方式会牵扯到其它的软件,比如keepalive软件,而且还要修改脚本master_ip_failover。这里先把app1.cnf  里面 master_ip_failover_script= /usr/local/bin/master_ip_failover这个选项屏蔽才可以通过。

#master_ip_failover_script= /usr/local/bin/master_ip_failover
Tue Aug :: - [info] Got exit code (Not master dead). MySQL Replication Health is OK.

检查MHA Manager的状态

[root@MHA-S2 mha]# masterha_check_status --conf=/etc/mha/app1.cnf
app1 is stopped(:NOT_RUNNING).

手动启动

[root@MHA-S2 mha]# nohup masterha_manager --conf=/etc/mha/app1.cnf --remove_dead_master_conf --ignore_last_failover < /dev/null > /var/log/masterha/app1/manager.log >& &
[]
[root@MHA-S2 mha]# ps -ef|grep masterha
root : pts/ :: perl /usr/local/bin/masterha_manager --conf=/etc/mha/app1.cnf --remove_dead_master_conf --ignore_last_failover
[root@MHA-S2 mha]# masterha_check_status --conf=/etc/mha/app1.cnf
app1 (pid:) is running(:PING_OK), master:MHA-M1

--remove_dead_master_conf      该参数代表当发生主从切换后,老的主库的ip将会从配置文件中移除。(如果发生异常切换之后修复了旧的master,要加进去新的MHA 的话,必须记得app1.cnf回补server1的信息)

--manger_log                            日志存放位置

--ignore_last_failover                 在缺省情况下,如果MHA检测到连续发生宕机,且两次宕机间隔不足8小时的话,则不会进行Failover,之所以这样限制是为了避免ping-pong效应。该参数代表忽略上次MHA触发切换产生的文件,默认情况下,MHA发生切换后会在日志目录,也就是上面我设置的/data产生app1.failover.complete文件,下次再次切换的时候如果发现该目录下存在该文件将不允许触发切换,除非在第一次切换后收到删除该文件,为了方便,这里设置为--ignore_last_failover。

 检查启动日志

[root@MHA-S2 app1]# vi manager.log 

Tue Aug   ::  - [warning] Global configuration file /etc/masterha_default.cnf not found. Skipping.
Tue Aug :: - [info] Reading application default configuration from /etc/mha/app1.cnf..
Tue Aug :: - [info] Reading server configuration from /etc/mha/app1.cnf..
Tue Aug :: - [info] MHA::MasterMonitor version 0.57.
Tue Aug :: - [info] GTID failover mode =
Tue Aug :: - [info] Dead Servers:
Tue Aug :: - [info] Alive Servers:
Tue Aug :: - [info] MHA-M1(10.180.2.163:)
Tue Aug :: - [info] MHA-S1(10.180.2.164:)
Tue Aug :: - [info] MHA-S2(10.180.2.165:)
Tue Aug :: - [info] Alive Slaves:
Tue Aug :: - [info] MHA-S1(10.180.2.164:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Tue Aug :: - [info] GTID ON
Tue Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Tue Aug :: - [info] Primary candidate for the new Master (candidate_master is set)
Tue Aug :: - [info] MHA-S2(10.180.2.165:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Tue Aug :: - [info] GTID ON
Tue Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Tue Aug :: - [info] Current Alive Master: MHA-M1(10.180.2.163:)
Tue Aug :: - [info] Checking slave configurations..
Tue Aug :: - [info] Checking replication filtering settings..
Tue Aug :: - [info] binlog_do_db= , binlog_ignore_db=
Tue Aug :: - [info] Replication filtering check ok.
Tue Aug :: - [info] GTID (with auto-pos) is supported. Skipping all SSH and Node package checking.
Tue Aug :: - [info] Checking SSH publickey authentication settings on the current master..
Tue Aug :: - [info] HealthCheck: SSH to MHA-M1 is reachable.
Tue Aug :: - [info]
MHA-M1(10.180.2.163:) (current master)
+--MHA-S1(10.180.2.164:)
+--MHA-S2(10.180.2.165:) Tue Aug :: - [warning] master_ip_failover_script is not defined.
Tue Aug :: - [warning] shutdown_script is not defined.
Tue Aug :: - [info] Set master ping interval seconds.
Tue Aug :: - [info] Set secondary check script: /usr/local/bin/masterha_secondary_check -s MHA-S1 -s MHA-S2
Tue Aug :: - [info] Starting ping health check on MHA-M1(10.180.2.163:)..
Tue Aug :: - [info] Ping(SELECT) succeeded, waiting until MySQL doesn't respond..

配置VIP

vip配置可以采用两种方式,一种通过keepalived的方式管理虚拟ip的浮动;另外一种通过脚本方式启动虚拟ip的方式(即不需要keepalived或者heartbeat类似的软件)。

这里仅演示使用脚本管理VIP 的方式, 修改master_ip_failover 脚本,使用脚本管理VIP

[root@MHA-M1 ~]# /sbin/ifconfig eth1: 10.180.2.168/

脚本:

[root@MHA-S2 bin]# cat master_ip_failover
#!/usr/bin/env perl use strict;
use warnings FATAL => 'all'; use Getopt::Long; my (
$command, $ssh_user, $orig_master_host, $orig_master_ip,
$orig_master_port, $new_master_host, $new_master_ip, $new_master_port
); my $vip = '10.180.2.168/19';
my $key = '';
my $ssh_start_vip = "/sbin/ifconfig eth1:$key $vip";
my $ssh_stop_vip = "/sbin/ifconfig eth1:$key down"; GetOptions(
'command=s' => \$command,
'ssh_user=s' => \$ssh_user,
'orig_master_host=s' => \$orig_master_host,
'orig_master_ip=s' => \$orig_master_ip,
'orig_master_port=i' => \$orig_master_port,
'new_master_host=s' => \$new_master_host,
'new_master_ip=s' => \$new_master_ip,
'new_master_port=i' => \$new_master_port,
); exit &main(); sub main { print "\n\nIN SCRIPT TEST====$ssh_stop_vip==$ssh_start_vip===\n\n"; if ( $command eq "stop" || $command eq "stopssh" ) { my $exit_code = ;
eval {
print "Disabling the VIP on old master: $orig_master_host \n";
&stop_vip();
$exit_code = ;
};
if ($@) {
warn "Got Error: $@\n";
exit $exit_code;
}
exit $exit_code;
}
elsif ( $command eq "start" ) { my $exit_code = ;
eval {
print "Enabling the VIP - $vip on the new master - $new_master_host \n";
&start_vip();
$exit_code = ;
};
if ($@) {
warn $@;
exit $exit_code;
}
exit $exit_code;
}
elsif ( $command eq "status" ) {
print "Checking the Status of the script.. OK \n";
exit ;
}
else {
&usage();
exit ;
}
} sub start_vip() {
`ssh $ssh_user\@$new_master_host \" $ssh_start_vip \"`;
}
sub stop_vip() {
return unless ($ssh_user);
`ssh $ssh_user\@$orig_master_host \" $ssh_stop_vip \"`;
} sub usage {
print
"Usage: master_ip_failover --command=start|stop|stopssh|status --orig_master_host=host --orig_master_ip=ip --orig_master_port=port --new_master_host=host --new_master_ip=ip --new_master_port=port\n";
}

在app1.cnf 文件中取消刚刚对master_ip_online_failover 的注释并测试:

再次检查MHA check
[root@MHA-S2 bin]# masterha_check_repl --conf=/etc/mha/app1.cnf
Wed Aug :: - [warning] Global configuration file /etc/masterha_default.cnf not found. Skipping.
Wed Aug :: - [info] Reading application default configuration from /etc/mha/app1.cnf..
Wed Aug :: - [info] Reading server configuration from /etc/mha/app1.cnf..
Wed Aug :: - [info] MHA::MasterMonitor version 0.57.
Wed Aug :: - [info] GTID failover mode =
Wed Aug :: - [info] Dead Servers:
Wed Aug :: - [info] Alive Servers:
Wed Aug :: - [info] MHA-M1(10.180.2.163:)
Wed Aug :: - [info] MHA-S1(10.180.2.164:)
Wed Aug :: - [info] MHA-S2(10.180.2.165:)
Wed Aug :: - [info] Alive Slaves:
Wed Aug :: - [info] MHA-S1(10.180.2.164:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Wed Aug :: - [info] GTID ON
Wed Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Wed Aug :: - [info] Primary candidate for the new Master (candidate_master is set)
Wed Aug :: - [info] MHA-S2(10.180.2.165:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Wed Aug :: - [info] GTID ON
Wed Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Wed Aug :: - [info] Current Alive Master: MHA-M1(10.180.2.163:)
Wed Aug :: - [info] Checking slave configurations..
Wed Aug :: - [info] Checking replication filtering settings..
Wed Aug :: - [info] binlog_do_db= , binlog_ignore_db=
Wed Aug :: - [info] Replication filtering check ok.
Wed Aug :: - [info] GTID (with auto-pos) is supported. Skipping all SSH and Node package checking.
Wed Aug :: - [info] Checking SSH publickey authentication settings on the current master..
Wed Aug :: - [info] HealthCheck: SSH to MHA-M1 is reachable.
Wed Aug :: - [info]
MHA-M1(10.180.2.163:) (current master)
+--MHA-S1(10.180.2.164:)
+--MHA-S2(10.180.2.165:) Wed Aug :: - [info] Checking replication health on MHA-S1..
Wed Aug :: - [info] ok.
Wed Aug :: - [info] Checking replication health on MHA-S2..
Wed Aug :: - [info] ok.
Wed Aug :: - [info] Checking master_ip_failover_script status:
Wed Aug :: - [info] /usr/local/bin/master_ip_failover --command=status --ssh_user=root --orig_master_host=MHA-M1 --orig_master_ip=10.180.2.163 --orig_master_port=

IN SCRIPT TEST====/sbin/ifconfig eth1:1 down==/sbin/ifconfig eth1:1 10.180.2.168/19===

Checking the Status of the script.. OK
Wed Aug 9 10:49:43 2017 - [info] OK.
Wed Aug 9 10:49:43 2017 - [warning] shutdown_script is not defined.
Wed Aug 9 10:49:43 2017 - [info] Got exit code 0 (Not master dead).

MySQL Replication Health is OK.

以上就是MHA 安装配置的全过程,以下进行简单的测试。

(1)failover 测试

手动kill 了master 上面的mysqld 进程,查看切换状态

[root@MHA-S2 tmp]# more manager.log
Wed Aug :: - [warning] Got error on MySQL select ping: (MySQL server has gone away)
Wed Aug :: - [info] Executing secondary network check script: /usr/local/bin/masterha_secondary_check -s MHA-S1 -s MHA-S2 --user=root --master_host
=MHA-M1 --master_ip=10.180.2.163 --master_port= --master_user=root --master_password= --ping_type=SELECT
Wed Aug :: - [info] Executing SSH check script: exit
Wed Aug :: - [info] HealthCheck: SSH to MHA-M1 is reachable.
Monitoring server MHA-S1 is reachable, Master is not reachable from MHA-S1. OK.
Monitoring server MHA-S2 is reachable, Master is not reachable from MHA-S2. OK.
Wed Aug :: - [info] Master is not reachable from all other monitoring servers. Failover should start.
Wed Aug :: - [warning] Got error on MySQL connect: (Lost connection to MySQL server at 'reading initial communication packet', system error:
)
Wed Aug :: - [warning] Connection failed time(s)..
Wed Aug :: - [warning] Got error on MySQL connect: (Lost connection to MySQL server at 'reading initial communication packet', system error:
)
Wed Aug :: - [warning] Connection failed time(s)..
Wed Aug :: - [warning] Got error on MySQL connect: (Lost connection to MySQL server at 'reading initial communication packet', system error:
)
Wed Aug :: - [warning] Connection failed time(s)..
Wed Aug :: - [warning] Master is not reachable from health checker!
Wed Aug :: - [warning] Master MHA-M1(10.180.2.163:) is not reachable!
Wed Aug :: - [warning] SSH is reachable.
Wed Aug :: - [info] Connecting to a master server failed. Reading configuration file /etc/masterha_default.cnf and /etc/mha/app1.cnf again, and tryin
g to connect to all servers to check server status..
Wed Aug :: - [warning] Global configuration file /etc/masterha_default.cnf not found. Skipping.
Wed Aug :: - [info] Reading application default configuration from /etc/mha/app1.cnf..
Wed Aug :: - [info] Reading server configuration from /etc/mha/app1.cnf..
Wed Aug :: - [info] GTID failover mode =
Wed Aug :: - [info] Dead Servers:
Wed Aug :: - [info] MHA-M1(10.180.2.163:)
Wed Aug :: - [info] Alive Servers:
Wed Aug :: - [info] MHA-S1(10.180.2.164:)
Wed Aug :: - [info] MHA-S2(10.180.2.165:)
Wed Aug :: - [info] Alive Slaves:
Wed Aug :: - [info] MHA-S1(10.180.2.164:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Wed Aug :: - [info] GTID ON
Wed Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Wed Aug :: - [info] Primary candidate for the new Master (candidate_master is set)
Wed Aug :: - [info] MHA-S2(10.180.2.165:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Wed Aug :: - [info] GTID ON
Wed Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Wed Aug :: - [info] Checking slave configurations..
Wed Aug :: - [info] Checking replication filtering settings..
Wed Aug :: - [info] Replication filtering check ok.
Wed Aug :: - [info] Master is down!
Wed Aug :: - [info] Terminating monitoring script.
Wed Aug :: - [info] Got exit code (Master dead).
Wed Aug :: - [info] MHA::MasterFailover version 0.57.
Wed Aug :: - [info] Starting master failover.
Wed Aug :: - [info]
Wed Aug :: - [info] * Phase : Configuration Check Phase..
Wed Aug :: - [info]
Wed Aug :: - [info] GTID failover mode =
Wed Aug :: - [info] Dead Servers:
Wed Aug :: - [info] MHA-M1(10.180.2.163:)
Wed Aug :: - [info] Checking master reachability via MySQL(double check)...
Wed Aug :: - [info] ok.
Wed Aug :: - [info] Alive Servers:
Wed Aug :: - [info] MHA-S1(10.180.2.164:)
Wed Aug :: - [info] MHA-S2(10.180.2.165:)
Wed Aug :: - [info] Alive Slaves:
Wed Aug :: - [info] MHA-S1(10.180.2.164:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Wed Aug :: - [info] GTID ON
Wed Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Wed Aug :: - [info] Primary candidate for the new Master (candidate_master is set)
Wed Aug :: - [info] MHA-S2(10.180.2.165:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Wed Aug :: - [info] GTID ON
Wed Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Wed Aug :: - [info] Starting GTID based failover.
Wed Aug :: - [info]
Wed Aug :: - [info] ** Phase : Configuration Check Phase completed.
Wed Aug :: - [info]
Wed Aug :: - [info] * Phase : Dead Master Shutdown Phase..
Wed Aug :: - [info]
Wed Aug :: - [info] Forcing shutdown so that applications never connect to the current master..
Wed Aug :: - [info] Executing master IP deactivation script:
Wed Aug :: - [info] /usr/local/bin/master_ip_failover --orig_master_host=MHA-M1 --orig_master_ip=10.180.2.163 --orig_master_port= --command=stopssh --ssh_user=root IN SCRIPT TEST====/sbin/ifconfig eth1: down==/sbin/ifconfig eth1: 10.180.2.168/=== Disabling the VIP on old master: MHA-M1
SIOCSIFFLAGS: Cannot assign requested address
Wed Aug :: - [info] done.
Wed Aug :: - [warning] shutdown_script is not set. Skipping explicit shutting down of the dead master.
Wed Aug :: - [info] * Phase : Dead Master Shutdown Phase completed.
Wed Aug :: - [info]
Wed Aug :: - [info] * Phase : Master Recovery Phase..
Wed Aug :: - [info]
Wed Aug :: - [info] * Phase 3.1: Getting Latest Slaves Phase..
Wed Aug :: - [info]
Wed Aug :: - [info] The latest binary log file/position on all slaves is -binlog.:
Wed Aug :: - [info] Retrieved Gtid Set: a5757eae--11e7-82c7-005056b662d3:-
Wed Aug :: - [info] Latest slaves (Slaves that received relay log files to the latest):
Wed Aug :: - [info] MHA-S1(10.180.2.164:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Wed Aug :: - [info] GTID ON
Wed Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Wed Aug :: - [info] Primary candidate for the new Master (candidate_master is set)
Wed Aug :: - [info] MHA-S2(10.180.2.165:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Wed Aug :: - [info] GTID ON
Wed Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Wed Aug :: - [info] The oldest binary log file/position on all slaves is -binlog.:
Wed Aug :: - [info] Retrieved Gtid Set: a5757eae--11e7-82c7-005056b662d3:-
Wed Aug :: - [info] Oldest slaves:
Wed Aug :: - [info] MHA-S1(10.180.2.164:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Wed Aug :: - [info] GTID ON
Wed Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Wed Aug :: - [info] Primary candidate for the new Master (candidate_master is set)
Wed Aug :: - [info] MHA-S2(10.180.2.165:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Wed Aug :: - [info] GTID ON
Wed Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Wed Aug :: - [info]
Wed Aug :: - [info] * Phase 3.3: Determining New Master Phase..
Wed Aug :: - [info]
Wed Aug :: - [info] Searching new master from slaves..
Wed Aug :: - [info] Candidate masters from the configuration file:
Wed Aug :: - [info] MHA-S1(10.180.2.164:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Wed Aug :: - [info] GTID ON
Wed Aug :: - [info] Replicating from MHA-M1(10.180.2.163:)
Wed Aug :: - [info] Primary candidate for the new Master (candidate_master is set)
Wed Aug :: - [info] Non-candidate masters:
Wed Aug :: - [info] Searching from candidate_master slaves which have received the latest relay log events..
Wed Aug :: - [info] New master is MHA-S1(10.180.2.164:)
Wed Aug :: - [info] Starting master failover..
Wed Aug :: - [info]
From:
MHA-M1(10.180.2.163:) (current master)
+--MHA-S1(10.180.2.164:)
+--MHA-S2(10.180.2.165:) To:
MHA-S1(10.180.2.164:) (new master)
+--MHA-S2(10.180.2.165:)
Wed Aug :: - [info]
Wed Aug :: - [info] * Phase 3.3: New Master Recovery Phase..
Wed Aug :: - [info]
Wed Aug :: - [info] Waiting all logs to be applied..
Wed Aug :: - [info] done.
Wed Aug :: - [info] Getting new master's binlog name and position..
Wed Aug :: - [info] -binlog.:
Wed Aug :: - [info] All other slaves should start replication from here. Statement should be: CHANGE MASTER TO MASTER_HOST='MHA-S1 or 10.180.2.164', MASTER_PORT=, MASTER_AUTO_POSITION=, MAS
TER_USER='repl', MASTER_PASSWORD='xxx';
Wed Aug :: - [info] Master Recovery succeeded. File:Pos:Exec_Gtid_Set: -binlog., , 1c2dc99f-7b57-11e7-a280-005056b665cb:-,
a5757eae--11e7-82c7-005056b662d3:-
Wed Aug :: - [info] Executing master IP activate script:
Wed Aug :: - [info] /usr/local/bin/master_ip_failover --command=start --ssh_user=root --orig_master_host=MHA-M1 --orig_master_ip=10.180.2.163 --orig_master_port= --new_master_host=MHA-S1 --
new_master_ip=10.180.2.164 --new_master_port= --new_master_user='root' --new_master_password=xxx
Unknown option: new_master_user
Unknown option: new_master_password IN SCRIPT TEST====/sbin/ifconfig eth1: down==/sbin/ifconfig eth1: 10.180.2.168/=== Enabling the VIP - 10.180.2.168/ on the new master - MHA-S1
Wed Aug :: - [info] OK.
Wed Aug :: - [info] Setting read_only= on MHA-S1(10.180.2.164:)..
Wed Aug :: - [info] ok.
Wed Aug :: - [info] ** Finished master recovery successfully.
Wed Aug :: - [info] * Phase : Master Recovery Phase completed.
Wed Aug :: - [info]
Wed Aug :: - [info] * Phase : Slaves Recovery Phase..
Wed Aug :: - [info]
Wed Aug :: - [info]
Wed Aug :: - [info] * Phase 4.1: Starting Slaves in parallel..
Wed Aug :: - [info]
Wed Aug :: - [info] -- Slave recovery on host MHA-S2(10.180.2.165:) started, pid: . Check tmp log /var/log/masterha/app1/MHA-S2_3306_20170809174714.log if it takes time..
Wed Aug :: - [info]
Wed Aug :: - [info] Log messages from MHA-S2 ...
Wed Aug :: - [info]
Wed Aug :: - [info] Resetting slave MHA-S2(10.180.2.165:) and starting replication from the new master MHA-S1(10.180.2.164:)..
Wed Aug :: - [info] Executed CHANGE MASTER.
Wed Aug :: - [info] Slave started.
Wed Aug :: - [info] gtid_wait(1c2dc99f-7b57-11e7-a280-005056b665cb:-,
a5757eae--11e7-82c7-005056b662d3:-) completed on MHA-S2(10.180.2.165:). Executed events.
Wed Aug :: - [info] End of log messages from MHA-S2.
Wed Aug :: - [info] -- Slave on host MHA-S2(10.180.2.165:) started.
Wed Aug :: - [info] All new slave servers recovered successfully.
Wed Aug :: - [info]
Wed Aug :: - [info] * Phase : New master cleanup phase..
Wed Aug :: - [info]
Wed Aug :: - [info] Resetting slave info on the new master..
Wed Aug :: - [info] MHA-S1: Resetting slave info succeeded.
Wed Aug :: - [info] Master failover to MHA-S1(10.180.2.164:) completed successfully.
Wed Aug :: - [info] Deleted server1 entry from /etc/mha/app1.cnf .
Wed Aug :: - [info] ----- Failover Report ----- app1: MySQL Master failover MHA-M1(10.180.2.163:) to MHA-S1(10.180.2.164:) succeeded Master MHA-M1(10.180.2.163:) is down! Check MHA Manager logs at MHA-S2:/var/log/masterha/app1/manager.log for details. Started automated(non-interactive) failover.
Invalidated master IP address on MHA-M1(10.180.2.163:)
Selected MHA-S1(10.180.2.164:) as a new master.
MHA-S1(10.180.2.164:): OK: Applying all logs succeeded.
MHA-S1(10.180.2.164:): OK: Activated master IP address.
MHA-S2(10.180.2.165:): OK: Slave started, replicating from MHA-S1(10.180.2.164:)
MHA-S1(10.180.2.164:): Resetting slave info succeeded.
Master failover to MHA-S1(10.180.2.164:) completed successfully.
Wed Aug :: - [info] Sending mail..
Unknown option: conf

以上是切换的全日志过程,我们可以看到MHA 切换主要经历以下步骤:

1.配置文件检查阶段,这个阶段会检查整个集群配置文件配置

2.宕机的master处理,这个阶段包括虚拟ip摘除操作,主机关机操作(这个我这里还没有实现,需要研究)

3.复制dead maste和最新slave相差的relay log,并保存到MHA Manger具体的目录下

4.识别含有最新更新的slave

5.应用从master保存的二进制日志事件(binlog events)

6.提升一个slave为新的master进行复制

7.使其他的slave连接新的master进行复制

MySQL高可用架构之MHA 原理与实践

注意:

1. 切换完之后你会发现MHA Manager 监控程序会自动死掉,官网有如下解释和解决方式:

Running MHA Manager from daemontools
Currently MHA Manager process does not run as a daemon. If failover completed successfully or the master process was killed by accident, the manager stops working. To run as a daemon, daemontool. or any external daemon program can be used. Here is an example to run from daemontools.

这里我们用shell 脚本的方式去执行就不会发生监控程序死掉的情况

[root@MHA-S2 bin]# more manager.sh
#!/bin/sh
nohup masterha_manager --conf=/etc/mha/app1.cnf --remove_dead_master_conf --ignore_last_failover < /dev/null > /var/log/masterha/app1/manager.log >& &

2. 当你修复完死掉的master想重新加入先有的两节点MHA 也是可以的

旧Master :

root@localhost:mysql3306.sock [tt]>show master status\G
*************************** . row ***************************
File: -binlog.
Position:
Binlog_Do_DB:
Binlog_Ignore_DB:
Executed_Gtid_Set: a5757eae--11e7-82c7-005056b662d3:-
row in set (0.00 sec)

现有master:

root@localhost:mysql3306.sock [tt]>show master status\G
*************************** . row ***************************
File: -binlog.
Position:
Binlog_Do_DB:
Binlog_Ignore_DB:
Executed_Gtid_Set: 1c2dc99f-7b57-11e7-a280-005056b665cb:-,
a5757eae--11e7-82c7-005056b662d3:-
row in set (0.00 sec)

由于有GTID,我们可以直接就change master 切换过去,先对比一下数据:

旧master:

root@localhost:mysql3306.sock [tt]>select * from t1;
+----+------+
| id | c1 |
+----+------+
| | a1 |
| | a2 |
| | a3 |
| | a4 |
+----+------+
rows in set (0.02 sec)

新master:

root@localhost:mysql3306.sock [tt]>select * from t1;
+----+------+
| id | c1 |
+----+------+
| | a1 |
| | a2 |
| | a3 |
| | a4 |
| 5 | a5 |
+----+------+

旧master 直接change master to:

change master to master_host='MHA-S1',master_user='repl',master_password='',master_port=,master_auto_position=;

start slave 看输出:

root@localhost:mysql3306.sock [tt]>show slave status\G
*************************** . row ***************************
Slave_IO_State: Waiting for master to send event
Master_Host: MHA-S1
Master_User: repl
Master_Port:
Connect_Retry:
Master_Log_File: -binlog.
Read_Master_Log_Pos:
Relay_Log_File: MHA-M1-relay-bin.
Relay_Log_Pos:
Relay_Master_Log_File: -binlog.
Slave_IO_Running: Yes
Slave_SQL_Running: Yes

看是否会补全数据:

root@localhost:mysql3306.sock [tt]>select * from t1;
+----+------+
| id | c1 |
+----+------+
| | a1 |
| | a2 |
| | a3 |
| | a4 |
| | a5 |
+----+------+

发现数据补全了,加入复制没问题。

最后还得修改app1.cnf 把server1 补上

[server1]
hostname=MHA-M1
port=

重启监控程序并查看MHA 状态

[root@MHA-S2 tmp]# masterha_check_repl --conf=/etc/mha/app1.cnf
Sat Aug :: - [info] Replication filtering check ok.
Sat Aug :: - [error][/usr/local/share/perl5/MHA/Server.pm, ln398] MHA-M1(10.180.2.163:): User repl does not exist or does not have REPLICATION SLAVE privilege! Other slaves can not start replication from this host.
Sat Aug :: - [error][/usr/local/share/perl5/MHA/MasterMonitor.pm, ln427] Error happened on checking configurations. at /usr/local/share/perl5/MHA/ServerManager.pm line
Sat Aug :: - [error][/usr/local/share/perl5/MHA/MasterMonitor.pm, ln525] Error happened on monitoring servers.
Sat Aug :: - [info] Got exit code (Not master dead).

发现权限有问题,赶紧修复一下:

MHA-M1:

set session sql_log_bin=OFF;
grant replication slave on *.* to repl@'%' identified by '';
set session sql_log_bin=ON;

再次执行MHA 状态检查:

masterha_check_repl --conf=/etc/mha/app1.cnf
Sat Aug :: - [info] Checking replication health on MHA-M1..
Sat Aug :: - [info] ok.
Sat Aug :: - [info] Checking replication health on MHA-S2..
Sat Aug :: - [info] ok.
Sat Aug :: - [info] Checking master_ip_failover_script status:
Sat Aug :: - [info] /usr/local/bin/master_ip_failover --command=status --ssh_user=root --orig_master_host=MHA-S1 --orig_master_ip=10.180.2.164 --orig_master_port= IN SCRIPT TEST====/sbin/ifconfig eth1: down==/sbin/ifconfig eth1: 10.180.2.168/=== Checking the Status of the script.. OK
Sat Aug :: - [info] OK.
Sat Aug :: - [warning] shutdown_script is not defined.
Sat Aug :: - [info] Got exit code (Not master dead). MySQL Replication Health is OK.

最后启动监控程序

[root@MHA-S2 bin]# nohup monitor.sh &
[root@MHA-S2 bin]# masterha_check_status --conf=/etc/mha/app1.cnf
app1 (pid:) is running(:PING_OK), master:MHA-S1

(2)手动在线切换测试

在许多情况下, 需要将现有的主服务器迁移到另外一台服务器上。 比如主服务器硬件故障,RAID 控制卡需要重建,将主服务器移到性能更好的服务器上等等。维护主服务器引起性能下降, 导致停机时间至少无法写入数据。 另外, 阻塞或杀掉当前运行的会话会导致主主之间数据不一致的问题发生。 MHA 提供快速切换和优雅的阻塞写入,这个切换过程只需要 0.5-2s 的时间,这段时间内数据是无法写入的。在很多情况下,0.5-2s 的阻塞写入是可以接受的。因此切换主服务器不需要计划分配维护时间窗口。

MHA在线切换的大概过程:
1.检测复制设置和确定当前主服务器
2.确定新的主服务器
3.阻塞写入到当前主服务器
4.等待所有从服务器赶上复制
5.授予写入到新的主服务器
6.重新设置从服务器

注意,在线切换的时候应用架构需要考虑以下两个问题:

1.自动识别master和slave的问题(master的机器可能会切换),如果采用了vip的方式,基本可以解决这个问题。

2.负载均衡的问题(可以定义大概的读写比例,每台机器可承担的负载比例,当有机器离开集群时,需要考虑这个问题)

为了保证数据完全一致性,在最快的时间内完成切换,MHA的在线切换必须满足以下条件才会切换成功,否则会切换失败。

1.所有slave的IO线程都在运行

2.所有slave的SQL线程都在运行

3.所有的show slave status的输出中Seconds_Behind_Master参数小于或者等于running_updates_limit秒,如果在切换过程中不指定running_updates_limit,那么默认情况下running_updates_limit为1秒。

4.在master端,通过show processlist输出,没有一个更新花费的时间大于running_updates_limit秒。

在线切换步骤如下:

先停止监控程序

[root@MHA-S2 app1]# masterha_stop --conf=/etc/mha/app1.cnf
Stopped app1 successfully.

修改master_ip_online_change脚本如下:

[root@MHA-S2 bin]# more master_ip_online_change
#!/usr/bin/env perl # Copyright (C) DeNA Co.,Ltd.
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc.,
# Franklin Street, Fifth Floor, Boston, MA - USA ## Note: This is a sample script and is not complete. Modify the script based on your environment. use strict;
use warnings FATAL => 'all'; use Getopt::Long;
use MHA::DBHelper;
use MHA::NodeUtil;
use Time::HiRes qw( sleep gettimeofday tv_interval );
use Data::Dumper; my $_tstart;
my $_running_interval = 0.1;
my (
$command, $orig_master_host, $orig_master_ip,
$orig_master_port, $orig_master_user,
$new_master_host, $new_master_ip, $new_master_port,
$new_master_user,
); my $vip = '10.180.2.168/19'; # Virtual IP
my $key = "";
my $ssh_start_vip = "/sbin/ifconfig eth1:$key $vip";
my $ssh_stop_vip = "/sbin/ifconfig eth1:$key down";
my $ssh_user = "root";
my $new_master_password='';
my $orig_master_password='';
GetOptions(
'command=s' => \$command,
#'ssh_user=s' => \$ssh_user,
'orig_master_host=s' => \$orig_master_host,
'orig_master_ip=s' => \$orig_master_ip,
'orig_master_port=i' => \$orig_master_port,
'orig_master_user=s' => \$orig_master_user,
#'orig_master_password=s' => \$orig_master_password,
'new_master_host=s' => \$new_master_host,
'new_master_ip=s' => \$new_master_ip,
'new_master_port=i' => \$new_master_port,
'new_master_user=s' => \$new_master_user,
#'new_master_password=s' => \$new_master_password,
); exit &main(); sub current_time_us {
my ( $sec, $microsec ) = gettimeofday();
my $curdate = localtime($sec);
return $curdate . " " . sprintf( "%06d", $microsec );
} sub sleep_until {
my $elapsed = tv_interval($_tstart);
if ( $_running_interval > $elapsed ) {
sleep( $_running_interval - $elapsed );
}
} sub get_threads_util {
my $dbh = shift;
my $my_connection_id = shift;
my $running_time_threshold = shift;
my $type = shift;
$running_time_threshold = unless ($running_time_threshold);
$type = unless ($type);
my @threads; my $sth = $dbh->prepare("SHOW PROCESSLIST");
$sth->execute(); while ( my $ref = $sth->fetchrow_hashref() ) {
my $id = $ref->{Id};
my $user = $ref->{User};
my $host = $ref->{Host};
my $command = $ref->{Command};
my $state = $ref->{State};
my $query_time = $ref->{Time};
my $info = $ref->{Info};
$info =~ s/^\s*(.*?)\s*$/$/ if defined($info);
next if ( $my_connection_id == $id );
next if ( defined($query_time) && $query_time < $running_time_threshold );
next if ( defined($command) && $command eq "Binlog Dump" );
next if ( defined($user) && $user eq "system user" );
next
if ( defined($command)
&& $command eq "Sleep"
&& defined($query_time)
&& $query_time >= ); if ( $type >= ) {
next if ( defined($command) && $command eq "Sleep" );
next if ( defined($command) && $command eq "Connect" );
} if ( $type >= ) {
next if ( defined($info) && $info =~ m/^select/i );
next if ( defined($info) && $info =~ m/^show/i );
} push @threads, $ref;
}
return @threads;
} sub main {
if ( $command eq "stop" ) {
## Gracefully killing connections on the current master
# . Set read_only= on the new master
# . DROP USER so that no app user can establish new connections
# . Set read_only= on the current master
# . Kill current queries
# * Any database access failure will result in script die.
my $exit_code = ;
eval {
## Setting read_only= on the new master (to avoid accident)
my $new_master_handler = new MHA::DBHelper(); # args: hostname, port, user, password, raise_error(die_on_error)_or_not
$new_master_handler->connect( $new_master_ip, $new_master_port,
$new_master_user, $new_master_password, );
print current_time_us() . " Set read_only on the new master.. ";
$new_master_handler->enable_read_only();
if ( $new_master_handler->is_read_only() ) {
print "ok.\n";
}
else {
die "Failed!\n";
}
$new_master_handler->disconnect(); # Connecting to the orig master, die if any database error happens
my $orig_master_handler = new MHA::DBHelper();
$orig_master_handler->connect( $orig_master_ip, $orig_master_port,
$orig_master_user, $orig_master_password, ); ## Drop application user so that nobody can connect. Disabling per-session binlog beforehand
#$orig_master_handler->disable_log_bin_local();
#print current_time_us() . " Drpping app user on the orig master..\n";
#FIXME_xxx_drop_app_user($orig_master_handler); ## Waiting for N * milliseconds so that current connections can exit
my $time_until_read_only = ;
$_tstart = [gettimeofday];
my @threads = get_threads_util( $orig_master_handler->{dbh},
$orig_master_handler->{connection_id} );
while ( $time_until_read_only > && $#threads >= ) {
if ( $time_until_read_only % == ) {
printf
"%s Waiting all running %d threads are disconnected.. (max %d milliseconds)\n",
current_time_us(), $#threads + , $time_until_read_only * ;
if ( $#threads < ) {
print Data::Dumper->new( [$_] )->Indent()->Terse()->Dump . "\n"
foreach (@threads);
}
}
sleep_until();
$_tstart = [gettimeofday];
$time_until_read_only--;
@threads = get_threads_util( $orig_master_handler->{dbh},
$orig_master_handler->{connection_id} );
} ## Setting read_only= on the current master so that nobody(except SUPER) can write
print current_time_us() . " Set read_only=1 on the orig master.. ";
$orig_master_handler->enable_read_only();
if ( $orig_master_handler->is_read_only() ) {
print "ok.\n";
}
else {
die "Failed!\n";
} ## Waiting for M * milliseconds so that current update queries can complete
my $time_until_kill_threads = ;
@threads = get_threads_util( $orig_master_handler->{dbh},
$orig_master_handler->{connection_id} );
while ( $time_until_kill_threads > && $#threads >= ) {
if ( $time_until_kill_threads % == ) {
printf
"%s Waiting all running %d queries are disconnected.. (max %d milliseconds)\n",
current_time_us(), $#threads + , $time_until_kill_threads * ;
if ( $#threads < ) {
print Data::Dumper->new( [$_] )->Indent()->Terse()->Dump . "\n"
foreach (@threads);
}
}
sleep_until();
$_tstart = [gettimeofday];
$time_until_kill_threads--;
@threads = get_threads_util( $orig_master_handler->{dbh},
$orig_master_handler->{connection_id} );
} print "Disabling the VIP on old master: $orig_master_host \n";
&stop_vip(); ## Terminating all threads
print current_time_us() . " Killing all application threads..\n";
$orig_master_handler->kill_threads(@threads) if ( $#threads >= );
print current_time_us() . " done.\n";
#$orig_master_handler->enable_log_bin_local();
$orig_master_handler->disconnect(); ## After finishing the script, MHA executes FLUSH TABLES WITH READ LOCK
$exit_code = ;
};
if ($@) {
warn "Got Error: $@\n";
exit $exit_code;
}
exit $exit_code;
}
elsif ( $command eq "start" ) {
## Activating master ip on the new master
# . Create app user with write privileges
# . Moving backup script if needed
# . Register new master's ip to the catalog database # We don't return error even though activating updatable accounts/ip failed so that we don't interrupt slaves' recovery.
# If exit code is or , MHA does not abort
my $exit_code = ;
eval {
my $new_master_handler = new MHA::DBHelper(); # args: hostname, port, user, password, raise_error_or_not
$new_master_handler->connect( $new_master_ip, $new_master_port,
$new_master_user, $new_master_password, ); ## Set read_only= on the new master
#$new_master_handler->disable_log_bin_local();
print current_time_us() . " Set read_only=0 on the new master.\n";
$new_master_handler->disable_read_only(); ## Creating an app user on the new master
#print current_time_us() . " Creating app user on the new master..\n";
#FIXME_xxx_create_app_user($new_master_handler);
#$new_master_handler->enable_log_bin_local();
$new_master_handler->disconnect(); ## Update master ip on the catalog database, etc
print "Enabling the VIP - $vip on the new master - $new_master_host \n";
&start_vip();
$exit_code = ;
};
if ($@) {
warn "Got Error: $@\n";
exit $exit_code;
}
exit $exit_code;
}
elsif ( $command eq "status" ) { # do nothing
exit ;
}
else {
&usage();
exit ;
}
} # A simple system call that enable the VIP on the new master
sub start_vip() {
`ssh $ssh_user\@$new_master_host \" $ssh_start_vip \"`;
}
# A simple system call that disable the VIP on the old_master
sub stop_vip() {
`ssh $ssh_user\@$orig_master_host \" $ssh_stop_vip \"`;
} sub usage {
print
"Usage: master_ip_online_change --command=start|stop|status --orig_master_host=host --orig_master_ip=ip --orig_master_port=port --new_master_host=host --new_master_ip=ip --new_master_port=port\n";
die;
}

执行切换

[root@MHA-S2 tmp]# masterha_master_switch --conf=/etc/mha/app1.cnf --master_state=alive --new_master_host=MHA-M1 --new_master_port= --orig_master_is_new_slave --running_updates_limit=

其中参数的意思:

--orig_master_is_new_slave 切换时加上此参数是将原 master 变为 slave 节点,如果不加此参数,原来的 master 将不启动

--running_updates_limit=10000,故障切换时,候选master 如果有延迟的话, mha 切换不能成功,加上此参数表示延迟在此时间范围内都可切换(单位为s),但是切换的时间长短是由recover 时relay 日志的大小决定

查看切换后各机器的状态:

S2:

root@localhost:mysql3306.sock [tt]>show slave status\G
*************************** . row ***************************
Slave_IO_State: Waiting for master to send event
Master_Host: MHA-M1
Master_User: repl
Master_Port:
Connect_Retry:
Master_Log_File: -binlog.
Read_Master_Log_Pos:
Relay_Log_File: MHA-S2-relay-bin.
Relay_Log_Pos:
Relay_Master_Log_File: -binlog.
Slave_IO_Running: Yes
Slave_SQL_Running: Yes

S1:

root@localhost:mysql3306.sock [tt]>show slave status\G
*************************** . row ***************************
Slave_IO_State: Waiting for master to send event
Master_Host: MHA-M1
Master_User: repl
Master_Port:
Connect_Retry:
Master_Log_File: -binlog.
Read_Master_Log_Pos:
Relay_Log_File: MHA-S1-relay-bin.
Relay_Log_Pos:
Relay_Master_Log_File: -binlog.
Slave_IO_Running: Yes
Slave_SQL_Running: Yes

M1:

root@localhost:mysql3306.sock [tt]>show slave status\G
Empty set (0.00 sec)

在线切换的日志:

[root@MHA-S2 tmp]# more sw.log
[root@MHA-S2 bin]# masterha_master_switch --conf=/etc/mha/app1.cnf --master_state=alive --new_master_host=MHA-M1 --new_master_port= --orig_master_is_new_slave --running_updates_limit=
Sat Aug :: - [info] MHA::MasterRotate version 0.57.
Sat Aug :: - [info] Starting online master switch..
Sat Aug :: - [info]
Sat Aug :: - [info] * Phase : Configuration Check Phase..
Sat Aug :: - [info]
Sat Aug :: - [warning] Global configuration file /etc/masterha_default.cnf not found. Skipping.
Sat Aug :: - [info] Reading application default configuration from /etc/mha/app1.cnf..
Sat Aug :: - [info] Reading server configuration from /etc/mha/app1.cnf..
Sat Aug :: - [info] GTID failover mode =
Sat Aug :: - [info] Current Alive Master: MHA-S1(10.180.2.164:)
Sat Aug :: - [info] Alive Slaves:
Sat Aug :: - [info] MHA-M1(10.180.2.163:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Sat Aug :: - [info] GTID ON
Sat Aug :: - [info] Replicating from MHA-S1(10.180.2.164:)
Sat Aug :: - [info] MHA-S2(10.180.2.165:) Version=5.7.-log (oldest major version between slaves) log-bin:enabled
Sat Aug :: - [info] GTID ON
Sat Aug :: - [info] Replicating from MHA-S1(10.180.2.164:) It is better to execute FLUSH NO_WRITE_TO_BINLOG TABLES on the master before switching. Is it ok to execute on MHA-S1(10.180.2.164:)? (YES/no): yes
Sat Aug :: - [info] Executing FLUSH NO_WRITE_TO_BINLOG TABLES. This may take long time..
Sat Aug :: - [info] ok.
Sat Aug :: - [info] Checking MHA is not monitoring or doing failover..
Sat Aug :: - [info] Checking replication health on MHA-M1..
Sat Aug :: - [info] ok.
Sat Aug :: - [info] Checking replication health on MHA-S2..
Sat Aug :: - [info] ok.
Sat Aug :: - [info] MHA-M1 can be new master.
Sat Aug :: - [info]
From:
MHA-S1(10.180.2.164:) (current master)
+--MHA-M1(10.180.2.163:)
+--MHA-S2(10.180.2.165:) To:
MHA-M1(10.180.2.163:) (new master)
+--MHA-S2(10.180.2.165:)
+--MHA-S1(10.180.2.164:) Starting master switch from MHA-S1(10.180.2.164:) to MHA-M1(10.180.2.163:)? (yes/NO): yes
Sat Aug :: - [info] Checking whether MHA-M1(10.180.2.163:) is ok for the new master..
Sat Aug :: - [info] ok.
Sat Aug :: - [info] MHA-S1(10.180.2.164:): SHOW SLAVE STATUS returned empty result. To check replication filtering rules, temporarily executing CHANGE MASTER to a dummy host.
Sat Aug :: - [info] MHA-S1(10.180.2.164:): Resetting slave pointing to the dummy host.
Sat Aug :: - [info] ** Phase : Configuration Check Phase completed.
Sat Aug :: - [info]
Sat Aug :: - [info] * Phase : Rejecting updates Phase..
Sat Aug :: - [info]
Sat Aug :: - [info] Executing master ip online change script to disable write on the current master:
Sat Aug :: - [info] /usr/local/bin/master_ip_online_change --command=stop --orig_master_host=MHA-S1 --orig_master_ip=10.180.2.164 --orig_master_port= --orig_master_user='root' --new_master_
host=MHA-M1 --new_master_ip=10.180.2.163 --new_master_port= --new_master_user='root' --orig_master_ssh_user=root --new_master_ssh_user=root --orig_master_is_new_slave --orig_master_password=xxx --new_mas
ter_password=xxx
Unknown option: orig_master_ssh_user
Unknown option: new_master_ssh_user
Unknown option: orig_master_is_new_slave
Unknown option: orig_master_password
Unknown option: new_master_password
Sat Aug :: Set read_only on the new master.. ok.
Sat Aug :: Waiting all running threads are disconnected.. (max milliseconds)
{'Time' => '','Command' => 'Binlog Dump GTID','db' => undef,'Id' => '','Info' => undef,'User' => 'repl','State' => 'Master has sent all binlog to slave; waiting for more updates','Host' => 'MHA-S2:4697
'}
{'Time' => '','Command' => 'Binlog Dump GTID','db' => undef,'Id' => '','Info' => undef,'User' => 'repl','State' => 'Master has sent all binlog to slave; waiting for more updates','Host' => 'MHA-M1:51506'
}
Sat Aug :: Waiting all running threads are disconnected.. (max milliseconds)
{'Time' => '','Command' => 'Binlog Dump GTID','db' => undef,'Id' => '','Info' => undef,'User' => 'repl','State' => 'Master has sent all binlog to slave; waiting for more updates','Host' => 'MHA-S2:4697
'}
{'Time' => '','Command' => 'Binlog Dump GTID','db' => undef,'Id' => '','Info' => undef,'User' => 'repl','State' => 'Master has sent all binlog to slave; waiting for more updates','Host' => 'MHA-M1:51506'
}
Sat Aug :: Waiting all running threads are disconnected.. (max milliseconds)
{'Time' => '','Command' => 'Binlog Dump GTID','db' => undef,'Id' => '','Info' => undef,'User' => 'repl','State' => 'Master has sent all binlog to slave; waiting for more updates','Host' => 'MHA-S2:4697
'}
{'Time' => '','Command' => 'Binlog Dump GTID','db' => undef,'Id' => '','Info' => undef,'User' => 'repl','State' => 'Master has sent all binlog to slave; waiting for more updates','Host' => 'MHA-M1:51506'
}
Sat Aug :: Set read_only= on the orig master.. ok.
Sat Aug :: Waiting all running queries are disconnected.. (max milliseconds)
{'Time' => '','Command' => 'Binlog Dump GTID','db' => undef,'Id' => '','Info' => undef,'User' => 'repl','State' => 'Master has sent all binlog to slave; waiting for more updates','Host' => 'MHA-S2:4697
'}
{'Time' => '','Command' => 'Binlog Dump GTID','db' => undef,'Id' => '','Info' => undef,'User' => 'repl','State' => 'Master has sent all binlog to slave; waiting for more updates','Host' => 'MHA-M1:51506'
}
Disabling the VIP on old master: MHA-S1
Sat Aug :: Killing all application threads..
Sat Aug :: done.
Sat Aug :: - [info] ok.
Sat Aug :: - [info] Locking all tables on the orig master to reject updates from everybody (including root):
Sat Aug :: - [info] Executing FLUSH TABLES WITH READ LOCK..
Sat Aug :: - [info] ok.
Sat Aug :: - [info] Orig master binlog:pos is -binlog.:.
Sat Aug :: - [info] Waiting to execute all relay logs on MHA-M1(10.180.2.163:)..
Sat Aug :: - [info] master_pos_wait(-binlog.:) completed on MHA-M1(10.180.2.163:). Executed events.
Sat Aug :: - [info] done.
Sat Aug :: - [info] Getting new master's binlog name and position..
Sat Aug :: - [info] -binlog.:
Sat Aug :: - [info] All other slaves should start replication from here. Statement should be: CHANGE MASTER TO MASTER_HOST='MHA-M1 or 10.180.2.163', MASTER_PORT=, MASTER_AUTO_POSITION=, MAS
TER_USER='repl', MASTER_PASSWORD='xxx';
Sat Aug :: - [info] Executing master ip online change script to allow write on the new master:
Sat Aug :: - [info] /usr/local/bin/master_ip_online_change --command=start --orig_master_host=MHA-S1 --orig_master_ip=10.180.2.164 --orig_master_port= --orig_master_user='root' --new_master
_host=MHA-M1 --new_master_ip=10.180.2.163 --new_master_port= --new_master_user='root' --orig_master_ssh_user=root --new_master_ssh_user=root --orig_master_is_new_slave --orig_master_password=xxx --new_ma
ster_password=xxx
Unknown option: orig_master_ssh_user
Unknown option: new_master_ssh_user
Unknown option: orig_master_is_new_slave
Unknown option: orig_master_password
Unknown option: new_master_password
Sat Aug :: Set read_only= on the new master.
Enabling the VIP - 10.180.2.168/ on the new master - MHA-M1
Sat Aug :: - [info] ok.
Sat Aug :: - [info]
Sat Aug :: - [info] * Switching slaves in parallel..
Sat Aug :: - [info]
Sat Aug :: - [info] -- Slave switch on host MHA-S2(10.180.2.165:) started, pid:
Sat Aug :: - [info]
Sat Aug :: - [info] Log messages from MHA-S2 ...
Sat Aug :: - [info]
Sat Aug :: - [info] Waiting to execute all relay logs on MHA-S2(10.180.2.165:)..
Sat Aug :: - [info] master_pos_wait(-binlog.:) completed on MHA-S2(10.180.2.165:). Executed events.
Sat Aug :: - [info] done.
Sat Aug :: - [info] Resetting slave MHA-S2(10.180.2.165:) and starting replication from the new master MHA-M1(10.180.2.163:)..
Sat Aug :: - [info] Executed CHANGE MASTER.
Sat Aug :: - [info] Slave started.
Sat Aug :: - [info] End of log messages from MHA-S2 ...
Sat Aug :: - [info]
Sat Aug :: - [info] -- Slave switch on host MHA-S2(10.180.2.165:) succeeded.
Sat Aug :: - [info] Unlocking all tables on the orig master:
Sat Aug :: - [info] Executing UNLOCK TABLES..
Sat Aug :: - [info] ok.
Sat Aug :: - [info] Starting orig master as a new slave..
Sat Aug :: - [info] Resetting slave MHA-S1(10.180.2.164:) and starting replication from the new master MHA-M1(10.180.2.163:)..
Sat Aug :: - [info] Executed CHANGE MASTER.
Sat Aug :: - [info] Slave started.
Sat Aug :: - [info] All new slave servers switched successfully.
Sat Aug :: - [info]
Sat Aug :: - [info] * Phase : New master cleanup phase..
Sat Aug :: - [info]
Sat Aug :: - [info] MHA-M1: Resetting slave info succeeded.
Sat Aug :: - [info] Switching master to MHA-M1(10.180.2.163:) completed successfully.

最后附上参考资料:

http://www.cnblogs.com/gomysql/p/3675429.html

http://www.178linux.com/61111