Contents

MySQL半同步

MySQL 5.7半同步复制技术

一、复制架构衍生史

在谈这个特性之前,我们先来看看MySQL的复制架构衍生史。

在2000年,MySQL 3.23.15版本引入了Replication。Replication作为一种准实时同步方式,得到广泛应用。这个时候的Replicaton的实现涉及到两个线程,一个在Master,一个在Slave。Slave的I/O和SQL功能是作为一个线程,从Master获取到event后直接apply,没有relay log。这种方式使得读取event的速度会被Slave replay速度拖慢,当主备存在较大延迟时候,会导致大量binary log没有备份到Slave端。

在2002年,MySQL 4.0.2版本将Slave端event读取和执行独立成两个线程(IO线程和SQL线程),同时引入了relay log。IO线程读取event后写入relay log,SQL线程从relay log中读取event然后执行。这样即使SQL线程执行慢,Master的binary log也会尽可能的同步到Slave。当Master宕机,切换到Slave,不会出现大量数据丢失。

在2010年MySQL 5.5版本之前,一直采用的是这种异步复制的方式。主库的事务执行不会管备库的同步进度,如果备库落后,主库不幸crash,那么就会导致数据丢失。于是在MySQL在5.5中就顺其自然地引入了半同步复制,主库在应答客户端提交的事务前需要保证至少一个从库接收并写到relay log中。那么半同步复制是否可以做到不丢失数据呢?下面分析。

在2016年,MySQL在5.7.17中引入了一个全新的技术,称之为InnoDB Group Replication。目前官方MySQL 5.7.17基于Group replication的全同步技术已经问世,全同步技术带来了更多的数据一致性保障。相信是未来同步技术一个重要方向,值得期待。MySQL 5.7 Group Replication

根据上面提到的这几种复制协议,分别对应MySQL几种复制类型,分别是异步、半同步、全同步。

[https://cdn.forsre.cn/mysql001.png

  • 对于异步复制,主库将事务Binlog事件写入到Binlog文件中,此时主库只会通知一下Dump线程发送这些新的Binlog,然后主库就会继续处理提交操作,而此时不会保证这些Binlog传到任何一个从库节点上。
  • 对于全同步复制,当主库提交事务之后,所有的从库节点必须收到,APPLY并且提交这些事务,然后主库线程才能继续做后续操作。这里面有一个很明显的缺点就是,主库完成一个事务的时间被拉长,性能降低。
  • 对于半同步复制,是介于全同步复制和异步复制之间的一种,主库只需要等待至少一个从库节点收到并且Flush Binlog到Relay Log文件即可,主库不需要等待所有从库给主库反馈。同时,这里只是一个收到的反馈,而不是已经完全执行并且提交的反馈,这样就节省了很多时间。

二、半同步复制技术

我们今天谈论第二种架构。我们知道,普通的replication,即MySQL的异步复制,依靠MySQL二进制日志也即binary log进行数据复制。比如两台机器,一台主机(master),另外一台是从机(slave)。

1)正常的复制为:事务一(t1)写入binlog buffer;dumper线程通知slave有新的事务t1;binlog buffer进行checkpoint;slave的io线程接收到t1并写入到自己的的relay log;slave的sql线程写入到本地数据库。 这时,master和slave都能看到这条新的事务,即使master挂了,slave可以提升为新的master。

2)异常的复制为:事务一(t1)写入binlog buffer;dumper线程通知slave有新的事务t1;binlog buffer进行checkpoint;slave因为网络不稳定,一直没有收到t1;master挂掉,slave提升为新的master,t1丢失。

3)很大的问题是:主机和从机事务更新的不同步,就算是没有网络或者其他系统的异常,当业务并发上来时,slave因为要顺序执行master批量事务,导致很大的延迟。

为了弥补以上几种场景的不足,MySQL从5.5开始推出了半同步复制。相比异步复制,半同步复制提高了数据完整性,因为很明确知道,在一个事务提交成功之后,这个事务就至少会存在于两个地方。即在master的dumper线程通知slave后,增加了一个ack(消息确认),即是否成功收到t1的标志码,也就是dumper线程除了发送t1到slave,还承担了接收slave的ack工作。如果出现异常,没有收到ack,那么将自动降级为普通的复制,直到异常修复后又会自动变为半同步复制。

