Browse Source
BUG#37148 Most callers of mysql_bin_log.write ignore the return result
BUG#37148 Most callers of mysql_bin_log.write ignore the return result
This is the non-ndb part of the patch. The return value of mysql_bin_log.write was ignored by most callers, which may lead to inconsistent on master and slave if the transaction was committed while the binlog was not correctly written. If my_error() is call in mysql_bin_log.write, this could also lead to assertion issue if my_ok() or my_error() is called after. This fixed the problem by let the caller to check and handle the return value of mysql_bin_log.write. This patch only adresses the simple cases. mysql-test/include/binlog_inject_error.inc: inject binlog write error when doing a query mysql-test/suite/binlog/t/binlog_write_error.test: Simple test case to check if proper error is reported when injecting binlog write errors. sql/events.cc: check return value of mysql_bin_log.write sql/log.cc: check return value of mysql_bin_log.write sql/log_event.cc: check return value of mysql_bin_log.write sql/log_event_old.cc: check return value of mysql_bin_log.write sql/mysql_priv.h: Change write_bin_log to return int instead of void sql/rpl_injector.cc: check return value of writing binlog sql/sp.cc: check return value of writing binlog sql/sp_head.cc: return 1 if writing binlog failed sql/sql_acl.cc: check return value of writing binlog sql/sql_base.cc: check return value of writing binlog sql/sql_class.h: Change binlog_show_create_table to return int sql/sql_db.cc: Change write_to_binlog to return int check return value of writing binlog sql/sql_delete.cc: check return value of writing binlog sql/sql_insert.cc: check return value of writing binlog sql/sql_load.cc: check return value of writing binlog sql/sql_parse.cc: check return value of writing binlog sql/sql_partition.cc: check return value of writing binlog sql/sql_rename.cc: check return value of writing binlog sql/sql_repl.cc: check return value of writing binlog sql/sql_table.cc: Change write_bin_log to return int, and return 1 if there was error writing binlog sql/sql_tablespace.cc: check return value of writing binlog sql/sql_trigger.cc: check return value of writing binlog sql/sql_udf.cc: check return value of writing binlog sql/sql_update.cc: check return value of writing binlog sql/sql_view.cc: check return value of writing binlogpull/374/head
28 changed files with 471 additions and 159 deletions
-
22mysql-test/include/binlog_inject_error.inc
-
108mysql-test/suite/binlog/r/binlog_write_error.result
-
101mysql-test/suite/binlog/t/binlog_write_error.test
-
6sql/events.cc
-
4sql/log.cc
-
6sql/log_event.cc
-
15sql/log_event_old.cc
-
4sql/mysql_priv.h
-
41sql/rpl_injector.cc
-
13sql/sp.cc
-
1sql/sp_head.cc
-
25sql/sql_acl.cc
-
18sql/sql_base.cc
-
2sql/sql_class.h
-
35sql/sql_db.cc
-
12sql/sql_delete.cc
-
51sql/sql_insert.cc
-
32sql/sql_load.cc
-
17sql/sql_parse.cc
-
5sql/sql_partition.cc
-
9sql/sql_rename.cc
-
13sql/sql_repl.cc
-
61sql/sql_table.cc
-
4sql/sql_tablespace.cc
-
2sql/sql_trigger.cc
-
8sql/sql_udf.cc
-
7sql/sql_update.cc
-
8sql/sql_view.cc
@ -0,0 +1,22 @@ |
|||
# |
|||
# === Name |
|||
# |
|||
# binlog_inject_error.inc |
|||
# |
|||
# === Description |
|||
# |
|||
# Inject binlog write error when running the query, verifies that the |
|||
# query is ended with the proper error (ER_ERROR_ON_WRITE). |
|||
# |
|||
# === Usage |
|||
# |
|||
# let query= 'CREATE TABLE t1 (a INT)'; |
|||
# source include/binlog_inject_error.inc; |
|||
# |
|||
|
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
--echo $query; |
|||
--replace_regex /(errno: .*)/(errno: #)/ |
|||
--error ER_ERROR_ON_WRITE |
|||
--eval $query |
|||
SET GLOBAL debug=''; |
@ -0,0 +1,108 @@ |
|||
# |
|||
# Initialization |
|||
# |
|||
DROP TABLE IF EXISTS t1, t2; |
|||
DROP FUNCTION IF EXISTS f1; |
|||
DROP FUNCTION IF EXISTS f2; |
|||
DROP PROCEDURE IF EXISTS p1; |
|||
DROP PROCEDURE IF EXISTS p2; |
|||
DROP TRIGGER IF EXISTS tr1; |
|||
DROP TRIGGER IF EXISTS tr2; |
|||
DROP VIEW IF EXISTS v1, v2; |
|||
# |
|||
# Test injecting binlog write error when executing queries |
|||
# |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
CREATE TABLE t1 (a INT); |
|||
CREATE TABLE t1 (a INT); |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
INSERT INTO t1 VALUES (1),(2),(3); |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
INSERT INTO t1 VALUES (4),(5),(6); |
|||
INSERT INTO t1 VALUES (4),(5),(6); |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
UPDATE t1 set a=a+1; |
|||
UPDATE t1 set a=a+1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
DELETE FROM t1; |
|||
DELETE FROM t1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
CREATE TRIGGER tr1 AFTER INSERT ON t1 FOR EACH ROW INSERT INTO t1 VALUES (new.a + 100); |
|||
CREATE TRIGGER tr1 AFTER INSERT ON t1 FOR EACH ROW INSERT INTO t1 VALUES (new.a + 100); |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
DROP TRIGGER tr1; |
|||
DROP TRIGGER tr1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
ALTER TABLE t1 ADD (b INT); |
|||
ALTER TABLE t1 ADD (b INT); |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
CREATE VIEW v1 AS SELECT a FROM t1; |
|||
CREATE VIEW v1 AS SELECT a FROM t1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
DROP VIEW v1; |
|||
DROP VIEW v1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
CREATE PROCEDURE p1(OUT rows INT) SELECT count(*) INTO rows FROM t1; |
|||
CREATE PROCEDURE p1(OUT rows INT) SELECT count(*) INTO rows FROM t1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
DROP PROCEDURE p1; |
|||
DROP PROCEDURE p1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
DROP TABLE t1; |
|||
DROP TABLE t1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
CREATE FUNCTION f1() RETURNS INT return 1; |
|||
CREATE FUNCTION f1() RETURNS INT return 1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
DROP FUNCTION f1; |
|||
DROP FUNCTION f1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
CREATE USER user1; |
|||
CREATE USER user1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
REVOKE ALL PRIVILEGES, GRANT OPTION FROM user1; |
|||
REVOKE ALL PRIVILEGES, GRANT OPTION FROM user1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
SET GLOBAL debug='d,injecting_fault_writing'; |
|||
DROP USER user1; |
|||
DROP USER user1; |
|||
ERROR HY000: Error writing file 'master-bin' ((errno: #) |
|||
SET GLOBAL debug=''; |
|||
# |
|||
# Cleanup |
|||
# |
|||
DROP TABLE IF EXISTS t1, t2; |
|||
DROP FUNCTION IF EXISTS f1; |
|||
DROP PROCEDURE IF EXISTS p1; |
|||
DROP TRIGGER IF EXISTS tr1; |
|||
DROP VIEW IF EXISTS v1, v2; |
@ -0,0 +1,101 @@ |
|||
# |
|||
# === Name === |
|||
# |
|||
# binlog_write_error.test |
|||
# |
|||
# === Description === |
|||
# |
|||
# This test case check if the error of writing binlog file is properly |
|||
# reported and handled when executing statements. |
|||
# |
|||
# === Related Bugs === |
|||
# |
|||
# BUG#37148 |
|||
# |
|||
|
|||
source include/have_log_bin.inc; |
|||
source include/have_debug.inc; |
|||
|
|||
--echo # |
|||
--echo # Initialization |
|||
--echo # |
|||
|
|||
disable_warnings; |
|||
DROP TABLE IF EXISTS t1, t2; |
|||
DROP FUNCTION IF EXISTS f1; |
|||
DROP FUNCTION IF EXISTS f2; |
|||
DROP PROCEDURE IF EXISTS p1; |
|||
DROP PROCEDURE IF EXISTS p2; |
|||
DROP TRIGGER IF EXISTS tr1; |
|||
DROP TRIGGER IF EXISTS tr2; |
|||
DROP VIEW IF EXISTS v1, v2; |
|||
enable_warnings; |
|||
|
|||
--echo # |
|||
--echo # Test injecting binlog write error when executing queries |
|||
--echo # |
|||
|
|||
let $query= CREATE TABLE t1 (a INT); |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
INSERT INTO t1 VALUES (1),(2),(3); |
|||
|
|||
let $query= INSERT INTO t1 VALUES (4),(5),(6); |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= UPDATE t1 set a=a+1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= DELETE FROM t1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= CREATE TRIGGER tr1 AFTER INSERT ON t1 FOR EACH ROW INSERT INTO t1 VALUES (new.a + 100); |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= DROP TRIGGER tr1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= ALTER TABLE t1 ADD (b INT); |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= CREATE VIEW v1 AS SELECT a FROM t1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= DROP VIEW v1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= CREATE PROCEDURE p1(OUT rows INT) SELECT count(*) INTO rows FROM t1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= DROP PROCEDURE p1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= DROP TABLE t1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= CREATE FUNCTION f1() RETURNS INT return 1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= DROP FUNCTION f1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= CREATE USER user1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= REVOKE ALL PRIVILEGES, GRANT OPTION FROM user1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
let $query= DROP USER user1; |
|||
source include/binlog_inject_error.inc; |
|||
|
|||
--echo # |
|||
--echo # Cleanup |
|||
--echo # |
|||
|
|||
disable_warnings; |
|||
DROP TABLE IF EXISTS t1, t2; |
|||
DROP FUNCTION IF EXISTS f1; |
|||
DROP PROCEDURE IF EXISTS p1; |
|||
DROP TRIGGER IF EXISTS tr1; |
|||
DROP VIEW IF EXISTS v1, v2; |
|||
enable_warnings; |
Write
Preview
Loading…
Cancel
Save
Reference in new issue