Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

*: support create global binding #9846

Merged
merged 54 commits into from
Apr 17, 2019
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
54 commits
Select commit Hold shift + click to select a range
0cebb75
plan binding:add global binding
iamzhoug37 Mar 21, 2019
4f5df56
fix tidy fmt lint
iamzhoug37 Mar 21, 2019
68ee4d8
remove blan
iamzhoug37 Mar 21, 2019
9747bff
fix fmt
iamzhoug37 Mar 21, 2019
44d4619
fix error-check
iamzhoug37 Mar 21, 2019
0d85387
fix createBindPlan originSql and bindSql unnormolize
iamzhoug37 Mar 22, 2019
150b574
fix import
iamzhoug37 Mar 22, 2019
e2dc348
add index for origin_sql and defaultDB
iamzhoug37 Mar 25, 2019
367c7cc
merge master
iamzhoug37 Mar 25, 2019
5aa25ef
fix comment
iamzhoug37 Mar 26, 2019
6c1bd81
add version update
iamzhoug37 Mar 28, 2019
ae25b95
fix go.sum
iamzhoug37 Mar 28, 2019
ea3940a
replace user session ctx with domain's session ctx
iamzhoug37 Mar 29, 2019
e9e9cb0
fix lint
iamzhoug37 Mar 29, 2019
9cadfff
fix comment
iamzhoug37 Apr 1, 2019
787564e
xx
iamzhoug37 Apr 1, 2019
2d7d6d3
fx comment
iamzhoug37 Apr 1, 2019
3416b81
fix ci
iamzhoug37 Apr 1, 2019
ecd00b4
fix version
iamzhoug37 Apr 1, 2019
1df98f0
replace originalSQL with rowId
iamzhoug37 Apr 2, 2019
91b7494
fix ci
iamzhoug37 Apr 2, 2019
8849d3d
fix comment
iamzhoug37 Apr 2, 2019
7e9d879
add privilage check
iamzhoug37 Apr 3, 2019
45c24f3
fix make lint
iamzhoug37 Apr 3, 2019
e89f6c9
create bind plan
iamzhoug37 Apr 3, 2019
d77a038
fix comment
iamzhoug37 Apr 3, 2019
520ea9e
fix comment
iamzhoug37 Apr 4, 2019
23e3b97
remove unused code
iamzhoug37 Apr 4, 2019
924145e
timestamp bugfix
iamzhoug37 Apr 4, 2019
9e539a4
Merge branch 'master' into plan-management-add-global-binding
alivxxx Apr 4, 2019
fca0e5d
Merge branch 'master' into plan-management-add-global-binding
iamzhoug37 Apr 4, 2019
debf0c3
fix undefined error
iamzhoug37 Apr 4, 2019
bd1ae07
cjrioe
iamzhoug37 Apr 4, 2019
9380e05
merge master
iamzhoug37 Apr 12, 2019
8c6e4ec
add create update cache
iamzhoug37 Apr 15, 2019
682e5af
fix comment
iamzhoug37 Apr 16, 2019
d22f5dc
fix comment
iamzhoug37 Apr 16, 2019
3527cbc
lock
iamzhoug37 Apr 16, 2019
f45010a
aa
iamzhoug37 Apr 16, 2019
fda7535
tiny refine
zz-jason Apr 16, 2019
927346a
tiny refine planner and executor
zz-jason Apr 16, 2019
d5ffa60
release lock
iamzhoug37 Apr 16, 2019
918d8d6
Merge branch 'plan-management-add-global-binding' of github.com:iamzh…
iamzhoug37 Apr 16, 2019
efc4ad3
refine mutex in handle
zz-jason Apr 16, 2019
1b59531
Merge branch 'plan-management-add-global-binding' of github.com:iamzh…
iamzhoug37 Apr 16, 2019
33a2af4
code refactor
iamzhoug37 Apr 17, 2019
8a6b7b0
fix vet
iamzhoug37 Apr 17, 2019
85fa209
remove db
iamzhoug37 Apr 17, 2019
07ccf7b
Merge branch 'master' into plan-management-add-global-binding
zz-jason Apr 17, 2019
2bbb6d4
update cache
iamzhoug37 Apr 17, 2019
3e4539a
Merge branch 'plan-management-add-global-binding' of github.com:iamzh…
iamzhoug37 Apr 17, 2019
4fac6c3
make fmt
iamzhoug37 Apr 17, 2019
603074c
add licensed
iamzhoug37 Apr 17, 2019
2f10bb5
move import
iamzhoug37 Apr 17, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
61 changes: 61 additions & 0 deletions bindinfo/bind.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
// Copyright 2019 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 bindinfo

import (
"github.com/pingcap/tidb/sessionctx"
)

var _ Manager = (*BindManager)(nil)

// BindManager implements Manager inferface.
type BindManager struct {
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
GlobalAccessor GlobalBindAccessor
}

