-
Notifications
You must be signed in to change notification settings - Fork 24
/
Copy pathplanner.go
121 lines (107 loc) · 3.37 KB
/
planner.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
package parplan
import (
"github.com/squareup/pranadb/errors"
"github.com/squareup/pranadb/tidb/planner"
"github.com/squareup/pranadb/tidb/sessionctx"
"github.com/squareup/pranadb/tidb/sessionctx/stmtctx"
"github.com/squareup/pranadb/sessctx"
"github.com/pingcap/parser/ast"
"github.com/squareup/pranadb/common"
"github.com/squareup/pranadb/tidb/infoschema"
)
type Planner struct {
pushQueryOptimizer *planner.Optimizer
pullQueryOptimizer *planner.Optimizer
parser *Parser
sessionCtx *sessctx.SessCtx
schema *common.Schema
is infoschema.InfoSchema
}
func NewPlanner(schema *common.Schema) *Planner {
is := schemaToInfoSchema(schema)
sessCtx := sessctx.NewSessionContext(is, schema.Name)
pl := &Planner{
pushQueryOptimizer: planner.NewPushQueryOptimizer(),
pullQueryOptimizer: planner.NewPullQueryOptimizer(),
parser: NewParser(),
sessionCtx: sessCtx,
schema: schema,
is: is,
}
return pl
}
func (p *Planner) StatementContext() *stmtctx.StatementContext {
return p.sessionCtx.GetSessionVars().StmtCtx
}
func (p *Planner) SessionContext() sessionctx.Context {
return p.sessionCtx
}
func (p *Planner) SetPSArgs(args []interface{}) {
p.sessionCtx.SetArgs(args)
}
func (p *Planner) Parse(query string) (AstHandle, int, error) {
return p.parser.Parse(query)
}
func (p *Planner) QueryToPlan(query string, prepare bool, pullQuery bool) (planner.PhysicalPlan, planner.LogicalPlan, int, error) {
phys, log, paramCount, err := p.doQueryToPlan(query, prepare, pullQuery)
if err != nil {
return nil, nil, 0, errors.MaybeConvertToPranaErrorf(err, errors.InvalidStatement, err.Error())
}
return phys, log, paramCount, nil
}
func (p *Planner) doQueryToPlan(query string, prepare bool, pullQuery bool) (planner.PhysicalPlan, planner.LogicalPlan, int, error) {
ast, paramCount, err := p.Parse(query)
if err != nil {
return nil, nil, 0, err
}
logic, err := p.BuildLogicalPlan(ast, prepare)
if err != nil {
return nil, nil, 0, err
}
physical, err := p.BuildPhysicalPlan(logic, pullQuery)
if err != nil {
return nil, nil, 0, err
}
return physical, logic, paramCount, nil
}
func (p *Planner) BuildLogicalPlan(stmt AstHandle, prepare bool) (planner.LogicalPlan, error) {
if err := p.preprocess(stmt.stmt, prepare); err != nil {
return nil, err
}
builder := planner.NewPlanBuilder(p.sessionCtx, p.is)
plan, err := builder.Build(stmt.stmt)
if err != nil {
return nil, errors.WithStack(err)
}
logicalPlan, isLogicalPlan := plan.(planner.LogicalPlan)
if !isLogicalPlan {
panic("Expected a logical plan")
}
return logicalPlan, nil
}
func (p *Planner) BuildPhysicalPlan(logicalPlan planner.LogicalPlan, pullQuery bool) (planner.PhysicalPlan, error) {
if !pullQuery {
physicalPlan, _, err := p.pushQueryOptimizer.FindBestPlan(p.sessionCtx, logicalPlan)
if err != nil {
return nil, errors.WithStack(err)
}
return physicalPlan, nil
}
physicalPlan, _, err := p.pullQueryOptimizer.FindBestPlan(p.sessionCtx, logicalPlan)
if err != nil {
return nil, errors.WithStack(err)
}
return physicalPlan, nil
}
func (p *Planner) preprocess(stmt ast.Node, prepare bool) error {
var err error
if prepare {
err = planner.Preprocess(p.sessionCtx, stmt, planner.InPrepare)
} else {
err = planner.Preprocess(p.sessionCtx, stmt)
}
if err != nil {
return errors.WithStack(err)
}
return nil
}