@@ -135,37 +135,37 @@ mysql> select count(*) from user;
1351351. 登录mysql,查看目前的binlog文件
136136
137137 ` ` ` bash
138- mysql> show master logs;
139- +------------------+-----------+
140- | Log_name | File_size |
141- +------------------+-----------+
142- | mysql-bin.000053 | 168652863 |
143- | mysql-bin.000054 | 504549 |
144- +------------------+-----------+
145- ` ` `
138+ mysql> show master logs;
139+ +------------------+-----------+
140+ | Log_name | File_size |
141+ +------------------+-----------+
142+ | mysql-bin.000053 | 168652863 |
143+ | mysql-bin.000054 | 504549 |
144+ +------------------+-----------+
145+ ` ` `
146146
1471472. 最新的binlog文件是mysql-bin.000054。我们的目标是筛选出需要回滚的SQL,由于误操作人只知道大致的误操作时间,我们首先根据时间做一次过滤。只需要解析test库user表。(注:如果有多个sql误操作,则生成的binlog可能分布在多个文件,需解析多个文件)
148148
149149 ` ` ` bash
150- 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
150+ 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
151151raw.sql 输出:
152- 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
153- 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
154- ...
155- 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
156- 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
157- ...
158- ` ` `
152+ 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
153+ 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
154+ ...
155+ 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
156+ 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
157+ ...
158+ ` ` `
159159
1601603. 根据位置信息,我们确定了误操作sql来自同一个事务,准确位置在257427-504272之间(binlog2sql对于同一个事务会输出同样的start position)。再根据位置过滤,使用 _** -B** _ 选项生成回滚sql,检查回滚sql是否正确。(注:真实场景下,生成的回滚SQL经常会需要进一步筛选。结合grep、编辑器等)
161161
162162 ` ` ` bash
163- 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
163+ 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
164164rollback.sql 输出:
165- 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
166- 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
167- ...
168- 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
165+ 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
166+ 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
167+ ...
168+ 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
169169
170170 shell> wc -l /tmp/rollback.sql
17117116128 /tmp/rollback.sql
@@ -177,30 +177,33 @@ INSERT INTO `test`.`user`(`addtime`, `id`, `name`) VALUES ('2014-11-11 00:04:48'
177177 shell> mysql -h127.0.0.1 -P3306 -uadmin -p' admin' < /tmp/rollback.sql
178178
179179 mysql> select count(*) from user;
180- +----------+
181- | count(* ) |
182- +----------+
183- | 16389 |
184- +----------+
185- ` ` `
180+ +----------+
181+ | count(* ) |
182+ +----------+
183+ | 16389 |
184+ +----------+
185+ ` ` `
186+
187+ # ## TIPS
186188
187- # ##TIPS
188189* 闪回的关键是快速筛选出真正需要回滚的SQL。
189190* 先根据库、表、时间做一次过滤,再根据位置做更准确的过滤。
190191* 由于数据一直在写入,要确保回滚sql中不包含其他数据。可根据是否是同一事务、误操作行数、字段值的特征等等来帮助判断。
191192* 执行回滚sql时如有报错,需要查实具体原因,一般是因为对应的数据已发生变化。由于是严格的行模式,只要有唯一键(包括主键)存在,就只会报某条数据不存在的错,不必担心会更新不该操作的数据。
192193* 如果待回滚的表与其他表有关联,要与开发说明回滚和不回滚各自的副作用,再确定方案。
193194* 回滚后数据变化,可能对用户和线上应用造成困惑(类似幻读)。
194195
195- # ###再重复下最重要的两点:**筛选出正确SQL**!**沟通清楚**!
196+ # ### 再重复下最重要的两点:**筛选出正确SQL**!**沟通清楚**!
196197
197198闪回工具
198199===
200+
199201MySQL闪回特性最早由阿里彭立勋开发,彭在2012年给官方提交了一个patch,并对[闪回设计思路](http://www.penglixun.com/tech/database/mysql_flashback_feature.html)做了说明(设计思路很有启发性,强烈推荐阅读)。但是因为种种原因,业内安装这个patch的团队至今还是少数,真正应用到线上的更是少之又少。彭之后,又有多位人员针对不同mysql版本不同语言开发了闪回工具,原理用的都是彭的思路。
200202
201203我将这些闪回工具按实现方式分成了三类。
202204
203205* 第一类是以patch形式集成到官方工具mysqlbinlog中。以彭提交的patch为代表。
206+
204207 > 优点
205208 >
206209 > * 上手成本低。mysqlbinlog原有的选项都能直接利用,只是多加了一个闪回选项。闪回特性未来有可能被官方收录。
@@ -215,6 +218,7 @@ MySQL闪回特性最早由阿里彭立勋开发,彭在2012年给官方提交
215218 > 这些缺点,可能都是闪回没有流行开来的原因。
216219
217220* 第二类是独立工具,通过伪装成slave拉取binlog来进行处理。以binlog2sql为代表。
221+
218222 > 优点
219223 >
220224 > * 兼容性好。伪装成slave拉binlog这项技术在业界应用的非常广泛,多个开发语言都有这样的活跃项目,MySQL版本的兼容性由这些项目搞定,闪回工具的兼容问题不再突出。
@@ -226,6 +230,7 @@ MySQL闪回特性最早由阿里彭立勋开发,彭在2012年给官方提交
226230 > * 必须开启MySQL server。
227231
228232* 第三类是简单脚本。先用mysqlbinlog解析出文本格式的binlog,再根据回滚原理用正则进行匹配并替换。
233+
229234 > 优点
230235 >
231236 > * 脚本写起来方便,往往能快速搞定某个特定问题。
@@ -240,7 +245,8 @@ MySQL闪回特性最早由阿里彭立勋开发,彭在2012年给官方提交
240245就目前的闪回工具而言,线上环境的闪回,笔者建议使用binlog2sql,离线解析使用mysqlbinlog。
241246
242247
243- # ##关于DDL的flashback
248+ # ## 关于DDL的flashback
249+
244250本文所述的flashback仅针对DML语句的快速回滚。但如果误操作是DDL的话,是无法利用binlog做快速回滚的,因为即使在row模式下,binlog对于DDL操作也不会记录每行数据的变化。要实现DDL快速回滚,必须修改MySQL源码,使得在执行DDL前先备份老数据。目前有多个mysql定制版本实现了DDL闪回特性,阿里林晓斌团队提交了patch给MySQL官方,MariaDB预计在不久后加入包含DDL的flashback特性。DDL闪回的副作用是会增加额外存储。考虑到其应用频次实在过低,本文不做详述,有兴趣的同学可以自己去了解,重要的几篇文章我在参考资料中做了引用。
245251
246252
@@ -249,6 +255,7 @@ MySQL闪回特性最早由阿里彭立勋开发,彭在2012年给官方提交
249255
250256参考资料
251257==============
258+
252259[1] MySQL Internals Manual
253260, [Chapter 20 The Binary Log](http://dev.mysql.com/doc/internals/en/binary-log.html)
254261
0 commit comments