type keyType int

func (k keyType) String() string {
return "bind-key"
}

// Manager is the interface for providing bind related operations.
type Manager interface {
AddGlobalBind(originSQL, bindSQL, defaultDB, charset, collation string) error
}

const key keyType = 0

// BindBinder2Manager binds Manager to context.
func BindBinder2Manager(ctx sessionctx.Context, pc Manager) {
ctx.SetValue(key, pc)
}

// GetBindManager gets Checker from context.
func GetBindManager(ctx sessionctx.Context) Manager {
if v, ok := ctx.Value(key).(Manager); ok {
return v
}
return nil
}

//AddGlobalBind implements Manager's AddGlobalBind interface.
func (b *BindManager) AddGlobalBind(originSQL, bindSQL, defaultDB, charset, collation string) error {
return b.GlobalAccessor.AddGlobalBind(originSQL, bindSQL, defaultDB, charset, collation)
}

// GlobalBindAccessor is the interface for accessing global bind info.
type GlobalBindAccessor interface {
AddGlobalBind(originSQL string, bindSQL string, defaultDB string, charset string, collation string) error
}
41 changes: 41 additions & 0 deletions bindinfo/bind_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -134,3 +134,44 @@ func (s *testSuite) TestBindParse(c *C) {
c.Check(bindData[0].CreateTime, NotNil)
c.Check(bindData[0].UpdateTime, NotNil)
}

func (s *testSuite) TestGlobalBinding(c *C) {
tk := testkit.NewTestKit(c, s.store)
s.cleanBindingEnv(tk)
tk.MustExec("use test")
tk.MustExec("drop table if exists t")
tk.MustExec("drop table if exists t1")
tk.MustExec("create table t(i int, s varchar(20))")
tk.MustExec("create table t1(i int, s varchar(20))")
tk.MustExec("create index index_t on t(i,s)")

_, err := tk.Exec("create global binding for select * from t where i>100 using select * from t use index(index_t) where i>100")
c.Assert(err, IsNil, Commentf("err %v", err))
_, err = tk.Exec("create global binding for select * from t where i>99 using select * from t use index(index_t) where i>99")
c.Assert(err, NotNil)

bindHandle := bindinfo.NewHandle()
bindCacheUpdater := bindinfo.NewBindCacheUpdater(tk.Se, bindHandle, s.Parser)
err = bindCacheUpdater.Update(true)
c.Check(err, IsNil)
c.Check(len(bindHandle.Get()), Equals, 1)

hash := parser.DigestHash("select * from t where i>100")
bindData := bindHandle.Get()[hash]
c.Check(bindData, NotNil)
c.Check(len(bindData), Equals, 1)
c.Check(bindData[0].OriginalSQL, Equals, "select * from t where i > ?")
c.Check(bindData[0].BindSQL, Equals, "select * from t use index(index_t) where i>100")
c.Check(bindData[0].Db, Equals, "test")
c.Check(bindData[0].Status, Equals, "using")
c.Check(bindData[0].Charset, NotNil)
c.Check(bindData[0].Collation, NotNil)
c.Check(bindData[0].CreateTime, NotNil)
c.Check(bindData[0].UpdateTime, NotNil)

_, err = tk.Exec("delete from mysql.bind_info")
c.Assert(err, IsNil)

_, err = tk.Exec("create global binding for select * from t using select * from t1 use index for join(index_t)")
c.Assert(err, NotNil, Commentf("err %v", err))
}
8 changes: 5 additions & 3 deletions bindinfo/cache.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,8 +29,10 @@ import (
)

const (
using = "using"
deleted = "deleted"
// BindUsing is the bind info's in use status.
BindUsing = "using"
// BindDeleted is the bind info's deleted status.
BindDeleted = "deleted"
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
)

