// Copyright 2016 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 tablecodec import ( "bytes" "encoding/binary" "math" "time" "github.com/pingcap/errors" "github.com/pingcap/parser/mysql" "github.com/pingcap/parser/terror" "github.com/pingcap/tidb/errno" "github.com/pingcap/tidb/kv" "github.com/pingcap/tidb/sessionctx/stmtctx" "github.com/pingcap/tidb/structure" "github.com/pingcap/tidb/types" "github.com/pingcap/tidb/util/codec" "github.com/pingcap/tidb/util/rowcodec" ) var ( errInvalidKey = terror.ClassXEval.New(errno.ErrInvalidKey, errno.MySQLErrName[errno.ErrInvalidKey]) errInvalidRecordKey = terror.ClassXEval.New(errno.ErrInvalidRecordKey, errno.MySQLErrName[errno.ErrInvalidRecordKey]) errInvalidIndexKey = terror.ClassXEval.New(errno.ErrInvalidIndexKey, errno.MySQLErrName[errno.ErrInvalidIndexKey]) ) var ( tablePrefix = []byte{'t'} recordPrefixSep = []byte("_r") indexPrefixSep = []byte("_i") metaPrefix = []byte{'m'} ) const ( idLen = 8 prefixLen = 1 + idLen /*tableID*/ + 2 // RecordRowKeyLen is public for calculating avgerage row size. RecordRowKeyLen = prefixLen + idLen /*handle*/ tablePrefixLength = 1 recordPrefixSepLength = 2 metaPrefixLength = 1 // MaxOldEncodeValueLen is the maximum len of the old encoding of index value. MaxOldEncodeValueLen = 9 // CommonHandleFlag is the flag used to decode the common handle in an unique index value. CommonHandleFlag byte = 127 ) // TableSplitKeyLen is the length of key 't{table_id}' which is used for table split. const TableSplitKeyLen = 1 + idLen // TablePrefix returns table's prefix 't'. func TablePrefix() []byte { return tablePrefix } // EncodeRowKey encodes the table id and record handle into a kv.Key func EncodeRowKey(tableID int64, encodedHandle []byte) kv.Key { buf := make([]byte, 0, prefixLen+len(encodedHandle)) buf = appendTableRecordPrefix(buf, tableID) buf = append(buf, encodedHandle...) return buf } // EncodeRowKeyWithHandle encodes the table id, row handle into a kv.Key func EncodeRowKeyWithHandle(tableID int64, handle kv.Handle) kv.Key { return EncodeRowKey(tableID, handle.Encoded()) } // CutRowKeyPrefix cuts the row key prefix. func CutRowKeyPrefix(key kv.Key) []byte { return key[prefixLen:] } // EncodeRecordKey encodes the recordPrefix, row handle into a kv.Key. func EncodeRecordKey(recordPrefix kv.Key, h kv.Handle) kv.Key { buf := make([]byte, 0, len(recordPrefix)+h.Len()) buf = append(buf, recordPrefix...) buf = append(buf, h.Encoded()...) return buf } func hasTablePrefix(key kv.Key) bool { return key[0] == tablePrefix[0] } func hasRecordPrefixSep(key kv.Key) bool { return key[0] == recordPrefixSep[0] && key[1] == recordPrefixSep[1] } // DecodeRecordKey decodes the key and gets the tableID, handle. func DecodeRecordKey(key kv.Key) (tableID int64, handle kv.Handle, err error) { if len(key) <= prefixLen { return 0, nil, errInvalidRecordKey.GenWithStack("invalid record key - %q", key) } k := key if !hasTablePrefix(key) { return 0, nil, errInvalidRecordKey.GenWithStack("invalid record key - %q", k) } key = key[tablePrefixLength:] key, tableID, err = codec.DecodeInt(key) if err != nil { return 0, nil, errors.Trace(err) } if !hasRecordPrefixSep(key) { return 0, nil, errInvalidRecordKey.GenWithStack("invalid record key - %q", k) } key = key[recordPrefixSepLength:] var intHandle int64 key, intHandle, err = codec.DecodeInt(key) if err != nil { return 0, nil, errors.Trace(err) } handle = kv.IntHandle(intHandle) return } // DecodeIndexKey decodes the key and gets the tableID, indexID, indexValues. func DecodeIndexKey(key kv.Key) (tableID int64, indexID int64, indexValues []string, err error) { k := key tableID, indexID, key, err = DecodeIndexKeyPrefix(key) if err != nil { return 0, 0, nil, errors.Trace(err) } for len(key) > 0 { // FIXME: Without the schema information, we can only decode the raw kind of // the column. For instance, MysqlTime is internally saved as uint64. remain, d, e := codec.DecodeOne(key) if e != nil { return 0, 0, nil, errInvalidIndexKey.GenWithStack("invalid index key - %q %v", k, e) } str, e1 := d.ToString() if e1 != nil { return 0, 0, nil, errInvalidIndexKey.GenWithStack("invalid index key - %q %v", k, e1) } indexValues = append(indexValues, str) key = remain } return } // DecodeMetaKey decodes the key and get the meta key and meta field. func DecodeMetaKey(ek kv.Key) (key []byte, field []byte, err error) { var tp uint64 if !bytes.HasPrefix(ek, metaPrefix) { return nil, nil, errors.New("invalid encoded hash data key prefix") } ek = ek[metaPrefixLength:] ek, key, err = codec.DecodeBytes(ek, nil) if err != nil { return nil, nil, errors.Trace(err) } ek, tp, err = codec.DecodeUint(ek) if err != nil { return nil, nil, errors.Trace(err) } else if structure.TypeFlag(tp) != structure.HashData { return nil, nil, errors.Errorf("invalid encoded hash data key flag %c", byte(tp)) } _, field, err = codec.DecodeBytes(ek, nil) return key, field, errors.Trace(err) } // DecodeIndexKeyPrefix decodes the key and gets the tableID, indexID, indexValues. func DecodeIndexKeyPrefix(key kv.Key) (tableID int64, indexID int64, indexValues []byte, err error) { k := key tableID, indexID, isRecord, err := DecodeKeyHead(key) if err != nil { return 0, 0, nil, errors.Trace(err) } if isRecord { return 0, 0, nil, errInvalidIndexKey.GenWithStack("invalid index key - %q", k) } indexValues = key[prefixLen+idLen:] return tableID, indexID, indexValues, nil } // DecodeKeyHead decodes the key's head and gets the tableID, indexID. isRecordKey is true when is a record key. func DecodeKeyHead(key kv.Key) (tableID int64, indexID int64, isRecordKey bool, err error) { isRecordKey = false k := key if !key.HasPrefix(tablePrefix) { err = errInvalidKey.GenWithStack("invalid key - %q", k) return } key = key[len(tablePrefix):] key, tableID, err = codec.DecodeInt(key) if err != nil { err = errors.Trace(err) return } if key.HasPrefix(recordPrefixSep) { isRecordKey = true return } if !key.HasPrefix(indexPrefixSep) { err = errInvalidKey.GenWithStack("invalid key - %q", k) return } key = key[len(indexPrefixSep):] key, indexID, err = codec.DecodeInt(key) if err != nil { err = errors.Trace(err) return } return } // DecodeTableID decodes the table ID of the key, if the key is not table key, returns 0. func DecodeTableID(key kv.Key) int64 { if !key.HasPrefix(tablePrefix) { return 0 } key = key[len(tablePrefix):] _, tableID, err := codec.DecodeInt(key) // TODO: return error. terror.Log(errors.Trace(err)) return tableID } // DecodeRowKey decodes the key and gets the handle. func DecodeRowKey(key kv.Key) (kv.Handle, error) { if len(key) < RecordRowKeyLen || !hasTablePrefix(key) || !hasRecordPrefixSep(key[prefixLen-2:]) { return kv.IntHandle(0), errInvalidKey.GenWithStack("invalid key - %q", key) } if len(key) == RecordRowKeyLen { u := binary.BigEndian.Uint64(key[prefixLen:]) return kv.IntHandle(codec.DecodeCmpUintToInt(u)), nil } return kv.NewCommonHandle(key[prefixLen:]) } // EncodeValue encodes a go value to bytes. func EncodeValue(sc *stmtctx.StatementContext, b []byte, raw types.Datum) ([]byte, error) { var v types.Datum err := flatten(sc, raw, &v) if err != nil { return nil, err } return codec.EncodeValue(sc, b, v) } // EncodeRow encode row data and column ids into a slice of byte. // valBuf and values pass by caller, for reducing EncodeRow allocates temporary bufs. If you pass valBuf and values as nil, // EncodeRow will allocate it. func EncodeRow(sc *stmtctx.StatementContext, row []types.Datum, colIDs []int64, valBuf []byte, values []types.Datum, e *rowcodec.Encoder) ([]byte, error) { if len(row) != len(colIDs) { return nil, errors.Errorf("EncodeRow error: data and columnID count not match %d vs %d", len(row), len(colIDs)) } if e.Enable { return e.Encode(sc, colIDs, row, valBuf) } return EncodeOldRow(sc, row, colIDs, valBuf, values) } // EncodeOldRow encode row data and column ids into a slice of byte. // Row layout: colID1, value1, colID2, value2, ..... // valBuf and values pass by caller, for reducing EncodeOldRow allocates temporary bufs. If you pass valBuf and values as nil, // EncodeOldRow will allocate it. func EncodeOldRow(sc *stmtctx.StatementContext, row []types.Datum, colIDs []int64, valBuf []byte, values []types.Datum) ([]byte, error) { if len(row) != len(colIDs) { return nil, errors.Errorf("EncodeRow error: data and columnID count not match %d vs %d", len(row), len(colIDs)) } valBuf = valBuf[:0] if values == nil { values = make([]types.Datum, len(row)*2) } for i, c := range row { id := colIDs[i] values[2*i].SetInt64(id) err := flatten(sc, c, &values[2*i+1]) if err != nil { return valBuf, errors.Trace(err) } } if len(values) == 0 { // We could not set nil value into kv. return append(valBuf, codec.NilFlag), nil } return codec.EncodeValue(sc, valBuf, values...) } func flatten(sc *stmtctx.StatementContext, data types.Datum, ret *types.Datum) error { switch data.Kind() { case types.KindMysqlTime: // for mysql datetime, timestamp and date type t := data.GetMysqlTime() if t.Type() == mysql.TypeTimestamp && sc.TimeZone != time.UTC { err := t.ConvertTimeZone(sc.TimeZone, time.UTC) if err != nil { return errors.Trace(err) } } v, err := t.ToPackedUint() ret.SetUint64(v) return errors.Trace(err) case types.KindMysqlDuration: // for mysql time type ret.SetInt64(int64(data.GetMysqlDuration().Duration)) return nil case types.KindMysqlEnum: ret.SetUint64(data.GetMysqlEnum().Value) return nil case types.KindMysqlSet: ret.SetUint64(data.GetMysqlSet().Value) return nil case types.KindBinaryLiteral, types.KindMysqlBit: // We don't need to handle errors here since the literal is ensured to be able to store in uint64 in convertToMysqlBit. val, err := data.GetBinaryLiteral().ToInt(sc) if err != nil { return errors.Trace(err) } ret.SetUint64(val) return nil default: *ret = data return nil } } // DecodeColumnValue decodes data to a Datum according to the column info. func DecodeColumnValue(data []byte, ft *types.FieldType, loc *time.Location) (types.Datum, error) { _, d, err := codec.DecodeOne(data) if err != nil { return types.Datum{}, errors.Trace(err) } colDatum, err := Unflatten(d, ft, loc) if err != nil { return types.Datum{}, errors.Trace(err) } return colDatum, nil } // DecodeRowWithMapNew decode a row to datum map. func DecodeRowWithMapNew(b []byte, cols map[int64]*types.FieldType, loc *time.Location, row map[int64]types.Datum) (map[int64]types.Datum, error) { if row == nil { row = make(map[int64]types.Datum, len(cols)) } if b == nil { return row, nil } if len(b) == 1 && b[0] == codec.NilFlag { return row, nil } reqCols := make([]rowcodec.ColInfo, len(cols)) var idx int for id, tp := range cols { reqCols[idx] = rowcodec.ColInfo{ ID: id, Ft: tp, } idx++ } // for decodeToMap: // - no need handle // - no need get default value rd := rowcodec.NewDatumMapDecoder(reqCols, nil, loc) return rd.DecodeToDatumMap(b, nil, row) } // DecodeRowWithMap decodes a byte slice into datums with a existing row map. // Row layout: colID1, value1, colID2, value2, ..... func DecodeRowWithMap(b []byte, cols map[int64]*types.FieldType, loc *time.Location, row map[int64]types.Datum) (map[int64]types.Datum, error) { if row == nil { row = make(map[int64]types.Datum, len(cols)) } if b == nil { return row, nil } if len(b) == 1 && b[0] == codec.NilFlag { return row, nil } cnt := 0 var ( data []byte err error ) for len(b) > 0 { // Get col id. data, b, err = codec.CutOne(b) if err != nil { return nil, errors.Trace(err) } _, cid, err := codec.DecodeOne(data) if err != nil { return nil, errors.Trace(err) } // Get col value. data, b, err = codec.CutOne(b) if err != nil { return nil, errors.Trace(err) } id := cid.GetInt64() ft, ok := cols[id] if ok { _, v, err := codec.DecodeOne(data) if err != nil { return nil, errors.Trace(err) } v, err = Unflatten(v, ft, loc) if err != nil { return nil, errors.Trace(err) } row[id] = v cnt++ if cnt == len(cols) { // Get enough data. break } } } return row, nil } // DecodeRow decodes a byte slice into datums. // Row layout: colID1, value1, colID2, value2, ..... func DecodeRow(b []byte, cols map[int64]*types.FieldType, loc *time.Location) (map[int64]types.Datum, error) { if !rowcodec.IsNewFormat(b) { return DecodeRowWithMap(b, cols, loc, nil) } return DecodeRowWithMapNew(b, cols, loc, nil) } // CutRowNew cuts encoded row into byte slices and return columns' byte slice. // Row layout: colID1, value1, colID2, value2, ..... func CutRowNew(data []byte, colIDs map[int64]int) ([][]byte, error) { if data == nil { return nil, nil } if len(data) == 1 && data[0] == codec.NilFlag { return nil, nil } var ( cnt int b []byte err error cid int64 ) row := make([][]byte, len(colIDs)) for len(data) > 0 && cnt < len(colIDs) { // Get col id. data, cid, err = codec.CutColumnID(data) if err != nil { return nil, errors.Trace(err) } // Get col value. b, data, err = codec.CutOne(data) if err != nil { return nil, errors.Trace(err) } offset, ok := colIDs[cid] if ok { row[offset] = b cnt++ } } return row, nil } // UnflattenDatums converts raw datums to column datums. func UnflattenDatums(datums []types.Datum, fts []*types.FieldType, loc *time.Location) ([]types.Datum, error) { for i, datum := range datums { ft := fts[i] uDatum, err := Unflatten(datum, ft, loc) if err != nil { return datums, errors.Trace(err) } datums[i] = uDatum } return datums, nil } // Unflatten converts a raw datum to a column datum. func Unflatten(datum types.Datum, ft *types.FieldType, loc *time.Location) (types.Datum, error) { if datum.IsNull() { return datum, nil } switch ft.Tp { case mysql.TypeFloat: datum.SetFloat32(float32(datum.GetFloat64())) return datum, nil case mysql.TypeVarchar, mysql.TypeString, mysql.TypeVarString: datum.SetString(datum.GetString(), ft.Collate) case mysql.TypeTiny, mysql.TypeShort, mysql.TypeYear, mysql.TypeInt24, mysql.TypeLong, mysql.TypeLonglong, mysql.TypeDouble, mysql.TypeTinyBlob, mysql.TypeMediumBlob, mysql.TypeBlob, mysql.TypeLongBlob: return datum, nil case mysql.TypeDate, mysql.TypeDatetime, mysql.TypeTimestamp: t := types.NewTime(types.ZeroCoreTime, ft.Tp, int8(ft.Decimal)) var err error err = t.FromPackedUint(datum.GetUint64()) if err != nil { return datum, errors.Trace(err) } if ft.Tp == mysql.TypeTimestamp && !t.IsZero() { err = t.ConvertTimeZone(time.UTC, loc) if err != nil { return datum, errors.Trace(err) } } datum.SetUint64(0) datum.SetMysqlTime(t) return datum, nil case mysql.TypeDuration: //duration should read fsp from column meta data dur := types.Duration{Duration: time.Duration(datum.GetInt64()), Fsp: int8(ft.Decimal)} datum.SetMysqlDuration(dur) return datum, nil case mysql.TypeEnum: // ignore error deliberately, to read empty enum value. enum, err := types.ParseEnumValue(ft.Elems, datum.GetUint64()) if err != nil { enum = types.Enum{} } datum.SetMysqlEnum(enum, ft.Collate) return datum, nil case mysql.TypeSet: set, err := types.ParseSetValue(ft.Elems, datum.GetUint64()) if err != nil { return datum, errors.Trace(err) } datum.SetMysqlSet(set, ft.Collate) return datum, nil case mysql.TypeBit: val := datum.GetUint64() byteSize := (ft.Flen + 7) >> 3 datum.SetUint64(0) datum.SetMysqlBit(types.NewBinaryLiteralFromUint(val, byteSize)) } return datum, nil } // EncodeIndexSeekKey encodes an index value to kv.Key. func EncodeIndexSeekKey(tableID int64, idxID int64, encodedValue []byte) kv.Key { key := make([]byte, 0, RecordRowKeyLen+len(encodedValue)) key = appendTableIndexPrefix(key, tableID) key = codec.EncodeInt(key, idxID) key = append(key, encodedValue...) return key } // EncodeCommonHandleSeekKey encodes a common handle value to kv.Key. func EncodeCommonHandleSeekKey(tableID int64, encodedValue []byte) kv.Key { key := make([]byte, 0, prefixLen+len(encodedValue)) key = appendTableRecordPrefix(key, tableID) key = append(key, encodedValue...) return key } // CutIndexKey cuts encoded index key into colIDs to bytes slices map. // The returned value b is the remaining bytes of the key which would be empty if it is unique index or handle data // if it is non-unique index. func CutIndexKey(key kv.Key, colIDs []int64) (values map[int64][]byte, b []byte, err error) { b = key[prefixLen+idLen:] values = make(map[int64][]byte, len(colIDs)) for _, id := range colIDs { var val []byte val, b, err = codec.CutOne(b) if err != nil { return nil, nil, errors.Trace(err) } values[id] = val } return } // CutIndexPrefix cuts the index prefix. func CutIndexPrefix(key kv.Key) []byte { return key[prefixLen+idLen:] } // CutIndexKeyNew cuts encoded index key into colIDs to bytes slices. // The returned value b is the remaining bytes of the key which would be empty if it is unique index or handle data // if it is non-unique index. func CutIndexKeyNew(key kv.Key, length int) (values [][]byte, b []byte, err error) { b = key[prefixLen+idLen:] values = make([][]byte, 0, length) for i := 0; i < length; i++ { var val []byte val, b, err = codec.CutOne(b) if err != nil { return nil, nil, errors.Trace(err) } values = append(values, val) } return } // CutCommonHandle cuts encoded common handle key into colIDs to bytes slices. // The returned value b is the remaining bytes of the key which would be empty if it is unique index or handle data // if it is non-unique index. func CutCommonHandle(key kv.Key, length int) (values [][]byte, b []byte, err error) { b = key[prefixLen:] values = make([][]byte, 0, length) for i := 0; i < length; i++ { var val []byte val, b, err = codec.CutOne(b) if err != nil { return nil, nil, errors.Trace(err) } values = append(values, val) } return } // HandleStatus is the handle status in index. type HandleStatus int const ( // HandleDefault means decode handle value as int64 or bytes when DecodeIndexKV. HandleDefault HandleStatus = iota // HandleIsUnsigned means decode handle value as uint64 when DecodeIndexKV. HandleIsUnsigned // HandleNotNeeded means no need to decode handle value when DecodeIndexKV. HandleNotNeeded ) // reEncodeHandle encodes the handle as a Datum so it can be properly decoded later. // If it is common handle, it is encoded as Bytes Datum. // If it is int handle, it is encoded as int Datum or uint Datum decided by the unsigned. func reEncodeHandle(handle kv.Handle, unsigned bool) ([]byte, error) { if !handle.IsInt() { return codec.EncodeValue(nil, nil, types.NewBytesDatum(handle.Encoded())) } handleDatum := types.NewIntDatum(handle.IntValue()) if unsigned { handleDatum.SetUint64(handleDatum.GetUint64()) } return codec.EncodeValue(nil, nil, handleDatum) } func decodeIndexKvNewCollation(key, value []byte, hdStatus HandleStatus, columns []rowcodec.ColInfo) ([][]byte, error) { vLen := len(value) tailLen := int(value[0]) restoredVal := value[1 : vLen-tailLen] resultValues, err := decodeRestoredValues(columns, restoredVal) if err != nil { return nil, errors.Trace(err) } if hdStatus == HandleNotNeeded { return resultValues, nil } if tailLen < 8 { // In non-unique index. _, keySuffix, err := CutIndexKeyNew(key, len(columns)) if err != nil { return nil, err } handle, err := decodeHandleInIndexKey(keySuffix) if err != nil { return nil, err } handleBytes, err := reEncodeHandle(handle, hdStatus == HandleIsUnsigned) if err != nil { return nil, err } resultValues = append(resultValues, handleBytes) } else { // In unique index. handle := decodeIntHandleInIndexValue(value[vLen-tailLen:]) handleBytes, err := reEncodeHandle(handle, hdStatus == HandleIsUnsigned) if err != nil { return nil, errors.Trace(err) } resultValues = append(resultValues, handleBytes) } return resultValues, nil } func decodeRestoredValues(columns []rowcodec.ColInfo, restoredVal []byte) ([][]byte, error) { colIDs := make(map[int64]int, len(columns)) for i, col := range columns { colIDs[col.ID] = i } // We don't need to decode handle here, and colIDs >= 0 always. rd := rowcodec.NewByteDecoder(columns, []int64{-1}, nil, nil) resultValues, err := rd.DecodeToBytesNoHandle(colIDs, restoredVal) if err != nil { return nil, errors.Trace(err) } return resultValues, nil } func decodeIndexKvOldCollation(key, value []byte, colsLen int, hdStatus HandleStatus) ([][]byte, error) { resultValues, b, err := CutIndexKeyNew(key, colsLen) if err != nil { return nil, errors.Trace(err) } if hdStatus == HandleNotNeeded { return resultValues, nil } var handle kv.Handle if len(b) > 0 { // non-unique index handle, err = decodeHandleInIndexKey(b) if err != nil { return nil, err } if !handle.IsInt() { // Re-encode handle to bytes datum. var handleBytes []byte handleBytes, err = codec.EncodeValue(nil, nil, types.NewBytesDatum(b)) if err != nil { return nil, err } resultValues = append(resultValues, handleBytes) } else { resultValues = append(resultValues, b) } } else { // unique index handle, err = decodeHandleInIndexValue(value) if err != nil { return nil, err } handleBytes, err := reEncodeHandle(handle, hdStatus == HandleIsUnsigned) if err != nil { return nil, errors.Trace(err) } resultValues = append(resultValues, handleBytes) } return resultValues, nil } // DecodeIndexKV uses to decode index key values. func DecodeIndexKV(key, value []byte, colsLen int, hdStatus HandleStatus, columns []rowcodec.ColInfo) ([][]byte, error) { if len(value) > MaxOldEncodeValueLen { if value[0] <= 1 && value[1] == CommonHandleFlag { return decodeIndexKVUniqueCommonHandle(key, value, colsLen, hdStatus, columns) } return decodeIndexKvNewCollation(key, value, hdStatus, columns) } return decodeIndexKvOldCollation(key, value, colsLen, hdStatus) } func decodeIndexKVUniqueCommonHandle(key, value []byte, colsLen int, hdStatus HandleStatus, columns []rowcodec.ColInfo) ([][]byte, error) { tailLen := int(value[0]) value = value[:len(value)-tailLen] handleLen := uint16(value[2])<<8 + uint16(value[3]) handleEndOff := 4 + handleLen var resultValues [][]byte var err error if int(handleEndOff) < len(value) { // new collation values. restoredValue := value[handleEndOff:] resultValues, err = decodeRestoredValues(columns, restoredValue) if err != nil { return nil, errors.Trace(err) } } else { resultValues, _, err = CutIndexKeyNew(key, colsLen) if err != nil { return nil, errors.Trace(err) } } if hdStatus == HandleNotNeeded { return resultValues, nil } // reEncode the common handle to a bytes column reEncodedHandle, err1 := codec.EncodeValue(nil, nil, types.NewBytesDatum(value[4:handleEndOff])) if err1 != nil { return nil, err1 } resultValues = append(resultValues, reEncodedHandle) return resultValues, nil } // DecodeIndexHandle uses to decode the handle from index key/value. func DecodeIndexHandle(key, value []byte, colsLen int) (kv.Handle, error) { _, b, err := CutIndexKeyNew(key, colsLen) if err != nil { return nil, errors.Trace(err) } if len(b) > 0 { return decodeHandleInIndexKey(b) } else if len(value) >= 8 { return decodeHandleInIndexValue(value) } // Should never execute to here. return nil, errors.Errorf("no handle in index key: %v, value: %v", key, value) } func decodeHandleInIndexKey(keySuffix []byte) (kv.Handle, error) { remain, d, err := codec.DecodeOne(keySuffix) if err != nil { return nil, errors.Trace(err) } if len(remain) == 0 && d.Kind() == types.KindInt64 { return kv.IntHandle(d.GetInt64()), nil } return kv.NewCommonHandle(keySuffix) } func decodeHandleInIndexValue(value []byte) (kv.Handle, error) { if len(value) > MaxOldEncodeValueLen { tailLen := value[0] if tailLen >= 8 { return decodeIntHandleInIndexValue(value[len(value)-int(tailLen):]), nil } handleLen := uint16(value[2])<<8 + uint16(value[3]) return kv.NewCommonHandle(value[4 : 4+handleLen]) } return decodeIntHandleInIndexValue(value), nil } // decodeIntHandleInIndexValue uses to decode index value as int handle id. func decodeIntHandleInIndexValue(data []byte) kv.Handle { return kv.IntHandle(binary.BigEndian.Uint64(data)) } // EncodeTableIndexPrefix encodes index prefix with tableID and idxID. func EncodeTableIndexPrefix(tableID, idxID int64) kv.Key { key := make([]byte, 0, prefixLen) key = appendTableIndexPrefix(key, tableID) key = codec.EncodeInt(key, idxID) return key } // EncodeTablePrefix encodes table prefix with table ID. func EncodeTablePrefix(tableID int64) kv.Key { var key kv.Key key = append(key, tablePrefix...) key = codec.EncodeInt(key, tableID) return key } // appendTableRecordPrefix appends table record prefix "t[tableID]_r". func appendTableRecordPrefix(buf []byte, tableID int64) []byte { buf = append(buf, tablePrefix...) buf = codec.EncodeInt(buf, tableID) buf = append(buf, recordPrefixSep...) return buf } // appendTableIndexPrefix appends table index prefix "t[tableID]_i". func appendTableIndexPrefix(buf []byte, tableID int64) []byte { buf = append(buf, tablePrefix...) buf = codec.EncodeInt(buf, tableID) buf = append(buf, indexPrefixSep...) return buf } // GenTableRecordPrefix composes record prefix with tableID: "t[tableID]_r". func GenTableRecordPrefix(tableID int64) kv.Key { buf := make([]byte, 0, len(tablePrefix)+8+len(recordPrefixSep)) return appendTableRecordPrefix(buf, tableID) } // GenTableIndexPrefix composes index prefix with tableID: "t[tableID]_i". func GenTableIndexPrefix(tableID int64) kv.Key { buf := make([]byte, 0, len(tablePrefix)+8+len(indexPrefixSep)) return appendTableIndexPrefix(buf, tableID) } // IsIndexKey is used to check whether the key is an index key. func IsIndexKey(k []byte) bool { return len(k) > 11 && k[0] == 't' && k[10] == 'i' } // IsUntouchedIndexKValue uses to check whether the key is index key, and the value is untouched, // since the untouched index key/value is no need to commit. func IsUntouchedIndexKValue(k, v []byte) bool { if !IsIndexKey(k) { return false } vLen := len(v) if vLen <= MaxOldEncodeValueLen { return (vLen == 1 || vLen == 9) && v[vLen-1] == kv.UnCommitIndexKVFlag } // New index value format tailLen := int(v[0]) if tailLen < 8 { // Non-unique index. return tailLen >= 1 && v[vLen-1] == kv.UnCommitIndexKVFlag } // Unique index return tailLen == 9 } // GenTablePrefix composes table record and index prefix: "t[tableID]". func GenTablePrefix(tableID int64) kv.Key { buf := make([]byte, 0, len(tablePrefix)+8) buf = append(buf, tablePrefix...) buf = codec.EncodeInt(buf, tableID) return buf } // TruncateToRowKeyLen truncates the key to row key length if the key is longer than row key. func TruncateToRowKeyLen(key kv.Key) kv.Key { if len(key) > RecordRowKeyLen { return key[:RecordRowKeyLen] } return key } // GetTableHandleKeyRange returns table handle's key range with tableID. func GetTableHandleKeyRange(tableID int64) (startKey, endKey []byte) { startKey = EncodeRowKeyWithHandle(tableID, kv.IntHandle(math.MinInt64)) endKey = EncodeRowKeyWithHandle(tableID, kv.IntHandle(math.MaxInt64)) return } // GetTableIndexKeyRange returns table index's key range with tableID and indexID. func GetTableIndexKeyRange(tableID, indexID int64) (startKey, endKey []byte) { startKey = EncodeIndexSeekKey(tableID, indexID, nil) endKey = EncodeIndexSeekKey(tableID, indexID, []byte{255}) return }