mysql基于binlog回滚工具_flashback(python版本)
update、delete的条件写错甚至没有写,导致数据操作错误,需要恢复被误操作的行记录。这种情形,其实时有发生,可以选择用备份文件+binlog来恢复到测试环境,然后再做数据修复,但是这样其实需要耗费一定的时间跟资源。
1 实现内容
- 回滚分为2个命令:第一个命令 分析binglog并存储进入数据库;第二个命令 执行回滚操作;
- 回滚的时候,可以把执行脚本跟回滚脚本统一存放到数据库中,可以查看 更新内容以及回滚内容;
- 根据存储的分析表格,方便指定事务或者指定表格来来恢复;
- 详细的日志输出,说明分析进度跟执行进度。
2 原理
- 判断一个事务的开始跟结束
- 同一个事务的执行顺序需要反序执行
- 解析回滚SQL
- 同一个事务操作不同表格处理
- 转义字符处理,比如换行符、tab符等等
- timestamp数据类型参数值转换
- 负数处理
- 单个事务涉及到行修改SQL操作了 max_allow
- 针对某个表格做回滚,而不是全库回滚
2.1 事务的开始与结束
2.2 事务内部反序处理
- 每行记录的修改SQL独立出来
- 将独立出来的SQL反序存储
2.3 解析回滚SQL
- 行记录的列名配对,binlog file存储的列序号,不能直接使用
- WHERE部分 跟 SET部分 之间并无关键字或者符号,需要添加 AND 或者 逗号
- DELETE SQL 需要反转为 INSERT
- UPDATE SQL 需要把WHERE 跟 SET的部分进行替换
- INSERT SQL需要反转为 DELETE
2.4 同事务不同表格处理
2.5 转义字符处理
2.6 timestamp数据类型处理
2.7 负数值处理
2.8 单个事务行记录总SQL超过max_allowed_package处理
2.9 针对性回滚
3 使用说明
3.1 参数说明
- 黄色区域:这6个参数,提供的是 分析并存储binlog file的相关值,说明存储分析结果的数据库的链接方式、binlog文件的位置以及存储结果的表格名字;
- 蓝色区域:这4个参数,提供 与线上数据库表结构一致的DB实例连接方式,仅需跟线上一模一样的表结构,不一定需要是主从库;
- 绿色区域:最最重要的选项 -a,0代表仅分析binlog文件,1代表仅执行回滚操作,必须先执行0才可以执行1;
- 紫色区域:举例说明。
3.2 应用场景说明
- 全库回滚某段时间
- 需要回滚某个时间段的所有SQL操作,回滚到某一个时间点
- 这种情况下呢,大多数是使用备份文件+binlog解决
- 但是这个脚本也可以满足,但请勿直接在线上操作,先 -a=0,看下分析结果,是否符合,符合的话,停掉某个从库,再在从库上执行,最后开发业务接入检查是否恢复到指定时间点,数据是否正常。
- 某段时间某些表格回滚某些操作
- 比如,开发提交了一个批量更新脚本,各个测试层面验证没有问题,提交线上执行,但是执行后,发现有个业务漏测试,导致某些字段更新后影响到其他业务,现在需要紧急 把被批量更新的表格回滚到原先的行记录
- 这个并不能单纯从技术角度来处理,要综合考虑
- 这种情况下,如何回顾tab A表格的修改操作呢?
- 个人觉得,这种方式比较行得通,dump tabA表格的数据到测试环境,然后再分析 binlog file 从11点-12点的undo sql,接着在测试环境回滚该表格到11点这个时刻,紧接着,由开发跟业务对比测试环境11点的数据跟线上现有的数据中,看下是哪些行哪些列需要在线上进行回滚,哪些是不需要的,然后开发再提交SQL脚本,再在线上执行。其实,这里边,DBA仅提供一个角色,就是把 表格 tab A 在一个新的环境上,回滚到某个时间点,但是不提供直接线上回滚SQL的处理。
- 回滚某个/些SQL
- 这种情况比较常见,某个update某个delete缺少where条件或者where条件执行错误
- 这种情况下,找到对应的事务,执行回滚即可,回滚流程请参考上面一说,对的,我就是这么胆小怕事
3.3 测试案例
3.3.1 全库回滚某段时间
- 准备测试环境实例存储分析后的数据
- 测试环境修改set global max_allowed_packet = 1073741824
- mysqlbinlog分析binlog文件
- python脚本分析文件,action=0
- 线上测试环境修改set global max_allowed_packet = 1073741824
- 回滚数据,action=1
- 线上测试环境修改set global max_allowed_packet = 4194304
--测试环境(请安装pymysql):IP: 192.168.9.242,PORT:3310 ,数据库:flashback,表格:tbevent
--具有线上表结构的db:IP:192.168.9.243 PORT:3310 mysql> show global variables like 'max_allowed_packet';
+--------------------+----------+
| Variable_name | Value |
+--------------------+----------+
| max_allowed_packet | 16777216 |
+--------------------+----------+
1 row in set (0.00 sec) mysql> set global max_allowed_packet = 1073741824;
Query OK, 0 rows affected (0.00 sec) [root@sutest244 ~]# mysqlbinlog --start-datetime='2017-06-19 09:00:00' --stop-datetime='2017-06-19 10:00:00' --base64-output=decode-rows -v ~/data/mysql/data/mysql-bin.007335 > /tmp/binlog.log [root@sutest242 pycharm]# python su_flashback.py -h=127.0.0.1 -P=3310 -u=root -p=**** -f=/tmp/binlog.log -t=flashback.tbevent -oh=192.168.9.244 -oP=3310 -u=root -op=**** -a=0
2017-06-19 10:59:39,041 INFO begin to assign values to parameters
2017-06-19 10:59:39,041 INFO assign values to parameters is done:host=127.0.0.1,user=root,password=***,port=3310,fpath=/tmp/binlog.log,tbevent=flashback.tbevent
2017-06-19 10:59:39,049 INFO MySQL which userd to store binlog event connection is ok
2017-06-19 10:59:39,050 INFO assign values to online mysql parameters is done:host=192.168.9.244,user=,password=***,port=3310
2017-06-19 10:59:39,054 INFO MySQL which userd to analyse online table schema connection is ok
2017-06-19 10:59:39,054 INFO MySQL connection is ok
2017-06-19 10:59:39,055 INFO creating table flashback.tbevent to store binlog event
2017-06-19 10:59:39,058 INFO created table flashback.tbevent
2017-06-19 10:59:39,060 INFO begining to analyze the binlog file ,this may be take a long time !!!
2017-06-19 10:59:39,061 INFO analyzing...
2017-06-19 11:49:53,781 INFO finished to analyze the binlog file !!!
2017-06-19 11:49:53,782 INFO release all db connections
2017-06-19 11:49:53,782 INFO All done,check the flashback.tbevent which stored binlog event on host 127.0.0.1 , port 3310 [root@sutest242 pycharm]# python su_flashback.py -h=127.0.0.1 -P=3310 -u=root -p=**** -f=/tmp/binlog.log -t=flashback.tbevent -oh=192.168.9.244 -oP=3310 -u=root -op=**** -a=1
2017-06-19 16:30:20,633 INFO begin to assign values to parameters
2017-06-19 16:30:20,635 INFO assign values to parameters is done:host=127.0.0.1,user=root,password=***,port=3310,fpath=/tmp/binlog.log,tbevent=flashback.tbevent
2017-06-19 16:30:20,865 INFO MySQL which userd to store binlog event connection is ok
2017-06-19 16:30:20,866 INFO assign values to online mysql parameters is done:host=192.168.9.244,user=,password=***,port=3310
2017-06-19 16:30:20,871 INFO MySQL which userd to analyse online table schema connection is ok
2017-06-19 16:30:20,871 INFO MySQL connection is ok
2017-06-19 16:30:21,243 INFO There has 347868 transactions ,need 35 batchs ,each batche doing 10000 transactions
2017-06-19 16:30:21,243 INFO doing batch : 1
2017-06-19 16:31:01,182 INFO doing batch : 2
2017-06-19 16:31:16,909 INFO doing batch : 3
-------省空间忽略不截图--------------
2017-06-19 16:41:11,287 INFO doing batch : 34
2017-06-19 16:41:25,577 INFO doing batch : 35
2017-06-19 16:41:44,629 INFO release all db connections
2017-06-19 16:41:44,630 INFO All done,check the flashback.tbevent which stored binlog event on host 127.0.0.1 , port 3310
3.3.2 某段时间某些表格回滚某些操作
- 准备测试环境实例存储分析后的数据
- 测试环境修改set global max_allowed_packet = 1073741824
- mysqlbinlog分析binlog文件
- python脚本分析文件,action=0
- 分析帅选需要的事务,rename表格
- dump 对应的表格到测试环境
- 回滚数据,action=1
- 提交给开发业务对比数据
3.3.3 回滚某个/些SQL
- 准备测试环境实例存储分析后的数据
- 测试环境修改set global max_allowed_packet = 1073741824
- mysqlbinlog分析binlog文件
- python脚本分析文件,action=0
- 分析帅选需要的事务,rename表格
- dump 对应的表格到测试环境
- 回滚数据,action=1
- 提交给开发业务对比数据
4 python脚本
脚本会不定期修复bug,若是感兴趣,可以往github下载:https://github.com/xinysu/mysql.git 中的 mysql_xinysu_flashback 。
# -*- coding: utf-8 -*-
__author__ = 'xinysu'
__date__ = '2017/6/15 10:30' import re
import os
import sys
import datetime
import time
import logging
import importlib
importlib.reload(logging)
logging.basicConfig(level=logging.DEBUG,format='%(asctime)s %(levelname)s %(message)s ') import pymysql
from pymysql.cursors import DictCursor usage='''\nusage: python [script's path] [option]
ALL options need to assign:
\033[1;33;40m
-h : host, the database host,which database will store the results after analysis
-u : user, the db user
-p : password, the db user's password
-P : port, the db port -f : file path, the binlog file
-t : table name, the table name to store the results after analysis , {dbname}.{tbname},
when you want to store in `test` db and the table name is `tbevent`,then this parameter
is test.tbevent
\033[1;34;40m
-oh : online host, the database host,which database have the online table schema
-ou : online user, the db user
-op : online password, the db user's password
-oP : online port, the db port
\033[1;32;40m
-a : action,
0 just analyse the binlog file ,and store sql in table;
1 after execute self.dotype=0, execute the undo_sql in the table
\033[0m
--help: help document
\033[1;35;40m
Example:
analysize binlog:
python su_flashback.py -h=127.0.0.1 -P=3310 -u=root -p=*** -f=/tmp/binlog.log -t=flashback.tbevent
-oh=192.168.9.244 -oP=3310 -u=root -op=***
-a=0 flash back:
python su_flashback.py -h=127.0.0.1 -P=3310 -u=root -p=*** -f=/tmp/binlog.log -t=flashback.tbevent
-oh=192.168.9.244 -oP=3310 -u=root -op=***
-a=1
\033[0m
''' class flashback:
def __init__(self):
self.host=''
self.user=''
self.password=''
self.port=''
self.fpath=''
self.tbevent='' self.on_host=''
self.on_user=''
self.on_password=''
self.on_port='' self.action=0 # 0 just analyse the binlog file ,and store sql in table;1 after execute self.dotype=0, execute the undo_sql in the table self._get_db() # 从输入参数获取连接数据库的相关参数值 # 连接数据库,该数据库是用来存储binlog文件分析后的内容
logging.info('assign values to parameters is done:host={},user={},password=***,port={},fpath={},tbevent={}'.format(self.host,self.user,self.port,self.fpath,self.tbevent))
self.mysqlconn = pymysql.connect(host=self.host, user=self.user, password=self.password, port=self.port,charset='utf8')
self.cur = self.mysqlconn.cursor(cursor=DictCursor)
logging.info('MySQL which userd to store binlog event connection is ok') # 连接数据库,该数据库的表结构必须跟binlogfile基于对数据库表结构一致
# 该数据库用于提供 binlog file 文件中涉及到表结构分析
logging.info('assign values to online mysql parameters is done:host={},user={},password=***,port={}'.format(self.on_host, self.on_user, self.on_port))
self.on_mysqlconn = pymysql.connect(host=self.on_host, user=self.on_user, password=self.on_password, port=self.on_port,charset='utf8')
self.on_cur = self.on_mysqlconn.cursor(cursor=DictCursor)
logging.info('MySQL which userd to analyse online table schema connection is ok') logging.info('\033[33mMySQL connection is ok\033[0m') self.dml_sql=''
self.undo_sql='' self.tbfield_where = []
self.tbfield_set = [] self.begin_time=''
self.db_name=''
self.tb_name=''
self.end_time=''
self.end_pos=''
self.sqltype=0 #_get_db用于获取执行命令的输入参数
def _get_db(self):
logging.info('begin to assign values to parameters')
if len(sys.argv) == 1:
print(usage)
sys.exit(1)
elif sys.argv[1] == '--help':
print(usage)
sys.exit()
elif len(sys.argv) > 2:
for i in sys.argv[1:]:
_argv = i.split('=')
if _argv[0] == '-h':
self.host = _argv[1]
elif _argv[0] == '-u':
self.user = _argv[1]
elif _argv[0] == '-P':
self.port = int(_argv[1])
elif _argv[0] == '-f':
self.fpath = _argv[1]
elif _argv[0] == '-t':
self.tbevent = _argv[1]
elif _argv[0] == '-p':
self.password = _argv[1] elif _argv[0] == '-oh':
self.on_host = _argv[1]
elif _argv[0] == '-ou':
self.on_user = _argv[1]
elif _argv[0] == '-oP':
self.on_port = int(_argv[1])
elif _argv[0] == '-op':
self.on_password = _argv[1] elif _argv[0] == '-a':
self.action = _argv[1] else:
print(usage) #创建表格,用于存储分析后的BINLOG内容
def create_tab(self):
logging.info('creating table {} to store binlog event'.format(self.tbevent))
create_tb_sql ='''
CREATE TABLE IF NOT EXISTS {}(
auto_id INT(10) UNSIGNED NOT NULL AUTO_INCREMENT,
binlog_name VARCHAR(100) NOT NULL COMMENT 'the binlog file path and name',
dml_start_time DATETIME NOT NULL COMMENT 'when to start this transaction ',
dml_end_time DATETIME NOT NULL COMMENT 'when to finish this transaction ',
end_log_pos BIGINT NOT NULL COMMENT 'the log position for finish this transaction',
db_name VARCHAR(100) NOT NULL COMMENT 'which database happened this transaction ',
table_name VARCHAR(200) NOT NULL COMMENT 'which table happened this transaction ',
sqltype INT NOT NULL COMMENT '1 is insert,2 is update,3 is delete',
dml_sql LONGTEXT NULL COMMENT 'what sql excuted',
undo_sql LONGTEXT NULL COMMENT 'rollback sql, this sql used for flashback',
PRIMARY KEY (auto_id),
INDEX sqltype(sqltype),
INDEX dml_start_time (dml_start_time),
INDEX dml_end_time (dml_end_time),
INDEX end_log_pos (end_log_pos),
INDEX db_name (db_name),
INDEX table_name (table_name)
)
COLLATE='utf8_general_ci' ENGINE=InnoDB;
TRUNCATE TABLE {}; '''.format(self.tbevent,self.tbevent)
self.cur.execute(create_tb_sql)
logging.info('created table {} '.format(self.tbevent)) #获取表格的列顺序对应的列名,并处理where set的时候,列与列之间的连接字符串是逗号还是 and
def tbschema(self,dbname,tbname):
self.tbfield_where = []
self.tbfield_set = [] sql_tb='desc {}.{}'.format(self.db_name,self.tb_name) self.on_cur.execute(sql_tb)
tbcol=self.on_cur.fetchall() i = 0
for l in tbcol:
#self.tbfield.append(l['Field'])
if i==0:
self.tbfield_where.append('`'+l['Field']+'`')
self.tbfield_set.append('`'+l['Field']+'`')
i+=1
else:
self.tbfield_where.append('/*where*/ and /*where*/' + '`'+l['Field']+'`')
self.tbfield_set.append( '/*set*/ , /*set*/'+'`'+l['Field']+'`' ) # 一个事务记录一行,若binlog file中的行记录包含 Table_map,则为事务的开始记录
def rowrecord(self,bl_line):
try:
if bl_line.find('Table_map:') != -1:
l = bl_line.index('server')
m = bl_line.index('end_log_pos')
n = bl_line.index('Table_map')
begin_time = bl_line[:l:].rstrip(' ').replace('#', '') self.begin_time = begin_time[0:4] + '-' + begin_time[4:6] + '-' + begin_time[6:]
self.db_name = bl_line[n::].split(' ')[1].replace('`', '').split('.')[0]
self.tb_name = bl_line[n::].split(' ')[1].replace('`', '').split('.')[1] self.tbschema(self.db_name,self.tb_name)
except Exception:
return 'funtion rowrecord error' def dml_tran(self,bl_line):
try: if bl_line.find('Xid =') != -1: l = bl_line.index('server')
m = bl_line.index('end_log_pos')
end_time = bl_line[:l:].rstrip(' ').replace('#', '')
self.end_time = end_time[0:4] + '-' + end_time[4:6] + '-' + end_time[6:]
self.end_pos = int(bl_line[m::].split(' ')[1]) self.undo_sql = self.dml_sql.replace(' INSERT INTO', ';DELETE FROM_su').replace(' UPDATE ',';UPDATE').replace(' DELETE FROM', ';INSERT INTO').replace(';DELETE FROM_su', ';DELETE FROM').replace('WHERE', 'WHERE_marksu').replace('SET', 'WHERE').replace('WHERE_marksu', 'SET').replace('/*set*/ , /*set*/', ' and ').replace('/*where*/ and /*where*/',' , ')
self.dml_sql=self.dml_sql.replace('/*set*/ , /*set*/', ' , ').replace('/*where*/ and /*where*/',' and ') if self.dml_sql.startswith(' INSERT INTO '):
self.sqltype=1
elif self.dml_sql.startswith(' UPDATE '):
self.sqltype=2
elif self.dml_sql.startswith(' DELETE '):
self.sqltype=3 record_sql = ''
undosql_desc = '' #同个事务内部的行记录修改SQL,反序存储
for l in self.undo_sql.splitlines():
if l.startswith(' ;UPDATE') or l.startswith(' ;INSERT') or l.startswith(' ;DELETE'):
undosql_desc = record_sql + undosql_desc
record_sql = ''
record_sql = record_sql + l
else:
record_sql = record_sql + l self.undo_sql = record_sql + undosql_desc
self.undo_sql = self.undo_sql.lstrip()[1:]+';' #处理非空格的空白特殊字符
self.dml_sql = self.esc_code(self.dml_sql)
self.undo_sql = self.esc_code(self.undo_sql) #单独处理 转移字符: \'
self.dml_sql = self.dml_sql.replace("'", "''").replace('\\x27',"''''") # + ';'
self.undo_sql = self.undo_sql.replace("'", "''").replace('\\x27',"''''") # + ';' if len(self.dml_sql)>500000000:
with open('/tmp/flashback_undosql/'+str(self.end_pos)+'.sql', 'w') as w_f:
w_f.write('begin;' + '\n')
w_f.write(self.undo_sql)
w_f.write('commit;' + '\n')
self.dml_sql=''
self.undo_sql='/tmp/flashback_undosql/'+str(self.end_pos)+'.sql'
logging.info("the size of this transaction is more than 500Mb ,the file location : {}".format(self.undo_file)) insert_sql = "INSERT INTO {}(binlog_name,dml_start_time,dml_end_time,end_log_pos,db_name,table_name,sqltype,dml_sql,undo_sql) select '{}','{}','{}','{}','{}','{}',{},'{}','{}'".format(
self.tbevent, self.fpath, self.begin_time, self.end_time, self.end_pos,
self.db_name, self.tb_name, self.sqltype, self.dml_sql, self.undo_sql) self.cur.execute(insert_sql)
self.mysqlconn.commit() self.dml_sql = ''
self.undo_sql = ''
except Exception:
print( 'funtion dml_tran error') def analyse_binlog(self):
try:
sqlcomma=0
self.create_tab() with open(self.fpath,'r') as binlog_file:
logging.info('\033[36mbegining to analyze the binlog file ,this may be take a long time !!!\033[0m')
logging.info('\033[36manalyzing...\033[0m')
for bline in binlog_file:
if bline.find('Table_map:') != -1:
self.rowrecord(bline)
bline=''
elif bline.rstrip()=='### SET':
bline = bline[3:]
sqlcomma=1
elif bline.rstrip()=='### WHERE':
bline = bline[3:]
sqlcomma = 2
elif bline.startswith('### @'):
len_f=len('### @')
i=bline[len_f:].split('=')[0] #处理timestamp类型
if bline[8+len(i):].split(' ')[2] == 'TIMESTAMP(0)':
stop_pos = bline.find(' /* TIMESTAMP(0) meta=')
bline = bline.split('=')[0] + '=from_unixtime(' + bline[:stop_pos].split('=')[1] + ')' #处理负数存储方式
if bline.split('=')[1].startswith('-'):
stop_pos = bline.find(' /* TIMESTAMP(0) meta=')
bline = bline.split('=')[0] + '=' + bline.split('=')[1].split(' ')[0]+'\n' if sqlcomma==1:
bline = self.tbfield_set[int(i) - 1]+bline[(len_f+len(i)):]
elif sqlcomma==2:
bline = self.tbfield_where[int(i) - 1] + bline[(len_f+len(i)):] elif bline.startswith('### DELETE') or bline.startswith('### INSERT') or bline.startswith('### UPDATE'):
bline = bline[3:] elif bline.find('Xid =') != -1:
self.dml_tran(bline)
bline=''
else:
bline = '' if bline.rstrip('\n') != '':
self.dml_sql = self.dml_sql + bline + ' '
except Exception:
return 'function do error' def esc_code(self,sql):
esc={
'\\x07':'\a','\\x08':'\b','\\x0c':'\f','\\x0a':'\n','\\x0d':'\r','\\x09':'\t','\\x0b':'\v','\\x5c':'\\',
#'\\x27':'\'',
'\\x22':'\"','\\x3f':'\?','\\x00':'\0'
} for k,v in esc.items():
sql=sql.replace(k,v)
return sql def binlogdesc(self): countsql='select sqltype , count(*) numbers from {} group by sqltype order by sqltype '.format(self.tbevent)
print(countsql)
self.cur.execute(countsql)
count_row=self.cur.fetchall() update_count=0
insert_couont=0
delete_count=0
for row in count_row:
if row['sqltype']==1:
insert_couont=row['numbers']
elif row['sqltype']==2:
update_count=row['numbers']
elif row['sqltype']==3:
delete_count=row['numbers']
logging.info('\033[1;35mTotal transactions number is {}: {} inserts, {} updates, {} deletes !\033[0m(all number is accurate, the other is approximate value) \033[0m'.format(insert_couont+update_count+delete_count,insert_couont,update_count,delete_count)) def undosql(self,number):
#这里会有几个问题:
#1 如果一共有几十万甚至更多的事务操作,那么这个python脚本,极为占用内存,有可能执行错误;
#2 如果单个事务中,涉及修改的行数高达几十万行,其binlog file 达好几G,这里也会有内存损耗问题;
#所以,针对第一点,这里考虑对超多事务进行一个分批执行处理,每个批次处理number个事务,避免一次性把所有事务放到python中;但是第2点,目前暂未处理 tran_num=1
id=0 tran_num_sql="select count(*) table_rows from {}".format(self.tbevent) self.cur.execute(tran_num_sql)
tran_rows=self.cur.fetchall() for num in tran_rows:
tran_num=num['table_rows'] logging.info('\033[32mThere has {} transactions ,need {} batchs ,each batche doing {} transactions \033[0m'.format(tran_num,int(tran_num/number)+1,number)) while id<=tran_num:
logging.info('doing batch : {} '.format(int(id/number)+1))
undo_sql='select auto_id,undo_sql from {} where auto_id > {} and auto_id <= {} order by auto_id desc;'.format(self.tbevent,tran_num-(id+number),tran_num-id)
self.cur.execute(undo_sql) undo_rows=self.cur.fetchall()
f_sql='' for u_row in undo_rows:
try:
self.on_cur.execute(u_row['undo_sql'])
self.on_mysqlconn.commit()
except Exception:
print('auto_id:',u_row['auto_id'])
id+=number def undo_file(self,number):
# 也可以选择私用undo_file将undo_sql导入到文件中,然后再source tran_num=1
id=0 tran_num_sql="select count(*) table_rows from {}".format(self.tbevent) self.cur.execute(tran_num_sql)
tran_rows=self.cur.fetchall() for num in tran_rows:
tran_num=num['table_rows'] logging.info('copy undo_sql to undo file on : /tmp/flashback_undosql/undo_file_flashback.sql')
logging.info('\033[32mThere has {} transactions ,need {} batchs to copy ,each batche doing {} transactions \033[0m'.format(tran_num,int(tran_num/number)+1,number)) with open('/tmp/flashback_undosql/undo_file_flashback.sql', 'w') as w_f:
while id<=tran_num:
logging.info('doing batch : {} '.format(int(id/number)+1))
undo_sql='select auto_id,undo_sql from {} where auto_id > {} and auto_id <= {} order by auto_id desc;'.format(self.tbevent,tran_num-(id+number),tran_num-id)
self.cur.execute(undo_sql) undo_rows=self.cur.fetchall()
for u_row in undo_rows:
try:
w_f.write('begin;' + '\n')
w_f.write('# auto_id'+str(u_row['auto_id']) + '\n')
w_f.write(u_row['undo_sql'] + '\n')
w_f.write('commit;' + '\n')
except Exception:
print('auto_id',u_row['auto_id'])
#time.sleep(2)
id+=number def do(self):
if self.action=='':
self.analyse_binlog()
logging.info('\033[36mfinished to analyze the binlog file !!!\033[0m')
#self.binlogdesc()
elif self.action=='':
self.undosql(10000) def closeconn(self):
self.cur.close()
self.on_cur.close()
logging.info('release all db connections')
logging.info('\033[33mAll done,check the {} which stored binlog event on host {} , port {} \033[0m'.format(self.tbevent,self.host,self.port)) def main():
p = flashback()
p.do()
p.closeconn() if __name__ == "__main__":
main()
mysql基于binlog回滚工具_flashback(python版本)的更多相关文章
- 第二百八十六节,MySQL数据库-MySQL事务操作(回滚)
MySQL数据库-MySQL事务操作(回滚) 事务用于将某些操作的多个SQL作为原子性操作,一旦有某一个出现错误,即可回滚到原来的状态,从而保证数据库数据完整性. 举例:有这样一张表 从表里可以看出张 ...
- MySQL基于binlog主从复制
MySQL复制介绍 默认情况 下复制是异步进行的,从库也不需要一直连接到主库来同步数据 MySQL复制的数据粒度可以是主实例上所有的数据库,也可以是指定的一个或多个数据库 ,也可以是一个数据库里的指定 ...
- [MySQL]mysql binlog回滚数据
1.先开启binlog log-bin = /var/log/mysql/mysql_bin #binlog日志文件,以mysql_bin开头,六个数字结尾的文件:mysql_bin.000001,并 ...
- 【MySQL】MySQL回滚工具
1.mysqlbinlog把事务从binlog中导出 2.从导出的binlog中找到要回滚的事务,去掉第一个DML语句前和最后一个DML语句后与DML无关的binlog信息 3.在目录中新建一个tab ...
- MySQL 存储引擎、锁、调优、失误与事务回滚、与python交互、orm
1.存储引擎(处理表的处理器) 1.基本操作 1.查看所有存储引擎 mysql> show engines; 2.查看已有表的存储引擎 mysql> show create table 表 ...
- MySQL利用binlog恢复误操作数据(python脚本)
在人工手动进行一些数据库写操作的时候(比方说数据订正),尤其是一些不可控的批量更新或删除,通常都建议备份后操作.不过不怕万一,就怕一万,有备无患总是好的.在线上或者测试环境误操作导致数据被删除或者更新 ...
- 记一次腾讯云MySQL数据库数据回滚
如题,因为操作人员的问题,需要对数据库数据进行回滚. 可以看到,设置了7天自动备份,且是物理冷备. 什么是物理冷备?科普一下: (1)热备:在数据库运行时,直接进行备份,对运行的数据库没有影响.(2) ...
- MySql事务无法回滚的原因
使用MySQL时.假设发现事务无法回滚,但Hibernate.Spring.JDBC等配置又没有明显问题时.不要苦恼,先看看MySQL创建的表有没有问题.即表的类型. InnoDB和MyISAM是在使 ...
- 【MySQL】 DB 回滚崩溃案例一则
一台mysql数据库在多次断电之后,长时间等待未果直接重启mysql 服务.之后错误日志中报错: 170119 :: InnoDB: Error: page log sequence number I ...
随机推荐
- 纯HTML课表
table标签构造课表 table标签常用于制作表格以及简单布局,于是我就玩了下table标签,用table标签也能构造出很漂亮的页面呢,虽然在博客页面加入与实际打开页面稍微有点出入,但还是可以接受的 ...
- 蓝桥杯-搭积木-java
/* (程序头部注释开始) * 程序的版权和版本声明部分 * Copyright (c) 2016, 广州科技贸易职业学院信息工程系学生 * All rights reserved. * 文件名称: ...
- godot新手中文系列教程1-打包安卓
国内godot qq群 302924317 我也是个新手,可以进群多多交流. 我想要吐槽一下,官方文档有点陈旧,细节缺乏,所以某些时候不要相信官方文档,可以向我们可爱的群友提问,他们一定很乐意回答. ...
- RocketMQ 介绍与基本使用
介绍 RocketMQ是阿里巴巴自研的第三代分布式消息中间件,是阿里系下开源的一款分布式.队列模型的消息中间件,原名Metaq,3.0 版本名称改为RocketMQ,是阿里参照kafka设计思想使用J ...
- SDN学习之OpenFlow协议分析
学习SDN相关的学习也已经有快半年了,期间从一无所知到懵懵懂懂,再到现在的有所熟悉,经历了许多,也走了不少弯路,其中,最为忌讳的便是,我在学习过程中,尚未搞明白OpenFlow协议的情况下,便开始对S ...
- LeetCode:36. Valid Sudoku,数独是否有效
LeetCode:36. Valid Sudoku,数独是否有效 : 题目: LeetCode:36. Valid Sudoku 描述: Determine if a Sudoku is valid, ...
- 去掉iframe默认滚动条后影响正常滚动以及js解决高度自适应。
对于iframe,相信大家都是知道存在很多弊端,比如说不利于搜索引擎的抓取:产生冗余结构体系不易管理等.不过在漫长的开发路上有时候使用是不可避免的. 前两天在做一个退弹的功能的时候使用了iframe, ...
- ar1220f-s四条拨号光纤做的策略路由实现负载均衡
acl number 3001 //内网数据流不需被重定向到外网出口. rule 5 permit ip source 192.168.0.0 0.0.255.255 destination 19 ...
- centos修改无法用用户名和密码登录
vi /etc/ssh/sshd_configPermitRootLogin这行改为PermitRootLogin yesPasswordAuthentication no上面的no改为yesUseP ...
- FiddlerScript高级技巧---自定义Fiddler菜单
Tips 书接上回, Fiddler插件 在团队内部试用后,效果很不错,小伙伴们也提出了很多改进的建议: 最近一段Fiddler使用的仍较为频繁,以前碰到一些特殊测试需求时,总是自己在FiddlerS ...