2006-02-24 23:50:36 +03:00
|
|
|
drop table if exists t1, t2, t3, t4;
|
2004-09-07 16:29:46 +04:00
|
|
|
drop view if exists v1;
|
2005-03-27 16:15:21 +04:00
|
|
|
drop database if exists mysqltest;
|
2005-07-09 21:51:59 +04:00
|
|
|
drop function if exists f1;
|
2006-03-29 14:53:00 +04:00
|
|
|
drop function if exists f2;
|
2005-08-10 10:31:32 +04:00
|
|
|
drop procedure if exists p1;
|
2004-09-07 16:29:46 +04:00
|
|
|
create table t1 (i int);
|
|
|
|
create trigger trg before insert on t1 for each row set @a:=1;
|
|
|
|
set @a:=0;
|
|
|
|
select @a;
|
|
|
|
@a
|
|
|
|
0
|
|
|
|
insert into t1 values (1);
|
|
|
|
select @a;
|
|
|
|
@a
|
|
|
|
1
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger trg;
|
2004-09-07 16:29:46 +04:00
|
|
|
create trigger trg before insert on t1 for each row set @a:=new.i;
|
|
|
|
insert into t1 values (123);
|
|
|
|
select @a;
|
|
|
|
@a
|
|
|
|
123
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger trg;
|
2004-09-07 16:29:46 +04:00
|
|
|
drop table t1;
|
|
|
|
create table t1 (i int not null, j int);
|
|
|
|
create trigger trg before insert on t1 for each row
|
|
|
|
begin
|
|
|
|
if isnull(new.j) then
|
|
|
|
set new.j:= new.i * 10;
|
|
|
|
end if;
|
|
|
|
end|
|
|
|
|
insert into t1 (i) values (1)|
|
|
|
|
insert into t1 (i,j) values (2, 3)|
|
|
|
|
select * from t1|
|
|
|
|
i j
|
|
|
|
1 10
|
|
|
|
2 3
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger trg|
|
2004-09-07 16:29:46 +04:00
|
|
|
drop table t1|
|
|
|
|
create table t1 (i int not null primary key);
|
|
|
|
create trigger trg after insert on t1 for each row
|
|
|
|
set @a:= if(@a,concat(@a, ":", new.i), new.i);
|
|
|
|
set @a:="";
|
|
|
|
insert into t1 values (2),(3),(4),(5);
|
|
|
|
select @a;
|
|
|
|
@a
|
|
|
|
2:3:4:5
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger trg;
|
2004-09-07 16:29:46 +04:00
|
|
|
drop table t1;
|
|
|
|
create table t1 (aid int not null primary key, balance int not null default 0);
|
|
|
|
insert into t1 values (1, 1000), (2,3000);
|
|
|
|
create trigger trg before update on t1 for each row
|
|
|
|
begin
|
|
|
|
declare loc_err varchar(255);
|
|
|
|
if abs(new.balance - old.balance) > 1000 then
|
|
|
|
set new.balance:= old.balance;
|
|
|
|
set loc_err := concat("Too big change for aid = ", new.aid);
|
|
|
|
set @update_failed:= if(@update_failed, concat(@a, ":", loc_err), loc_err);
|
|
|
|
end if;
|
|
|
|
end|
|
|
|
|
set @update_failed:=""|
|
|
|
|
update t1 set balance=1500|
|
|
|
|
select @update_failed;
|
|
|
|
select * from t1|
|
|
|
|
@update_failed
|
|
|
|
Too big change for aid = 2
|
|
|
|
aid balance
|
|
|
|
1 1500
|
|
|
|
2 3000
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger trg|
|
2004-09-07 16:29:46 +04:00
|
|
|
drop table t1|
|
|
|
|
create table t1 (i int);
|
|
|
|
insert into t1 values (1),(2),(3),(4);
|
|
|
|
create trigger trg after update on t1 for each row
|
|
|
|
set @total_change:=@total_change + new.i - old.i;
|
|
|
|
set @total_change:=0;
|
|
|
|
update t1 set i=3;
|
|
|
|
select @total_change;
|
|
|
|
@total_change
|
|
|
|
2
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger trg;
|
2004-09-07 16:29:46 +04:00
|
|
|
drop table t1;
|
|
|
|
create table t1 (i int);
|
|
|
|
insert into t1 values (1),(2),(3),(4);
|
|
|
|
create trigger trg before delete on t1 for each row
|
|
|
|
set @del_sum:= @del_sum + old.i;
|
|
|
|
set @del_sum:= 0;
|
|
|
|
delete from t1 where i <= 3;
|
|
|
|
select @del_sum;
|
|
|
|
@del_sum
|
|
|
|
6
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger trg;
|
2004-09-07 16:29:46 +04:00
|
|
|
drop table t1;
|
|
|
|
create table t1 (i int);
|
|
|
|
insert into t1 values (1),(2),(3),(4);
|
|
|
|
create trigger trg after delete on t1 for each row set @del:= 1;
|
|
|
|
set @del:= 0;
|
|
|
|
delete from t1 where i <> 0;
|
|
|
|
select @del;
|
|
|
|
@del
|
|
|
|
1
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger trg;
|
2004-09-07 16:29:46 +04:00
|
|
|
drop table t1;
|
|
|
|
create table t1 (i int, j int);
|
|
|
|
create trigger trg1 before insert on t1 for each row
|
|
|
|
begin
|
|
|
|
if new.j > 10 then
|
|
|
|
set new.j := 10;
|
|
|
|
end if;
|
|
|
|
end|
|
|
|
|
create trigger trg2 before update on t1 for each row
|
|
|
|
begin
|
|
|
|
if old.i % 2 = 0 then
|
|
|
|
set new.j := -1;
|
|
|
|
end if;
|
|
|
|
end|
|
|
|
|
create trigger trg3 after update on t1 for each row
|
|
|
|
begin
|
|
|
|
if new.j = -1 then
|
|
|
|
set @fired:= "Yes";
|
|
|
|
end if;
|
|
|
|
end|
|
|
|
|
set @fired:="";
|
|
|
|
insert into t1 values (1,2),(2,3),(3,14);
|
|
|
|
select @fired;
|
|
|
|
@fired
|
|
|
|
|
|
|
|
select * from t1;
|
|
|
|
i j
|
|
|
|
1 2
|
|
|
|
2 3
|
|
|
|
3 10
|
|
|
|
update t1 set j= 20;
|
|
|
|
select @fired;
|
|
|
|
@fired
|
|
|
|
Yes
|
|
|
|
select * from t1;
|
|
|
|
i j
|
|
|
|
1 20
|
|
|
|
2 -1
|
|
|
|
3 20
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger trg1;
|
|
|
|
drop trigger trg2;
|
|
|
|
drop trigger trg3;
|
2004-09-07 16:29:46 +04:00
|
|
|
drop table t1;
|
2005-05-24 22:19:33 +04:00
|
|
|
create table t1 (id int not null primary key, data int);
|
|
|
|
create trigger t1_bi before insert on t1 for each row
|
|
|
|
set @log:= concat(@log, "(BEFORE_INSERT: new=(id=", new.id, ", data=", new.data,"))");
|
|
|
|
create trigger t1_ai after insert on t1 for each row
|
|
|
|
set @log:= concat(@log, "(AFTER_INSERT: new=(id=", new.id, ", data=", new.data,"))");
|
|
|
|
create trigger t1_bu before update on t1 for each row
|
|
|
|
set @log:= concat(@log, "(BEFORE_UPDATE: old=(id=", old.id, ", data=", old.data,
|
|
|
|
") new=(id=", new.id, ", data=", new.data,"))");
|
|
|
|
create trigger t1_au after update on t1 for each row
|
|
|
|
set @log:= concat(@log, "(AFTER_UPDATE: old=(id=", old.id, ", data=", old.data,
|
|
|
|
") new=(id=", new.id, ", data=", new.data,"))");
|
|
|
|
create trigger t1_bd before delete on t1 for each row
|
|
|
|
set @log:= concat(@log, "(BEFORE_DELETE: old=(id=", old.id, ", data=", old.data,"))");
|
|
|
|
create trigger t1_ad after delete on t1 for each row
|
|
|
|
set @log:= concat(@log, "(AFTER_DELETE: old=(id=", old.id, ", data=", old.data,"))");
|
|
|
|
set @log:= "";
|
|
|
|
insert into t1 values (1, 1);
|
|
|
|
select @log;
|
|
|
|
@log
|
|
|
|
(BEFORE_INSERT: new=(id=1, data=1))(AFTER_INSERT: new=(id=1, data=1))
|
|
|
|
set @log:= "";
|
|
|
|
insert ignore t1 values (1, 2);
|
|
|
|
select @log;
|
|
|
|
@log
|
|
|
|
(BEFORE_INSERT: new=(id=1, data=2))
|
|
|
|
set @log:= "";
|
2006-06-16 20:21:25 +04:00
|
|
|
insert into t1 (id, data) values (1, 3), (2, 2) on duplicate key update data= data + 1;
|
2005-05-24 22:19:33 +04:00
|
|
|
select @log;
|
|
|
|
@log
|
2006-06-16 20:21:25 +04:00
|
|
|
(BEFORE_INSERT: new=(id=1, data=3))(BEFORE_UPDATE: old=(id=1, data=1) new=(id=1, data=2))(AFTER_UPDATE: old=(id=1, data=1) new=(id=1, data=2))(BEFORE_INSERT: new=(id=2, data=2))(AFTER_INSERT: new=(id=2, data=2))
|
2005-05-24 22:19:33 +04:00
|
|
|
set @log:= "";
|
2006-06-16 20:21:25 +04:00
|
|
|
replace t1 values (1, 4), (3, 3);
|
2005-05-24 22:19:33 +04:00
|
|
|
select @log;
|
|
|
|
@log
|
2006-06-16 20:21:25 +04:00
|
|
|
(BEFORE_INSERT: new=(id=1, data=4))(BEFORE_DELETE: old=(id=1, data=2))(AFTER_DELETE: old=(id=1, data=2))(AFTER_INSERT: new=(id=1, data=4))(BEFORE_INSERT: new=(id=3, data=3))(AFTER_INSERT: new=(id=3, data=3))
|
|
|
|
drop trigger t1_bd;
|
|
|
|
drop trigger t1_ad;
|
2005-05-24 22:19:33 +04:00
|
|
|
set @log:= "";
|
2006-06-16 20:21:25 +04:00
|
|
|
replace t1 values (1, 5);
|
2005-05-24 22:19:33 +04:00
|
|
|
select @log;
|
|
|
|
@log
|
2006-06-16 20:21:25 +04:00
|
|
|
(BEFORE_INSERT: new=(id=1, data=5))(AFTER_INSERT: new=(id=1, data=5))
|
2005-05-24 22:19:33 +04:00
|
|
|
drop table t1;
|
2005-07-09 21:51:59 +04:00
|
|
|
create table t1 (id int primary key, data varchar(10), fk int);
|
|
|
|
create table t2 (event varchar(100));
|
|
|
|
create table t3 (id int primary key);
|
|
|
|
create trigger t1_ai after insert on t1 for each row
|
|
|
|
insert into t2 values (concat("INSERT INTO t1 id=", new.id, " data='", new.data, "'"));
|
|
|
|
insert into t1 (id, data) values (1, "one"), (2, "two");
|
|
|
|
select * from t1;
|
|
|
|
id data fk
|
|
|
|
1 one NULL
|
|
|
|
2 two NULL
|
|
|
|
select * from t2;
|
|
|
|
event
|
|
|
|
INSERT INTO t1 id=1 data='one'
|
|
|
|
INSERT INTO t1 id=2 data='two'
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger t1_ai;
|
2005-07-09 21:51:59 +04:00
|
|
|
create trigger t1_bi before insert on t1 for each row
|
|
|
|
begin
|
|
|
|
if exists (select id from t3 where id=new.fk) then
|
|
|
|
insert into t2 values (concat("INSERT INTO t1 id=", new.id, " data='", new.data, "' fk=", new.fk));
|
|
|
|
else
|
|
|
|
insert into t2 values (concat("INSERT INTO t1 FAILED id=", new.id, " data='", new.data, "' fk=", new.fk));
|
|
|
|
set new.id= NULL;
|
|
|
|
end if;
|
|
|
|
end|
|
|
|
|
insert into t3 values (1);
|
|
|
|
insert into t1 values (4, "four", 1), (5, "five", 2);
|
|
|
|
ERROR 23000: Column 'id' cannot be null
|
|
|
|
select * from t1;
|
|
|
|
id data fk
|
|
|
|
1 one NULL
|
|
|
|
2 two NULL
|
|
|
|
4 four 1
|
|
|
|
select * from t2;
|
|
|
|
event
|
|
|
|
INSERT INTO t1 id=1 data='one'
|
|
|
|
INSERT INTO t1 id=2 data='two'
|
|
|
|
INSERT INTO t1 id=4 data='four' fk=1
|
|
|
|
INSERT INTO t1 FAILED id=5 data='five' fk=2
|
|
|
|
drop table t1, t2, t3;
|
|
|
|
create table t1 (id int primary key, data varchar(10));
|
|
|
|
create table t2 (seq int);
|
|
|
|
insert into t2 values (10);
|
|
|
|
create function f1 () returns int return (select max(seq) from t2);
|
|
|
|
create trigger t1_bi before insert on t1 for each row
|
|
|
|
begin
|
|
|
|
if new.id > f1() then
|
|
|
|
set new.id:= f1();
|
|
|
|
end if;
|
|
|
|
end|
|
|
|
|
insert into t1 values (1, "first");
|
|
|
|
insert into t1 values (f1(), "max");
|
|
|
|
select * from t1;
|
|
|
|
id data
|
|
|
|
1 first
|
|
|
|
10 max
|
|
|
|
drop table t1, t2;
|
|
|
|
drop function f1;
|
|
|
|
create table t1 (id int primary key, fk_t2 int);
|
|
|
|
create table t2 (id int primary key, fk_t3 int);
|
|
|
|
create table t3 (id int primary key);
|
|
|
|
insert into t1 values (1,1), (2,1), (3,2);
|
|
|
|
insert into t2 values (1,1), (2,2);
|
|
|
|
insert into t3 values (1), (2);
|
|
|
|
create trigger t3_ad after delete on t3 for each row
|
|
|
|
delete from t2 where fk_t3=old.id;
|
|
|
|
create trigger t2_ad after delete on t2 for each row
|
|
|
|
delete from t1 where fk_t2=old.id;
|
|
|
|
delete from t3 where id = 1;
|
|
|
|
select * from t1 left join (t2 left join t3 on t2.fk_t3 = t3.id) on t1.fk_t2 = t2.id;
|
|
|
|
id fk_t2 id fk_t3 id
|
|
|
|
3 2 2 2 2
|
|
|
|
drop table t1, t2, t3;
|
|
|
|
create table t1 (id int primary key, copy int);
|
|
|
|
create table t2 (id int primary key, data int);
|
|
|
|
insert into t2 values (1,1), (2,2);
|
|
|
|
create trigger t1_bi before insert on t1 for each row
|
|
|
|
set new.copy= (select data from t2 where id = new.id);
|
|
|
|
create trigger t1_bu before update on t1 for each row
|
|
|
|
set new.copy= (select data from t2 where id = new.id);
|
|
|
|
insert into t1 values (1,3), (2,4), (3,3);
|
|
|
|
update t1 set copy= 1 where id = 2;
|
|
|
|
select * from t1;
|
|
|
|
id copy
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
3 NULL
|
|
|
|
drop table t1, t2;
|
2004-09-07 16:29:46 +04:00
|
|
|
create table t1 (i int);
|
2005-07-19 20:06:49 +04:00
|
|
|
create table t3 (i int);
|
2004-09-07 16:29:46 +04:00
|
|
|
create trigger trg before insert on t1 for each row set @a:= old.i;
|
|
|
|
ERROR HY000: There is no OLD row in on INSERT trigger
|
|
|
|
create trigger trg before delete on t1 for each row set @a:= new.i;
|
|
|
|
ERROR HY000: There is no NEW row in on DELETE trigger
|
|
|
|
create trigger trg before update on t1 for each row set old.i:=1;
|
|
|
|
ERROR HY000: Updating of OLD row is not allowed in trigger
|
|
|
|
create trigger trg before delete on t1 for each row set new.i:=1;
|
|
|
|
ERROR HY000: There is no NEW row in on DELETE trigger
|
|
|
|
create trigger trg after update on t1 for each row set new.i:=1;
|
|
|
|
ERROR HY000: Updating of NEW row is not allowed in after trigger
|
2004-11-24 12:24:02 +03:00
|
|
|
create trigger trg before update on t1 for each row set new.j:=1;
|
|
|
|
ERROR 42S22: Unknown column 'j' in 'NEW'
|
|
|
|
create trigger trg before update on t1 for each row set @a:=old.j;
|
|
|
|
ERROR 42S22: Unknown column 'j' in 'OLD'
|
2004-09-07 16:29:46 +04:00
|
|
|
create trigger trg before insert on t2 for each row set @a:=1;
|
|
|
|
ERROR 42S02: Table 'test.t2' doesn't exist
|
|
|
|
create trigger trg before insert on t1 for each row set @a:=1;
|
|
|
|
create trigger trg after insert on t1 for each row set @a:=1;
|
|
|
|
ERROR HY000: Trigger already exists
|
|
|
|
create trigger trg2 before insert on t1 for each row set @a:=1;
|
2006-06-28 23:50:50 +04:00
|
|
|
ERROR 42000: This version of MySQL doesn't yet support 'multiple triggers with the same action time and event for one table'
|
2005-07-19 20:06:49 +04:00
|
|
|
create trigger trg before insert on t3 for each row set @a:=1;
|
|
|
|
ERROR HY000: Trigger already exists
|
|
|
|
create trigger trg2 before insert on t3 for each row set @a:=1;
|
|
|
|
drop trigger trg2;
|
|
|
|
drop trigger trg;
|
|
|
|
drop trigger trg;
|
2004-09-07 16:29:46 +04:00
|
|
|
ERROR HY000: Trigger does not exist
|
|
|
|
create view v1 as select * from t1;
|
|
|
|
create trigger trg before insert on v1 for each row set @a:=1;
|
2005-07-13 23:51:09 +04:00
|
|
|
ERROR HY000: 'test.v1' is not BASE TABLE
|
2004-09-07 16:29:46 +04:00
|
|
|
drop view v1;
|
|
|
|
drop table t1;
|
2005-07-19 20:06:49 +04:00
|
|
|
drop table t3;
|
2004-09-07 16:29:46 +04:00
|
|
|
create temporary table t1 (i int);
|
|
|
|
create trigger trg before insert on t1 for each row set @a:=1;
|
|
|
|
ERROR HY000: Trigger's 't1' is view or temporary table
|
|
|
|
drop table t1;
|
2004-10-08 15:16:03 +04:00
|
|
|
create table t1 (x1col char);
|
|
|
|
create trigger tx1 before insert on t1 for each row set new.x1col = 'x';
|
|
|
|
insert into t1 values ('y');
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger tx1;
|
2004-10-08 15:16:03 +04:00
|
|
|
drop table t1;
|
2004-11-12 17:04:07 +03:00
|
|
|
create table t1 (i int) engine=myisam;
|
|
|
|
insert into t1 values (1), (2);
|
|
|
|
create trigger trg1 before delete on t1 for each row set @del_before:= @del_before + old.i;
|
|
|
|
create trigger trg2 after delete on t1 for each row set @del_after:= @del_after + old.i;
|
|
|
|
set @del_before:=0, @del_after:= 0;
|
|
|
|
delete from t1;
|
|
|
|
select @del_before, @del_after;
|
|
|
|
@del_before @del_after
|
|
|
|
3 3
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger trg1;
|
|
|
|
drop trigger trg2;
|
2004-11-12 17:04:07 +03:00
|
|
|
drop table t1;
|
2005-03-27 16:15:21 +04:00
|
|
|
create table t1 (a int);
|
|
|
|
create trigger trg1 before insert on t1 for each row set new.a= 10;
|
|
|
|
drop table t1;
|
|
|
|
create table t1 (a int);
|
|
|
|
insert into t1 values ();
|
|
|
|
select * from t1;
|
|
|
|
a
|
|
|
|
NULL
|
|
|
|
drop table t1;
|
|
|
|
create database mysqltest;
|
|
|
|
use mysqltest;
|
|
|
|
create table t1 (i int);
|
|
|
|
create trigger trg1 before insert on t1 for each row set @a:= 1;
|
|
|
|
drop database mysqltest;
|
|
|
|
use test;
|
2005-07-19 20:06:49 +04:00
|
|
|
create database mysqltest;
|
|
|
|
create table mysqltest.t1 (i int);
|
|
|
|
create trigger trg1 before insert on mysqltest.t1 for each row set @a:= 1;
|
|
|
|
ERROR HY000: Trigger in wrong schema
|
|
|
|
use mysqltest;
|
|
|
|
create trigger test.trg1 before insert on t1 for each row set @a:= 1;
|
A fix and a test case for Bug#29050 Creation of a legal stored procedure
fails if a database is not selected prior.
The problem manifested itself when a user tried to
create a routine that had non-fully-qualified identifiers in its bodies
and there was no current database selected.
This is a regression introduced by the fix for Bug 19022:
The patch for Bug 19022 changes the code to always produce a warning
if we can't resolve the current database in the parser.
In this case this was not necessary, since even though the produced
parsed tree was incorrect, we never re-use sphead
that was obtained at first parsing of CREATE PROCEDURE.
The sphead that is anyhow used is always obtained through db_load_routine,
and there we change the current database to sphead->m_db before
calling yyparse.
The idea of the fix is to resolve the current database directly using
lex->sphead->m_db member when parsing a stored routine body, when
such is present.
This patch removes the need to reset the current database
when loading a trigger or routine definition into SP cache.
The redundant code will be removed in 5.1.
mysql-test/r/sp.result:
Update test results (Bug#29050)
mysql-test/r/trigger.result:
Update results.
mysql-test/t/sp.test:
Add a test case for Bug#29050
mysql-test/t/trigger.test:
Fix wrong behavior covered with tests.
sql/sql_lex.cc:
Implement st_lex::copy_db_to().
sql/sql_lex.h:
Declare st_lex::copy_db_to().
sql/sql_parse.cc:
Use st_lex::copy_db_to() in add_table_to_list, rather than
THD::copy_db_to(). The former will use the database of the sphead,
if we're parsing a stored routine, not the default database in
THD. The default database is needed to initialize tables->db
when the database part was not explicitly specified in the identifier.
sql/sql_yacc.yy:
Use st_lex::copy_db_to() in the parser, rather than
THD::copy_db_to(). The former will use the database of the sphead,
if we're parsing a stored routine, not the default database in
THD.
2007-07-05 11:34:04 +04:00
|
|
|
ERROR 42S02: Table 'test.t1' doesn't exist
|
2005-07-19 20:06:49 +04:00
|
|
|
drop database mysqltest;
|
|
|
|
use test;
|
2005-05-24 22:19:33 +04:00
|
|
|
create table t1 (i int, j int default 10, k int not null, key (k));
|
|
|
|
create table t2 (i int);
|
|
|
|
insert into t1 (i, k) values (1, 1);
|
|
|
|
insert into t2 values (1);
|
|
|
|
create trigger trg1 before update on t1 for each row set @a:= @a + new.j - old.j;
|
|
|
|
create trigger trg2 after update on t1 for each row set @b:= "Fired";
|
|
|
|
set @a:= 0, @b:= "";
|
|
|
|
update t1, t2 set j = j + 10 where t1.i = t2.i;
|
|
|
|
select @a, @b;
|
|
|
|
@a @b
|
|
|
|
10 Fired
|
|
|
|
insert into t1 values (2, 13, 2);
|
|
|
|
insert into t2 values (2);
|
|
|
|
set @a:= 0, @b:= "";
|
|
|
|
update t1, t2 set j = j + 15 where t1.i = t2.i and t1.k >= 2;
|
|
|
|
select @a, @b;
|
|
|
|
@a @b
|
|
|
|
15 Fired
|
|
|
|
create trigger trg3 before delete on t1 for each row set @c:= @c + old.j;
|
|
|
|
create trigger trg4 before delete on t2 for each row set @d:= @d + old.i;
|
|
|
|
create trigger trg5 after delete on t1 for each row set @e:= "After delete t1 fired";
|
|
|
|
create trigger trg6 after delete on t2 for each row set @f:= "After delete t2 fired";
|
|
|
|
set @c:= 0, @d:= 0, @e:= "", @f:= "";
|
|
|
|
delete t1, t2 from t1, t2 where t1.i = t2.i;
|
|
|
|
select @c, @d, @e, @f;
|
|
|
|
@c @d @e @f
|
|
|
|
48 3 After delete t1 fired After delete t2 fired
|
|
|
|
drop table t1, t2;
|
|
|
|
create table t1 (i int, j int default 10)|
|
|
|
|
create table t2 (i int)|
|
|
|
|
insert into t2 values (1), (2)|
|
|
|
|
create trigger trg1 before insert on t1 for each row
|
|
|
|
begin
|
|
|
|
if new.i = 1 then
|
|
|
|
set new.j := 1;
|
|
|
|
end if;
|
|
|
|
end|
|
|
|
|
create trigger trg2 after insert on t1 for each row set @a:= 1|
|
|
|
|
set @a:= 0|
|
|
|
|
insert into t1 (i) select * from t2|
|
|
|
|
select * from t1|
|
|
|
|
i j
|
|
|
|
1 1
|
|
|
|
2 10
|
|
|
|
select @a|
|
|
|
|
@a
|
|
|
|
1
|
|
|
|
drop table t1, t2|
|
|
|
|
create table t1 (i int, j int, k int);
|
|
|
|
create trigger trg1 before insert on t1 for each row set new.k = new.i;
|
|
|
|
create trigger trg2 after insert on t1 for each row set @b:= "Fired";
|
|
|
|
set @b:="";
|
2006-01-24 08:30:54 +01:00
|
|
|
load data infile '../std_data_ln/rpl_loaddata.dat' into table t1 (@a, i);
|
2005-05-24 22:19:33 +04:00
|
|
|
select *, @b from t1;
|
|
|
|
i j k @b
|
|
|
|
10 NULL 10 Fired
|
|
|
|
15 NULL 15 Fired
|
|
|
|
set @b:="";
|
2006-01-24 08:30:54 +01:00
|
|
|
load data infile '../std_data_ln/loaddata5.dat' into table t1 fields terminated by '' enclosed by '' (i, j);
|
2005-05-24 22:19:33 +04:00
|
|
|
select *, @b from t1;
|
|
|
|
i j k @b
|
|
|
|
10 NULL 10 Fired
|
|
|
|
15 NULL 15 Fired
|
|
|
|
1 2 1 Fired
|
|
|
|
3 4 3 Fired
|
|
|
|
5 6 5 Fired
|
|
|
|
drop table t1;
|
2005-05-30 18:55:56 +04:00
|
|
|
create table t1 (i int, at int, k int, key(k)) engine=myisam;
|
|
|
|
create table t2 (i int);
|
|
|
|
insert into t1 values (1, 1, 1);
|
|
|
|
insert into t2 values (1), (2), (3);
|
|
|
|
create trigger ai after insert on t1 for each row set @a:= new.at;
|
|
|
|
create trigger au after update on t1 for each row set @a:= new.at;
|
|
|
|
create trigger ad after delete on t1 for each row set @a:= old.at;
|
|
|
|
alter table t1 drop column at;
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
insert into t1 values (2, 1);
|
|
|
|
ERROR 42S22: Unknown column 'at' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 1
|
|
|
|
update t1 set k = 2 where i = 2;
|
|
|
|
ERROR 42S22: Unknown column 'at' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
delete from t1 where i = 2;
|
|
|
|
ERROR 42S22: Unknown column 'at' in 'OLD'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
2006-01-24 08:30:54 +01:00
|
|
|
load data infile '../std_data_ln/loaddata5.dat' into table t1 fields terminated by '' enclosed by '' (i, k);
|
2005-05-30 18:55:56 +04:00
|
|
|
ERROR 42S22: Unknown column 'at' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
1 2
|
|
|
|
insert into t1 select 3, 3;
|
|
|
|
ERROR 42S22: Unknown column 'at' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
1 2
|
|
|
|
3 3
|
|
|
|
update t1, t2 set k = k + 10 where t1.i = t2.i;
|
|
|
|
ERROR 42S22: Unknown column 'at' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 11
|
|
|
|
1 2
|
|
|
|
3 3
|
|
|
|
update t1, t2 set k = k + 10 where t1.i = t2.i and k < 3;
|
|
|
|
ERROR 42S22: Unknown column 'at' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 11
|
|
|
|
1 12
|
|
|
|
3 3
|
|
|
|
delete t1, t2 from t1 straight_join t2 where t1.i = t2.i;
|
|
|
|
ERROR 42S22: Unknown column 'at' in 'OLD'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 12
|
|
|
|
3 3
|
|
|
|
delete t2, t1 from t2 straight_join t1 where t1.i = t2.i;
|
|
|
|
ERROR 42S22: Unknown column 'at' in 'OLD'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
3 3
|
|
|
|
alter table t1 add primary key (i);
|
|
|
|
insert into t1 values (3, 4) on duplicate key update k= k + 10;
|
|
|
|
ERROR 42S22: Unknown column 'at' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
3 13
|
|
|
|
replace into t1 values (3, 3);
|
|
|
|
ERROR 42S22: Unknown column 'at' in 'OLD'
|
|
|
|
select * from t1;
|
2006-06-16 20:21:25 +04:00
|
|
|
i k
|
2005-05-30 18:55:56 +04:00
|
|
|
drop table t1, t2;
|
|
|
|
create table t1 (i int, bt int, k int, key(k)) engine=myisam;
|
|
|
|
create table t2 (i int);
|
|
|
|
insert into t1 values (1, 1, 1), (2, 2, 2);
|
|
|
|
insert into t2 values (1), (2), (3);
|
|
|
|
create trigger bi before insert on t1 for each row set @a:= new.bt;
|
|
|
|
create trigger bu before update on t1 for each row set @a:= new.bt;
|
|
|
|
create trigger bd before delete on t1 for each row set @a:= old.bt;
|
|
|
|
alter table t1 drop column bt;
|
|
|
|
insert into t1 values (3, 3);
|
|
|
|
ERROR 42S22: Unknown column 'bt' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
update t1 set i = 2;
|
|
|
|
ERROR 42S22: Unknown column 'bt' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
delete from t1;
|
|
|
|
ERROR 42S22: Unknown column 'bt' in 'OLD'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 2
|
2006-01-24 08:30:54 +01:00
|
|
|
load data infile '../std_data_ln/loaddata5.dat' into table t1 fields terminated by '' enclosed by '' (i, k);
|
2005-05-30 18:55:56 +04:00
|
|
|
ERROR 42S22: Unknown column 'bt' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
insert into t1 select 3, 3;
|
|
|
|
ERROR 42S22: Unknown column 'bt' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
update t1, t2 set k = k + 10 where t1.i = t2.i;
|
|
|
|
ERROR 42S22: Unknown column 'bt' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
update t1, t2 set k = k + 10 where t1.i = t2.i and k < 2;
|
|
|
|
ERROR 42S22: Unknown column 'bt' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
delete t1, t2 from t1 straight_join t2 where t1.i = t2.i;
|
|
|
|
ERROR 42S22: Unknown column 'bt' in 'OLD'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
delete t2, t1 from t2 straight_join t1 where t1.i = t2.i;
|
|
|
|
ERROR 42S22: Unknown column 'bt' in 'OLD'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
alter table t1 add primary key (i);
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger bi;
|
2005-05-30 18:55:56 +04:00
|
|
|
insert into t1 values (2, 4) on duplicate key update k= k + 10;
|
|
|
|
ERROR 42S22: Unknown column 'bt' in 'NEW'
|
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
replace into t1 values (2, 4);
|
2006-06-16 20:21:25 +04:00
|
|
|
ERROR 42S22: Unknown column 'bt' in 'OLD'
|
2005-05-30 18:55:56 +04:00
|
|
|
select * from t1;
|
|
|
|
i k
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
drop table t1, t2;
|
2005-07-13 16:22:36 +04:00
|
|
|
drop function if exists bug5893;
|
|
|
|
create table t1 (col1 int, col2 int);
|
|
|
|
insert into t1 values (1, 2);
|
|
|
|
create function bug5893 () returns int return 5;
|
|
|
|
create trigger t1_bu before update on t1 for each row set new.col1= bug5893();
|
|
|
|
drop function bug5893;
|
|
|
|
update t1 set col2 = 4;
|
|
|
|
ERROR 42000: FUNCTION test.bug5893 does not exist
|
2005-07-19 20:06:49 +04:00
|
|
|
drop trigger t1_bu;
|
2005-07-13 16:22:36 +04:00
|
|
|
drop table t1;
|
2005-07-28 22:39:11 +03:00
|
|
|
set sql_mode='ansi';
|
|
|
|
create table t1 ("t1 column" int);
|
|
|
|
create trigger t1_bi before insert on t1 for each row set new."t1 column" = 5;
|
2005-07-30 09:24:51 +03:00
|
|
|
set sql_mode="";
|
2005-07-28 22:39:11 +03:00
|
|
|
insert into t1 values (0);
|
|
|
|
create trigger t1_af after insert on t1 for each row set @a=10;
|
|
|
|
insert into t1 values (0);
|
|
|
|
select * from t1;
|
|
|
|
t1 column
|
|
|
|
5
|
|
|
|
5
|
|
|
|
select @a;
|
|
|
|
@a
|
|
|
|
10
|
|
|
|
show triggers;
|
Patch for the following bugs:
- BUG#11986: Stored routines and triggers can fail if the code
has a non-ascii symbol
- BUG#16291: mysqldump corrupts string-constants with non-ascii-chars
- BUG#19443: INFORMATION_SCHEMA does not support charsets properly
- BUG#21249: Character set of SP-var can be ignored
- BUG#25212: Character set of string constant is ignored (stored routines)
- BUG#25221: Character set of string constant is ignored (triggers)
There were a few general problems that caused these bugs:
1. Character set information of the original (definition) query for views,
triggers, stored routines and events was lost.
2. mysqldump output query in client character set, which can be
inappropriate to encode definition-query.
3. INFORMATION_SCHEMA used strings with mixed encodings to display object
definition;
1. No query-definition-character set.
In order to compile query into execution code, some extra data (such as
environment variables or the database character set) is used. The problem
here was that this context was not preserved. So, on the next load it can
differ from the original one, thus the result will be different.
The context contains the following data:
- client character set;
- connection collation (character set and collation);
- collation of the owner database;
The fix is to store this context and use it each time we parse (compile)
and execute the object (stored routine, trigger, ...).
2. Wrong mysqldump-output.
The original query can contain several encodings (by means of character set
introducers). The problem here was that we tried to convert original query
to the mysqldump-client character set.
Moreover, we stored queries in different character sets for different
objects (views, for one, used UTF8, triggers used original character set).
The solution is
- to store definition queries in the original character set;
- to change SHOW CREATE statement to output definition query in the
binary character set (i.e. without any conversion);
- introduce SHOW CREATE TRIGGER statement;
- to dump special statements to switch the context to the original one
before dumping and restore it afterwards.
Note, in order to preserve the database collation at the creation time,
additional ALTER DATABASE might be used (to temporary switch the database
collation back to the original value). In this case, ALTER DATABASE
privilege will be required. This is a backward-incompatible change.
3. INFORMATION_SCHEMA showed non-UTF8 strings
The fix is to generate UTF8-query during the parsing, store it in the object
and show it in the INFORMATION_SCHEMA.
Basically, the idea is to create a copy of the original query convert it to
UTF8. Character set introducers are removed and all text literals are
converted to UTF8.
This UTF8 query is intended to provide user-readable output. It must not be
used to recreate the object. Specialized SHOW CREATE statements should be
used for this.
The reason for this limitation is the following: the original query can
contain symbols from several character sets (by means of character set
introducers).
Example:
- original query:
CREATE VIEW v1 AS SELECT _cp1251 'Hello' AS c1;
- UTF8 query (for INFORMATION_SCHEMA):
CREATE VIEW v1 AS SELECT 'Hello' AS c1;
client/mysqldump.c:
Set original character set and collation before dumping definition query.
include/my_sys.h:
Move out-parameter to the end of list.
mysql-test/lib/mtr_report.pl:
Ignore server-warnings during the test case.
mysql-test/r/create.result:
Update result file.
mysql-test/r/ctype_cp932_binlog_stm.result:
Update result file.
mysql-test/r/events.result:
Update result file.
mysql-test/r/events_bugs.result:
Update result file.
mysql-test/r/events_grant.result:
Update result file.
mysql-test/r/func_in.result:
Update result file.
mysql-test/r/gis.result:
Update result file.
mysql-test/r/grant.result:
Update result file.
mysql-test/r/information_schema.result:
Update result file.
mysql-test/r/information_schema_db.result:
Update result file.
mysql-test/r/lowercase_view.result:
Update result file.
mysql-test/r/mysqldump.result:
Update result file.
mysql-test/r/ndb_sp.result:
Update result file.
mysql-test/r/ps.result:
Update result file.
mysql-test/r/rpl_replicate_do.result:
Update result file.
mysql-test/r/rpl_sp.result:
Update result file.
mysql-test/r/rpl_trigger.result:
Update result file.
mysql-test/r/rpl_view.result:
Update result file.
mysql-test/r/show_check.result:
Update result file.
mysql-test/r/skip_grants.result:
Update result file.
mysql-test/r/sp-destruct.result:
Update result file.
mysql-test/r/sp-error.result:
Update result file.
mysql-test/r/sp-security.result:
Update result file.
mysql-test/r/sp.result:
Update result file.
mysql-test/r/sql_mode.result:
Update result file.
mysql-test/r/system_mysql_db.result:
Update result file.
mysql-test/r/temp_table.result:
Update result file.
mysql-test/r/trigger-compat.result:
Update result file.
mysql-test/r/trigger-grant.result:
Update result file.
mysql-test/r/trigger.result:
Update result file.
mysql-test/r/view.result:
Update result file.
mysql-test/r/view_grant.result:
Update result file.
mysql-test/t/events.test:
Update test case (new columns added).
mysql-test/t/information_schema.test:
Update test case (new columns added).
mysql-test/t/show_check.test:
Test case for SHOW CREATE TRIGGER in prepared statements and
stored routines.
mysql-test/t/sp-destruct.test:
Update test case (new columns added).
mysql-test/t/sp.test:
Update test case (new columns added).
mysql-test/t/view.test:
Update test.
mysys/charset.c:
Move out-parameter to the end of list.
scripts/mysql_system_tables.sql:
Add new columns to mysql.proc and mysql.event.
scripts/mysql_system_tables_fix.sql:
Add new columns to mysql.proc and mysql.event.
sql/event_data_objects.cc:
Support new attributes for events.
sql/event_data_objects.h:
Support new attributes for events.
sql/event_db_repository.cc:
Support new attributes for events.
sql/event_db_repository.h:
Support new attributes for events.
sql/events.cc:
Add new columns to SHOW CREATE event resultset.
sql/mysql_priv.h:
1. Introduce Object_creation_ctx;
2. Introduce SHOW CREATE TRIGGER;
3. Introduce auxilary functions.
sql/sp.cc:
Add support for new store routines attributes.
sql/sp_head.cc:
Add support for new store routines attributes.
sql/sp_head.h:
Add support for new store routines attributes.
sql/sql_lex.cc:
Generate UTF8-body on parsing/lexing.
sql/sql_lex.h:
1. Generate UTF8-body on parsing/lexing.
2. Introduce SHOW CREATE TRIGGER.
sql/sql_parse.cc:
Introduce SHOW CREATE TRIGGER.
sql/sql_partition.cc:
Update parse_sql().
sql/sql_prepare.cc:
Update parse_sql().
sql/sql_show.cc:
Support new attributes for views
sql/sql_trigger.cc:
Support new attributes for views
sql/sql_trigger.h:
Support new attributes for views
sql/sql_view.cc:
Support new attributes for views
sql/sql_yacc.yy:
1. Add SHOW CREATE TRIGGER statement.
2. Generate UTF8-body for views, stored routines, triggers and events.
sql/table.cc:
Introduce Object_creation_ctx.
sql/table.h:
Introduce Object_creation_ctx.
sql/share/errmsg.txt:
Add new errors.
mysql-test/include/ddl_i18n.check_events.inc:
Aux file for test suite.
mysql-test/include/ddl_i18n.check_sp.inc:
Aux file for test suite.
mysql-test/include/ddl_i18n.check_triggers.inc:
Aux file for test suite.
mysql-test/include/ddl_i18n.check_views.inc:
Aux file for test suite.
mysql-test/include/have_cp1251.inc:
Aux file for test suite.
mysql-test/include/have_cp866.inc:
Aux file for test suite.
mysql-test/include/have_koi8r.inc:
Aux file for test suite.
mysql-test/include/have_utf8.inc:
Aux file for test suite.
mysql-test/r/ddl_i18n_koi8r.result:
Result file.
mysql-test/r/ddl_i18n_utf8.result:
Result file.
mysql-test/r/have_cp1251.require:
Aux file for test suite.
mysql-test/r/have_cp866.require:
Aux file for test suite.
mysql-test/r/have_koi8r.require:
Aux file for test suite.
mysql-test/r/have_utf8.require:
Aux file for test suite.
mysql-test/t/ddl_i18n_koi8r.test:
Complete koi8r test case for the CS patch.
mysql-test/t/ddl_i18n_utf8.test:
Complete utf8 test case for the CS patch.
2007-06-28 21:34:54 +04:00
|
|
|
Trigger Event Table Statement Timing Created sql_mode Definer character_set_client collation_connection Database Collation
|
|
|
|
t1_bi INSERT t1 set new."t1 column" = 5 BEFORE # REAL_AS_FLOAT,PIPES_AS_CONCAT,ANSI_QUOTES,IGNORE_SPACE,ANSI root@localhost latin1 latin1_swedish_ci latin1_swedish_ci
|
|
|
|
t1_af INSERT t1 set @a=10 AFTER # root@localhost latin1 latin1_swedish_ci latin1_swedish_ci
|
2005-07-28 22:39:11 +03:00
|
|
|
drop table t1;
|
|
|
|
set sql_mode="traditional";
|
|
|
|
create table t1 (a date);
|
|
|
|
insert into t1 values ('2004-01-00');
|
|
|
|
ERROR 22007: Incorrect date value: '2004-01-00' for column 'a' at row 1
|
|
|
|
set sql_mode="";
|
|
|
|
create trigger t1_bi before insert on t1 for each row set new.a = '2004-01-00';
|
|
|
|
set sql_mode="traditional";
|
|
|
|
insert into t1 values ('2004-01-01');
|
|
|
|
select * from t1;
|
|
|
|
a
|
|
|
|
2004-01-00
|
|
|
|
set sql_mode=default;
|
|
|
|
show create table t1;
|
|
|
|
Table Create Table
|
|
|
|
t1 CREATE TABLE `t1` (
|
2006-02-22 10:09:59 +01:00
|
|
|
`a` date DEFAULT NULL
|
2005-07-28 22:39:11 +03:00
|
|
|
) ENGINE=MyISAM DEFAULT CHARSET=latin1
|
|
|
|
show triggers;
|
Patch for the following bugs:
- BUG#11986: Stored routines and triggers can fail if the code
has a non-ascii symbol
- BUG#16291: mysqldump corrupts string-constants with non-ascii-chars
- BUG#19443: INFORMATION_SCHEMA does not support charsets properly
- BUG#21249: Character set of SP-var can be ignored
- BUG#25212: Character set of string constant is ignored (stored routines)
- BUG#25221: Character set of string constant is ignored (triggers)
There were a few general problems that caused these bugs:
1. Character set information of the original (definition) query for views,
triggers, stored routines and events was lost.
2. mysqldump output query in client character set, which can be
inappropriate to encode definition-query.
3. INFORMATION_SCHEMA used strings with mixed encodings to display object
definition;
1. No query-definition-character set.
In order to compile query into execution code, some extra data (such as
environment variables or the database character set) is used. The problem
here was that this context was not preserved. So, on the next load it can
differ from the original one, thus the result will be different.
The context contains the following data:
- client character set;
- connection collation (character set and collation);
- collation of the owner database;
The fix is to store this context and use it each time we parse (compile)
and execute the object (stored routine, trigger, ...).
2. Wrong mysqldump-output.
The original query can contain several encodings (by means of character set
introducers). The problem here was that we tried to convert original query
to the mysqldump-client character set.
Moreover, we stored queries in different character sets for different
objects (views, for one, used UTF8, triggers used original character set).
The solution is
- to store definition queries in the original character set;
- to change SHOW CREATE statement to output definition query in the
binary character set (i.e. without any conversion);
- introduce SHOW CREATE TRIGGER statement;
- to dump special statements to switch the context to the original one
before dumping and restore it afterwards.
Note, in order to preserve the database collation at the creation time,
additional ALTER DATABASE might be used (to temporary switch the database
collation back to the original value). In this case, ALTER DATABASE
privilege will be required. This is a backward-incompatible change.
3. INFORMATION_SCHEMA showed non-UTF8 strings
The fix is to generate UTF8-query during the parsing, store it in the object
and show it in the INFORMATION_SCHEMA.
Basically, the idea is to create a copy of the original query convert it to
UTF8. Character set introducers are removed and all text literals are
converted to UTF8.
This UTF8 query is intended to provide user-readable output. It must not be
used to recreate the object. Specialized SHOW CREATE statements should be
used for this.
The reason for this limitation is the following: the original query can
contain symbols from several character sets (by means of character set
introducers).
Example:
- original query:
CREATE VIEW v1 AS SELECT _cp1251 'Hello' AS c1;
- UTF8 query (for INFORMATION_SCHEMA):
CREATE VIEW v1 AS SELECT 'Hello' AS c1;
client/mysqldump.c:
Set original character set and collation before dumping definition query.
include/my_sys.h:
Move out-parameter to the end of list.
mysql-test/lib/mtr_report.pl:
Ignore server-warnings during the test case.
mysql-test/r/create.result:
Update result file.
mysql-test/r/ctype_cp932_binlog_stm.result:
Update result file.
mysql-test/r/events.result:
Update result file.
mysql-test/r/events_bugs.result:
Update result file.
mysql-test/r/events_grant.result:
Update result file.
mysql-test/r/func_in.result:
Update result file.
mysql-test/r/gis.result:
Update result file.
mysql-test/r/grant.result:
Update result file.
mysql-test/r/information_schema.result:
Update result file.
mysql-test/r/information_schema_db.result:
Update result file.
mysql-test/r/lowercase_view.result:
Update result file.
mysql-test/r/mysqldump.result:
Update result file.
mysql-test/r/ndb_sp.result:
Update result file.
mysql-test/r/ps.result:
Update result file.
mysql-test/r/rpl_replicate_do.result:
Update result file.
mysql-test/r/rpl_sp.result:
Update result file.
mysql-test/r/rpl_trigger.result:
Update result file.
mysql-test/r/rpl_view.result:
Update result file.
mysql-test/r/show_check.result:
Update result file.
mysql-test/r/skip_grants.result:
Update result file.
mysql-test/r/sp-destruct.result:
Update result file.
mysql-test/r/sp-error.result:
Update result file.
mysql-test/r/sp-security.result:
Update result file.
mysql-test/r/sp.result:
Update result file.
mysql-test/r/sql_mode.result:
Update result file.
mysql-test/r/system_mysql_db.result:
Update result file.
mysql-test/r/temp_table.result:
Update result file.
mysql-test/r/trigger-compat.result:
Update result file.
mysql-test/r/trigger-grant.result:
Update result file.
mysql-test/r/trigger.result:
Update result file.
mysql-test/r/view.result:
Update result file.
mysql-test/r/view_grant.result:
Update result file.
mysql-test/t/events.test:
Update test case (new columns added).
mysql-test/t/information_schema.test:
Update test case (new columns added).
mysql-test/t/show_check.test:
Test case for SHOW CREATE TRIGGER in prepared statements and
stored routines.
mysql-test/t/sp-destruct.test:
Update test case (new columns added).
mysql-test/t/sp.test:
Update test case (new columns added).
mysql-test/t/view.test:
Update test.
mysys/charset.c:
Move out-parameter to the end of list.
scripts/mysql_system_tables.sql:
Add new columns to mysql.proc and mysql.event.
scripts/mysql_system_tables_fix.sql:
Add new columns to mysql.proc and mysql.event.
sql/event_data_objects.cc:
Support new attributes for events.
sql/event_data_objects.h:
Support new attributes for events.
sql/event_db_repository.cc:
Support new attributes for events.
sql/event_db_repository.h:
Support new attributes for events.
sql/events.cc:
Add new columns to SHOW CREATE event resultset.
sql/mysql_priv.h:
1. Introduce Object_creation_ctx;
2. Introduce SHOW CREATE TRIGGER;
3. Introduce auxilary functions.
sql/sp.cc:
Add support for new store routines attributes.
sql/sp_head.cc:
Add support for new store routines attributes.
sql/sp_head.h:
Add support for new store routines attributes.
sql/sql_lex.cc:
Generate UTF8-body on parsing/lexing.
sql/sql_lex.h:
1. Generate UTF8-body on parsing/lexing.
2. Introduce SHOW CREATE TRIGGER.
sql/sql_parse.cc:
Introduce SHOW CREATE TRIGGER.
sql/sql_partition.cc:
Update parse_sql().
sql/sql_prepare.cc:
Update parse_sql().
sql/sql_show.cc:
Support new attributes for views
sql/sql_trigger.cc:
Support new attributes for views
sql/sql_trigger.h:
Support new attributes for views
sql/sql_view.cc:
Support new attributes for views
sql/sql_yacc.yy:
1. Add SHOW CREATE TRIGGER statement.
2. Generate UTF8-body for views, stored routines, triggers and events.
sql/table.cc:
Introduce Object_creation_ctx.
sql/table.h:
Introduce Object_creation_ctx.
sql/share/errmsg.txt:
Add new errors.
mysql-test/include/ddl_i18n.check_events.inc:
Aux file for test suite.
mysql-test/include/ddl_i18n.check_sp.inc:
Aux file for test suite.
mysql-test/include/ddl_i18n.check_triggers.inc:
Aux file for test suite.
mysql-test/include/ddl_i18n.check_views.inc:
Aux file for test suite.
mysql-test/include/have_cp1251.inc:
Aux file for test suite.
mysql-test/include/have_cp866.inc:
Aux file for test suite.
mysql-test/include/have_koi8r.inc:
Aux file for test suite.
mysql-test/include/have_utf8.inc:
Aux file for test suite.
mysql-test/r/ddl_i18n_koi8r.result:
Result file.
mysql-test/r/ddl_i18n_utf8.result:
Result file.
mysql-test/r/have_cp1251.require:
Aux file for test suite.
mysql-test/r/have_cp866.require:
Aux file for test suite.
mysql-test/r/have_koi8r.require:
Aux file for test suite.
mysql-test/r/have_utf8.require:
Aux file for test suite.
mysql-test/t/ddl_i18n_koi8r.test:
Complete koi8r test case for the CS patch.
mysql-test/t/ddl_i18n_utf8.test:
Complete utf8 test case for the CS patch.
2007-06-28 21:34:54 +04:00
|
|
|
Trigger Event Table Statement Timing Created sql_mode Definer character_set_client collation_connection Database Collation
|
|
|
|
t1_bi INSERT t1 set new.a = '2004-01-00' BEFORE # root@localhost latin1 latin1_swedish_ci latin1_swedish_ci
|
2005-07-28 22:39:11 +03:00
|
|
|
drop table t1;
|
2005-08-10 10:31:32 +04:00
|
|
|
create table t1 (id int);
|
2006-08-25 15:51:29 +02:00
|
|
|
create trigger t1_ai after insert on t1 for each row reset query cache;
|
|
|
|
ERROR 0A000: RESET is not allowed in stored function or trigger
|
|
|
|
create trigger t1_ai after insert on t1 for each row reset master;
|
|
|
|
ERROR 0A000: RESET is not allowed in stored function or trigger
|
|
|
|
create trigger t1_ai after insert on t1 for each row reset slave;
|
|
|
|
ERROR 0A000: RESET is not allowed in stored function or trigger
|
|
|
|
create trigger t1_ai after insert on t1 for each row flush hosts;
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
create trigger t1_ai after insert on t1 for each row flush tables with read lock;
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
create trigger t1_ai after insert on t1 for each row flush logs;
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
create trigger t1_ai after insert on t1 for each row flush status;
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
create trigger t1_ai after insert on t1 for each row flush slave;
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
create trigger t1_ai after insert on t1 for each row flush master;
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
create trigger t1_ai after insert on t1 for each row flush des_key_file;
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
create trigger t1_ai after insert on t1 for each row flush user_resources;
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
2005-08-10 10:31:32 +04:00
|
|
|
create trigger t1_ai after insert on t1 for each row flush tables;
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
create trigger t1_ai after insert on t1 for each row flush privileges;
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
2006-08-25 15:51:29 +02:00
|
|
|
drop procedure if exists p1;
|
2005-08-10 10:31:32 +04:00
|
|
|
create trigger t1_ai after insert on t1 for each row call p1();
|
2006-08-25 15:51:29 +02:00
|
|
|
create procedure p1() flush tables;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
|
|
|
create procedure p1() reset query cache;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: RESET is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
|
|
|
create procedure p1() reset master;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: RESET is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
|
|
|
create procedure p1() reset slave;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: RESET is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
|
|
|
create procedure p1() flush hosts;
|
2005-08-10 10:31:32 +04:00
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
|
|
|
create procedure p1() flush privileges;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
2006-08-25 15:51:29 +02:00
|
|
|
create procedure p1() flush tables with read lock;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
|
|
|
create procedure p1() flush tables;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
|
|
|
create procedure p1() flush logs;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
|
|
|
create procedure p1() flush status;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
|
|
|
create procedure p1() flush slave;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
|
|
|
create procedure p1() flush master;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
|
|
|
create procedure p1() flush des_key_file;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
|
|
|
create procedure p1() flush user_resources;
|
|
|
|
insert into t1 values (0);
|
|
|
|
ERROR 0A000: FLUSH is not allowed in stored function or trigger
|
|
|
|
drop procedure p1;
|
2005-08-10 10:31:32 +04:00
|
|
|
drop table t1;
|
2005-08-10 00:23:56 +04:00
|
|
|
create table t1 (id int, data int, username varchar(16));
|
|
|
|
insert into t1 (id, data) values (1, 0);
|
|
|
|
create trigger t1_whoupdated before update on t1 for each row
|
|
|
|
begin
|
|
|
|
declare user varchar(32);
|
|
|
|
declare i int;
|
|
|
|
select user() into user;
|
|
|
|
set NEW.username = user;
|
|
|
|
select count(*) from ((select 1) union (select 2)) as d1 into i;
|
|
|
|
end|
|
|
|
|
update t1 set data = 1;
|
|
|
|
update t1 set data = 2;
|
|
|
|
drop table t1;
|
2005-08-15 18:15:12 +03:00
|
|
|
create table t1 (c1 int, c2 datetime);
|
|
|
|
create trigger tr1 before insert on t1 for each row
|
|
|
|
begin
|
|
|
|
set new.c2= '2004-04-01';
|
|
|
|
select 'hello';
|
|
|
|
end|
|
|
|
|
ERROR 0A000: Not allowed to return a result set from a trigger
|
|
|
|
insert into t1 (c1) values (1),(2),(3);
|
|
|
|
select * from t1;
|
|
|
|
c1 c2
|
|
|
|
1 NULL
|
|
|
|
2 NULL
|
|
|
|
3 NULL
|
|
|
|
drop procedure if exists bug11587;
|
|
|
|
create procedure bug11587(x char(16))
|
|
|
|
begin
|
|
|
|
select "hello";
|
|
|
|
select "hello again";
|
|
|
|
end|
|
|
|
|
create trigger tr1 before insert on t1 for each row
|
|
|
|
begin
|
|
|
|
call bug11587();
|
|
|
|
set new.c2= '2004-04-02';
|
|
|
|
end|
|
|
|
|
insert into t1 (c1) values (4),(5),(6);
|
Implement WL#2661 "Prepared Statements: Dynamic SQL in Stored Procedures".
The idea of the patch is to separate statement processing logic,
such as parsing, validation of the parsed tree, execution and cleanup,
from global query processing logic, such as logging, resetting
priorities of a thread, resetting stored procedure cache, resetting
thread count of errors and warnings.
This makes PREPARE and EXECUTE behave similarly to the rest of SQL
statements and allows their use in stored procedures.
This patch contains a change in behaviour:
until recently for each SQL prepared statement command, 2 queries
were written to the general log, e.g.
[Query] prepare stmt from @stmt_text;
[Prepare] select * from t1 <-- contents of @stmt_text
The chagne was necessary to prevent [Prepare] commands from being written
to the general log when executing a stored procedure with Dynamic SQL.
We should consider whether the old behavior is preferrable and probably
restore it.
This patch refixes Bug#7115, Bug#10975 (partially), Bug#10605 (various bugs
in Dynamic SQL reported before it was disabled).
mysql-test/r/not_embedded_server.result:
Since we don't want to log Dynamic SQL in stored procedures,
now the general log gets only one log entry per SQL statement.
mysql-test/r/sp-error.result:
- remove obsolete tests
- a better error message for the case when a stored procedure that
returns a result set is called from a function
mysql-test/r/trigger.result:
- a better error message for the case when a stored procedure that
returns a result set is called from a trigger
mysql-test/t/sp-error.test:
- a better error message for the case when a stored procedure that
returns a result set is called from a function.
- move the comment to its place (end of file).
mysql-test/t/trigger.test:
- a better error message for the case when a stored procedure that
returns a result set is called from a trigger
sql/item_func.cc:
- we need to pass sql_command explicitly to get_var_with_binlog, because
when creating a query for SQL prepared statement thd->lex->sql_command
points at SQLCOM_EXECUTE, which is not listed in the list of update
queries.
sql/log_event.h:
- remove an extra copy of the previous sentence
sql/mysql_priv.h:
- fix declarations of sql_prepare.cc API
sql/share/errmsg.txt:
- a new error message, when one attempts to execute a prepared statement
which is currently being executed (this can happen only in Dynamic SQL
at the moment).
sql/sp_head.cc:
- extend sp_multi_results_command to return different flags for a
command (and rename it)
- add support for SQLCOM_PREPARE,SQLCOM_EXECUTE, SQLCOM_DEALLOCATE
to sp_get_flags_for_command
- replace multiple boolean sp_head members with uint m_flags
- a fix for a crash when user variables are used in a stored procedure
and binlog is on. A temporary fix for Bug#12637 "SP crashes the server
if it has update query with user var & binlog is enabled", which actually
stands for stored functions: now instead of a crash we break
replication if a user variable is used in a stored function which
is executed in prelocked mode.
sql/sp_head.h:
- replace multiple boolean flags of sp_head with uint m_flags;
- add flag CONTAINS_DYNAMIC_SQL
- use this flag to error if a stored procedure with Dynamic SQL is
called from a function or trigger.
sql/sql_class.cc:
- Statement_map::insert should not delete a statement if it exists,
now it's done externally to be able to handle the case when the
statement being deleted is in use.
- remove extra code (free_list is already reset in free_items)
sql/sql_lex.cc:
- add lex->stmt_prepare_mode; we can't rely on thd->command any more,
because we don't reset it any more (Dynamic SQL requirement is that
PS are as little intrusive as possible).
sql/sql_lex.h:
- declare bool LEX::stmt_prepare_mode
sql/sql_parse.cc:
- move prepared statement code to sql_prepare.cc
- change declarations (refactored code)
- better error message when one attempts to use Dynamic SQL or a
stored procedure that returns a result set in a function or trigger.
sql/sql_prepare.cc:
- major refactoring to ensure PREPARE/EXECUTE commands do not reset global THD
state and allow their use in stored procedures.
- add Prepared_statement::flags and use it to ensure no recursive execution
of a prepared statement is possible
- better comments
sql/sql_yacc.yy:
- enable PREPARE/EXECUTE/DEALLOCATE in stored procedures
- produce an error message on attempt to use PREPARE/EXECUTE/DEALLOCATE
in a stored function or trigger
mysql-test/r/sp-dynamic.result:
- sp-dynamic.test results
mysql-test/t/sp-dynamic.test:
- a new test for PREPARE/EXECUTE/DEALLOCATE in stored procedures.
2005-09-03 03:13:18 +04:00
|
|
|
ERROR 0A000: Not allowed to return a result set from a trigger
|
2005-08-15 18:15:12 +03:00
|
|
|
select * from t1;
|
|
|
|
c1 c2
|
|
|
|
1 NULL
|
|
|
|
2 NULL
|
|
|
|
3 NULL
|
|
|
|
drop procedure bug11587;
|
|
|
|
drop table t1;
|
2005-08-18 19:07:23 +04:00
|
|
|
create table t1 (f1 integer);
|
|
|
|
create table t2 (f2 integer);
|
|
|
|
create trigger t1_ai after insert on t1
|
|
|
|
for each row insert into t2 values (new.f1+1);
|
|
|
|
create trigger t2_ai after insert on t2
|
|
|
|
for each row insert into t1 values (new.f2+1);
|
2005-11-23 01:11:19 +02:00
|
|
|
set @SAVE_SP_RECURSION_LEVELS=@@max_sp_recursion_depth;
|
|
|
|
set @@max_sp_recursion_depth=100;
|
2005-08-18 19:07:23 +04:00
|
|
|
insert into t1 values (1);
|
|
|
|
ERROR HY000: Can't update table 't1' in stored function/trigger because it is already used by statement which invoked this stored function/trigger.
|
2005-11-23 01:11:19 +02:00
|
|
|
set @@max_sp_recursion_depth=@SAVE_SP_RECURSION_LEVELS;
|
2005-08-18 19:07:23 +04:00
|
|
|
select * from t1;
|
|
|
|
f1
|
|
|
|
1
|
|
|
|
select * from t2;
|
|
|
|
f2
|
|
|
|
2
|
|
|
|
drop trigger t1_ai;
|
|
|
|
drop trigger t2_ai;
|
|
|
|
create trigger t1_bu before update on t1
|
|
|
|
for each row insert into t1 values (2);
|
|
|
|
update t1 set f1= 10;
|
|
|
|
ERROR HY000: Can't update table 't1' in stored function/trigger because it is already used by statement which invoked this stored function/trigger.
|
|
|
|
select * from t1;
|
|
|
|
f1
|
|
|
|
1
|
|
|
|
drop trigger t1_bu;
|
|
|
|
create trigger t1_bu before update on t1
|
|
|
|
for each row delete from t1 where f1=new.f1;
|
|
|
|
update t1 set f1= 10;
|
|
|
|
ERROR HY000: Can't update table 't1' in stored function/trigger because it is already used by statement which invoked this stored function/trigger.
|
|
|
|
select * from t1;
|
|
|
|
f1
|
|
|
|
1
|
|
|
|
drop trigger t1_bu;
|
|
|
|
create trigger t1_bi before insert on t1
|
|
|
|
for each row set new.f1=(select sum(f1) from t1);
|
|
|
|
insert into t1 values (3);
|
|
|
|
select * from t1;
|
|
|
|
f1
|
|
|
|
1
|
|
|
|
1
|
|
|
|
drop trigger t1_bi;
|
|
|
|
drop tables t1, t2;
|
2005-11-17 03:51:14 +03:00
|
|
|
create table t1 (id int);
|
|
|
|
create table t2 (id int);
|
|
|
|
create trigger t1_bi before insert on t1 for each row insert into t2 values (new.id);
|
|
|
|
prepare stmt1 from "insert into t1 values (10)";
|
|
|
|
create procedure p1() insert into t1 values (10);
|
|
|
|
call p1();
|
|
|
|
drop trigger t1_bi;
|
|
|
|
execute stmt1;
|
|
|
|
call p1();
|
|
|
|
deallocate prepare stmt1;
|
|
|
|
drop procedure p1;
|
|
|
|
create table t3 (id int);
|
|
|
|
create trigger t1_bi after insert on t1 for each row insert into t2 values (new.id);
|
|
|
|
prepare stmt1 from "insert into t1 values (10)";
|
|
|
|
create procedure p1() insert into t1 values (10);
|
|
|
|
call p1();
|
|
|
|
drop trigger t1_bi;
|
|
|
|
create trigger t1_bi after insert on t1 for each row insert into t3 values (new.id);
|
|
|
|
execute stmt1;
|
2007-05-18 12:29:06 +04:00
|
|
|
ERROR 42S02: Table 'test.t3' doesn't exist
|
2005-11-17 03:51:14 +03:00
|
|
|
call p1();
|
2007-05-18 12:29:06 +04:00
|
|
|
ERROR 42S02: Table 'test.t3' doesn't exist
|
2005-11-17 03:51:14 +03:00
|
|
|
deallocate prepare stmt1;
|
|
|
|
drop procedure p1;
|
|
|
|
drop table t1, t2, t3;
|
2005-11-23 00:50:37 +02:00
|
|
|
create table t1 (a int);
|
2006-01-06 00:47:49 +02:00
|
|
|
CREATE PROCEDURE `p1`()
|
2005-11-23 00:50:37 +02:00
|
|
|
begin
|
|
|
|
insert into t1 values (1);
|
|
|
|
end//
|
|
|
|
create trigger trg before insert on t1 for each row
|
|
|
|
begin
|
|
|
|
declare done int default 0;
|
|
|
|
set done= not done;
|
|
|
|
end//
|
2006-01-06 00:47:49 +02:00
|
|
|
CALL p1();
|
|
|
|
drop procedure p1;
|
2005-11-23 00:50:37 +02:00
|
|
|
drop table t1;
|
2005-12-11 15:26:15 +03:00
|
|
|
create trigger t1_bi before insert on test.t1 for each row set @a:=0;
|
|
|
|
ERROR 3D000: No database selected
|
|
|
|
create trigger test.t1_bi before insert on t1 for each row set @a:=0;
|
A fix and a test case for Bug#29050 Creation of a legal stored procedure
fails if a database is not selected prior.
The problem manifested itself when a user tried to
create a routine that had non-fully-qualified identifiers in its bodies
and there was no current database selected.
This is a regression introduced by the fix for Bug 19022:
The patch for Bug 19022 changes the code to always produce a warning
if we can't resolve the current database in the parser.
In this case this was not necessary, since even though the produced
parsed tree was incorrect, we never re-use sphead
that was obtained at first parsing of CREATE PROCEDURE.
The sphead that is anyhow used is always obtained through db_load_routine,
and there we change the current database to sphead->m_db before
calling yyparse.
The idea of the fix is to resolve the current database directly using
lex->sphead->m_db member when parsing a stored routine body, when
such is present.
This patch removes the need to reset the current database
when loading a trigger or routine definition into SP cache.
The redundant code will be removed in 5.1.
mysql-test/r/sp.result:
Update test results (Bug#29050)
mysql-test/r/trigger.result:
Update results.
mysql-test/t/sp.test:
Add a test case for Bug#29050
mysql-test/t/trigger.test:
Fix wrong behavior covered with tests.
sql/sql_lex.cc:
Implement st_lex::copy_db_to().
sql/sql_lex.h:
Declare st_lex::copy_db_to().
sql/sql_parse.cc:
Use st_lex::copy_db_to() in add_table_to_list, rather than
THD::copy_db_to(). The former will use the database of the sphead,
if we're parsing a stored routine, not the default database in
THD. The default database is needed to initialize tables->db
when the database part was not explicitly specified in the identifier.
sql/sql_yacc.yy:
Use st_lex::copy_db_to() in the parser, rather than
THD::copy_db_to(). The former will use the database of the sphead,
if we're parsing a stored routine, not the default database in
THD.
2007-07-05 11:34:04 +04:00
|
|
|
ERROR 42S02: Table 'test.t1' doesn't exist
|
2005-12-11 15:26:15 +03:00
|
|
|
drop trigger t1_bi;
|
|
|
|
ERROR 3D000: No database selected
|
2006-02-24 23:50:36 +03:00
|
|
|
create table t1 (id int);
|
|
|
|
create trigger t1_bi before insert on t1 for each row set @a:=new.id;
|
2006-03-04 16:55:06 +03:00
|
|
|
create trigger t1_ai after insert on test.t1 for each row set @b:=new.id;
|
2006-02-24 23:50:36 +03:00
|
|
|
insert into t1 values (101);
|
2006-03-04 16:55:06 +03:00
|
|
|
select @a, @b;
|
|
|
|
@a @b
|
|
|
|
101 101
|
2006-02-24 23:50:36 +03:00
|
|
|
select trigger_schema, trigger_name, event_object_schema,
|
|
|
|
event_object_table, action_statement from information_schema.triggers
|
|
|
|
where event_object_schema = 'test';
|
|
|
|
trigger_schema trigger_name event_object_schema event_object_table action_statement
|
2006-02-26 20:25:24 +03:00
|
|
|
test t1_bi test t1 set @a:=new.id
|
2006-03-04 16:55:06 +03:00
|
|
|
test t1_ai test t1 set @b:=new.id
|
2006-02-24 23:50:36 +03:00
|
|
|
rename table t1 to t2;
|
|
|
|
insert into t2 values (102);
|
2006-03-04 16:55:06 +03:00
|
|
|
select @a, @b;
|
|
|
|
@a @b
|
|
|
|
102 102
|
2006-02-24 23:50:36 +03:00
|
|
|
select trigger_schema, trigger_name, event_object_schema,
|
|
|
|
event_object_table, action_statement from information_schema.triggers
|
|
|
|
where event_object_schema = 'test';
|
|
|
|
trigger_schema trigger_name event_object_schema event_object_table action_statement
|
2006-02-26 20:25:24 +03:00
|
|
|
test t1_bi test t2 set @a:=new.id
|
2006-03-04 16:55:06 +03:00
|
|
|
test t1_ai test t2 set @b:=new.id
|
2006-02-24 23:50:36 +03:00
|
|
|
alter table t2 rename to t3;
|
|
|
|
insert into t3 values (103);
|
2006-03-04 16:55:06 +03:00
|
|
|
select @a, @b;
|
|
|
|
@a @b
|
|
|
|
103 103
|
2006-02-24 23:50:36 +03:00
|
|
|
select trigger_schema, trigger_name, event_object_schema,
|
|
|
|
event_object_table, action_statement from information_schema.triggers
|
|
|
|
where event_object_schema = 'test';
|
|
|
|
trigger_schema trigger_name event_object_schema event_object_table action_statement
|
2006-02-26 20:25:24 +03:00
|
|
|
test t1_bi test t3 set @a:=new.id
|
2006-03-04 16:55:06 +03:00
|
|
|
test t1_ai test t3 set @b:=new.id
|
2006-02-24 23:50:36 +03:00
|
|
|
alter table t3 rename to t4, add column val int default 0;
|
|
|
|
insert into t4 values (104, 1);
|
2006-03-04 16:55:06 +03:00
|
|
|
select @a, @b;
|
|
|
|
@a @b
|
|
|
|
104 104
|
2006-02-24 23:50:36 +03:00
|
|
|
select trigger_schema, trigger_name, event_object_schema,
|
|
|
|
event_object_table, action_statement from information_schema.triggers
|
|
|
|
where event_object_schema = 'test';
|
|
|
|
trigger_schema trigger_name event_object_schema event_object_table action_statement
|
2006-02-26 20:25:24 +03:00
|
|
|
test t1_bi test t4 set @a:=new.id
|
2006-03-04 16:55:06 +03:00
|
|
|
test t1_ai test t4 set @b:=new.id
|
2006-02-24 23:50:36 +03:00
|
|
|
drop trigger t1_bi;
|
2006-03-04 16:55:06 +03:00
|
|
|
drop trigger t1_ai;
|
2006-02-24 23:50:36 +03:00
|
|
|
drop table t4;
|
|
|
|
create database mysqltest;
|
|
|
|
use mysqltest;
|
|
|
|
create table t1 (id int);
|
|
|
|
create trigger t1_bi before insert on t1 for each row set @a:=new.id;
|
|
|
|
insert into t1 values (101);
|
|
|
|
select @a;
|
|
|
|
@a
|
|
|
|
101
|
|
|
|
select trigger_schema, trigger_name, event_object_schema,
|
|
|
|
event_object_table, action_statement from information_schema.triggers
|
|
|
|
where event_object_schema = 'test' or event_object_schema = 'mysqltest';
|
|
|
|
trigger_schema trigger_name event_object_schema event_object_table action_statement
|
2006-02-26 20:25:24 +03:00
|
|
|
mysqltest t1_bi mysqltest t1 set @a:=new.id
|
2006-02-24 23:50:36 +03:00
|
|
|
rename table t1 to test.t2;
|
|
|
|
ERROR HY000: Trigger in wrong schema
|
|
|
|
insert into t1 values (102);
|
|
|
|
select @a;
|
|
|
|
@a
|
|
|
|
102
|
|
|
|
select trigger_schema, trigger_name, event_object_schema,
|
|
|
|
event_object_table, action_statement from information_schema.triggers
|
|
|
|
where event_object_schema = 'test' or event_object_schema = 'mysqltest';
|
|
|
|
trigger_schema trigger_name event_object_schema event_object_table action_statement
|
2006-02-26 20:25:24 +03:00
|
|
|
mysqltest t1_bi mysqltest t1 set @a:=new.id
|
2006-02-24 23:50:36 +03:00
|
|
|
drop trigger test.t1_bi;
|
|
|
|
ERROR HY000: Trigger does not exist
|
2006-03-24 14:58:18 +03:00
|
|
|
alter table t1 rename to test.t1;
|
|
|
|
ERROR HY000: Trigger in wrong schema
|
|
|
|
insert into t1 values (103);
|
|
|
|
select @a;
|
|
|
|
@a
|
|
|
|
103
|
|
|
|
select trigger_schema, trigger_name, event_object_schema,
|
|
|
|
event_object_table, action_statement from information_schema.triggers
|
|
|
|
where event_object_schema = 'test' or event_object_schema = 'mysqltest';
|
|
|
|
trigger_schema trigger_name event_object_schema event_object_table action_statement
|
|
|
|
mysqltest t1_bi mysqltest t1 set @a:=new.id
|
|
|
|
drop trigger test.t1_bi;
|
|
|
|
ERROR HY000: Trigger does not exist
|
|
|
|
alter table t1 rename to test.t1, add column val int default 0;
|
|
|
|
ERROR HY000: Trigger in wrong schema
|
|
|
|
insert into t1 values (104);
|
|
|
|
select @a;
|
|
|
|
@a
|
|
|
|
104
|
|
|
|
select trigger_schema, trigger_name, event_object_schema,
|
|
|
|
event_object_table, action_statement from information_schema.triggers
|
|
|
|
where event_object_schema = 'test' or event_object_schema = 'mysqltest';
|
|
|
|
trigger_schema trigger_name event_object_schema event_object_table action_statement
|
|
|
|
mysqltest t1_bi mysqltest t1 set @a:=new.id
|
|
|
|
show create table t1;
|
|
|
|
Table Create Table
|
|
|
|
t1 CREATE TABLE `t1` (
|
2006-03-25 12:07:45 +03:00
|
|
|
`id` int(11) DEFAULT NULL
|
2006-03-24 14:58:18 +03:00
|
|
|
) ENGINE=MyISAM DEFAULT CHARSET=latin1
|
|
|
|
drop trigger test.t1_bi;
|
|
|
|
ERROR HY000: Trigger does not exist
|
2006-02-24 23:50:36 +03:00
|
|
|
drop trigger t1_bi;
|
|
|
|
drop table t1;
|
|
|
|
drop database mysqltest;
|
|
|
|
use test;
|
|
|
|
create table t1 (id int);
|
|
|
|
create trigger t1_bi before insert on t1 for each row set @a:=new.id;
|
|
|
|
create trigger t1_ai after insert on t1 for each row set @b:=new.id;
|
|
|
|
insert into t1 values (101);
|
|
|
|
select @a, @b;
|
|
|
|
@a @b
|
|
|
|
101 101
|
|
|
|
select trigger_schema, trigger_name, event_object_schema,
|
|
|
|
event_object_table, action_statement from information_schema.triggers
|
|
|
|
where event_object_schema = 'test';
|
|
|
|
trigger_schema trigger_name event_object_schema event_object_table action_statement
|
2006-02-26 20:25:24 +03:00
|
|
|
test t1_bi test t1 set @a:=new.id
|
|
|
|
test t1_ai test t1 set @b:=new.id
|
2006-02-24 23:50:36 +03:00
|
|
|
rename table t1 to t2;
|
|
|
|
ERROR HY000: Can't create/write to file './test/t1_ai.TRN~' (Errcode: 13)
|
|
|
|
insert into t1 values (102);
|
|
|
|
select @a, @b;
|
|
|
|
@a @b
|
|
|
|
102 102
|
|
|
|
select trigger_schema, trigger_name, event_object_schema,
|
|
|
|
event_object_table, action_statement from information_schema.triggers
|
|
|
|
where event_object_schema = 'test';
|
|
|
|
trigger_schema trigger_name event_object_schema event_object_table action_statement
|
2006-02-26 20:25:24 +03:00
|
|
|
test t1_bi test t1 set @a:=new.id
|
|
|
|
test t1_ai test t1 set @b:=new.id
|
2006-02-24 23:50:36 +03:00
|
|
|
drop trigger t1_bi;
|
|
|
|
drop trigger t1_ai;
|
|
|
|
drop table t1;
|
2006-01-28 12:50:16 +03:00
|
|
|
create table t1 (i int);
|
|
|
|
create trigger t1_bi before insert on t1 for each row return 0;
|
|
|
|
ERROR 42000: RETURN is only allowed in a FUNCTION
|
|
|
|
insert into t1 values (1);
|
|
|
|
drop table t1;
|
2006-03-29 14:53:00 +04:00
|
|
|
create table t1 (a varchar(64), b int);
|
|
|
|
create table t2 like t1;
|
|
|
|
create trigger t1_ai after insert on t1 for each row
|
|
|
|
set @a:= (select max(a) from t1);
|
|
|
|
insert into t1 (a) values
|
|
|
|
("Twas"),("brillig"),("and"),("the"),("slithy"),("toves"),
|
|
|
|
("Did"),("gyre"),("and"),("gimble"),("in"),("the"),("wabe");
|
|
|
|
create trigger t2_ai after insert on t2 for each row
|
|
|
|
set @a:= (select max(a) from t2);
|
|
|
|
insert into t2 select * from t1;
|
|
|
|
load data infile '../std_data_ln/words.dat' into table t1 (a);
|
|
|
|
drop trigger t1_ai;
|
|
|
|
drop trigger t2_ai;
|
|
|
|
create function f1() returns int return (select max(b) from t1);
|
|
|
|
insert into t1 values
|
|
|
|
("All",f1()),("mimsy",f1()),("were",f1()),("the",f1()),("borogoves",f1()),
|
|
|
|
("And",f1()),("the",f1()),("mome", f1()),("raths",f1()),("outgrabe",f1());
|
|
|
|
create function f2() returns int return (select max(b) from t2);
|
|
|
|
insert into t2 select a, f2() from t1;
|
|
|
|
load data infile '../std_data_ln/words.dat' into table t1 (a) set b:= f1();
|
|
|
|
drop function f1;
|
|
|
|
drop function f2;
|
2006-05-10 21:35:05 +02:00
|
|
|
drop table t1, t2;
|
2006-04-19 19:13:03 +04:00
|
|
|
create table t1(i int not null, j int not null, n numeric(15,2), primary key(i,j));
|
|
|
|
create table t2(i int not null, n numeric(15,2), primary key(i));
|
|
|
|
create trigger t1_ai after insert on t1 for each row
|
|
|
|
begin
|
|
|
|
declare sn numeric(15,2);
|
|
|
|
select sum(n) into sn from t1 where i=new.i;
|
|
|
|
replace into t2 values(new.i, sn);
|
|
|
|
end|
|
|
|
|
insert into t1 values
|
|
|
|
(1,1,10.00),(1,2,10.00),(1,3,10.00),(1,4,10.00),(1,5,10.00),
|
|
|
|
(1,6,10.00),(1,7,10.00),(1,8,10.00),(1,9,10.00),(1,10,10.00),
|
|
|
|
(1,11,10.00),(1,12,10.00),(1,13,10.00),(1,14,10.00),(1,15,10.00);
|
|
|
|
select * from t1;
|
|
|
|
i j n
|
|
|
|
1 1 10.00
|
|
|
|
1 2 10.00
|
|
|
|
1 3 10.00
|
|
|
|
1 4 10.00
|
|
|
|
1 5 10.00
|
|
|
|
1 6 10.00
|
|
|
|
1 7 10.00
|
|
|
|
1 8 10.00
|
|
|
|
1 9 10.00
|
|
|
|
1 10 10.00
|
|
|
|
1 11 10.00
|
|
|
|
1 12 10.00
|
|
|
|
1 13 10.00
|
|
|
|
1 14 10.00
|
|
|
|
1 15 10.00
|
|
|
|
select * from t2;
|
|
|
|
i n
|
|
|
|
1 150.00
|
|
|
|
drop tables t1, t2;
|
2006-04-12 19:31:00 +04:00
|
|
|
DROP TABLE IF EXISTS t1;
|
|
|
|
CREATE TABLE t1 (
|
|
|
|
conn_id INT,
|
|
|
|
trigger_conn_id INT
|
|
|
|
);
|
|
|
|
CREATE TRIGGER t1_bi BEFORE INSERT ON t1 FOR EACH ROW
|
|
|
|
SET NEW.trigger_conn_id = CONNECTION_ID();
|
|
|
|
INSERT INTO t1 (conn_id, trigger_conn_id) VALUES (CONNECTION_ID(), -1);
|
|
|
|
INSERT INTO t1 (conn_id, trigger_conn_id) VALUES (CONNECTION_ID(), -1);
|
|
|
|
SELECT * FROM t1 WHERE conn_id != trigger_conn_id;
|
|
|
|
conn_id trigger_conn_id
|
|
|
|
DROP TRIGGER t1_bi;
|
|
|
|
DROP TABLE t1;
|
2006-04-19 14:27:59 +04:00
|
|
|
DROP TABLE IF EXISTS t1;
|
|
|
|
CREATE TABLE t1 (i1 INT);
|
|
|
|
SET @save_sql_mode=@@sql_mode;
|
|
|
|
SET SQL_MODE='';
|
|
|
|
CREATE TRIGGER t1_ai AFTER INSERT ON t1 FOR EACH ROW
|
|
|
|
SET @x = 5/0;
|
|
|
|
SET SQL_MODE='traditional';
|
|
|
|
CREATE TRIGGER t1_au AFTER UPDATE ON t1 FOR EACH ROW
|
|
|
|
SET @x = 5/0;
|
|
|
|
SET @x=1;
|
|
|
|
INSERT INTO t1 VALUES (@x);
|
|
|
|
SELECT @x;
|
|
|
|
@x
|
|
|
|
NULL
|
|
|
|
SET @x=2;
|
|
|
|
UPDATE t1 SET i1 = @x;
|
2006-10-19 11:39:51 -07:00
|
|
|
Warnings:
|
|
|
|
Error 1365 Division by 0
|
2006-04-19 14:27:59 +04:00
|
|
|
SELECT @x;
|
|
|
|
@x
|
2006-10-19 11:39:51 -07:00
|
|
|
NULL
|
2006-04-19 14:27:59 +04:00
|
|
|
SET SQL_MODE='';
|
|
|
|
SET @x=3;
|
|
|
|
INSERT INTO t1 VALUES (@x);
|
|
|
|
SELECT @x;
|
|
|
|
@x
|
|
|
|
NULL
|
|
|
|
SET @x=4;
|
|
|
|
UPDATE t1 SET i1 = @x;
|
2006-10-19 11:39:51 -07:00
|
|
|
Warnings:
|
|
|
|
Error 1365 Division by 0
|
|
|
|
Error 1365 Division by 0
|
2006-04-19 14:27:59 +04:00
|
|
|
SELECT @x;
|
|
|
|
@x
|
2006-10-19 11:39:51 -07:00
|
|
|
NULL
|
2006-04-19 14:27:59 +04:00
|
|
|
SET @@sql_mode=@save_sql_mode;
|
|
|
|
DROP TRIGGER t1_ai;
|
|
|
|
DROP TRIGGER t1_au;
|
|
|
|
DROP TABLE t1;
|
2006-05-12 13:55:21 +04:00
|
|
|
DROP TABLE IF EXISTS t1;
|
|
|
|
DROP PROCEDURE IF EXISTS p1;
|
|
|
|
DROP PROCEDURE IF EXISTS p2;
|
|
|
|
CREATE TABLE t1 (i1 INT);
|
|
|
|
INSERT INTO t1 VALUES (3);
|
|
|
|
CREATE PROCEDURE p1(OUT i1 INT) DETERMINISTIC NO SQL SET i1 = 5;
|
|
|
|
CREATE PROCEDURE p2(INOUT i1 INT) DETERMINISTIC NO SQL SET i1 = i1 * 7;
|
|
|
|
CREATE TRIGGER t1_bu BEFORE UPDATE ON t1 FOR EACH ROW
|
|
|
|
BEGIN
|
|
|
|
CALL p1(NEW.i1);
|
|
|
|
CALL p2(NEW.i1);
|
|
|
|
END//
|
|
|
|
UPDATE t1 SET i1 = 11 WHERE i1 = 3;
|
|
|
|
DROP TRIGGER t1_bu;
|
|
|
|
DROP PROCEDURE p2;
|
|
|
|
DROP PROCEDURE p1;
|
|
|
|
INSERT INTO t1 VALUES (13);
|
|
|
|
CREATE PROCEDURE p1(OUT i1 INT) DETERMINISTIC NO SQL SET @a = 17;
|
|
|
|
CREATE TRIGGER t1_bu BEFORE UPDATE ON t1 FOR EACH ROW
|
|
|
|
CALL p1(OLD.i1);
|
|
|
|
UPDATE t1 SET i1 = 19 WHERE i1 = 13;
|
|
|
|
ERROR 42000: OUT or INOUT argument 1 for routine test.p1 is not a variable or NEW pseudo-variable in BEFORE trigger
|
|
|
|
DROP TRIGGER t1_bu;
|
|
|
|
DROP PROCEDURE p1;
|
|
|
|
INSERT INTO t1 VALUES (23);
|
|
|
|
CREATE PROCEDURE p1(INOUT i1 INT) DETERMINISTIC NO SQL SET @a = i1 * 29;
|
|
|
|
CREATE TRIGGER t1_bu BEFORE UPDATE ON t1 FOR EACH ROW
|
|
|
|
CALL p1(OLD.i1);
|
|
|
|
UPDATE t1 SET i1 = 31 WHERE i1 = 23;
|
|
|
|
ERROR 42000: OUT or INOUT argument 1 for routine test.p1 is not a variable or NEW pseudo-variable in BEFORE trigger
|
|
|
|
DROP TRIGGER t1_bu;
|
|
|
|
DROP PROCEDURE p1;
|
|
|
|
INSERT INTO t1 VALUES (37);
|
|
|
|
CREATE PROCEDURE p1(OUT i1 INT) DETERMINISTIC NO SQL SET @a = 41;
|
|
|
|
CREATE TRIGGER t1_au AFTER UPDATE ON t1 FOR EACH ROW
|
|
|
|
CALL p1(NEW.i1);
|
|
|
|
UPDATE t1 SET i1 = 43 WHERE i1 = 37;
|
|
|
|
ERROR 42000: OUT or INOUT argument 1 for routine test.p1 is not a variable or NEW pseudo-variable in BEFORE trigger
|
|
|
|
DROP TRIGGER t1_au;
|
|
|
|
DROP PROCEDURE p1;
|
|
|
|
INSERT INTO t1 VALUES (47);
|
|
|
|
CREATE PROCEDURE p1(INOUT i1 INT) DETERMINISTIC NO SQL SET @a = i1 * 49;
|
|
|
|
CREATE TRIGGER t1_au AFTER UPDATE ON t1 FOR EACH ROW
|
|
|
|
CALL p1(NEW.i1);
|
|
|
|
UPDATE t1 SET i1 = 51 WHERE i1 = 47;
|
|
|
|
ERROR 42000: OUT or INOUT argument 1 for routine test.p1 is not a variable or NEW pseudo-variable in BEFORE trigger
|
|
|
|
DROP TRIGGER t1_au;
|
|
|
|
DROP PROCEDURE p1;
|
|
|
|
SELECT * FROM t1;
|
|
|
|
i1
|
|
|
|
35
|
|
|
|
13
|
|
|
|
23
|
|
|
|
43
|
|
|
|
51
|
|
|
|
DROP TABLE t1;
|
2006-06-27 17:16:02 -07:00
|
|
|
create trigger wont_work after update on mysql.user for each row
|
|
|
|
begin
|
|
|
|
set @a:= 1;
|
|
|
|
end|
|
|
|
|
ERROR HY000: Triggers can not be created on system tables
|
|
|
|
use mysql|
|
|
|
|
create trigger wont_work after update on event for each row
|
|
|
|
begin
|
|
|
|
set @a:= 1;
|
|
|
|
end|
|
|
|
|
ERROR HY000: Triggers can not be created on system tables
|
2006-08-23 21:31:00 +04:00
|
|
|
use test|
|
|
|
|
DROP TABLE IF EXISTS t1;
|
|
|
|
DROP TABLE IF EXISTS t2;
|
|
|
|
CREATE TABLE t1(c INT);
|
|
|
|
CREATE TABLE t2(c INT);
|
|
|
|
CREATE DEFINER=1234567890abcdefGHIKL@localhost
|
|
|
|
TRIGGER t1_bi BEFORE INSERT ON t1 FOR EACH ROW SET @a = 1;
|
|
|
|
ERROR HY000: String '1234567890abcdefGHIKL' is too long for user name (should be no longer than 16)
|
|
|
|
CREATE DEFINER=some_user_name@1234567890abcdefghij1234567890abcdefghij1234567890abcdefghijQWERTY
|
|
|
|
TRIGGER t2_bi BEFORE INSERT ON t2 FOR EACH ROW SET @a = 2;
|
|
|
|
ERROR HY000: String '1234567890abcdefghij1234567890abcdefghij1234567890abcdefghijQWERTY' is too long for host name (should be no longer than 60)
|
|
|
|
DROP TABLE t1;
|
|
|
|
DROP TABLE t2;
|
2006-09-21 11:35:38 +04:00
|
|
|
drop table if exists t1;
|
2006-10-19 11:39:51 -07:00
|
|
|
drop table if exists t2;
|
|
|
|
drop table if exists t3;
|
|
|
|
drop table if exists t4;
|
|
|
|
SET @save_sql_mode=@@sql_mode;
|
|
|
|
SET sql_mode='TRADITIONAL'|
|
|
|
|
create table t1 (id int(10) not null primary key, v int(10) )|
|
|
|
|
create table t2 (id int(10) not null primary key, v int(10) )|
|
|
|
|
create table t3 (id int(10) not null primary key, v int(10) )|
|
|
|
|
create table t4 (c int)|
|
|
|
|
create trigger t4_bi before insert on t4 for each row set @t4_bi_called:=1|
|
|
|
|
create trigger t4_bu before update on t4 for each row set @t4_bu_called:=1|
|
|
|
|
insert into t1 values(10, 10)|
|
|
|
|
set @a:=1/0|
|
|
|
|
Warnings:
|
|
|
|
Error 1365 Division by 0
|
|
|
|
select 1/0 from t1|
|
|
|
|
1/0
|
|
|
|
NULL
|
|
|
|
Warnings:
|
|
|
|
Error 1365 Division by 0
|
|
|
|
create trigger t1_bi before insert on t1 for each row set @a:=1/0|
|
|
|
|
insert into t1 values(20, 20)|
|
|
|
|
Warnings:
|
|
|
|
Error 1365 Division by 0
|
|
|
|
drop trigger t1_bi|
|
|
|
|
create trigger t1_bi before insert on t1 for each row
|
|
|
|
begin
|
|
|
|
insert into t2 values (new.id, new.v);
|
|
|
|
update t2 set v=v+1 where id= new.id;
|
|
|
|
replace t3 values (new.id, 0);
|
|
|
|
update t2, t3 set t2.v=new.v, t3.v=new.v where t2.id=t3.id;
|
|
|
|
create temporary table t5 select * from t1;
|
|
|
|
delete from t5;
|
|
|
|
insert into t5 select * from t1;
|
|
|
|
insert into t4 values (0);
|
|
|
|
set @check= (select count(*) from t5);
|
|
|
|
update t4 set c= @check;
|
|
|
|
drop temporary table t5;
|
|
|
|
set @a:=1/0;
|
|
|
|
end|
|
|
|
|
set @check=0, @t4_bi_called=0, @t4_bu_called=0|
|
|
|
|
insert into t1 values(30, 30)|
|
|
|
|
Warnings:
|
|
|
|
Error 1365 Division by 0
|
|
|
|
select @check, @t4_bi_called, @t4_bu_called|
|
|
|
|
@check @t4_bi_called @t4_bu_called
|
|
|
|
2 1 1
|
|
|
|
SET @@sql_mode=@save_sql_mode;
|
|
|
|
drop table t1;
|
|
|
|
drop table t2;
|
|
|
|
drop table t3;
|
|
|
|
drop table t4;
|
|
|
|
drop table if exists t1;
|
2006-09-21 11:35:38 +04:00
|
|
|
create table t1 (i int, j int key);
|
|
|
|
insert into t1 values (1,1), (2,2), (3,3);
|
|
|
|
create trigger t1_bu before update on t1 for each row
|
|
|
|
set new.j = new.j + 10;
|
|
|
|
update t1 set i= i+ 10 where j > 2;
|
|
|
|
select * from t1;
|
|
|
|
i j
|
|
|
|
1 1
|
|
|
|
2 2
|
|
|
|
13 13
|
|
|
|
drop table t1;
|
2007-02-01 13:25:50 +02:00
|
|
|
CREATE TABLE t1 (a INT PRIMARY KEY);
|
|
|
|
CREATE TABLE t2 (a INT PRIMARY KEY);
|
|
|
|
INSERT INTO t1 VALUES (1),(2),(3),(4),(5),(6),(7),(8);
|
|
|
|
CREATE TRIGGER trg_t1 BEFORE DELETE on t1 FOR EACH ROW
|
|
|
|
INSERT INTO t2 VALUES (OLD.a);
|
|
|
|
FLUSH STATUS;
|
|
|
|
TRUNCATE t1;
|
|
|
|
SHOW STATUS LIKE 'handler_delete';
|
|
|
|
Variable_name Value
|
|
|
|
Handler_delete 0
|
|
|
|
SELECT COUNT(*) FROM t2;
|
|
|
|
COUNT(*)
|
|
|
|
0
|
|
|
|
INSERT INTO t1 VALUES (1),(2),(3),(4),(5),(6),(7),(8);
|
|
|
|
DELETE FROM t2;
|
|
|
|
FLUSH STATUS;
|
|
|
|
DELETE FROM t1;
|
|
|
|
SHOW STATUS LIKE 'handler_delete';
|
|
|
|
Variable_name Value
|
|
|
|
Handler_delete 8
|
|
|
|
SELECT COUNT(*) FROM t2;
|
|
|
|
COUNT(*)
|
|
|
|
8
|
|
|
|
DROP TRIGGER trg_t1;
|
|
|
|
DROP TABLE t1,t2;
|
2006-11-13 11:10:49 +03:00
|
|
|
drop table if exists t1;
|
|
|
|
drop function if exists f1;
|
|
|
|
create table t1 (i int);
|
|
|
|
create function f1() returns int return 10;
|
|
|
|
create trigger t1_bi before insert on t1 for each row set @a:= f1() + 10;
|
|
|
|
insert into t1 values ();
|
|
|
|
select @a;
|
|
|
|
@a
|
|
|
|
20
|
|
|
|
insert into t1 values ();
|
|
|
|
select @a;
|
|
|
|
@a
|
|
|
|
20
|
|
|
|
drop table t1;
|
|
|
|
drop function f1;
|
Bug#23703 (DROP TRIGGER needs an IF EXISTS)
This change set implements the DROP TRIGGER IF EXISTS functionality.
This fix is considered a bug and not a feature, because without it,
there is no known method to write a database creation script that can create
a trigger without failing, when executed on a database that may or may not
contain already a trigger of the same name.
Implementing this functionality closes an orthogonality gap between triggers
and stored procedures / stored functions (which do support the DROP IF
EXISTS syntax).
In sql_trigger.cc, in mysql_create_or_drop_trigger,
the code has been reordered to:
- perform the tests that do not depend on the file system (access()),
- get the locks (wait_if_global_read_lock, LOCK_open)
- call access()
- perform the operation
- write to the binlog
- unlock (LOCK_open, start_waiting_global_read_lock)
This is to ensure that all the code that depends on the presence of the
trigger file is executed in the same critical section,
and prevents race conditions similar to the case fixed by Bug 14262 :
- thread 1 executes DROP TRIGGER IF EXISTS, access() returns a failure
- thread 2 executes CREATE TRIGGER
- thread 2 logs CREATE TRIGGER
- thread 1 logs DROP TRIGGER IF EXISTS
The patch itself is based on code contributed by the MySQL community,
under the terms of the Contributor License Agreement (See Bug 18161).
mysql-test/r/rpl_trigger.result:
DROP TRIGGER IF EXISTS
mysql-test/r/trigger.result:
DROP TRIGGER IF EXISTS
mysql-test/t/rpl_trigger.test:
DROP TRIGGER IF EXISTS
mysql-test/t/trigger.test:
DROP TRIGGER IF EXISTS
sql/sql_trigger.cc:
DROP TRIGGER IF EXISTS
sql/sql_yacc.yy:
DROP TRIGGER IF EXISTS
2006-11-13 15:40:22 -07:00
|
|
|
drop table if exists t1;
|
|
|
|
create table t1(a int, b varchar(50));
|
|
|
|
drop trigger not_a_trigger;
|
|
|
|
ERROR HY000: Trigger does not exist
|
|
|
|
drop trigger if exists not_a_trigger;
|
|
|
|
Warnings:
|
|
|
|
Note 1360 Trigger does not exist
|
|
|
|
create trigger t1_bi before insert on t1
|
|
|
|
for each row set NEW.b := "In trigger t1_bi";
|
|
|
|
insert into t1 values (1, "a");
|
|
|
|
drop trigger if exists t1_bi;
|
|
|
|
insert into t1 values (2, "b");
|
|
|
|
drop trigger if exists t1_bi;
|
|
|
|
Warnings:
|
|
|
|
Note 1360 Trigger does not exist
|
|
|
|
insert into t1 values (3, "c");
|
|
|
|
select * from t1;
|
|
|
|
a b
|
|
|
|
1 In trigger t1_bi
|
|
|
|
2 b
|
|
|
|
3 c
|
|
|
|
drop table t1;
|
2007-01-12 13:43:25 -08:00
|
|
|
CREATE TABLE t1 (
|
|
|
|
id int NOT NULL DEFAULT '0',
|
|
|
|
a varchar(10) NOT NULL,
|
|
|
|
b varchar(10),
|
|
|
|
c varchar(10),
|
|
|
|
d timestamp NOT NULL,
|
|
|
|
PRIMARY KEY (id, a)
|
|
|
|
);
|
|
|
|
CREATE TABLE t2 (
|
|
|
|
fubar_id int unsigned NOT NULL DEFAULT '0',
|
|
|
|
last_change_time datetime NOT NULL DEFAULT '0000-00-00 00:00:00',
|
|
|
|
PRIMARY KEY (fubar_id)
|
|
|
|
);
|
|
|
|
CREATE TRIGGER fubar_change
|
|
|
|
AFTER UPDATE ON t1
|
|
|
|
FOR EACH ROW
|
|
|
|
BEGIN
|
|
|
|
INSERT INTO t2 (fubar_id, last_change_time)
|
|
|
|
SELECT DISTINCT NEW.id AS fubar_id, NOW() AS last_change_time
|
|
|
|
FROM t1 WHERE (id = NEW.id) AND (OLD.c != NEW.c)
|
|
|
|
ON DUPLICATE KEY UPDATE
|
|
|
|
last_change_time =
|
|
|
|
IF((fubar_id = NEW.id)AND(OLD.c != NEW.c),NOW(),last_change_time);
|
|
|
|
END
|
|
|
|
|
|
|
|
|
INSERT INTO t1 (id,a, b,c,d) VALUES
|
|
|
|
(1,'a','b','c',now()),(2,'a','b','c',now());
|
|
|
|
UPDATE t1 SET c='Bang!' WHERE id=1;
|
|
|
|
SELECT fubar_id FROM t2;
|
|
|
|
fubar_id
|
|
|
|
1
|
|
|
|
DROP TABLE t1,t2;
|
2007-03-06 13:46:33 -07:00
|
|
|
DROP TABLE IF EXISTS bug21825_A;
|
|
|
|
DROP TABLE IF EXISTS bug21825_B;
|
|
|
|
CREATE TABLE bug21825_A (id int(10));
|
|
|
|
CREATE TABLE bug21825_B (id int(10));
|
|
|
|
CREATE TRIGGER trgA AFTER INSERT ON bug21825_A
|
|
|
|
FOR EACH ROW
|
|
|
|
BEGIN
|
|
|
|
INSERT INTO bug21825_B (id) values (1);
|
|
|
|
END//
|
|
|
|
INSERT INTO bug21825_A (id) VALUES (10);
|
|
|
|
INSERT INTO bug21825_A (id) VALUES (20);
|
|
|
|
DROP TABLE bug21825_B;
|
|
|
|
DELETE FROM bug21825_A WHERE id = 20;
|
|
|
|
DROP TABLE bug21825_A;
|
|
|
|
DROP TABLE IF EXISTS bug22580_t1;
|
|
|
|
DROP PROCEDURE IF EXISTS bug22580_proc_1;
|
|
|
|
DROP PROCEDURE IF EXISTS bug22580_proc_2;
|
|
|
|
CREATE TABLE bug22580_t1 (a INT, b INT);
|
|
|
|
CREATE PROCEDURE bug22580_proc_2()
|
|
|
|
BEGIN
|
|
|
|
DROP TABLE IF EXISTS bug22580_tmp;
|
|
|
|
CREATE TEMPORARY TABLE bug22580_tmp (a INT);
|
|
|
|
DROP TABLE bug22580_tmp;
|
|
|
|
END||
|
|
|
|
CREATE PROCEDURE bug22580_proc_1()
|
|
|
|
BEGIN
|
|
|
|
CALL bug22580_proc_2();
|
|
|
|
END||
|
|
|
|
CREATE TRIGGER t1bu BEFORE UPDATE ON bug22580_t1
|
|
|
|
FOR EACH ROW
|
|
|
|
BEGIN
|
|
|
|
CALL bug22580_proc_1();
|
|
|
|
END||
|
|
|
|
INSERT INTO bug22580_t1 VALUES (1,1);
|
|
|
|
DROP TABLE bug22580_t1;
|
|
|
|
DROP PROCEDURE bug22580_proc_1;
|
|
|
|
DROP PROCEDURE bug22580_proc_2;
|
2007-03-16 17:23:26 +03:00
|
|
|
DROP TRIGGER IF EXISTS trg27006_a_update;
|
|
|
|
DROP TRIGGER IF EXISTS trg27006_a_insert;
|
|
|
|
CREATE TABLE t1 (
|
|
|
|
`id` int(10) unsigned NOT NULL auto_increment,
|
|
|
|
`val` varchar(10) NOT NULL,
|
|
|
|
PRIMARY KEY (`id`)
|
|
|
|
);
|
|
|
|
CREATE TABLE t2 like t1;
|
|
|
|
CREATE TRIGGER trg27006_a_insert AFTER INSERT ON t1 FOR EACH ROW
|
|
|
|
BEGIN
|
|
|
|
insert into t2 values (NULL,new.val);
|
|
|
|
END |
|
|
|
|
CREATE TRIGGER trg27006_a_update AFTER UPDATE ON t1 FOR EACH ROW
|
|
|
|
BEGIN
|
|
|
|
insert into t2 values (NULL,new.val);
|
|
|
|
END |
|
|
|
|
INSERT INTO t1(val) VALUES ('test1'),('test2');
|
|
|
|
SELECT * FROM t1;
|
|
|
|
id val
|
|
|
|
1 test1
|
|
|
|
2 test2
|
|
|
|
SELECT * FROM t2;
|
|
|
|
id val
|
|
|
|
1 test1
|
|
|
|
2 test2
|
|
|
|
INSERT INTO t1 VALUES (2,'test2') ON DUPLICATE KEY UPDATE val=VALUES(val);
|
2007-03-20 00:46:19 +03:00
|
|
|
INSERT INTO t1 VALUES (2,'test3') ON DUPLICATE KEY UPDATE val=VALUES(val);
|
|
|
|
INSERT INTO t1 VALUES (3,'test4') ON DUPLICATE KEY UPDATE val=VALUES(val);
|
2007-03-16 17:23:26 +03:00
|
|
|
SELECT * FROM t1;
|
|
|
|
id val
|
|
|
|
1 test1
|
2007-03-20 00:46:19 +03:00
|
|
|
2 test3
|
|
|
|
3 test4
|
2007-03-16 17:23:26 +03:00
|
|
|
SELECT * FROM t2;
|
|
|
|
id val
|
|
|
|
1 test1
|
|
|
|
2 test2
|
2007-03-20 00:46:19 +03:00
|
|
|
3 test3
|
|
|
|
4 test4
|
2007-03-16 17:23:26 +03:00
|
|
|
DROP TRIGGER trg27006_a_insert;
|
|
|
|
DROP TRIGGER trg27006_a_update;
|
|
|
|
drop table t1,t2;
|
2007-05-11 21:51:03 +04:00
|
|
|
drop table if exists t1, t2, t3;
|
|
|
|
create table t1 (i int);
|
|
|
|
create trigger t1_bi before insert on t1 for each row set new.i = 7;
|
|
|
|
create trigger t1_ai after insert on t1 for each row set @a := 7;
|
|
|
|
create table t2 (j int);
|
|
|
|
insert into t2 values (1), (2);
|
|
|
|
set @a:="";
|
|
|
|
create table if not exists t1 select * from t2;
|
|
|
|
Warnings:
|
|
|
|
Note 1050 Table 't1' already exists
|
|
|
|
select * from t1;
|
|
|
|
i
|
|
|
|
7
|
|
|
|
7
|
|
|
|
select @a;
|
|
|
|
@a
|
|
|
|
7
|
|
|
|
drop trigger t1_bi;
|
|
|
|
drop trigger t1_ai;
|
|
|
|
create table t3 (isave int);
|
|
|
|
create trigger t1_bi before insert on t1 for each row insert into t3 values (new.i);
|
|
|
|
create table if not exists t1 select * from t2;
|
|
|
|
Warnings:
|
|
|
|
Note 1050 Table 't1' already exists
|
|
|
|
select * from t1;
|
|
|
|
i
|
|
|
|
7
|
|
|
|
7
|
|
|
|
1
|
|
|
|
2
|
|
|
|
select * from t3;
|
|
|
|
isave
|
|
|
|
1
|
|
|
|
2
|
|
|
|
drop table t1, t2, t3;
|
2007-06-03 09:40:00 +03:00
|
|
|
CREATE TABLE t1 (id INTEGER);
|
|
|
|
CREATE TABLE t2 (id INTEGER);
|
|
|
|
INSERT INTO t2 VALUES (1),(2);
|
|
|
|
CREATE TRIGGER t1_test AFTER INSERT ON t1 FOR EACH ROW
|
|
|
|
INSERT INTO t2 VALUES (new.id);
|
2007-06-12 14:35:36 +03:00
|
|
|
SELECT GET_LOCK('B26162',120);
|
|
|
|
GET_LOCK('B26162',120)
|
2007-06-03 09:40:00 +03:00
|
|
|
1
|
2007-06-12 14:35:36 +03:00
|
|
|
SELECT 'rl_acquirer', GET_LOCK('B26162',120), id FROM t2 WHERE id = 1;
|
2007-06-03 09:40:00 +03:00
|
|
|
SET SESSION LOW_PRIORITY_UPDATES=1;
|
|
|
|
SET GLOBAL LOW_PRIORITY_UPDATES=1;
|
|
|
|
INSERT INTO t1 VALUES (5);
|
|
|
|
SELECT 'rl_contender', id FROM t2 WHERE id > 1;
|
|
|
|
SELECT RELEASE_LOCK('B26162');
|
|
|
|
RELEASE_LOCK('B26162')
|
2007-06-12 14:35:36 +03:00
|
|
|
1
|
|
|
|
rl_acquirer GET_LOCK('B26162',120) id
|
|
|
|
rl_acquirer 1 1
|
|
|
|
SELECT RELEASE_LOCK('B26162');
|
|
|
|
RELEASE_LOCK('B26162')
|
|
|
|
1
|
2007-06-03 09:40:00 +03:00
|
|
|
rl_contender id
|
|
|
|
rl_contender 2
|
|
|
|
DROP TRIGGER t1_test;
|
|
|
|
DROP TABLE t1,t2;
|
|
|
|
SET SESSION LOW_PRIORITY_UPDATES=DEFAULT;
|
|
|
|
SET GLOBAL LOW_PRIORITY_UPDATES=DEFAULT;
|
A fix and a test case for Bug#26141 mixing table types in trigger
causes full table lock on innodb table.
Also fixes Bug#28502 Triggers that update another innodb table
will block on X lock unnecessarily (duplciate).
Code review fixes.
Both bugs' synopses are misleading: InnoDB table is
not X locked. The statements, however, cannot proceed concurrently,
but this happens due to lock conflicts for tables used in triggers,
not for the InnoDB table.
If a user had an InnoDB table, and two triggers, AFTER UPDATE and
AFTER INSERT, competing for different resources (e.g. two distinct
MyISAM tables), then these two triggers would not be able to execute
concurrently. Moreover, INSERTS/UPDATES of the InnoDB table would
not be able to run concurrently.
The problem had other side-effects (see respective bug reports).
This behavior was a consequence of a shortcoming of the pre-locking
algorithm, which would not distinguish between different DML operations
(e.g. INSERT and DELETE) and pre-lock all the tables
that are used by any trigger defined on the subject table.
The idea of the fix is to extend the pre-locking algorithm to keep track,
for each table, what DML operation it is used for and not
load triggers that are known to never be fired.
mysql-test/r/trigger-trans.result:
Update results (Bug#26141)
mysql-test/r/trigger.result:
Update results (Bug#28502)
mysql-test/t/trigger-trans.test:
Add a test case for Bug#26141 mixing table types in trigger causes
full table lock on innodb table.
mysql-test/t/trigger.test:
Add a test case for Bug#28502 Triggers that update another innodb
table will block echo on X lock unnecessarily. Add more test
coverage for triggers.
sql/item.h:
enum trg_event_type is needed in table.h
sql/sp.cc:
Take into account table_list->trg_event_map when determining
what tables to pre-lock.
After this change, if we attempt to fire a
trigger for which we had not pre-locked any tables, error
'Table was not locked with LOCK TABLES' will be printed.
This, however, should never happen, provided the pre-locking
algorithm has no programming bugs.
Previously a trigger key in the sroutines hash was based on the name
of the table the trigger belongs to. This was possible because we would
always add to the pre-locking list all the triggers defined for a table when
handling this table.
Now the key is based on the name of the trigger, owing
to the fact that a trigger name must be unique in the database it
belongs to.
sql/sp_head.cc:
Generate sroutines hash key in init_spname(). This is a convenient
place since there we have all the necessary information and can
avoid an extra alloc.
Maintain and merge trg_event_map when adding and merging elements
of the pre-locking list.
sql/sp_head.h:
Add ,m_sroutines_key member, used when inserting the sphead for a
trigger into the cache of routines used by a statement.
Previously the key was based on the table name the trigger belonged
to, since for a given table we would add to the sroutines list
all the triggers defined on it.
sql/sql_lex.cc:
Introduce a new lex step: set_trg_event_type_for_tables().
It is called when we have finished parsing but before opening
and locking tables. Now this step is used to evaluate for each
TABLE_LIST instance which INSERT/UPDATE/DELETE operation, if any,
it is used in.
In future this method could be extended to aggregate other information
that is hard to aggregate during parsing.
sql/sql_lex.h:
Add declaration for set_trg_event_type_for_tables().
sql/sql_parse.cc:
Call set_trg_event_type_for_tables() after MYSQLparse(). Remove tabs.
sql/sql_prepare.cc:
Call set_trg_event_type_for_tables() after MYSQLparse().
sql/sql_trigger.cc:
Call set_trg_event_type_for_tables() after MYSQLparse().
sql/sql_trigger.h:
Remove an obsolete member.
sql/sql_view.cc:
Call set_trg_event_type_for_tables() after MYSQLparse().
sql/sql_yacc.yy:
Move assignment of sp_head::m_type before calling sp_head::init_spname(),
one is now used inside another.
sql/table.cc:
Implement TABLE_LIST::set_trg_event_map() - a method that calculates
wh triggers may be fired on this table when executing a statement.
sql/table.h:
Add missing declarations.
Move declaration of trg_event_type from item.h (it will be needed for
trg_event_map bitmap when we start using Bitmap template instead
of uint8).
2007-07-12 22:26:41 +04:00
|
|
|
|
|
|
|
Bug#28502 Triggers that update another innodb table will block
|
|
|
|
on X lock unnecessarily
|
|
|
|
|
|
|
|
Ensure we do not open and lock tables for triggers we do not fire.
|
|
|
|
|
|
|
|
drop table if exists t1, t2;
|
|
|
|
drop trigger if exists trg_bug28502_au;
|
|
|
|
create table t1 (id int, count int);
|
|
|
|
create table t2 (id int);
|
|
|
|
create trigger trg_bug28502_au before update on t2
|
|
|
|
for each row
|
|
|
|
begin
|
|
|
|
if (new.id is not null) then
|
|
|
|
update t1 set count= count + 1 where id = old.id;
|
|
|
|
end if;
|
|
|
|
end|
|
|
|
|
insert into t1 (id, count) values (1, 0);
|
|
|
|
lock table t1 write;
|
|
|
|
insert into t2 set id=1;
|
|
|
|
unlock tables;
|
|
|
|
update t2 set id=1 where id=1;
|
|
|
|
select * from t1;
|
|
|
|
id count
|
|
|
|
1 1
|
|
|
|
select * from t2;
|
|
|
|
id
|
|
|
|
1
|
|
|
|
drop table t1, t2;
|
|
|
|
|
|
|
|
Additionally, provide test coverage for triggers and
|
|
|
|
all MySQL data changing commands.
|
|
|
|
|
|
|
|
drop table if exists t1, t2, t1_op_log;
|
|
|
|
drop view if exists v1;
|
|
|
|
drop trigger if exists trg_bug28502_bi;
|
|
|
|
drop trigger if exists trg_bug28502_ai;
|
|
|
|
drop trigger if exists trg_bug28502_bu;
|
|
|
|
drop trigger if exists trg_bug28502_au;
|
|
|
|
drop trigger if exists trg_bug28502_bd;
|
|
|
|
drop trigger if exists trg_bug28502_ad;
|
|
|
|
create table t1 (id int primary key auto_increment, operation varchar(255));
|
|
|
|
create table t2 (id int primary key);
|
|
|
|
create table t1_op_log(operation varchar(255));
|
|
|
|
create view v1 as select * from t1;
|
|
|
|
create trigger trg_bug28502_bi before insert on t1
|
|
|
|
for each row
|
|
|
|
insert into t1_op_log (operation)
|
|
|
|
values (concat("Before INSERT, new=", new.operation));
|
|
|
|
create trigger trg_bug28502_ai after insert on t1
|
|
|
|
for each row
|
|
|
|
insert into t1_op_log (operation)
|
|
|
|
values (concat("After INSERT, new=", new.operation));
|
|
|
|
create trigger trg_bug28502_bu before update on t1
|
|
|
|
for each row
|
|
|
|
insert into t1_op_log (operation)
|
|
|
|
values (concat("Before UPDATE, new=", new.operation,
|
|
|
|
", old=", old.operation));
|
|
|
|
create trigger trg_bug28502_au after update on t1
|
|
|
|
for each row
|
|
|
|
insert into t1_op_log (operation)
|
|
|
|
values (concat("After UPDATE, new=", new.operation,
|
|
|
|
", old=", old.operation));
|
|
|
|
create trigger trg_bug28502_bd before delete on t1
|
|
|
|
for each row
|
|
|
|
insert into t1_op_log (operation)
|
|
|
|
values (concat("Before DELETE, old=", old.operation));
|
|
|
|
create trigger trg_bug28502_ad after delete on t1
|
|
|
|
for each row
|
|
|
|
insert into t1_op_log (operation)
|
|
|
|
values (concat("After DELETE, old=", old.operation));
|
|
|
|
insert into t1 (operation) values ("INSERT");
|
|
|
|
set @id=last_insert_id();
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 INSERT
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT
|
|
|
|
After INSERT, new=INSERT
|
|
|
|
truncate t1_op_log;
|
|
|
|
update t1 set operation="UPDATE" where id=@id;
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 UPDATE
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before UPDATE, new=UPDATE, old=INSERT
|
|
|
|
After UPDATE, new=UPDATE, old=INSERT
|
|
|
|
truncate t1_op_log;
|
|
|
|
delete from t1 where id=@id;
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before DELETE, old=UPDATE
|
|
|
|
After DELETE, old=UPDATE
|
|
|
|
truncate t1;
|
|
|
|
truncate t1_op_log;
|
|
|
|
insert into t1 (id, operation) values
|
|
|
|
(NULL, "INSERT ON DUPLICATE KEY UPDATE, inserting a new key")
|
|
|
|
on duplicate key update id=NULL, operation="Should never happen";
|
|
|
|
set @id=last_insert_id();
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 INSERT ON DUPLICATE KEY UPDATE, inserting a new key
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT ON DUPLICATE KEY UPDATE, inserting a new key
|
|
|
|
After INSERT, new=INSERT ON DUPLICATE KEY UPDATE, inserting a new key
|
|
|
|
truncate t1_op_log;
|
|
|
|
insert into t1 (id, operation) values
|
|
|
|
(@id, "INSERT ON DUPLICATE KEY UPDATE, the key value is the same")
|
|
|
|
on duplicate key update id=NULL,
|
|
|
|
operation="INSERT ON DUPLICATE KEY UPDATE, updating the duplicate";
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
0 INSERT ON DUPLICATE KEY UPDATE, updating the duplicate
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT ON DUPLICATE KEY UPDATE, the key value is the same
|
|
|
|
Before UPDATE, new=INSERT ON DUPLICATE KEY UPDATE, updating the duplicate, old=INSERT ON DUPLICATE KEY UPDATE, inserting a new key
|
|
|
|
After UPDATE, new=INSERT ON DUPLICATE KEY UPDATE, updating the duplicate, old=INSERT ON DUPLICATE KEY UPDATE, inserting a new key
|
|
|
|
truncate t1;
|
|
|
|
truncate t1_op_log;
|
|
|
|
replace into t1 values (NULL, "REPLACE, inserting a new key");
|
|
|
|
set @id=last_insert_id();
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 REPLACE, inserting a new key
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=REPLACE, inserting a new key
|
|
|
|
After INSERT, new=REPLACE, inserting a new key
|
|
|
|
truncate t1_op_log;
|
|
|
|
replace into t1 values (@id, "REPLACE, deleting the duplicate");
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 REPLACE, deleting the duplicate
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=REPLACE, deleting the duplicate
|
|
|
|
Before DELETE, old=REPLACE, inserting a new key
|
|
|
|
After DELETE, old=REPLACE, inserting a new key
|
|
|
|
After INSERT, new=REPLACE, deleting the duplicate
|
|
|
|
truncate t1;
|
|
|
|
truncate t1_op_log;
|
|
|
|
create table if not exists t1
|
|
|
|
select NULL, "CREATE TABLE ... SELECT, inserting a new key";
|
|
|
|
Warnings:
|
|
|
|
Note 1050 Table 't1' already exists
|
|
|
|
set @id=last_insert_id();
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 CREATE TABLE ... SELECT, inserting a new key
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=CREATE TABLE ... SELECT, inserting a new key
|
|
|
|
After INSERT, new=CREATE TABLE ... SELECT, inserting a new key
|
|
|
|
truncate t1_op_log;
|
|
|
|
create table if not exists t1 replace
|
|
|
|
select @id, "CREATE TABLE ... REPLACE SELECT, deleting a duplicate key";
|
|
|
|
Warnings:
|
|
|
|
Note 1050 Table 't1' already exists
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 CREATE TABLE ... REPLACE SELECT, deleting a duplicate key
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=CREATE TABLE ... REPLACE SELECT, deleting a duplicate key
|
|
|
|
Before DELETE, old=CREATE TABLE ... SELECT, inserting a new key
|
|
|
|
After DELETE, old=CREATE TABLE ... SELECT, inserting a new key
|
|
|
|
After INSERT, new=CREATE TABLE ... REPLACE SELECT, deleting a duplicate key
|
|
|
|
truncate t1;
|
|
|
|
truncate t1_op_log;
|
|
|
|
insert into t1 (id, operation)
|
|
|
|
select NULL, "INSERT ... SELECT, inserting a new key";
|
|
|
|
set @id=last_insert_id();
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 INSERT ... SELECT, inserting a new key
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT ... SELECT, inserting a new key
|
|
|
|
After INSERT, new=INSERT ... SELECT, inserting a new key
|
|
|
|
truncate t1_op_log;
|
|
|
|
insert into t1 (id, operation)
|
|
|
|
select @id,
|
|
|
|
"INSERT ... SELECT ... ON DUPLICATE KEY UPDATE, updating a duplicate"
|
|
|
|
on duplicate key update id=NULL,
|
|
|
|
operation="INSERT ... SELECT ... ON DUPLICATE KEY UPDATE, updating a duplicate";
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
0 INSERT ... SELECT ... ON DUPLICATE KEY UPDATE, updating a duplicate
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT ... SELECT ... ON DUPLICATE KEY UPDATE, updating a duplicate
|
|
|
|
Before UPDATE, new=INSERT ... SELECT ... ON DUPLICATE KEY UPDATE, updating a duplicate, old=INSERT ... SELECT, inserting a new key
|
|
|
|
After UPDATE, new=INSERT ... SELECT ... ON DUPLICATE KEY UPDATE, updating a duplicate, old=INSERT ... SELECT, inserting a new key
|
|
|
|
truncate t1;
|
|
|
|
truncate t1_op_log;
|
|
|
|
replace into t1 (id, operation)
|
|
|
|
select NULL, "REPLACE ... SELECT, inserting a new key";
|
|
|
|
set @id=last_insert_id();
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 REPLACE ... SELECT, inserting a new key
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=REPLACE ... SELECT, inserting a new key
|
|
|
|
After INSERT, new=REPLACE ... SELECT, inserting a new key
|
|
|
|
truncate t1_op_log;
|
|
|
|
replace into t1 (id, operation)
|
|
|
|
select @id, "REPLACE ... SELECT, deleting a duplicate";
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 REPLACE ... SELECT, deleting a duplicate
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=REPLACE ... SELECT, deleting a duplicate
|
|
|
|
Before DELETE, old=REPLACE ... SELECT, inserting a new key
|
|
|
|
After DELETE, old=REPLACE ... SELECT, inserting a new key
|
|
|
|
After INSERT, new=REPLACE ... SELECT, deleting a duplicate
|
|
|
|
truncate t1;
|
|
|
|
truncate t1_op_log;
|
|
|
|
insert into t1 (id, operation) values (1, "INSERT for multi-DELETE");
|
|
|
|
insert into t2 (id) values (1);
|
|
|
|
delete t1.*, t2.* from t1, t2 where t1.id=1;
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
select * from t2;
|
|
|
|
id
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT for multi-DELETE
|
|
|
|
After INSERT, new=INSERT for multi-DELETE
|
|
|
|
Before DELETE, old=INSERT for multi-DELETE
|
|
|
|
After DELETE, old=INSERT for multi-DELETE
|
|
|
|
truncate t1;
|
|
|
|
truncate t2;
|
|
|
|
truncate t1_op_log;
|
|
|
|
insert into t1 (id, operation) values (1, "INSERT for multi-UPDATE");
|
|
|
|
insert into t2 (id) values (1);
|
|
|
|
update t1, t2 set t1.id=2, operation="multi-UPDATE" where t1.id=1;
|
|
|
|
update t1, t2
|
|
|
|
set t2.id=3, operation="multi-UPDATE, SET for t2, but the trigger is fired" where t1.id=2;
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
2 multi-UPDATE, SET for t2, but the trigger is fired
|
|
|
|
select * from t2;
|
|
|
|
id
|
|
|
|
3
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT for multi-UPDATE
|
|
|
|
After INSERT, new=INSERT for multi-UPDATE
|
|
|
|
Before UPDATE, new=multi-UPDATE, old=INSERT for multi-UPDATE
|
|
|
|
After UPDATE, new=multi-UPDATE, old=INSERT for multi-UPDATE
|
|
|
|
Before UPDATE, new=multi-UPDATE, SET for t2, but the trigger is fired, old=multi-UPDATE
|
|
|
|
After UPDATE, new=multi-UPDATE, SET for t2, but the trigger is fired, old=multi-UPDATE
|
|
|
|
truncate table t1;
|
|
|
|
truncate table t2;
|
|
|
|
truncate table t1_op_log;
|
|
|
|
|
|
|
|
Now do the same but use a view instead of the base table.
|
|
|
|
|
|
|
|
insert into v1 (operation) values ("INSERT");
|
|
|
|
set @id=last_insert_id();
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 INSERT
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT
|
|
|
|
After INSERT, new=INSERT
|
|
|
|
truncate t1_op_log;
|
|
|
|
update v1 set operation="UPDATE" where id=@id;
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 UPDATE
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before UPDATE, new=UPDATE, old=INSERT
|
|
|
|
After UPDATE, new=UPDATE, old=INSERT
|
|
|
|
truncate t1_op_log;
|
|
|
|
delete from v1 where id=@id;
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before DELETE, old=UPDATE
|
|
|
|
After DELETE, old=UPDATE
|
|
|
|
truncate t1;
|
|
|
|
truncate t1_op_log;
|
|
|
|
insert into v1 (id, operation) values
|
|
|
|
(NULL, "INSERT ON DUPLICATE KEY UPDATE, inserting a new key")
|
|
|
|
on duplicate key update id=NULL, operation="Should never happen";
|
|
|
|
set @id=last_insert_id();
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 INSERT ON DUPLICATE KEY UPDATE, inserting a new key
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT ON DUPLICATE KEY UPDATE, inserting a new key
|
|
|
|
After INSERT, new=INSERT ON DUPLICATE KEY UPDATE, inserting a new key
|
|
|
|
truncate t1_op_log;
|
|
|
|
insert into v1 (id, operation) values
|
|
|
|
(@id, "INSERT ON DUPLICATE KEY UPDATE, the key value is the same")
|
|
|
|
on duplicate key update id=NULL,
|
|
|
|
operation="INSERT ON DUPLICATE KEY UPDATE, updating the duplicate";
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
0 INSERT ON DUPLICATE KEY UPDATE, updating the duplicate
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT ON DUPLICATE KEY UPDATE, the key value is the same
|
|
|
|
Before UPDATE, new=INSERT ON DUPLICATE KEY UPDATE, updating the duplicate, old=INSERT ON DUPLICATE KEY UPDATE, inserting a new key
|
|
|
|
After UPDATE, new=INSERT ON DUPLICATE KEY UPDATE, updating the duplicate, old=INSERT ON DUPLICATE KEY UPDATE, inserting a new key
|
|
|
|
truncate t1;
|
|
|
|
truncate t1_op_log;
|
|
|
|
replace into v1 values (NULL, "REPLACE, inserting a new key");
|
|
|
|
set @id=last_insert_id();
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 REPLACE, inserting a new key
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=REPLACE, inserting a new key
|
|
|
|
After INSERT, new=REPLACE, inserting a new key
|
|
|
|
truncate t1_op_log;
|
|
|
|
replace into v1 values (@id, "REPLACE, deleting the duplicate");
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 REPLACE, deleting the duplicate
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=REPLACE, deleting the duplicate
|
|
|
|
Before DELETE, old=REPLACE, inserting a new key
|
|
|
|
After DELETE, old=REPLACE, inserting a new key
|
|
|
|
After INSERT, new=REPLACE, deleting the duplicate
|
|
|
|
truncate t1;
|
|
|
|
truncate t1_op_log;
|
|
|
|
create table if not exists v1
|
|
|
|
select NULL, "CREATE TABLE ... SELECT, inserting a new key";
|
|
|
|
Warnings:
|
|
|
|
Note 1050 Table 'v1' already exists
|
|
|
|
set @id=last_insert_id();
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 CREATE TABLE ... SELECT, inserting a new key
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=CREATE TABLE ... SELECT, inserting a new key
|
|
|
|
After INSERT, new=CREATE TABLE ... SELECT, inserting a new key
|
|
|
|
truncate t1_op_log;
|
|
|
|
create table if not exists v1 replace
|
|
|
|
select @id, "CREATE TABLE ... REPLACE SELECT, deleting a duplicate key";
|
|
|
|
Warnings:
|
|
|
|
Note 1050 Table 'v1' already exists
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 CREATE TABLE ... REPLACE SELECT, deleting a duplicate key
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=CREATE TABLE ... REPLACE SELECT, deleting a duplicate key
|
|
|
|
Before DELETE, old=CREATE TABLE ... SELECT, inserting a new key
|
|
|
|
After DELETE, old=CREATE TABLE ... SELECT, inserting a new key
|
|
|
|
After INSERT, new=CREATE TABLE ... REPLACE SELECT, deleting a duplicate key
|
|
|
|
truncate t1;
|
|
|
|
truncate t1_op_log;
|
|
|
|
insert into v1 (id, operation)
|
|
|
|
select NULL, "INSERT ... SELECT, inserting a new key";
|
|
|
|
set @id=last_insert_id();
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 INSERT ... SELECT, inserting a new key
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT ... SELECT, inserting a new key
|
|
|
|
After INSERT, new=INSERT ... SELECT, inserting a new key
|
|
|
|
truncate t1_op_log;
|
|
|
|
insert into v1 (id, operation)
|
|
|
|
select @id,
|
|
|
|
"INSERT ... SELECT ... ON DUPLICATE KEY UPDATE, updating a duplicate"
|
|
|
|
on duplicate key update id=NULL,
|
|
|
|
operation="INSERT ... SELECT ... ON DUPLICATE KEY UPDATE, updating a duplicate";
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
0 INSERT ... SELECT ... ON DUPLICATE KEY UPDATE, updating a duplicate
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT ... SELECT ... ON DUPLICATE KEY UPDATE, updating a duplicate
|
|
|
|
Before UPDATE, new=INSERT ... SELECT ... ON DUPLICATE KEY UPDATE, updating a duplicate, old=INSERT ... SELECT, inserting a new key
|
|
|
|
After UPDATE, new=INSERT ... SELECT ... ON DUPLICATE KEY UPDATE, updating a duplicate, old=INSERT ... SELECT, inserting a new key
|
|
|
|
truncate t1;
|
|
|
|
truncate t1_op_log;
|
|
|
|
replace into v1 (id, operation)
|
|
|
|
select NULL, "REPLACE ... SELECT, inserting a new key";
|
|
|
|
set @id=last_insert_id();
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 REPLACE ... SELECT, inserting a new key
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=REPLACE ... SELECT, inserting a new key
|
|
|
|
After INSERT, new=REPLACE ... SELECT, inserting a new key
|
|
|
|
truncate t1_op_log;
|
|
|
|
replace into v1 (id, operation)
|
|
|
|
select @id, "REPLACE ... SELECT, deleting a duplicate";
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
1 REPLACE ... SELECT, deleting a duplicate
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=REPLACE ... SELECT, deleting a duplicate
|
|
|
|
Before DELETE, old=REPLACE ... SELECT, inserting a new key
|
|
|
|
After DELETE, old=REPLACE ... SELECT, inserting a new key
|
|
|
|
After INSERT, new=REPLACE ... SELECT, deleting a duplicate
|
|
|
|
truncate t1;
|
|
|
|
truncate t1_op_log;
|
|
|
|
insert into v1 (id, operation) values (1, "INSERT for multi-DELETE");
|
|
|
|
insert into t2 (id) values (1);
|
|
|
|
delete v1.*, t2.* from v1, t2 where v1.id=1;
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
select * from t2;
|
|
|
|
id
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT for multi-DELETE
|
|
|
|
After INSERT, new=INSERT for multi-DELETE
|
|
|
|
Before DELETE, old=INSERT for multi-DELETE
|
|
|
|
After DELETE, old=INSERT for multi-DELETE
|
|
|
|
truncate t1;
|
|
|
|
truncate t2;
|
|
|
|
truncate t1_op_log;
|
|
|
|
insert into v1 (id, operation) values (1, "INSERT for multi-UPDATE");
|
|
|
|
insert into t2 (id) values (1);
|
|
|
|
update v1, t2 set v1.id=2, operation="multi-UPDATE" where v1.id=1;
|
|
|
|
update v1, t2
|
|
|
|
set t2.id=3, operation="multi-UPDATE, SET for t2, but the trigger is fired" where v1.id=2;
|
|
|
|
select * from t1;
|
|
|
|
id operation
|
|
|
|
2 multi-UPDATE, SET for t2, but the trigger is fired
|
|
|
|
select * from t2;
|
|
|
|
id
|
|
|
|
3
|
|
|
|
select * from t1_op_log;
|
|
|
|
operation
|
|
|
|
Before INSERT, new=INSERT for multi-UPDATE
|
|
|
|
After INSERT, new=INSERT for multi-UPDATE
|
|
|
|
Before UPDATE, new=multi-UPDATE, old=INSERT for multi-UPDATE
|
|
|
|
After UPDATE, new=multi-UPDATE, old=INSERT for multi-UPDATE
|
|
|
|
Before UPDATE, new=multi-UPDATE, SET for t2, but the trigger is fired, old=multi-UPDATE
|
|
|
|
After UPDATE, new=multi-UPDATE, SET for t2, but the trigger is fired, old=multi-UPDATE
|
|
|
|
drop view v1;
|
|
|
|
drop table t1, t2, t1_op_log;
|
2007-07-18 14:42:06 +04:00
|
|
|
|
|
|
|
Bug#27248 Triggers: error if insert affects temporary table
|
|
|
|
|
|
|
|
The bug was fixed by the fix for Bug#26141
|
|
|
|
|
|
|
|
drop table if exists t1;
|
|
|
|
drop temporary table if exists t2;
|
|
|
|
create table t1 (s1 int);
|
|
|
|
create temporary table t2 (s1 int);
|
|
|
|
create trigger t1_bi before insert on t1 for each row insert into t2 values (0);
|
|
|
|
create trigger t1_bd before delete on t1 for each row delete from t2;
|
|
|
|
insert into t1 values (0);
|
|
|
|
insert into t1 values (0);
|
|
|
|
select * from t1;
|
|
|
|
s1
|
|
|
|
0
|
|
|
|
0
|
|
|
|
select * from t2;
|
|
|
|
s1
|
|
|
|
0
|
|
|
|
0
|
|
|
|
delete from t1;
|
|
|
|
select * from t1;
|
|
|
|
s1
|
|
|
|
select * from t2;
|
|
|
|
s1
|
|
|
|
drop table t1;
|
|
|
|
drop temporary table t2;
|
2006-06-27 17:16:02 -07:00
|
|
|
End of 5.0 tests
|
Bug#25411 (trigger code truncated), PART II
Bug 28127 (Some valid identifiers names are not parsed correctly)
Bug 26302 (MySQL server cuts off trailing "*/" from comments in SP/func)
This patch is the second part of a major cleanup, required to fix
Bug 25411 (trigger code truncated).
The root cause of the issue stems from the function skip_rear_comments,
which was a work around to remove "extra" "*/" characters from the query
text, when parsing a query and reusing the text fragments to represent a
view, trigger, function or stored procedure.
The reason for this work around is that "special comments",
like /*!50002 XXX */, were not parsed properly, so that a query like:
AAA /*!50002 BBB */ CCC
would be seen by the parser as "AAA BBB */ CCC" when the current version
is greater or equal to 5.0.2
The root cause of this stems from how special comments are parsed.
Special comments are really out-of-bound text that appear inside a query,
that affects how the parser behave.
In nature, /*!50002 XXX */ in MySQL is similar to the C concept
of preprocessing :
#if VERSION >= 50002
XXX
#endif
Depending on the current VERSION of the server, either the special comment
should be expanded or it should be ignored, but in all cases the "text" of
the query should be re-written to strip the "/*!50002" and "*/" markers,
which does not belong to the SQL language itself.
Prior to this fix, these markers would leak into :
- the storage format for VIEW,
- the storage format for FUNCTION,
- the storage format for FUNCTION parameters, in mysql.proc (param_list),
- the storage format for PROCEDURE,
- the storage format for PROCEDURE parameters, in mysql.proc (param_list),
- the storage format for TRIGGER,
- the binary log used for replication.
In all cases, not only this cause format corruption, but also provide a vector
for dormant security issues, by allowing to tunnel code that will be activated
after an upgrade.
The proper solution is to deal with special comments strictly during parsing,
when accepting a query from the outside world.
Once a query is parsed and an object is created with a persistant
representation, this object should not arbitrarily mutate after an upgrade.
In short, special comments are a useful but limited feature for MYSQLdump,
when used at an *interface* level to facilitate import/export,
but bloating the server *internal* storage format is *not* the proper way
to deal with configuration management of the user logic.
With this fix:
- the Lex_input_stream class now acts as a comment pre-processor,
and either expands or ignore special comments on the fly.
- MYSQLlex and sql_yacc.yy have been cleaned up to strictly use the
public interface of Lex_input_stream. In particular, how the input stream
accepts or rejects a character is private to Lex_input_stream, and the
internal buffer pointers of that class are strictly private, and should not
be tempered with during parsing.
This caused many changes mostly in sql_lex.cc.
During the code cleanup in case MY_LEX_NUMBER_IDENT,
Bug 28127 (Some valid identifiers names are not parsed correctly)
was found and fixed.
By parsing special comments properly, and removing the function
'skip_rear_comments' [sic],
Bug 26302 (MySQL server cuts off trailing "*/" from comments in SP/func)
has been fixed as well.
sql/event_data_objects.cc:
Cleanup of the code that extracts the query text
sql/sp.cc:
Cleanup of the code that extracts the query text
sql/sp_head.cc:
Cleanup of the code that extracts the query text
sql/sql_trigger.cc:
Cleanup of the code that extracts the query text
sql/sql_view.cc:
Cleanup of the code that extracts the query text
mysql-test/r/comments.result:
Bug#25411 (trigger code truncated)
mysql-test/r/sp.result:
Bug#25411 (trigger code truncated)
Bug 26302 (MySQL server cuts off trailing "*/" from comments in SP/func)
mysql-test/r/trigger.result:
Bug#25411 (trigger code truncated)
mysql-test/r/varbinary.result:
Bug 28127 (Some valid identifiers names are not parsed correctly)
mysql-test/t/comments.test:
Bug#25411 (trigger code truncated)
mysql-test/t/sp.test:
Bug#25411 (trigger code truncated)
Bug 26302 (MySQL server cuts off trailing "*/" from comments in SP/func)
mysql-test/t/trigger.test:
Bug#25411 (trigger code truncated)
mysql-test/t/varbinary.test:
Bug 28127 (Some valid identifiers names are not parsed correctly)
sql/sql_lex.cc:
Implemented comment pre-processing in Lex_input_stream,
major cleanup of the lex/yacc code to not use Lex_input_stream private members.
sql/sql_lex.h:
Implemented comment pre-processing in Lex_input_stream,
major cleanup of the lex/yacc code to not use Lex_input_stream private members.
sql/sql_yacc.yy:
post merge fix : view_check_options must be parsed before signaling the end of the query
2007-06-12 15:23:58 -06:00
|
|
|
drop table if exists table_25411_a;
|
|
|
|
drop table if exists table_25411_b;
|
|
|
|
create table table_25411_a(a int);
|
|
|
|
create table table_25411_b(b int);
|
|
|
|
create trigger trg_25411a_ai after insert on table_25411_a
|
|
|
|
for each row
|
|
|
|
insert into table_25411_b select new.*;
|
|
|
|
select * from table_25411_a;
|
|
|
|
a
|
|
|
|
insert into table_25411_a values (1);
|
|
|
|
ERROR 42S02: Unknown table 'new'
|
|
|
|
select * from table_25411_a;
|
|
|
|
a
|
|
|
|
1
|
|
|
|
drop table table_25411_a;
|
|
|
|
drop table table_25411_b;
|