Commit 013512ab authored by bell@sanja.is.com.ua's avatar bell@sanja.is.com.ua

merge

parents 3bddaaf8 392292a4
...@@ -380,4 +380,6 @@ ...@@ -380,4 +380,6 @@
#define ER_TRG_ON_VIEW_OR_TEMP_TABLE 1361 #define ER_TRG_ON_VIEW_OR_TEMP_TABLE 1361
#define ER_TRG_CANT_CHANGE_ROW 1362 #define ER_TRG_CANT_CHANGE_ROW 1362
#define ER_TRG_NO_SUCH_ROW_IN_TRG 1363 #define ER_TRG_NO_SUCH_ROW_IN_TRG 1363
#define ER_ERROR_MESSAGES 364 #define ER_VIEW_NONUPD_CHECK 1364
#define ER_VIEW_CHECK_FAILED 1365
#define ER_ERROR_MESSAGES 366
...@@ -475,8 +475,10 @@ drop view v1; ...@@ -475,8 +475,10 @@ drop view v1;
drop table t1; drop table t1;
create table t1 (a int); create table t1 (a int);
create view v1 as select distinct a from t1 WITH CHECK OPTION; create view v1 as select distinct a from t1 WITH CHECK OPTION;
create view v2 as select distinct a from t1 WITH CASCADED CHECK OPTION; ERROR HY000: CHECK OPTION on non-updatable view 'test.v1'
create view v3 as select distinct a from t1 WITH LOCAL CHECK OPTION; create view v1 as select a from t1 WITH CHECK OPTION;
create view v2 as select a from t1 WITH CASCADED CHECK OPTION;
create view v3 as select a from t1 WITH LOCAL CHECK OPTION;
drop view v3 RESTRICT; drop view v3 RESTRICT;
drop view v2 CASCADE; drop view v2 CASCADE;
drop view v1; drop view v1;
...@@ -1338,3 +1340,87 @@ select * from t2; ...@@ -1338,3 +1340,87 @@ select * from t2;
ERROR HY000: Table 't2' was not locked with LOCK TABLES ERROR HY000: Table 't2' was not locked with LOCK TABLES
drop view v1; drop view v1;
drop table t1, t2; drop table t1, t2;
create table t1 (a int);
create view v1 as select * from t1 where a < 2 with check option;
insert into v1 values(1);
insert into v1 values(3);
ERROR HY000: CHECK OPTION failed 'test.v1'
insert ignore into v1 values (2),(3),(0);
Warnings:
Error 1359 CHECK OPTION failed 'test.v1'
Error 1359 CHECK OPTION failed 'test.v1'
select * from t1;
a
1
0
delete from t1;
insert into v1 SELECT 1;
insert into v1 SELECT 3;
ERROR HY000: CHECK OPTION failed 'test.v1'
create table t2 (a int);
insert into t2 values (2),(3),(0);
insert ignore into v1 SELECT a from t2;
Warnings:
Error 1359 CHECK OPTION failed 'test.v1'
Error 1359 CHECK OPTION failed 'test.v1'
select * from t1;
a
1
0
update v1 set a=-1 where a=0;
update v1 set a=2 where a=1;
ERROR HY000: CHECK OPTION failed 'test.v1'
select * from t1;
a
1
-1
update v1 set a=0 where a=0;
insert into t2 values (1);
update v1,t2 set v1.a=v1.a-1 where v1.a=t2.a;
select * from t1;
a
0
-1
update v1 set a=a+1;
update ignore v1,t2 set v1.a=v1.a+1 where v1.a=t2.a;
Warnings:
Error 1359 CHECK OPTION failed 'test.v1'
select * from t1;
a
1
1
drop view v1;
drop table t1, t2;
create table t1 (a int);
create view v1 as select * from t1 where a < 2 with check option;
create view v2 as select * from v1 where a > 0 with local check option;
create view v3 as select * from v1 where a > 0 with cascaded check option;
insert into v2 values (1);
insert into v3 values (1);
insert into v2 values (0);
ERROR HY000: CHECK OPTION failed 'test.v2'
insert into v3 values (0);
ERROR HY000: CHECK OPTION failed 'test.v3'
insert into v2 values (2);
insert into v3 values (2);
ERROR HY000: CHECK OPTION failed 'test.v3'
select * from t1;
a
1
1
2
drop view v3,v2,v1;
drop table t1;
create table t1 (a int, primary key (a));
create view v1 as select * from t1 where a < 2 with check option;
insert into v1 values (1) on duplicate key update a=2;
insert into v1 values (1) on duplicate key update a=2;
ERROR HY000: CHECK OPTION failed 'test.v1'
insert ignore into v1 values (1) on duplicate key update a=2;
Warnings:
Error 1359 CHECK OPTION failed 'test.v1'
select * from t1;
a
1
drop view v1;
drop table t1;
...@@ -394,9 +394,11 @@ drop table t1; ...@@ -394,9 +394,11 @@ drop table t1;
# syntax compatibility # syntax compatibility
# #
create table t1 (a int); create table t1 (a int);
-- error 1358
create view v1 as select distinct a from t1 WITH CHECK OPTION; create view v1 as select distinct a from t1 WITH CHECK OPTION;
create view v2 as select distinct a from t1 WITH CASCADED CHECK OPTION; create view v1 as select a from t1 WITH CHECK OPTION;
create view v3 as select distinct a from t1 WITH LOCAL CHECK OPTION; create view v2 as select a from t1 WITH CASCADED CHECK OPTION;
create view v3 as select a from t1 WITH LOCAL CHECK OPTION;
drop view v3 RESTRICT; drop view v3 RESTRICT;
drop view v2 CASCADE; drop view v2 CASCADE;
drop view v1; drop view v1;
...@@ -1308,3 +1310,80 @@ select * from v1; ...@@ -1308,3 +1310,80 @@ select * from v1;
select * from t2; select * from t2;
drop view v1; drop view v1;
drop table t1, t2; drop table t1, t2;
#
# WITH CHECK OPTION insert/update test
#
create table t1 (a int);
create view v1 as select * from t1 where a < 2 with check option;
# simple insert
insert into v1 values(1);
-- error 1359
insert into v1 values(3);
# simple insert with ignore
insert ignore into v1 values (2),(3),(0);
select * from t1;
# prepare data for next check
delete from t1;
# INSERT SELECT test
insert into v1 SELECT 1;
-- error 1359
insert into v1 SELECT 3;
# prepare data for next check
create table t2 (a int);
insert into t2 values (2),(3),(0);
# INSERT SELECT with ignore test
insert ignore into v1 SELECT a from t2;
select * from t1;
#simple UPDATE test
update v1 set a=-1 where a=0;
-- error 1359
update v1 set a=2 where a=1;
select * from t1;
# prepare data for next check
update v1 set a=0 where a=0;
insert into t2 values (1);
# multiupdate test
update v1,t2 set v1.a=v1.a-1 where v1.a=t2.a;
select * from t1;
# prepare data for next check
update v1 set a=a+1;
# multiupdate with ignore test
update ignore v1,t2 set v1.a=v1.a+1 where v1.a=t2.a;
select * from t1;
drop view v1;
drop table t1, t2;
#
# CASCADED/LOCAL CHECK OPTION test
#
create table t1 (a int);
create view v1 as select * from t1 where a < 2 with check option;
create view v2 as select * from v1 where a > 0 with local check option;
create view v3 as select * from v1 where a > 0 with cascaded check option;
insert into v2 values (1);
insert into v3 values (1);
-- error 1359
insert into v2 values (0);
-- error 1359
insert into v3 values (0);
insert into v2 values (2);
-- error 1359
insert into v3 values (2);
select * from t1;
drop view v3,v2,v1;
drop table t1;
#
# CHECK OPTION with INSERT ... ON DUPLICATE KEY UPDATE
#
create table t1 (a int, primary key (a));
create view v1 as select * from t1 where a < 2 with check option;
insert into v1 values (1) on duplicate key update a=2;
-- error 1359
insert into v1 values (1) on duplicate key update a=2;
insert ignore into v1 values (1) on duplicate key update a=2;
select * from t1;
drop view v1;
drop table t1;
...@@ -392,3 +392,5 @@ character-set=latin2 ...@@ -392,3 +392,5 @@ character-set=latin2
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -383,3 +383,5 @@ character-set=latin1 ...@@ -383,3 +383,5 @@ character-set=latin1
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -392,3 +392,5 @@ character-set=latin1 ...@@ -392,3 +392,5 @@ character-set=latin1
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -380,3 +380,5 @@ character-set=latin1 ...@@ -380,3 +380,5 @@ character-set=latin1
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -385,3 +385,5 @@ character-set=latin7 ...@@ -385,3 +385,5 @@ character-set=latin7
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -380,3 +380,5 @@ character-set=latin1 ...@@ -380,3 +380,5 @@ character-set=latin1
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -393,3 +393,5 @@ character-set=latin1 ...@@ -393,3 +393,5 @@ character-set=latin1
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -380,3 +380,5 @@ character-set=greek ...@@ -380,3 +380,5 @@ character-set=greek
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -385,3 +385,5 @@ character-set=latin2 ...@@ -385,3 +385,5 @@ character-set=latin2
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -380,3 +380,5 @@ character-set=latin1 ...@@ -380,3 +380,5 @@ character-set=latin1
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -384,3 +384,5 @@ character-set=ujis ...@@ -384,3 +384,5 @@ character-set=ujis
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -380,3 +380,5 @@ character-set=euckr ...@@ -380,3 +380,5 @@ character-set=euckr
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -382,3 +382,5 @@ character-set=latin1 ...@@ -382,3 +382,5 @@ character-set=latin1
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -382,3 +382,5 @@ character-set=latin1 ...@@ -382,3 +382,5 @@ character-set=latin1
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -385,3 +385,5 @@ character-set=latin2 ...@@ -385,3 +385,5 @@ character-set=latin2
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -382,3 +382,5 @@ character-set=latin1 ...@@ -382,3 +382,5 @@ character-set=latin1
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -385,3 +385,5 @@ character-set=latin2 ...@@ -385,3 +385,5 @@ character-set=latin2
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -385,3 +385,5 @@ character-set=koi8r ...@@ -385,3 +385,5 @@ character-set=koi8r
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION VIEW '%-.64s.%-.64s'"
" CHECK OPTION VIEW '%-.64s.%-.64s' "
...@@ -373,3 +373,5 @@ character-set=cp1250 ...@@ -373,3 +373,5 @@ character-set=cp1250
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -388,3 +388,5 @@ character-set=latin2 ...@@ -388,3 +388,5 @@ character-set=latin2
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -384,3 +384,5 @@ character-set=latin1 ...@@ -384,3 +384,5 @@ character-set=latin1
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -380,3 +380,5 @@ character-set=latin1 ...@@ -380,3 +380,5 @@ character-set=latin1
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION on non-updatable view '%-.64s.%-.64s'"
"CHECK OPTION failed '%-.64s.%-.64s'"
...@@ -386,3 +386,5 @@ character-set=koi8u ...@@ -386,3 +386,5 @@ character-set=koi8u
"Trigger's '%-.64s' is view or temporary table" "Trigger's '%-.64s' is view or temporary table"
"Updating of %s row is not allowed in %strigger" "Updating of %s row is not allowed in %strigger"
"There is no %s row in %s trigger" "There is no %s row in %s trigger"
"CHECK OPTION VIEW '%-.64s.%-.64s' "
"צ CHECK OPTION VIEW '%-.64s.%-.64s' "
...@@ -227,6 +227,9 @@ typedef struct st_copy_info { ...@@ -227,6 +227,9 @@ typedef struct st_copy_info {
/* for INSERT ... UPDATE */ /* for INSERT ... UPDATE */
List<Item> *update_fields; List<Item> *update_fields;
List<Item> *update_values; List<Item> *update_values;
/* for VIEW ... WITH CHECK OPTION */
TABLE_LIST *view;
bool ignore;
} COPY_INFO; } COPY_INFO;
...@@ -1271,14 +1274,8 @@ class select_insert :public select_result { ...@@ -1271,14 +1274,8 @@ class select_insert :public select_result {
bool insert_into_view; bool insert_into_view;
select_insert(TABLE_LIST *table_list_par, TABLE *table_par, select_insert(TABLE_LIST *table_list_par, TABLE *table_par,
List<Item> *fields_par, enum_duplicates duplic) List<Item> *fields_par, enum_duplicates duplic,
:table_list(table_list_par), table(table_par), fields(fields_par), bool ignore_check_option_errors);
last_insert_id(0),
insert_into_view(table_list_par && table_list_par->view != 0)
{
bzero((char*) &info,sizeof(info));
info.handle_duplicates=duplic;
}
~select_insert(); ~select_insert();
int prepare(List<Item> &list, SELECT_LEX_UNIT *u); int prepare(List<Item> &list, SELECT_LEX_UNIT *u);
bool send_fields(List<Item> &list, uint flags) { return 0; } bool send_fields(List<Item> &list, uint flags) { return 0; }
...@@ -1304,7 +1301,7 @@ public: ...@@ -1304,7 +1301,7 @@ public:
List<create_field> &fields_par, List<create_field> &fields_par,
List<Key> &keys_par, List<Key> &keys_par,
List<Item> &select_fields,enum_duplicates duplic) List<Item> &select_fields,enum_duplicates duplic)
:select_insert (NULL, NULL, &select_fields, duplic), create_table(table), :select_insert (NULL, NULL, &select_fields, duplic, 0), create_table(table),
extra_fields(&fields_par),keys(&keys_par), create_info(create_info_par), extra_fields(&fields_par),keys(&keys_par), create_info(create_info_par),
lock(0) lock(0)
{} {}
......
...@@ -130,6 +130,7 @@ int mysql_insert(THD *thd,TABLE_LIST *table_list, ...@@ -130,6 +130,7 @@ int mysql_insert(THD *thd,TABLE_LIST *table_list,
*/ */
bool log_on= (thd->options & OPTION_BIN_LOG) || (!(thd->master_access & SUPER_ACL)); bool log_on= (thd->options & OPTION_BIN_LOG) || (!(thd->master_access & SUPER_ACL));
bool transactional_table, log_delayed; bool transactional_table, log_delayed;
bool ignore_err= (thd->lex->duplicates == DUP_IGNORE);
uint value_count; uint value_count;
ulong counter = 1; ulong counter = 1;
ulonglong id; ulonglong id;
...@@ -243,6 +244,8 @@ int mysql_insert(THD *thd,TABLE_LIST *table_list, ...@@ -243,6 +244,8 @@ int mysql_insert(THD *thd,TABLE_LIST *table_list,
info.handle_duplicates=duplic; info.handle_duplicates=duplic;
info.update_fields=&update_fields; info.update_fields=&update_fields;
info.update_values=&update_values; info.update_values=&update_values;
info.view= (table_list->view ? table_list : 0);
info.ignore= ignore_err;
/* /*
Count warnings for all inserts. Count warnings for all inserts.
For single line insert, generate an error if try to set a NOT NULL field For single line insert, generate an error if try to set a NOT NULL field
...@@ -304,6 +307,14 @@ int mysql_insert(THD *thd,TABLE_LIST *table_list, ...@@ -304,6 +307,14 @@ int mysql_insert(THD *thd,TABLE_LIST *table_list,
break; break;
} }
} }
if ((res= table_list->view_check_option(thd, ignore_err)) ==
VIEW_CHECK_SKIP)
continue;
else if (res == VIEW_CHECK_ERROR)
{
error= 1;
break;
}
// FIXME: Actually we should do this before check_null_fields. // FIXME: Actually we should do this before check_null_fields.
// Or even go into write_record ? // Or even go into write_record ?
...@@ -699,6 +710,7 @@ int write_record(TABLE *table,COPY_INFO *info) ...@@ -699,6 +710,7 @@ int write_record(TABLE *table,COPY_INFO *info)
} }
if (info->handle_duplicates == DUP_UPDATE) if (info->handle_duplicates == DUP_UPDATE)
{ {
int res= 0;
/* we don't check for other UNIQUE keys - the first row /* we don't check for other UNIQUE keys - the first row
that matches, is updated. If update causes a conflict again, that matches, is updated. If update causes a conflict again,
an error is returned an error is returned
...@@ -707,6 +719,15 @@ int write_record(TABLE *table,COPY_INFO *info) ...@@ -707,6 +719,15 @@ int write_record(TABLE *table,COPY_INFO *info)
restore_record(table,record[1]); restore_record(table,record[1]);
if (fill_record(*info->update_fields, *info->update_values, 0)) if (fill_record(*info->update_fields, *info->update_values, 0))
goto err; goto err;
/* CHECK OPTION for VIEW ... ON DUPLICATE KEY UPDATE ... */
if (info->view &&
(res= info->view->view_check_option(current_thd, info->ignore)) ==
VIEW_CHECK_SKIP)
break;
else if (res == VIEW_CHECK_ERROR)
goto err;
if ((error=table->file->update_row(table->record[1],table->record[0]))) if ((error=table->file->update_row(table->record[1],table->record[0])))
goto err; goto err;
info->updated++; info->updated++;
...@@ -1614,6 +1635,11 @@ int mysql_insert_select_prepare(THD *thd) ...@@ -1614,6 +1635,11 @@ int mysql_insert_select_prepare(THD *thd)
{ {
LEX *lex= thd->lex; LEX *lex= thd->lex;
DBUG_ENTER("mysql_insert_select_prepare"); DBUG_ENTER("mysql_insert_select_prepare");
/*
SELECT_LEX do not belong to INSERT statement, so we can't add WHERE
clasue if table is VIEW
*/
lex->query_tables->no_where_clause= 1;
if (mysql_prepare_insert_check_table(thd, lex->query_tables, if (mysql_prepare_insert_check_table(thd, lex->query_tables,
lex->field_list, lex->field_list,
&lex->select_lex.where)) &lex->select_lex.where))
...@@ -1622,6 +1648,21 @@ int mysql_insert_select_prepare(THD *thd) ...@@ -1622,6 +1648,21 @@ int mysql_insert_select_prepare(THD *thd)
} }
select_insert::select_insert(TABLE_LIST *table_list_par, TABLE *table_par,
List<Item> *fields_par, enum_duplicates duplic,
bool ignore_check_option_errors)
:table_list(table_list_par), table(table_par), fields(fields_par),
last_insert_id(0),
insert_into_view(table_list_par && table_list_par->view != 0)
{
bzero((char*) &info,sizeof(info));
info.handle_duplicates=duplic;
if (table_list_par)
info.view= (table_list_par->view ? table_list_par : 0);
info.ignore= ignore_check_option_errors;
}
int int
select_insert::prepare(List<Item> &values, SELECT_LEX_UNIT *u) select_insert::prepare(List<Item> &values, SELECT_LEX_UNIT *u)
{ {
...@@ -1675,6 +1716,14 @@ bool select_insert::send_data(List<Item> &values) ...@@ -1675,6 +1716,14 @@ bool select_insert::send_data(List<Item> &values)
fill_record(*fields, values, 1); fill_record(*fields, values, 1);
else else
fill_record(table->field, values, 1); fill_record(table->field, values, 1);
switch (table_list->view_check_option(thd,
thd->lex->duplicates == DUP_IGNORE))
{
case VIEW_CHECK_SKIP:
DBUG_RETURN(0);
case VIEW_CHECK_ERROR:
DBUG_RETURN(1);
}
if (thd->net.report_error || write_record(table,&info)) if (thd->net.report_error || write_record(table,&info))
DBUG_RETURN(1); DBUG_RETURN(1);
if (table->next_number_field) // Clear for next record if (table->next_number_field) // Clear for next record
......
...@@ -691,6 +691,7 @@ typedef struct st_lex ...@@ -691,6 +691,7 @@ typedef struct st_lex
uint8 describe; uint8 describe;
uint8 derived_tables; uint8 derived_tables;
uint8 create_view_algorithm; uint8 create_view_algorithm;
uint8 create_view_check;
bool drop_if_exists, drop_temporary, local_file, one_shot_set; bool drop_if_exists, drop_temporary, local_file, one_shot_set;
bool in_comment, ignore_space, verbose, no_write_to_binlog; bool in_comment, ignore_space, verbose, no_write_to_binlog;
/* special JOIN::prepare mode: changing of query is prohibited */ /* special JOIN::prepare mode: changing of query is prohibited */
......
...@@ -2818,7 +2818,8 @@ unsent_create_error: ...@@ -2818,7 +2818,8 @@ unsent_create_error:
if ((res= mysql_insert_select_prepare(thd))) if ((res= mysql_insert_select_prepare(thd)))
break; break;
if ((result= new select_insert(first_table, first_table->table, if ((result= new select_insert(first_table, first_table->table,
&lex->field_list, lex->duplicates))) &lex->field_list, lex->duplicates,
lex->duplicates == DUP_IGNORE)))
{ {
/* Skip first table, which is the table we are inserting in */ /* Skip first table, which is the table we are inserting in */
lex->select_lex.table_list.first= (byte*) first_table->next_local; lex->select_lex.table_list.first= (byte*) first_table->next_local;
......
...@@ -97,10 +97,12 @@ int mysql_update(THD *thd, ...@@ -97,10 +97,12 @@ int mysql_update(THD *thd,
ha_rows limit, ha_rows limit,
enum enum_duplicates handle_duplicates) enum enum_duplicates handle_duplicates)
{ {
bool using_limit=limit != HA_POS_ERROR; bool using_limit= limit != HA_POS_ERROR;
bool safe_update= thd->options & OPTION_SAFE_UPDATES; bool safe_update= thd->options & OPTION_SAFE_UPDATES;
bool used_key_is_modified, transactional_table, log_delayed; bool used_key_is_modified, transactional_table, log_delayed;
bool ignore_err= (thd->lex->duplicates == DUP_IGNORE);
int error=0; int error=0;
int res;
uint used_index; uint used_index;
#ifndef NO_EMBEDDED_ACCESS_CHECKS #ifndef NO_EMBEDDED_ACCESS_CHECKS
uint want_privilege; uint want_privilege;
...@@ -152,7 +154,7 @@ int mysql_update(THD *thd, ...@@ -152,7 +154,7 @@ int mysql_update(THD *thd,
#endif #endif
{ {
thd->lex->select_lex.no_wrap_view_item= 1; thd->lex->select_lex.no_wrap_view_item= 1;
int res= setup_fields(thd, 0, table_list, fields, 1, 0, 0); res= setup_fields(thd, 0, table_list, fields, 1, 0, 0);
thd->lex->select_lex.no_wrap_view_item= 0; thd->lex->select_lex.no_wrap_view_item= 0;
if (res) if (res)
DBUG_RETURN(-1); /* purecov: inspected */ DBUG_RETURN(-1); /* purecov: inspected */
...@@ -363,6 +365,18 @@ int mysql_update(THD *thd, ...@@ -363,6 +365,18 @@ int mysql_update(THD *thd,
if (compare_record(table, query_id)) if (compare_record(table, query_id))
{ {
if ((res= table_list->view_check_option(thd, ignore_err)) !=
VIEW_CHECK_OK)
{
found--;
if (res == VIEW_CHECK_SKIP)
continue;
else if (res == VIEW_CHECK_ERROR)
{
error= 1;
break;
}
}
if (!(error=table->file->update_row((byte*) table->record[1], if (!(error=table->file->update_row((byte*) table->record[1],
(byte*) table->record[0]))) (byte*) table->record[0])))
{ {
...@@ -966,6 +980,7 @@ multi_update::~multi_update() ...@@ -966,6 +980,7 @@ multi_update::~multi_update()
bool multi_update::send_data(List<Item> &not_used_values) bool multi_update::send_data(List<Item> &not_used_values)
{ {
TABLE_LIST *cur_table; TABLE_LIST *cur_table;
bool ignore_err= (thd->lex->duplicates == DUP_IGNORE);
DBUG_ENTER("multi_update::send_data"); DBUG_ENTER("multi_update::send_data");
for (cur_table= update_tables; cur_table; cur_table= cur_table->next_local) for (cur_table= update_tables; cur_table; cur_table= cur_table->next_local)
...@@ -998,6 +1013,15 @@ bool multi_update::send_data(List<Item> &not_used_values) ...@@ -998,6 +1013,15 @@ bool multi_update::send_data(List<Item> &not_used_values)
if (compare_record(table, thd->query_id)) if (compare_record(table, thd->query_id))
{ {
int error; int error;
if ((error= cur_table->view_check_option(thd, ignore_err)) !=
VIEW_CHECK_OK)
{
found--;
if (error == VIEW_CHECK_SKIP)
continue;
else if (error == VIEW_CHECK_ERROR)
DBUG_RETURN(1);
}
if (!updated++) if (!updated++)
{ {
/* /*
......
...@@ -302,7 +302,6 @@ static const int required_view_parameters= 7; ...@@ -302,7 +302,6 @@ static const int required_view_parameters= 7;
Note that one should NOT change the order for this, as it's used by Note that one should NOT change the order for this, as it's used by
parse() parse()
*/ */
static File_option view_parameters[]= static File_option view_parameters[]=
{{{(char*) "query", 5}, offsetof(TABLE_LIST, query), {{{(char*) "query", 5}, offsetof(TABLE_LIST, query),
FILE_OPTIONS_STRING}, FILE_OPTIONS_STRING},
...@@ -312,6 +311,8 @@ static File_option view_parameters[]= ...@@ -312,6 +311,8 @@ static File_option view_parameters[]=
FILE_OPTIONS_ULONGLONG}, FILE_OPTIONS_ULONGLONG},
{{(char*) "algorithm", 9}, offsetof(TABLE_LIST, algorithm), {{(char*) "algorithm", 9}, offsetof(TABLE_LIST, algorithm),
FILE_OPTIONS_ULONGLONG}, FILE_OPTIONS_ULONGLONG},
{{"with_check_option", 17}, offsetof(TABLE_LIST, with_check),
FILE_OPTIONS_ULONGLONG},
{{(char*) "revision", 8}, offsetof(TABLE_LIST, revision), {{(char*) "revision", 8}, offsetof(TABLE_LIST, revision),
FILE_OPTIONS_REV}, FILE_OPTIONS_REV},
{{(char*) "timestamp", 9}, offsetof(TABLE_LIST, timestamp), {{(char*) "timestamp", 9}, offsetof(TABLE_LIST, timestamp),
...@@ -393,7 +394,7 @@ static int mysql_register_view(THD *thd, TABLE_LIST *view, ...@@ -393,7 +394,7 @@ static int mysql_register_view(THD *thd, TABLE_LIST *view,
if (mode == VIEW_CREATE_NEW) if (mode == VIEW_CREATE_NEW)
{ {
my_error(ER_TABLE_EXISTS_ERROR, MYF(0), view->alias); my_error(ER_TABLE_EXISTS_ERROR, MYF(0), view->alias);
DBUG_RETURN(1); DBUG_RETURN(-1);
} }
if (!(parser= sql_parse_prepare(&path, &thd->mem_root, 0))) if (!(parser= sql_parse_prepare(&path, &thd->mem_root, 0)))
...@@ -404,7 +405,7 @@ static int mysql_register_view(THD *thd, TABLE_LIST *view, ...@@ -404,7 +405,7 @@ static int mysql_register_view(THD *thd, TABLE_LIST *view,
{ {
my_error(ER_WRONG_OBJECT, MYF(0), (view->db ? view->db : thd->db), my_error(ER_WRONG_OBJECT, MYF(0), (view->db ? view->db : thd->db),
view->real_name, "VIEW"); view->real_name, "VIEW");
DBUG_RETURN(1); DBUG_RETURN(-1);
} }
/* /*
...@@ -416,7 +417,7 @@ static int mysql_register_view(THD *thd, TABLE_LIST *view, ...@@ -416,7 +417,7 @@ static int mysql_register_view(THD *thd, TABLE_LIST *view,
if (parser->parse((gptr)view, &thd->mem_root, if (parser->parse((gptr)view, &thd->mem_root,
view_parameters + revision_number_position, 1)) view_parameters + revision_number_position, 1))
{ {
DBUG_RETURN(1); DBUG_RETURN(thd->net.report_error? -1 : 0);
} }
} }
else else
...@@ -424,7 +425,7 @@ static int mysql_register_view(THD *thd, TABLE_LIST *view, ...@@ -424,7 +425,7 @@ static int mysql_register_view(THD *thd, TABLE_LIST *view,
if (mode == VIEW_ALTER) if (mode == VIEW_ALTER)
{ {
my_error(ER_NO_SUCH_TABLE, MYF(0), view->db, view->alias); my_error(ER_NO_SUCH_TABLE, MYF(0), view->db, view->alias);
DBUG_RETURN(1); DBUG_RETURN(-1);
} }
} }
} }
...@@ -446,6 +447,7 @@ static int mysql_register_view(THD *thd, TABLE_LIST *view, ...@@ -446,6 +447,7 @@ static int mysql_register_view(THD *thd, TABLE_LIST *view,
thd->lex->create_view_algorithm= VIEW_ALGORITHM_UNDEFINED; thd->lex->create_view_algorithm= VIEW_ALGORITHM_UNDEFINED;
} }
view->algorithm= thd->lex->create_view_algorithm; view->algorithm= thd->lex->create_view_algorithm;
view->with_check= thd->lex->create_view_check;
if ((view->updatable_view= (can_be_merged && if ((view->updatable_view= (can_be_merged &&
view->algorithm != VIEW_ALGORITHM_TMPTABLE))) view->algorithm != VIEW_ALGORITHM_TMPTABLE)))
{ {
...@@ -461,10 +463,18 @@ static int mysql_register_view(THD *thd, TABLE_LIST *view, ...@@ -461,10 +463,18 @@ static int mysql_register_view(THD *thd, TABLE_LIST *view,
} }
} }
} }
if (view->with_check != VIEW_CHECK_NONE &&
!view->updatable_view)
{
my_error(ER_VIEW_NONUPD_CHECK, MYF(0), view->db, view->real_name);
DBUG_RETURN(-1);
}
if (sql_create_definition_file(&dir, &file, view_file_type, if (sql_create_definition_file(&dir, &file, view_file_type,
(gptr)view, view_parameters, 3)) (gptr)view, view_parameters, 3))
{ {
DBUG_RETURN(1); DBUG_RETURN(thd->net.report_error? -1 : 1);
} }
DBUG_RETURN(0); DBUG_RETURN(0);
} }
...@@ -720,6 +730,7 @@ mysql_make_view(File_parser *parser, TABLE_LIST *table) ...@@ -720,6 +730,7 @@ mysql_make_view(File_parser *parser, TABLE_LIST *table)
DBUG_PRINT("info", ("algorithm: TEMPORARY TABLE")); DBUG_PRINT("info", ("algorithm: TEMPORARY TABLE"));
lex->select_lex.linkage= DERIVED_TABLE_TYPE; lex->select_lex.linkage= DERIVED_TABLE_TYPE;
table->updatable= 0; table->updatable= 0;
table->with_check= VIEW_CHECK_NONE;
/* SELECT tree link */ /* SELECT tree link */
lex->unit.include_down(table->select_lex); lex->unit.include_down(table->select_lex);
......
...@@ -7898,9 +7898,13 @@ algorithm: ...@@ -7898,9 +7898,13 @@ algorithm:
{ Lex->create_view_algorithm= VIEW_ALGORITHM_TMPTABLE; } { Lex->create_view_algorithm= VIEW_ALGORITHM_TMPTABLE; }
; ;
check_option: check_option:
/* empty */ {} /* empty */
| WITH CHECK_SYM OPTION {} { Lex->create_view_check= VIEW_CHECK_NONE; }
| WITH CASCADED CHECK_SYM OPTION {} | WITH CHECK_SYM OPTION
| WITH LOCAL_SYM CHECK_SYM OPTION {} { Lex->create_view_check= VIEW_CHECK_LOCAL; }
| WITH CASCADED CHECK_SYM OPTION
{ Lex->create_view_check= VIEW_CHECK_CASCADED; }
| WITH LOCAL_SYM CHECK_SYM OPTION
{ Lex->create_view_check= VIEW_CHECK_LOCAL; }
; ;
...@@ -1594,6 +1594,14 @@ bool st_table_list::setup_ancestor(THD *thd, Item **conds) ...@@ -1594,6 +1594,14 @@ bool st_table_list::setup_ancestor(THD *thd, Item **conds)
if (arena) if (arena)
thd->set_n_backup_item_arena(arena, &backup); thd->set_n_backup_item_arena(arena, &backup);
if (with_check)
{
check_option= where->copy_andor_structure(thd);
if (with_check == VIEW_CHECK_CASCADED)
{
check_option= and_conds(check_option, ancestor->check_option);
}
}
/* Go up to join tree and try to find left join */ /* Go up to join tree and try to find left join */
for (; tbl; tbl= tbl->embedding) for (; tbl; tbl= tbl->embedding)
{ {
...@@ -1610,17 +1618,37 @@ bool st_table_list::setup_ancestor(THD *thd, Item **conds) ...@@ -1610,17 +1618,37 @@ bool st_table_list::setup_ancestor(THD *thd, Item **conds)
} }
} }
if (tbl == 0) if (tbl == 0)
{
if (outer_join)
{ {
/* /*
It is conds of JOIN, but it will be stored in st_select_lex::prep_where Store WHERE condition to ON expression for outer join, because we
for next reexecution can't use WHERE to correctly execute jeft joins on VIEWs and this
expression will not be moved to WHERE condition (i.e. will be
clean correctly for PS/SP)
*/
on_expr= and_conds(on_expr, where);
}
else
{
/*
It is conds of JOIN, but it will be stored in
st_select_lex::prep_where for next reexecution
*/ */
*conds= and_conds(*conds, where); *conds= and_conds(*conds, where);
} }
}
if (arena) if (arena)
thd->restore_backup_item_arena(arena, &backup); thd->restore_backup_item_arena(arena, &backup);
} }
/*
fix_fields do not need tables, because new are only AND operation and we
just need recollect statistics
*/
if (check_option && !check_option->fixed &&
check_option->fix_fields(thd, 0, &check_option))
goto err;
/* full text function moving to current select */ /* full text function moving to current select */
if (view->select_lex.ftfunc_list->elements) if (view->select_lex.ftfunc_list->elements)
...@@ -1654,6 +1682,40 @@ err: ...@@ -1654,6 +1682,40 @@ err:
} }
/*
check CHECK OPTION condition
SYNOPSIS
check_option()
ignore_failure ignore check option fail
RETURN
VIEW_CHECK_OK OK
VIEW_CHECK_ERROR FAILED
VIEW_CHECK_SKIP FAILED, but continue
*/
int st_table_list::view_check_option(THD *thd, bool ignore_failure)
{
if (check_option && check_option->val_int() == 0)
{
if (ignore_failure)
{
push_warning_printf(thd, MYSQL_ERROR::WARN_LEVEL_ERROR,
ER_VIEW_CHECK_FAILED, ER(ER_VIEW_CHECK_FAILED),
view_db.str, view_name.str);
return(VIEW_CHECK_SKIP);
}
else
{
my_error(ER_VIEW_CHECK_FAILED, MYF(0), view_db.str, view_name.str);
return(VIEW_CHECK_ERROR);
}
}
return(VIEW_CHECK_OK);
}
void Field_iterator_view::set(TABLE_LIST *table) void Field_iterator_view::set(TABLE_LIST *table)
{ {
ptr= table->field_translation; ptr= table->field_translation;
......
...@@ -188,6 +188,16 @@ struct st_table { ...@@ -188,6 +188,16 @@ struct st_table {
#define VIEW_ALGORITHM_TMPTABLE 1 #define VIEW_ALGORITHM_TMPTABLE 1
#define VIEW_ALGORITHM_MERGE 2 #define VIEW_ALGORITHM_MERGE 2
/* view WITH CHECK OPTION parameter options */
#define VIEW_CHECK_NONE 0
#define VIEW_CHECK_LOCAL 1
#define VIEW_CHECK_CASCADED 2
/* result of view WITH CHECK OPTION parameter check */
#define VIEW_CHECK_OK 0
#define VIEW_CHECK_ERROR 1
#define VIEW_CHECK_SKIP 2
struct st_lex; struct st_lex;
typedef struct st_table_list typedef struct st_table_list
...@@ -223,6 +233,7 @@ typedef struct st_table_list ...@@ -223,6 +233,7 @@ typedef struct st_table_list
/* next_global before adding VIEW tables */ /* next_global before adding VIEW tables */
st_table_list *old_next; st_table_list *old_next;
Item *where; /* VIEW WHERE clause condition */ Item *where; /* VIEW WHERE clause condition */
Item *check_option; /* WITH CHECK OPTION condition */
LEX_STRING query; /* text of (CRETE/SELECT) statement */ LEX_STRING query; /* text of (CRETE/SELECT) statement */
LEX_STRING md5; /* md5 of query tesxt */ LEX_STRING md5; /* md5 of query tesxt */
LEX_STRING source; /* source of CREATE VIEW */ LEX_STRING source; /* source of CREATE VIEW */
...@@ -233,6 +244,7 @@ typedef struct st_table_list ...@@ -233,6 +244,7 @@ typedef struct st_table_list
ulonglong updatable_view; /* VIEW can be updated */ ulonglong updatable_view; /* VIEW can be updated */
ulonglong revision; /* revision control number */ ulonglong revision; /* revision control number */
ulonglong algorithm; /* 0 any, 1 tmp tables , 2 merging */ ulonglong algorithm; /* 0 any, 1 tmp tables , 2 merging */
ulonglong with_check; /* WITH CHECK OPTION */
uint effective_algorithm; /* which algorithm was really used */ uint effective_algorithm; /* which algorithm was really used */
GRANT_INFO grant; GRANT_INFO grant;
thr_lock_type lock_type; thr_lock_type lock_type;
...@@ -244,6 +256,7 @@ typedef struct st_table_list ...@@ -244,6 +256,7 @@ typedef struct st_table_list
bool updating; /* for replicate-do/ignore table */ bool updating; /* for replicate-do/ignore table */
bool force_index; /* prefer index over table scan */ bool force_index; /* prefer index over table scan */
bool ignore_leaves; /* preload only non-leaf nodes */ bool ignore_leaves; /* preload only non-leaf nodes */
bool no_where_clause; /* do not attach WHERE to SELECT */
table_map dep_tables; /* tables the table depends on */ table_map dep_tables; /* tables the table depends on */
table_map on_expr_dep_tables; /* tables on expression depends on */ table_map on_expr_dep_tables; /* tables on expression depends on */
struct st_nested_join *nested_join; /* if the element is a nested join */ struct st_nested_join *nested_join; /* if the element is a nested join */
...@@ -262,6 +275,7 @@ typedef struct st_table_list ...@@ -262,6 +275,7 @@ typedef struct st_table_list
void calc_md5(char *buffer); void calc_md5(char *buffer);
void set_ancestor(); void set_ancestor();
int view_check_option(THD *thd, bool ignore_failure);
bool setup_ancestor(THD *thd, Item **conds); bool setup_ancestor(THD *thd, Item **conds);
bool placeholder() {return derived || view; } bool placeholder() {return derived || view; }
void print(THD *thd, String *str); void print(THD *thd, String *str);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment