// Copyright 2016 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, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. package executor_test import ( "context" "crypto/tls" "fmt" "math" "testing" "time" "github.com/pingcap/tidb/executor" "github.com/pingcap/tidb/infoschema" "github.com/pingcap/tidb/metrics" "github.com/pingcap/tidb/parser/mysql" plannercore "github.com/pingcap/tidb/planner/core" "github.com/pingcap/tidb/session" "github.com/pingcap/tidb/session/txninfo" "github.com/pingcap/tidb/testkit" "github.com/pingcap/tidb/types" "github.com/pingcap/tidb/util" "github.com/pingcap/tidb/util/kvcache" dto "github.com/prometheus/client_model/go" "github.com/stretchr/testify/require" ) func TestPrepared(t *testing.T) { store, clean := testkit.CreateMockStore(t) defer clean() orgEnable := plannercore.PreparedPlanCacheEnabled() defer func() { plannercore.SetPreparedPlanCache(orgEnable) }() flags := []bool{false, true} ctx := context.Background() for _, flag := range flags { plannercore.SetPreparedPlanCache(flag) tk := testkit.NewTestKit(t, store) se, err := session.CreateSession4TestWithOpt(store, &session.Opt{ PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64), }) require.NoError(t, err) tk.SetSession(se) tk.MustExec("use test") tk.MustExec("drop table if exists prepare_test") tk.MustExec("create table prepare_test (id int PRIMARY KEY AUTO_INCREMENT, c1 int, c2 int, c3 int default 1)") tk.MustExec("insert prepare_test (c1) values (1),(2),(NULL)") tk.MustExec(`prepare stmt_test_1 from 'select id from prepare_test where id > ?';`) tk.MustExec(`set @a = 1;`) tk.MustExec(`execute stmt_test_1 using @a;`) tk.MustExec(`prepare stmt_test_2 from 'select 1'`) // Prepare multiple statement is not allowed. _, err = tk.Exec(`prepare stmt_test_3 from 'select id from prepare_test where id > ?;select id from prepare_test where id > ?;'`) require.True(t, executor.ErrPrepareMulti.Equal(err)) // The variable count does not match. tk.MustExec(`prepare stmt_test_4 from 'select id from prepare_test where id > ? and id < ?';`) tk.MustExec(`set @a = 1;`) _, err = tk.Exec(`execute stmt_test_4 using @a;`) require.True(t, plannercore.ErrWrongParamCount.Equal(err)) // Prepare and deallocate prepared statement immediately. tk.MustExec(`prepare stmt_test_5 from 'select id from prepare_test where id > ?';`) tk.MustExec(`deallocate prepare stmt_test_5;`) // Statement not found. _, err = tk.Exec("deallocate prepare stmt_test_5") require.True(t, plannercore.ErrStmtNotFound.Equal(err)) // incorrect SQLs in prepare. issue #3738, SQL in prepare stmt is parsed in DoPrepare. _, err = tk.Exec(`prepare p from "delete from t where a = 7 or 1=1/*' and b = 'p'";`) require.EqualError(t, err, `[parser:1064]You have an error in your SQL syntax; check the manual that corresponds to your TiDB version for the right syntax to use near '/*' and b = 'p'' at line 1`) // The `stmt_test5` should not be found. _, err = tk.Exec(`set @a = 1; execute stmt_test_5 using @a;`) require.True(t, plannercore.ErrStmtNotFound.Equal(err)) // Use parameter marker with argument will run prepared statement. result := tk.MustQuery("select distinct c1, c2 from prepare_test where c1 = ?", 1) result.Check(testkit.Rows("1 ")) // Call Session PrepareStmt directly to get stmtID. query := "select c1, c2 from prepare_test where c1 = ?" stmtID, _, _, err := tk.Session().PrepareStmt(query) require.NoError(t, err) rs, err := tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1)}) require.NoError(t, err) tk.ResultSetToResult(rs, fmt.Sprintf("%v", rs)).Check(testkit.Rows("1 ")) tk.MustExec("delete from prepare_test") query = "select c1 from prepare_test where c1 = (select c1 from prepare_test where c1 = ?)" stmtID, _, _, err = tk.Session().PrepareStmt(query) require.NoError(t, err) tk1 := testkit.NewTestKit(t, store) se, err = session.CreateSession4TestWithOpt(store, &session.Opt{ PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64), }) require.NoError(t, err) tk1.SetSession(se) tk1.MustExec("use test") tk1.MustExec("insert prepare_test (c1) values (3)") rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(3)}) require.NoError(t, err) tk.ResultSetToResult(rs, fmt.Sprintf("%v", rs)).Check(testkit.Rows("3")) tk.MustExec("delete from prepare_test") query = "select c1 from prepare_test where c1 = (select c1 from prepare_test where c1 = ?)" stmtID, _, _, err = tk.Session().PrepareStmt(query) require.NoError(t, err) rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(3)}) require.NoError(t, err) require.NoError(t, rs.Close()) tk1.MustExec("insert prepare_test (c1) values (3)") rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(3)}) require.NoError(t, err) tk.ResultSetToResult(rs, fmt.Sprintf("%v", rs)).Check(testkit.Rows("3")) tk.MustExec("delete from prepare_test") query = "select c1 from prepare_test where c1 in (select c1 from prepare_test where c1 = ?)" stmtID, _, _, err = tk.Session().PrepareStmt(query) require.NoError(t, err) rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(3)}) require.NoError(t, err) require.NoError(t, rs.Close()) tk1.MustExec("insert prepare_test (c1) values (3)") rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(3)}) require.NoError(t, err) tk.ResultSetToResult(rs, fmt.Sprintf("%v", rs)).Check(testkit.Rows("3")) tk.MustExec("begin") tk.MustExec("insert prepare_test (c1) values (4)") query = "select c1, c2 from prepare_test where c1 = ?" stmtID, _, _, err = tk.Session().PrepareStmt(query) require.NoError(t, err) tk.MustExec("rollback") rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(4)}) require.NoError(t, err) tk.ResultSetToResult(rs, fmt.Sprintf("%v", rs)).Check(testkit.Rows()) // Check that ast.Statement created by executor.CompileExecutePreparedStmt has query text. stmt, _, _, err := executor.CompileExecutePreparedStmt(context.TODO(), tk.Session(), stmtID, tk.Session().GetInfoSchema().(infoschema.InfoSchema), 0, []types.Datum{types.NewDatum(1)}) require.NoError(t, err) require.Equal(t, query, stmt.OriginText()) // Check that rebuild plan works. err = tk.Session().PrepareTxnCtx(ctx) require.NoError(t, err) _, err = stmt.RebuildPlan(ctx) require.NoError(t, err) rs, err = stmt.Exec(ctx) require.NoError(t, err) req := rs.NewChunk(nil) err = rs.Next(ctx, req) require.NoError(t, err) require.NoError(t, rs.Close()) // Make schema change. tk.MustExec("drop table if exists prepare2") _, err = tk.Exec("create table prepare2 (a int)") require.NoError(t, err) // Should success as the changed schema do not affect the prepared statement. rs, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1)}) require.NoError(t, err) if rs != nil { require.NoError(t, rs.Close()) } // Drop a column so the prepared statement become invalid. query = "select c1, c2 from prepare_test where c1 = ?" stmtID, _, _, err = tk.Session().PrepareStmt(query) require.NoError(t, err) tk.MustExec("alter table prepare_test drop column c2") _, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1)}) require.True(t, plannercore.ErrUnknownColumn.Equal(err)) tk.MustExec("drop table prepare_test") _, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1)}) require.True(t, plannercore.ErrSchemaChanged.Equal(err)) // issue 3381 tk.MustExec("drop table if exists prepare3") tk.MustExec("create table prepare3 (a decimal(1))") tk.MustExec("prepare stmt from 'insert into prepare3 value(123)'") _, err = tk.Exec("execute stmt") require.Error(t, err) _, _, fields, err := tk.Session().PrepareStmt("select a from prepare3") require.NoError(t, err) require.Equal(t, "test", fields[0].DBName.L) require.Equal(t, "prepare3", fields[0].TableAsName.L) require.Equal(t, "a", fields[0].ColumnAsName.L) _, _, fields, err = tk.Session().PrepareStmt("select a from prepare3 where ?") require.NoError(t, err) require.Equal(t, "test", fields[0].DBName.L) require.Equal(t, "prepare3", fields[0].TableAsName.L) require.Equal(t, "a", fields[0].ColumnAsName.L) _, _, fields, err = tk.Session().PrepareStmt("select (1,1) in (select 1,1)") require.NoError(t, err) require.Equal(t, "", fields[0].DBName.L) require.Equal(t, "", fields[0].TableAsName.L) require.Equal(t, "(1,1) in (select 1,1)", fields[0].ColumnAsName.L) _, _, fields, err = tk.Session().PrepareStmt("select a from prepare3 where a = (" + "select a from prepare2 where a = ?)") require.NoError(t, err) require.Equal(t, "test", fields[0].DBName.L) require.Equal(t, "prepare3", fields[0].TableAsName.L) require.Equal(t, "a", fields[0].ColumnAsName.L) _, _, fields, err = tk.Session().PrepareStmt("select * from prepare3 as t1 join prepare3 as t2") require.NoError(t, err) require.Equal(t, "test", fields[0].DBName.L) require.Equal(t, "t1", fields[0].TableAsName.L) require.Equal(t, "a", fields[0].ColumnAsName.L) require.Equal(t, "test", fields[1].DBName.L) require.Equal(t, "t2", fields[1].TableAsName.L) require.Equal(t, "a", fields[1].ColumnAsName.L) _, _, fields, err = tk.Session().PrepareStmt("update prepare3 set a = ?") require.NoError(t, err) require.Len(t, fields, 0) // issue 8074 tk.MustExec("drop table if exists prepare1;") tk.MustExec("create table prepare1 (a decimal(1))") tk.MustExec("insert into prepare1 values(1);") _, err = tk.Exec("prepare stmt FROM @sql1") require.EqualError(t, err, "[parser:1064]You have an error in your SQL syntax; check the manual that corresponds to your TiDB version for the right syntax to use line 1 column 4 near \"NULL\" ") tk.MustExec("SET @sql = 'update prepare1 set a=5 where a=?';") _, err = tk.Exec("prepare stmt FROM @sql") require.NoError(t, err) tk.MustExec("set @var=1;") _, err = tk.Exec("execute stmt using @var") require.NoError(t, err) tk.MustQuery("select a from prepare1;").Check(testkit.Rows("5")) // issue 19371 tk.MustExec("SET @sql = 'update prepare1 set a=a+1';") _, err = tk.Exec("prepare stmt FROM @SQL") require.NoError(t, err) _, err = tk.Exec("execute stmt") require.NoError(t, err) tk.MustQuery("select a from prepare1;").Check(testkit.Rows("6")) _, err = tk.Exec("prepare stmt FROM @Sql") require.NoError(t, err) _, err = tk.Exec("execute stmt") require.NoError(t, err) tk.MustQuery("select a from prepare1;").Check(testkit.Rows("7")) // Coverage. exec := &executor.ExecuteExec{} err = exec.Next(ctx, nil) require.NoError(t, err) err = exec.Close() require.NoError(t, err) // issue 8065 stmtID, _, _, err = tk.Session().PrepareStmt("select ? from dual") require.NoError(t, err) _, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1)}) require.NoError(t, err) stmtID, _, _, err = tk.Session().PrepareStmt("update prepare1 set a = ? where a = ?") require.NoError(t, err) _, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1), types.NewDatum(1)}) require.NoError(t, err) } } func TestPreparedLimitOffset(t *testing.T) { store, clean := testkit.CreateMockStore(t) defer clean() orgEnable := plannercore.PreparedPlanCacheEnabled() defer func() { plannercore.SetPreparedPlanCache(orgEnable) }() flags := []bool{false, true} ctx := context.Background() for _, flag := range flags { plannercore.SetPreparedPlanCache(flag) tk := testkit.NewTestKit(t, store) se, err := session.CreateSession4TestWithOpt(store, &session.Opt{ PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64), }) require.NoError(t, err) tk.SetSession(se) tk.MustExec("use test") tk.MustExec("drop table if exists prepare_test") tk.MustExec("create table prepare_test (id int PRIMARY KEY AUTO_INCREMENT, c1 int, c2 int, c3 int default 1)") tk.MustExec("insert prepare_test (c1) values (1),(2),(NULL)") tk.MustExec(`prepare stmt_test_1 from 'select id from prepare_test limit ? offset ?'; set @a = 1, @b=1;`) r := tk.MustQuery(`execute stmt_test_1 using @a, @b;`) r.Check(testkit.Rows("2")) tk.MustExec(`set @a=1.1`) r = tk.MustQuery(`execute stmt_test_1 using @a, @b;`) r.Check(testkit.Rows("2")) tk.MustExec(`set @c="-1"`) _, err = tk.Exec("execute stmt_test_1 using @c, @c") require.True(t, plannercore.ErrWrongArguments.Equal(err)) stmtID, _, _, err := tk.Session().PrepareStmt("select id from prepare_test limit ?") require.NoError(t, err) _, err = tk.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{types.NewDatum(1)}) require.NoError(t, err) } } func TestPreparedNullParam(t *testing.T) { store, clean := testkit.CreateMockStore(t) defer clean() orgEnable := plannercore.PreparedPlanCacheEnabled() defer func() { plannercore.SetPreparedPlanCache(orgEnable) }() flags := []bool{false, true} for _, flag := range flags { plannercore.SetPreparedPlanCache(flag) tk := testkit.NewTestKit(t, store) se, err := session.CreateSession4TestWithOpt(store, &session.Opt{ PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64), }) require.NoError(t, err) tk.SetSession(se) tk.MustExec("use test") tk.MustExec("drop table if exists t") tk.MustExec("create table t (id int, KEY id (id))") tk.MustExec("insert into t values (1), (2), (3)") tk.MustExec(`prepare stmt from 'select * from t use index(id) where id = ?'`) r := tk.MustQuery(`execute stmt using @id;`) r.Check(nil) r = tk.MustQuery(`execute stmt using @id;`) r.Check(nil) tk.MustExec(`set @id="1"`) r = tk.MustQuery(`execute stmt using @id;`) r.Check(testkit.Rows("1")) r = tk.MustQuery(`execute stmt using @id2;`) r.Check(nil) r = tk.MustQuery(`execute stmt using @id;`) r.Check(testkit.Rows("1")) } } func TestPrepareWithAggregation(t *testing.T) { store, clean := testkit.CreateMockStore(t) defer clean() orgEnable := plannercore.PreparedPlanCacheEnabled() defer func() { plannercore.SetPreparedPlanCache(orgEnable) }() flags := []bool{false, true} for _, flag := range flags { plannercore.SetPreparedPlanCache(flag) tk := testkit.NewTestKit(t, store) se, err := session.CreateSession4TestWithOpt(store, &session.Opt{ PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64), }) require.NoError(t, err) tk.SetSession(se) tk.MustExec("use test") tk.MustExec("drop table if exists t") tk.MustExec("create table t (id int primary key)") tk.MustExec("insert into t values (1), (2), (3)") tk.MustExec(`prepare stmt from 'select sum(id) from t where id = ?'`) tk.MustExec(`set @id="1"`) r := tk.MustQuery(`execute stmt using @id;`) r.Check(testkit.Rows("1")) r = tk.MustQuery(`execute stmt using @id;`) r.Check(testkit.Rows("1")) } } func TestPreparedIssue7579(t *testing.T) { store, clean := testkit.CreateMockStore(t) defer clean() orgEnable := plannercore.PreparedPlanCacheEnabled() defer func() { plannercore.SetPreparedPlanCache(orgEnable) }() flags := []bool{false, true} for _, flag := range flags { plannercore.SetPreparedPlanCache(flag) tk := testkit.NewTestKit(t, store) se, err := session.CreateSession4TestWithOpt(store, &session.Opt{ PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64), }) require.NoError(t, err) tk.SetSession(se) tk.MustExec("use test") tk.MustExec("drop table if exists t") tk.MustExec("create table t (a int, b int, index a_idx(a))") tk.MustExec("insert into t values (1,1), (2,2), (null,3)") r := tk.MustQuery("select a, b from t order by b asc;") r.Check(testkit.Rows("1 1", "2 2", " 3")) tk.MustExec(`prepare stmt from 'select a, b from t where ? order by b asc'`) r = tk.MustQuery(`execute stmt using @param;`) r.Check(nil) tk.MustExec(`set @param = true`) r = tk.MustQuery(`execute stmt using @param;`) r.Check(testkit.Rows("1 1", "2 2", " 3")) tk.MustExec(`set @param = false`) r = tk.MustQuery(`execute stmt using @param;`) r.Check(nil) tk.MustExec(`set @param = 1`) r = tk.MustQuery(`execute stmt using @param;`) r.Check(testkit.Rows("1 1", "2 2", " 3")) tk.MustExec(`set @param = 0`) r = tk.MustQuery(`execute stmt using @param;`) r.Check(nil) } } func TestPreparedInsert(t *testing.T) { store, clean := testkit.CreateMockStore(t) defer clean() orgEnable := plannercore.PreparedPlanCacheEnabled() defer func() { plannercore.SetPreparedPlanCache(orgEnable) }() metrics.ResettablePlanCacheCounterFortTest = true metrics.PlanCacheCounter.Reset() counter := metrics.PlanCacheCounter.WithLabelValues("prepare") pb := &dto.Metric{} flags := []bool{false, true} for _, flag := range flags { plannercore.SetPreparedPlanCache(flag) tk := testkit.NewTestKit(t, store) se, err := session.CreateSession4TestWithOpt(store, &session.Opt{ PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64), }) require.NoError(t, err) tk.SetSession(se) tk.MustExec("use test") tk.MustExec("drop table if exists prepare_test") tk.MustExec("create table prepare_test (id int PRIMARY KEY, c1 int)") tk.MustExec(`prepare stmt_insert from 'insert into prepare_test values (?, ?)'`) tk.MustExec(`set @a=1,@b=1; execute stmt_insert using @a, @b;`) if flag { err = counter.Write(pb) require.NoError(t, err) hit := pb.GetCounter().GetValue() require.Equal(t, float64(0), hit) } tk.MustExec(`set @a=2,@b=2; execute stmt_insert using @a, @b;`) if flag { err = counter.Write(pb) require.NoError(t, err) hit := pb.GetCounter().GetValue() require.Equal(t, float64(1), hit) } tk.MustExec(`set @a=3,@b=3; execute stmt_insert using @a, @b;`) if flag { err = counter.Write(pb) require.NoError(t, err) hit := pb.GetCounter().GetValue() require.Equal(t, float64(2), hit) } result := tk.MustQuery("select id, c1 from prepare_test where id = ?", 1) result.Check(testkit.Rows("1 1")) result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 2) result.Check(testkit.Rows("2 2")) result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 3) result.Check(testkit.Rows("3 3")) tk.MustExec(`prepare stmt_insert_select from 'insert into prepare_test (id, c1) select id + 100, c1 + 100 from prepare_test where id = ?'`) tk.MustExec(`set @a=1; execute stmt_insert_select using @a;`) if flag { err = counter.Write(pb) require.NoError(t, err) hit := pb.GetCounter().GetValue() require.Equal(t, float64(2), hit) } tk.MustExec(`set @a=2; execute stmt_insert_select using @a;`) if flag { err = counter.Write(pb) require.NoError(t, err) hit := pb.GetCounter().GetValue() require.Equal(t, float64(3), hit) } tk.MustExec(`set @a=3; execute stmt_insert_select using @a;`) if flag { err = counter.Write(pb) require.NoError(t, err) hit := pb.GetCounter().GetValue() require.Equal(t, float64(4), hit) } result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 101) result.Check(testkit.Rows("101 101")) result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 102) result.Check(testkit.Rows("102 102")) result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 103) result.Check(testkit.Rows("103 103")) } } func TestPreparedUpdate(t *testing.T) { store, clean := testkit.CreateMockStore(t) defer clean() orgEnable := plannercore.PreparedPlanCacheEnabled() defer func() { plannercore.SetPreparedPlanCache(orgEnable) }() metrics.ResettablePlanCacheCounterFortTest = true metrics.PlanCacheCounter.Reset() counter := metrics.PlanCacheCounter.WithLabelValues("prepare") pb := &dto.Metric{} flags := []bool{false, true} for _, flag := range flags { plannercore.SetPreparedPlanCache(flag) tk := testkit.NewTestKit(t, store) se, err := session.CreateSession4TestWithOpt(store, &session.Opt{ PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64), }) require.NoError(t, err) tk.SetSession(se) tk.MustExec("use test") tk.MustExec("drop table if exists prepare_test") tk.MustExec("create table prepare_test (id int PRIMARY KEY, c1 int)") tk.MustExec(`insert into prepare_test values (1, 1)`) tk.MustExec(`insert into prepare_test values (2, 2)`) tk.MustExec(`insert into prepare_test values (3, 3)`) tk.MustExec(`prepare stmt_update from 'update prepare_test set c1 = c1 + ? where id = ?'`) tk.MustExec(`set @a=1,@b=100; execute stmt_update using @b,@a;`) if flag { err = counter.Write(pb) require.NoError(t, err) hit := pb.GetCounter().GetValue() require.Equal(t, float64(0), hit) } tk.MustExec(`set @a=2,@b=200; execute stmt_update using @b,@a;`) if flag { err = counter.Write(pb) require.NoError(t, err) hit := pb.GetCounter().GetValue() require.Equal(t, float64(1), hit) } tk.MustExec(`set @a=3,@b=300; execute stmt_update using @b,@a;`) if flag { err = counter.Write(pb) require.NoError(t, err) hit := pb.GetCounter().GetValue() require.Equal(t, float64(2), hit) } result := tk.MustQuery("select id, c1 from prepare_test where id = ?", 1) result.Check(testkit.Rows("1 101")) result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 2) result.Check(testkit.Rows("2 202")) result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 3) result.Check(testkit.Rows("3 303")) } } func TestIssue21884(t *testing.T) { store, clean := testkit.CreateMockStore(t) defer clean() orgEnable := plannercore.PreparedPlanCacheEnabled() defer func() { plannercore.SetPreparedPlanCache(orgEnable) }() plannercore.SetPreparedPlanCache(false) tk := testkit.NewTestKit(t, store) tk.MustExec("use test") tk.MustExec("drop table if exists prepare_test") tk.MustExec("create table prepare_test(a bigint primary key, status bigint, last_update_time datetime)") tk.MustExec("insert into prepare_test values (100, 0, '2020-12-18 20:00:00')") tk.MustExec("prepare stmt from 'update prepare_test set status = ?, last_update_time = now() where a = 100'") tk.MustExec("set @status = 1") tk.MustExec("execute stmt using @status") updateTime := tk.MustQuery("select last_update_time from prepare_test").Rows()[0][0] // Sleep 1 second to make sure `last_update_time` is updated. time.Sleep(1 * time.Second) tk.MustExec("execute stmt using @status") newUpdateTime := tk.MustQuery("select last_update_time from prepare_test").Rows()[0][0] require.NotEqual(t, newUpdateTime, updateTime) } func TestPreparedDelete(t *testing.T) { store, clean := testkit.CreateMockStore(t) defer clean() orgEnable := plannercore.PreparedPlanCacheEnabled() defer func() { plannercore.SetPreparedPlanCache(orgEnable) }() metrics.ResettablePlanCacheCounterFortTest = true metrics.PlanCacheCounter.Reset() counter := metrics.PlanCacheCounter.WithLabelValues("prepare") pb := &dto.Metric{} flags := []bool{false, true} for _, flag := range flags { plannercore.SetPreparedPlanCache(flag) tk := testkit.NewTestKit(t, store) se, err := session.CreateSession4TestWithOpt(store, &session.Opt{ PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64), }) require.NoError(t, err) tk.SetSession(se) tk.MustExec("use test") tk.MustExec("drop table if exists prepare_test") tk.MustExec("create table prepare_test (id int PRIMARY KEY, c1 int)") tk.MustExec(`insert into prepare_test values (1, 1)`) tk.MustExec(`insert into prepare_test values (2, 2)`) tk.MustExec(`insert into prepare_test values (3, 3)`) tk.MustExec(`prepare stmt_delete from 'delete from prepare_test where id = ?'`) tk.MustExec(`set @a=1; execute stmt_delete using @a;`) if flag { err = counter.Write(pb) require.NoError(t, err) hit := pb.GetCounter().GetValue() require.Equal(t, float64(0), hit) } tk.MustExec(`set @a=2; execute stmt_delete using @a;`) if flag { err = counter.Write(pb) require.NoError(t, err) hit := pb.GetCounter().GetValue() require.Equal(t, float64(1), hit) } tk.MustExec(`set @a=3; execute stmt_delete using @a;`) if flag { err = counter.Write(pb) require.NoError(t, err) hit := pb.GetCounter().GetValue() require.Equal(t, float64(2), hit) } result := tk.MustQuery("select id, c1 from prepare_test where id = ?", 1) result.Check(nil) result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 2) result.Check(nil) result = tk.MustQuery("select id, c1 from prepare_test where id = ?", 3) result.Check(nil) } } func TestPrepareDealloc(t *testing.T) { store, clean := testkit.CreateMockStore(t) defer clean() orgEnable := plannercore.PreparedPlanCacheEnabled() defer func() { plannercore.SetPreparedPlanCache(orgEnable) }() plannercore.SetPreparedPlanCache(true) tk := testkit.NewTestKit(t, store) se, err := session.CreateSession4TestWithOpt(store, &session.Opt{ PreparedPlanCache: kvcache.NewSimpleLRUCache(3, 0.1, math.MaxUint64), }) require.NoError(t, err) tk.SetSession(se) tk.MustExec("use test") tk.MustExec("drop table if exists prepare_test") tk.MustExec("create table prepare_test (id int PRIMARY KEY, c1 int)") require.Equal(t, 0, tk.Session().PreparedPlanCache().Size()) tk.MustExec(`prepare stmt1 from 'select * from prepare_test'`) tk.MustExec("execute stmt1") tk.MustExec(`prepare stmt2 from 'select * from prepare_test'`) tk.MustExec("execute stmt2") tk.MustExec(`prepare stmt3 from 'select * from prepare_test'`) tk.MustExec("execute stmt3") tk.MustExec(`prepare stmt4 from 'select * from prepare_test'`) tk.MustExec("execute stmt4") require.Equal(t, 3, tk.Session().PreparedPlanCache().Size()) tk.MustExec("deallocate prepare stmt1") require.Equal(t, 3, tk.Session().PreparedPlanCache().Size()) tk.MustExec("deallocate prepare stmt2") tk.MustExec("deallocate prepare stmt3") tk.MustExec("deallocate prepare stmt4") require.Equal(t, 0, tk.Session().PreparedPlanCache().Size()) } func TestPreparedIssue8153(t *testing.T) { store, clean := testkit.CreateMockStore(t) defer clean() orgEnable := plannercore.PreparedPlanCacheEnabled() defer func() { plannercore.SetPreparedPlanCache(orgEnable) }() flags := []bool{false, true} for _, flag := range flags { plannercore.SetPreparedPlanCache(flag) tk := testkit.NewTestKit(t, store) se, err := session.CreateSession4TestWithOpt(store, &session.Opt{ PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64), }) require.NoError(t, err) tk.SetSession(se) tk.MustExec("use test") tk.MustExec("drop table if exists t") tk.MustExec("create table t (a int, b int)") tk.MustExec("insert into t (a, b) values (1,3), (2,2), (3,1)") tk.MustExec(`prepare stmt from 'select * from t order by ? asc'`) r := tk.MustQuery(`execute stmt using @param;`) r.Check(testkit.Rows("1 3", "2 2", "3 1")) tk.MustExec(`set @param = 1`) r = tk.MustQuery(`execute stmt using @param;`) r.Check(testkit.Rows("1 3", "2 2", "3 1")) tk.MustExec(`set @param = 2`) r = tk.MustQuery(`execute stmt using @param;`) r.Check(testkit.Rows("3 1", "2 2", "1 3")) tk.MustExec(`set @param = 3`) _, err = tk.Exec(`execute stmt using @param;`) require.EqualError(t, err, "[planner:1054]Unknown column '?' in 'order clause'") tk.MustExec(`set @param = '##'`) r = tk.MustQuery(`execute stmt using @param;`) r.Check(testkit.Rows("1 3", "2 2", "3 1")) tk.MustExec("insert into t (a, b) values (1,1), (1,2), (2,1), (2,3), (3,2), (3,3)") tk.MustExec(`prepare stmt from 'select ?, sum(a) from t group by ?'`) tk.MustExec(`set @a=1,@b=1`) r = tk.MustQuery(`execute stmt using @a,@b;`) r.Check(testkit.Rows("1 18")) tk.MustExec(`set @a=1,@b=2`) _, err = tk.Exec(`execute stmt using @a,@b;`) require.EqualError(t, err, "[planner:1056]Can't group on 'sum(a)'") } } func TestPreparedIssue8644(t *testing.T) { store, clean := testkit.CreateMockStore(t) defer clean() orgEnable := plannercore.PreparedPlanCacheEnabled() defer func() { plannercore.SetPreparedPlanCache(orgEnable) }() flags := []bool{false, true} for _, flag := range flags { plannercore.SetPreparedPlanCache(flag) tk := testkit.NewTestKit(t, store) se, err := session.CreateSession4TestWithOpt(store, &session.Opt{ PreparedPlanCache: kvcache.NewSimpleLRUCache(100, 0.1, math.MaxUint64), }) require.NoError(t, err) tk.SetSession(se) tk.MustExec("use test") tk.MustExec("drop table if exists t") tk.MustExec("create table t(data mediumblob)") tk.MustExec(`prepare stmt from 'insert t (data) values (?)'`) tk.MustExec(`set @a = 'a'`) tk.MustExec(`execute stmt using @a;`) tk.MustExec(`set @b = 'aaaaaaaaaaaaaaaaaa'`) tk.MustExec(`execute stmt using @b;`) r := tk.MustQuery(`select * from t`) r.Check(testkit.Rows("a", "aaaaaaaaaaaaaaaaaa")) tk.MustExec("drop table if exists t") tk.MustExec("create table t(data decimal)") tk.MustExec(`prepare stmt from 'insert t (data) values (?)'`) tk.MustExec(`set @a = '1'`) tk.MustExec(`execute stmt using @a;`) tk.MustExec(`set @b = '11111.11111'`) // '.11111' will be truncated. tk.MustExec(`execute stmt using @b;`) r = tk.MustQuery(`select * from t`) r.Check(testkit.Rows("1", "11111")) tk.MustExec("drop table if exists t") tk.MustExec("create table t(data decimal(10,3));") tk.MustExec("prepare stmt from 'insert t (data) values (?)';") tk.MustExec("set @a = 1.1;") tk.MustExec("execute stmt using @a;") tk.MustExec("set @b = 11.11;") tk.MustExec("execute stmt using @b;") r = tk.MustQuery(`select * from t`) r.Check(testkit.Rows("1.100", "11.110")) } } // mockSessionManager is a mocked session manager which is used for test. type mockSessionManager1 struct { Se session.Session } func (msm *mockSessionManager1) ShowTxnList() []*txninfo.TxnInfo { panic("unimplemented!") } // ShowProcessList implements the SessionManager.ShowProcessList interface. func (msm *mockSessionManager1) ShowProcessList() map[uint64]*util.ProcessInfo { ret := make(map[uint64]*util.ProcessInfo) return ret } func (msm *mockSessionManager1) GetProcessInfo(_ uint64) (*util.ProcessInfo, bool) { pi := msm.Se.ShowProcess() return pi, true } // Kill implements the SessionManager.Kill interface. func (msm *mockSessionManager1) Kill(_ uint64, _ bool) {} func (msm *mockSessionManager1) KillAllConnections() {} func (msm *mockSessionManager1) ServerID() uint64 { return 1 } func (msm *mockSessionManager1) UpdateTLSConfig(_ *tls.Config) {} func TestPreparedIssue17419(t *testing.T) { store, dom, clean := testkit.CreateMockStoreAndDomain(t) defer clean() ctx := context.Background() tk := testkit.NewTestKit(t, store) tk.MustExec("use test") tk.MustExec("drop table if exists t") tk.MustExec("create table t (a int)") tk.MustExec("insert into t (a) values (1), (2), (3)") tk1 := testkit.NewTestKit(t, store) query := "select * from test.t" stmtID, _, _, err := tk1.Session().PrepareStmt(query) require.NoError(t, err) sm := &mockSessionManager1{ Se: tk1.Session(), } tk1.Session().SetSessionManager(sm) dom.ExpensiveQueryHandle().SetSessionManager(sm) rs, err := tk1.Session().ExecutePreparedStmt(ctx, stmtID, []types.Datum{}) require.NoError(t, err) tk1.ResultSetToResult(rs, fmt.Sprintf("%v", rs)).Check(testkit.Rows("1", "2", "3")) tk1.Session().SetProcessInfo("", time.Now(), mysql.ComStmtExecute, 0) dom.ExpensiveQueryHandle().LogOnQueryExceedMemQuota(tk.Session().GetSessionVars().ConnectionID) // After entirely fixing https://github.com/pingcap/tidb/issues/17419 // require.NotNil(t, tk1.Session().ShowProcess().Plan) // _, ok := tk1.Session().ShowProcess().Plan.(*plannercore.Execute) // require.True(t, ok) }