// bindMeta stores the basic bind info and bindSql astNode.
Expand Down Expand Up @@ -177,7 +179,7 @@ func (b cache) appendNode(newBindRecord *bindRecord, sparser *parser.Parser) err
}
}
}
if newBindRecord.Status == deleted {
if newBindRecord.Status == BindDeleted {
return nil
}
stmtNodes, _, err := sparser.Parse(newBindRecord.BindSQL, newBindRecord.Charset, newBindRecord.Collation)
Expand Down
50 changes: 50 additions & 0 deletions executor/bind.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
// Copyright 2019 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

import (
"context"

"github.com/pingcap/errors"
"github.com/pingcap/parser/ast"
"github.com/pingcap/tidb/bindinfo"
"github.com/pingcap/tidb/util/chunk"
)

// CreateBindExec represents a create bind executor.
type CreateBindExec struct {
baseExecutor

originSQL string
bindSQL string
defaultDB string
charset string
collation string
isGlobal bool
bindAst ast.StmtNode
}

// Next implements the Executor Next interface.
func (e *CreateBindExec) Next(ctx context.Context, req *chunk.RecordBatch) error {
req.Reset()
bm := bindinfo.GetBindManager(e.ctx)
if bm == nil {
return errors.New("bind manager is nil")
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
}
var err error
if e.isGlobal {
err = bm.AddGlobalBind(e.originSQL, e.bindSQL, e.defaultDB, e.charset, e.collation)
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
}
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
return err
}
20 changes: 20 additions & 0 deletions executor/builder.go
Original file line number Diff line number Diff line change
Expand Up @@ -180,6 +180,8 @@ func (b *executorBuilder) build(p plannercore.Plan) Executor {
return b.buildIndexLookUpReader(v)
case *plannercore.PhysicalWindow:
return b.buildWindow(v)
case *plannercore.CreateBindPlan:
return b.buildCreateBind(v)
default:
if mp, ok := p.(MockPhysicalPlan); ok {
return mp.GetExecutor()
Expand Down Expand Up @@ -1974,6 +1976,24 @@ func (b *executorBuilder) buildWindow(v *plannercore.PhysicalWindow) *WindowExec
}
}

func (b *executorBuilder) buildCreateBind(v *plannercore.CreateBindPlan) Executor {
base := newBaseExecutor(b.ctx, v.Schema(), v.ExplainID())
base.initCap = chunk.ZeroCapacity

charset, collation := b.ctx.GetSessionVars().GetCharsetInfo()
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
e := &CreateBindExec{
baseExecutor: base,
originSQL: v.OriginSQL,
bindSQL: v.BindSQL,
defaultDB: v.DefaultDB,
charset: charset,
collation: collation,
isGlobal: v.IsGlobal,
bindAst: v.BindStmt,
}
return e
}

func getPhysicalTableID(t table.Table) int64 {
if p, ok := t.(table.PhysicalTable); ok {
return p.GetPhysicalID()
Expand Down
48 changes: 48 additions & 0 deletions executor/compiler.go
Original file line number Diff line number Diff line change
Expand Up @@ -42,6 +42,15 @@ func (c *Compiler) Compile(ctx context.Context, stmtNode ast.StmtNode) (*ExecStm
defer span1.Finish()
}

var needDefaultDb bool
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
switch x := stmtNode.(type) {
case *ast.CreateBindingStmt:
needDefaultDb = checkDefaultDb(x.OriginSel)
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
if !needDefaultDb {
needDefaultDb = checkDefaultDb(x.HintedSel)
}
}

infoSchema := GetInfoSchema(c.Ctx)
if err := plannercore.Preprocess(c.Ctx, stmtNode, infoSchema); err != nil {
return nil, errors.Trace(err)
Expand All @@ -52,6 +61,20 @@ func (c *Compiler) Compile(ctx context.Context, stmtNode ast.StmtNode) (*ExecStm
return nil, errors.Trace(err)
}

if needDefaultDb {
switch x := finalPlan.(type) {
case *plannercore.CreateBindPlan:
if c.Ctx.GetSessionVars().CurrentDB != "" {
x.DefaultDB = c.Ctx.GetSessionVars().CurrentDB
} else {
err = errors.Trace(plannercore.ErrNoDB)
}
}
if err != nil {
return nil, errors.Trace(err)
alivxxx marked this conversation as resolved.
Show resolved Hide resolved
}
}

CountStmtNode(stmtNode, c.Ctx.GetSessionVars().InRestrictedSQL)
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
isExpensive := logExpensiveQuery(stmtNode, finalPlan)

Expand Down Expand Up @@ -314,3 +337,28 @@ func GetInfoSchema(ctx sessionctx.Context) infoschema.InfoSchema {
}
return is
}

func checkDefaultDb(stmtNode ast.ResultSetNode) bool {
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
switch x := stmtNode.(type) {
case *ast.TableSource:
return checkDefaultDb(x.Source)
case *ast.SelectStmt:
return checkDefaultDb(x.From.TableRefs)
case *ast.TableName:
if x.Schema.O == "" {
return true
}
case *ast.Join:
var need bool
if x.Left != nil {
need = checkDefaultDb(x.Left)
if !need {
if x.Right != nil {
return checkDefaultDb(x.Right)
}
}
}
return need
}
return false
}
11 changes: 11 additions & 0 deletions planner/core/common_plans.go
Original file line number Diff line number Diff line change
Expand Up @@ -340,6 +340,17 @@ type Set struct {
VarAssigns []*expression.VarAssignment
}

// CreateBindPlan represents a plan for createBind stmt.
type CreateBindPlan struct {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it's better to use only one sql operator to support all the operations on the sql-bind table. how about change this operator name to SQLBindPlan, and use an operation type to specify the operation this operator need to perform? for example, for operation to create a sql bind, we can use OpSQLBindCreate, for operation to delete a sql bind, we can use OpSQLBindDelete.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

But we have 2 ast, so I think it's better to split them.

baseSchemaProducer

OriginSQL string
BindSQL string
DefaultDB string
IsGlobal bool
BindStmt ast.StmtNode
}

// Simple represents a simple statement plan which doesn't need any optimization.
type Simple struct {
baseSchemaProducer
Expand Down
13 changes: 13 additions & 0 deletions planner/core/planbuilder.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ import (

"github.com/cznic/mathutil"
"github.com/pingcap/errors"
"github.com/pingcap/parser"
"github.com/pingcap/parser/ast"
"github.com/pingcap/parser/charset"
"github.com/pingcap/parser/model"
Expand Down Expand Up @@ -217,6 +218,8 @@ func (b *PlanBuilder) Build(node ast.Node) (Plan, error) {
return b.buildSimple(node.(ast.StmtNode))
case ast.DDLNode:
return b.buildDDL(x)
case *ast.CreateBindingStmt:
return b.buildCreateBind(x)
case *ast.ChangeStmt:
return b.buildChange(x)
}
Expand Down Expand Up @@ -306,6 +309,16 @@ func (b *PlanBuilder) buildSet(v *ast.SetStmt) (Plan, error) {
return p, nil
}

func (b *PlanBuilder) buildCreateBind(v *ast.CreateBindingStmt) (Plan, error) {
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
p := &CreateBindPlan{
OriginSQL: parser.Normalize(v.OriginSel.Text()),
BindSQL: v.HintedSel.Text(),
IsGlobal: v.GlobalScope,
BindStmt: v.HintedSel,
}
return p, nil
}

// detectSelectAgg detects an aggregate function or GROUP BY clause.
func (b *PlanBuilder) detectSelectAgg(sel *ast.SelectStmt) bool {
if sel.GroupBy != nil {
Expand Down
11 changes: 11 additions & 0 deletions planner/core/preprocess.go
Original file line number Diff line number Diff line change
Expand Up @@ -116,12 +116,23 @@ func (p *preprocessor) Enter(in ast.Node) (out ast.Node, skipChildren bool) {
// So skip check table name here, otherwise, admin restore table [table_name] syntax will return
// table not exists error. But admin restore is use to restore the dropped table. So skip children here.
return in, node.Tp == ast.AdminRestoreTable
case *ast.CreateBindingStmt:
p.checkBindGrammar(node)
default:
p.flag &= ^parentIsJoin
}
return in, p.err != nil
}

func (p *preprocessor) checkBindGrammar(createBindingStmt *ast.CreateBindingStmt) {
originSQL := parser.Normalize(createBindingStmt.OriginSel.(*ast.SelectStmt).Text())
hintedSQL := parser.Normalize(createBindingStmt.HintedSel.(*ast.SelectStmt).Text())

if originSQL != hintedSQL {
p.err = errors.New("hinted sql and origin sql don't match when hinted sql erase the hint info")
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
}
}

func (p *preprocessor) Leave(in ast.Node) (out ast.Node, ok bool) {
switch x := in.(type) {
case *ast.CreateTableStmt:
Expand Down
10 changes: 10 additions & 0 deletions session/bootstrap.go
Original file line number Diff line number Diff line change
Expand Up @@ -225,6 +225,7 @@ const (
update_time timestamp NOT NULL,
charset text NOT NULL,
collation text NOT NULL,
INDEX primary_index(original_sql(1024),default_db(1024)) COMMENT "accelerate the speed when add global binding query",
XuHuaiyu marked this conversation as resolved.
Show resolved Hide resolved
INDEX time_index(update_time) COMMENT "accelerate the speed when querying with last update time"
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_bin;`

Expand Down Expand Up @@ -304,6 +305,7 @@ const (
version26 = 26
version27 = 27
version28 = 28
version29 = 29
)

func checkBootstrapped(s Session) (bool, error) {
Expand Down Expand Up @@ -475,6 +477,10 @@ func upgrade(s Session) {
upgradeToVer28(s)
}

if ver < version29 {
upgradeToVer29(s)
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
}

updateBootstrapVer(s)
_, err = s.Execute(context.Background(), "COMMIT")

Expand Down Expand Up @@ -757,6 +763,10 @@ func upgradeToVer28(s Session) {
doReentrantDDL(s, CreateBindInfoTable)
}

func upgradeToVer29(s Session) {
doReentrantDDL(s, "ALTER TABLE mysql.bind_info ADD INDEX primary_index (original_sql, default_db)", ddl.ErrDupKeyName)
iamzhoug37 marked this conversation as resolved.
Show resolved Hide resolved
}

// updateBootstrapVer updates bootstrap version variable in mysql.TiDB table.
func updateBootstrapVer(s Session) {
// Update bootstrap version.
Expand Down
Loading