From 90940053663d7f5155528b0c9cf814775d91f290 Mon Sep 17 00:00:00 2001 From: yanweiqi <592838129@qq.com> Date: Fri, 22 Jul 2022 14:05:09 +0800 Subject: [PATCH 1/2] cherry pick #36208 to release-5.4 Signed-off-by: ti-srebot --- executor/aggregate.go | 14 ++++++ executor/aggregate_test.go | 90 ++++++++++++++++++++++++++++++++++++++ 2 files changed, 104 insertions(+) diff --git a/executor/aggregate.go b/executor/aggregate.go index 8a6b83d089e58..5bed2cbad1ab8 100644 --- a/executor/aggregate.go +++ b/executor/aggregate.go @@ -572,11 +572,24 @@ func getGroupKey(ctx sessionctx.Context, input *chunk.Chunk, groupKey [][]byte, for _, item := range groupByItems { tp := item.GetType() + buf, err := expression.GetColumn(tp.EvalType(), numRows) if err != nil { return nil, err } + // In strict sql mode like ‘STRICT_TRANS_TABLES’,can not insert an invalid enum value like 0. + // While in sql mode like '', can insert an invalid enum value like 0, + // then the enum value 0 will have the enum name '', which maybe conflict with user defined enum ''. + // Ref to issue #26885. + // This check is used to handle invalid enum name same with user defined enum name. + // Use enum value as groupKey instead of enum name. + if item.GetType().GetType() == mysql.TypeEnum { + newTp := *tp + newTp.AddFlag(mysql.EnumSetAsIntFlag) + tp = &newTp + } + if err := expression.EvalExpr(ctx, item, tp.EvalType(), input, buf); err != nil { expression.PutColumn(buf) return nil, err @@ -587,6 +600,7 @@ func getGroupKey(ctx sessionctx.Context, input *chunk.Chunk, groupKey [][]byte, newTp.Flen = 0 tp = &newTp } + groupKey, err = codec.HashGroupKey(ctx.GetSessionVars().StmtCtx, input.NumRows(), buf, groupKey, tp) if err != nil { expression.PutColumn(buf) diff --git a/executor/aggregate_test.go b/executor/aggregate_test.go index 976775ce5dc4c..b3a69d61f11f9 100644 --- a/executor/aggregate_test.go +++ b/executor/aggregate_test.go @@ -1597,3 +1597,93 @@ func TestRandomPanicAggConsume(t *testing.T) { require.EqualError(t, err, "failpoint panic: ERROR 1105 (HY000): Out Of Memory Quota![conn_id=1]") } } +<<<<<<< HEAD +======= + +func TestIssue35295(t *testing.T) { + store, clean := testkit.CreateMockStore(t) + defer clean() + tk := testkit.NewTestKit(t, store) + tk.MustExec("use test") + tk.MustExec("drop table if exists t100") + // This bug only happens on partition prune mode = 'static' + tk.MustExec("set @@tidb_partition_prune_mode = 'static'") + tk.MustExec(`CREATE TABLE t100 ( +ID bigint(20) unsigned NOT NULL AUTO_INCREMENT, +col1 int(10) NOT NULL DEFAULT '0' COMMENT 'test', +money bigint(20) NOT NULL COMMENT 'test', +logtime datetime NOT NULL COMMENT '记录时间', +PRIMARY KEY (ID,logtime) +) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin AUTO_INCREMENT=1 COMMENT='test' +PARTITION BY RANGE COLUMNS(logtime) ( +PARTITION p20220608 VALUES LESS THAN ("20220609"), +PARTITION p20220609 VALUES LESS THAN ("20220610"), +PARTITION p20220610 VALUES LESS THAN ("20220611"), +PARTITION p20220611 VALUES LESS THAN ("20220612"), +PARTITION p20220612 VALUES LESS THAN ("20220613"), +PARTITION p20220613 VALUES LESS THAN ("20220614"), +PARTITION p20220614 VALUES LESS THAN ("20220615"), +PARTITION p20220615 VALUES LESS THAN ("20220616"), +PARTITION p20220616 VALUES LESS THAN ("20220617"), +PARTITION p20220617 VALUES LESS THAN ("20220618"), +PARTITION p20220618 VALUES LESS THAN ("20220619"), +PARTITION p20220619 VALUES LESS THAN ("20220620"), +PARTITION p20220620 VALUES LESS THAN ("20220621"), +PARTITION p20220621 VALUES LESS THAN ("20220622"), +PARTITION p20220622 VALUES LESS THAN ("20220623"), +PARTITION p20220623 VALUES LESS THAN ("20220624"), +PARTITION p20220624 VALUES LESS THAN ("20220625") + );`) + tk.MustExec("insert into t100(col1,money,logtime) values (100,10,'2022-06-09 00:00:00');") + tk.MustExec("insert into t100(col1,money,logtime) values (100,10,'2022-06-10 00:00:00');") + tk.MustQuery("SELECT /*+STREAM_AGG()*/ col1,sum(money) FROM t100 WHERE logtime>='2022-06-09 00:00:00' AND col1=100 ;").Check(testkit.Rows("100 20")) + tk.MustQuery("SELECT /*+HASH_AGG()*/ col1,sum(money) FROM t100 WHERE logtime>='2022-06-09 00:00:00' AND col1=100 ;").Check(testkit.Rows("100 20")) +} + +// https://github.com/pingcap/tidb/issues/27751 +func TestIssue27751(t *testing.T) { + store, clean := testkit.CreateMockStore(t) + defer clean() + tk := testkit.NewTestKit(t, store) + tk.MustExec("use test") + tk.MustExec("drop table if exists t") + tk.MustExec("create table test.t(nname char(20));") + tk.MustExec("insert into test.t values ('2'),(null),('11'),('2'),(null),('2'),(null),('11'),('33');") + tk.MustExec("set @@group_concat_max_len=0;") + tk.MustQuery("select group_concat(nname order by 1 separator '#' ) from t;").Check(testkit.Rows("11#1")) + tk.MustQuery("select group_concat(nname order by 1 desc separator '#' ) from t;").Check(testkit.Rows("33#2")) +} + +func TestIssue26885(t *testing.T) { + store, clean := testkit.CreateMockStore(t) + defer clean() + tk := testkit.NewTestKit(t, store) + tk.MustExec(`use test`) + tk.MustExec(`SET sql_mode = 'NO_ENGINE_SUBSTITUTION';`) + tk.MustExec(`DROP TABLE IF EXISTS t1;`) + + tk.MustExec("CREATE TABLE t1 (c1 ENUM('a', '', 'b'));") + tk.MustExec("INSERT INTO t1 (c1) VALUES ('b');") + tk.MustExec("INSERT INTO t1 (c1) VALUES ('');") + tk.MustExec("INSERT INTO t1 (c1) VALUES ('a');") + tk.MustExec("INSERT INTO t1 (c1) VALUES ('');") + tk.MustExec("INSERT INTO t1 (c1) VALUES (0);") + tk.MustQuery("select * from t1").Check(testkit.Rows("b", "", "a", "", "")) + tk.MustQuery("select c1 + 0 from t1").Check(testkit.Rows("3", "2", "1", "2", "0")) + tk.MustQuery("SELECT c1 + 0, COUNT(c1) FROM t1 GROUP BY c1 order by c1;").Check(testkit.Rows("0 1", "1 1", "2 2", "3 1")) + + tk.MustExec("alter table t1 add index idx(c1); ") + tk.MustQuery("select c1 + 0 from t1").Check(testkit.Rows("3", "2", "1", "2", "0")) + tk.MustQuery("SELECT c1 + 0, COUNT(c1) FROM t1 GROUP BY c1 order by c1;").Check(testkit.Rows("0 1", "1 1", "2 2", "3 1")) + + tk.MustExec(`DROP TABLE IF EXISTS t1;`) + tk.MustExec("CREATE TABLE t1 (c1 ENUM('a', 'b', 'c'));") + tk.MustExec("INSERT INTO t1 (c1) VALUES ('b');") + tk.MustExec("INSERT INTO t1 (c1) VALUES ('a');") + tk.MustExec("INSERT INTO t1 (c1) VALUES ('b');") + tk.MustExec("INSERT INTO t1 (c1) VALUES ('c');") + tk.MustExec("INSERT INTO t1 (c1) VALUES (0);") + tk.MustQuery("select * from t1").Check(testkit.Rows("b", "a", "b", "c", "")) + tk.MustQuery("SELECT c1 + 0, COUNT(c1) FROM t1 GROUP BY c1 order by c1;").Check(testkit.Rows("0 1", "1 1", "2 2", "3 1")) +} +>>>>>>> 065563a8e... executor: fix aggregating enum zero value gets different results from mysql (#36208) From 3867d0eba46c0244e853087b99b74ad2c32a30dd Mon Sep 17 00:00:00 2001 From: ywqzzy <592838129@qq.com> Date: Fri, 22 Jul 2022 15:07:41 +0800 Subject: [PATCH 2/2] resolve --- executor/aggregate.go | 4 +-- executor/aggregate_test.go | 57 -------------------------------------- 2 files changed, 2 insertions(+), 59 deletions(-) diff --git a/executor/aggregate.go b/executor/aggregate.go index 5bed2cbad1ab8..85c7f64862b8e 100644 --- a/executor/aggregate.go +++ b/executor/aggregate.go @@ -584,9 +584,9 @@ func getGroupKey(ctx sessionctx.Context, input *chunk.Chunk, groupKey [][]byte, // Ref to issue #26885. // This check is used to handle invalid enum name same with user defined enum name. // Use enum value as groupKey instead of enum name. - if item.GetType().GetType() == mysql.TypeEnum { + if item.GetType().Tp == mysql.TypeEnum { newTp := *tp - newTp.AddFlag(mysql.EnumSetAsIntFlag) + newTp.Flag |= mysql.EnumSetAsIntFlag tp = &newTp } diff --git a/executor/aggregate_test.go b/executor/aggregate_test.go index b3a69d61f11f9..a4e5c9cee0ac8 100644 --- a/executor/aggregate_test.go +++ b/executor/aggregate_test.go @@ -1597,62 +1597,6 @@ func TestRandomPanicAggConsume(t *testing.T) { require.EqualError(t, err, "failpoint panic: ERROR 1105 (HY000): Out Of Memory Quota![conn_id=1]") } } -<<<<<<< HEAD -======= - -func TestIssue35295(t *testing.T) { - store, clean := testkit.CreateMockStore(t) - defer clean() - tk := testkit.NewTestKit(t, store) - tk.MustExec("use test") - tk.MustExec("drop table if exists t100") - // This bug only happens on partition prune mode = 'static' - tk.MustExec("set @@tidb_partition_prune_mode = 'static'") - tk.MustExec(`CREATE TABLE t100 ( -ID bigint(20) unsigned NOT NULL AUTO_INCREMENT, -col1 int(10) NOT NULL DEFAULT '0' COMMENT 'test', -money bigint(20) NOT NULL COMMENT 'test', -logtime datetime NOT NULL COMMENT '记录时间', -PRIMARY KEY (ID,logtime) -) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_bin AUTO_INCREMENT=1 COMMENT='test' -PARTITION BY RANGE COLUMNS(logtime) ( -PARTITION p20220608 VALUES LESS THAN ("20220609"), -PARTITION p20220609 VALUES LESS THAN ("20220610"), -PARTITION p20220610 VALUES LESS THAN ("20220611"), -PARTITION p20220611 VALUES LESS THAN ("20220612"), -PARTITION p20220612 VALUES LESS THAN ("20220613"), -PARTITION p20220613 VALUES LESS THAN ("20220614"), -PARTITION p20220614 VALUES LESS THAN ("20220615"), -PARTITION p20220615 VALUES LESS THAN ("20220616"), -PARTITION p20220616 VALUES LESS THAN ("20220617"), -PARTITION p20220617 VALUES LESS THAN ("20220618"), -PARTITION p20220618 VALUES LESS THAN ("20220619"), -PARTITION p20220619 VALUES LESS THAN ("20220620"), -PARTITION p20220620 VALUES LESS THAN ("20220621"), -PARTITION p20220621 VALUES LESS THAN ("20220622"), -PARTITION p20220622 VALUES LESS THAN ("20220623"), -PARTITION p20220623 VALUES LESS THAN ("20220624"), -PARTITION p20220624 VALUES LESS THAN ("20220625") - );`) - tk.MustExec("insert into t100(col1,money,logtime) values (100,10,'2022-06-09 00:00:00');") - tk.MustExec("insert into t100(col1,money,logtime) values (100,10,'2022-06-10 00:00:00');") - tk.MustQuery("SELECT /*+STREAM_AGG()*/ col1,sum(money) FROM t100 WHERE logtime>='2022-06-09 00:00:00' AND col1=100 ;").Check(testkit.Rows("100 20")) - tk.MustQuery("SELECT /*+HASH_AGG()*/ col1,sum(money) FROM t100 WHERE logtime>='2022-06-09 00:00:00' AND col1=100 ;").Check(testkit.Rows("100 20")) -} - -// https://github.com/pingcap/tidb/issues/27751 -func TestIssue27751(t *testing.T) { - store, clean := testkit.CreateMockStore(t) - defer clean() - tk := testkit.NewTestKit(t, store) - tk.MustExec("use test") - tk.MustExec("drop table if exists t") - tk.MustExec("create table test.t(nname char(20));") - tk.MustExec("insert into test.t values ('2'),(null),('11'),('2'),(null),('2'),(null),('11'),('33');") - tk.MustExec("set @@group_concat_max_len=0;") - tk.MustQuery("select group_concat(nname order by 1 separator '#' ) from t;").Check(testkit.Rows("11#1")) - tk.MustQuery("select group_concat(nname order by 1 desc separator '#' ) from t;").Check(testkit.Rows("33#2")) -} func TestIssue26885(t *testing.T) { store, clean := testkit.CreateMockStore(t) @@ -1686,4 +1630,3 @@ func TestIssue26885(t *testing.T) { tk.MustQuery("select * from t1").Check(testkit.Rows("b", "a", "b", "c", "")) tk.MustQuery("SELECT c1 + 0, COUNT(c1) FROM t1 GROUP BY c1 order by c1;").Check(testkit.Rows("0 1", "1 1", "2 2", "3 1")) } ->>>>>>> 065563a8e... executor: fix aggregating enum zero value gets different results from mysql (#36208)