Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
M
mariadb
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
mariadb
Commits
08bb33db
Commit
08bb33db
authored
Mar 22, 2007
by
mats@romeo.(none)
Browse files
Options
Browse Files
Download
Plain Diff
Merge romeo.(none):/home/bkroot/mysql-5.1-new-rpl
into romeo.(none):/home/bk/b22583-mysql-5.1-new-rpl
parents
b1f69b48
070c147b
Changes
16
Show whitespace changes
Inline
Side-by-side
Showing
16 changed files
with
1008 additions
and
403 deletions
+1008
-403
mysql-test/r/rpl_misc_functions.result
mysql-test/r/rpl_misc_functions.result
+25
-2
mysql-test/r/rpl_ndb_basic.result
mysql-test/r/rpl_ndb_basic.result
+2
-2
mysql-test/r/rpl_row_tabledefs_2myisam.result
mysql-test/r/rpl_row_tabledefs_2myisam.result
+1
-1
mysql-test/r/rpl_row_tabledefs_3innodb.result
mysql-test/r/rpl_row_tabledefs_3innodb.result
+1
-1
mysql-test/t/rpl_misc_functions.test
mysql-test/t/rpl_misc_functions.test
+66
-2
sql/log.cc
sql/log.cc
+14
-4
sql/log_event.cc
sql/log_event.cc
+429
-235
sql/log_event.h
sql/log_event.h
+292
-46
sql/rpl_rli.cc
sql/rpl_rli.cc
+22
-16
sql/rpl_rli.h
sql/rpl_rli.h
+18
-4
sql/rpl_utility.cc
sql/rpl_utility.cc
+2
-1
sql/rpl_utility.h
sql/rpl_utility.h
+1
-1
sql/slave.cc
sql/slave.cc
+111
-83
sql/slave.h
sql/slave.h
+3
-3
sql/sql_binlog.cc
sql/sql_binlog.cc
+10
-2
sql/sql_class.cc
sql/sql_class.cc
+11
-0
No files found.
mysql-test/r/rpl_misc_functions.result
View file @
08bb33db
...
@@ -18,6 +18,29 @@ create table t2 like t1;
...
@@ -18,6 +18,29 @@ create table t2 like t1;
load data local infile 'MYSQLTEST_VARDIR/master-data/test/rpl_misc_functions.outfile' into table t2;
load data local infile 'MYSQLTEST_VARDIR/master-data/test/rpl_misc_functions.outfile' into table t2;
select * from t1, t2 where (t1.id=t2.id) and not(t1.i=t2.i and t1.r1=t2.r1 and t1.r2=t2.r2 and t1.p=t2.p);
select * from t1, t2 where (t1.id=t2.id) and not(t1.i=t2.i and t1.r1=t2.r1 and t1.r2=t2.r2 and t1.p=t2.p);
id i r1 r2 p id i r1 r2 p
id i r1 r2 p id i r1 r2 p
stop slave;
drop table t1;
drop table t1;
drop table t1;
DROP TABLE IF EXISTS t1;
CREATE TABLE t1 (col_a double default NULL);
CREATE PROCEDURE test_replication_sp1()
BEGIN
INSERT INTO t1 VALUES (rand()), (rand());
INSERT INTO t1 VALUES (rand());
END|
CREATE PROCEDURE test_replication_sp2()
BEGIN
CALL test_replication_sp1();
CALL test_replication_sp1();
END|
CREATE FUNCTION test_replication_sf() RETURNS DOUBLE DETERMINISTIC
BEGIN
RETURN (rand() + rand());
END|
CALL test_replication_sp1();
CALL test_replication_sp2();
INSERT INTO t1 VALUES (test_replication_sf());
INSERT INTO t1 VALUES (test_replication_sf());
INSERT INTO t1 VALUES (test_replication_sf());
DROP PROCEDURE IF EXISTS test_replication_sp1;
DROP PROCEDURE IF EXISTS test_replication_sp2;
DROP FUNCTION IF EXISTS test_replication_sf;
DROP TABLE IF EXISTS t1;
mysql-test/r/rpl_ndb_basic.result
View file @
08bb33db
...
@@ -97,8 +97,8 @@ Replicate_Do_Table
...
@@ -97,8 +97,8 @@ Replicate_Do_Table
Replicate_Ignore_Table <Replicate_Ignore_Table>
Replicate_Ignore_Table <Replicate_Ignore_Table>
Replicate_Wild_Do_Table
Replicate_Wild_Do_Table
Replicate_Wild_Ignore_Table
Replicate_Wild_Ignore_Table
Last_Errno 1
46
Last_Errno 1
105
Last_Error
Error in Write_rows event: error during transaction execution on table test.t1
Last_Error
Unknown error
Skip_Counter 0
Skip_Counter 0
Exec_Master_Log_Pos <Exec_Master_Log_Pos>
Exec_Master_Log_Pos <Exec_Master_Log_Pos>
Relay_Log_Space <Relay_Log_Space>
Relay_Log_Space <Relay_Log_Space>
...
...
mysql-test/r/rpl_row_tabledefs_2myisam.result
View file @
08bb33db
...
@@ -122,7 +122,7 @@ Replicate_Do_Table
...
@@ -122,7 +122,7 @@ Replicate_Do_Table
Replicate_Ignore_Table
Replicate_Ignore_Table
Replicate_Wild_Do_Table
Replicate_Wild_Do_Table
Replicate_Wild_Ignore_Table
Replicate_Wild_Ignore_Table
Last_Errno 1
364
Last_Errno 1
105
Last_Error Error in Write_rows event: error during transaction execution on table test.t1_nodef
Last_Error Error in Write_rows event: error during transaction execution on table test.t1_nodef
Skip_Counter 0
Skip_Counter 0
Exec_Master_Log_Pos #
Exec_Master_Log_Pos #
...
...
mysql-test/r/rpl_row_tabledefs_3innodb.result
View file @
08bb33db
...
@@ -122,7 +122,7 @@ Replicate_Do_Table
...
@@ -122,7 +122,7 @@ Replicate_Do_Table
Replicate_Ignore_Table
Replicate_Ignore_Table
Replicate_Wild_Do_Table
Replicate_Wild_Do_Table
Replicate_Wild_Ignore_Table
Replicate_Wild_Ignore_Table
Last_Errno 1
364
Last_Errno 1
105
Last_Error Error in Write_rows event: error during transaction execution on table test.t1_nodef
Last_Error Error in Write_rows event: error during transaction execution on table test.t1_nodef
Skip_Counter 0
Skip_Counter 0
Exec_Master_Log_Pos #
Exec_Master_Log_Pos #
...
...
mysql-test/t/rpl_misc_functions.test
View file @
08bb33db
...
@@ -28,10 +28,74 @@ create table t2 like t1;
...
@@ -28,10 +28,74 @@ create table t2 like t1;
eval
load
data
local
infile
'$MYSQLTEST_VARDIR/master-data/test/rpl_misc_functions.outfile'
into
table
t2
;
eval
load
data
local
infile
'$MYSQLTEST_VARDIR/master-data/test/rpl_misc_functions.outfile'
into
table
t2
;
# compare them with the replica; the SELECT below should return no row
# compare them with the replica; the SELECT below should return no row
select
*
from
t1
,
t2
where
(
t1
.
id
=
t2
.
id
)
and
not
(
t1
.
i
=
t2
.
i
and
t1
.
r1
=
t2
.
r1
and
t1
.
r2
=
t2
.
r2
and
t1
.
p
=
t2
.
p
);
select
*
from
t1
,
t2
where
(
t1
.
id
=
t2
.
id
)
and
not
(
t1
.
i
=
t2
.
i
and
t1
.
r1
=
t2
.
r1
and
t1
.
r2
=
t2
.
r2
and
t1
.
p
=
t2
.
p
);
stop
slave
;
drop
table
t1
;
connection
master
;
connection
master
;
drop
table
t1
;
drop
table
t1
;
# End of 4.1 tests
# End of 4.1 tests
#
# BUG#25543 test calling rand() multiple times on the master in
# a stored procedure.
#
--
disable_warnings
DROP
TABLE
IF
EXISTS
t1
;
--
enable_warnings
CREATE
TABLE
t1
(
col_a
double
default
NULL
);
DELIMITER
|
;
# Use a SP that calls rand() multiple times
CREATE
PROCEDURE
test_replication_sp1
()
BEGIN
INSERT
INTO
t1
VALUES
(
rand
()),
(
rand
());
INSERT
INTO
t1
VALUES
(
rand
());
END
|
# Use a SP that calls another SP to call rand() multiple times
CREATE
PROCEDURE
test_replication_sp2
()
BEGIN
CALL
test_replication_sp1
();
CALL
test_replication_sp1
();
END
|
# Use a SF that calls rand() multiple times
CREATE
FUNCTION
test_replication_sf
()
RETURNS
DOUBLE
DETERMINISTIC
BEGIN
RETURN
(
rand
()
+
rand
());
END
|
DELIMITER
;
|
# Exercise the functions and procedures then compare the results on
# the master to those on the slave.
CALL
test_replication_sp1
();
CALL
test_replication_sp2
();
INSERT
INTO
t1
VALUES
(
test_replication_sf
());
INSERT
INTO
t1
VALUES
(
test_replication_sf
());
INSERT
INTO
t1
VALUES
(
test_replication_sf
());
# Record the results of the query on the master
--
exec
$MYSQL
--
port
=
$MASTER_MYPORT
test
-
e
"SELECT * FROM test.t1"
>
$MYSQLTEST_VARDIR
/
tmp
/
rpl_rand_master
.
sql
--
sync_slave_with_master
# Record the results of the query on the slave
--
exec
$MYSQL
--
port
=
$SLAVE_MYPORT
test
-
e
"SELECT * FROM test.t1"
>
$MYSQLTEST_VARDIR
/
tmp
/
rpl_rand_slave
.
sql
# Compare the results from the master to the slave.
--
exec
diff
$MYSQLTEST_VARDIR
/
tmp
/
rpl_rand_master
.
sql
$MYSQLTEST_VARDIR
/
tmp
/
rpl_rand_slave
.
sql
# Cleanup
--
disable_warnings
DROP
PROCEDURE
IF
EXISTS
test_replication_sp1
;
DROP
PROCEDURE
IF
EXISTS
test_replication_sp2
;
DROP
FUNCTION
IF
EXISTS
test_replication_sf
;
DROP
TABLE
IF
EXISTS
t1
;
--
enable_warnings
# If all is good, when can cleanup our dump files.
--
system
rm
$MYSQLTEST_VARDIR
/
tmp
/
rpl_rand_master
.
sql
--
system
rm
$MYSQLTEST_VARDIR
/
tmp
/
rpl_rand_slave
.
sql
sql/log.cc
View file @
08bb33db
...
@@ -1548,7 +1548,13 @@ static int binlog_commit(handlerton *hton, THD *thd, bool all)
...
@@ -1548,7 +1548,13 @@ static int binlog_commit(handlerton *hton, THD *thd, bool all)
(
binlog_trx_data
*
)
thd
->
ha_data
[
binlog_hton
->
slot
];
(
binlog_trx_data
*
)
thd
->
ha_data
[
binlog_hton
->
slot
];
DBUG_ASSERT
(
mysql_bin_log
.
is_open
());
DBUG_ASSERT
(
mysql_bin_log
.
is_open
());
if
(
all
&&
trx_data
->
empty
())
/*
The condition here has to be identical to the one inside
binlog_end_trans(), guarding the write of the transaction cache to
the binary log.
*/
if
((
all
||
!
(
thd
->
options
&
(
OPTION_BEGIN
|
OPTION_NOT_AUTOCOMMIT
)))
&&
trx_data
->
empty
())
{
{
// we're here because trans_log was flushed in MYSQL_BIN_LOG::log_xid()
// we're here because trans_log was flushed in MYSQL_BIN_LOG::log_xid()
trx_data
->
reset
();
trx_data
->
reset
();
...
@@ -2499,7 +2505,7 @@ bool MYSQL_BIN_LOG::open(const char *log_name,
...
@@ -2499,7 +2505,7 @@ bool MYSQL_BIN_LOG::open(const char *log_name,
/*
/*
Set 'created' to 0, so that in next relay logs this event does not
Set 'created' to 0, so that in next relay logs this event does not
trigger cleaning actions on the slave in
trigger cleaning actions on the slave in
Format_description_log_event::
exec_event
().
Format_description_log_event::
apply_event_impl
().
*/
*/
description_event_for_queue
->
created
=
0
;
description_event_for_queue
->
created
=
0
;
/* Don't set log_pos in event header */
/* Don't set log_pos in event header */
...
@@ -3206,8 +3212,10 @@ void MYSQL_BIN_LOG::new_file_impl(bool need_lock)
...
@@ -3206,8 +3212,10 @@ void MYSQL_BIN_LOG::new_file_impl(bool need_lock)
{
{
tc_log_page_waits
++
;
tc_log_page_waits
++
;
pthread_mutex_lock
(
&
LOCK_prep_xids
);
pthread_mutex_lock
(
&
LOCK_prep_xids
);
while
(
prepared_xids
)
while
(
prepared_xids
)
{
DBUG_PRINT
(
"info"
,
(
"prepared_xids=%lu"
,
prepared_xids
));
pthread_cond_wait
(
&
COND_prep_xids
,
&
LOCK_prep_xids
);
pthread_cond_wait
(
&
COND_prep_xids
,
&
LOCK_prep_xids
);
}
pthread_mutex_unlock
(
&
LOCK_prep_xids
);
pthread_mutex_unlock
(
&
LOCK_prep_xids
);
}
}
...
@@ -5061,8 +5069,10 @@ void TC_LOG_BINLOG::unlog(ulong cookie, my_xid xid)
...
@@ -5061,8 +5069,10 @@ void TC_LOG_BINLOG::unlog(ulong cookie, my_xid xid)
{
{
pthread_mutex_lock
(
&
LOCK_prep_xids
);
pthread_mutex_lock
(
&
LOCK_prep_xids
);
DBUG_ASSERT
(
prepared_xids
>
0
);
DBUG_ASSERT
(
prepared_xids
>
0
);
if
(
--
prepared_xids
==
0
)
if
(
--
prepared_xids
==
0
)
{
DBUG_PRINT
(
"info"
,
(
"prepared_xids=%lu"
,
prepared_xids
));
pthread_cond_signal
(
&
COND_prep_xids
);
pthread_cond_signal
(
&
COND_prep_xids
);
}
pthread_mutex_unlock
(
&
LOCK_prep_xids
);
pthread_mutex_unlock
(
&
LOCK_prep_xids
);
rotate_and_purge
(
0
);
// as ::write() did not rotate
rotate_and_purge
(
0
);
// as ::write() did not rotate
}
}
...
...
sql/log_event.cc
View file @
08bb33db
...
@@ -88,9 +88,10 @@ public:
...
@@ -88,9 +88,10 @@ public:
operator&()
operator&()
DESCRIPTION
DESCRIPTION
Function to return a pointer to the internal, so that the object
can be treated as a IO_CACHE and used with the my_b_* IO_CACHE
Function to return a pointer to the internal cache, so that the
functions
object can be treated as a IO_CACHE and used with the my_b_*
IO_CACHE functions
RETURN VALUE
RETURN VALUE
A pointer to the internal IO_CACHE.
A pointer to the internal IO_CACHE.
...
@@ -531,25 +532,19 @@ Log_event::Log_event(const char* buf,
...
@@ -531,25 +532,19 @@ Log_event::Log_event(const char* buf,
#ifndef MYSQL_CLIENT
#ifndef MYSQL_CLIENT
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
/*
int
Log_event
::
do_update_pos
(
RELAY_LOG_INFO
*
rli
)
Log_event::exec_event()
*/
int
Log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
{
{
DBUG_ENTER
(
"Log_event::exec_event"
);
/*
/*
rli is null when (as far as I (Guilhem) know)
rli is null when (as far as I (Guilhem) know)
the caller is
the caller is
Load_log_event::do_apply_event *and* that one is called from
Load_log_event::exec_event *and* that one is called from
Execute_load_log_event::do_apply_event. In this case, we don't
Execute_load_log_event::exec_event.
do anything here ; Execute_load_log_event::do_apply_event will
In this case, we don't do anything here ;
call Log_event::do_apply_event again later with the proper rli.
Execute_load_log_event::exec_event will call Log_event::exec_event
Strictly speaking, if we were sure that rli is null only in the
again later with the proper rli.
case discussed above, 'if (rli)' is useless here. But as we are
Strictly speaking, if we were sure that rli is null
not 100% sure, keep it for now.
only in the case discussed above, 'if (rli)' is useless here.
But as we are not 100% sure, keep it for now
.
Matz: I don't think we will need this check with this refactoring
.
*/
*/
if
(
rli
)
if
(
rli
)
{
{
...
@@ -585,17 +580,36 @@ int Log_event::exec_event(struct st_relay_log_info* rli)
...
@@ -585,17 +580,36 @@ int Log_event::exec_event(struct st_relay_log_info* rli)
rli
->
inc_group_relay_log_pos
(
log_pos
);
rli
->
inc_group_relay_log_pos
(
log_pos
);
flush_relay_log_info
(
rli
);
flush_relay_log_info
(
rli
);
/*
/*
Note that Rotate_log_event::
exec_event() does not call this
Note that Rotate_log_event::
do_apply_event() does not call
function, so there is no chance that a fake rotate event resets
this function, so there is no chance that a fake rotate event
last_master_timestamp.
resets last_master_timestamp. Note that we update without
Note that we update without mutex (probably ok - except in some ver
y
mutex (probably ok - except in some very rare cases, onl
y
rare cases, only consequence is that value may take some time to
consequence is that value may take some time to display in
display in
Seconds_Behind_Master - not critical).
Seconds_Behind_Master - not critical).
*/
*/
rli
->
last_master_timestamp
=
when
;
rli
->
last_master_timestamp
=
when
;
}
}
}
}
DBUG_RETURN
(
0
);
return
0
;
// Cannot fail currently
}
Log_event
::
enum_skip_reason
Log_event
::
do_shall_skip
(
RELAY_LOG_INFO
*
rli
)
{
DBUG_PRINT
(
"info"
,
(
"ev->server_id=%lu, ::server_id=%lu,"
" rli->replicate_same_server_id=%d,"
" rli->slave_skip_counter=%d"
,
(
ulong
)
server_id
,
(
ulong
)
::
server_id
,
rli
->
replicate_same_server_id
,
rli
->
slave_skip_counter
));
if
(
server_id
==
::
server_id
&&
!
rli
->
replicate_same_server_id
)
return
EVENT_SKIP_IGNORE
;
else
if
(
rli
->
slave_skip_counter
>
0
)
return
EVENT_SKIP_COUNT
;
else
return
EVENT_SKIP_NOT
;
}
}
...
@@ -742,7 +756,7 @@ int Log_event::read_log_event(IO_CACHE* file, String* packet,
...
@@ -742,7 +756,7 @@ int Log_event::read_log_event(IO_CACHE* file, String* packet,
ulong
data_len
;
ulong
data_len
;
int
result
=
0
;
int
result
=
0
;
char
buf
[
LOG_EVENT_MINIMAL_HEADER_LEN
];
char
buf
[
LOG_EVENT_MINIMAL_HEADER_LEN
];
DBUG_ENTER
(
"read_log_event"
);
DBUG_ENTER
(
"
Log_event::
read_log_event"
);
if
(
log_lock
)
if
(
log_lock
)
pthread_mutex_lock
(
log_lock
);
pthread_mutex_lock
(
log_lock
);
...
@@ -817,7 +831,7 @@ Log_event* Log_event::read_log_event(IO_CACHE* file,
...
@@ -817,7 +831,7 @@ Log_event* Log_event::read_log_event(IO_CACHE* file,
const
Format_description_log_event
*
description_event
)
const
Format_description_log_event
*
description_event
)
#endif
#endif
{
{
DBUG_ENTER
(
"Log_event::read_log_event
(IO_CACHE *, Format_description_log_event *
"
);
DBUG_ENTER
(
"Log_event::read_log_event"
);
DBUG_ASSERT
(
description_event
!=
0
);
DBUG_ASSERT
(
description_event
!=
0
);
char
head
[
LOG_EVENT_MINIMAL_HEADER_LEN
];
char
head
[
LOG_EVENT_MINIMAL_HEADER_LEN
];
/*
/*
...
@@ -1888,27 +1902,28 @@ void Query_log_event::print(FILE* file, PRINT_EVENT_INFO* print_event_info)
...
@@ -1888,27 +1902,28 @@ void Query_log_event::print(FILE* file, PRINT_EVENT_INFO* print_event_info)
/*
/*
Query_log_event::
exec
_event()
Query_log_event::
do_apply
_event()
*/
*/
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
int
Query_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Query_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
return
exec
_event
(
rli
,
query
,
q_len
);
return
do_apply
_event
(
rli
,
query
,
q_len
);
}
}
int
Query_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
,
int
Query_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
,
const
char
*
query_arg
,
uint32
q_len_arg
)
const
char
*
query_arg
,
uint32
q_len_arg
)
{
{
LEX_STRING
new_db
;
LEX_STRING
new_db
;
int
expected_error
,
actual_error
=
0
;
int
expected_error
,
actual_error
=
0
;
/*
/*
Colleagues: please never free(thd->catalog) in MySQL. This would lead to
Colleagues: please never free(thd->catalog) in MySQL. This would
bugs as here thd->catalog is a part of an alloced block, not an entire
lead to bugs as here thd->catalog is a part of an alloced block,
alloced block (see Query_log_event::exec_event()). Same for thd->db.
not an entire alloced block (see
Thank you.
Query_log_event::do_apply_event()). Same for thd->db. Thank
you.
*/
*/
thd
->
catalog
=
catalog_len
?
(
char
*
)
catalog
:
(
char
*
)
""
;
thd
->
catalog
=
catalog_len
?
(
char
*
)
catalog
:
(
char
*
)
""
;
new_db
.
length
=
db_len
;
new_db
.
length
=
db_len
;
...
@@ -1927,11 +1942,11 @@ int Query_log_event::exec_event(struct st_relay_log_info* rli,
...
@@ -1927,11 +1942,11 @@ int Query_log_event::exec_event(struct st_relay_log_info* rli,
END of the current log event (COMMIT). We save it in rli so that InnoDB can
END of the current log event (COMMIT). We save it in rli so that InnoDB can
access it.
access it.
*/
*/
rli
->
future_group_master_log_pos
=
log_pos
;
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
future_group_master_log_pos
=
log_pos
;
DBUG_PRINT
(
"info"
,
(
"log_pos: %lu"
,
(
ulong
)
log_pos
));
DBUG_PRINT
(
"info"
,
(
"log_pos: %lu"
,
(
ulong
)
log_pos
));
clear_all_errors
(
thd
,
rli
);
clear_all_errors
(
thd
,
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
);
rli
->
clear_tables_to_lock
();
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
clear_tables_to_lock
();
/*
/*
Note: We do not need to execute reset_one_shot_variables() if this
Note: We do not need to execute reset_one_shot_variables() if this
...
@@ -1940,8 +1955,8 @@ int Query_log_event::exec_event(struct st_relay_log_info* rli,
...
@@ -1940,8 +1955,8 @@ int Query_log_event::exec_event(struct st_relay_log_info* rli,
its companion query. If the SET is ignored because of
its companion query. If the SET is ignored because of
db_ok(), the companion query will also be ignored, and if
db_ok(), the companion query will also be ignored, and if
the companion query is ignored in the db_ok() test of
the companion query is ignored in the db_ok() test of
::
exec_event(), then the companion SET also have so we
::
do_apply_event(), then the companion SET also have so
don't need to reset_one_shot_variables().
we
don't need to reset_one_shot_variables().
*/
*/
if
(
rpl_filter
->
db_ok
(
thd
->
db
))
if
(
rpl_filter
->
db_ok
(
thd
->
db
))
{
{
...
@@ -2057,7 +2072,7 @@ int Query_log_event::exec_event(struct st_relay_log_info* rli,
...
@@ -2057,7 +2072,7 @@ int Query_log_event::exec_event(struct st_relay_log_info* rli,
to check/fix it.
to check/fix it.
*/
*/
if
(
mysql_test_parse_for_slave
(
thd
,
thd
->
query
,
thd
->
query_length
))
if
(
mysql_test_parse_for_slave
(
thd
,
thd
->
query
,
thd
->
query_length
))
clear_all_errors
(
thd
,
rli
);
/* Can ignore query */
clear_all_errors
(
thd
,
const_cast
<
RELAY_LOG_INFO
*>
(
rli
));
/* Can ignore query */
else
else
{
{
slave_print_msg
(
ERROR_LEVEL
,
rli
,
expected_error
,
slave_print_msg
(
ERROR_LEVEL
,
rli
,
expected_error
,
...
@@ -2108,7 +2123,7 @@ Default database: '%s'. Query: '%s'",
...
@@ -2108,7 +2123,7 @@ Default database: '%s'. Query: '%s'",
ignored_error_code
(
actual_error
))
ignored_error_code
(
actual_error
))
{
{
DBUG_PRINT
(
"info"
,(
"error ignored"
));
DBUG_PRINT
(
"info"
,(
"error ignored"
));
clear_all_errors
(
thd
,
rli
);
clear_all_errors
(
thd
,
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
);
}
}
/*
/*
Other cases: mostly we expected no error and get one.
Other cases: mostly we expected no error and get one.
...
@@ -2175,16 +2190,26 @@ end:
...
@@ -2175,16 +2190,26 @@ end:
thd
->
first_successful_insert_id_in_prev_stmt
=
0
;
thd
->
first_successful_insert_id_in_prev_stmt
=
0
;
thd
->
stmt_depends_on_first_successful_insert_id_in_prev_stmt
=
0
;
thd
->
stmt_depends_on_first_successful_insert_id_in_prev_stmt
=
0
;
free_root
(
thd
->
mem_root
,
MYF
(
MY_KEEP_PREALLOC
));
free_root
(
thd
->
mem_root
,
MYF
(
MY_KEEP_PREALLOC
));
return
thd
->
query_error
;
}
int
Query_log_event
::
do_update_pos
(
RELAY_LOG_INFO
*
rli
)
{
/*
/*
If there was an error we stop. Otherwise we increment positions. Note that
Note that we will not increment group* positions if we are just
we will not increment group* positions if we are just after a SET
after a SET ONE_SHOT, because SET ONE_SHOT should not be separated
ONE_SHOT, because SET ONE_SHOT should not be separated from its following
from its following updating query.
updating query.
*/
*/
return
(
thd
->
query_error
?
thd
->
query_error
:
if
(
thd
->
one_shot_set
)
(
thd
->
one_shot_set
?
(
rli
->
inc_event_relay_log_pos
(),
0
)
:
{
Log_event
::
exec_event
(
rli
)));
rli
->
inc_event_relay_log_pos
();
return
0
;
}
else
return
Log_event
::
do_update_pos
(
rli
);
}
}
#endif
#endif
...
@@ -2313,7 +2338,7 @@ bool Start_log_event_v3::write(IO_CACHE* file)
...
@@ -2313,7 +2338,7 @@ bool Start_log_event_v3::write(IO_CACHE* file)
/*
/*
Start_log_event_v3::
exec
_event()
Start_log_event_v3::
do_apply
_event()
The master started
The master started
...
@@ -2332,9 +2357,9 @@ bool Start_log_event_v3::write(IO_CACHE* file)
...
@@ -2332,9 +2357,9 @@ bool Start_log_event_v3::write(IO_CACHE* file)
*/
*/
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
int
Start_log_event_v3
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Start_log_event_v3
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
DBUG_ENTER
(
"Start_log_event_v3::
exec
_event"
);
DBUG_ENTER
(
"Start_log_event_v3::
do_apply
_event"
);
switch
(
binlog_version
)
switch
(
binlog_version
)
{
{
case
3
:
case
3
:
...
@@ -2376,7 +2401,7 @@ int Start_log_event_v3::exec_event(struct st_relay_log_info* rli)
...
@@ -2376,7 +2401,7 @@ int Start_log_event_v3::exec_event(struct st_relay_log_info* rli)
/* this case is impossible */
/* this case is impossible */
DBUG_RETURN
(
1
);
DBUG_RETURN
(
1
);
}
}
DBUG_RETURN
(
Log_event
::
exec_event
(
rli
)
);
DBUG_RETURN
(
0
);
}
}
#endif
/* defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT) */
#endif
/* defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT) */
...
@@ -2567,24 +2592,10 @@ bool Format_description_log_event::write(IO_CACHE* file)
...
@@ -2567,24 +2592,10 @@ bool Format_description_log_event::write(IO_CACHE* file)
}
}
#endif
#endif
/*
SYNOPSIS
Format_description_log_event::exec_event()
IMPLEMENTATION
Save the information which describes the binlog's format, to be able to
read all coming events.
Call Start_log_event_v3::exec_event().
*/
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
int
Format_description_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Format_description_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
DBUG_ENTER
(
"Format_description_log_event::exec_event"
);
DBUG_ENTER
(
"Format_description_log_event::do_apply_event"
);
/* save the information describing this binlog */
delete
rli
->
relay_log
.
description_event_for_exec
;
rli
->
relay_log
.
description_event_for_exec
=
this
;
#ifdef USING_TRANSACTIONS
#ifdef USING_TRANSACTIONS
/*
/*
...
@@ -2606,14 +2617,36 @@ int Format_description_log_event::exec_event(struct st_relay_log_info* rli)
...
@@ -2606,14 +2617,36 @@ int Format_description_log_event::exec_event(struct st_relay_log_info* rli)
"or ROLLBACK in relay log). A probable cause is that "
"or ROLLBACK in relay log). A probable cause is that "
"the master died while writing the transaction to "
"the master died while writing the transaction to "
"its binary log, thus rolled back too."
);
"its binary log, thus rolled back too."
);
rli
->
cleanup_context
(
thd
,
1
);
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
cleanup_context
(
thd
,
1
);
}
}
#endif
#endif
/*
/*
If this event comes from ourselves, there is no cleaning task to perform,
If this event comes from ourselves, there is no cleaning task to
we don't call Start_log_event_v3::exec_event() (this was just to update the
perform, we don't call Start_log_event_v3::do_apply_event()
log's description event).
(this was just to update the log's description event).
*/
if
(
server_id
!=
(
uint32
)
::
server_id
)
{
/*
If the event was not requested by the slave i.e. the master sent
it while the slave asked for a position >4, the event will make
rli->group_master_log_pos advance. Say that the slave asked for
position 1000, and the Format_desc event's end is 96. Then in
the beginning of replication rli->group_master_log_pos will be
0, then 96, then jump to first really asked event (which is
>96). So this is ok.
*/
*/
DBUG_RETURN
(
Start_log_event_v3
::
do_apply_event
(
rli
));
}
DBUG_RETURN
(
0
);
}
int
Format_description_log_event
::
do_update_pos
(
RELAY_LOG_INFO
*
rli
)
{
/* save the information describing this binlog */
delete
rli
->
relay_log
.
description_event_for_exec
;
rli
->
relay_log
.
description_event_for_exec
=
this
;
if
(
server_id
==
(
uint32
)
::
server_id
)
if
(
server_id
==
(
uint32
)
::
server_id
)
{
{
/*
/*
...
@@ -2630,19 +2663,20 @@ int Format_description_log_event::exec_event(struct st_relay_log_info* rli)
...
@@ -2630,19 +2663,20 @@ int Format_description_log_event::exec_event(struct st_relay_log_info* rli)
the Intvar_log_event respectively.
the Intvar_log_event respectively.
*/
*/
rli
->
inc_event_relay_log_pos
();
rli
->
inc_event_relay_log_pos
();
DBUG_RETURN
(
0
)
;
return
0
;
}
}
else
{
return
Log_event
::
do_update_pos
(
rli
);
}
}
/*
Log_event
::
enum_skip_reason
If the event was not requested by the slave i.e. the master sent it while
Format_description_log_event
::
do_shall_skip
(
RELAY_LOG_INFO
*
rli
)
the slave asked for a position >4, the event will make
{
rli->group_master_log_pos advance. Say that the slave asked for position
return
Log_event
::
EVENT_SKIP_NOT
;
1000, and the Format_desc event's end is 96. Then in the beginning of
replication rli->group_master_log_pos will be 0, then 96, then jump to
first really asked event (which is >96). So this is ok.
*/
DBUG_RETURN
(
Start_log_event_v3
::
exec_event
(
rli
));
}
}
#endif
#endif
...
@@ -3156,19 +3190,21 @@ void Load_log_event::set_fields(const char* affected_db,
...
@@ -3156,19 +3190,21 @@ void Load_log_event::set_fields(const char* affected_db,
Does the data loading job when executing a LOAD DATA on the slave
Does the data loading job when executing a LOAD DATA on the slave
SYNOPSIS
SYNOPSIS
Load_log_event::
exec
_event
Load_log_event::
do_apply
_event
net
net
rli
rli
use_rli_only_for_errors - if set to 1, rli is provided to
use_rli_only_for_errors - if set to 1, rli is provided to
Load_log_event::exec_event only for this
Load_log_event::do_apply_event
function to have RPL_LOG_NAME and
only for this function to have
rli->last_slave_error, both being used by
RPL_LOG_NAME and
error reports. rli's position advancing
rli->last_slave_error, both being
is skipped (done by the caller which is
used by error reports. rli's
Execute_load_log_event::exec_event).
position advancing is skipped (done
- if set to 0, rli is provided for full use,
by the caller which is
i.e. for error reports and position
Execute_load_log_event::do_apply_event).
advancing.
- if set to 0, rli is provided for
full use, i.e. for error reports and
position advancing.
DESCRIPTION
DESCRIPTION
Does the data loading job when executing a LOAD DATA on the slave
Does the data loading job when executing a LOAD DATA on the slave
...
@@ -3178,7 +3214,7 @@ void Load_log_event::set_fields(const char* affected_db,
...
@@ -3178,7 +3214,7 @@ void Load_log_event::set_fields(const char* affected_db,
1 Failure
1 Failure
*/
*/
int
Load_log_event
::
exec_event
(
NET
*
net
,
struct
st_relay_log_info
*
rli
,
int
Load_log_event
::
do_apply_event
(
NET
*
net
,
RELAY_LOG_INFO
const
*
rli
,
bool
use_rli_only_for_errors
)
bool
use_rli_only_for_errors
)
{
{
LEX_STRING
new_db
;
LEX_STRING
new_db
;
...
@@ -3188,9 +3224,9 @@ int Load_log_event::exec_event(NET* net, struct st_relay_log_info* rli,
...
@@ -3188,9 +3224,9 @@ int Load_log_event::exec_event(NET* net, struct st_relay_log_info* rli,
DBUG_ASSERT
(
thd
->
query
==
0
);
DBUG_ASSERT
(
thd
->
query
==
0
);
thd
->
query_length
=
0
;
// Should not be needed
thd
->
query_length
=
0
;
// Should not be needed
thd
->
query_error
=
0
;
thd
->
query_error
=
0
;
clear_all_errors
(
thd
,
rli
);
clear_all_errors
(
thd
,
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
);
/* see Query_log_event::
exec
_event() and BUG#13360 */
/* see Query_log_event::
do_apply
_event() and BUG#13360 */
DBUG_ASSERT
(
!
rli
->
m_table_map
.
count
());
DBUG_ASSERT
(
!
rli
->
m_table_map
.
count
());
/*
/*
Usually mysql_init_query() is called by mysql_parse(), but we need it here
Usually mysql_init_query() is called by mysql_parse(), but we need it here
...
@@ -3199,22 +3235,26 @@ int Load_log_event::exec_event(NET* net, struct st_relay_log_info* rli,
...
@@ -3199,22 +3235,26 @@ int Load_log_event::exec_event(NET* net, struct st_relay_log_info* rli,
mysql_init_query
(
thd
,
0
,
0
);
mysql_init_query
(
thd
,
0
,
0
);
if
(
!
use_rli_only_for_errors
)
if
(
!
use_rli_only_for_errors
)
{
{
/* Saved for InnoDB, see comment in Query_log_event::exec_event() */
/*
rli
->
future_group_master_log_pos
=
log_pos
;
Saved for InnoDB, see comment in
Query_log_event::do_apply_event()
*/
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
future_group_master_log_pos
=
log_pos
;
DBUG_PRINT
(
"info"
,
(
"log_pos: %lu"
,
(
ulong
)
log_pos
));
DBUG_PRINT
(
"info"
,
(
"log_pos: %lu"
,
(
ulong
)
log_pos
));
}
}
/*
/*
We test replicate_*_db rules. Note that we have already prepared the file
We test replicate_*_db rules. Note that we have already prepared
to load, even if we are going to ignore and delete it now. So it is
the file to load, even if we are going to ignore and delete it
possible that we did a lot of disk writes for nothing. In other words, a
now. So it is possible that we did a lot of disk writes for
big LOAD DATA INFILE on the master will still consume a lot of space on
nothing. In other words, a big LOAD DATA INFILE on the master will
the slave (space in the relay log + space of temp files: twice the space
still consume a lot of space on the slave (space in the relay log
of the file to load...) even if it will finally be ignored.
+ space of temp files: twice the space of the file to load...)
TODO: fix this; this can be done by testing rules in
even if it will finally be ignored. TODO: fix this; this can be
Create_file_log_event::exec_event() and then discarding Append_block and
done by testing rules in Create_file_log_event::do_apply_event()
al. Another way is do the filtering in the I/O thread (more efficient: no
and then discarding Append_block and al. Another way is do the
disk writes at all).
filtering in the I/O thread (more efficient: no disk writes at
all).
Note: We do not need to execute reset_one_shot_variables() if this
Note: We do not need to execute reset_one_shot_variables() if this
...
@@ -3223,8 +3263,8 @@ int Load_log_event::exec_event(NET* net, struct st_relay_log_info* rli,
...
@@ -3223,8 +3263,8 @@ int Load_log_event::exec_event(NET* net, struct st_relay_log_info* rli,
its companion query. If the SET is ignored because of
its companion query. If the SET is ignored because of
db_ok(), the companion query will also be ignored, and if
db_ok(), the companion query will also be ignored, and if
the companion query is ignored in the db_ok() test of
the companion query is ignored in the db_ok() test of
::
exec_event(), then the companion SET also have so we
::
do_apply_event(), then the companion SET also have so
don't need to reset_one_shot_variables().
we
don't need to reset_one_shot_variables().
*/
*/
if
(
rpl_filter
->
db_ok
(
thd
->
db
))
if
(
rpl_filter
->
db_ok
(
thd
->
db
))
{
{
...
@@ -3420,7 +3460,7 @@ Fatal error running LOAD DATA INFILE on table '%s'. Default database: '%s'",
...
@@ -3420,7 +3460,7 @@ Fatal error running LOAD DATA INFILE on table '%s'. Default database: '%s'",
return
1
;
return
1
;
}
}
return
(
use_rli_only_for_errors
?
0
:
Log_event
::
exec
_event
(
rli
)
);
return
(
use_rli_only_for_errors
?
0
:
Log_event
::
do_apply
_event
(
rli
)
);
}
}
#endif
#endif
...
@@ -3514,6 +3554,7 @@ Rotate_log_event::Rotate_log_event(const char* buf, uint event_len,
...
@@ -3514,6 +3554,7 @@ Rotate_log_event::Rotate_log_event(const char* buf, uint event_len,
ident_offset
=
post_header_len
;
ident_offset
=
post_header_len
;
set_if_smaller
(
ident_len
,
FN_REFLEN
-
1
);
set_if_smaller
(
ident_len
,
FN_REFLEN
-
1
);
new_log_ident
=
my_strndup
(
buf
+
ident_offset
,
(
uint
)
ident_len
,
MYF
(
MY_WME
));
new_log_ident
=
my_strndup
(
buf
+
ident_offset
,
(
uint
)
ident_len
,
MYF
(
MY_WME
));
DBUG_PRINT
(
"debug"
,
(
"new_log_ident: '%s'"
,
new_log_ident
));
DBUG_VOID_RETURN
;
DBUG_VOID_RETURN
;
}
}
...
@@ -3533,8 +3574,20 @@ bool Rotate_log_event::write(IO_CACHE* file)
...
@@ -3533,8 +3574,20 @@ bool Rotate_log_event::write(IO_CACHE* file)
}
}
#endif
#endif
/**
Helper function to detect if the event is inside a group.
*/
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
static
bool
is_in_group
(
THD
*
const
thd
,
RELAY_LOG_INFO
*
const
rli
)
{
return
(
thd
->
options
&
OPTION_BEGIN
)
!=
0
||
(
rli
->
last_event_start_time
>
0
);
}
#endif
/*
/*
Rotate_log_event::
exec
_event()
Rotate_log_event::
do_apply
_event()
Got a rotate log event from the master
Got a rotate log event from the master
...
@@ -3551,34 +3604,49 @@ bool Rotate_log_event::write(IO_CACHE* file)
...
@@ -3551,34 +3604,49 @@ bool Rotate_log_event::write(IO_CACHE* file)
*/
*/
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
int
Rotate_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Rotate_log_event
::
do_update_pos
(
RELAY_LOG_INFO
*
rli
)
{
{
DBUG_ENTER
(
"Rotate_log_event::exec_event"
);
DBUG_ENTER
(
"Rotate_log_event::do_update_pos"
);
#ifndef DBUG_OFF
char
buf
[
32
];
#endif
DBUG_PRINT
(
"info"
,
(
"server_id=%lu; ::server_id=%lu"
,
(
ulong
)
this
->
server_id
,
(
ulong
)
::
server_id
));
DBUG_PRINT
(
"info"
,
(
"new_log_ident: %s"
,
this
->
new_log_ident
));
DBUG_PRINT
(
"info"
,
(
"pos: %s"
,
llstr
(
this
->
pos
,
buf
)));
pthread_mutex_lock
(
&
rli
->
data_lock
);
pthread_mutex_lock
(
&
rli
->
data_lock
);
rli
->
event_relay_log_pos
=
my_b_tell
(
rli
->
cur_log
);
rli
->
event_relay_log_pos
=
my_b_tell
(
rli
->
cur_log
);
/*
/*
If we are in a transaction: the only normal case is when the I/O thread was
If we are in a transaction or in a group: the only normal case is
copying a big transaction, then it was stopped and restarted: we have this
when the I/O thread was copying a big transaction, then it was
in the relay log:
stopped and restarted: we have this in the relay log:
BEGIN
BEGIN
...
...
ROTATE (a fake one)
ROTATE (a fake one)
...
...
COMMIT or ROLLBACK
COMMIT or ROLLBACK
In that case, we don't want to touch the coordinates which correspond to
the beginning of the transaction.
In that case, we don't want to touch the coordinates which
Starting from 5.0.0, there also are some rotates from the slave itself, in
correspond to the beginning of the transaction. Starting from
the relay log.
5.0.0, there also are some rotates from the slave itself, in the
relay log, which shall not change the group positions.
*/
*/
if
(
!
(
thd
->
options
&
OPTION_BEGIN
))
if
((
server_id
!=
::
server_id
||
rli
->
replicate_same_server_id
)
&&
!
is_in_group
(
thd
,
rli
))
{
{
DBUG_PRINT
(
"info"
,
(
"old group_master_log_name: '%s' "
"old group_master_log_pos: %lu"
,
rli
->
group_master_log_name
,
(
ulong
)
rli
->
group_master_log_pos
));
memcpy
(
rli
->
group_master_log_name
,
new_log_ident
,
ident_len
+
1
);
memcpy
(
rli
->
group_master_log_name
,
new_log_ident
,
ident_len
+
1
);
rli
->
notify_group_master_log_name_update
();
rli
->
notify_group_master_log_name_update
();
rli
->
group_master_log_pos
=
pos
;
rli
->
group_master_log_pos
=
pos
;
rli
->
group_relay_log_pos
=
rli
->
event_relay_log_pos
;
rli
->
group_relay_log_pos
=
rli
->
event_relay_log_pos
;
DBUG_PRINT
(
"info"
,
(
"group_master_log_name: '%s' "
DBUG_PRINT
(
"info"
,
(
"
new
group_master_log_name: '%s' "
"group_master_log_pos: %lu"
,
"
new
group_master_log_pos: %lu"
,
rli
->
group_master_log_name
,
rli
->
group_master_log_name
,
(
ulong
)
rli
->
group_master_log_pos
));
(
ulong
)
rli
->
group_master_log_pos
));
/*
/*
...
@@ -3597,8 +3665,27 @@ int Rotate_log_event::exec_event(struct st_relay_log_info* rli)
...
@@ -3597,8 +3665,27 @@ int Rotate_log_event::exec_event(struct st_relay_log_info* rli)
pthread_mutex_unlock
(
&
rli
->
data_lock
);
pthread_mutex_unlock
(
&
rli
->
data_lock
);
pthread_cond_broadcast
(
&
rli
->
data_cond
);
pthread_cond_broadcast
(
&
rli
->
data_cond
);
flush_relay_log_info
(
rli
);
flush_relay_log_info
(
rli
);
DBUG_RETURN
(
0
);
DBUG_RETURN
(
0
);
}
}
Log_event
::
enum_skip_reason
Rotate_log_event
::
do_shall_skip
(
RELAY_LOG_INFO
*
rli
)
{
enum_skip_reason
reason
=
Log_event
::
do_shall_skip
(
rli
);
switch
(
reason
)
{
case
Log_event
:
:
EVENT_SKIP_NOT
:
case
Log_event
:
:
EVENT_SKIP_COUNT
:
return
Log_event
::
EVENT_SKIP_NOT
;
case
Log_event
:
:
EVENT_SKIP_IGNORE
:
return
Log_event
::
EVENT_SKIP_IGNORE
;
}
DBUG_ASSERT
(
0
);
}
#endif
#endif
...
@@ -3705,11 +3792,11 @@ void Intvar_log_event::print(FILE* file, PRINT_EVENT_INFO* print_event_info)
...
@@ -3705,11 +3792,11 @@ void Intvar_log_event::print(FILE* file, PRINT_EVENT_INFO* print_event_info)
/*
/*
Intvar_log_event::
exec
_event()
Intvar_log_event::
do_apply
_event()
*/
*/
#if defined(HAVE_REPLICATION)&& !defined(MYSQL_CLIENT)
#if defined(HAVE_REPLICATION)&& !defined(MYSQL_CLIENT)
int
Intvar_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Intvar_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
switch
(
type
)
{
switch
(
type
)
{
case
LAST_INSERT_ID_EVENT
:
case
LAST_INSERT_ID_EVENT
:
...
@@ -3720,9 +3807,33 @@ int Intvar_log_event::exec_event(struct st_relay_log_info* rli)
...
@@ -3720,9 +3807,33 @@ int Intvar_log_event::exec_event(struct st_relay_log_info* rli)
thd
->
force_one_auto_inc_interval
(
val
);
thd
->
force_one_auto_inc_interval
(
val
);
break
;
break
;
}
}
return
0
;
}
int
Intvar_log_event
::
do_update_pos
(
RELAY_LOG_INFO
*
rli
)
{
rli
->
inc_event_relay_log_pos
();
rli
->
inc_event_relay_log_pos
();
return
0
;
return
0
;
}
}
Log_event
::
enum_skip_reason
Intvar_log_event
::
do_shall_skip
(
RELAY_LOG_INFO
*
rli
)
{
/*
It is a common error to set the slave skip counter to 1 instead of
2 when recovering from an insert which used a auto increment,
rand, or user var. Therefore, if the slave skip counter is 1, we
just say that this event should be skipped by ignoring it, meaning
that we do not change the value of the slave skip counter since it
will be decreased by the following insert event.
*/
if
(
rli
->
slave_skip_counter
==
1
)
return
Log_event
::
EVENT_SKIP_IGNORE
;
else
return
Log_event
::
do_shall_skip
(
rli
);
}
#endif
#endif
...
@@ -3785,13 +3896,37 @@ void Rand_log_event::print(FILE* file, PRINT_EVENT_INFO* print_event_info)
...
@@ -3785,13 +3896,37 @@ void Rand_log_event::print(FILE* file, PRINT_EVENT_INFO* print_event_info)
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
int
Rand_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Rand_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
thd
->
rand
.
seed1
=
(
ulong
)
seed1
;
thd
->
rand
.
seed1
=
(
ulong
)
seed1
;
thd
->
rand
.
seed2
=
(
ulong
)
seed2
;
thd
->
rand
.
seed2
=
(
ulong
)
seed2
;
return
0
;
}
int
Rand_log_event
::
do_update_pos
(
RELAY_LOG_INFO
*
rli
)
{
rli
->
inc_event_relay_log_pos
();
rli
->
inc_event_relay_log_pos
();
return
0
;
return
0
;
}
}
Log_event
::
enum_skip_reason
Rand_log_event
::
do_shall_skip
(
RELAY_LOG_INFO
*
rli
)
{
/*
It is a common error to set the slave skip counter to 1 instead of
2 when recovering from an insert which used a auto increment,
rand, or user var. Therefore, if the slave skip counter is 1, we
just say that this event should be skipped by ignoring it, meaning
that we do not change the value of the slave skip counter since it
will be decreased by the following insert event.
*/
if
(
rli
->
slave_skip_counter
==
1
)
return
Log_event
::
EVENT_SKIP_IGNORE
;
else
return
Log_event
::
do_shall_skip
(
rli
);
}
#endif
/* !MYSQL_CLIENT */
#endif
/* !MYSQL_CLIENT */
...
@@ -3858,12 +3993,12 @@ void Xid_log_event::print(FILE* file, PRINT_EVENT_INFO* print_event_info)
...
@@ -3858,12 +3993,12 @@ void Xid_log_event::print(FILE* file, PRINT_EVENT_INFO* print_event_info)
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
int
Xid_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Xid_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
/* For a slave Xid_log_event is COMMIT */
/* For a slave Xid_log_event is COMMIT */
general_log_print
(
thd
,
COM_QUERY
,
general_log_print
(
thd
,
COM_QUERY
,
"COMMIT /* implicit, from Xid_log_event */"
);
"COMMIT /* implicit, from Xid_log_event */"
);
return
end_trans
(
thd
,
COMMIT
)
||
Log_event
::
exec_event
(
rli
)
;
return
end_trans
(
thd
,
COMMIT
);
}
}
#endif
/* !MYSQL_CLIENT */
#endif
/* !MYSQL_CLIENT */
...
@@ -4141,11 +4276,11 @@ void User_var_log_event::print(FILE* file, PRINT_EVENT_INFO* print_event_info)
...
@@ -4141,11 +4276,11 @@ void User_var_log_event::print(FILE* file, PRINT_EVENT_INFO* print_event_info)
/*
/*
User_var_log_event::
exec
_event()
User_var_log_event::
do_apply
_event()
*/
*/
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
int
User_var_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
User_var_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
Item
*
it
=
0
;
Item
*
it
=
0
;
CHARSET_INFO
*
charset
;
CHARSET_INFO
*
charset
;
...
@@ -4207,9 +4342,31 @@ int User_var_log_event::exec_event(struct st_relay_log_info* rli)
...
@@ -4207,9 +4342,31 @@ int User_var_log_event::exec_event(struct st_relay_log_info* rli)
e
.
update_hash
(
val
,
val_len
,
type
,
charset
,
DERIVATION_IMPLICIT
,
0
);
e
.
update_hash
(
val
,
val_len
,
type
,
charset
,
DERIVATION_IMPLICIT
,
0
);
free_root
(
thd
->
mem_root
,
0
);
free_root
(
thd
->
mem_root
,
0
);
return
0
;
}
int
User_var_log_event
::
do_update_pos
(
RELAY_LOG_INFO
*
rli
)
{
rli
->
inc_event_relay_log_pos
();
rli
->
inc_event_relay_log_pos
();
return
0
;
return
0
;
}
}
Log_event
::
enum_skip_reason
User_var_log_event
::
do_shall_skip
(
RELAY_LOG_INFO
*
rli
)
{
/*
It is a common error to set the slave skip counter to 1 instead
of 2 when recovering from an insert which used a auto increment,
rand, or user var. Therefore, if the slave skip counter is 1, we
just say that this event should be skipped by ignoring it, meaning
that we do not change the value of the slave skip counter since it
will be decreased by the following insert event.
*/
if
(
rli
->
slave_skip_counter
==
1
)
return
Log_event
::
EVENT_SKIP_IGNORE
;
else
return
Log_event
::
do_shall_skip
(
rli
);
}
#endif
/* !MYSQL_CLIENT */
#endif
/* !MYSQL_CLIENT */
...
@@ -4249,7 +4406,7 @@ void Slave_log_event::pack_info(Protocol *protocol)
...
@@ -4249,7 +4406,7 @@ void Slave_log_event::pack_info(Protocol *protocol)
#ifndef MYSQL_CLIENT
#ifndef MYSQL_CLIENT
Slave_log_event
::
Slave_log_event
(
THD
*
thd_arg
,
Slave_log_event
::
Slave_log_event
(
THD
*
thd_arg
,
struct
st_relay_log_info
*
rli
)
RELAY_LOG_INFO
*
rli
)
:
Log_event
(
thd_arg
,
0
,
0
)
,
mem_pool
(
0
),
master_host
(
0
)
:
Log_event
(
thd_arg
,
0
,
0
)
,
mem_pool
(
0
),
master_host
(
0
)
{
{
DBUG_ENTER
(
"Slave_log_event"
);
DBUG_ENTER
(
"Slave_log_event"
);
...
@@ -4359,11 +4516,11 @@ Slave_log_event::Slave_log_event(const char* buf, uint event_len)
...
@@ -4359,11 +4516,11 @@ Slave_log_event::Slave_log_event(const char* buf, uint event_len)
#ifndef MYSQL_CLIENT
#ifndef MYSQL_CLIENT
int
Slave_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Slave_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
if
(
mysql_bin_log
.
is_open
())
if
(
mysql_bin_log
.
is_open
())
mysql_bin_log
.
write
(
this
);
mysql_bin_log
.
write
(
this
);
return
Log_event
::
exec_event
(
rli
)
;
return
0
;
}
}
#endif
/* !MYSQL_CLIENT */
#endif
/* !MYSQL_CLIENT */
...
@@ -4392,21 +4549,21 @@ void Stop_log_event::print(FILE* file, PRINT_EVENT_INFO* print_event_info)
...
@@ -4392,21 +4549,21 @@ void Stop_log_event::print(FILE* file, PRINT_EVENT_INFO* print_event_info)
/*
/*
Stop_log_event::
exec
_event()
Stop_log_event::
do_apply
_event()
The master stopped.
The master stopped.
We used to clean up all temporary tables but
We used to clean up all temporary tables but this is useless as, as the
this is useless as, as the master has shut down properly, it has
master has shut down properly, it has written all DROP TEMPORARY TABLE
written all DROP TEMPORARY TABLE (prepared statements' deletion is
(prepared statements' deletion is TODO only when we binlog prep stmts).
TODO only when we binlog prep stmts). We used to clean up
We used to clean up slave_load_tmpdir, but this is useless as it has been
slave_load_tmpdir, but this is useless as it has been cleared at the
cleared at the end of LOAD DATA INFILE.
end of LOAD DATA INFILE. So we have nothing to do here. The place
So we have nothing to do here.
were we must do this cleaning is in
The place were we must do this cleaning is in Start_log_event_v3::exec_event(),
Start_log_event_v3::do_apply_event(), not here. Because if we come
not here. Because if we come
here, the master was sane.
here, the master was sane.
*/
*/
#ifndef MYSQL_CLIENT
#ifndef MYSQL_CLIENT
int
Stop_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Stop_log_event
::
do_update_pos
(
RELAY_LOG_INFO
*
rli
)
{
{
/*
/*
We do not want to update master_log pos because we get a rotate event
We do not want to update master_log pos because we get a rotate event
...
@@ -4424,6 +4581,7 @@ int Stop_log_event::exec_event(struct st_relay_log_info* rli)
...
@@ -4424,6 +4581,7 @@ int Stop_log_event::exec_event(struct st_relay_log_info* rli)
}
}
return
0
;
return
0
;
}
}
#endif
/* !MYSQL_CLIENT */
#endif
/* !MYSQL_CLIENT */
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
...
@@ -4614,11 +4772,11 @@ void Create_file_log_event::pack_info(Protocol *protocol)
...
@@ -4614,11 +4772,11 @@ void Create_file_log_event::pack_info(Protocol *protocol)
/*
/*
Create_file_log_event::
exec
_event()
Create_file_log_event::
do_apply
_event()
*/
*/
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
int
Create_file_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Create_file_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
char
proc_info
[
17
+
FN_REFLEN
+
10
],
*
fname_buf
;
char
proc_info
[
17
+
FN_REFLEN
+
10
],
*
fname_buf
;
char
*
ext
;
char
*
ext
;
...
@@ -4680,7 +4838,7 @@ err:
...
@@ -4680,7 +4838,7 @@ err:
if
(
fd
>=
0
)
if
(
fd
>=
0
)
my_close
(
fd
,
MYF
(
0
));
my_close
(
fd
,
MYF
(
0
));
thd
->
proc_info
=
0
;
thd
->
proc_info
=
0
;
return
error
?
1
:
Log_event
::
exec_event
(
rli
)
;
return
error
==
0
;
}
}
#endif
/* defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT) */
#endif
/* defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT) */
...
@@ -4788,15 +4946,15 @@ int Append_block_log_event::get_create_or_append() const
...
@@ -4788,15 +4946,15 @@ int Append_block_log_event::get_create_or_append() const
}
}
/*
/*
Append_block_log_event::
exec
_event()
Append_block_log_event::
do_apply
_event()
*/
*/
int
Append_block_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Append_block_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
char
proc_info
[
17
+
FN_REFLEN
+
10
],
*
fname
=
proc_info
+
17
;
char
proc_info
[
17
+
FN_REFLEN
+
10
],
*
fname
=
proc_info
+
17
;
int
fd
;
int
fd
;
int
error
=
1
;
int
error
=
1
;
DBUG_ENTER
(
"Append_block_log_event::
exec
_event"
);
DBUG_ENTER
(
"Append_block_log_event::
do_apply
_event"
);
fname
=
strmov
(
proc_info
,
"Making temp file "
);
fname
=
strmov
(
proc_info
,
"Making temp file "
);
slave_load_file_stem
(
fname
,
file_id
,
server_id
,
".data"
);
slave_load_file_stem
(
fname
,
file_id
,
server_id
,
".data"
);
...
@@ -4835,7 +4993,7 @@ err:
...
@@ -4835,7 +4993,7 @@ err:
if
(
fd
>=
0
)
if
(
fd
>=
0
)
my_close
(
fd
,
MYF
(
0
));
my_close
(
fd
,
MYF
(
0
));
thd
->
proc_info
=
0
;
thd
->
proc_info
=
0
;
DBUG_RETURN
(
error
?
error
:
Log_event
::
exec_event
(
rli
)
);
DBUG_RETURN
(
error
);
}
}
#endif
#endif
...
@@ -4919,18 +5077,18 @@ void Delete_file_log_event::pack_info(Protocol *protocol)
...
@@ -4919,18 +5077,18 @@ void Delete_file_log_event::pack_info(Protocol *protocol)
#endif
#endif
/*
/*
Delete_file_log_event::
exec
_event()
Delete_file_log_event::
do_apply
_event()
*/
*/
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
#if defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT)
int
Delete_file_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Delete_file_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
char
fname
[
FN_REFLEN
+
10
];
char
fname
[
FN_REFLEN
+
10
];
char
*
ext
=
slave_load_file_stem
(
fname
,
file_id
,
server_id
,
".data"
);
char
*
ext
=
slave_load_file_stem
(
fname
,
file_id
,
server_id
,
".data"
);
(
void
)
my_delete
(
fname
,
MYF
(
MY_WME
));
(
void
)
my_delete
(
fname
,
MYF
(
MY_WME
));
strmov
(
ext
,
".info"
);
strmov
(
ext
,
".info"
);
(
void
)
my_delete
(
fname
,
MYF
(
MY_WME
));
(
void
)
my_delete
(
fname
,
MYF
(
MY_WME
));
return
Log_event
::
exec_event
(
rli
)
;
return
0
;
}
}
#endif
/* defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT) */
#endif
/* defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT) */
...
@@ -5016,10 +5174,10 @@ void Execute_load_log_event::pack_info(Protocol *protocol)
...
@@ -5016,10 +5174,10 @@ void Execute_load_log_event::pack_info(Protocol *protocol)
/*
/*
Execute_load_log_event::
exec
_event()
Execute_load_log_event::
do_apply
_event()
*/
*/
int
Execute_load_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
int
Execute_load_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
char
fname
[
FN_REFLEN
+
10
];
char
fname
[
FN_REFLEN
+
10
];
char
*
ext
;
char
*
ext
;
...
@@ -5050,14 +5208,15 @@ int Execute_load_log_event::exec_event(struct st_relay_log_info* rli)
...
@@ -5050,14 +5208,15 @@ int Execute_load_log_event::exec_event(struct st_relay_log_info* rli)
lev
->
thd
=
thd
;
lev
->
thd
=
thd
;
/*
/*
lev->exec_event should use rli only for errors
lev->do_apply_event should use rli only for errors i.e. should
i.e. should not advance rli's position.
not advance rli's position.
lev->exec_event is the place where the table is loaded (it calls
mysql_load()).
lev->do_apply_event is the place where the table is loaded (it
calls mysql_load()).
*/
*/
rli
->
future_group_master_log_pos
=
log_pos
;
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
future_group_master_log_pos
=
log_pos
;
if
(
lev
->
exec
_event
(
0
,
rli
,
1
))
if
(
lev
->
do_apply
_event
(
0
,
rli
,
1
))
{
{
/*
/*
We want to indicate the name of the file that could not be loaded
We want to indicate the name of the file that could not be loaded
...
@@ -5100,7 +5259,7 @@ err:
...
@@ -5100,7 +5259,7 @@ err:
my_close
(
fd
,
MYF
(
0
));
my_close
(
fd
,
MYF
(
0
));
end_io_cache
(
&
file
);
end_io_cache
(
&
file
);
}
}
return
error
?
error
:
Log_event
::
exec_event
(
rli
)
;
return
error
;
}
}
#endif
/* defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT) */
#endif
/* defined(HAVE_REPLICATION) && !defined(MYSQL_CLIENT) */
...
@@ -5268,7 +5427,7 @@ void Execute_load_query_log_event::pack_info(Protocol *protocol)
...
@@ -5268,7 +5427,7 @@ void Execute_load_query_log_event::pack_info(Protocol *protocol)
int
int
Execute_load_query_log_event
::
exec_event
(
struct
st_relay_log_info
*
rli
)
Execute_load_query_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
char
*
p
;
char
*
p
;
char
*
buf
;
char
*
buf
;
...
@@ -5305,7 +5464,7 @@ Execute_load_query_log_event::exec_event(struct st_relay_log_info* rli)
...
@@ -5305,7 +5464,7 @@ Execute_load_query_log_event::exec_event(struct st_relay_log_info* rli)
p
=
strmake
(
p
,
STRING_WITH_LEN
(
" INTO"
));
p
=
strmake
(
p
,
STRING_WITH_LEN
(
" INTO"
));
p
=
strmake
(
p
,
query
+
fn_pos_end
,
q_len
-
fn_pos_end
);
p
=
strmake
(
p
,
query
+
fn_pos_end
,
q_len
-
fn_pos_end
);
error
=
Query_log_event
::
exec
_event
(
rli
,
buf
,
p
-
buf
);
error
=
Query_log_event
::
do_apply
_event
(
rli
,
buf
,
p
-
buf
);
/* Forging file name for deletion in same buffer */
/* Forging file name for deletion in same buffer */
*
fname_end
=
0
;
*
fname_end
=
0
;
...
@@ -5625,7 +5784,7 @@ int Rows_log_event::do_add_row_data(byte *const row_data,
...
@@ -5625,7 +5784,7 @@ int Rows_log_event::do_add_row_data(byte *const row_data,
the master does not have a default value (and isn't nullable)
the master does not have a default value (and isn't nullable)
*/
*/
static
int
static
int
unpack_row
(
RELAY_LOG_INFO
*
rli
,
unpack_row
(
RELAY_LOG_INFO
const
*
rli
,
TABLE
*
table
,
uint
const
colcnt
,
TABLE
*
table
,
uint
const
colcnt
,
char
const
*
const
row_data
,
MY_BITMAP
const
*
cols
,
char
const
*
const
row_data
,
MY_BITMAP
const
*
cols
,
char
const
**
const
row_end
,
ulong
*
const
master_reclength
,
char
const
**
const
row_end
,
ulong
*
const
master_reclength
,
...
@@ -5743,17 +5902,17 @@ unpack_row(RELAY_LOG_INFO *rli,
...
@@ -5743,17 +5902,17 @@ unpack_row(RELAY_LOG_INFO *rli,
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
}
}
int
Rows_log_event
::
exec_event
(
st_relay_log_info
*
rli
)
int
Rows_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
DBUG_ENTER
(
"Rows_log_event::
exec
_event(st_relay_log_info*)"
);
DBUG_ENTER
(
"Rows_log_event::
do_apply
_event(st_relay_log_info*)"
);
int
error
=
0
;
int
error
=
0
;
char
const
*
row_start
=
(
char
const
*
)
m_rows_buf
;
char
const
*
row_start
=
(
char
const
*
)
m_rows_buf
;
/*
/*
If m_table_id == ~0UL, then we have a dummy event that does
If m_table_id == ~0UL, then we have a dummy event that does
not
not contain any data. In that case, we just remove all tables in
contain any data. In that case, we just remove all tables in the
t
he tables_to_lock list, close the thread tables, step the relay
t
ables_to_lock list, close the thread tables, and return with
log position, and return with success.
success. The relay log position will be stepped in
*/
*/
if
(
m_table_id
==
~
0UL
)
if
(
m_table_id
==
~
0UL
)
{
{
...
@@ -5763,16 +5922,16 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -5763,16 +5922,16 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
*/
*/
DBUG_ASSERT
(
get_flags
(
STMT_END_F
));
DBUG_ASSERT
(
get_flags
(
STMT_END_F
));
rli
->
clear_tables_to_lock
();
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
clear_tables_to_lock
();
close_thread_tables
(
thd
);
close_thread_tables
(
thd
);
thd
->
clear_error
();
thd
->
clear_error
();
rli
->
inc_event_relay_log_pos
();
DBUG_RETURN
(
0
);
DBUG_RETURN
(
0
);
}
}
/*
/*
'thd' has been set by exec_relay_log_event(), just before calling
'thd' has been set by exec_relay_log_event(), just before calling
exec_event(). We still check here to prevent future coding errors.
do_apply_event(). We still check here to prevent future coding
errors.
*/
*/
DBUG_ASSERT
(
rli
->
sql_thd
==
thd
);
DBUG_ASSERT
(
rli
->
sql_thd
==
thd
);
...
@@ -5788,8 +5947,9 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -5788,8 +5947,9 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
/*
/*
lock_tables() reads the contents of thd->lex, so they must be
lock_tables() reads the contents of thd->lex, so they must be
initialized. Contrary to in Table_map_log_event::exec_event() we don't
initialized. Contrary to in
call mysql_init_query() as that may reset the binlog format.
Table_map_log_event::do_apply_event() we don't call
mysql_init_query() as that may reset the binlog format.
*/
*/
lex_start
(
thd
,
NULL
,
0
);
lex_start
(
thd
,
NULL
,
0
);
...
@@ -5818,7 +5978,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -5818,7 +5978,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
"Error in %s event: when locking tables"
,
"Error in %s event: when locking tables"
,
get_type_str
());
get_type_str
());
}
}
rli
->
clear_tables_to_lock
();
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
clear_tables_to_lock
();
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
}
}
...
@@ -5839,7 +5999,8 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -5839,7 +5999,8 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
TABLE_LIST
*
tables
=
rli
->
tables_to_lock
;
TABLE_LIST
*
tables
=
rli
->
tables_to_lock
;
close_tables_for_reopen
(
thd
,
&
tables
);
close_tables_for_reopen
(
thd
,
&
tables
);
if
((
error
=
open_tables
(
thd
,
&
tables
,
&
rli
->
tables_to_lock_count
,
0
)))
uint
tables_count
=
rli
->
tables_to_lock_count
;
if
((
error
=
open_tables
(
thd
,
&
tables
,
&
tables_count
,
0
)))
{
{
if
(
thd
->
query_error
||
thd
->
is_fatal_error
)
if
(
thd
->
query_error
||
thd
->
is_fatal_error
)
{
{
...
@@ -5854,7 +6015,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -5854,7 +6015,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
"unexpected success or fatal error"
));
"unexpected success or fatal error"
));
thd
->
query_error
=
1
;
thd
->
query_error
=
1
;
}
}
rli
->
clear_tables_to_lock
();
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
clear_tables_to_lock
();
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
}
}
}
}
...
@@ -5876,7 +6037,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -5876,7 +6037,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
mysql_unlock_tables
(
thd
,
thd
->
lock
);
mysql_unlock_tables
(
thd
,
thd
->
lock
);
thd
->
lock
=
0
;
thd
->
lock
=
0
;
thd
->
query_error
=
1
;
thd
->
query_error
=
1
;
rli
->
clear_tables_to_lock
();
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
clear_tables_to_lock
();
DBUG_RETURN
(
ERR_BAD_TABLE_DEF
);
DBUG_RETURN
(
ERR_BAD_TABLE_DEF
);
}
}
}
}
...
@@ -5898,24 +6059,24 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -5898,24 +6059,24 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
*/
*/
for
(
TABLE_LIST
*
ptr
=
rli
->
tables_to_lock
;
ptr
;
ptr
=
ptr
->
next_global
)
for
(
TABLE_LIST
*
ptr
=
rli
->
tables_to_lock
;
ptr
;
ptr
=
ptr
->
next_global
)
{
{
rli
->
m_table_map
.
set_table
(
ptr
->
table_id
,
ptr
->
table
);
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
m_table_map
.
set_table
(
ptr
->
table_id
,
ptr
->
table
);
}
}
#ifdef HAVE_QUERY_CACHE
#ifdef HAVE_QUERY_CACHE
query_cache
.
invalidate_locked_for_write
(
rli
->
tables_to_lock
);
query_cache
.
invalidate_locked_for_write
(
rli
->
tables_to_lock
);
#endif
#endif
rli
->
clear_tables_to_lock
();
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
clear_tables_to_lock
();
}
}
DBUG_ASSERT
(
rli
->
tables_to_lock
==
NULL
&&
rli
->
tables_to_lock_count
==
0
);
DBUG_ASSERT
(
rli
->
tables_to_lock
==
NULL
&&
rli
->
tables_to_lock_count
==
0
);
TABLE
*
table
=
rli
->
m_table_map
.
get_table
(
m_table_id
);
TABLE
*
table
=
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
m_table_map
.
get_table
(
m_table_id
);
if
(
table
)
if
(
table
)
{
{
/*
/*
table == NULL means that this table should not be replicated
table == NULL means that this table should not be replicated
(this was set up by Table_map_log_event::
exec_event() which
(this was set up by Table_map_log_event::
do_apply_event()
tested replicate-* rules).
which
tested replicate-* rules).
*/
*/
/*
/*
...
@@ -5972,7 +6133,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -5972,7 +6133,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
break
;
break
;
default:
default:
slave_print_msg
(
ERROR_LEVEL
,
rli
,
error
,
slave_print_msg
(
ERROR_LEVEL
,
rli
,
thd
->
net
.
last_errno
,
"Error in %s event: row application failed"
,
"Error in %s event: row application failed"
,
get_type_str
());
get_type_str
());
thd
->
query_error
=
1
;
thd
->
query_error
=
1
;
...
@@ -5982,7 +6143,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -5982,7 +6143,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
row_start
=
row_end
;
row_start
=
row_end
;
}
}
DBUG_EXECUTE_IF
(
"STOP_SLAVE_after_first_Rows_event"
,
DBUG_EXECUTE_IF
(
"STOP_SLAVE_after_first_Rows_event"
,
rli
->
abort_slave
=
1
;);
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
abort_slave
=
1
;);
error
=
do_after_row_operations
(
table
,
error
);
error
=
do_after_row_operations
(
table
,
error
);
if
(
!
cache_stmt
)
if
(
!
cache_stmt
)
{
{
...
@@ -5993,11 +6154,12 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -5993,11 +6154,12 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
if
(
error
)
if
(
error
)
{
/* error has occured during the transaction */
{
/* error has occured during the transaction */
slave_print_msg
(
ERROR_LEVEL
,
rli
,
error
,
slave_print_msg
(
ERROR_LEVEL
,
rli
,
thd
->
net
.
last_errno
,
"Error in %s event: error during transaction execution "
"Error in %s event: error during transaction execution "
"on table %s.%s"
,
"on table %s.%s"
,
get_type_str
(),
table
->
s
->
db
.
str
,
get_type_str
(),
table
->
s
->
db
.
str
,
table
->
s
->
table_name
.
str
);
table
->
s
->
table_name
.
str
);
/*
/*
If one day we honour --skip-slave-errors in row-based replication, and
If one day we honour --skip-slave-errors in row-based replication, and
the error should be skipped, then we would clear mappings, rollback,
the error should be skipped, then we would clear mappings, rollback,
...
@@ -6010,7 +6172,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -6010,7 +6172,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
rollback at the caller along with sbr.
rollback at the caller along with sbr.
*/
*/
thd
->
reset_current_stmt_binlog_row_based
();
thd
->
reset_current_stmt_binlog_row_based
();
rli
->
cleanup_context
(
thd
,
0
);
/* rollback at caller in step with sbr */
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
cleanup_context
(
thd
,
error
);
thd
->
query_error
=
1
;
thd
->
query_error
=
1
;
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
}
}
...
@@ -6054,8 +6216,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -6054,8 +6216,7 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
*/
*/
thd
->
reset_current_stmt_binlog_row_based
();
thd
->
reset_current_stmt_binlog_row_based
();
rli
->
cleanup_context
(
thd
,
0
);
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
cleanup_context
(
thd
,
0
);
rli
->
transaction_end
(
thd
);
if
(
error
==
0
)
if
(
error
==
0
)
{
{
...
@@ -6068,7 +6229,6 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -6068,7 +6229,6 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
do not become visible. We still prefer to wipe them out.
do not become visible. We still prefer to wipe them out.
*/
*/
thd
->
clear_error
();
thd
->
clear_error
();
error
=
Log_event
::
exec_event
(
rli
);
}
}
else
else
slave_print_msg
(
ERROR_LEVEL
,
rli
,
error
,
slave_print_msg
(
ERROR_LEVEL
,
rli
,
error
,
...
@@ -6095,17 +6255,17 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
...
@@ -6095,17 +6255,17 @@ int Rows_log_event::exec_event(st_relay_log_info *rli)
wait (reached end of last relay log and nothing gets appended
wait (reached end of last relay log and nothing gets appended
there), we timeout after one minute, and notify DBA about the
there), we timeout after one minute, and notify DBA about the
problem. When WL#2975 is implemented, just remove the member
problem. When WL#2975 is implemented, just remove the member
st_relay_log_info::
unsafe_to_stop_at
and all its occurences.
st_relay_log_info::
last_event_start_time
and all its occurences.
*/
*/
rli
->
unsafe_to_stop_at
=
time
(
0
);
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
last_event_start_time
=
time
(
0
);
}
}
DBUG_ASSERT
(
error
==
0
);
DBUG_ASSERT
(
error
==
0
);
thd
->
clear_error
();
thd
->
clear_error
();
rli
->
inc_event_relay_log_pos
();
DBUG_RETURN
(
0
);
DBUG_RETURN
(
0
);
}
}
#endif
/* !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION) */
#endif
/* !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION) */
#ifndef MYSQL_CLIENT
#ifndef MYSQL_CLIENT
...
@@ -6285,15 +6445,15 @@ Table_map_log_event::Table_map_log_event(const char *buf, uint event_len,
...
@@ -6285,15 +6445,15 @@ Table_map_log_event::Table_map_log_event(const char *buf, uint event_len,
const
char
*
const
vpart
=
buf
+
common_header_len
+
post_header_len
;
const
char
*
const
vpart
=
buf
+
common_header_len
+
post_header_len
;
/* Extract the length of the various parts from the buffer */
/* Extract the length of the various parts from the buffer */
byte
const
*
const
ptr_dblen
=
(
byte
const
*
)
vpart
+
0
;
byte
const
*
const
ptr_dblen
=
(
byte
const
*
)
vpart
+
0
;
m_dblen
=
*
(
uchar
*
)
ptr_dblen
;
m_dblen
=
*
(
uchar
*
)
ptr_dblen
;
/* Length of database name + counter + terminating null */
/* Length of database name + counter + terminating null */
byte
const
*
const
ptr_tbllen
=
ptr_dblen
+
m_dblen
+
2
;
byte
const
*
const
ptr_tbllen
=
ptr_dblen
+
m_dblen
+
2
;
m_tbllen
=
*
(
uchar
*
)
ptr_tbllen
;
m_tbllen
=
*
(
uchar
*
)
ptr_tbllen
;
/* Length of table name + counter + terminating null */
/* Length of table name + counter + terminating null */
byte
const
*
const
ptr_colcnt
=
ptr_tbllen
+
m_tbllen
+
2
;
byte
const
*
const
ptr_colcnt
=
ptr_tbllen
+
m_tbllen
+
2
;
uchar
*
ptr_after_colcnt
=
(
uchar
*
)
ptr_colcnt
;
uchar
*
ptr_after_colcnt
=
(
uchar
*
)
ptr_colcnt
;
m_colcnt
=
net_field_length
(
&
ptr_after_colcnt
);
m_colcnt
=
net_field_length
(
&
ptr_after_colcnt
);
...
@@ -6338,9 +6498,9 @@ Table_map_log_event::~Table_map_log_event()
...
@@ -6338,9 +6498,9 @@ Table_map_log_event::~Table_map_log_event()
*/
*/
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
int
Table_map_log_event
::
exec_event
(
st_relay_log_info
*
rli
)
int
Table_map_log_event
::
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
{
DBUG_ENTER
(
"Table_map_log_event::
exec
_event(st_relay_log_info*)"
);
DBUG_ENTER
(
"Table_map_log_event::
do_apply
_event(st_relay_log_info*)"
);
DBUG_ASSERT
(
rli
->
sql_thd
==
thd
);
DBUG_ASSERT
(
rli
->
sql_thd
==
thd
);
...
@@ -6463,29 +6623,24 @@ int Table_map_log_event::exec_event(st_relay_log_info *rli)
...
@@ -6463,29 +6623,24 @@ int Table_map_log_event::exec_event(st_relay_log_info *rli)
locked by linking the table into the list of tables to lock.
locked by linking the table into the list of tables to lock.
*/
*/
table_list
->
next_global
=
table_list
->
next_local
=
rli
->
tables_to_lock
;
table_list
->
next_global
=
table_list
->
next_local
=
rli
->
tables_to_lock
;
rli
->
tables_to_lock
=
table_list
;
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
tables_to_lock
=
table_list
;
rli
->
tables_to_lock_count
++
;
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
tables_to_lock_count
++
;
/* 'memory' is freed in clear_tables_to_lock */
/* 'memory' is freed in clear_tables_to_lock */
}
}
/*
We explicitly do not call Log_event::exec_event() here since we do not
want the relay log position to be flushed to disk. The flushing will be
done by the last Rows_log_event that either ends a statement (outside a
transaction) or a transaction.
A table map event can *never* end a transaction or a statement, so we
just step the relay log position.
*/
if
(
likely
(
!
error
))
rli
->
inc_event_relay_log_pos
();
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
err:
err:
my_free
((
gptr
)
memory
,
MYF
(
MY_WME
));
my_free
((
gptr
)
memory
,
MYF
(
MY_WME
));
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
}
}
int
Table_map_log_event
::
do_update_pos
(
RELAY_LOG_INFO
*
rli
)
{
rli
->
inc_event_relay_log_pos
();
return
0
;
}
#endif
/* !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION) */
#endif
/* !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION) */
#ifndef MYSQL_CLIENT
#ifndef MYSQL_CLIENT
...
@@ -6650,7 +6805,7 @@ int Write_rows_log_event::do_after_row_operations(TABLE *table, int error)
...
@@ -6650,7 +6805,7 @@ int Write_rows_log_event::do_after_row_operations(TABLE *table, int error)
return
error
;
return
error
;
}
}
int
Write_rows_log_event
::
do_prepare_row
(
THD
*
thd
,
RELAY_LOG_INFO
*
rli
,
int
Write_rows_log_event
::
do_prepare_row
(
THD
*
thd
,
RELAY_LOG_INFO
const
*
rli
,
TABLE
*
table
,
TABLE
*
table
,
char
const
*
const
row_start
,
char
const
*
const
row_start
,
char
const
**
const
row_end
)
char
const
**
const
row_end
)
...
@@ -6800,6 +6955,33 @@ copy_extra_record_fields(TABLE *table,
...
@@ -6800,6 +6955,33 @@ copy_extra_record_fields(TABLE *table,
DBUG_PRINT((N), ((FRM), buf)); \
DBUG_PRINT((N), ((FRM), buf)); \
} while (0)
} while (0)
/**
Check if an error is a duplicate key error.
This function is used to check if an error code is one of the
duplicate key error, i.e., and error code for which it is sensible
to do a <code>get_dup_key()</code> to retrieve the duplicate key.
@param errcode The error code to check.
@return <code>true</code> if the error code is such that
<code>get_dup_key()</code> will return true, <code>false</code>
otherwise.
*/
bool
is_duplicate_key_error
(
int
errcode
)
{
switch
(
errcode
)
{
case
HA_ERR_FOUND_DUPP_KEY
:
case
HA_ERR_FOUND_DUPP_UNIQUE
:
return
true
;
}
return
false
;
}
/*
/*
Replace the provided record in the database.
Replace the provided record in the database.
...
@@ -6848,7 +7030,7 @@ replace_record(THD *thd, TABLE *table,
...
@@ -6848,7 +7030,7 @@ replace_record(THD *thd, TABLE *table,
if
((
keynum
=
table
->
file
->
get_dup_key
(
error
))
<
0
)
if
((
keynum
=
table
->
file
->
get_dup_key
(
error
))
<
0
)
{
{
/* We failed to retrieve the duplicate key */
/* We failed to retrieve the duplicate key */
DBUG_RETURN
(
HA_ERR_FOUND_DUPP_KEY
);
DBUG_RETURN
(
error
);
}
}
/*
/*
...
@@ -6865,8 +7047,11 @@ replace_record(THD *thd, TABLE *table,
...
@@ -6865,8 +7047,11 @@ replace_record(THD *thd, TABLE *table,
{
{
error
=
table
->
file
->
rnd_pos
(
table
->
record
[
1
],
table
->
file
->
dup_ref
);
error
=
table
->
file
->
rnd_pos
(
table
->
record
[
1
],
table
->
file
->
dup_ref
);
if
(
error
)
if
(
error
)
{
table
->
file
->
print_error
(
error
,
MYF
(
0
));
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
}
}
}
else
else
{
{
if
(
table
->
file
->
extra
(
HA_EXTRA_FLUSH_CACHE
))
if
(
table
->
file
->
extra
(
HA_EXTRA_FLUSH_CACHE
))
...
@@ -6887,8 +7072,11 @@ replace_record(THD *thd, TABLE *table,
...
@@ -6887,8 +7072,11 @@ replace_record(THD *thd, TABLE *table,
table
->
key_info
[
keynum
].
key_length
,
table
->
key_info
[
keynum
].
key_length
,
HA_READ_KEY_EXACT
);
HA_READ_KEY_EXACT
);
if
(
error
)
if
(
error
)
{
table
->
file
->
print_error
(
error
,
MYF
(
0
));
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
}
}
}
/*
/*
Now, table->record[1] should contain the offending row. That
Now, table->record[1] should contain the offending row. That
...
@@ -6920,15 +7108,21 @@ replace_record(THD *thd, TABLE *table,
...
@@ -6920,15 +7108,21 @@ replace_record(THD *thd, TABLE *table,
{
{
error
=
table
->
file
->
ha_update_row
(
table
->
record
[
1
],
error
=
table
->
file
->
ha_update_row
(
table
->
record
[
1
],
table
->
record
[
0
]);
table
->
record
[
0
]);
if
(
error
)
table
->
file
->
print_error
(
error
,
MYF
(
0
));
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
}
}
else
else
{
{
if
((
error
=
table
->
file
->
ha_delete_row
(
table
->
record
[
1
])))
if
((
error
=
table
->
file
->
ha_delete_row
(
table
->
record
[
1
])))
{
table
->
file
->
print_error
(
error
,
MYF
(
0
));
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
}
/* Will retry ha_write_row() with the offending row removed. */
/* Will retry ha_write_row() with the offending row removed. */
}
}
}
}
DBUG_RETURN
(
error
);
DBUG_RETURN
(
error
);
}
}
...
@@ -7327,7 +7521,7 @@ int Delete_rows_log_event::do_after_row_operations(TABLE *table, int error)
...
@@ -7327,7 +7521,7 @@ int Delete_rows_log_event::do_after_row_operations(TABLE *table, int error)
return
error
;
return
error
;
}
}
int
Delete_rows_log_event
::
do_prepare_row
(
THD
*
thd
,
RELAY_LOG_INFO
*
rli
,
int
Delete_rows_log_event
::
do_prepare_row
(
THD
*
thd
,
RELAY_LOG_INFO
const
*
rli
,
TABLE
*
table
,
TABLE
*
table
,
char
const
*
const
row_start
,
char
const
*
const
row_start
,
char
const
**
const
row_end
)
char
const
**
const
row_end
)
...
@@ -7462,7 +7656,7 @@ int Update_rows_log_event::do_after_row_operations(TABLE *table, int error)
...
@@ -7462,7 +7656,7 @@ int Update_rows_log_event::do_after_row_operations(TABLE *table, int error)
return
error
;
return
error
;
}
}
int
Update_rows_log_event
::
do_prepare_row
(
THD
*
thd
,
RELAY_LOG_INFO
*
rli
,
int
Update_rows_log_event
::
do_prepare_row
(
THD
*
thd
,
RELAY_LOG_INFO
const
*
rli
,
TABLE
*
table
,
TABLE
*
table
,
char
const
*
const
row_start
,
char
const
*
const
row_start
,
char
const
**
const
row_end
)
char
const
**
const
row_end
)
...
...
sql/log_event.h
View file @
08bb33db
...
@@ -516,6 +516,7 @@ class THD;
...
@@ -516,6 +516,7 @@ class THD;
class
Format_description_log_event
;
class
Format_description_log_event
;
struct
st_relay_log_info
;
struct
st_relay_log_info
;
typedef
st_relay_log_info
RELAY_LOG_INFO
;
#ifdef MYSQL_CLIENT
#ifdef MYSQL_CLIENT
/*
/*
...
@@ -604,6 +605,33 @@ typedef struct st_print_event_info
...
@@ -604,6 +605,33 @@ typedef struct st_print_event_info
class
Log_event
class
Log_event
{
{
public:
public:
/**
Enumeration of what kinds of skipping (and non-skipping) that can
occur when the slave executes an event.
@see shall_skip
@see do_shall_skip
*/
enum
enum_skip_reason
{
/**
Don't skip event.
*/
EVENT_SKIP_NOT
,
/**
Skip event by ignoring it.
This means that the slave skip counter will not be changed.
*/
EVENT_SKIP_IGNORE
,
/**
Skip event and decrease skip counter.
*/
EVENT_SKIP_COUNT
};
/*
/*
The following type definition is to be used whenever data is placed
The following type definition is to be used whenever data is placed
and manipulated in a common buffer. Use this typedef for buffers
and manipulated in a common buffer. Use this typedef for buffers
...
@@ -685,16 +713,14 @@ public:
...
@@ -685,16 +713,14 @@ public:
static
void
init_show_field_list
(
List
<
Item
>*
field_list
);
static
void
init_show_field_list
(
List
<
Item
>*
field_list
);
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
int
net_send
(
Protocol
*
protocol
,
const
char
*
log_name
,
my_off_t
pos
);
int
net_send
(
Protocol
*
protocol
,
const
char
*
log_name
,
my_off_t
pos
);
/*
/*
pack_info() is used by SHOW BINLOG EVENTS; as print() it prepares and sends
pack_info() is used by SHOW BINLOG EVENTS; as print() it prepares and sends
a string to display to the user, so it resembles print().
a string to display to the user, so it resembles print().
*/
*/
virtual
void
pack_info
(
Protocol
*
protocol
);
virtual
void
pack_info
(
Protocol
*
protocol
);
/*
The SQL slave thread calls exec_event() to execute the event; this is where
the slave's data is modified.
*/
virtual
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
virtual
const
char
*
get_db
()
virtual
const
char
*
get_db
()
{
{
...
@@ -767,6 +793,127 @@ public:
...
@@ -767,6 +793,127 @@ public:
*
description_event
);
*
description_event
);
/* returns the human readable name of the event's type */
/* returns the human readable name of the event's type */
const
char
*
get_type_str
();
const
char
*
get_type_str
();
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
public:
/**
Apply the event to the database.
This function represents the public interface for applying an
event.
@see do_apply_event
*/
int
apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
return
do_apply_event
(
rli
);
}
/**
Update the relay log position.
This function represents the public interface for "stepping over"
the event and will update the relay log information.
@see do_update_pos
*/
int
update_pos
(
RELAY_LOG_INFO
*
rli
)
{
return
do_update_pos
(
rli
);
}
/**
Decide if the event shall be skipped, and the reason for skipping
it.
@see do_shall_skip
*/
enum_skip_reason
shall_skip
(
RELAY_LOG_INFO
*
rli
)
{
return
do_shall_skip
(
rli
);
}
protected:
/**
Primitive to apply an event to the database.
This is where the change to the database is made.
@note The primitive is protected instead of private, since there
is a hierarchy of actions to be performed in some cases.
@see Format_description_log_event::do_apply_event()
@param rli Pointer to relay log info structure
@retval 0 Event applied successfully
@retval errno Error code if event application failed
*/
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
return
0
;
/* Default implementation does nothing */
}
/**
Advance relay log coordinates.
This function is called to advance the relay log coordinates to
just after the event. It is essential that both the relay log
coordinate and the group log position is updated correctly, since
this function is used also for skipping events.
Normally, each implementation of do_update_pos() shall:
- Update the event position to refer to the position just after
the event.
- Update the group log position to refer to the position just
after the event <em>if the event is last in a group</em>
@param rli Pointer to relay log info structure
@retval 0 Coordinates changed successfully
@retval errno Error code if advancing failed (usually just
1). Observe that handler errors are returned by the
do_apply_event() function, and not by this one.
*/
virtual
int
do_update_pos
(
RELAY_LOG_INFO
*
rli
);
/**
Decide if this event shall be skipped or not and the reason for
skipping it.
The default implementation decide that the event shall be skipped
if either:
- the server id of the event is the same as the server id of the
server and <code>rli->replicate_same_server_id</code> is true,
or
- if <code>rli->slave_skip_counter</code> is greater than zero.
@see do_apply_event
@see do_update_pos
@retval Log_event::EVENT_SKIP_NOT
The event shall not be skipped and should be applied.
@retval Log_event::EVENT_SKIP_IGNORE
The event shall be skipped by just ignoring it, i.e., the slave
skip counter shall not be changed. This happends if, for example,
the originating server id of the event is the same as the server
id of the slave.
@retval Log_event::EVENT_SKIP_COUNT
The event shall be skipped because the slave skip counter was
non-zero. The caller shall decrease the counter by one.
*/
virtual
enum_skip_reason
do_shall_skip
(
RELAY_LOG_INFO
*
rli
);
#endif
};
};
/*
/*
...
@@ -807,10 +954,10 @@ public:
...
@@ -807,10 +954,10 @@ public:
uint16
error_code
;
uint16
error_code
;
ulong
thread_id
;
ulong
thread_id
;
/*
/*
For events created by Query_log_event::
exec
_event (and
For events created by Query_log_event::
do_apply
_event (and
Load_log_event::
exec_event()) we need the *original* thread id, to be able
Load_log_event::
do_apply_event()) we need the *original* thread
to log the event with the original (=master's) thread id (fix for
id, to be able to log the event with the original (=master's)
BUG#1686).
thread id (fix for
BUG#1686).
*/
*/
ulong
slave_proxy_id
;
ulong
slave_proxy_id
;
...
@@ -873,9 +1020,6 @@ public:
...
@@ -873,9 +1020,6 @@ public:
const
char
*
get_db
()
{
return
db
;
}
const
char
*
get_db
()
{
return
db
;
}
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
);
int
exec_event
(
struct
st_relay_log_info
*
rli
,
const
char
*
query_arg
,
uint32
q_len_arg
);
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
#else
#else
void
print_query_header
(
IO_CACHE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print_query_header
(
IO_CACHE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
...
@@ -904,6 +1048,16 @@ public:
...
@@ -904,6 +1048,16 @@ public:
*/
*/
virtual
ulong
get_post_header_size_for_derived
()
{
return
0
;
}
virtual
ulong
get_post_header_size_for_derived
()
{
return
0
;
}
/* Writes derived event-specific part of post header. */
/* Writes derived event-specific part of post header. */
public:
/* !!! Public in this patch to allow old usage */
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
virtual
int
do_update_pos
(
RELAY_LOG_INFO
*
rli
);
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
,
const
char
*
query_arg
,
uint32
q_len_arg
);
#endif
/* HAVE_REPLICATION */
};
};
...
@@ -952,9 +1106,8 @@ public:
...
@@ -952,9 +1106,8 @@ public:
uint16
master_port
;
uint16
master_port
;
#ifndef MYSQL_CLIENT
#ifndef MYSQL_CLIENT
Slave_log_event
(
THD
*
thd_arg
,
struct
st_relay_log_info
*
rli
);
Slave_log_event
(
THD
*
thd_arg
,
RELAY_LOG_INFO
*
rli
);
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#else
#else
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
#endif
#endif
...
@@ -967,6 +1120,11 @@ public:
...
@@ -967,6 +1120,11 @@ public:
#ifndef MYSQL_CLIENT
#ifndef MYSQL_CLIENT
bool
write
(
IO_CACHE
*
file
);
bool
write
(
IO_CACHE
*
file
);
#endif
#endif
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
#endif
};
};
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
...
@@ -1036,12 +1194,6 @@ public:
...
@@ -1036,12 +1194,6 @@ public:
const
char
*
get_db
()
{
return
db
;
}
const
char
*
get_db
()
{
return
db
;
}
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
)
{
return
exec_event
(
thd
->
slave_net
,
rli
,
0
);
}
int
exec_event
(
NET
*
net
,
struct
st_relay_log_info
*
rli
,
bool
use_rli_only_for_errors
);
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
#else
#else
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
...
@@ -1073,6 +1225,17 @@ public:
...
@@ -1073,6 +1225,17 @@ public:
+
LOAD_HEADER_LEN
+
LOAD_HEADER_LEN
+
sql_ex
.
data_size
()
+
field_block_len
+
num_fields
);
+
sql_ex
.
data_size
()
+
field_block_len
+
num_fields
);
}
}
public:
/* !!! Public in this patch to allow old usage */
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
)
{
return
do_apply_event
(
thd
->
slave_net
,
rli
,
0
);
}
int
do_apply_event
(
NET
*
net
,
RELAY_LOG_INFO
const
*
rli
,
bool
use_rli_only_for_errors
);
#endif
};
};
extern
char
server_version
[
SERVER_VERSION_LENGTH
];
extern
char
server_version
[
SERVER_VERSION_LENGTH
];
...
@@ -1130,7 +1293,6 @@ public:
...
@@ -1130,7 +1293,6 @@ public:
Start_log_event_v3
();
Start_log_event_v3
();
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
#else
#else
Start_log_event_v3
()
{}
Start_log_event_v3
()
{}
...
@@ -1150,6 +1312,22 @@ public:
...
@@ -1150,6 +1312,22 @@ public:
return
START_V3_HEADER_LEN
;
//no variable-sized part
return
START_V3_HEADER_LEN
;
//no variable-sized part
}
}
virtual
bool
is_artificial_event
()
{
return
artificial_event
;
}
virtual
bool
is_artificial_event
()
{
return
artificial_event
;
}
protected:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
virtual
enum_skip_reason
do_shall_skip
(
RELAY_LOG_INFO
*
)
{
/*
Events from ourself should be skipped, but they should not
decrease the slave skip counter.
*/
if
(
this
->
server_id
==
::
server_id
)
return
Log_event
::
EVENT_SKIP_IGNORE
;
else
return
Log_event
::
EVENT_SKIP_NOT
;
}
#endif
};
};
...
@@ -1175,13 +1353,6 @@ public:
...
@@ -1175,13 +1353,6 @@ public:
uchar
server_version_split
[
3
];
uchar
server_version_split
[
3
];
Format_description_log_event
(
uint8
binlog_ver
,
const
char
*
server_ver
=
0
);
Format_description_log_event
(
uint8
binlog_ver
,
const
char
*
server_ver
=
0
);
#ifndef MYSQL_CLIENT
#ifdef HAVE_REPLICATION
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#endif
/* HAVE_REPLICATION */
#endif
Format_description_log_event
(
const
char
*
buf
,
uint
event_len
,
Format_description_log_event
(
const
char
*
buf
,
uint
event_len
,
const
Format_description_log_event
*
description_event
);
const
Format_description_log_event
*
description_event
);
~
Format_description_log_event
()
{
my_free
((
gptr
)
post_header_len
,
MYF
(
0
));
}
~
Format_description_log_event
()
{
my_free
((
gptr
)
post_header_len
,
MYF
(
0
));
}
...
@@ -1204,7 +1375,15 @@ public:
...
@@ -1204,7 +1375,15 @@ public:
*/
*/
return
FORMAT_DESCRIPTION_HEADER_LEN
;
return
FORMAT_DESCRIPTION_HEADER_LEN
;
}
}
void
calc_server_version_split
();
void
calc_server_version_split
();
protected:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
virtual
int
do_update_pos
(
RELAY_LOG_INFO
*
rli
);
virtual
enum_skip_reason
do_shall_skip
(
RELAY_LOG_INFO
*
rli
);
#endif
};
};
...
@@ -1228,7 +1407,6 @@ public:
...
@@ -1228,7 +1407,6 @@ public:
{}
{}
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
#else
#else
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
...
@@ -1243,6 +1421,13 @@ public:
...
@@ -1243,6 +1421,13 @@ public:
bool
write
(
IO_CACHE
*
file
);
bool
write
(
IO_CACHE
*
file
);
#endif
#endif
bool
is_valid
()
const
{
return
1
;
}
bool
is_valid
()
const
{
return
1
;
}
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
virtual
int
do_update_pos
(
RELAY_LOG_INFO
*
rli
);
virtual
enum_skip_reason
do_shall_skip
(
RELAY_LOG_INFO
*
rli
);
#endif
};
};
...
@@ -1269,7 +1454,6 @@ class Rand_log_event: public Log_event
...
@@ -1269,7 +1454,6 @@ class Rand_log_event: public Log_event
{}
{}
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
#else
#else
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
...
@@ -1283,6 +1467,13 @@ class Rand_log_event: public Log_event
...
@@ -1283,6 +1467,13 @@ class Rand_log_event: public Log_event
bool
write
(
IO_CACHE
*
file
);
bool
write
(
IO_CACHE
*
file
);
#endif
#endif
bool
is_valid
()
const
{
return
1
;
}
bool
is_valid
()
const
{
return
1
;
}
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
virtual
int
do_update_pos
(
RELAY_LOG_INFO
*
rli
);
virtual
enum_skip_reason
do_shall_skip
(
RELAY_LOG_INFO
*
rli
);
#endif
};
};
/*****************************************************************************
/*****************************************************************************
...
@@ -1306,7 +1497,6 @@ class Xid_log_event: public Log_event
...
@@ -1306,7 +1497,6 @@ class Xid_log_event: public Log_event
Xid_log_event
(
THD
*
thd_arg
,
my_xid
x
)
:
Log_event
(
thd_arg
,
0
,
0
),
xid
(
x
)
{}
Xid_log_event
(
THD
*
thd_arg
,
my_xid
x
)
:
Log_event
(
thd_arg
,
0
,
0
),
xid
(
x
)
{}
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
#else
#else
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
...
@@ -1320,6 +1510,11 @@ class Xid_log_event: public Log_event
...
@@ -1320,6 +1510,11 @@ class Xid_log_event: public Log_event
bool
write
(
IO_CACHE
*
file
);
bool
write
(
IO_CACHE
*
file
);
#endif
#endif
bool
is_valid
()
const
{
return
1
;
}
bool
is_valid
()
const
{
return
1
;
}
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
#endif
};
};
/*****************************************************************************
/*****************************************************************************
...
@@ -1349,7 +1544,6 @@ public:
...
@@ -1349,7 +1544,6 @@ public:
val_len
(
val_len_arg
),
type
(
type_arg
),
charset_number
(
charset_number_arg
)
val_len
(
val_len_arg
),
type
(
type_arg
),
charset_number
(
charset_number_arg
)
{
is_null
=
!
val
;
}
{
is_null
=
!
val
;
}
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#else
#else
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
#endif
#endif
...
@@ -1361,6 +1555,13 @@ public:
...
@@ -1361,6 +1555,13 @@ public:
bool
write
(
IO_CACHE
*
file
);
bool
write
(
IO_CACHE
*
file
);
#endif
#endif
bool
is_valid
()
const
{
return
1
;
}
bool
is_valid
()
const
{
return
1
;
}
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
virtual
int
do_update_pos
(
RELAY_LOG_INFO
*
rli
);
virtual
enum_skip_reason
do_shall_skip
(
RELAY_LOG_INFO
*
rli
);
#endif
};
};
...
@@ -1375,7 +1576,6 @@ public:
...
@@ -1375,7 +1576,6 @@ public:
#ifndef MYSQL_CLIENT
#ifndef MYSQL_CLIENT
Stop_log_event
()
:
Log_event
()
Stop_log_event
()
:
Log_event
()
{}
{}
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#else
#else
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
#endif
#endif
...
@@ -1386,6 +1586,22 @@ public:
...
@@ -1386,6 +1586,22 @@ public:
~
Stop_log_event
()
{}
~
Stop_log_event
()
{}
Log_event_type
get_type_code
()
{
return
STOP_EVENT
;}
Log_event_type
get_type_code
()
{
return
STOP_EVENT
;}
bool
is_valid
()
const
{
return
1
;
}
bool
is_valid
()
const
{
return
1
;
}
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_update_pos
(
RELAY_LOG_INFO
*
rli
);
virtual
enum_skip_reason
do_shall_skip
(
RELAY_LOG_INFO
*
rli
)
{
/*
Events from ourself should be skipped, but they should not
decrease the slave skip counter.
*/
if
(
this
->
server_id
==
::
server_id
)
return
Log_event
::
EVENT_SKIP_IGNORE
;
else
return
Log_event
::
EVENT_SKIP_NOT
;
}
#endif
};
};
/*****************************************************************************
/*****************************************************************************
...
@@ -1412,7 +1628,6 @@ public:
...
@@ -1412,7 +1628,6 @@ public:
ulonglong
pos_arg
,
uint
flags
);
ulonglong
pos_arg
,
uint
flags
);
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
#else
#else
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
...
@@ -1431,6 +1646,12 @@ public:
...
@@ -1431,6 +1646,12 @@ public:
#ifndef MYSQL_CLIENT
#ifndef MYSQL_CLIENT
bool
write
(
IO_CACHE
*
file
);
bool
write
(
IO_CACHE
*
file
);
#endif
#endif
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_update_pos
(
RELAY_LOG_INFO
*
rli
);
virtual
enum_skip_reason
do_shall_skip
(
RELAY_LOG_INFO
*
rli
);
#endif
};
};
...
@@ -1465,7 +1686,6 @@ public:
...
@@ -1465,7 +1686,6 @@ public:
bool
using_trans
);
bool
using_trans
);
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
#else
#else
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
...
@@ -1499,6 +1719,11 @@ public:
...
@@ -1499,6 +1719,11 @@ public:
*/
*/
bool
write_base
(
IO_CACHE
*
file
);
bool
write_base
(
IO_CACHE
*
file
);
#endif
#endif
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
#endif
};
};
...
@@ -1531,7 +1756,6 @@ public:
...
@@ -1531,7 +1756,6 @@ public:
Append_block_log_event
(
THD
*
thd
,
const
char
*
db_arg
,
char
*
block_arg
,
Append_block_log_event
(
THD
*
thd
,
const
char
*
db_arg
,
char
*
block_arg
,
uint
block_len_arg
,
bool
using_trans
);
uint
block_len_arg
,
bool
using_trans
);
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
int
exec_event
(
struct
st_relay_log_info
*
rli
);
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
virtual
int
get_create_or_append
()
const
;
virtual
int
get_create_or_append
()
const
;
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
...
@@ -1549,6 +1773,11 @@ public:
...
@@ -1549,6 +1773,11 @@ public:
bool
write
(
IO_CACHE
*
file
);
bool
write
(
IO_CACHE
*
file
);
const
char
*
get_db
()
{
return
db
;
}
const
char
*
get_db
()
{
return
db
;
}
#endif
#endif
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
#endif
};
};
...
@@ -1568,7 +1797,6 @@ public:
...
@@ -1568,7 +1797,6 @@ public:
Delete_file_log_event
(
THD
*
thd
,
const
char
*
db_arg
,
bool
using_trans
);
Delete_file_log_event
(
THD
*
thd
,
const
char
*
db_arg
,
bool
using_trans
);
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
#else
#else
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
...
@@ -1585,6 +1813,11 @@ public:
...
@@ -1585,6 +1813,11 @@ public:
bool
write
(
IO_CACHE
*
file
);
bool
write
(
IO_CACHE
*
file
);
const
char
*
get_db
()
{
return
db
;
}
const
char
*
get_db
()
{
return
db
;
}
#endif
#endif
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
#endif
};
};
...
@@ -1604,7 +1837,6 @@ public:
...
@@ -1604,7 +1837,6 @@ public:
Execute_load_log_event
(
THD
*
thd
,
const
char
*
db_arg
,
bool
using_trans
);
Execute_load_log_event
(
THD
*
thd
,
const
char
*
db_arg
,
bool
using_trans
);
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
#else
#else
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
...
@@ -1620,6 +1852,11 @@ public:
...
@@ -1620,6 +1852,11 @@ public:
bool
write
(
IO_CACHE
*
file
);
bool
write
(
IO_CACHE
*
file
);
const
char
*
get_db
()
{
return
db
;
}
const
char
*
get_db
()
{
return
db
;
}
#endif
#endif
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
#endif
};
};
...
@@ -1689,7 +1926,6 @@ public:
...
@@ -1689,7 +1926,6 @@ public:
bool
using_trans
,
bool
suppress_use
);
bool
using_trans
,
bool
suppress_use
);
#ifdef HAVE_REPLICATION
#ifdef HAVE_REPLICATION
void
pack_info
(
Protocol
*
protocol
);
void
pack_info
(
Protocol
*
protocol
);
int
exec_event
(
struct
st_relay_log_info
*
rli
);
#endif
/* HAVE_REPLICATION */
#endif
/* HAVE_REPLICATION */
#else
#else
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
void
print
(
FILE
*
file
,
PRINT_EVENT_INFO
*
print_event_info
);
...
@@ -1708,7 +1944,12 @@ public:
...
@@ -1708,7 +1944,12 @@ public:
#ifndef MYSQL_CLIENT
#ifndef MYSQL_CLIENT
bool
write_post_header_for_derived
(
IO_CACHE
*
file
);
bool
write_post_header_for_derived
(
IO_CACHE
*
file
);
#endif
#endif
};
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
#endif
};
#ifdef MYSQL_CLIENT
#ifdef MYSQL_CLIENT
...
@@ -1806,7 +2047,6 @@ public:
...
@@ -1806,7 +2047,6 @@ public:
#endif
#endif
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
exec_event
(
struct
st_relay_log_info
*
rli
);
virtual
void
pack_info
(
Protocol
*
protocol
);
virtual
void
pack_info
(
Protocol
*
protocol
);
#endif
#endif
...
@@ -1816,6 +2056,11 @@ public:
...
@@ -1816,6 +2056,11 @@ public:
private:
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
virtual
int
do_update_pos
(
RELAY_LOG_INFO
*
rli
);
#endif
#ifndef MYSQL_CLIENT
#ifndef MYSQL_CLIENT
TABLE
*
m_table
;
TABLE
*
m_table
;
#endif
#endif
...
@@ -1899,7 +2144,6 @@ public:
...
@@ -1899,7 +2144,6 @@ public:
flag_set
get_flags
(
flag_set
flags
)
const
{
return
m_flags
&
flags
;
}
flag_set
get_flags
(
flag_set
flags
)
const
{
return
m_flags
&
flags
;
}
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
exec_event
(
struct
st_relay_log_info
*
rli
);
virtual
void
pack_info
(
Protocol
*
protocol
);
virtual
void
pack_info
(
Protocol
*
protocol
);
#endif
#endif
...
@@ -1983,6 +2227,8 @@ protected:
...
@@ -1983,6 +2227,8 @@ protected:
private:
private:
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
virtual
int
do_apply_event
(
RELAY_LOG_INFO
const
*
rli
);
/*
/*
Primitive to prepare for a sequence of row executions.
Primitive to prepare for a sequence of row executions.
...
@@ -2030,7 +2276,7 @@ private:
...
@@ -2030,7 +2276,7 @@ private:
RETURN VALUE
RETURN VALUE
Error code, if something went wrong, 0 otherwise.
Error code, if something went wrong, 0 otherwise.
*/
*/
virtual
int
do_prepare_row
(
THD
*
,
RELAY_LOG_INFO
*
,
TABLE
*
,
virtual
int
do_prepare_row
(
THD
*
,
RELAY_LOG_INFO
const
*
,
TABLE
*
,
char
const
*
row_start
,
char
const
**
row_end
)
=
0
;
char
const
*
row_start
,
char
const
**
row_end
)
=
0
;
/*
/*
...
@@ -2101,7 +2347,7 @@ private:
...
@@ -2101,7 +2347,7 @@ private:
virtual
int
do_before_row_operations
(
TABLE
*
table
);
virtual
int
do_before_row_operations
(
TABLE
*
table
);
virtual
int
do_after_row_operations
(
TABLE
*
table
,
int
error
);
virtual
int
do_after_row_operations
(
TABLE
*
table
,
int
error
);
virtual
int
do_prepare_row
(
THD
*
,
RELAY_LOG_INFO
*
,
TABLE
*
,
virtual
int
do_prepare_row
(
THD
*
,
RELAY_LOG_INFO
const
*
,
TABLE
*
,
char
const
*
row_start
,
char
const
**
row_end
);
char
const
*
row_start
,
char
const
**
row_end
);
virtual
int
do_exec_row
(
TABLE
*
table
);
virtual
int
do_exec_row
(
TABLE
*
table
);
#endif
#endif
...
@@ -2166,7 +2412,7 @@ private:
...
@@ -2166,7 +2412,7 @@ private:
virtual
int
do_before_row_operations
(
TABLE
*
table
);
virtual
int
do_before_row_operations
(
TABLE
*
table
);
virtual
int
do_after_row_operations
(
TABLE
*
table
,
int
error
);
virtual
int
do_after_row_operations
(
TABLE
*
table
,
int
error
);
virtual
int
do_prepare_row
(
THD
*
,
RELAY_LOG_INFO
*
,
TABLE
*
,
virtual
int
do_prepare_row
(
THD
*
,
RELAY_LOG_INFO
const
*
,
TABLE
*
,
char
const
*
row_start
,
char
const
**
row_end
);
char
const
*
row_start
,
char
const
**
row_end
);
virtual
int
do_exec_row
(
TABLE
*
table
);
virtual
int
do_exec_row
(
TABLE
*
table
);
#endif
/* !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION) */
#endif
/* !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION) */
...
@@ -2237,7 +2483,7 @@ private:
...
@@ -2237,7 +2483,7 @@ private:
virtual
int
do_before_row_operations
(
TABLE
*
table
);
virtual
int
do_before_row_operations
(
TABLE
*
table
);
virtual
int
do_after_row_operations
(
TABLE
*
table
,
int
error
);
virtual
int
do_after_row_operations
(
TABLE
*
table
,
int
error
);
virtual
int
do_prepare_row
(
THD
*
,
RELAY_LOG_INFO
*
,
TABLE
*
,
virtual
int
do_prepare_row
(
THD
*
,
RELAY_LOG_INFO
const
*
,
TABLE
*
,
char
const
*
row_start
,
char
const
**
row_end
);
char
const
*
row_start
,
char
const
**
row_end
);
virtual
int
do_exec_row
(
TABLE
*
table
);
virtual
int
do_exec_row
(
TABLE
*
table
);
#endif
#endif
...
...
sql/rpl_rli.cc
View file @
08bb33db
...
@@ -29,14 +29,15 @@ int init_strvar_from_file(char *var, int max_size, IO_CACHE *f,
...
@@ -29,14 +29,15 @@ int init_strvar_from_file(char *var, int max_size, IO_CACHE *f,
st_relay_log_info
::
st_relay_log_info
()
st_relay_log_info
::
st_relay_log_info
()
:
no_storage
(
FALSE
),
info_fd
(
-
1
),
cur_log_fd
(
-
1
),
save_temporary_tables
(
0
),
:
no_storage
(
FALSE
),
replicate_same_server_id
(
::
replicate_same_server_id
),
info_fd
(
-
1
),
cur_log_fd
(
-
1
),
save_temporary_tables
(
0
),
cur_log_old_open_count
(
0
),
group_master_log_pos
(
0
),
log_space_total
(
0
),
cur_log_old_open_count
(
0
),
group_master_log_pos
(
0
),
log_space_total
(
0
),
ignore_log_space_limit
(
0
),
last_master_timestamp
(
0
),
slave_skip_counter
(
0
),
ignore_log_space_limit
(
0
),
last_master_timestamp
(
0
),
slave_skip_counter
(
0
),
abort_pos_wait
(
0
),
slave_run_id
(
0
),
sql_thd
(
0
),
last_slave_errno
(
0
),
abort_pos_wait
(
0
),
slave_run_id
(
0
),
sql_thd
(
0
),
last_slave_errno
(
0
),
inited
(
0
),
abort_slave
(
0
),
slave_running
(
0
),
until_condition
(
UNTIL_NONE
),
inited
(
0
),
abort_slave
(
0
),
slave_running
(
0
),
until_condition
(
UNTIL_NONE
),
until_log_pos
(
0
),
retried_trans
(
0
),
until_log_pos
(
0
),
retried_trans
(
0
),
tables_to_lock
(
0
),
tables_to_lock_count
(
0
),
tables_to_lock
(
0
),
tables_to_lock_count
(
0
),
unsafe_to_stop_at
(
0
)
last_event_start_time
(
0
)
{
{
DBUG_ENTER
(
"st_relay_log_info::st_relay_log_info"
);
DBUG_ENTER
(
"st_relay_log_info::st_relay_log_info"
);
...
@@ -1001,6 +1002,22 @@ bool st_relay_log_info::is_until_satisfied()
...
@@ -1001,6 +1002,22 @@ bool st_relay_log_info::is_until_satisfied()
log_pos
=
group_relay_log_pos
;
log_pos
=
group_relay_log_pos
;
}
}
#ifndef DBUG_OFF
{
char
buf
[
32
];
DBUG_PRINT
(
"info"
,
(
"group_master_log_name='%s', group_master_log_pos=%s"
,
group_master_log_name
,
llstr
(
group_master_log_pos
,
buf
)));
DBUG_PRINT
(
"info"
,
(
"group_relay_log_name='%s', group_relay_log_pos=%s"
,
group_relay_log_name
,
llstr
(
group_relay_log_pos
,
buf
)));
DBUG_PRINT
(
"info"
,
(
"(%s) log_name='%s', log_pos=%s"
,
until_condition
==
UNTIL_MASTER_POS
?
"master"
:
"relay"
,
log_name
,
llstr
(
log_pos
,
buf
)));
DBUG_PRINT
(
"info"
,
(
"(%s) until_log_name='%s', until_log_pos=%s"
,
until_condition
==
UNTIL_MASTER_POS
?
"master"
:
"relay"
,
until_log_name
,
llstr
(
until_log_pos
,
buf
)));
}
#endif
if
(
until_log_names_cmp_result
==
UNTIL_LOG_NAMES_CMP_UNKNOWN
)
if
(
until_log_names_cmp_result
==
UNTIL_LOG_NAMES_CMP_UNKNOWN
)
{
{
/*
/*
...
@@ -1056,30 +1073,19 @@ void st_relay_log_info::cached_charset_invalidate()
...
@@ -1056,30 +1073,19 @@ void st_relay_log_info::cached_charset_invalidate()
}
}
bool
st_relay_log_info
::
cached_charset_compare
(
char
*
charset
)
bool
st_relay_log_info
::
cached_charset_compare
(
char
*
charset
)
const
{
{
DBUG_ENTER
(
"st_relay_log_info::cached_charset_compare"
);
DBUG_ENTER
(
"st_relay_log_info::cached_charset_compare"
);
if
(
bcmp
(
cached_charset
,
charset
,
sizeof
(
cached_charset
)))
if
(
bcmp
(
cached_charset
,
charset
,
sizeof
(
cached_charset
)))
{
{
memcpy
(
c
ached_charset
,
charset
,
sizeof
(
cached_charset
));
memcpy
(
c
onst_cast
<
char
*>
(
cached_charset
)
,
charset
,
sizeof
(
cached_charset
));
DBUG_RETURN
(
1
);
DBUG_RETURN
(
1
);
}
}
DBUG_RETURN
(
0
);
DBUG_RETURN
(
0
);
}
}
void
st_relay_log_info
::
transaction_end
(
THD
*
thd
)
{
DBUG_ENTER
(
"st_relay_log_info::transaction_end"
);
/*
Nothing to do here right now.
*/
DBUG_VOID_RETURN
;
}
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
#if !defined(MYSQL_CLIENT) && defined(HAVE_REPLICATION)
void
st_relay_log_info
::
cleanup_context
(
THD
*
thd
,
bool
error
)
void
st_relay_log_info
::
cleanup_context
(
THD
*
thd
,
bool
error
)
{
{
...
@@ -1106,7 +1112,7 @@ void st_relay_log_info::cleanup_context(THD *thd, bool error)
...
@@ -1106,7 +1112,7 @@ void st_relay_log_info::cleanup_context(THD *thd, bool error)
m_table_map
.
clear_tables
();
m_table_map
.
clear_tables
();
close_thread_tables
(
thd
);
close_thread_tables
(
thd
);
clear_tables_to_lock
();
clear_tables_to_lock
();
unsafe_to_stop_at
=
0
;
last_event_start_time
=
0
;
DBUG_VOID_RETURN
;
DBUG_VOID_RETURN
;
}
}
...
...
sql/rpl_rli.h
View file @
08bb33db
...
@@ -58,6 +58,15 @@ typedef struct st_relay_log_info
...
@@ -58,6 +58,15 @@ typedef struct st_relay_log_info
*/
*/
bool
no_storage
;
bool
no_storage
;
/*
If true, events with the same server id should be replicated. This
field is set on creation of a relay log info structure by copying
the value of ::replicate_same_server_id and can be overridden if
necessary. For example of when this is done, check sql_binlog.cc,
where the BINLOG statement can be used to execute "raw" events.
*/
bool
replicate_same_server_id
;
/*** The following variables can only be read when protect by data lock ****/
/*** The following variables can only be read when protect by data lock ****/
/*
/*
...
@@ -292,14 +301,19 @@ typedef struct st_relay_log_info
...
@@ -292,14 +301,19 @@ typedef struct st_relay_log_info
When the 6 bytes are equal to 0 is used to mean "cache is invalidated".
When the 6 bytes are equal to 0 is used to mean "cache is invalidated".
*/
*/
void
cached_charset_invalidate
();
void
cached_charset_invalidate
();
bool
cached_charset_compare
(
char
*
charset
);
bool
cached_charset_compare
(
char
*
charset
)
const
;
void
transaction_end
(
THD
*
);
void
cleanup_context
(
THD
*
,
bool
);
void
cleanup_context
(
THD
*
,
bool
);
void
clear_tables_to_lock
();
void
clear_tables_to_lock
();
time_t
unsafe_to_stop_at
;
/*
Used by row-based replication to detect that it should not stop at
this event, but give it a chance to send more events. The time
where the last event inside a group started is stored here. If the
variable is zero, we are not in a group (but may be in a
transaction).
*/
time_t
last_event_start_time
;
}
RELAY_LOG_INFO
;
}
RELAY_LOG_INFO
;
...
...
sql/rpl_utility.cc
View file @
08bb33db
...
@@ -108,7 +108,7 @@ field_length_from_packed(enum_field_types const field_type,
...
@@ -108,7 +108,7 @@ field_length_from_packed(enum_field_types const field_type,
*/
*/
int
int
table_def
::
compatible_with
(
RELAY_LOG_INFO
*
rli
,
TABLE
*
table
)
table_def
::
compatible_with
(
RELAY_LOG_INFO
const
*
rli_arg
,
TABLE
*
table
)
const
const
{
{
/*
/*
...
@@ -116,6 +116,7 @@ table_def::compatible_with(RELAY_LOG_INFO *rli, TABLE *table)
...
@@ -116,6 +116,7 @@ table_def::compatible_with(RELAY_LOG_INFO *rli, TABLE *table)
*/
*/
uint
const
cols_to_check
=
min
(
table
->
s
->
fields
,
size
());
uint
const
cols_to_check
=
min
(
table
->
s
->
fields
,
size
());
int
error
=
0
;
int
error
=
0
;
RELAY_LOG_INFO
const
*
rli
=
const_cast
<
RELAY_LOG_INFO
*>
(
rli_arg
);
TABLE_SHARE
const
*
const
tsh
=
table
->
s
;
TABLE_SHARE
const
*
const
tsh
=
table
->
s
;
...
...
sql/rpl_utility.h
View file @
08bb33db
...
@@ -117,7 +117,7 @@ public:
...
@@ -117,7 +117,7 @@ public:
@retval 1 if the table definition is not compatible with @c table
@retval 1 if the table definition is not compatible with @c table
@retval 0 if the table definition is compatible with @c table
@retval 0 if the table definition is compatible with @c table
*/
*/
int
compatible_with
(
RELAY_LOG_INFO
*
rli
,
TABLE
*
table
)
const
;
int
compatible_with
(
RELAY_LOG_INFO
const
*
rli
,
TABLE
*
table
)
const
;
private:
private:
my_size_t
m_size
;
// Number of elements in the types array
my_size_t
m_size
;
// Number of elements in the types array
...
...
sql/slave.cc
View file @
08bb33db
...
@@ -519,11 +519,11 @@ static bool sql_slave_killed(THD* thd, RELAY_LOG_INFO* rli)
...
@@ -519,11 +519,11 @@ static bool sql_slave_killed(THD* thd, RELAY_LOG_INFO* rli)
really one minute of idleness, we don't timeout if the slave SQL thread
really one minute of idleness, we don't timeout if the slave SQL thread
is actively working.
is actively working.
*/
*/
if
(
!
rli
->
unsafe_to_stop_at
)
if
(
rli
->
last_event_start_time
==
0
)
DBUG_RETURN
(
1
);
DBUG_RETURN
(
1
);
DBUG_PRINT
(
"info"
,
(
"Slave SQL thread is in an unsafe situation, giving "
DBUG_PRINT
(
"info"
,
(
"Slave SQL thread is in an unsafe situation, giving "
"it some grace period"
));
"it some grace period"
));
if
(
difftime
(
time
(
0
),
rli
->
unsafe_to_stop_at
)
>
60
)
if
(
difftime
(
time
(
0
),
rli
->
last_event_start_time
)
>
60
)
{
{
slave_print_msg
(
ERROR_LEVEL
,
rli
,
0
,
slave_print_msg
(
ERROR_LEVEL
,
rli
,
0
,
"SQL thread had to stop in an unsafe situation, in "
"SQL thread had to stop in an unsafe situation, in "
...
@@ -557,7 +557,7 @@ static bool sql_slave_killed(THD* thd, RELAY_LOG_INFO* rli)
...
@@ -557,7 +557,7 @@ static bool sql_slave_killed(THD* thd, RELAY_LOG_INFO* rli)
void
void
*/
*/
void
slave_print_msg
(
enum
loglevel
level
,
RELAY_LOG_INFO
*
rli
,
void
slave_print_msg
(
enum
loglevel
level
,
RELAY_LOG_INFO
const
*
rli
,
int
err_code
,
const
char
*
msg
,
...)
int
err_code
,
const
char
*
msg
,
...)
{
{
void
(
*
report_function
)(
const
char
*
,
...);
void
(
*
report_function
)(
const
char
*
,
...);
...
@@ -579,9 +579,9 @@ void slave_print_msg(enum loglevel level, RELAY_LOG_INFO* rli,
...
@@ -579,9 +579,9 @@ void slave_print_msg(enum loglevel level, RELAY_LOG_INFO* rli,
It's an error, it must be reported in Last_error and Last_errno in SHOW
It's an error, it must be reported in Last_error and Last_errno in SHOW
SLAVE STATUS.
SLAVE STATUS.
*/
*/
pbuff
=
rli
->
last_slave_error
;
pbuff
=
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
last_slave_error
;
pbuffsize
=
sizeof
(
rli
->
last_slave_error
);
pbuffsize
=
sizeof
(
rli
->
last_slave_error
);
rli
->
last_slave_errno
=
err_code
;
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
last_slave_errno
=
err_code
;
report_function
=
sql_print_error
;
report_function
=
sql_print_error
;
break
;
break
;
case
WARNING_LEVEL
:
case
WARNING_LEVEL
:
...
@@ -813,7 +813,7 @@ do not trust column Seconds_Behind_Master of SHOW SLAVE STATUS");
...
@@ -813,7 +813,7 @@ do not trust column Seconds_Behind_Master of SHOW SLAVE STATUS");
{
{
if
((
master_row
=
mysql_fetch_row
(
master_res
))
&&
if
((
master_row
=
mysql_fetch_row
(
master_res
))
&&
(
::
server_id
==
strtoul
(
master_row
[
1
],
0
,
10
))
&&
(
::
server_id
==
strtoul
(
master_row
[
1
],
0
,
10
))
&&
!
replicate_same_server_id
)
!
mi
->
rli
.
replicate_same_server_id
)
errmsg
=
"The slave I/O thread stops because master and slave have equal \
errmsg
=
"The slave I/O thread stops because master and slave have equal \
MySQL server ids; these ids must be different for replication to work (or \
MySQL server ids; these ids must be different for replication to work (or \
the --replicate-same-server-id option must be used on slave but this does \
the --replicate-same-server-id option must be used on slave but this does \
...
@@ -1390,7 +1390,7 @@ void set_slave_thread_options(THD* thd)
...
@@ -1390,7 +1390,7 @@ void set_slave_thread_options(THD* thd)
DBUG_VOID_RETURN
;
DBUG_VOID_RETURN
;
}
}
void
set_slave_thread_default_charset
(
THD
*
thd
,
RELAY_LOG_INFO
*
rli
)
void
set_slave_thread_default_charset
(
THD
*
thd
,
RELAY_LOG_INFO
const
*
rli
)
{
{
DBUG_ENTER
(
"set_slave_thread_default_charset"
);
DBUG_ENTER
(
"set_slave_thread_default_charset"
);
...
@@ -1401,7 +1401,14 @@ void set_slave_thread_default_charset(THD* thd, RELAY_LOG_INFO *rli)
...
@@ -1401,7 +1401,14 @@ void set_slave_thread_default_charset(THD* thd, RELAY_LOG_INFO *rli)
thd
->
variables
.
collation_server
=
thd
->
variables
.
collation_server
=
global_system_variables
.
collation_server
;
global_system_variables
.
collation_server
;
thd
->
update_charset
();
thd
->
update_charset
();
rli
->
cached_charset_invalidate
();
/*
We use a const cast here since the conceptual (and externally
visible) behavior of the function is to set the default charset of
the thread. That the cache has to be invalidated is a secondary
effect.
*/
const_cast
<
RELAY_LOG_INFO
*>
(
rli
)
->
cached_charset_invalidate
();
DBUG_VOID_RETURN
;
DBUG_VOID_RETURN
;
}
}
...
@@ -1609,7 +1616,8 @@ static ulong read_event(MYSQL* mysql, MASTER_INFO *mi, bool* suppress_warnings)
...
@@ -1609,7 +1616,8 @@ static ulong read_event(MYSQL* mysql, MASTER_INFO *mi, bool* suppress_warnings)
}
}
int
check_expected_error
(
THD
*
thd
,
RELAY_LOG_INFO
*
rli
,
int
expected_error
)
int
check_expected_error
(
THD
*
thd
,
RELAY_LOG_INFO
const
*
rli
,
int
expected_error
)
{
{
DBUG_ENTER
(
"check_expected_error"
);
DBUG_ENTER
(
"check_expected_error"
);
...
@@ -1715,77 +1723,42 @@ static int exec_relay_log_event(THD* thd, RELAY_LOG_INFO* rli)
...
@@ -1715,77 +1723,42 @@ static int exec_relay_log_event(THD* thd, RELAY_LOG_INFO* rli)
if
(
ev
)
if
(
ev
)
{
{
int
type_code
=
ev
->
get_type_code
();
int
type_code
=
ev
->
get_type_code
();
int
exec_res
;
int
exec_res
=
0
;
/*
/*
Queries originating from this server must be skipped.
*/
Low-level events (Format_desc, Rotate, Stop) from this server
must also be skipped. But for those we don't want to modify
DBUG_PRINT
(
"info"
,(
"type_code=%d (%s), server_id=%d"
,
group_master_log_pos, because these events did not exist on the master.
type_code
,
ev
->
get_type_str
(),
ev
->
server_id
));
Format_desc is not completely skipped.
DBUG_PRINT
(
"info"
,
(
"thd->options={ %s%s}"
,
Skip queries specified by the user in slave_skip_counter.
FLAGSTR
(
thd
->
options
,
OPTION_NOT_AUTOCOMMIT
),
We can't however skip events that has something to do with the
FLAGSTR
(
thd
->
options
,
OPTION_BEGIN
)));
log files themselves.
Filtering on own server id is extremely important, to ignore execution of
events created by the creation/rotation of the relay log (remember that
now the relay log starts with its Format_desc, has a Rotate etc).
/*
*/
Execute the event to change the database and update the binary
log coordinates, but first we set some data that is needed for
DBUG_PRINT
(
"info"
,(
"type_code=%d, server_id=%d"
,
type_code
,
ev
->
server_id
));
the thread.
if
((
ev
->
server_id
==
(
uint32
)
::
server_id
&&
The event will be executed unless it is supposed to be skipped.
!
replicate_same_server_id
&&
type_code
!=
FORMAT_DESCRIPTION_EVENT
)
||
Queries originating from this server must be skipped. Low-level
(
rli
->
slave_skip_counter
&&
events (Format_description_log_event, Rotate_log_event,
type_code
!=
ROTATE_EVENT
&&
type_code
!=
STOP_EVENT
&&
Stop_log_event) from this server must also be skipped. But for
type_code
!=
START_EVENT_V3
&&
type_code
!=
FORMAT_DESCRIPTION_EVENT
))
those we don't want to modify 'group_master_log_pos', because
{
these events did not exist on the master.
DBUG_PRINT
(
"info"
,
(
"event skipped"
));
Format_description_log_event is not completely skipped.
/*
We only skip the event here and do not increase the group log
Skip queries specified by the user in 'slave_skip_counter'. We
position. In the event that we have to restart, this means
can't however skip events that has something to do with the log
that we might have to skip the event again, but that is a
files themselves.
minor issue.
Filtering on own server id is extremely important, to ignore
If we were to increase the group log position when skipping an
execution of events created by the creation/rotation of the relay
event, it might be that we are restarting at the wrong
log (remember that now the relay log starts with its Format_desc,
position and have events before that we should have executed,
has a Rotate etc).
so not increasing the group log position is a sure bet in this
*/
case.
In this way, we just step the group log position when we
*know* that we are at the end of a group.
*/
rli
->
inc_event_relay_log_pos
();
/*
Protect against common user error of setting the counter to 1
instead of 2 while recovering from an insert which used auto_increment,
rand or user var.
*/
if
(
rli
->
slave_skip_counter
&&
!
((
type_code
==
INTVAR_EVENT
||
type_code
==
RAND_EVENT
||
type_code
==
USER_VAR_EVENT
)
&&
rli
->
slave_skip_counter
==
1
)
&&
/*
The events from ourselves which have something to do with the relay
log itself must be skipped, true, but they mustn't decrement
rli->slave_skip_counter, because the user is supposed to not see
these events (they are not in the master's binlog) and if we
decremented, START SLAVE would for example decrement when it sees
the Rotate, so the event which the user probably wanted to skip
would not be skipped.
*/
!
(
ev
->
server_id
==
(
uint32
)
::
server_id
&&
(
type_code
==
ROTATE_EVENT
||
type_code
==
STOP_EVENT
||
type_code
==
START_EVENT_V3
||
type_code
==
FORMAT_DESCRIPTION_EVENT
)))
--
rli
->
slave_skip_counter
;
pthread_mutex_unlock
(
&
rli
->
data_lock
);
delete
ev
;
DBUG_RETURN
(
0
);
// avoid infinite update loops
}
pthread_mutex_unlock
(
&
rli
->
data_lock
);
thd
->
server_id
=
ev
->
server_id
;
// use the original server id for logging
thd
->
server_id
=
ev
->
server_id
;
// use the original server id for logging
thd
->
set_time
();
// time the query
thd
->
set_time
();
// time the query
...
@@ -1793,13 +1766,63 @@ static int exec_relay_log_event(THD* thd, RELAY_LOG_INFO* rli)
...
@@ -1793,13 +1766,63 @@ static int exec_relay_log_event(THD* thd, RELAY_LOG_INFO* rli)
if
(
!
ev
->
when
)
if
(
!
ev
->
when
)
ev
->
when
=
time
(
NULL
);
ev
->
when
=
time
(
NULL
);
ev
->
thd
=
thd
;
// because up to this point, ev->thd == 0
ev
->
thd
=
thd
;
// because up to this point, ev->thd == 0
DBUG_PRINT
(
"info"
,
(
"thd->options={ %s%s}"
,
FLAGSTR
(
thd
->
options
,
OPTION_NOT_AUTOCOMMIT
),
FLAGSTR
(
thd
->
options
,
OPTION_BEGIN
)));
exec_res
=
ev
->
exec_event
(
rli
);
int
reason
=
ev
->
shall_skip
(
rli
);
DBUG_PRINT
(
"info"
,
(
"exec_event result: %d"
,
exec_res
));
if
(
reason
==
Log_event
::
EVENT_SKIP_COUNT
)
DBUG_ASSERT
(
rli
->
sql_thd
==
thd
);
--
rli
->
slave_skip_counter
;
pthread_mutex_unlock
(
&
rli
->
data_lock
);
if
(
reason
==
Log_event
::
EVENT_SKIP_NOT
)
exec_res
=
ev
->
apply_event
(
rli
);
#ifndef DBUG_OFF
else
{
/*
This only prints information to the debug trace.
TODO: Print an informational message to the error log?
*/
static
const
char
*
const
explain
[]
=
{
"event was not skipped"
,
// EVENT_SKIP_NOT,
"event originated from this server"
,
// EVENT_SKIP_IGNORE,
"event skip counter was non-zero"
// EVENT_SKIP_COUNT
};
DBUG_PRINT
(
"info"
,
(
"%s was skipped because %s"
,
ev
->
get_type_str
(),
explain
[
reason
]));
}
#endif
DBUG_PRINT
(
"info"
,
(
"apply_event error = %d"
,
exec_res
));
if
(
exec_res
==
0
)
{
int
error
=
ev
->
update_pos
(
rli
);
char
buf
[
22
];
DBUG_PRINT
(
"info"
,
(
"update_pos error = %d"
,
error
));
DBUG_PRINT
(
"info"
,
(
"group %s %s"
,
llstr
(
rli
->
group_relay_log_pos
,
buf
),
rli
->
group_relay_log_name
));
DBUG_PRINT
(
"info"
,
(
"event %s %s"
,
llstr
(
rli
->
event_relay_log_pos
,
buf
),
rli
->
event_relay_log_name
));
/*
The update should not fail, so print an error message and
return an error code.
TODO: Replace this with a decent error message when merged
with BUG#24954 (which adds several new error message).
*/
if
(
error
)
{
slave_print_msg
(
ERROR_LEVEL
,
rli
,
ER_UNKNOWN_ERROR
,
"It was not possible to update the positions"
" of the relay log information: the slave may"
" be in an inconsistent state."
" Stopped in %s position %s"
,
rli
->
group_relay_log_name
,
llstr
(
rli
->
group_relay_log_pos
,
buf
));
DBUG_RETURN
(
1
);
}
}
/*
/*
Format_description_log_event should not be deleted because it will be
Format_description_log_event should not be deleted because it will be
used to read info about the relay log's format; it will be deleted when
used to read info about the relay log's format; it will be deleted when
...
@@ -2366,13 +2389,17 @@ Slave SQL thread aborted. Can't execute init_slave query");
...
@@ -2366,13 +2389,17 @@ Slave SQL thread aborted. Can't execute init_slave query");
THD_CHECK_SENTRY
(
thd
);
THD_CHECK_SENTRY
(
thd
);
if
(
exec_relay_log_event
(
thd
,
rli
))
if
(
exec_relay_log_event
(
thd
,
rli
))
{
{
DBUG_PRINT
(
"info"
,
(
"exec_relay_log_event() failed"
));
// do not scare the user if SQL thread was simply killed or stopped
// do not scare the user if SQL thread was simply killed or stopped
if
(
!
sql_slave_killed
(
thd
,
rli
))
if
(
!
sql_slave_killed
(
thd
,
rli
))
{
{
/*
/*
retrieve as much info as possible from the thd and, error codes and warnings
retrieve as much info as possible from the thd and, error
and print this to the error log as to allow the user to locate the error
codes and warnings and print this to the error log as to
allow the user to locate the error
*/
*/
DBUG_PRINT
(
"info"
,
(
"thd->net.last_errno=%d; rli->last_slave_errno=%d"
,
thd
->
net
.
last_errno
,
rli
->
last_slave_errno
));
if
(
thd
->
net
.
last_errno
!=
0
)
if
(
thd
->
net
.
last_errno
!=
0
)
{
{
if
(
rli
->
last_slave_errno
==
0
)
if
(
rli
->
last_slave_errno
==
0
)
...
@@ -2699,6 +2726,7 @@ static int queue_binlog_ver_1_event(MASTER_INFO *mi, const char *buf,
...
@@ -2699,6 +2726,7 @@ static int queue_binlog_ver_1_event(MASTER_INFO *mi, const char *buf,
my_free
((
char
*
)
tmp_buf
,
MYF
(
MY_ALLOW_ZERO_PTR
));
my_free
((
char
*
)
tmp_buf
,
MYF
(
MY_ALLOW_ZERO_PTR
));
DBUG_RETURN
(
1
);
DBUG_RETURN
(
1
);
}
}
pthread_mutex_lock
(
&
mi
->
data_lock
);
pthread_mutex_lock
(
&
mi
->
data_lock
);
ev
->
log_pos
=
mi
->
master_log_pos
;
/* 3.23 events don't contain log_pos */
ev
->
log_pos
=
mi
->
master_log_pos
;
/* 3.23 events don't contain log_pos */
switch
(
ev
->
get_type_code
())
{
switch
(
ev
->
get_type_code
())
{
...
@@ -2962,7 +2990,7 @@ int queue_event(MASTER_INFO* mi,const char* buf, ulong event_len)
...
@@ -2962,7 +2990,7 @@ int queue_event(MASTER_INFO* mi,const char* buf, ulong event_len)
pthread_mutex_lock
(
log_lock
);
pthread_mutex_lock
(
log_lock
);
if
((
uint4korr
(
buf
+
SERVER_ID_OFFSET
)
==
::
server_id
)
&&
if
((
uint4korr
(
buf
+
SERVER_ID_OFFSET
)
==
::
server_id
)
&&
!
replicate_same_server_id
)
!
mi
->
rli
.
replicate_same_server_id
)
{
{
/*
/*
Do not write it to the relay log.
Do not write it to the relay log.
...
...
sql/slave.h
View file @
08bb33db
...
@@ -162,9 +162,9 @@ bool show_binlog_info(THD* thd);
...
@@ -162,9 +162,9 @@ bool show_binlog_info(THD* thd);
bool
rpl_master_has_bug
(
RELAY_LOG_INFO
*
rli
,
uint
bug_id
);
bool
rpl_master_has_bug
(
RELAY_LOG_INFO
*
rli
,
uint
bug_id
);
const
char
*
print_slave_db_safe
(
const
char
*
db
);
const
char
*
print_slave_db_safe
(
const
char
*
db
);
int
check_expected_error
(
THD
*
thd
,
RELAY_LOG_INFO
*
rli
,
int
error_code
);
int
check_expected_error
(
THD
*
thd
,
RELAY_LOG_INFO
const
*
rli
,
int
error_code
);
void
skip_load_data_infile
(
NET
*
net
);
void
skip_load_data_infile
(
NET
*
net
);
void
slave_print_msg
(
enum
loglevel
level
,
RELAY_LOG_INFO
*
rli
,
void
slave_print_msg
(
enum
loglevel
level
,
RELAY_LOG_INFO
const
*
rli
,
int
err_code
,
const
char
*
msg
,
...)
int
err_code
,
const
char
*
msg
,
...)
ATTRIBUTE_FORMAT
(
printf
,
4
,
5
);
ATTRIBUTE_FORMAT
(
printf
,
4
,
5
);
...
@@ -182,7 +182,7 @@ int init_relay_log_pos(RELAY_LOG_INFO* rli,const char* log,ulonglong pos,
...
@@ -182,7 +182,7 @@ int init_relay_log_pos(RELAY_LOG_INFO* rli,const char* log,ulonglong pos,
int
purge_relay_logs
(
RELAY_LOG_INFO
*
rli
,
THD
*
thd
,
bool
just_reset
,
int
purge_relay_logs
(
RELAY_LOG_INFO
*
rli
,
THD
*
thd
,
bool
just_reset
,
const
char
**
errmsg
);
const
char
**
errmsg
);
void
set_slave_thread_options
(
THD
*
thd
);
void
set_slave_thread_options
(
THD
*
thd
);
void
set_slave_thread_default_charset
(
THD
*
thd
,
RELAY_LOG_INFO
*
rli
);
void
set_slave_thread_default_charset
(
THD
*
thd
,
RELAY_LOG_INFO
const
*
rli
);
void
rotate_relay_log
(
MASTER_INFO
*
mi
);
void
rotate_relay_log
(
MASTER_INFO
*
mi
);
pthread_handler_t
handle_slave_io
(
void
*
arg
);
pthread_handler_t
handle_slave_io
(
void
*
arg
);
...
...
sql/sql_binlog.cc
View file @
08bb33db
...
@@ -163,9 +163,17 @@ void mysql_client_binlog_statement(THD* thd)
...
@@ -163,9 +163,17 @@ void mysql_client_binlog_statement(THD* thd)
(
ulong
)
uint4korr
(
bufptr
+
EVENT_LEN_OFFSET
)));
(
ulong
)
uint4korr
(
bufptr
+
EVENT_LEN_OFFSET
)));
#endif
#endif
ev
->
thd
=
thd
;
ev
->
thd
=
thd
;
if
(
IF_DBUG
(
int
err
=
)
ev
->
exec_event
(
thd
->
rli_fake
))
/*
We go directly to the application phase, since we don't need
to check if the event shall be skipped or not.
Neither do we have to update the log positions, since that is
not used at all: the rli_fake instance is used only for error
reporting.
*/
if
(
IF_DBUG
(
int
err
=
)
ev
->
apply_event
(
thd
->
rli_fake
))
{
{
DBUG_PRINT
(
"
error"
,
(
"exec
_event() returned: %d"
,
err
));
DBUG_PRINT
(
"
info"
,
(
"apply
_event() returned: %d"
,
err
));
/*
/*
TODO: Maybe a better error message since the BINLOG statement
TODO: Maybe a better error message since the BINLOG statement
now contains several events.
now contains several events.
...
...
sql/sql_class.cc
View file @
08bb33db
...
@@ -679,11 +679,22 @@ bool THD::store_globals()
...
@@ -679,11 +679,22 @@ bool THD::store_globals()
void
THD
::
cleanup_after_query
()
void
THD
::
cleanup_after_query
()
{
{
/*
Reset rand_used so that detection of calls to rand() will save random
seeds if needed by the slave.
Do not reset rand_used if inside a stored function or trigger because
only the call to these operations is logged. Thus only the calling
statement needs to detect rand() calls made by its substatements. These
substatements must not set rand_used to 0 because it would remove the
detection of rand() by the calling statement.
*/
if
(
!
in_sub_stmt
)
/* stored functions and triggers are a special case */
if
(
!
in_sub_stmt
)
/* stored functions and triggers are a special case */
{
{
/* Forget those values, for next binlogger: */
/* Forget those values, for next binlogger: */
stmt_depends_on_first_successful_insert_id_in_prev_stmt
=
0
;
stmt_depends_on_first_successful_insert_id_in_prev_stmt
=
0
;
auto_inc_intervals_in_cur_stmt_for_binlog
.
empty
();
auto_inc_intervals_in_cur_stmt_for_binlog
.
empty
();
rand_used
=
0
;
}
}
if
(
first_successful_insert_id_in_cur_stmt
>
0
)
if
(
first_successful_insert_id_in_cur_stmt
>
0
)
{
{
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment