mirror of
				https://github.com/MariaDB/server.git
				synced 2025-10-26 16:38:11 +01:00 
			
		
		
		
	 40b3525fcc
			
		
	
	
	40b3525fcc
	
	
	
		
			
			Workaround patch: Do not remove GROUP BY clause when it has
subquer(ies) in it.
remove_redundant_subquery_clauses() removes redundant GROUP BY clause
from queries in form:
  expr IN (SELECT no_aggregates GROUP BY ...)
  expr {CMP} {ALL|ANY|SOME} (SELECT no_aggregates GROUP BY ...)
This hits problems when the GROUP BY clause itself has subquer(y/ies).
This patch is just a workaround: it disables removal of GROUP BY clause
if the clause has one or more subqueries in it.
Tests:
- subselect_elimination.test has all known crashing cases.
- subselect4.result, insert_select.result are updated.
Note that in some cases results of SELECT are changed too (not just
EXPLAINs). These are caused by non-deterministic SQL: when running a
query like:
  x > ANY( SELECT col1 FROM t1 GROUP BY constant_expression)
without removing the GROUP BY, the executor is free to pick the value
of t1.col1 from any row in the GROUP BY group (denote it $COL1_VAL).
Then, it computes x > ANY(SELECT $COL1_VAL).
When running the same query and removing the GROUP BY:
   x > ANY( SELECT col1 FROM t1)
the executor will actually check all rows of t1.
		
	
			
		
			
				
	
	
		
			268 lines
		
	
	
	
		
			8.4 KiB
		
	
	
	
		
			Text
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
	
		
			8.4 KiB
		
	
	
	
		
			Text
		
	
	
	
	
	
| #
 | |
| # MDEV-28621 group by optimization incorrectly removing subquery where
 | |
| #  subject buried in a function
 | |
| #
 | |
| CREATE TABLE t1 (i int) ;
 | |
| INSERT INTO t1 VALUES (1),(2),(3);
 | |
| SELECT 1 FROM t1
 | |
| WHERE i in
 | |
| ( SELECT  a+1
 | |
| FROM
 | |
| (SELECT (SELECT i FROM (SELECT 1 FROM t1) dt) AS a FROM t1) dt2
 | |
| GROUP BY a
 | |
| );
 | |
| ERROR 21000: Subquery returns more than 1 row
 | |
| DROP TABLE t1;
 | |
| create table t1 (a int, b int, c int);
 | |
| insert into t1 select seq, seq, seq from seq_1_to_10;
 | |
| create table t2 as select * from t1;
 | |
| create table t20 as select * from t1;
 | |
| create table t21 as select * from t1;
 | |
| create table t3 as select * from t1;
 | |
| select a, a in
 | |
| (
 | |
| select
 | |
| (
 | |
| select max(c) from t20 where t20.a<=t2.a
 | |
| ) as SUBQ1 from t2 group by SUBQ1+1
 | |
| ) as COL
 | |
| from t1;
 | |
| a	COL
 | |
| 1	1
 | |
| 2	1
 | |
| 3	1
 | |
| 4	1
 | |
| 5	1
 | |
| 6	1
 | |
| 7	1
 | |
| 8	1
 | |
| 9	1
 | |
| 10	1
 | |
| create view v2 as
 | |
| select
 | |
| a, b,
 | |
| (select max(c) from t20 where t20.a<=t2.a) as SUBQ1,
 | |
| (select max(c) from t21 where t21.a<=t2.a) as SUBQ2
 | |
| from t2;
 | |
| # test partial elimination
 | |
| explain
 | |
| select
 | |
| a,
 | |
| a in (select a from v2 where a>3 and v2.SUBQ2>=0 group by v2.SUBQ1, v2.SUBQ2)
 | |
| from t1;
 | |
| id	select_type	table	type	possible_keys	key	key_len	ref	rows	Extra
 | |
| 1	PRIMARY	t1	ALL	NULL	NULL	NULL	NULL	10	
 | |
| 2	MATERIALIZED	t2	ALL	NULL	NULL	NULL	NULL	10	Using where; Using temporary
 | |
| 5	DEPENDENT SUBQUERY	t21	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| 4	DEPENDENT SUBQUERY	t20	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| # test buried subselects in group by clause
 | |
| select a, a in
 | |
| (
 | |
| select
 | |
| (
 | |
| select max(c) from t20 where t20.a<=t2.a
 | |
| )*2 as SUBQ1 from t2 group by SUBQ1+1
 | |
| ) as COL
 | |