半同步复制具体特性:

  • 从库会在连接到主库时告诉主库,它是不是配置了半同步。
  • 如果半同步复制在主库端是开启了的,并且至少有一个半同步复制的从库节点,那么此时主库的事务线程在提交时会被阻塞并等待,结果有两种可能,要么至少一个从库节点通知它已经收到了所有这个事务的Binlog事件,要么一直等待直到超过配置的某一个时间点为止,而此时,半同步复制将自动关闭,转换为异步复制。
  • 从库节点只有在接收到某一个事务的所有Binlog,将其写入并Flush到Relay Log文件之后,才会通知对应主库上面的等待线程。
  • 如果在等待过程中,等待时间已经超过了配置的超时时间,没有任何一个从节点通知当前事务,那么此时主库会自动转换为异步复制,当至少一个半同步从节点赶上来时,主库便会自动转换为半同步方式的复制。
  • 半同步复制必须是在主库和从库两端都开启时才行,如果在主库上没打开,或者在主库上开启了而在从库上没有开启,主库都会使用异步方式复制。

半同步复制潜在问题:

先看一下半同步复制原理图,如下:

[https://cdn.forsre.cn/mysql002.png

master将每个事务写入binlog(sync_binlog=1),传递到slave刷新到磁盘(sync_relay=1),同时主库提交事务(commit)。master等待slave反馈收到relay log,只有收到ACK后master才将commit OK结果反馈给客户端。

在MySQL 5.5~5.6使用after_commit的模式下,客户端事务在存储引擎层提交后,在得到从库确认的过程中,主库宕机了。此时,即主库在等待Slave ACK的时候,虽然没有返回当前客户端,但事务已经提交,其他客户端会读取到已提交事务。如果Slave端还没有读到该事务的events,同时主库发生了crash,然后切换到备库。那么之前读到的事务就不见了,出现了幻读。如下图所示,图片引自Loss-less Semi-Synchronous Replication on MySQL 5.7.2

[https://cdn.forsre.cn/mysql003.png

如果主库永远启动不了,那么实际上在主库已经成功提交的事务,在从库上是找不到的,也就是数据丢失了,这是MySQL不愿意看到的。所以在MySQL 5.7版本中增加了after_sync(无损复制)参数,并将其设置为默认半同步方式,解决了数据丢失的问题。

三、MySQL 5.6半同步复制配置

具体完整配置可参考:MySQL基于日志点做主从复制(二)

Master配置

1)安装半同步模块并启动(此模块就在/usr/local/mysql/lib/plugin/semisync_master.so)

1
mysql> install plugin rpl_semi_sync_master soname ``'semisync_master.so'``;

[https://cdn.forsre.cn/mysql004.png

1
mysql> ``set` `global rpl_semi_sync_master_enabled = 1;``mysql> ``set` `global rpl_semi_sync_master_timeout = 2000;

安装后启动和定制主从连接错误的超时时间默认是10s可改为2s,一旦有一次超时自动降级为异步。(以上内容要想永久有效需要写到配置文件中)

1
[root@localhost ~]# cat /etc/my.cnf``[mysqld]``rpl_semi_sync_master_enabled = 1;``rpl_semi_sync_master_timeout = 2000;

Slave配置

1)安装半同步模块并启动

1
mysql> install plugin rpl_semi_sync_slave soname ``'semisync_slave.so'``;``mysql> ``set` `global rpl_semi_sync_slave_enabled = 1;``mysql> show global variables like ``'%semi%'``;``+---------------------------------+-------+``| Variable_name          | Value |``+---------------------------------+-------+``| rpl_semi_sync_slave_enabled   | ON  |``| rpl_semi_sync_slave_trace_level | 32  |``+---------------------------------+-------+``2 rows ``in` `set` `(0.00 sec)

  

2)从节点需要重新连接主服务器半同步才会生效

1
mysql> stop slave io_thread;``mysql> start slave io_thread;

PS:如果想卸载异步模块就使用uninstall即可。

Master上查看是否启用了半同步

