-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathmysql-slave-skip-one-table
executable file
·210 lines (176 loc) · 6.79 KB
/
mysql-slave-skip-one-table
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
#!/usr/bin/env python3
# mysql-slave-skip-one-table (part of ossobv/vcutil) // wdoekes/2023
# // Public Domain
#
# This mysql-slave-skip-one-table script is a stop gap to force
# old-fashioned MySQL replication to continue in the face of errors like:
#
# Last_SQL_Error: Could not execute Delete_rows_v1 event on table
# mydatabase.mytable; Can't find record in 'mytable', Error_code: 1032;
# handler error HA_ERR_KEY_NOT_FOUND; the event's master log
# mariadb-bin.009713, end_log_pos 22310985
#
# If you're confident you can skip these, you can run this script.
#
# It will:
# - check if SQL slaving is stopped because of an INSERT/UPDATE/DELETE
# constraint on the table (duplicate key or missing record);
# - if so, it will "SET GLOBAL sql_slave_skip_one = 1" + "START SLAVE";
# - this will loop until you stop the script.
#
# It will NOT:
# - make things in sync again.
#
# But if you're lucky you can get the slaving of the rest of the DB in
# sync. And in the mean time you figure out how to fix this one table.
#
# Usage:
#
# mysql-slave-skip-one-table /etc/default/my.cnf client DATABASE TABLE
#
# See also:
#
# mysql-slave-sync-table
#
from configparser import ConfigParser
from datetime import datetime
from fnmatch import fnmatch
from pprint import pprint
from time import sleep
from MySQLdb import connect
def now():
return datetime.now().strftime('%Y-%m-%dT%H:%M:%S')
def read_my_cnf(conffile, section):
# Read mysql config file:
# [section]
# host=host
# user=user
# password=password
confparser = ConfigParser(interpolation=None)
assert confparser.read([conffile]) == [conffile], (
f'Error reading {conffile!r}')
assert section in confparser.sections(), confparser.sections()
return [
confparser[section][i] for i in ('host', 'user', 'password')]
def get_slave_status(cursor):
cursor.execute('SHOW SLAVE STATUS')
columns = [column[0] for column in cursor.description]
res = cursor.fetchall()
assert len(res) == 1, res
res = res[0]
ret = {}
for i, c in enumerate(res):
ret[columns[i]] = c
return ret
def slaving_is_up(slaving):
return bool(
slaving['Slave_IO_Running'] == 'Yes' and
slaving['Slave_SQL_Running'] == 'Yes')
def slaving_is_down_because_of_sql_error(slaving):
return bool(
slaving['Slave_IO_Running'] == 'Yes' and
slaving['Slave_SQL_Running'] == 'No' and
slaving['Last_Errno'] == slaving['Last_SQL_Errno'] and
slaving['Last_Error'] == slaving['Last_SQL_Error'])
def make_slaving_is_down_because_of_sql_reason(error_code, error_text):
def _slaving_is_down_because_of_sql_reason(slaving):
if not slaving_is_down_because_of_sql_error(slaving):
return False
if slaving['Last_SQL_Errno'] != error_code:
return False
return fnmatch(slaving['Last_SQL_Error'], error_text)
return _slaving_is_down_because_of_sql_reason
def sql_slave_skip_one(cursor):
cursor.execute('SET GLOBAL sql_slave_skip_counter = 1')
cursor.execute('START SLAVE')
def sql_slave_skip_if_reasons_match(cursor, reasons):
slaving = get_slave_status(cursor)
if slaving_is_up(slaving):
return False
if any(reason(slaving) for reason in reasons):
print(f'{now()}: {slaving["Last_SQL_Error"]} -- skipping..')
sql_slave_skip_one(cursor)
return True
print(f'{now()}: slaving is down because of something unhandled...')
pprint(slaving)
exit(1)
def sql_slave_skip_loop(cursor, reasons):
no_skips = 0
while True:
if sql_slave_skip_if_reasons_match(cursor, reasons):
no_skips = 0
else:
no_skips += 1
if no_skips >= 500:
if no_skips == 500:
print(f'{now()}: nothing to skip for a while, starting sleep')
sleep(0.2)
else:
sleep(1)
def sql_slave_skip_start(conffile, section, database, table, binlog_prefix):
# Define valid reasons to skip a single slave statement
reasons = [
make_slaving_is_down_because_of_sql_reason(1062, (
f"Could not execute Write_rows_v1 event on table "
f"{database}.{table}; Duplicate entry '*' "
f"for key '*', Error_code: 1062; handler error "
f"HA_ERR_FOUND_DUPP_KEY; the event's master log "
f"{binlog_prefix}.*, end_log_pos *")),
make_slaving_is_down_because_of_sql_reason(1032, (
f"Could not execute Update_rows_v1 event on table "
f"{database}.{table}; Can't find record in "
f"'{table}', Error_code: 1032; handler error "
f"HA_ERR_KEY_NOT_FOUND; the event's master log "
f"{binlog_prefix}.*, end_log_pos *")),
make_slaving_is_down_because_of_sql_reason(1032, (
f"Could not execute Delete_rows_v1 event on table "
f"{database}.{table}; Can't find record in "
f"'{table}', Error_code: 1032; handler error "
f"HA_ERR_KEY_NOT_FOUND; the event's master log "
f"{binlog_prefix}.*, end_log_pos *")),
]
# Test reasons to skip a single slave statement
if 0:
test_error_string = (
f"Could not execute Delete_rows_v1 event on table "
f"{database}.{table}; Can't find record in "
f"'{table}', Error_code: 1032; handler error "
f"HA_ERR_KEY_NOT_FOUND; the event's master log "
f"{binlog_prefix}.012345, end_log_pos 12345")
print('TEST REASON:', test_error_string)
test_slaving = {
'Last_Errno': 1032,
'Last_Error': test_error_string,
'Last_SQL_Errno': 1032,
'Last_SQL_Error': test_error_string,
'Slave_IO_Running': 'Yes',
'Slave_SQL_Running': 'No',
}
print('TEST RESULT:', [reason(test_slaving) for reason in reasons])
exit()
# Open connection, start infinite loop
host, user, password = read_my_cnf(conffile, section)
conn = connect(host=host, user=user, password=password)
with conn.cursor() as cursor:
sql_slave_skip_loop(cursor, reasons)
if __name__ == '__main__':
import sys
if len(sys.argv) in (5, 6):
kwargs = {
'conffile': sys.argv[1],
'section': sys.argv[2],
'database': sys.argv[3],
'table': sys.argv[4],
'binlog_prefix': 'mariadb-bin',
}
if len(sys.argv) == 6:
kwargs['binlog_prefix'] = sys.argv[5]
sql_slave_skip_start(**kwargs)
else:
print(
f'Usage: {sys.argv[0]} /etc/mysql/debian.cnf client '
f'DATABASE TABLE [mariadb-bin]\n'
f' client = the [client] section in /etc/mysql/debian.cnf\n'
f' mariadb-bin = the binlog filename prefix\n',
end='', file=sys.stderr)
exit(1)