| from t1;
 | |
| a	COL
 | |
| 1	0
 | |
| 2	1
 | |
| 3	0
 | |
| 4	1
 | |
| 5	0
 | |
| 6	1
 | |
| 7	0
 | |
| 8	1
 | |
| 9	0
 | |
| 10	1
 | |
| drop view v2;
 | |
| drop table t1, t2, t20, t21, t3;
 | |
| # Testcase from MDEV-32311
 | |
| SELECT (
 | |
| ( WITH x ( x ) AS
 | |
| (SELECT ( SELECT 'x' UNION SELECT 'x' ) FROM ( SELECT ( 'x' ) ) x)
 | |
| SELECT x FROM x
 | |
| WHERE x IN ( ( SELECT 'x' AND x GROUP BY x ) )
 | |
| )
 | |
| ) AS SUBQ;
 | |
| SUBQ
 | |
| x
 | |
| # MDEV-32390:
 | |
| CREATE TABLE t0 ( c43 DECIMAL ( 31 ) DEFAULT ( 45 ) ) ;
 | |
| INSERT INTO t0 VALUES ( 13 ) , ( 29 ) ;
 | |
| ALTER TABLE t0 ADD COLUMN c24 INT AFTER c43 ;
 | |
| INSERT INTO t0 VALUES ( DEFAULT , DEFAULT ) , ( DEFAULT , DEFAULT ) ;
 | |
| SELECT t1 . c22 AS c9 FROM ( SELECT ( SELECT + EXISTS ( SELECT -128 AS c29 ) <<
 | |
| LOCATE ( t0 . c43 , t0 . c24 <= t0 . c24 NOT BETWEEN 4642475734208631537 AND
 | |
| -108 , NULLIF ( 57 , -8 ) SOUNDS LIKE TRIM( TRAILING FROM 6107036197732405580 )
 | |
| ) - t0 . c43 AS c57 FROM t0 LIMIT 1 ) AS c22 FROM t0 ) AS t1 HAVING TRIM( CASE
 | |
| t1 . c22 WHEN -16 THEN RAND ( ) % HEX ( t1 . c22 ) - SUBSTRING_INDEX ( t1 . c22,
 | |
| ':A9SEZxtjN,fKN*zR' , 'V*vhJb}&c%Op,[T[S,j`F9NDsK;\'8 4;m"
 | |
| P,ce}1r"3ID1DN' ) >> NULLIF ( t1 . c22 , -95 ) ELSE -2 END IS TRUE
 | |
| FROM t1 . c22 >= EXISTS ( SELECT t2 . c57 AS c59 FROM ( SELECT CASE c24 WHEN
 | |
| -103 THEN 85 ELSE 22 END IS TRUE AS c57 FROM t0 ) AS t2 WHERE MOD ( 64 , 46 ) =
 | |
| CONVERT ( 73 , BINARY ) % RAND ( ) IS NOT NULL = -65 GROUP BY c57 , c22 , c22
 | |
| WINDOW w0 AS ( PARTITION BY t2 . c57 ) ) & PI ( ) ) ;
 | |
| c9
 | |
| DROP TABLE t0;
 | |
| # MDEV-32309
 | |
| SELECT
 | |
| ( WITH x ( x ) AS
 | |
| (
 | |
| WITH x ( x ) AS ( SELECT 1 ) SELECT ( SELECT x ) FROM x
 | |
| )
 | |
| SELECT x FROM x WHERE x IN ( SELECT NULL GROUP BY x )
 | |
| ) as col1 ;
 | |
| col1
 | |
| NULL
 | |
| # MDEV-32391
 | |
| CREATE TABLE t0 ( c15 INT , c33 INT ) engine=innodb;
 | |
| INSERT INTO t0 ( c15 ) WITH t1 AS ( SELECT SQRT ( 123 ) NOT
 | |
| REGEXP MOD ( 91 , -121 ) = ALL ( SELECT c15 AS c33 FROM t0 ) AS c49 FROM t0 )
 | |
| SELECT t1 . c49 IS UNKNOWN AS c59 FROM t1 CROSS JOIN t0 AS t2
 | |
| WHERE t1 . c49 = + EXISTS ( SELECT -5839312620871436105 AS c17 GROUP BY c49 )
 | |
| BETWEEN -109 AND CHAR_LENGTH ( 2694839150676403988 ) - - LOWER ( -13 ) ;
 | |
| DROP TABLE t0;
 | |
