update README

This commit is contained in:
danfengcao 2016-12-29 23:03:27 +08:00
parent 36618af652
commit 0d1c89bdbe
5 changed files with 271 additions and 209 deletions

View File

@ -13,7 +13,7 @@ binlog2sql
项目状态
===
正常维护。已在大众点评线上环境使用。线上环境的操作请在对MySQL熟悉的同学指导下进行
正常维护。应用于大众点评线上环境。线上环境的操作请在对MySQL**相当熟悉**的同学指导下进行
* 已测试环境
* Python 2.6, 2.7
@ -112,9 +112,7 @@ UPDATE `test`.`test3` SET `addtime`='2016-12-10 13:03:22', `data`='中文', `id`
#### **误删整张表数据,需要紧急回滚**
详细描述可参见[example/mysql-rollback-your-data.md](./example/mysql-rollback-your-data.md)
更多应用案例可参见[example/](./example/)
闪回详细介绍可参见example目录下《闪回原理与实战》[example/mysql-flashback-priciple-and-practice.md](./example/mysql-flashback-priciple-and-practice.md)
```bash
test库tbl表原有数据
@ -156,7 +154,6 @@ mysql> show master status;
```bash
shell> python binlog2sql/binlog2sql.py -h127.0.0.1 -P3306 -uadmin -p'admin' -dtest -ttbl --start-file='mysql-bin.000052' --start-datetime='2016-12-13 20:25:00' --stop-datetime='2016-12-13 20:30:00'
输出:
INSERT INTO `test`.`tbl`(`addtime`, `id`, `name`) VALUES ('2016-12-13 20:25:00', 3, '小孙'); #start 4 end 290 time 2016-12-13 20:25:46
INSERT INTO `test`.`tbl`(`addtime`, `id`, `name`) VALUES ('2016-12-13 20:26:00', 4, '小李'); #start 317 end 487 time 2016-12-13 20:26:26
UPDATE `test`.`tbl` SET `addtime`='2016-12-12 00:00:00', `id`=4, `name`='小李' WHERE `addtime`='2016-12-13 20:26:00' AND `id`=4 AND `name`='小李' LIMIT 1; #start 514 end 701 time 2016-12-13 20:27:07
DELETE FROM `test`.`tbl` WHERE `addtime`='2016-12-10 00:04:33' AND `id`=1 AND `name`='小赵' LIMIT 1; #start 728 end 938 time 2016-12-13 20:28:05

View File

