2016-06-16 11:15:56 +02:00
|
|
|
package mysql
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
2021-06-24 20:19:37 +02:00
|
|
|
|
2021-07-15 21:49:50 +02:00
|
|
|
"github.com/go-mysql-org/go-mysql/utils"
|
2016-06-16 11:15:56 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type FieldData []byte
|
|
|
|
|
|
|
|
type Field struct {
|
|
|
|
Data FieldData
|
|
|
|
Schema []byte
|
|
|
|
Table []byte
|
|
|
|
OrgTable []byte
|
|
|
|
Name []byte
|
|
|
|
OrgName []byte
|
|
|
|
Charset uint16
|
|
|
|
ColumnLength uint32
|
|
|
|
Type uint8
|
|
|
|
Flag uint16
|
|
|
|
Decimal uint8
|
|
|
|
|
|
|
|
DefaultValueLength uint64
|
|
|
|
DefaultValue []byte
|
|
|
|
}
|
|
|
|
|
2021-06-24 20:19:37 +02:00
|
|
|
type FieldValueType uint8
|
|
|
|
|
|
|
|
type FieldValue struct {
|
|
|
|
Type FieldValueType
|
|
|
|
value uint64 // Also for int64 and float64
|
|
|
|
str []byte
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
FieldValueTypeNull = iota
|
|
|
|
FieldValueTypeUnsigned
|
|
|
|
FieldValueTypeSigned
|
|
|
|
FieldValueTypeFloat
|
|
|
|
FieldValueTypeString
|
|
|
|
)
|
2016-06-16 11:15:56 +02:00
|
|
|
|
2021-06-24 20:19:37 +02:00
|
|
|
func (f *Field) Parse(p FieldData) (err error) {
|
2016-06-16 11:15:56 +02:00
|
|
|
f.Data = p
|
|
|
|
|
|
|
|
var n int
|
|
|
|
pos := 0
|
|
|
|
//skip catelog, always def
|
2019-01-01 10:57:46 +02:00
|
|
|
n, err = SkipLengthEncodedString(p)
|
2016-06-16 11:15:56 +02:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
pos += n
|
|
|
|
|
|
|
|
//schema
|
2019-01-01 10:57:46 +02:00
|
|
|
f.Schema, _, n, err = LengthEncodedString(p[pos:])
|
2016-06-16 11:15:56 +02:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
pos += n
|
|
|
|
|
|
|
|
//table
|
2019-01-01 10:57:46 +02:00
|
|
|
f.Table, _, n, err = LengthEncodedString(p[pos:])
|
2016-06-16 11:15:56 +02:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
pos += n
|
|
|
|
|
|
|
|
//org_table
|
2019-01-01 10:57:46 +02:00
|
|
|
f.OrgTable, _, n, err = LengthEncodedString(p[pos:])
|
2016-06-16 11:15:56 +02:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
pos += n
|
|
|
|
|
|
|
|
//name
|
2019-01-01 10:57:46 +02:00
|
|
|
f.Name, _, n, err = LengthEncodedString(p[pos:])
|
2016-06-16 11:15:56 +02:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
pos += n
|
|
|
|
|
|
|
|
//org_name
|
2019-01-01 10:57:46 +02:00
|
|
|
f.OrgName, _, n, err = LengthEncodedString(p[pos:])
|
2016-06-16 11:15:56 +02:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
pos += n
|
|
|
|
|
|
|
|
//skip oc
|
|
|
|
pos += 1
|
|
|
|
|
|
|
|
//charset
|
|
|
|
f.Charset = binary.LittleEndian.Uint16(p[pos:])
|
|
|
|
pos += 2
|
|
|
|
|
|
|
|
//column length
|
|
|
|
f.ColumnLength = binary.LittleEndian.Uint32(p[pos:])
|
|
|
|
pos += 4
|
|
|
|
|
|
|
|
//type
|
|
|
|
f.Type = p[pos]
|
|
|
|
pos++
|
|
|
|
|
|
|
|
//flag
|
|
|
|
f.Flag = binary.LittleEndian.Uint16(p[pos:])
|
|
|
|
pos += 2
|
|
|
|
|
|
|
|
//decimals 1
|
|
|
|
f.Decimal = p[pos]
|
|
|
|
pos++
|
|
|
|
|
|
|
|
//filter [0x00][0x00]
|
|
|
|
pos += 2
|
|
|
|
|
|
|
|
f.DefaultValue = nil
|
|
|
|
//if more data, command was field list
|
|
|
|
if len(p) > pos {
|
|
|
|
//length of default value lenenc-int
|
|
|
|
f.DefaultValueLength, _, n = LengthEncodedInt(p[pos:])
|
|
|
|
pos += n
|
|
|
|
|
|
|
|
if pos+int(f.DefaultValueLength) > len(p) {
|
|
|
|
err = ErrMalformPacket
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
//default value string[$len]
|
|
|
|
f.DefaultValue = p[pos:(pos + int(f.DefaultValueLength))]
|
|
|
|
}
|
|
|
|
|
2021-07-15 21:49:50 +02:00
|
|
|
return nil
|
2016-06-16 11:15:56 +02:00
|
|
|
}
|
|
|
|
|
2021-06-24 20:19:37 +02:00
|
|
|
func (p FieldData) Parse() (f *Field, err error) {
|
|
|
|
f = new(Field)
|
|
|
|
if err = f.Parse(p); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return f, nil
|
|
|
|
}
|
|
|
|
|
2016-06-16 11:15:56 +02:00
|
|
|
func (f *Field) Dump() []byte {
|
|
|
|
if f == nil {
|
|
|
|
f = &Field{}
|
|
|
|
}
|
|
|
|
if f.Data != nil {
|
|
|
|
return []byte(f.Data)
|
|
|
|
}
|
|
|
|
|
|
|
|
l := len(f.Schema) + len(f.Table) + len(f.OrgTable) + len(f.Name) + len(f.OrgName) + len(f.DefaultValue) + 48
|
|
|
|
|
|
|
|
data := make([]byte, 0, l)
|
|
|
|
|
|
|
|
data = append(data, PutLengthEncodedString([]byte("def"))...)
|
|
|
|
|
|
|
|
data = append(data, PutLengthEncodedString(f.Schema)...)
|
|
|
|
|
|
|
|
data = append(data, PutLengthEncodedString(f.Table)...)
|
|
|
|
data = append(data, PutLengthEncodedString(f.OrgTable)...)
|
|
|
|
|
|
|
|
data = append(data, PutLengthEncodedString(f.Name)...)
|
|
|
|
data = append(data, PutLengthEncodedString(f.OrgName)...)
|
|
|
|
|
|
|
|
data = append(data, 0x0c)
|
|
|
|
|
|
|
|
data = append(data, Uint16ToBytes(f.Charset)...)
|
|
|
|
data = append(data, Uint32ToBytes(f.ColumnLength)...)
|
|
|
|
data = append(data, f.Type)
|
|
|
|
data = append(data, Uint16ToBytes(f.Flag)...)
|
|
|
|
data = append(data, f.Decimal)
|
|
|
|
data = append(data, 0, 0)
|
|
|
|
|
|
|
|
if f.DefaultValue != nil {
|
|
|
|
data = append(data, Uint64ToBytes(f.DefaultValueLength)...)
|
|
|
|
data = append(data, f.DefaultValue...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return data
|
|
|
|
}
|
2021-06-24 20:19:37 +02:00
|
|
|
|
|
|
|
func (fv *FieldValue) AsUint64() uint64 {
|
|
|
|
return fv.value
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fv *FieldValue) AsInt64() int64 {
|
|
|
|
return utils.Uint64ToInt64(fv.value)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fv *FieldValue) AsFloat64() float64 {
|
|
|
|
return utils.Uint64ToFloat64(fv.value)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fv *FieldValue) AsString() []byte {
|
|
|
|
return fv.str
|
|
|
|
}
|
|
|
|
|
|
|
|
func (fv *FieldValue) Value() interface{} {
|
|
|
|
switch fv.Type {
|
|
|
|
case FieldValueTypeUnsigned:
|
|
|
|
return fv.AsUint64()
|
|
|
|
case FieldValueTypeSigned:
|
|
|
|
return fv.AsInt64()
|
|
|
|
case FieldValueTypeFloat:
|
|
|
|
return fv.AsFloat64()
|
|
|
|
case FieldValueTypeString:
|
|
|
|
return fv.AsString()
|
|
|
|
default: // FieldValueTypeNull
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|