229 lines
10 KiB
Go
229 lines
10 KiB
Go
// 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,
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package executor_test
|
|
|
|
import (
|
|
"fmt"
|
|
"strings"
|
|
|
|
. "github.com/pingcap/check"
|
|
"github.com/pingcap/parser/auth"
|
|
"github.com/pingcap/parser/model"
|
|
"github.com/pingcap/tidb/domain"
|
|
plannercore "github.com/pingcap/tidb/planner/core"
|
|
"github.com/pingcap/tidb/util"
|
|
"github.com/pingcap/tidb/util/testkit"
|
|
)
|
|
|
|
func (s *testSuite1) TestPreparedNameResolver(c *C) {
|
|
tk := testkit.NewTestKit(c, s.store)
|
|
tk.MustExec("use test")
|
|
tk.MustExec("drop table if exists t")
|
|
tk.MustExec("create table t (id int, KEY id (id))")
|
|
tk.MustExec("prepare stmt from 'select * from t limit ? offset ?'")
|
|
_, err := tk.Exec("prepare stmt from 'select b from t'")
|
|
c.Assert(err.Error(), Equals, "[planner:1054]Unknown column 'b' in 'field list'")
|
|
|
|
_, err = tk.Exec("prepare stmt from '(select * FROM t) union all (select * FROM t) order by a limit ?'")
|
|
c.Assert(err.Error(), Equals, "[planner:1054]Unknown column 'a' in 'order clause'")
|
|
}
|
|
|
|
// a 'create table' DDL statement should be accepted if it has no parameters.
|
|
func (s *testSuite1) TestPreparedDDL(c *C) {
|
|
tk := testkit.NewTestKit(c, s.store)
|
|
tk.MustExec("use test")
|
|
tk.MustExec("drop table if exists t")
|
|
tk.MustExec("prepare stmt from 'create table t (id int, KEY id (id))'")
|
|
}
|
|
|
|
func (s *testSuite1) TestIgnorePlanCache(c *C) {
|
|
tk := testkit.NewTestKit(c, s.store)
|
|
tk.MustExec("use test")
|
|
tk.MustExec("drop table if exists t")
|
|
|
|
tk.MustExec("create table t (id int primary key, num int)")
|
|
tk.MustExec("insert into t values (1, 1)")
|
|
tk.MustExec("insert into t values (2, 2)")
|
|
tk.MustExec("insert into t values (3, 3)")
|
|
tk.MustExec("prepare stmt from 'select /*+ IGNORE_PLAN_CACHE() */ * from t where id=?'")
|
|
tk.MustExec("set @ignore_plan_doma = 1")
|
|
tk.MustExec("execute stmt using @ignore_plan_doma")
|
|
c.Assert(tk.Se.GetSessionVars().StmtCtx.UseCache, IsFalse)
|
|
}
|
|
|
|
func (s *testSuite1) TestPrepareStmtAfterIsolationReadChange(c *C) {
|
|
tk := testkit.NewTestKitWithInit(c, s.store)
|
|
tk.Se.Auth(&auth.UserIdentity{Username: "root", Hostname: "localhost", CurrentUser: true, AuthUsername: "root", AuthHostname: "%"}, nil, []byte("012345678901234567890"))
|
|
|
|
tk.MustExec("drop table if exists t")
|
|
tk.MustExec("create table t(a int)")
|
|
// create virtual tiflash replica.
|
|
dom := domain.GetDomain(tk.Se)
|
|
is := dom.InfoSchema()
|
|
db, exists := is.SchemaByName(model.NewCIStr("test"))
|
|
c.Assert(exists, IsTrue)
|
|
for _, tblInfo := range db.Tables {
|
|
if tblInfo.Name.L == "t" {
|
|
tblInfo.TiFlashReplica = &model.TiFlashReplicaInfo{
|
|
Count: 1,
|
|
Available: true,
|
|
}
|
|
}
|
|
}
|
|
|
|
tk.MustExec("set @@session.tidb_isolation_read_engines='tikv'")
|
|
tk.MustExec("prepare stmt from \"select * from t\"")
|
|
tk.MustQuery("execute stmt")
|
|
tkProcess := tk.Se.ShowProcess()
|
|
ps := []*util.ProcessInfo{tkProcess}
|
|
tk.Se.SetSessionManager(&mockSessionManager1{PS: ps})
|
|
rows := tk.MustQuery(fmt.Sprintf("explain for connection %d", tkProcess.ID)).Rows()
|
|
c.Assert(rows[len(rows)-1][2], Equals, "cop[tikv]")
|
|
|
|
tk.MustExec("set @@session.tidb_isolation_read_engines='tiflash'")
|
|
tk.MustExec("execute stmt")
|
|
tkProcess = tk.Se.ShowProcess()
|
|
ps = []*util.ProcessInfo{tkProcess}
|
|
tk.Se.SetSessionManager(&mockSessionManager1{PS: ps})
|
|
rows = tk.MustQuery(fmt.Sprintf("explain for connection %d", tkProcess.ID)).Rows()
|
|
c.Assert(rows[len(rows)-1][2], Equals, "cop[tiflash]")
|
|
|
|
c.Assert(len(tk.Se.GetSessionVars().PreparedStmts), Equals, 1)
|
|
c.Assert(tk.Se.GetSessionVars().PreparedStmts[1].(*plannercore.CachedPrepareStmt).NormalizedSQL, Equals, "select * from t")
|
|
c.Assert(tk.Se.GetSessionVars().PreparedStmts[1].(*plannercore.CachedPrepareStmt).NormalizedPlan, Equals, "")
|
|
}
|
|
|
|
func (s *testSuite9) TestPlanCacheClusterIndex(c *C) {
|
|
store, dom, err := newStoreWithBootstrap()
|
|
c.Assert(err, IsNil)
|
|
tk := testkit.NewTestKit(c, store)
|
|
defer func() {
|
|
dom.Close()
|
|
store.Close()
|
|
}()
|
|
orgEnable := plannercore.PreparedPlanCacheEnabled()
|
|
defer func() {
|
|
plannercore.SetPreparedPlanCache(orgEnable)
|
|
}()
|
|
plannercore.SetPreparedPlanCache(true)
|
|
tk.MustExec("use test")
|
|
tk.MustExec("drop table if exists t1")
|
|
tk.MustExec("set @@tidb_enable_clustered_index = 1")
|
|
tk.MustExec("create table t1(a varchar(20), b varchar(20), c varchar(20), primary key(a, b))")
|
|
tk.MustExec("insert into t1 values('1','1','111'),('2','2','222'),('3','3','333')")
|
|
|
|
// For table scan
|
|
tk.MustExec(`prepare stmt1 from "select * from t1 where t1.a = ? and t1.b > ?"`)
|
|
tk.MustExec("set @v1 = 1")
|
|
tk.MustExec("set @v2 = 0")
|
|
tk.MustQuery("execute stmt1 using @v1,@v2").Check(testkit.Rows("1 1 111"))
|
|
tk.MustQuery("select @@last_plan_from_cache").Check(testkit.Rows("0"))
|
|
tk.MustExec("set @v1 = 2")
|
|
tk.MustExec("set @v2 = 1")
|
|
tk.MustQuery("execute stmt1 using @v1,@v2").Check(testkit.Rows("2 2 222"))
|
|
tk.MustQuery("select @@last_plan_from_cache").Check(testkit.Rows("1"))
|
|
tk.MustExec("set @v1 = 3")
|
|
tk.MustExec("set @v2 = 2")
|
|
tk.MustQuery("execute stmt1 using @v1,@v2").Check(testkit.Rows("3 3 333"))
|
|
tkProcess := tk.Se.ShowProcess()
|
|
ps := []*util.ProcessInfo{tkProcess}
|
|
tk.Se.SetSessionManager(&mockSessionManager1{PS: ps})
|
|
rows := tk.MustQuery(fmt.Sprintf("explain for connection %d", tkProcess.ID)).Rows()
|
|
c.Assert(strings.Index(rows[len(rows)-1][4].(string), `range:("3" "2","3" +inf]`), Equals, 0)
|
|
|
|
// For point get
|
|
tk.MustExec(`prepare stmt2 from "select * from t1 where t1.a = ? and t1.b = ?"`)
|
|
tk.MustExec("set @v1 = 1")
|
|
tk.MustExec("set @v2 = 1")
|
|
tk.MustQuery("execute stmt2 using @v1,@v2").Check(testkit.Rows("1 1 111"))
|
|
tk.MustQuery("select @@last_plan_from_cache").Check(testkit.Rows("0"))
|
|
tk.MustExec("set @v1 = 2")
|
|
tk.MustExec("set @v2 = 2")
|
|
tk.MustQuery("execute stmt2 using @v1,@v2").Check(testkit.Rows("2 2 222"))
|
|
tk.MustQuery("select @@last_plan_from_cache").Check(testkit.Rows("1"))
|
|
tk.MustExec("set @v1 = 3")
|
|
tk.MustExec("set @v2 = 3")
|
|
tk.MustQuery("execute stmt2 using @v1,@v2").Check(testkit.Rows("3 3 333"))
|
|
tkProcess = tk.Se.ShowProcess()
|
|
ps = []*util.ProcessInfo{tkProcess}
|
|
tk.Se.SetSessionManager(&mockSessionManager1{PS: ps})
|
|
rows = tk.MustQuery(fmt.Sprintf("explain for connection %d", tkProcess.ID)).Rows()
|
|
c.Assert(strings.Index(rows[len(rows)-1][0].(string), `Point_Get`), Equals, 0)
|
|
|
|
// For CBO point get and batch point get
|
|
// case 1:
|
|
tk.MustExec(`drop table if exists ta, tb`)
|
|
tk.MustExec(`create table ta (a varchar(8) primary key, b int)`)
|
|
tk.MustExec(`insert ta values ('a', 1), ('b', 2)`)
|
|
tk.MustExec(`create table tb (a varchar(8) primary key, b int)`)
|
|
tk.MustExec(`insert tb values ('a', 1), ('b', 2)`)
|
|
tk.MustExec(`prepare stmt1 from "select * from ta, tb where ta.a = tb.a and ta.a = ?"`)
|
|
tk.MustExec(`set @v1 = 'a', @v2 = 'b'`)
|
|
tk.MustQuery(`execute stmt1 using @v1`).Check(testkit.Rows("a 1 a 1"))
|
|
tk.MustQuery(`execute stmt1 using @v2`).Check(testkit.Rows("b 2 b 2"))
|
|
tk.MustQuery("select @@last_plan_from_cache").Check(testkit.Rows("1"))
|
|
|
|
// case 2:
|
|
tk.MustExec(`drop table if exists ta, tb`)
|
|
tk.MustExec(`create table ta (a varchar(10) primary key, b int not null)`)
|
|
tk.MustExec(`insert ta values ('a', 1), ('b', 2)`)
|
|
tk.MustExec(`create table tb (b int primary key, c int)`)
|
|
tk.MustExec(`insert tb values (1, 1), (2, 2)`)
|
|
tk.MustExec(`prepare stmt1 from "select * from ta, tb where ta.b = tb.b and ta.a = ?"`)
|
|
tk.MustExec(`set @v1 = 'a', @v2 = 'b'`)
|
|
tk.MustQuery(`execute stmt1 using @v1`).Check(testkit.Rows("a 1 1 1"))
|
|
tk.MustQuery(`execute stmt1 using @v2`).Check(testkit.Rows("b 2 2 2"))
|
|
tk.MustQuery("select @@last_plan_from_cache").Check(testkit.Rows("1"))
|
|
tk.MustQuery(`execute stmt1 using @v2`).Check(testkit.Rows("b 2 2 2"))
|
|
tkProcess = tk.Se.ShowProcess()
|
|
ps = []*util.ProcessInfo{tkProcess}
|
|
tk.Se.SetSessionManager(&mockSessionManager1{PS: ps})
|
|
rows = tk.MustQuery(fmt.Sprintf("explain for connection %d", tkProcess.ID)).Rows()
|
|
c.Assert(strings.Index(rows[1][0].(string), `Point_Get`), Equals, 6)
|
|
|
|
// case 3:
|
|
tk.MustExec(`drop table if exists ta, tb`)
|
|
tk.MustExec(`create table ta (a varchar(10), b varchar(10), c int, primary key (a, b))`)
|
|
tk.MustExec(`insert ta values ('a', 'a', 1), ('b', 'b', 2), ('c', 'c', 3)`)
|
|
tk.MustExec(`create table tb (b int primary key, c int)`)
|
|
tk.MustExec(`insert tb values (1, 1), (2, 2), (3,3)`)
|
|
tk.MustExec(`prepare stmt1 from "select * from ta, tb where ta.c = tb.b and ta.a = ? and ta.b = ?"`)
|
|
tk.MustExec(`set @v1 = 'a', @v2 = 'b', @v3 = 'c'`)
|
|
tk.MustQuery(`execute stmt1 using @v1, @v1`).Check(testkit.Rows("a a 1 1 1"))
|
|
tk.MustQuery(`execute stmt1 using @v2, @v2`).Check(testkit.Rows("b b 2 2 2"))
|
|
tk.MustExec(`prepare stmt2 from "select * from ta, tb where ta.c = tb.b and (ta.a, ta.b) in ((?, ?), (?, ?))"`)
|
|
tk.MustQuery(`execute stmt2 using @v1, @v1, @v2, @v2`).Check(testkit.Rows("a a 1 1 1", "b b 2 2 2"))
|
|
tk.MustQuery(`execute stmt2 using @v2, @v2, @v3, @v3`).Check(testkit.Rows("b b 2 2 2", "c c 3 3 3"))
|
|
|
|
// For issue 19002
|
|
tk.MustExec(`set @@tidb_enable_clustered_index = 1`)
|
|
tk.MustExec(`drop table if exists t1`)
|
|
tk.MustExec(`create table t1(a int, b int, c int, primary key(a, b))`)
|
|
tk.MustExec(`insert into t1 values(1,1,111),(2,2,222),(3,3,333)`)
|
|
// Point Get:
|
|
tk.MustExec(`prepare stmt1 from "select * from t1 where t1.a = ? and t1.b = ?"`)
|
|
tk.MustExec(`set @v1=1, @v2=1`)
|
|
tk.MustQuery(`execute stmt1 using @v1,@v2`).Check(testkit.Rows("1 1 111"))
|
|
tk.MustExec(`set @v1=2, @v2=2`)
|
|
tk.MustQuery(`execute stmt1 using @v1,@v2`).Check(testkit.Rows("2 2 222"))
|
|
tk.MustQuery(`select @@last_plan_from_cache`).Check(testkit.Rows("1"))
|
|
// Batch Point Get:
|
|
tk.MustExec(`prepare stmt2 from "select * from t1 where (t1.a,t1.b) in ((?,?),(?,?))"`)
|
|
tk.MustExec(`set @v1=1, @v2=1, @v3=2, @v4=2`)
|
|
tk.MustQuery(`execute stmt2 using @v1,@v2,@v3,@v4`).Check(testkit.Rows("1 1 111", "2 2 222"))
|
|
tk.MustExec(`set @v1=2, @v2=2, @v3=3, @v4=3`)
|
|
tk.MustQuery(`execute stmt2 using @v1,@v2,@v3,@v4`).Check(testkit.Rows("2 2 222", "3 3 333"))
|
|
tk.MustQuery(`select @@last_plan_from_cache`).Check(testkit.Rows("1"))
|
|
}
|