@ -1,63 +0,0 @@
MySQL主从切换后如何进行数据修复
========================
目前的MySQL高可用方案多是基于replication实现。MySQL replication默认是异步复制它有个重要缺陷主从延迟。当出现主从延迟时如果master出现宕机且无法将全部数据发送至slave则slave将丢失这部分数据。半同步复制和同步复制能避免主从延迟发生但会使得性能大大下降以至于诸多采用mysql存储的高性能系统仍旧采用异步复制。本文对主从切换后的数据不一致提供了一种新的解决办法。
### 主从切换后的状态
如下图所示切换完成后old master(10.1.1.1:3306)有一部分binlog(unreplicated binlog)未同步到new master(10.1.1.2:3306)new master已经有新数据进入。此时主从数据已经不一致。
![](./static/master-slave-inconsistency.jpg)
### 常见办法
将old master指向new master开启同步。执行命令如下
```bash
> CHANGE MASTER TO MASTER_HOST='10.1.1.2', MASTER_PORT=3306, MASTER_USER='xxx', MASTER_PASSWORD=xxx;
> CHANGE MASTER TO master_log_file='mysql-bin.000001',master_log_pos= 2448;
> start slave;
```
此时一般会报错duplicate entry for key 'primary'。DBA们往往会选择以下几种方式来处理
1. 粗暴跳过所有错误,直到复制不再报错。
缺点:主从数据不一致,留下安全隐患。
2. 根据报错信息逐次修old master数据。如报duplicate entry for key 'primary' for table a where id=12345, 则将old master 表a id=12345的数据删除再继续开启同步。
缺点手工修复起来繁琐unreplicated binlog部分事务丢失如果unreplicated binlog较多的话主从数据很大概率还是会不一致。
3. 将old master回滚至mysql-bin.00040 120再开启同步。同步正常。
缺点unreplicated binlog丢失。
### 新办法
**步骤**
1. 提取old master未同步的数据并对其中的insert语句去除主键为了防止步骤3中出现主键冲突
2. 将old master回滚至mysql-bin.00040 120开启同步。同步正常
3. 在new master重新导入改造后的sql
我们使用了纯Python开发的工具[binlog2sql](https://github.com/danfengcao/binlog2sql)来操作,执行命令如下
```bash
$ python binlog2sql.py --no-primary-key -h10.1.1.1 -P3306 -uadmin -p'admin' --start-file='mysql-bin.000040' --start-pos=125466 --end-file='mysql-bin.000041' > oldMaster.sql
$ python binlog2sql.py --flashback -h10.1.1.1 -P3306 -uadmin -p'admin' --start-file='mysql-bin.mysql-bin.000040' --start-pos=125466 --end-file='mysql-bin.000041' | mysql -h10.1.1.1 -P3306 -uadmin -p'admin'
$ mysql -h10.1.1.2 -P3306 -uadmin -p'admin' < oldMaster.sql
```
**优点**
数据丢失最少,操作简单快捷。
**限制**
如果表的主键id是有业务含义的则务必与业务方确认可行后再操作。
###参考资料
[1] 易固武, [MySQL数据库的高可用性分析](https://www.qcloud.com/community/article/203)
[2] danfengcao, [binlog2sql: Parse MySQL binlog to SQL you want](https://github.com/danfengcao/binlog2sql)

View File

@ -0,0 +1,269 @@
MySQL闪回原理与实战
========================
DBA或开发人员有时会误删或者误更新数据如果是线上环境并且影响较大就需要能快速回滚。传统恢复方法是利用备份重搭实例再应用去除错误sql后的binlog来恢复数据。此法费时费力甚至需要停机维护并不适合快速回滚。也有团队利用LVM快照来缩短恢复时间但快照的缺点是会影响mysql的性能。
MySQL闪回(flashback)利用binlog直接进行回滚能快速恢复且不用停机。本文将介绍闪回原理给出笔者的实战经验并对现存的闪回工具作比较。
开胃菜
===
某天小明因种种原因误删了大批线上用户表的数据。他急忙找到公司DBA请求帮助“客服电话已被打爆大量用户投诉无法登陆领导非常恼火。请问多久能恢复数据”DBA一脸懵逼沉默十秒后伸出一根手指。“你的意思是一分钟就能恢复太好了。”小明终于有些放松露出了一丝笑容。“不我们中有个人将会离开公司。”DBA沉痛的说道。
勿让悲剧发生尽早将此文转给公司DBA。
闪回原理
===
**binlog概述**
MySQL binlog以event的形式记录了MySQL server从启用binlog以来所有的变更信息能够帮助重现这之间的所有变化。MySQL引入binlog主要有两个目的一是为了主从复制二是某些备份还原操作后需要重新应用binlog。
有三种可选的binlog格式各有优缺点
* statement基于SQL语句的模式binlog数据量小但是某些语句和函数在复制过程可能导致数据不一致甚至出错
* row基于行的模式记录的是行的完整变化。很安全但是binlog会比其他两种模式大很多
* mixed混合模式根据语句来选用是statement还是row模式
利用binlog闪回**需要将binlog格式设置为row**。row模式下一条使用innodb的insert会产生如下格式的binlog
```
# 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从后往前回滚回去即可。
对于单个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](https://github.com/danfengcao/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表大批的数据需要紧急回滚。
```bash
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 |
+----------+
```
**恢复数据步骤**
1. 登录mysql查看目前的binlog文件
```bash
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
...
```
3. 根据位置信息我们确定了误操作sql来自同一个事务准确位置在257427-504272之间(binlog2sql对于同一个事务会输出同样的start position)。再根据位置过滤,使用 _**-B**_ 选项生成回滚sql检查回滚sql是否正确。(注真实场景下生成的回滚SQL经常会需要进一步筛选。结合grep、编辑器等)
```bash
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
```
4. 与业务方确认回滚sql没问题执行回滚语句。登录mysql确认回滚成功。
```bash
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.55.65.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)

View File

@ -1,141 +0,0 @@
MySQL误操作后如何快速恢复数据
========================
基本上每个跟数据库打交道的程序员当然也可能是你同事都会碰一个问题MySQL误操作后如何快速回滚比如不小心update了整张表的某个字段或者delete一张表忘加限制条件整张表都没了。假如这还是线上环境核心业务数据那这事就闹大了。误操作后能快速回滚数据是非常重要的。
传统解法
===
用全量备份重搭实例再利用增量binlog备份恢复到误操作之前的状态。然后跳过误操作的SQL再继续应用binlog。对于DML的回滚此法费时费力不值得再推荐。
利用binlog2sql快速闪回
===
首先确认你的MySQL server开启了binlog设置了以下参数:
[mysqld]
server-id = 1
log_bin = /var/log/mysql/mysql-bin.log
max_binlog_size = 100M
binlog-format = row
如果没有开启binlog也没有预先生成回滚SQL那真的无法快速回滚了。对存放重要业务数据的MySQL强烈建议开启binlog。
随后,安装开源工具[binlog2sql](https://github.com/danfengcao/binlog2sql)。binlog2sql是一款简单易用的binlog解析工具其中一个功能就是生成回滚SQL。
```
shell> git clone https://github.com/danfengcao/binlog2sql.git
shell> pip install -r requirements.txt
```
然后我们就可以生成回滚SQL了。
**背景**小明在20点多时误删了test库tbl表整张表的数据需要紧急回滚。
```bash
test库tbl表原有数据
mysql> select * from tbl;
+----+--------+---------------------+
| id | name | addtime |
+----+--------+---------------------+
| 1 | 小赵 | 2016-12-10 00:04:33 |
| 2 | 小钱 | 2016-12-10 00:04:48 |
| 3 | 小孙 | 2016-12-13 20:25:00 |
| 4 | 小李 | 2016-12-12 00:00:00 |
+----+--------+---------------------+
4 rows in set (0.00 sec)
mysql> delete from tbl;
Query OK, 4 rows affected (0.00 sec)
20:28时tbl表误操作被清空
mysql> select * from tbl;
Empty set (0.00 sec)
```
**恢复数据步骤**
1. 登录mysql查看目前的binlog文件
```bash
mysql> show master status;
+------------------+-----------+
| Log_name | File_size |
+------------------+-----------+
| mysql-bin.000051 | 967 |
| mysql-bin.000052 | 965 |
+------------------+-----------+
```
2. 最新的binlog文件是mysql-bin.000052我们再定位误操作SQL的binlog位置。误操作人只能知道大致的误操作时间我们根据大致时间过滤数据。
```bash
shell> python binlog2sql/binlog2sql.py -h127.0.0.1 -P3306 -uadmin -p'admin' -dtest -ttbl --start-file='mysql-bin.000052' --start-datetime='2016-12-13 20:25:00' --stop-datetime='2016-12-13 20:30:00'
输出:
INSERT INTO `test`.`tbl`(`addtime`, `id`, `name`) VALUES ('2016-12-13 20:25:00', 3, '小孙'); #start 4 end 290 time 2016-12-13 20:25:46
INSERT INTO `test`.`tbl`(`addtime`, `id`, `name`) VALUES ('2016-12-13 20:26:00', 4, '小李'); #start 317 end 487 time 2016-12-13 20:26:26
UPDATE `test`.`tbl` SET `addtime`='2016-12-12 00:00:00', `id`=4, `name`='小李' WHERE `addtime`='2016-12-13 20:26:00' AND `id`=4 AND `name`='小李' LIMIT 1; #start 514 end 701 time 2016-12-13 20:27:07
DELETE FROM `test`.`tbl` WHERE `addtime`='2016-12-10 00:04:33' AND `id`=1 AND `name`='小赵' LIMIT 1; #start 728 end 938 time 2016-12-13 20:28:05
DELETE FROM `test`.`tbl` WHERE `addtime`='2016-12-10 00:04:48' AND `id`=2 AND `name`='小钱' LIMIT 1; #start 728 end 938 time 2016-12-13 20:28:05
DELETE FROM `test`.`tbl` WHERE `addtime`='2016-12-13 20:25:00' AND `id`=3 AND `name`='小孙' LIMIT 1; #start 728 end 938 time 2016-12-13 20:28:05
DELETE FROM `test`.`tbl` WHERE `addtime`='2016-12-12 00:00:00' AND `id`=4 AND `name`='小李' LIMIT 1; #start 728 end 938 time 2016-12-13 20:28:05
```
3. 我们得到了误操作sql的准确位置在728-938之间再根据位置过滤使用flashback模式生成回滚sql检查回滚sql是否正确(注真实环境下此步经常会进一步筛选出需要的sql。结合grep、编辑器等)
```bash
shell> python binlog2sql/binlog2sql.py -h127.0.0.1 -P3306 -uadmin -p'admin' -dtest -ttbl --start-file='mysql-bin.000052' --start-position=3346 --stop-position=3556 -B > rollback.sql | cat
输出:
INSERT INTO `test`.`tbl`(`addtime`, `id`, `name`) VALUES ('2016-12-12 00:00:00', 4, '小李'); #start 728 end 938 time 2016-12-13 20:28:05
INSERT INTO `test`.`tbl`(`addtime`, `id`, `name`) VALUES ('2016-12-13 20:25:00', 3, '小孙'); #start 728 end 938 time 2016-12-13 20:28:05
INSERT INTO `test`.`tbl`(`addtime`, `id`, `name`) VALUES ('2016-12-10 00:04:48', 2, '小钱'); #start 728 end 938 time 2016-12-13 20:28:05
INSERT INTO `test`.`tbl`(`addtime`, `id`, `name`) VALUES ('2016-12-10 00:04:33', 1, '小赵'); #start 728 end 938 time 2016-12-13 20:28:05
```
4. 确认回滚sql正确执行回滚语句。登录mysql确认数据回滚成功。
```bash
shell> mysql -h127.0.0.1 -P3306 -uadmin -p'admin' < rollback.sql
mysql> select * from tbl;
+----+--------+---------------------+
| id | name | addtime |
+----+--------+---------------------+
| 1 | 小赵 | 2016-12-10 00:04:33 |
| 2 | 小钱 | 2016-12-10 00:04:48 |
| 3 | 小孙 | 2016-12-13 20:25:00 |
| 4 | 小李 | 2016-12-12 00:00:00 |
+----+--------+---------------------+
```
至此,不用再担心被炒鱿鱼了。
常见问题
===
* 有人会问我DDL误操作了怎么快速回滚比如drop了一张大表。
> 很难做到。因为即使在在row模式下DDL操作也不会把每行数据的变化记录到binlog所以DDL无法通过binlog回滚。实现DDL回滚必须要在执行DDL前先备份老数据。确实有人通过修改mysql server源码实现了DDL的快速回滚我找到阿里的xiaobin lin提交了一个patch。但据我所知国内很少有互联网公司应用了这个特性。原因的话我认为最主要还是懒的去折腾没必要搞这个低频功能次要原因是会增加一些额外存储。
>
所以DDL误操作的话一般只能通过备份来恢复。如果公司连备份也不能用了那真的建议去买张飞机票了。干啥跑呗
* mysql除了[binlog2sql](https://github.com/danfengcao/binlog2sql),是否还有其他回滚工具?
>当然有。阿里彭立勋对mysqlbinlog增加了flashback的特性这应该是mysql最早有的flashback功能彭解决的是DML的回滚并说明了利用binlog进行DML闪回的设计思路。DDL回滚特性也是由阿里团队提出并实现的。这两个功能是有创新精神的此后出现的闪回工具基本都是对上面两者的模仿。另外去哪儿开源的Inception是一套MySQL自动化运维工具这个就比较重了支持DML回滚还不是从binlog回滚的是从备份回滚的也支持DDL回滚表结构数据是回滚不了滴~ 还有一种做法叫slave延时备份搞台不加业务流量的slave故意延迟一段时间这其实是在传统办法的基础上去除了实例恢复这步。此法会额外消耗一台机器我们不推荐这么做。
如有mysql回滚相关的优秀工具优秀文章遗漏烦请告知。
我的邮箱 danfengcao.info@gmail.com
参考资料
==============
[1] 彭立勋, [MySQL下实现闪回的设计思路](http://www.penglixun.com/tech/database/mysql_flashback_feature.html)
[2] Lixun Peng, [Provide the flashback feature by binlog](https://bugs.mysql.com/bug.php?id=65178)
[3] 丁奇, [MySQL闪回方案讨论及实现](http://dinglin.iteye.com/blog/1539167)
[4] xiaobin lin, [flashback from binlog for MySQL](https://bugs.mysql.com/bug.php?id=65861)
[5] 王竹峰, [去哪儿inception](https://github.com/mysql-inception/inception)
[6] danfengcao, [binlog2sql: Parse MySQL binlog to SQL you want](https://github.com/danfengcao/binlog2sql)

Binary file not shown.

Before

Width:  |  Height:  |  Size: 38 KiB