[https://cdn.forsre.cn/mysql005.png

现在半同步已经正常工作了,主要看Rpl_semi_sync_master_clients是否不为0,Rpl_semi_sync_master_status是否为ON。如果Rpl_semi_sync_master_status为OFF,说明出现了网络延迟或Slave IO线程延迟。

那么可以验证一下半同步超时,是否会自动降为异步工作。可以在Slave上停掉半同步协议,然后在Master上创建数据库看一下能不能复制到Slave上。

Slave

1
# 关闭半同步;``mysql> ``set` `global rpl_semi_sync_slave_enabled = 0 ;``mysql> stop slave io_thread;``mysql> start slave io_thread;

Master

1
mysql> create database dbtest;``Query OK, 1 row affected (2.01 sec)<br>``mysql> create database dbtest01;``Query OK, 1 row affected (0.01 sec)

  

创建第一个数据库花了2.01秒,而我们前面设置的超时时间是2秒,而创建第二个数据库花了0.01秒,由此得出结论是超时转换为异步传送。可以在Master上查看半同步相关的参数值Rpl_semi_sync_master_clients和Rpl_semi_sync_master_status是否正常。

1
mysql> show global status like ``'%semi%'``; ``+--------------------------------------------+-----------+``| Variable_name               | Value   |``+--------------------------------------------+-----------+``| Rpl_semi_sync_master_clients        | 0     |``| Rpl_semi_sync_master_net_avg_wait_time   | 0     |``| Rpl_semi_sync_master_net_wait_time     | 0     |``| Rpl_semi_sync_master_net_waits       | 37490   |``| Rpl_semi_sync_master_no_times       | 3     |``| Rpl_semi_sync_master_no_tx         | 197542  |``| Rpl_semi_sync_master_status        | OFF    |``| Rpl_semi_sync_master_timefunc_failures   | 0     |``| Rpl_semi_sync_master_tx_avg_wait_time   | 51351   |``| Rpl_semi_sync_master_tx_wait_time     | 362437445 |``| Rpl_semi_sync_master_tx_waits       | 7058   |``| Rpl_semi_sync_master_wait_pos_backtraverse | 0     |``| Rpl_semi_sync_master_wait_sessions     | 0     |``| Rpl_semi_sync_master_yes_tx        | 7472   |``+--------------------------------------------+-----------+``14 rows ``in` `set` `(0.00 sec)

  

可以看到都自动关闭了,需要注意一点的是,当Slave开启半同步后,或者当主从之间网络延迟恢复正常的时候,半同步复制会自动从异步复制又转为半同步复制,还是相当智能的。

另外个人在实际使用中还碰到一种情况从库IO线程有延迟时,主库会自动把半同步复制降为异步复制;当从库IO延迟没有时,主库又会把异步复制升级为半同步复制。可以进行压测模拟,但是此时查看Master的状态跟上面直接关闭Slave半同步有些不同,会发现Rpl_semi_sync_master_clients仍然等于1,而Rpl_semi_sync_master_status等于OFF。

随着MySQL 5.7版本的发布,半同步复制技术升级为全新的Loss-less Semi-Synchronous Replication架构,其成熟度、数据一致性与执行效率得到显著的提升。

四、MySQL 5.7半同步复制的改进

现在我们已经知道,在半同步环境下,主库是在事务提交之后等待Slave ACK,所以才会有数据不一致问题。所以这个Slave ACK在什么时间去等待,也是一个很关键的问题了。因此MySQL针对半同步复制的问题,在5.7.2引入了Loss-less Semi-Synchronous,在调用binlog sync之后,engine层commit之前等待Slave ACK。这样只有在确认Slave收到事务events后,事务才会提交。在commit之前等待Slave ACK,同时可以堆积事务,利于group commit,有利于提升性能。

MySQL 5.7安装半同步模块,命令如下:

1
mysql> install plugin rpl_semi_sync_master soname ``'semisync_master.so'``;``Query OK, 0 rows affected (0.00 sec)

  

看一下相关状态信息

1
mysql> show global variables like ``'%semi%'``;``+-------------------------------------------+------------+``| Variable_name               | Value   |``+-------------------------------------------+------------+``| rpl_semi_sync_master_enabled       | OFF    |``| rpl_semi_sync_master_timeout       | 10000   |``| rpl_semi_sync_master_trace_level     | 32     |``| rpl_semi_sync_master_wait_for_slave_count | 1     |``| rpl_semi_sync_master_wait_no_slave    | ON     |``| rpl_semi_sync_master_wait_point      | AFTER_SYNC |``+-------------------------------------------+------------+``6 rows ``in` `set` `(0.00 sec)

  

  • 支持无损复制(Loss-less Semi-Synchronous)

在Loss-less Semi-Synchronous模式下,master在调用binlog sync之后,engine层commit之前等待Slave ACK(需要收到至少一个Slave节点回复的ACK后)。这样只有在确认Slave收到事务events后,master事务才会提交,然后把结果返回给客户端。此时此事务才对其他事务可见。在这种模式下解决了after_commit模式带来的幻读和数据丢失问题,因为主库没有提交事务。但也会有个问题,假设主库在存储引擎提交之前挂了,那么很明显这个事务是不成功的,但由于对应的Binlog已经做了Sync操作,从库已经收到了这些Binlog,并且执行成功,相当于在从库上多了数据,也算是有问题的,但多了数据,问题一般不算严重。这个问题可以这样理解,作为MySQL,在没办法解决分布式数据一致性问题的情况下,它能保证的是不丢数据,多了数据总比丢数据要好。

无损复制其实就是对semi sync增加了rpl_semi_sync_master_wait_point参数,来控制半同步模式下主库在返回给会话事务成功之前提交事务的方式。rpl_semi_sync_master_wait_point该参数有两个值:AFTER_COMMIT和AFTER_SYNC

第一个值:AFTER_COMMIT(5.6默认值)

master将每个事务写入binlog(sync_binlog=1),传递到slave刷新到磁盘(sync_relay=1),同时主库提交事务。master等待slave反馈收到relay log,只有收到ACK后master才将commit OK结果反馈给客户端。

[https://cdn.forsre.cn/mysql006.png

第二个值:AFTER_SYNC(5.7默认值,但5.6中无此模式)

master将每个事务写入binlog , 传递到slave刷新到磁盘(relay log)。master等待slave反馈接收到relay log的ack之后,再提交事务并且返回commit OK结果给客户端。 即使主库crash,所有在主库上已经提交的事务都能保证已经同步到slave的relay log中。

[https://cdn.forsre.cn/mysql007.png

半同步复制与无损复制的对比

1.1 ACK的时间点不同

  • 半同步复制在InnoDB层的Commit Log后等待ACK,主从切换会有数据丢失风险。
  • 无损复制在MySQL Server层的Write binlog后等待ACK,主从切换会有数据变多风险。

1.2 主从数据一致性

  • 半同步复制意味着在Master节点上,这个刚刚提交的事物对数据库的修改,对其他事物是可见的。因此,如果在等待Slave ACK的时候crash了,那么会对其他事务出现幻读,数据丢失。
  • 无损复制在write binlog完成后,就传输binlog,但还没有去写commit log,意味着当前这个事物对数据库的修改,其他事物也是不可见的。因此,不会出现幻读,数据丢失风险。

因此5.7引入了无损复制(after_sync)模式,带来的主要收益是解决after_commit导致的master crash后数据丢失问题,因此在引入after_sync模式后,所有提交的数据已经都被复制,故障切换时数据一致性将得到提升。

  • 性能提升,支持发送binlog和接受ack的异步化

旧版本的semi sync受限于dump thread ,原因是dump thread承担了两份不同且又十分频繁的任务:传送binlog给slave ,还需要等待slave反馈信息,而且这两个任务是串行的,dump thread必须等待slave返回之后才会传送下一个events事务。dump thread已然成为整个半同步提高性能的瓶颈。在高并发业务场景下,这样的机制会影响数据库整体的TPS 。

[https://cdn.forsre.cn/mysql008.png

为了解决上述问题,在5.7版本的semi sync框架中,独立出一个Ack Receiver线程 ,专门用于接收slave返回的ack请求,这将之前dump线程的发送和接受工作分为了两个线程来处理。这样master上有两个线程独立工作,可以同时发送binlog到slave,和接收slave的ack信息。因此半同步复制得到了极大的性能提升。这也是MySQL 5.7发布时号称的Faster semi-sync replication。

[https://cdn.forsre.cn/mysql009.png

但是在MySQL 5.7.17之前,这个Ack Receiver线程采用了select机制来监听slave返回的结果,然而select机制监控的文件句柄只能是0-1024,当超过1024时,用户在MySQL的错误日志中或许会收到类似如下的报错,更有甚者会导致MySQL发生宕机。

semi-sync master failed on net_flush() before waiting for slave reply.

MySQL 5.7.17版本开始,官方修复了这个bug,开始使用poll机制来替换原来的select机制,从而可以避免上面的问题。其实poll调用本质上和select没有区别,只是在I/O句柄数理论上没有上限了,原因是它是基于链表来存储的。但是同样有缺点:比如大量的fd的数组被整体复制于用户态和内核地址空间之间,而不管这样的复制是不是有意义。poll还有一个特点是“水平触发”,如果报告了fd后,没有被处理,那么下次poll时会再次报告该fd。

其实在高性能软件中都是用另外一种调用机制,名为epoll,高性能的代表,比如Nginx,haproxy等都是使用epoll。可能poll的复杂性比epoll低,另外对于ack receiver线程来说可能poll足矣。

  • 性能提升,控制主库接收slave写事务成功反馈数量

MySQL 5.7新增了rpl_semi_sync_master_wait_slave_count参数,可以用来控制主库接受多少个slave写事务成功反馈,给高可用架构切换提供了灵活性。如图所示,当count值为2时,master需等待两个slave的ack。

[https://cdn.forsre.cn/mysql010.png

  • 性能提升, Binlog互斥锁改进

旧版本半同步复制在主提交binlog的写会话和dump thread读binlog的操作都会对binlog添加互斥锁,导致binlog文件的读写是串行化的,存在并发度的问题。

[https://cdn.forsre.cn/mysql011.png

MySQL 5.7对binlog lock进行了以下两方面优化:

\1. 移除了dump thread对binlog的互斥锁。

\2. 加入了安全边际保证binlog的读安全。

[https://cdn.forsre.cn/mysql012.png

可以看到从replication功能引入后,官方MySQL一直在不停的完善,前进。同时我们可以发现当前原生的MySQL主备复制实现实际上很难在满足数据一致性的前提下做到高可用、高性能。

五、参数sync_binlog/sync_relay与半同步复制

sync_binlog的配置

其实无损复制流程中也会存在着会导致主备数据不一致的情况,使主备同步失败的情形。见下面sync_binlog配置的分析。

源码剖析

1
sql/binlog.cc ordered_commit``9002  update_binlog_end_pos_after_sync= (get_sync_period() == 1);``    ``...``     ``//sync_period(sync_binlog)1时,在sync之后update binlog end pos``9021   ``if` `(!update_binlog_end_pos_after_sync)``      ``//更新binlog end position,dump线程会发送更新后的events``9022    update_binlog_end_pos();``    ``...``     ``//``9057   std::pair<``bool``, ``bool``> result= sync_binlog_file(``false``);``    ``...``     ``//``9061   ``if` `(update_binlog_end_pos_after_sync)``9062  {``    ``...``9068    update_binlog_end_pos(tmp_thd->get_trans_pos());``9069  }`` ` ` ` `sql/binlog.cc sync_binlog_file``8618 std::pair<``bool``, ``bool``>``8619 MYSQL_BIN_LOG::sync_binlog_file(``bool` `force)``8620 {``8621  ``bool` `synced= ``false``;``8622  unsigned ``int` `sync_period= get_sync_period(); ``//sync_binlog``    ``//sync_period0不做sync操作,其他值为达到sync调用次数后sync``8623  ``if` `(force || (sync_period && ++sync_counter >= sync_period))``8624  {

  

配置分析

当sync_binlog为0的时候,binlog sync磁盘由操作系统负责。当不为0的时候,其数值为定期sync磁盘的binlog commit group数。通过源码我们知道,sync_binlog值不等于1的时候事务在FLUSH阶段就传输binlog到从库了,而值为1时,binlog同步操作是在SYNC阶段后。当sync_binlog值大于1的时候,sync binlog操作可能并没有使binlog落盘。如果没有落盘,事务在提交前,Master掉电,然后恢复,那么这个时候该事务被回滚。但是Slave上可能已经收到了该事务的events并且执行,这个时候就会出现Slave事务比Master多的情况,主备同步会失败。所以如果要保持主备一致,需要设置sync_binlog为1。

WAIT_AFTER_SYNC和WAIT_AFTER_COMMIT两图中Send Events的位置,也可能导致主备数据不一致,出现同步失败的情形。实际在rpl_semi_sync_master_wait_point分析的图中是sync binlog大于1的情况。根据上面源码,流程如下图所示。Master依次执行flush binlog, update binlog position, sync binlog。如果Master在update binlog position后,sync binlog前掉电,Master再次启动后原事务就会被回滚。但可能出现Slave获取到Events,这也会导致Slave数据比Master多,主备同步失败。

[D:/md/pic/mysql013.png

由于上面的原因,sync_binlog设置为1的时候,MySQL会update binlog end pos after sync。流程如下图所示。这时候,对于每一个事务都需要sync binlog,同时sync binlog和网络发送events会是一个串行的过程,性能下降明显。

[D:/md/pic/mysql014.png

sync_relay_log的配置

源码剖析

1
sql/rpl_slave.cc handle_slave_io`` ` `5764    ``if` `(queue_event(mi, event_buf, event_len))``      ``...``5771    ``if` `(RUN_HOOK(binlog_relay_io, after_queue_event,``5772          (thd, mi, event_buf, event_len, synced)))`` ` `after_queue_event``->plugin/semisync/semisync_slave_plugin.cc repl_semi_slave_queue_event``->plugin/semisync/semisync_slave.cc ReplSemiSyncSlave::slaveReply`` ` `queue_event``->sql/binlog.cc MYSQL_BIN_LOG::append_buffer(``const` `char``* buf, ``uint` `len, Master_info *mi)``->sql/binlog.cc after_append_to_relay_log(mi);``->sql/binlog.cc flush_and_sync(0)``->sql/binlog.cc sync_binlog_file(force)

  

配置分析

在Slave的IO线程中get_sync_period获得的是sync_relay_log的值,与sync_binlog对sync控制一样。当sync_relay_log不是1的时候,semisync返回给Master的position可能没有sync到磁盘。在gtid_mode下,在保证前面两个配置正确的情况下,sync_relay_log不是1的时候,仅发生Master或Slave的一次Crash并不会发生数据丢失或者主备同步失败情况。如果发生Slave没有sync relay log,Master端事务提交,客户端观察到事务提交,然后Slave端Crash。这样Slave端就会丢失掉已经回复Master ACK的事务events。

[D:/md/pic/mysql015.png

但当Slave再次启动,如果没有来得及从Master端同步丢失的事务Events,Master就Crash。这个时候,用户访问Slave就会发现数据丢失。

[D:/md/pic/mysql016.png

通过上面这个Case,MySQL semisync如果要保证任意时刻发生一台机器宕机都不丢失数据,需要同时设置sync_relay_log为1。对relay log的sync操作是在queue_event中,对每个event都要sync,所以sync_relay_log设置为1的时候,事务响应时间会受到影响,对于涉及数据比较多的事务延迟会增加很多。

MySQL三节点

在一主一从的主备semisync的数据一致性分析中放弃了高可用,当主备之间网络抖动或者一台宕机的情况下停止提供服务。要做到高可用,很自然我们可以想到一主两从,这样解决某一网络抖动或一台宕机时候的可用性问题。但是,前文叙述要保证数据一致性配置要求依然存在,即正常情况下的性能不会有改善。同时需要解决Master宕机时候,如何选取新主机的问题,如何避免多主的情形。

[D:/md/pic/mysql017.png

选取新主机时一定要读取两个从机,看哪一个从机有最新的日志,否则可能导致数据丢失。这样的三节点方案就类似分布式Quorum机制,写的时候需要保证写成功三节点中的法定集合,确定新主的时候需要读取法定集合。利用分布式一致性协议Paxos/Raft可以解决数据一致性问题,选主问题和多主问题,因此近些年,国内数据库团队大多实现了基于Paxos/Raft的三节点方案。近来MySQL官方也以插件形式引入了支持多主集群的Group Replication方案。