173 lines
5.4 KiB
Go
173 lines
5.4 KiB
Go
// Copyright 2024 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
|
|
|
|
import (
|
|
"github.com/pingcap/tidb/pkg/executor/internal/exec"
|
|
"github.com/pingcap/tidb/pkg/expression"
|
|
"github.com/pingcap/tidb/pkg/expression/contextsession"
|
|
)
|
|
|
|
// Detach detaches the current executor from the session context. After detaching, the session context
|
|
// can be used to execute another statement while this executor is still running. The returning value
|
|
// shows whether this executor is able to be detached.
|
|
//
|
|
// NOTE: the implementation of `Detach` should guarantee that no matter whether it returns true or false,
|
|
// both the original executor and the returning executor should be able to be used correctly. This restriction
|
|
// is to make sure that if `Detach(a)` returns `true`, while other children of `a`'s parent returns `false`,
|
|
// the caller can still use the original one.
|
|
func Detach(originalExecutor exec.Executor) (exec.Executor, bool) {
|
|
newExecutor, ok := originalExecutor.Detach()
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
|
|
children := originalExecutor.AllChildren()
|
|
newChildren := make([]exec.Executor, len(children))
|
|
for i, child := range children {
|
|
detached, ok := Detach(child)
|
|
if !ok {
|
|
return nil, false
|
|
}
|
|
newChildren[i] = detached
|
|
}
|
|
newExecutor.SetAllChildren(newChildren)
|
|
|
|
return newExecutor, true
|
|
}
|
|
|
|
func (treCtx tableReaderExecutorContext) Detach() tableReaderExecutorContext {
|
|
newCtx := treCtx
|
|
|
|
if ctx, ok := treCtx.ectx.(*contextsession.SessionExprContext); ok {
|
|
staticExprCtx := ctx.IntoStatic()
|
|
|
|
newCtx.dctx = newCtx.dctx.Detach()
|
|
newCtx.rctx = newCtx.rctx.Detach(staticExprCtx)
|
|
newCtx.buildPBCtx = newCtx.buildPBCtx.Detach(staticExprCtx)
|
|
newCtx.ectx = staticExprCtx
|
|
return newCtx
|
|
}
|
|
|
|
return treCtx
|
|
}
|
|
|
|
func (ireCtx indexReaderExecutorContext) Detach() indexReaderExecutorContext {
|
|
newCtx := ireCtx
|
|
|
|
if ctx, ok := ireCtx.ectx.(*contextsession.SessionExprContext); ok {
|
|
staticExprCtx := ctx.IntoStatic()
|
|
|
|
newCtx.dctx = newCtx.dctx.Detach()
|
|
newCtx.rctx = newCtx.rctx.Detach(staticExprCtx)
|
|
newCtx.buildPBCtx = newCtx.buildPBCtx.Detach(staticExprCtx)
|
|
newCtx.ectx = staticExprCtx
|
|
return newCtx
|
|
}
|
|
|
|
return ireCtx
|
|
}
|
|
|
|
func (iluCtx indexLookUpExecutorContext) Detach() indexLookUpExecutorContext {
|
|
newCtx := iluCtx
|
|
newCtx.tableReaderExecutorContext = newCtx.tableReaderExecutorContext.Detach()
|
|
|
|
return iluCtx
|
|
}
|
|
|
|
func (pCtx projectionExecutorContext) Detach() projectionExecutorContext {
|
|
newCtx := pCtx
|
|
if ctx, ok := pCtx.evalCtx.(*contextsession.SessionEvalContext); ok {
|
|
newCtx.evalCtx = ctx.IntoStatic()
|
|
}
|
|
|
|
return newCtx
|
|
}
|
|
|
|
func (sCtx selectionExecutorContext) Detach() selectionExecutorContext {
|
|
newCtx := sCtx
|
|
if ctx, ok := sCtx.evalCtx.(*contextsession.SessionEvalContext); ok {
|
|
newCtx.evalCtx = ctx.IntoStatic()
|
|
}
|
|
|
|
return newCtx
|
|
}
|
|
|
|
// Detach detaches the current executor from the session context.
|
|
func (e *TableReaderExecutor) Detach() (exec.Executor, bool) {
|
|
newExec := new(TableReaderExecutor)
|
|
*newExec = *e
|
|
|
|
newExec.tableReaderExecutorContext = newExec.tableReaderExecutorContext.Detach()
|
|
|
|
return newExec, true
|
|
}
|
|
|
|
// Detach detaches the current executor from the session context.
|
|
func (e *IndexReaderExecutor) Detach() (exec.Executor, bool) {
|
|
newExec := new(IndexReaderExecutor)
|
|
*newExec = *e
|
|
|
|
newExec.indexReaderExecutorContext = newExec.indexReaderExecutorContext.Detach()
|
|
|
|
return newExec, true
|
|
}
|
|
|
|
// Detach detaches the current executor from the session context.
|
|
func (e *IndexLookUpExecutor) Detach() (exec.Executor, bool) {
|
|
newExec := new(IndexLookUpExecutor)
|
|
*newExec = *e
|
|
|
|
newExec.indexLookUpExecutorContext = newExec.indexLookUpExecutorContext.Detach()
|
|
|
|
return newExec, true
|
|
}
|
|
|
|
// Detach detaches the current executor from the session context.
|
|
func (e *ProjectionExec) Detach() (exec.Executor, bool) {
|
|
// check whether the `Projection` requires any optional property
|
|
// Now, no optional property is copied, so if it requires any optional property, it should return false.
|
|
// TODO: some optional property can be detached. If they are implemented in the future, this check needs to be changed.
|
|
if !e.evaluatorSuit.RequiredOptionalEvalProps().IsEmpty() {
|
|
return nil, false
|
|
}
|
|
|
|
newExec := new(ProjectionExec)
|
|
*newExec = *e
|
|
|
|
newExec.projectionExecutorContext = newExec.projectionExecutorContext.Detach()
|
|
|
|
return newExec, true
|
|
}
|
|
|
|
// Detach detaches the current executor from the session context.
|
|
func (e *SelectionExec) Detach() (exec.Executor, bool) {
|
|
// check whether the `Selection` requires any optional property
|
|
// Now, no optional property is copied, so if it requires any optional property, it should return false.
|
|
// TODO: some optional property can be detached. If they are implemented in the future, this check needs to be changed.
|
|
for _, expr := range e.filters {
|
|
if !expression.GetOptionalEvalPropsForExpr(expr).IsEmpty() {
|
|
return nil, false
|
|
}
|
|
}
|
|
|
|
newExec := new(SelectionExec)
|
|
*newExec = *e
|
|
|
|
newExec.selectionExecutorContext = newExec.selectionExecutorContext.Detach()
|
|
|
|
return newExec, true
|
|
}
|