DBA或开发人员,有时会误删或者误更新数据,若是是线上环境而且影响较大,就须要能快速回滚。传统恢复方法是利用备份重搭实例,再应用去除错误sql后的binlog来恢复数据。此法费时费力,甚至须要停机维护,并不适合快速回滚。也有团队利用LVM快照来缩短恢复时间,但快照的缺点是会影响mysql的性能。php
MySQL闪回(flashback)利用binlog直接进行回滚,能快速恢复且不用停机。本文将介绍闪回原理,给出笔者的实战经验,并对现存的闪回工具做比较。html
某天,小明因种种缘由,误删了大批线上用户表的数据。他急忙找到公司DBA请求帮助,“客服电话已被打爆,大量用户投诉没法登录,领导很是恼火。请问多久能恢复数据?”DBA一脸懵逼,沉默十秒后,伸出一根手指。“你的意思是一分钟就能恢复?太好了。”小明终于有些放松,露出了一丝笑容。“不,咱们中有我的将会离开公司。”DBA沉痛的说道。python
勿让悲剧发生,尽早将此文转给公司DBA。mysql
binlog概述git
MySQL binlog以event的形式,记录了MySQL server从启用binlog以来全部的变动信息,可以帮助重现这之间的全部变化。MySQL引入binlog主要有两个目的:一是为了主从复制;二是某些备份还原操做后须要从新应用binlog。github
有三种可选的binlog格式,各有优缺点:sql
利用binlog闪回,须要将binlog格式设置为row。row模式下,一条使用innodb的insert会产生以下格式的binlog:shell
# at 1129 #161225 23:15:38 server id 3773306082 end_log_pos 1197 Query thread_id=1903021 exec_time=0 error_code=0 SET TIMESTAMP=1482678938/*!*/; BEGIN /*!*/; # at 1197 #161225 23:15:38 server id 3773306082 end_log_pos 1245 Table_map: `test`.`user` mapped to number 290 # at 1245 #161225 23:15:38 server id 3773306082 end_log_pos 1352 Write_rows: table id 290 flags: STMT_END_F BINLOG ' muJfWBPiFOjgMAAAAN0EAAAAACIBAAAAAAEABHRlc3QABHVzZXIAAwMPEQMeAAAC muJfWB7iFOjgawAAAEgFAAAAACIBAAAAAAEAAgAD//gBAAAABuWwj+i1tVhK1hH4AgAAAAblsI/p krFYStYg+AMAAAAG5bCP5a2ZWE/onPgEAAAABuWwj+adjlhNeAD4BQAAAAJ0dFhRYJM= '/*!*/; # at 1352 #161225 23:15:38 server id 3773306082 end_log_pos 1379 Xid = 5327954 COMMIT/*!*/;
闪回原理安全
既然binlog以event形式记录了全部的变动信息,那么咱们把须要回滚的event,从后往前回滚回去便可。bash
对于单个event的回滚,咱们以表test.user来演示原理
mysql> show create table test.user\G *************************** 1. row *************************** Table: user Create Table: CREATE TABLE `user` ( `id` int(11) NOT NULL AUTO_INCREMENT, `name` varchar(10) DEFAULT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=utf8
对于delete操做,咱们从binlog提取出delete信息,生成的回滚语句是insert。(注:为了方便解释,咱们用binlog2sql将原始binlog转化成了可读SQL)
原始:DELETE FROM `test`.`user` WHERE `id`=1 AND `name`='小赵'; 回滚:INSERT INTO `test`.`user`(`id`, `name`) VALUES (1, '小赵');
对于insert操做,回滚SQL是delete。
原始:INSERT INTO `test`.`user`(`id`, `name`) VALUES (2, '小钱'); 回滚:DELETE FROM `test`.`user` WHERE `id`=2 AND `name`='小钱';
对于update操做,回滚sql应该交换SET和WHERE的值。
原始:UPDATE `test`.`user` SET `id`=3, `name`='小李' WHERE `id`=3 AND `name`='小孙'; 回滚:UPDATE `test`.`user` SET `id`=3, `name`='小孙' WHERE `id`=3 AND `name`='小李';
真实的闪回场景中,最关键的是能快速筛选出真正须要回滚的SQL。
咱们使用开源工具binlog2sql来进行实战演练。binlog2sql由美团点评DBA团队(上海)出品,屡次在线上环境作快速回滚。
首先咱们安装binlog2sql:
shell> git clone https://github.com/danfengcao/binlog2sql.git && cd binlog2sql shell> pip install -r requirements.txt
背景:小明在11:44时误删了test库user表大批的数据,须要紧急回滚。
test库user表原有数据 mysql> select * from user; +----+--------+---------------------+ | id | name | addtime | +----+--------+---------------------+ | 1 | 小赵 | 2013-11-11 00:04:33 | | 2 | 小钱 | 2014-11-11 00:04:48 | | 3 | 小孙 | 2016-11-11 20:25:00 | | 4 | 小李 | 2013-11-11 00:00:00 | ......... +----+--------+---------------------+ 16384 rows in set (0.04 sec) 11:44时,user表大批数据被误删除。与此同时,正常业务数据是在继续写入的 mysql> delete from user where addtime>'2014-01-01'; Query OK, 16128 rows affected (0.18 sec) mysql> select count(*) from user; +----------+ | count(*) | +----------+ | 261 | +----------+
恢复数据步骤:
登陆mysql,查看目前的binlog文件
mysql> show master logs; +------------------+-----------+ | Log_name | File_size | +------------------+-----------+ | mysql-bin.000053 | 168652863 | | mysql-bin.000054 | 504549 | +------------------+-----------+
2. 最新的binlog文件是mysql-bin.000054。咱们的目标是筛选出须要回滚的SQL,因为误操做人只知道大体的误操做时间,咱们首先根据时间作一次过滤。只须要解析test库user表。(注:若是有多个sql误操做,则生成的binlog可能分布在多个文件,需解析多个文件) ```bash shell> python binlog2sql/binlog2sql.py -h127.0.0.1 -P3306 -uadmin -p'admin' -dtest -tuser --start-file='mysql-bin.000054' --start-datetime='2016-12-26 11:44:00' --stop-datetime='2016-12-26 11:50:00' > /tmp/raw.sql raw.sql 输出: DELETE FROM `test`.`user` WHERE `addtime`='2014-11-11 00:04:48' AND `id`=2 AND `name`='小钱' LIMIT 1; #start 257427 end 265754 time 2016-12-26 11:44:56 DELETE FROM `test`.`user` WHERE `addtime`='2015-11-11 20:25:00' AND `id`=3 AND `name`='小孙' LIMIT 1; #start 257427 end 265754 time 2016-12-26 11:44:56 ... DELETE FROM `test`.`user` WHERE `addtime`='2016-12-14 23:09:07' AND `id`=24530 AND `name`='tt' LIMIT 1; #start 257427 end 504272 time 2016-12-26 11:44:56 INSERT INTO `test`.`user`(`addtime`, `id`, `name`) VALUES ('2016-12-10 00:04:33', 32722, '小王'); #start 504299 end 504522 time 2016-12-26 11:49:42 ...
根据位置信息,咱们肯定了误操做sql来自同一个事务,准确位置在257427-504272之间(binlog2sql对于同一个事务会输出一样的start position)。再根据位置过滤,使用 -B 选项生成回滚sql,检查回滚sql是否正确。(注:真实场景下,生成的回滚SQL常常会须要进一步筛选。结合grep、编辑器等)
shell> python binlog2sql/binlog2sql.py -h127.0.0.1 -P3306 -uadmin -p'admin' -dtest -tuser --start-file='mysql-bin.000054' --start-position=257427 --stop-position=504272 -B > /tmp/rollback.sql rollback.sql 输出: INSERT INTO test
.user
(addtime
, id
, name
) VALUES ('2016-12-14 23:09:07', 24530, 'tt'); #start 257427 end 504272 time 2016-12-26 11:44:56 INSERT INTO test
.user
(addtime
, id
, name
) VALUES ('2016-12-12 00:00:00', 24529, '小李'); #start 257427 end 504272 time 2016-12-26 11:44:56 ... INSERT INTO test
.user
(addtime
, id
, name
) VALUES ('2014-11-11 00:04:48', 2, '小钱'); #start 257427 end 265754 time 2016-12-26 11:44:56
shell> wc -l /tmp/rollback.sql
16128 /tmp/rollback.sql ```
与业务方确认回滚sql没问题,执行回滚语句。登陆mysql,确认回滚成功。
shell> mysql -h127.0.0.1 -P3306 -uadmin -p'admin' < /tmp/rollback.sql mysql> select count(*) from user;
+----------+ | count(*) | +----------+ | 16389 | +----------+
###TIPS * 闪回的关键是快速筛选出真正须要回滚的SQL。 * 先根据库、表、时间作一次过滤,再根据位置作更准确的过滤。 * 因为数据一直在写入,要确保回滚sql中不包含其余数据。可根据是不是同一事务、误操做行数、字段值的特征等等来帮助判断。 * 执行回滚sql时若有报错,须要查实具体缘由,通常是由于对应的数据已发生变化。因为是严格的行模式,只要有惟一键(包括主键)存在,就只会报某条数据不存在的错,没必要担忧会更新不应操做的数据。 * 若是待回滚的表与其余表有关联,要与开发说明回滚和不回滚各自的反作用,再肯定方案。 * 回滚后数据变化,可能对用户和线上应用形成困惑(相似幻读)。 ####再重复下最重要的两点:**筛选出正确SQL**!**沟通清楚**! 闪回工具 === MySQL闪回特性最先由阿里彭立勋开发,彭在2012年给官方提交了一个patch,并对[闪回设计思路](http://www.penglixun.com/tech/database/mysql_flashback_feature.html)作了说明(设计思路颇有启发性,强烈推荐阅读)。可是由于种种缘由,业内安装这个patch的团队至今仍是少数,真正应用到线上的更是少之又少。彭以后,又有多位人员针对不一样mysql版本不一样语言开发了闪回工具,原理用的都是彭的思路。 我将这些闪回工具按实现方式分红了三类。 * 第一类是以patch形式集成到官方工具mysqlbinlog中。以彭提交的patch为表明。 > 优势 > > * 上手成本低。mysqlbinlog原有的选项都能直接利用,只是多加了一个闪回选项。闪回特性将来有可能被官方收录。 > * 支持离线解析。 > > 缺点 > > * 兼容性差、项目活跃度不高。因为binlog格式的变更,若是闪回工具做者不及时对补丁升级,则闪回工具将没法使用。目前已有多位人员分别针对mysql5.5,5.6,5.7开发了patch,部分项目代码公开,但整体上活跃度都不高。 > * 难以添加新功能,实战效果欠佳。在实战中,常常会遇到现有patch不知足需求的状况,好比要加个表过滤,很简单的一个需求,代码改动也不会大,但对大部分DBA来讲,改mysql源码仍是很困难的事。 > * 安装稍显麻烦。须要对mysql源码打补丁再编译生成。 > 这些缺点,可能都是闪回没有流行开来的缘由。 * 第二类是独立工具,经过假装成slave拉取binlog来进行处理。以binlog2sql为表明。 > 优势 > > * 兼容性好。假装成slave拉binlog这项技术在业界应用的很是普遍,多个开发语言都有这样的活跃项目,MySQL版本的兼容性由这些项目搞定,闪回工具的兼容问题再也不突出。 > * 添加新功能的难度小。更容易被改形成DBA本身喜欢的形式。更适合实战。 > * 安装和使用简单。 > > 缺点 > > * 必须开启MySQL server。 * 第三类是简单脚本。先用mysqlbinlog解析出文本格式的binlog,再根据回滚原理用正则进行匹配并替换。 > 优势 > > * 脚本写起来方便,每每能快速搞定某个特定问题。 > * 安装和使用简单。 > * 支持离线解析。 > > 缺点 > > * 通用性很差。 > * 可靠性很差。 就目前的闪回工具而言,线上环境的闪回,笔者建议使用binlog2sql,离线解析使用mysqlbinlog。 ###关于DDL的flashback 本文所述的flashback仅针对DML语句的快速回滚。但若是误操做是DDL的话,是没法利用binlog作快速回滚的,由于即便在row模式下,binlog对于DDL操做也不会记录每行数据的变化。要实现DDL快速回滚,必须修改MySQL源码,使得在执行DDL前先备份老数据。目前有多个mysql定制版本实现了DDL闪回特性,阿里林晓斌团队提交了patch给MySQL官方,MariaDB预计在不久后加入包含DDL的flashback特性。DDL闪回的反作用是会增长额外存储。考虑到其应用频次实在太低,本文不作详述,有兴趣的同窗能够本身去了解,重要的几篇文章我在参考资料中作了引用。 有任何问题,或有mysql闪回相关的优秀工具优秀文章遗漏,烦请告知。 danfengcao.info@gmail.com 参考资料 ============== [1] MySQL Internals Manual , [Chapter 20 The Binary Log](http://dev.mysql.com/doc/internals/en/binary-log.html) [2] 彭立勋,[MySQL下实现闪回的设计思路](http://www.penglixun.com/tech/database/mysql_flashback_feature.html) [3] Lixun Peng, [Provide the flashback feature by binlog](https://bugs.mysql.com/bug.php?id=65178) [4] 王广友,[mysqlbinlog flashback 5.6彻底使用手册与原理](http://www.cnblogs.com/youge-OneSQL/p/5249736.html) [5] 姜承尧, [拿走不谢,Flashback for MySQL 5.7](http://mp.weixin.qq.com/s?__biz=MjM5MjIxNDA4NA==&mid=2649737874&idx=1&sn=a993322ae58db541c2cf4d9a1efa3063&chksm=beb2d7b989c55eafb7ddcadb28f45bb6018b3e9e65df20b30217fe8cb26d3d444d58076f2d76&mpshare=1&scene=1&srcid=1228ta3qs3QIN6FS4AUCuCKm#rd) [6] 林晓斌, [MySQL闪回方案讨论及实现](http://dinglin.iteye.com/blog/1539167) [7] xiaobin lin, [flashback from binlog for MySQL](https://bugs.mysql.com/bug.php?id=65861) [8] mariadb.com, [AliSQL and some features that have made it into MariaDB Server](https://mariadb.com/resources/blog/alisql-and-some-features-have-made-it-mariadb-server) [9] danfengcao, [binlog2sql: Parse MySQL binlog to SQL you want](https://github.com/danfengcao/binlog2sql)