| # MDEV-28620
 | |
| CREATE TABLE t1 ( a int);
 | |
| INSERT INTO t1 VALUES (1),(2);
 | |
| SELECT EXISTS
 | |
| ( SELECT 1 FROM t1 GROUP BY 1 IN (SELECT a FROM t1)
 | |
| ORDER BY a + (SELECT 1 FROM t1 WHERE (1,2) NOT IN (SELECT 1,0))
 | |
| ) as SUBQ;
 | |
| ERROR 21000: Subquery returns more than 1 row
 | |
| DROP TABLE t1;
 | |
| # MDEV-30842 Item_subselect::get_cache_parameters and UBSAN member
 | |
| #  access within null pointer
 | |
| CREATE TABLE x (x INT) ENGINE=InnoDB;
 | |
| INSERT INTO x (x) VALUES (0);
 | |
| INSERT INTO x (x) VALUES (x IN (SELECT (SELECT x FROM (SELECT x FROM
 | |
| (SELECT 0 IN (SELECT x=0 FROM (SELECT x FROM (SELECT (SELECT (SELECT (SELECT
 | |
| (SELECT 0 AS x) FROM x AS x) IN (SELECT 0 AS x) AS x) FROM x AS x) IN
 | |
| (SELECT x WHERE x=0) AS x FROM x AS x) AS x) AS x GROUP BY x) AS x FROM x) AS x)
 | |
| AS x) IN (SELECT 0 AS x) AS x FROM x));
 | |
| ERROR HY000: Table 'x' is specified twice, both as a target for 'INSERT' and as a separate source for data
 | |
| DROP TABLE x;
 | |
| # MDEV-28622: Item_subselect eliminated flag set but Item still
 | |
| #   evaluated/used.
 | |
| CREATE TABLE t1 (  a int) ;
 | |
| CREATE VIEW v1 (i) AS SELECT EXISTS(SELECT 1) FROM t1;
 | |
| SELECT 1 FROM v1 WHERE i NOT IN (SELECT i = 0 FROM v1 WHERE i = -1 GROUP BY i);
 | |
| 1
 | |
| DROP TABLE t1;
 | |
| DROP VIEW v1;
 | |
| CREATE TABLE t(c1 INT);
 | |
| SELECT 0
 | |
| WHERE 0 IN
 | |
| (
 | |
| SELECT 0 FROM
 | |
| (
 | |
| SELECT 0 IN
 | |
| (
 | |
| SELECT
 | |
| (
 | |
| SELECT c1 FROM t
 | |
| )
 | |
| ) AS c
 | |
| FROM t
 | |
| ) AS dt
 | |
| WHERE c GROUP BY c
 | |
| );
 | |
| 0
 | |
| DROP TABLE t;
 | |
| create table t1 (a int, b int, c int);
 | |
| insert into t1 select seq, seq, seq from seq_1_to_10;
 | |
| create table t2 as select * from t1;
 | |
| create table t20 as select * from t1;
 | |
| create table t3 as select * from t1;
 | |
| create view v2 as
 | |
| select
 | |
| a, b, (select max(c) from t20 where t20.a<=t2.a) as SUBQ1
 | |
| from t2;
 | |
| explain
 | |
| select
 | |
| a, a in (select a from v2 where a>3 group by v2.SUBQ1)
 | |
| from t1;
 | |
| id	select_type	table	type	possible_keys	key	key_len	ref	rows	Extra
 | |
| 1	PRIMARY	t1	ALL	NULL	NULL	NULL	NULL	10	
 | |
| 2	MATERIALIZED	t2	ALL	NULL	NULL	NULL	NULL	10	Using where; Using temporary
 | |
| 4	DEPENDENT SUBQUERY	t20	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| prepare s from '
 | |
| explain
 | |
| select
 | |
|   a, a in (select a from v2 where a>3 group by v2.SUBQ1)
 | |
| from t1';
 | |
| execute s;
 | |
| id	select_type	table	type	possible_keys	key	key_len	ref	rows	Extra
 | |
| 1	PRIMARY	t1	ALL	NULL	NULL	NULL	NULL	10	
 | |
| 2	MATERIALIZED	t2	ALL	NULL	NULL	NULL	NULL	10	Using where; Using temporary
 | |
| 4	DEPENDENT SUBQUERY	t20	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| execute s;
 | |
| id	select_type	table	type	possible_keys	key	key_len	ref	rows	Extra
 | |
