Decide whether a executor support streaming, TableScan/IndexScan/Selection support it. If a coprocessor request need some operations like aggregation or topn, it's meanless to use streaming for the request.
214 lines
6.7 KiB
Go
214 lines
6.7 KiB
Go
// 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 distsql
|
|
|
|
import (
|
|
"math"
|
|
|
|
"github.com/juju/errors"
|
|
"github.com/pingcap/tidb/ast"
|
|
"github.com/pingcap/tidb/kv"
|
|
"github.com/pingcap/tidb/sessionctx/stmtctx"
|
|
"github.com/pingcap/tidb/sessionctx/variable"
|
|
"github.com/pingcap/tidb/tablecodec"
|
|
"github.com/pingcap/tidb/util/codec"
|
|
"github.com/pingcap/tidb/util/ranger"
|
|
"github.com/pingcap/tipb/go-tipb"
|
|
)
|
|
|
|
// RequestBuilder is used to build a "kv.Request".
|
|
// It is called before we issue a kv request by "Select".
|
|
type RequestBuilder struct {
|
|
kv.Request
|
|
err error
|
|
}
|
|
|
|
// Build builds a "kv.Request".
|
|
func (builder *RequestBuilder) Build() (*kv.Request, error) {
|
|
return &builder.Request, errors.Trace(builder.err)
|
|
}
|
|
|
|
// SetTableRanges sets "KeyRanges" for "kv.Request" by converting "tableRanges"
|
|
// to "KeyRanges" firstly.
|
|
func (builder *RequestBuilder) SetTableRanges(tid int64, tableRanges []*ranger.NewRange) *RequestBuilder {
|
|
if builder.err != nil {
|
|
return builder
|
|
}
|
|
builder.Request.KeyRanges = TableRangesToKVRanges(tid, tableRanges)
|
|
return builder
|
|
}
|
|
|
|
// SetIndexRanges sets "KeyRanges" for "kv.Request" by converting index range
|
|
// "ranges" to "KeyRanges" firstly.
|
|
func (builder *RequestBuilder) SetIndexRanges(sc *stmtctx.StatementContext, tid, idxID int64, ranges []*ranger.NewRange) *RequestBuilder {
|
|
if builder.err != nil {
|
|
return builder
|
|
}
|
|
builder.Request.KeyRanges, builder.err = IndexRangesToKVRanges(sc, tid, idxID, ranges)
|
|
return builder
|
|
}
|
|
|
|
// SetTableHandles sets "KeyRanges" for "kv.Request" by converting table handles
|
|
// "handles" to "KeyRanges" firstly.
|
|
func (builder *RequestBuilder) SetTableHandles(tid int64, handles []int64) *RequestBuilder {
|
|
builder.Request.KeyRanges = TableHandlesToKVRanges(tid, handles)
|
|
return builder
|
|
}
|
|
|
|
// SetDAGRequest sets the request type to "ReqTypeDAG" and cosntruct request data.
|
|
func (builder *RequestBuilder) SetDAGRequest(dag *tipb.DAGRequest) *RequestBuilder {
|
|
if builder.err != nil {
|
|
return builder
|
|
}
|
|
|
|
builder.Request.Tp = kv.ReqTypeDAG
|
|
builder.Request.StartTs = dag.StartTs
|
|
builder.Request.Data, builder.err = dag.Marshal()
|
|
return builder
|
|
}
|
|
|
|
// SetAnalyzeRequest sets the request type to "ReqTypeAnalyze" and cosntruct request data.
|
|
func (builder *RequestBuilder) SetAnalyzeRequest(ana *tipb.AnalyzeReq) *RequestBuilder {
|
|
if builder.err != nil {
|
|
return builder
|
|
}
|
|
|
|
builder.Request.Tp = kv.ReqTypeAnalyze
|
|
builder.Request.StartTs = ana.StartTs
|
|
builder.Request.Data, builder.err = ana.Marshal()
|
|
builder.Request.NotFillCache = true
|
|
return builder
|
|
}
|
|
|
|
// SetKeyRanges sets "KeyRanges" for "kv.Request".
|
|
func (builder *RequestBuilder) SetKeyRanges(keyRanges []kv.KeyRange) *RequestBuilder {
|
|
builder.Request.KeyRanges = keyRanges
|
|
return builder
|
|
}
|
|
|
|
// SetDesc sets "Desc" for "kv.Request".
|
|
func (builder *RequestBuilder) SetDesc(desc bool) *RequestBuilder {
|
|
builder.Request.Desc = desc
|
|
return builder
|
|
}
|
|
|
|
// SetKeepOrder sets "KeepOrder" for "kv.Request".
|
|
func (builder *RequestBuilder) SetKeepOrder(order bool) *RequestBuilder {
|
|
builder.Request.KeepOrder = order
|
|
return builder
|
|
}
|
|
|
|
func (builder *RequestBuilder) getIsolationLevel(sv *variable.SessionVars) kv.IsoLevel {
|
|
isoLevel, _ := sv.GetSystemVar(variable.TxnIsolation)
|
|
if isoLevel == ast.ReadCommitted {
|
|
return kv.RC
|
|
}
|
|
return kv.SI
|
|
}
|
|
|
|
// SetFromSessionVars sets the following fields for "kv.Request" from session variables:
|
|
// "Concurrency", "IsolationLevel", "NotFillCache".
|
|
func (builder *RequestBuilder) SetFromSessionVars(sv *variable.SessionVars) *RequestBuilder {
|
|
builder.Request.Concurrency = sv.DistSQLScanConcurrency
|
|
builder.Request.IsolationLevel = builder.getIsolationLevel(sv)
|
|
builder.Request.NotFillCache = sv.StmtCtx.NotFillCache
|
|
return builder
|
|
}
|
|
|
|
// SetPriority sets "Priority" for "kv.Request".
|
|
func (builder *RequestBuilder) SetPriority(priority int) *RequestBuilder {
|
|
builder.Request.Priority = priority
|
|
return builder
|
|
}
|
|
|
|
// SetStreaming sets "Streaming" flag for "kv.Request".
|
|
func (builder *RequestBuilder) SetStreaming(streaming bool) *RequestBuilder {
|
|
builder.Request.Streaming = streaming
|
|
return builder
|
|
}
|
|
|
|
// TableRangesToKVRanges converts table ranges to "KeyRange".
|
|
func TableRangesToKVRanges(tid int64, ranges []*ranger.NewRange) []kv.KeyRange {
|
|
krs := make([]kv.KeyRange, 0, len(ranges))
|
|
for _, ran := range ranges {
|
|
var low, high []byte
|
|
low = codec.EncodeInt(nil, ran.LowVal[0].GetInt64())
|
|
high = codec.EncodeInt(nil, ran.HighVal[0].GetInt64())
|
|
if ran.LowExclude {
|
|
low = []byte(kv.Key(low).PrefixNext())
|
|
}
|
|
if !ran.HighExclude {
|
|
high = []byte(kv.Key(high).PrefixNext())
|
|
}
|
|
startKey := tablecodec.EncodeRowKey(tid, low)
|
|
endKey := tablecodec.EncodeRowKey(tid, high)
|
|
krs = append(krs, kv.KeyRange{StartKey: startKey, EndKey: endKey})
|
|
}
|
|
return krs
|
|
}
|
|
|
|
// TableHandlesToKVRanges converts sorted handle to kv ranges.
|
|
// For continuous handles, we should merge them to a single key range.
|
|
func TableHandlesToKVRanges(tid int64, handles []int64) []kv.KeyRange {
|
|
krs := make([]kv.KeyRange, 0, len(handles))
|
|
i := 0
|
|
for i < len(handles) {
|
|
h := handles[i]
|
|
if h == math.MaxInt64 {
|
|
// We can't convert MaxInt64 into an left closed, right open range.
|
|
i++
|
|
continue
|
|
}
|
|
j := i + 1
|
|
endHandle := h + 1
|
|
for ; j < len(handles); j++ {
|
|
if handles[j] == endHandle {
|
|
endHandle = handles[j] + 1
|
|
continue
|
|
}
|
|
break
|
|
}
|
|
startKey := tablecodec.EncodeRowKeyWithHandle(tid, h)
|
|
endKey := tablecodec.EncodeRowKeyWithHandle(tid, endHandle)
|
|
krs = append(krs, kv.KeyRange{StartKey: startKey, EndKey: endKey})
|
|
i = j
|
|
}
|
|
return krs
|
|
}
|
|
|
|
// IndexRangesToKVRanges converts index ranges to "KeyRange".
|
|
func IndexRangesToKVRanges(sc *stmtctx.StatementContext, tid, idxID int64, ranges []*ranger.NewRange) ([]kv.KeyRange, error) {
|
|
krs := make([]kv.KeyRange, 0, len(ranges))
|
|
for _, ran := range ranges {
|
|
low, err := codec.EncodeKey(sc, nil, ran.LowVal...)
|
|
if err != nil {
|
|
return nil, errors.Trace(err)
|
|
}
|
|
if ran.LowExclude {
|
|
low = []byte(kv.Key(low).PrefixNext())
|
|
}
|
|
high, err := codec.EncodeKey(sc, nil, ran.HighVal...)
|
|
if err != nil {
|
|
return nil, errors.Trace(err)
|
|
}
|
|
if !ran.HighExclude {
|
|
high = []byte(kv.Key(high).PrefixNext())
|
|
}
|
|
startKey := tablecodec.EncodeIndexSeekKey(tid, idxID, low)
|
|
endKey := tablecodec.EncodeIndexSeekKey(tid, idxID, high)
|
|
krs = append(krs, kv.KeyRange{StartKey: startKey, EndKey: endKey})
|
|
}
|
|
return krs, nil
|
|
}
|