1 В избранное 0 Ответвления 0

OSCHINA-MIRROR/hanchuanchuan-goInception

Присоединиться к Gitlife
Откройте для себя и примите участие в публичных проектах с открытым исходным кодом с участием более 10 миллионов разработчиков. Приватные репозитории также полностью бесплатны :)
Присоединиться бесплатно
Клонировать/Скачать
point_get_plan.go 13 КБ
Копировать Редактировать Исходные данные Просмотреть построчно История
hanchuanchuan Отправлено 6 лет назад 27f3c5a
// 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
import (
"bytes"
"fmt"
"github.com/hanchuanchuan/goInception/ast"
"github.com/hanchuanchuan/goInception/expression"
"github.com/hanchuanchuan/goInception/model"
"github.com/hanchuanchuan/goInception/mysql"
"github.com/hanchuanchuan/goInception/parser/opcode"
"github.com/hanchuanchuan/goInception/planner/property"
"github.com/hanchuanchuan/goInception/privilege"
"github.com/hanchuanchuan/goInception/sessionctx"
"github.com/hanchuanchuan/goInception/types"
"github.com/pingcap/errors"
"github.com/pingcap/tipb/go-tipb"
)
// PointGetPlan is a fast plan for simple point get.
// When we detect that the statement has a unique equal access condition, this plan is used.
// This plan is much faster to build and to execute because it avoid the optimization and coprocessor cost.
type PointGetPlan struct {
basePlan
schema *expression.Schema
TblInfo *model.TableInfo
IndexInfo *model.IndexInfo
Handle int64
IndexValues []types.Datum
expr expression.Expression
ctx sessionctx.Context
}
type nameValuePair struct {
colName string
value types.Datum
}
// Schema implements the Plan interface.
func (p *PointGetPlan) Schema() *expression.Schema {
return p.schema
}
// attach2Task makes the current physical plan as the father of task's physicalPlan and updates the cost of
// current task. If the child's task is cop task, some operator may close this task and return a new rootTask.
func (p *PointGetPlan) attach2Task(...task) task {
return nil
}
// ToPB converts physical plan to tipb executor.
func (p *PointGetPlan) ToPB(ctx sessionctx.Context) (*tipb.Executor, error) {
return nil, nil
}
// ExplainInfo returns operator information to be explained.
func (p *PointGetPlan) ExplainInfo() string {
buffer := bytes.NewBufferString("")
tblName := p.TblInfo.Name.O
fmt.Fprintf(buffer, "table:%s", tblName)
if p.IndexInfo != nil {
fmt.Fprintf(buffer, ", index:")
for i, col := range p.IndexInfo.Columns {
buffer.WriteString(col.Name.O)
if i < len(p.IndexInfo.Columns)-1 {
buffer.WriteString(" ")
}
}
} else {
fmt.Fprintf(buffer, ", handle:%d", p.Handle)
}
return buffer.String()
}
// getChildReqProps gets the required property by child index.
func (p *PointGetPlan) getChildReqProps(idx int) *property.PhysicalProperty {
return nil
}
// StatsCount will return the the RowCount of property.StatsInfo for this plan.
func (p *PointGetPlan) StatsCount() float64 {
return 1
}
// StatsCount will return the the RowCount of property.StatsInfo for this plan.
func (p *PointGetPlan) statsInfo() *property.StatsInfo {
if p.stats == nil {
p.stats = &property.StatsInfo{}
}
p.stats.RowCount = 1
return p.stats
}
// Children gets all the children.
func (p *PointGetPlan) Children() []PhysicalPlan {
return nil
}
// SetChildren sets the children for the plan.
func (p *PointGetPlan) SetChildren(...PhysicalPlan) {}
// ResolveIndices resolves the indices for columns. After doing this, the columns can evaluate the rows by their indices.
func (p *PointGetPlan) ResolveIndices() {}
// TryFastPlan tries to use the PointGetPlan for the query.
func TryFastPlan(ctx sessionctx.Context, node ast.Node) Plan {
if PreparedPlanCacheEnabled() {
// Do not support plan cache.
return nil
}
switch x := node.(type) {
case *ast.SelectStmt:
fp := tryPointGetPlan(ctx, x)
if fp != nil {
if checkFastPlanPrivilege(ctx, fp, mysql.SelectPriv) != nil {
return nil
}
return fp
}
case *ast.UpdateStmt:
return tryUpdatePointPlan(ctx, x)
case *ast.DeleteStmt:
return tryDeletePointPlan(ctx, x)
}
return nil
}
// tryPointGetPlan determine if the SelectStmt can use a PointGetPlan.
// Returns nil if not applicable.
// To use the PointGetPlan the following rules must be satisfied:
// 1. For the limit clause, the count should at least 1 and the offset is 0.
// 2. It must be a single table select.
// 3. All the columns must be public and generated.
// 4. The condition is an access path that the range is a unique key.
func tryPointGetPlan(ctx sessionctx.Context, selStmt *ast.SelectStmt) *PointGetPlan {
if selStmt.Having != nil || selStmt.LockTp != ast.SelectLockNone {
return nil
} else if selStmt.Limit != nil {
sc := ctx.GetSessionVars().StmtCtx
count, offset, err := extractLimitCountOffset(sc, selStmt.Limit)
if err != nil || count == 0 || offset > 0 {
return nil
}
}
tblName := getSingleTableName(selStmt.From)
if tblName == nil {
return nil
}
tbl := tblName.TableInfo
if tbl == nil {
return nil
}
// Do not handle partitioned table.
// Table partition implementation translates LogicalPlan from `DataSource` to
// `Union -> DataSource` in the logical plan optimization pass, since PointGetPlan
// bypass the logical plan optimization, it can't support partitioned table.
if tbl.GetPartitionInfo() != nil {
return nil
}
for _, col := range tbl.Columns {
// Do not handle generated columns.
if col.IsGenerated() {
return nil
}
// Only handle tables that all columns are public.
if col.State != model.StatePublic {
return nil
}
}
pairs := make([]nameValuePair, 0, 4)
pairs = getNameValuePairs(pairs, selStmt.Where)
if pairs == nil {
return nil
}
handleDatum := findPKHandle(tbl, pairs)
if handleDatum.Kind() != types.KindNull {
if len(pairs) != 1 {
return nil
}
schema := buildSchemaFromFields(ctx, tblName.Schema, tbl, selStmt.Fields.Fields)
if schema == nil {
return nil
}
p := newPointGetPlan(ctx, schema, tbl)
var err error
p.Handle, err = handleDatum.ToInt64(ctx.GetSessionVars().StmtCtx)
if err != nil {
return nil
}
return p
}
for _, idxInfo := range tbl.Indices {
if !idxInfo.Unique {
continue
}
if idxInfo.State != model.StatePublic {
continue
}
idxValues := getIndexValues(idxInfo, pairs)
if idxValues == nil {
continue
}
schema := buildSchemaFromFields(ctx, tblName.Schema, tbl, selStmt.Fields.Fields)
if schema == nil {
return nil
}
p := newPointGetPlan(ctx, schema, tbl)
p.IndexInfo = idxInfo
p.IndexValues = idxValues
return p
}
return nil
}
func newPointGetPlan(ctx sessionctx.Context, schema *expression.Schema, tbl *model.TableInfo) *PointGetPlan {
p := &PointGetPlan{
basePlan: newBasePlan(ctx, "Point_Get"),
schema: schema,
TblInfo: tbl,
}
return p
}
func checkFastPlanPrivilege(ctx sessionctx.Context, fastPlan *PointGetPlan, checkTypes ...mysql.PrivilegeType) error {
pm := privilege.GetPrivilegeManager(ctx)
if pm == nil {
return nil
}
dbName := ctx.GetSessionVars().CurrentDB
for _, checkType := range checkTypes {
if !pm.RequestVerification(dbName, fastPlan.TblInfo.Name.L, "", checkType) {
return errors.New("privilege check fail")
}
}
return nil
}
func buildSchemaFromFields(ctx sessionctx.Context, dbName model.CIStr, tbl *model.TableInfo, fields []*ast.SelectField) *expression.Schema {
if dbName.L == "" {
dbName = model.NewCIStr(ctx.GetSessionVars().CurrentDB)
}
columns := make([]*expression.Column, 0, len(tbl.Columns)+1)
if len(fields) == 1 && fields[0].WildCard != nil {
for _, col := range tbl.Columns {
columns = append(columns, colInfoToColumn(dbName, tbl.Name, col.Name, col, len(columns)))
}
return expression.NewSchema(columns...)
}
if len(fields) > 0 {
for _, field := range fields {
colNameExpr, ok := field.Expr.(*ast.ColumnNameExpr)
if !ok {
return nil
}
col := findCol(tbl, colNameExpr.Name)
if col == nil {
return nil
}
asName := col.Name
if field.AsName.L != "" {
asName = field.AsName
}
columns = append(columns, colInfoToColumn(dbName, tbl.Name, asName, col, len(columns)))
}
return expression.NewSchema(columns...)
}
// fields len is 0 for update and delete.
var handleCol *expression.Column
for _, col := range tbl.Columns {
column := colInfoToColumn(dbName, tbl.Name, col.Name, col, len(columns))
if tbl.PKIsHandle && mysql.HasPriKeyFlag(col.Flag) {
handleCol = column
}
columns = append(columns, column)
}
if handleCol == nil {
handleCol = colInfoToColumn(dbName, tbl.Name, model.ExtraHandleName, model.NewExtraHandleColInfo(), len(columns))
columns = append(columns, handleCol)
}
schema := expression.NewSchema(columns...)
schema.TblID2Handle = make(map[int64][]*expression.Column)
schema.TblID2Handle[tbl.ID] = []*expression.Column{handleCol}
return schema
}
func getSingleTableName(tableRefs *ast.TableRefsClause) *ast.TableName {
if tableRefs == nil || tableRefs.TableRefs == nil || tableRefs.TableRefs.Right != nil {
return nil
}
tblSrc, ok := tableRefs.TableRefs.Left.(*ast.TableSource)
if !ok {
return nil
}
if tblSrc.AsName.L != "" {
return nil
}
tblName, ok := tblSrc.Source.(*ast.TableName)
if !ok {
return nil
}
return tblName
}
// getNameValuePairs extracts `column = constant/paramMarker` conditions from expr as name value pairs.
func getNameValuePairs(nvPairs []nameValuePair, expr ast.ExprNode) []nameValuePair {
binOp, ok := expr.(*ast.BinaryOperationExpr)
if !ok {
return nil
}
if binOp.Op == opcode.LogicAnd {
nvPairs = getNameValuePairs(nvPairs, binOp.L)
if nvPairs == nil {
return nil
}
nvPairs = getNameValuePairs(nvPairs, binOp.R)
if nvPairs == nil {
return nil
}
return nvPairs
} else if binOp.Op == opcode.EQ {
colName, ok := binOp.L.(*ast.ColumnNameExpr)
if !ok {
return nil
}
var d types.Datum
switch x := binOp.R.(type) {
case *ast.ValueExpr:
d = x.Datum
case *ast.ParamMarkerExpr:
d = x.Datum
}
if d.IsNull() {
return nil
}
return append(nvPairs, nameValuePair{colName: colName.Name.Name.L, value: d})
}
return nil
}
func findPKHandle(tblInfo *model.TableInfo, pairs []nameValuePair) (d types.Datum) {
if !tblInfo.PKIsHandle {
return d
}
for _, col := range tblInfo.Columns {
if mysql.HasPriKeyFlag(col.Flag) {
i := findInPairs(col.Name.L, pairs)
if i == -1 {
return d
}
return pairs[i].value
}
}
return d
}
func getIndexValues(idxInfo *model.IndexInfo, pairs []nameValuePair) []types.Datum {
idxValues := make([]types.Datum, 0, 4)
if len(idxInfo.Columns) != len(pairs) {
return nil
}
if idxInfo.HasPrefixIndex() {
return nil
}
for _, idxCol := range idxInfo.Columns {
i := findInPairs(idxCol.Name.L, pairs)
if i == -1 {
return nil
}
idxValues = append(idxValues, pairs[i].value)
}
if len(idxValues) > 0 {
return idxValues
}
return nil
}
func findInPairs(colName string, pairs []nameValuePair) int {
for i, pair := range pairs {
if pair.colName == colName {
return i
}
}
return -1
}
func tryUpdatePointPlan(ctx sessionctx.Context, updateStmt *ast.UpdateStmt) Plan {
selStmt := &ast.SelectStmt{
Fields: &ast.FieldList{},
From: updateStmt.TableRefs,
Where: updateStmt.Where,
OrderBy: updateStmt.Order,
Limit: updateStmt.Limit,
}
fastSelect := tryPointGetPlan(ctx, selStmt)
if fastSelect == nil {
return nil
}
if checkFastPlanPrivilege(ctx, fastSelect, mysql.SelectPriv, mysql.UpdatePriv) != nil {
return nil
}
orderedList := buildOrderedList(ctx, fastSelect, updateStmt.List)
if orderedList == nil {
return nil
}
updatePlan := &Update{
SelectPlan: fastSelect,
OrderedList: orderedList,
}
updatePlan.SetSchema(fastSelect.schema)
return updatePlan
}
func buildOrderedList(ctx sessionctx.Context, fastSelect *PointGetPlan, list []*ast.Assignment) []*expression.Assignment {
orderedList := make([]*expression.Assignment, 0, len(list))
for _, assign := range list {
col, err := fastSelect.schema.FindColumn(assign.Column)
if err != nil {
return nil
}
if col == nil {
return nil
}
newAssign := &expression.Assignment{
Col: col,
}
expr, err := expression.RewriteSimpleExprWithSchema(ctx, assign.Expr, fastSelect.schema)
if err != nil {
return nil
}
expr = expression.BuildCastFunction(ctx, expr, col.GetType())
newAssign.Expr = expr.ResolveIndices(fastSelect.schema)
orderedList = append(orderedList, newAssign)
}
return orderedList
}
func tryDeletePointPlan(ctx sessionctx.Context, delStmt *ast.DeleteStmt) Plan {
if delStmt.IsMultiTable {
return nil
}
selStmt := &ast.SelectStmt{
Fields: &ast.FieldList{},
From: delStmt.TableRefs,
Where: delStmt.Where,
OrderBy: delStmt.Order,
Limit: delStmt.Limit,
}
fastSelect := tryPointGetPlan(ctx, selStmt)
if fastSelect == nil {
return nil
}
if checkFastPlanPrivilege(ctx, fastSelect, mysql.SelectPriv, mysql.DeletePriv) != nil {
return nil
}
delPlan := &Delete{
SelectPlan: fastSelect,
}
delPlan.SetSchema(fastSelect.schema)
return delPlan
}
func findCol(tbl *model.TableInfo, colName *ast.ColumnName) *model.ColumnInfo {
for _, col := range tbl.Columns {
if col.Name.L == colName.Name.L {
return col
}
}
return nil
}
func colInfoToColumn(db model.CIStr, tblName model.CIStr, asName model.CIStr, col *model.ColumnInfo, idx int) *expression.Column {
return &expression.Column{
ColName: asName,
OrigTblName: tblName,
DBName: db,
TblName: tblName,
RetType: &col.FieldType,
ID: col.ID,
UniqueID: int64(col.Offset),
Index: idx,
}
}

Опубликовать ( 0 )

Вы можете оставить комментарий после Вход в систему

1
https://gitlife.ru/oschina-mirror/hanchuanchuan-goInception.git
git@gitlife.ru:oschina-mirror/hanchuanchuan-goInception.git
oschina-mirror
hanchuanchuan-goInception
hanchuanchuan-goInception
v1.3.0