| 1	PRIMARY	t1	ALL	NULL	NULL	NULL	NULL	10	
 | |
| 2	MATERIALIZED	t2	ALL	NULL	NULL	NULL	NULL	10	Using where; Using temporary
 | |
| 4	DEPENDENT SUBQUERY	t20	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| execute s;
 | |
| id	select_type	table	type	possible_keys	key	key_len	ref	rows	Extra
 | |
| 1	PRIMARY	t1	ALL	NULL	NULL	NULL	NULL	10	
 | |
| 2	MATERIALIZED	t2	ALL	NULL	NULL	NULL	NULL	10	Using where; Using temporary
 | |
| 4	DEPENDENT SUBQUERY	t20	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| prepare s from '
 | |
| explain
 | |
| select
 | |
|   a, a in (select a from v2 where a>3 and SUBQ1+1 group by v2.SUBQ1)
 | |
| from t1';
 | |
| execute s;
 | |
| id	select_type	table	type	possible_keys	key	key_len	ref	rows	Extra
 | |
| 1	PRIMARY	t1	ALL	NULL	NULL	NULL	NULL	10	
 | |
| 2	MATERIALIZED	t2	ALL	NULL	NULL	NULL	NULL	10	Using where; Using temporary
 | |
| 4	DEPENDENT SUBQUERY	t20	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| execute s;
 | |
| id	select_type	table	type	possible_keys	key	key_len	ref	rows	Extra
 | |
| 1	PRIMARY	t1	ALL	NULL	NULL	NULL	NULL	10	
 | |
| 2	MATERIALIZED	t2	ALL	NULL	NULL	NULL	NULL	10	Using where; Using temporary
 | |
| 4	DEPENDENT SUBQUERY	t20	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| execute s;
 | |
| id	select_type	table	type	possible_keys	key	key_len	ref	rows	Extra
 | |
| 1	PRIMARY	t1	ALL	NULL	NULL	NULL	NULL	10	
 | |
| 2	MATERIALIZED	t2	ALL	NULL	NULL	NULL	NULL	10	Using where; Using temporary
 | |
| 4	DEPENDENT SUBQUERY	t20	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| create procedure p1()
 | |
| begin
 | |
| explain
 | |
| select
 | |
| a, a in (select a from v2 where a>3 group by v2.SUBQ1)
 | |
| from t1;
 | |
| end//
 | |
| create procedure p2()
 | |
| begin
 | |
| explain
 | |
| select
 | |
| a, a in (select a from v2 where a>3 and SUBQ1+1 group by v2.SUBQ1)
 | |
| from t1;
 | |
| end//
 | |
| call p1();
 | |
| id	select_type	table	type	possible_keys	key	key_len	ref	rows	Extra
 | |
| 1	PRIMARY	t1	ALL	NULL	NULL	NULL	NULL	10	
 | |
| 2	MATERIALIZED	t2	ALL	NULL	NULL	NULL	NULL	10	Using where; Using temporary
 | |
| 4	DEPENDENT SUBQUERY	t20	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| call p1();
 | |
| id	select_type	table	type	possible_keys	key	key_len	ref	rows	Extra
 | |
| 1	PRIMARY	t1	ALL	NULL	NULL	NULL	NULL	10	
 | |
| 2	MATERIALIZED	t2	ALL	NULL	NULL	NULL	NULL	10	Using where; Using temporary
 | |
| 4	DEPENDENT SUBQUERY	t20	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| call p2();
 | |
| id	select_type	table	type	possible_keys	key	key_len	ref	rows	Extra
 | |
| 1	PRIMARY	t1	ALL	NULL	NULL	NULL	NULL	10	
 | |
| 2	MATERIALIZED	t2	ALL	NULL	NULL	NULL	NULL	10	Using where; Using temporary
 | |
| 4	DEPENDENT SUBQUERY	t20	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| call p2();
 | |
| id	select_type	table	type	possible_keys	key	key_len	ref	rows	Extra
 | |
| 1	PRIMARY	t1	ALL	NULL	NULL	NULL	NULL	10	
 | |
| 2	MATERIALIZED	t2	ALL	NULL	NULL	NULL	NULL	10	Using where; Using temporary
 | |
| 4	DEPENDENT SUBQUERY	t20	ALL	NULL	NULL	NULL	NULL	10	Using where
 | |
| drop procedure p1;
 | |
| drop procedure p2;
 | |
| drop view v2;
 | |
| drop table t1,t2,t3,t20;
 | |
| # end of 10.4 tests
 |