Files
tidb/pkg/table/tblsession/table.go

192 lines
6.7 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 tblsession
import (
"github.com/pingcap/tidb/pkg/expression/exprctx"
infoschema "github.com/pingcap/tidb/pkg/infoschema/context"
"github.com/pingcap/tidb/pkg/meta/autoid"
"github.com/pingcap/tidb/pkg/meta/model"
"github.com/pingcap/tidb/pkg/sessionctx"
"github.com/pingcap/tidb/pkg/sessionctx/stmtctx"
"github.com/pingcap/tidb/pkg/sessionctx/variable"
"github.com/pingcap/tidb/pkg/table/tblctx"
"github.com/pingcap/tidb/pkg/util/intest"
)
var _ tblctx.MutateContext = &MutateContext{}
var _ tblctx.AllocatorContext = &MutateContext{}
// MutateContext is used to provide context for table operations.
type MutateContext struct {
sessionctx.Context
// mutateBuffers is a memory pool for table related memory allocation that aims to reuse memory
// and saves allocation
// The buffers are supposed to be used inside AddRecord/UpdateRecord/RemoveRecord.
mutateBuffers *tblctx.MutateBuffers
}
// NewMutateContext creates a new MutateContext.
func NewMutateContext(sctx sessionctx.Context) *MutateContext {
return &MutateContext{
Context: sctx,
mutateBuffers: tblctx.NewMutateBuffers(sctx.GetSessionVars().GetWriteStmtBufs()),
}
}
// AlternativeAllocators implements the AllocatorContext interface
func (ctx *MutateContext) AlternativeAllocators(tbl *model.TableInfo) (allocators autoid.Allocators, ok bool) {
// Use an independent allocator for global temporary tables.
if tbl.TempTableType == model.TempTableGlobal {
if tempTbl := ctx.vars().GetTemporaryTable(tbl); tempTbl != nil {
if alloc := tempTbl.GetAutoIDAllocator(); alloc != nil {
return autoid.NewAllocators(false, alloc), true
}
}
// If the session is not in a txn, for example, in "show create table", use the original allocator.
}
return
}
// GetExprCtx returns the ExprContext
func (ctx *MutateContext) GetExprCtx() exprctx.ExprContext {
return ctx.Context.GetExprCtx()
}
// ConnectionID implements the MutateContext interface.
func (ctx *MutateContext) ConnectionID() uint64 {
return ctx.vars().ConnectionID
}
// InRestrictedSQL returns whether the current context is used in restricted SQL.
func (ctx *MutateContext) InRestrictedSQL() bool {
return ctx.vars().InRestrictedSQL
}
// TxnAssertionLevel implements the MutateContext interface.
func (ctx *MutateContext) TxnAssertionLevel() variable.AssertionLevel {
return ctx.vars().AssertionLevel
}
// EnableMutationChecker implements the MutateContext interface.
func (ctx *MutateContext) EnableMutationChecker() bool {
return ctx.vars().EnableMutationChecker
}
// GetRowEncodingConfig returns the RowEncodingConfig.
func (ctx *MutateContext) GetRowEncodingConfig() tblctx.RowEncodingConfig {
vars := ctx.vars()
return tblctx.RowEncodingConfig{
IsRowLevelChecksumEnabled: vars.IsRowLevelChecksumEnabled(),
RowEncoder: &vars.RowEncoder,
}
}
// GetMutateBuffers implements the MutateContext interface.
func (ctx *MutateContext) GetMutateBuffers() *tblctx.MutateBuffers {
return ctx.mutateBuffers
}
// GetRowIDShardGenerator implements the MutateContext interface.
func (ctx *MutateContext) GetRowIDShardGenerator() *variable.RowIDShardGenerator {
return ctx.vars().GetRowIDShardGenerator()
}
// GetReservedRowIDAlloc implements the MutateContext interface.
func (ctx *MutateContext) GetReservedRowIDAlloc() (*stmtctx.ReservedRowIDAlloc, bool) {
if sc := ctx.vars().StmtCtx; sc != nil {
return &sc.ReservedRowIDAlloc, true
}
// `StmtCtx` should not be nil in the `variable.SessionVars`.
// We just put an assertion that will panic only if in test here.
// In production code, here returns (nil, false) to make code safe
// because some old code checks `StmtCtx != nil` but we don't know why.
intest.Assert(false, "SessionVars.StmtCtx should not be nil")
return nil, false
}
// GetStatisticsSupport implements the MutateContext interface.
func (ctx *MutateContext) GetStatisticsSupport() (tblctx.StatisticsSupport, bool) {
if ctx.vars().TxnCtx != nil {
return ctx, true
}
return nil, false
}
// UpdatePhysicalTableDelta implements the StatisticsSupport interface.
func (ctx *MutateContext) UpdatePhysicalTableDelta(
physicalTableID int64, delta int64, count int64,
) {
if txnCtx := ctx.vars().TxnCtx; txnCtx != nil {
txnCtx.UpdateDeltaForTable(physicalTableID, delta, count)
}
}
// GetCachedTableSupport implements the MutateContext interface.
func (ctx *MutateContext) GetCachedTableSupport() (tblctx.CachedTableSupport, bool) {
if ctx.vars().TxnCtx != nil {
return ctx, true
}
return nil, false
}
// AddCachedTableHandleToTxn implements `CachedTableSupport` interface
func (ctx *MutateContext) AddCachedTableHandleToTxn(tableID int64, handle any) {
txnCtx := ctx.vars().TxnCtx
if txnCtx.CachedTables == nil {
txnCtx.CachedTables = make(map[int64]any)
}
if _, ok := txnCtx.CachedTables[tableID]; !ok {
txnCtx.CachedTables[tableID] = handle
}
}
// GetTemporaryTableSupport implements the MutateContext interface.
func (ctx *MutateContext) GetTemporaryTableSupport() (tblctx.TemporaryTableSupport, bool) {
if ctx.vars().TxnCtx == nil {
return nil, false
}
return ctx, true
}
// GetInfoSchemaToCheckExchangeConstraint implements the ExchangePartitionDMLSupport interface.
func (ctx *MutateContext) GetInfoSchemaToCheckExchangeConstraint() infoschema.MetaOnlyInfoSchema {
return ctx.Context.GetLatestInfoSchema()
}
// GetExchangePartitionDMLSupport implements the MutateContext interface.
func (ctx *MutateContext) GetExchangePartitionDMLSupport() (tblctx.ExchangePartitionDMLSupport, bool) {
return ctx, true
}
// GetTemporaryTableSizeLimit implements TemporaryTableSupport interface.
func (ctx *MutateContext) GetTemporaryTableSizeLimit() int64 {
return ctx.vars().TMPTableSize
}
// AddTemporaryTableToTxn implements the TemporaryTableSupport interface.
func (ctx *MutateContext) AddTemporaryTableToTxn(tblInfo *model.TableInfo) (tblctx.TemporaryTableHandler, bool) {
vars := ctx.vars()
if tbl := vars.GetTemporaryTable(tblInfo); tbl != nil {
tbl.SetModified(true)
return tblctx.NewTemporaryTableHandler(tbl, vars.TemporaryTableData), true
}
return tblctx.TemporaryTableHandler{}, false
}
func (ctx *MutateContext) vars() *variable.SessionVars {
return ctx.Context.GetSessionVars()
}