mirror of
https://github.com/MariaDB/server.git
synced 2025-01-26 00:34:18 +01:00
0594e1b84b
this is a cleanup patch for our current auto_increment handling: new names for auto_increment variables in THD, new methods to manipulate them (see sql_class.h), some move into handler::, causing less backup/restore work when executing substatements. This makes the logic hopefully clearer, less work is is needed in mysql_insert(). By cleaning up, using different variables for different purposes (instead of one for 3 things...), we fix those bugs, which someone may want to fix in 5.0 too: BUG#20339 "stored procedure using LAST_INSERT_ID() does not replicate statement-based" BUG#20341 "stored function inserting into one auto_increment puts bad data in slave" BUG#19243 "wrong LAST_INSERT_ID() after ON DUPLICATE KEY UPDATE" (now if a row is updated, LAST_INSERT_ID() will return its id) and re-fixes: BUG#6880 "LAST_INSERT_ID() value changes during multi-row INSERT" (already fixed differently by Ramil in 4.1) Test of documented behaviour of mysql_insert_id() (there was no test). The behaviour changes introduced are: - LAST_INSERT_ID() now returns "the first autogenerated auto_increment value successfully inserted", instead of "the first autogenerated auto_increment value if any row was successfully inserted", see auto_increment.test. Same for mysql_insert_id(), see mysql_client_test.c. - LAST_INSERT_ID() returns the id of the updated row if ON DUPLICATE KEY UPDATE, see auto_increment.test. Same for mysql_insert_id(), see mysql_client_test.c. - LAST_INSERT_ID() does not change if no autogenerated value was successfully inserted (it used to then be 0), see auto_increment.test. - if in INSERT SELECT no autogenerated value was successfully inserted, mysql_insert_id() now returns the id of the last inserted row (it already did this for INSERT VALUES), see mysql_client_test.c. - if INSERT SELECT uses LAST_INSERT_ID(X), mysql_insert_id() now returns X (it already did this for INSERT VALUES), see mysql_client_test.c. - NDB now behaves like other engines wrt SET INSERT_ID: with INSERT IGNORE, the id passed in SET INSERT_ID is re-used until a row succeeds; SET INSERT_ID influences not only the first row now. Additionally, when unlocking a table we check that the thread is not keeping a next_insert_id (as the table is unlocked that id is potentially out-of-date); forgetting about this next_insert_id is done in a new handler::ha_release_auto_increment(). Finally we prepare for engines capable of reserving finite-length intervals of auto_increment values: we store such intervals in THD. The next step (to be done by the replication team in 5.1) is to read those intervals from THD and actually store them in the statement-based binary log. NDB will be a good engine to test that.
170 lines
2.6 KiB
Text
170 lines
2.6 KiB
Text
stop slave;
|
|
drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9;
|
|
reset master;
|
|
reset slave;
|
|
drop table if exists t1,t2,t3,t4,t5,t6,t7,t8,t9;
|
|
start slave;
|
|
***************** Test 1 ************************
|
|
|
|
CREATE TABLE t1 (a INT NOT NULL auto_increment,b INT, PRIMARY KEY (a)) ENGINE=NDB auto_increment=3;
|
|
insert into t1 values (NULL,1),(NULL,2),(NULL,3);
|
|
******* Select from Master *************
|
|
|
|
select * from t1 ORDER BY a;
|
|
a b
|
|
3 1
|
|
4 2
|
|
5 3
|
|
******* Select from Slave *************
|
|
|
|
select * from t1 ORDER BY a;
|
|
a b
|
|
3 1
|
|
4 2
|
|
5 3
|
|
drop table t1;
|
|
create table t1 (a int not null auto_increment,b int, primary key (a)) engine=NDB;
|
|
insert into t1 values (1,1),(NULL,2),(3,3),(NULL,4);
|
|
delete from t1 where b=4;
|
|
insert into t1 values (NULL,5),(NULL,6);
|
|
******* Select from Master *************
|
|
|
|
select * from t1 ORDER BY a;
|
|
a b
|
|
1 1
|
|
2 2
|
|
3 3
|
|
5 5
|
|
6 6
|
|
******* Select from Slave *************
|
|
|
|
select * from t1 ORDER BY a;
|
|
a b
|
|
1 1
|
|
2 2
|
|
3 3
|
|
5 5
|
|
6 6
|
|
drop table t1;
|
|
create table t1 (a int not null auto_increment, primary key (a)) engine=NDB;
|
|
insert into t1 values (NULL),(5),(NULL);
|
|
insert into t1 values (250),(NULL);
|
|
******* Select from Master *************
|
|
|
|
select * from t1 ORDER BY a;
|
|
a
|
|
1
|
|
5
|
|
6
|
|
250
|
|
251
|
|
insert into t1 values (1000);
|
|
set @@insert_id=400;
|
|
insert into t1 values(NULL),(NULL);
|
|
******* Select from Master *************
|
|
|
|
select * from t1 ORDER BY a;
|
|
a
|
|
1
|
|
5
|
|
6
|
|
250
|
|
251
|
|
400
|
|
401
|
|
1000
|
|
******* Select from Slave *************
|
|
|
|
select * from t1 ORDER BY a;
|
|
a
|
|
1
|
|
5
|
|
6
|
|
250
|
|
251
|
|
400
|
|
401
|
|
1000
|
|
drop table t1;
|
|
create table t1 (a int not null auto_increment, primary key (a)) engine=NDB;
|
|
insert into t1 values (NULL),(5),(NULL),(NULL);
|
|
insert into t1 values (500),(NULL),(502),(NULL),(600);
|
|
******* Select from Master *************
|
|
|
|
select * from t1 ORDER BY a;
|
|
a
|
|
1
|
|
5
|
|
6
|
|
7
|
|
500
|
|
501
|
|
502
|
|
503
|
|
600
|
|
set @@insert_id=600;
|
|
insert into t1 values(600),(NULL),(NULL);
|
|
ERROR 23000: Can't write; duplicate key in table 't1'
|
|
set @@insert_id=600;
|
|
insert ignore into t1 values(600),(NULL),(NULL),(610),(NULL);
|
|
******* Select from Master *************
|
|
|
|
select * from t1 ORDER BY a;
|
|
a
|
|
1
|
|
5
|
|
6
|
|
7
|
|
500
|
|
501
|
|
502
|
|
503
|
|
600
|
|
610
|
|
611
|
|
******* Select from Slave *************
|
|
|
|
select * from t1 ORDER BY a;
|
|
a
|
|
1
|
|
5
|
|
6
|
|
7
|
|
500
|
|
501
|
|
502
|
|
503
|
|
600
|
|
610
|
|
611
|
|
drop table t1;
|
|
create table t1 (a int not null auto_increment, primary key (a)) engine=NDB;
|
|
insert into t1 values(2),(12),(22),(32),(42);
|
|
insert into t1 values (NULL),(NULL);
|
|
insert into t1 values (3),(NULL),(NULL);
|
|
******* Select from Master *************
|
|
|
|
select * from t1 ORDER BY a;
|
|
a
|
|
1
|
|
2
|
|
3
|
|
4
|
|
5
|
|
******* Select from Slave *************
|
|
|
|
** Slave should have 2, 12, 22, 32, 42 **
|
|
** Master will have 2 but not 12, 22, 32, 42 **
|
|
|
|
select * from t1 ORDER BY a;
|
|
a
|
|
1
|
|
2
|
|
3
|
|
4
|
|
5
|
|
12
|
|
22
|
|
32
|
|
42
|
|
drop table t1;
|