// Copyright 2018 PingCAP, Inc. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // See the License for the specific language governing permissions and // limitations under the License. package core_test import ( . "github.com/pingcap/check" "github.com/pingcap/parser/terror" "github.com/pingcap/tidb/planner/core" "github.com/pingcap/tidb/util/testkit" "github.com/pingcap/tidb/util/testleak" "github.com/pingcap/tidb/util/testutil" ) var _ = Suite(&testExpressionRewriterSuite{}) type testExpressionRewriterSuite struct { } func (s *testExpressionRewriterSuite) TestIfNullEliminateColName(c *C) { defer testleak.AfterTest(c)() store, dom, err := newStoreWithBootstrap() c.Assert(err, IsNil) tk := testkit.NewTestKit(c, store) defer func() { dom.Close() store.Close() }() tk.MustExec("use test") tk.MustExec("drop table if exists t") tk.MustExec("create table t(a int not null, b int not null)") rs, err := tk.Exec("select ifnull(a,b) from t") c.Assert(err, IsNil) fields := rs.Fields() c.Assert(fields[0].Column.Name.L, Equals, "ifnull(a,b)") } func (s *testExpressionRewriterSuite) TestBinaryOpFunction(c *C) { defer testleak.AfterTest(c)() store, dom, err := newStoreWithBootstrap() c.Assert(err, IsNil) tk := testkit.NewTestKit(c, store) defer func() { dom.Close() store.Close() }() tk.MustExec("use test") tk.MustExec("drop table if exists t") tk.MustExec("CREATE TABLE t(a int, b int, c int);") tk.MustExec("INSERT INTO t VALUES (1, 2, 3), (NULL, 2, 3 ), (1, NULL, 3),(1, 2, NULL),(NULL, 2, 3+1), (1, NULL, 3+1), (1, 2+1, NULL),(NULL, 2, 3-1), (1, NULL, 3-1), (1, 2-1, NULL)") tk.MustQuery("SELECT * FROM t WHERE (a,b,c) <= (1,2,3) order by b").Check(testkit.Rows("1 1 ", "1 2 3")) tk.MustQuery("SELECT * FROM t WHERE (a,b,c) > (1,2,3) order by b").Check(testkit.Rows("1 3 ")) } func (s *testExpressionRewriterSuite) TestDefaultFunction(c *C) { defer testleak.AfterTest(c)() store, dom, err := newStoreWithBootstrap() c.Assert(err, IsNil) tk := testkit.NewTestKit(c, store) defer func() { dom.Close() store.Close() }() tk.MustExec("use test") tk.MustExec("drop table if exists t1") tk.MustExec(`create table t1( a varchar(10) default 'def', b varchar(10), c int default '10', d double default '3.14', e datetime default '20180101', f datetime default current_timestamp);`) tk.MustExec("insert into t1(a, b, c, d) values ('1', '1', 1, 1)") tk.MustQuery(`select default(a) as defa, default(b) as defb, default(c) as defc, default(d) as defd, default(e) as defe, default(f) as deff from t1`).Check(testutil.RowsWithSep("|", "def||10|3.14|2018-01-01 00:00:00|")) err = tk.ExecToErr("select default(x) from t1") c.Assert(err.Error(), Equals, "[planner:1054]Unknown column 'x' in 'field list'") tk.MustQuery("select default(a0) from (select a as a0 from t1) as t0").Check(testkit.Rows("def")) err = tk.ExecToErr("select default(a0) from (select a+1 as a0 from t1) as t0") c.Assert(err.Error(), Equals, "[table:1364]Field 'a0' doesn't have a default value") tk.MustExec("create table t2(a varchar(10), b varchar(10))") tk.MustExec("insert into t2 values ('1', '1')") err = tk.ExecToErr("select default(a) from t1, t2") c.Assert(err.Error(), Equals, "[planner:1052]Column 'a' in field list is ambiguous") tk.MustQuery("select default(t1.a) from t1, t2").Check(testkit.Rows("def")) tk.MustExec(`create table t3( a datetime default current_timestamp, b timestamp default current_timestamp, c timestamp(6) default current_timestamp(6), d varchar(20) default 'current_timestamp')`) tk.MustExec("insert into t3 values ()") tk.MustQuery(`select default(a) as defa, default(b) as defb, default(c) as defc, default(d) as defd from t3`).Check(testutil.RowsWithSep("|", "|0000-00-00 00:00:00|0000-00-00 00:00:00.000000|current_timestamp")) tk.MustExec(`create table t4(a int default 1, b varchar(5))`) tk.MustExec(`insert into t4 values (0, 'B'), (1, 'B'), (2, 'B')`) tk.MustExec(`create table t5(d int default 0, e varchar(5))`) tk.MustExec(`insert into t5 values (5, 'B')`) tk.MustQuery(`select a from t4 where a > (select default(d) from t5 where t4.b = t5.e)`).Check(testkit.Rows("1", "2")) tk.MustQuery(`select a from t4 where a > (select default(a) from t5 where t4.b = t5.e)`).Check(testkit.Rows("2")) tk.MustExec("prepare stmt from 'select default(a) from t1';") tk.MustQuery("execute stmt").Check(testkit.Rows("def")) tk.MustExec("alter table t1 modify a varchar(10) default 'DEF'") tk.MustQuery("execute stmt").Check(testkit.Rows("DEF")) tk.MustExec("update t1 set c = c + default(c)") tk.MustQuery("select c from t1").Check(testkit.Rows("11")) } func (s *testExpressionRewriterSuite) TestCompareSubquery(c *C) { defer testleak.AfterTest(c)() store, dom, err := newStoreWithBootstrap() c.Assert(err, IsNil) tk := testkit.NewTestKit(c, store) defer func() { dom.Close() store.Close() }() tk.MustExec("use test") tk.MustExec("drop table if exists t") tk.MustExec("drop table if exists s") tk.MustExec("create table t(a int, b int)") tk.MustExec("create table s(a int, b int)") tk.MustExec("insert into t values(1, null), (2, null)") // Test empty checker. tk.MustQuery("select a != any (select a from s) from t").Check(testkit.Rows( "0", "0", )) tk.MustQuery("select b != any (select a from s) from t").Check(testkit.Rows( "0", "0", )) tk.MustQuery("select a = all (select a from s) from t").Check(testkit.Rows( "1", "1", )) tk.MustQuery("select b = all (select a from s) from t").Check(testkit.Rows( "1", "1", )) tk.MustQuery("select * from t where a != any (select a from s)").Check(testkit.Rows()) tk.MustQuery("select * from t where b != any (select a from s)").Check(testkit.Rows()) tk.MustQuery("select * from t where a = all (select a from s)").Check(testkit.Rows( "1 ", "2 ", )) tk.MustQuery("select * from t where b = all (select a from s)").Check(testkit.Rows( "1 ", "2 ", )) // Test outer null checker. tk.MustQuery("select b != any (select a from t t2) from t t1").Check(testkit.Rows( "", "", )) tk.MustQuery("select b = all (select a from t t2) from t t1").Check(testkit.Rows( "", "", )) tk.MustQuery("select * from t t1 where b != any (select a from t t2)").Check(testkit.Rows()) tk.MustQuery("select * from t t1 where b = all (select a from t t2)").Check(testkit.Rows()) tk.MustExec("delete from t where a = 2") tk.MustQuery("select b != any (select a from t t2) from t t1").Check(testkit.Rows( "", )) tk.MustQuery("select b = all (select a from t t2) from t t1").Check(testkit.Rows( "", )) tk.MustQuery("select * from t t1 where b != any (select a from t t2)").Check(testkit.Rows()) tk.MustQuery("select * from t t1 where b = all (select a from t t2)").Check(testkit.Rows()) // Test inner null checker. tk.MustExec("insert into t values(null, 1)") tk.MustQuery("select b != any (select a from t t2) from t t1").Check(testkit.Rows( "", "", )) tk.MustQuery("select b = all (select a from t t2) from t t1").Check(testkit.Rows( "", "", )) tk.MustQuery("select * from t t1 where b != any (select a from t t2)").Check(testkit.Rows()) tk.MustQuery("select * from t t1 where b = all (select a from t t2)").Check(testkit.Rows()) tk.MustExec("delete from t where b = 1") tk.MustExec("insert into t values(null, 2)") tk.MustQuery("select b != any (select a from t t2) from t t1").Check(testkit.Rows( "", "1", )) tk.MustQuery("select b = all (select a from t t2) from t t1").Check(testkit.Rows( "", "0", )) tk.MustQuery("select * from t t1 where b != any (select a from t t2)").Check(testkit.Rows( " 2", )) tk.MustQuery("select * from t t1 where b = all (select a from t t2)").Check(testkit.Rows()) } func (s *testExpressionRewriterSuite) TestCheckFullGroupBy(c *C) { defer testleak.AfterTest(c)() store, dom, err := newStoreWithBootstrap() c.Assert(err, IsNil) tk := testkit.NewTestKit(c, store) defer func() { dom.Close() store.Close() }() tk.MustExec("use test") tk.MustExec("drop table if exists t") tk.MustExec("create table t(a int, b int)") tk.MustQuery("select t1.a, (select max(t2.b) from t t2) from t t1").Check(testkit.Rows()) err = tk.ExecToErr("select t1.a, (select t2.a, max(t2.b) from t t2) from t t1") c.Assert(terror.ErrorEqual(err, core.ErrMixOfGroupFuncAndFields), IsTrue, Commentf("err %v", err)) }