mirror of https://github.com/k3s-io/k3s
661 lines
14 KiB
Go
661 lines
14 KiB
Go
|
package protocol
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"database/sql/driver"
|
||
|
"encoding/binary"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"math"
|
||
|
"strings"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
// NamedValues is a type alias of a slice of driver.NamedValue. It's used by
|
||
|
// schema.sh to generate encoding logic for statement parameters.
|
||
|
type NamedValues = []driver.NamedValue
|
||
|
|
||
|
// Nodes is a type alias of a slice of NodeInfo. It's used by schema.sh to
|
||
|
// generate decoding logic for the heartbeat response.
|
||
|
type Nodes []NodeInfo
|
||
|
|
||
|
// Message holds data about a single request or response.
|
||
|
type Message struct {
|
||
|
words uint32
|
||
|
mtype uint8
|
||
|
flags uint8
|
||
|
extra uint16
|
||
|
header []byte // Statically allocated header buffer
|
||
|
body1 buffer // Statically allocated body data, using bytes
|
||
|
body2 buffer // Dynamically allocated body data
|
||
|
}
|
||
|
|
||
|
// Init initializes the message using the given size of the statically
|
||
|
// allocated buffer (i.e. a buffer which is re-used across requests or
|
||
|
// responses encoded or decoded using this message object).
|
||
|
func (m *Message) Init(staticSize int) {
|
||
|
if (staticSize % messageWordSize) != 0 {
|
||
|
panic("static size is not aligned to word boundary")
|
||
|
}
|
||
|
m.header = make([]byte, messageHeaderSize)
|
||
|
m.body1.Bytes = make([]byte, staticSize)
|
||
|
m.Reset()
|
||
|
}
|
||
|
|
||
|
// Reset the state of the message so it can be used to encode or decode again.
|
||
|
func (m *Message) Reset() {
|
||
|
m.words = 0
|
||
|
m.mtype = 0
|
||
|
m.flags = 0
|
||
|
m.extra = 0
|
||
|
for i := 0; i < messageHeaderSize; i++ {
|
||
|
m.header[i] = 0
|
||
|
}
|
||
|
m.body1.Offset = 0
|
||
|
m.body2.Bytes = nil
|
||
|
m.body2.Offset = 0
|
||
|
}
|
||
|
|
||
|
// Append a byte slice to the message.
|
||
|
func (m *Message) putBlob(v []byte) {
|
||
|
size := len(v)
|
||
|
m.putUint64(uint64(size))
|
||
|
|
||
|
pad := 0
|
||
|
if (size % messageWordSize) != 0 {
|
||
|
// Account for padding
|
||
|
pad = messageWordSize - (size % messageWordSize)
|
||
|
size += pad
|
||
|
}
|
||
|
|
||
|
b := m.bufferForPut(size)
|
||
|
defer b.Advance(size)
|
||
|
|
||
|
// Copy the bytes into the buffer.
|
||
|
offset := b.Offset
|
||
|
copy(b.Bytes[offset:], v)
|
||
|
offset += len(v)
|
||
|
|
||
|
// Add padding
|
||
|
for i := 0; i < pad; i++ {
|
||
|
b.Bytes[offset] = 0
|
||
|
offset++
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Append a string to the message.
|
||
|
func (m *Message) putString(v string) {
|
||
|
size := len(v) + 1
|
||
|
pad := 0
|
||
|
if (size % messageWordSize) != 0 {
|
||
|
// Account for padding
|
||
|
pad = messageWordSize - (size % messageWordSize)
|
||
|
size += pad
|
||
|
}
|
||
|
|
||
|
b := m.bufferForPut(size)
|
||
|
defer b.Advance(size)
|
||
|
|
||
|
// Copy the string bytes into the buffer.
|
||
|
offset := b.Offset
|
||
|
copy(b.Bytes[offset:], v)
|
||
|
offset += len(v)
|
||
|
|
||
|
// Add a nul byte
|
||
|
b.Bytes[offset] = 0
|
||
|
offset++
|
||
|
|
||
|
// Add padding
|
||
|
for i := 0; i < pad; i++ {
|
||
|
b.Bytes[offset] = 0
|
||
|
offset++
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Append a byte to the message.
|
||
|
func (m *Message) putUint8(v uint8) {
|
||
|
b := m.bufferForPut(1)
|
||
|
defer b.Advance(1)
|
||
|
|
||
|
b.Bytes[b.Offset] = v
|
||
|
}
|
||
|
|
||
|
// Append a 2-byte word to the message.
|
||
|
func (m *Message) putUint16(v uint16) {
|
||
|
b := m.bufferForPut(2)
|
||
|
defer b.Advance(2)
|
||
|
|
||
|
binary.LittleEndian.PutUint16(b.Bytes[b.Offset:], v)
|
||
|
}
|
||
|
|
||
|
// Append a 4-byte word to the message.
|
||
|
func (m *Message) putUint32(v uint32) {
|
||
|
b := m.bufferForPut(4)
|
||
|
defer b.Advance(4)
|
||
|
|
||
|
binary.LittleEndian.PutUint32(b.Bytes[b.Offset:], v)
|
||
|
}
|
||
|
|
||
|
// Append an 8-byte word to the message.
|
||
|
func (m *Message) putUint64(v uint64) {
|
||
|
b := m.bufferForPut(8)
|
||
|
defer b.Advance(8)
|
||
|
|
||
|
binary.LittleEndian.PutUint64(b.Bytes[b.Offset:], v)
|
||
|
}
|
||
|
|
||
|
// Append a signed 8-byte word to the message.
|
||
|
func (m *Message) putInt64(v int64) {
|
||
|
b := m.bufferForPut(8)
|
||
|
defer b.Advance(8)
|
||
|
|
||
|
binary.LittleEndian.PutUint64(b.Bytes[b.Offset:], uint64(v))
|
||
|
}
|
||
|
|
||
|
// Append a floating point number to the message.
|
||
|
func (m *Message) putFloat64(v float64) {
|
||
|
b := m.bufferForPut(8)
|
||
|
defer b.Advance(8)
|
||
|
|
||
|
binary.LittleEndian.PutUint64(b.Bytes[b.Offset:], math.Float64bits(v))
|
||
|
}
|
||
|
|
||
|
// Encode the given driver values as binding parameters.
|
||
|
func (m *Message) putNamedValues(values NamedValues) {
|
||
|
n := uint8(len(values)) // N of params
|
||
|
if n == 0 {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
m.putUint8(n)
|
||
|
|
||
|
for i := range values {
|
||
|
if values[i].Ordinal != i+1 {
|
||
|
panic("unexpected ordinal")
|
||
|
}
|
||
|
|
||
|
switch values[i].Value.(type) {
|
||
|
case int64:
|
||
|
m.putUint8(Integer)
|
||
|
case float64:
|
||
|
m.putUint8(Float)
|
||
|
case bool:
|
||
|
m.putUint8(Boolean)
|
||
|
case []byte:
|
||
|
m.putUint8(Blob)
|
||
|
case string:
|
||
|
m.putUint8(Text)
|
||
|
case nil:
|
||
|
m.putUint8(Null)
|
||
|
case time.Time:
|
||
|
m.putUint8(ISO8601)
|
||
|
default:
|
||
|
panic("unsupported value type")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
b := m.bufferForPut(1)
|
||
|
|
||
|
if trailing := b.Offset % messageWordSize; trailing != 0 {
|
||
|
// Skip padding bytes
|
||
|
b.Advance(messageWordSize - trailing)
|
||
|
}
|
||
|
|
||
|
for i := range values {
|
||
|
switch v := values[i].Value.(type) {
|
||
|
case int64:
|
||
|
m.putInt64(v)
|
||
|
case float64:
|
||
|
m.putFloat64(v)
|
||
|
case bool:
|
||
|
if v {
|
||
|
m.putUint64(1)
|
||
|
} else {
|
||
|
m.putUint64(0)
|
||
|
}
|
||
|
case []byte:
|
||
|
m.putBlob(v)
|
||
|
case string:
|
||
|
m.putString(v)
|
||
|
case nil:
|
||
|
m.putInt64(0)
|
||
|
case time.Time:
|
||
|
timestamp := v.Format(iso8601Formats[0])
|
||
|
m.putString(timestamp)
|
||
|
default:
|
||
|
panic("unsupported value type")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
// Finalize the message by setting the message type and the number
|
||
|
// of words in the body (calculated from the body size).
|
||
|
func (m *Message) putHeader(mtype uint8) {
|
||
|
if m.body1.Offset <= 0 {
|
||
|
panic("static offset is not positive")
|
||
|
}
|
||
|
|
||
|
if (m.body1.Offset % messageWordSize) != 0 {
|
||
|
panic("static body is not aligned")
|
||
|
}
|
||
|
|
||
|
m.mtype = mtype
|
||
|
m.flags = 0
|
||
|
m.extra = 0
|
||
|
|
||
|
m.words = uint32(m.body1.Offset) / messageWordSize
|
||
|
|
||
|
if m.body2.Bytes == nil {
|
||
|
m.finalize()
|
||
|
return
|
||
|
}
|
||
|
|
||
|
if m.body2.Offset <= 0 {
|
||
|
panic("dynamic offset is not positive")
|
||
|
}
|
||
|
|
||
|
if (m.body2.Offset % messageWordSize) != 0 {
|
||
|
panic("dynamic body is not aligned")
|
||
|
}
|
||
|
|
||
|
m.words += uint32(m.body2.Offset) / messageWordSize
|
||
|
|
||
|
m.finalize()
|
||
|
}
|
||
|
|
||
|
func (m *Message) finalize() {
|
||
|
if m.words == 0 {
|
||
|
panic("empty message body")
|
||
|
}
|
||
|
|
||
|
binary.LittleEndian.PutUint32(m.header[0:], m.words)
|
||
|
m.header[4] = m.mtype
|
||
|
m.header[5] = m.flags
|
||
|
binary.LittleEndian.PutUint16(m.header[6:], m.extra)
|
||
|
}
|
||
|
|
||
|
func (m *Message) bufferForPut(size int) *buffer {
|
||
|
if m.body2.Bytes != nil {
|
||
|
if (m.body2.Offset + size) > len(m.body2.Bytes) {
|
||
|
// Grow body2.
|
||
|
//
|
||
|
// TODO: find a good grow strategy.
|
||
|
bytes := make([]byte, m.body2.Offset+size)
|
||
|
copy(bytes, m.body2.Bytes)
|
||
|
m.body2.Bytes = bytes
|
||
|
}
|
||
|
|
||
|
return &m.body2
|
||
|
}
|
||
|
|
||
|
if (m.body1.Offset + size) > len(m.body1.Bytes) {
|
||
|
m.body2.Bytes = make([]byte, size)
|
||
|
m.body2.Offset = 0
|
||
|
|
||
|
return &m.body2
|
||
|
}
|
||
|
|
||
|
return &m.body1
|
||
|
}
|
||
|
|
||
|
// Return the message type and its flags.
|
||
|
func (m *Message) getHeader() (uint8, uint8) {
|
||
|
return m.mtype, m.flags
|
||
|
}
|
||
|
|
||
|
// Read a string from the message body.
|
||
|
func (m *Message) getString() string {
|
||
|
b := m.bufferForGet()
|
||
|
|
||
|
index := bytes.IndexByte(b.Bytes[b.Offset:], 0)
|
||
|
if index == -1 {
|
||
|
// Check if the string overflows in the dynamic buffer.
|
||
|
if b == &m.body1 && m.body2.Bytes != nil {
|
||
|
// Assert that this is the first read of the dynamic buffer.
|
||
|
if m.body2.Offset != 0 {
|
||
|
panic("static buffer read after dynamic buffer one")
|
||
|
}
|
||
|
index = bytes.IndexByte(m.body2.Bytes[0:], 0)
|
||
|
if index != -1 {
|
||
|
// We found the trailing part of the string.
|
||
|
data := b.Bytes[b.Offset:]
|
||
|
data = append(data, m.body2.Bytes[0:index]...)
|
||
|
|
||
|
index++
|
||
|
|
||
|
if trailing := index % messageWordSize; trailing != 0 {
|
||
|
// Account for padding, moving index to the next word boundary.
|
||
|
index += messageWordSize - trailing
|
||
|
}
|
||
|
|
||
|
m.body1.Offset = len(m.body1.Bytes)
|
||
|
m.body2.Advance(index)
|
||
|
|
||
|
return string(data)
|
||
|
}
|
||
|
}
|
||
|
panic("no string found")
|
||
|
}
|
||
|
s := string(b.Bytes[b.Offset : b.Offset+index])
|
||
|
|
||
|
index++
|
||
|
|
||
|
if trailing := index % messageWordSize; trailing != 0 {
|
||
|
// Account for padding, moving index to the next word boundary.
|
||
|
index += messageWordSize - trailing
|
||
|
}
|
||
|
|
||
|
b.Advance(index)
|
||
|
|
||
|
return s
|
||
|
}
|
||
|
|
||
|
func (m *Message) getBlob() []byte {
|
||
|
size := m.getUint64()
|
||
|
data := make([]byte, size)
|
||
|
for i := range data {
|
||
|
data[i] = m.getUint8()
|
||
|
}
|
||
|
pad := 0
|
||
|
if (size % messageWordSize) != 0 {
|
||
|
// Account for padding
|
||
|
pad = int(messageWordSize - (size % messageWordSize))
|
||
|
}
|
||
|
// Consume padding
|
||
|
for i := 0; i < pad; i++ {
|
||
|
m.getUint8()
|
||
|
}
|
||
|
return data
|
||
|
}
|
||
|
|
||
|
// Read a byte from the message body.
|
||
|
func (m *Message) getUint8() uint8 {
|
||
|
b := m.bufferForGet()
|
||
|
defer b.Advance(1)
|
||
|
|
||
|
return b.Bytes[b.Offset]
|
||
|
}
|
||
|
|
||
|
// Read a 2-byte word from the message body.
|
||
|
func (m *Message) getUint16() uint16 {
|
||
|
b := m.bufferForGet()
|
||
|
defer b.Advance(2)
|
||
|
|
||
|
return binary.LittleEndian.Uint16(b.Bytes[b.Offset:])
|
||
|
}
|
||
|
|
||
|
// Read a 4-byte word from the message body.
|
||
|
func (m *Message) getUint32() uint32 {
|
||
|
b := m.bufferForGet()
|
||
|
defer b.Advance(4)
|
||
|
|
||
|
return binary.LittleEndian.Uint32(b.Bytes[b.Offset:])
|
||
|
}
|
||
|
|
||
|
// Read reads an 8-byte word from the message body.
|
||
|
func (m *Message) getUint64() uint64 {
|
||
|
b := m.bufferForGet()
|
||
|
defer b.Advance(8)
|
||
|
|
||
|
return binary.LittleEndian.Uint64(b.Bytes[b.Offset:])
|
||
|
}
|
||
|
|
||
|
// Read a signed 8-byte word from the message body.
|
||
|
func (m *Message) getInt64() int64 {
|
||
|
b := m.bufferForGet()
|
||
|
defer b.Advance(8)
|
||
|
|
||
|
return int64(binary.LittleEndian.Uint64(b.Bytes[b.Offset:]))
|
||
|
}
|
||
|
|
||
|
// Read a floating point number from the message body.
|
||
|
func (m *Message) getFloat64() float64 {
|
||
|
b := m.bufferForGet()
|
||
|
defer b.Advance(8)
|
||
|
|
||
|
return math.Float64frombits(binary.LittleEndian.Uint64(b.Bytes[b.Offset:]))
|
||
|
}
|
||
|
|
||
|
// Decode a list of server objects from the message body.
|
||
|
func (m *Message) getNodes() Nodes {
|
||
|
n := m.getUint64()
|
||
|
servers := make(Nodes, n)
|
||
|
|
||
|
for i := 0; i < int(n); i++ {
|
||
|
servers[i].ID = m.getUint64()
|
||
|
servers[i].Address = m.getString()
|
||
|
}
|
||
|
|
||
|
return servers
|
||
|
}
|
||
|
|
||
|
// Decode a statement result object from the message body.
|
||
|
func (m *Message) getResult() Result {
|
||
|
return Result{
|
||
|
LastInsertID: m.getUint64(),
|
||
|
RowsAffected: m.getUint64(),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Decode a query result set object from the message body.
|
||
|
func (m *Message) getRows() Rows {
|
||
|
// Read the column count and column names.
|
||
|
columns := make([]string, m.getUint64())
|
||
|
|
||
|
for i := range columns {
|
||
|
columns[i] = m.getString()
|
||
|
}
|
||
|
|
||
|
rows := Rows{
|
||
|
Columns: columns,
|
||
|
message: m,
|
||
|
}
|
||
|
return rows
|
||
|
}
|
||
|
|
||
|
func (m *Message) getFiles() Files {
|
||
|
files := Files{
|
||
|
n: m.getUint64(),
|
||
|
message: m,
|
||
|
}
|
||
|
return files
|
||
|
}
|
||
|
|
||
|
func (m *Message) hasBeenConsumed() bool {
|
||
|
size := int(m.words * messageWordSize)
|
||
|
return (m.body1.Offset == size || m.body1.Offset == len(m.body1.Bytes)) &&
|
||
|
m.body1.Offset+m.body2.Offset == size
|
||
|
}
|
||
|
|
||
|
func (m *Message) lastByte() byte {
|
||
|
size := int(m.words * messageWordSize)
|
||
|
if size > len(m.body1.Bytes) {
|
||
|
size = size - m.body1.Offset
|
||
|
return m.body2.Bytes[size-1]
|
||
|
}
|
||
|
return m.body1.Bytes[size-1]
|
||
|
}
|
||
|
|
||
|
func (m *Message) bufferForGet() *buffer {
|
||
|
size := int(m.words * messageWordSize)
|
||
|
if m.body1.Offset == size || m.body1.Offset == len(m.body1.Bytes) {
|
||
|
// The static body has been exahusted, use the dynamic one.
|
||
|
if m.body1.Offset+m.body2.Offset == size {
|
||
|
err := fmt.Errorf("short message: type=%d words=%d off=%d", m.mtype, m.words, m.body1.Offset)
|
||
|
panic(err)
|
||
|
}
|
||
|
return &m.body2
|
||
|
}
|
||
|
|
||
|
return &m.body1
|
||
|
}
|
||
|
|
||
|
// Result holds the result of a statement.
|
||
|
type Result struct {
|
||
|
LastInsertID uint64
|
||
|
RowsAffected uint64
|
||
|
}
|
||
|
|
||
|
// Rows holds a result set encoded in a message body.
|
||
|
type Rows struct {
|
||
|
Columns []string
|
||
|
message *Message
|
||
|
}
|
||
|
|
||
|
// Next returns the next row in the result set.
|
||
|
func (r *Rows) Next(dest []driver.Value) error {
|
||
|
types := make([]uint8, len(r.Columns))
|
||
|
|
||
|
// Each column needs a 4 byte slot to store the column type. The row
|
||
|
// header must be padded to reach word boundary.
|
||
|
headerBits := len(types) * 4
|
||
|
padBits := 0
|
||
|
if trailingBits := (headerBits % messageWordBits); trailingBits != 0 {
|
||
|
padBits = (messageWordBits - trailingBits)
|
||
|
}
|
||
|
|
||
|
headerSize := (headerBits + padBits) / messageWordBits * messageWordSize
|
||
|
|
||
|
for i := 0; i < headerSize; i++ {
|
||
|
slot := r.message.getUint8()
|
||
|
|
||
|
if slot == 0xee {
|
||
|
// More rows are available.
|
||
|
return ErrRowsPart
|
||
|
}
|
||
|
|
||
|
if slot == 0xff {
|
||
|
// Rows EOF marker
|
||
|
return io.EOF
|
||
|
}
|
||
|
|
||
|
index := i * 2
|
||
|
|
||
|
if index >= len(types) {
|
||
|
continue // This is padding.
|
||
|
}
|
||
|
|
||
|
types[index] = slot & 0x0f
|
||
|
|
||
|
index++
|
||
|
|
||
|
if index >= len(types) {
|
||
|
continue // This is padding byte.
|
||
|
}
|
||
|
|
||
|
types[index] = slot >> 4
|
||
|
}
|
||
|
|
||
|
for i := range types {
|
||
|
switch types[i] {
|
||
|
case Integer:
|
||
|
dest[i] = r.message.getInt64()
|
||
|
case Float:
|
||
|
dest[i] = r.message.getFloat64()
|
||
|
case Blob:
|
||
|
dest[i] = r.message.getBlob()
|
||
|
case Text:
|
||
|
dest[i] = r.message.getString()
|
||
|
case Null:
|
||
|
r.message.getUint64()
|
||
|
dest[i] = nil
|
||
|
case UnixTime:
|
||
|
timestamp := time.Unix(r.message.getInt64(), 0)
|
||
|
dest[i] = timestamp
|
||
|
case ISO8601:
|
||
|
value := r.message.getString()
|
||
|
if value == "" {
|
||
|
dest[i] = time.Time{}
|
||
|
break
|
||
|
}
|
||
|
var t time.Time
|
||
|
var timeVal time.Time
|
||
|
var err error
|
||
|
value = strings.TrimSuffix(value, "Z")
|
||
|
for _, format := range iso8601Formats {
|
||
|
if timeVal, err = time.ParseInLocation(format, value, time.UTC); err == nil {
|
||
|
t = timeVal
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
t = t.In(time.Local)
|
||
|
dest[i] = t
|
||
|
case Boolean:
|
||
|
dest[i] = r.message.getInt64() != 0
|
||
|
default:
|
||
|
panic("unknown data type")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// Close the result set and reset the underlying message.
|
||
|
func (r *Rows) Close() error {
|
||
|
// If we didn't go through all rows, let's look at the last byte.
|
||
|
var err error
|
||
|
if !r.message.hasBeenConsumed() {
|
||
|
slot := r.message.lastByte()
|
||
|
if slot == 0xee {
|
||
|
// More rows are available.
|
||
|
err = ErrRowsPart
|
||
|
} else if slot == 0xff {
|
||
|
// Rows EOF marker
|
||
|
err = io.EOF
|
||
|
} else {
|
||
|
err = fmt.Errorf("unexpected end of message")
|
||
|
}
|
||
|
}
|
||
|
r.message.Reset()
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
// Files holds a set of files encoded in a message body.
|
||
|
type Files struct {
|
||
|
n uint64
|
||
|
message *Message
|
||
|
}
|
||
|
|
||
|
func (f *Files) Next() (string, []byte) {
|
||
|
if f.n == 0 {
|
||
|
return "", nil
|
||
|
}
|
||
|
f.n--
|
||
|
name := f.message.getString()
|
||
|
length := f.message.getUint64()
|
||
|
data := make([]byte, length)
|
||
|
for i := 0; i < int(length); i++ {
|
||
|
data[i] = f.message.getUint8()
|
||
|
}
|
||
|
return name, data
|
||
|
}
|
||
|
|
||
|
func (f *Files) Close() {
|
||
|
f.message.Reset()
|
||
|
}
|
||
|
|
||
|
const (
|
||
|
messageWordSize = 8
|
||
|
messageWordBits = messageWordSize * 8
|
||
|
messageHeaderSize = messageWordSize
|
||
|
messageMaxConsecutiveEmptyReads = 100
|
||
|
)
|
||
|
|
||
|
var iso8601Formats = []string{
|
||
|
// By default, store timestamps with whatever timezone they come with.
|
||
|
// When parsed, they will be returned with the same timezone.
|
||
|
"2006-01-02 15:04:05.999999999-07:00",
|
||
|
"2006-01-02T15:04:05.999999999-07:00",
|
||
|
"2006-01-02 15:04:05.999999999",
|
||
|
"2006-01-02T15:04:05.999999999",
|
||
|
"2006-01-02 15:04:05",
|
||
|
"2006-01-02T15:04:05",
|
||
|
"2006-01-02 15:04",
|
||
|
"2006-01-02T15:04",
|
||
|
"2006-01-02",
|
||
|
}
|