redis/parser.go

638 lines
13 KiB
Go
Raw Normal View History

package redis
import (
"errors"
"fmt"
2015-09-03 17:55:31 +03:00
"io"
2015-01-24 15:12:48 +03:00
"net"
"strconv"
)
2015-10-07 17:09:20 +03:00
const (
errorReply = '-'
statusReply = '+'
intReply = ':'
stringReply = '$'
arrayReply = '*'
)
2015-09-03 17:55:31 +03:00
type multiBulkParser func(cn *conn, n int64) (interface{}, error)
2014-05-11 11:42:40 +04:00
var (
2014-07-05 14:46:27 +04:00
errReaderTooSmall = errors.New("redis: reader is too small")
)
//------------------------------------------------------------------------------
// Copy of encoding.BinaryMarshaler.
type binaryMarshaler interface {
MarshalBinary() (data []byte, err error)
}
// Copy of encoding.BinaryUnmarshaler.
type binaryUnmarshaler interface {
UnmarshalBinary(data []byte) error
}
func appendString(b []byte, s string) []byte {
b = append(b, '$')
b = strconv.AppendUint(b, uint64(len(s)), 10)
b = append(b, '\r', '\n')
b = append(b, s...)
b = append(b, '\r', '\n')
return b
}
func appendBytes(b, bb []byte) []byte {
b = append(b, '$')
b = strconv.AppendUint(b, uint64(len(bb)), 10)
b = append(b, '\r', '\n')
b = append(b, bb...)
b = append(b, '\r', '\n')
return b
}
func appendArg(b []byte, val interface{}) ([]byte, error) {
switch v := val.(type) {
case nil:
b = appendString(b, "")
case string:
b = appendString(b, v)
case []byte:
b = appendBytes(b, v)
case int:
b = appendString(b, formatInt(int64(v)))
case int8:
b = appendString(b, formatInt(int64(v)))
case int16:
b = appendString(b, formatInt(int64(v)))
case int32:
b = appendString(b, formatInt(int64(v)))
case int64:
b = appendString(b, formatInt(v))
case uint:
b = appendString(b, formatUint(uint64(v)))
case uint8:
b = appendString(b, formatUint(uint64(v)))
case uint16:
b = appendString(b, formatUint(uint64(v)))
case uint32:
b = appendString(b, formatUint(uint64(v)))
case uint64:
b = appendString(b, formatUint(v))
case float32:
b = appendString(b, formatFloat(float64(v)))
case float64:
b = appendString(b, formatFloat(v))
case bool:
if v {
b = appendString(b, "1")
} else {
b = appendString(b, "0")
}
default:
if bm, ok := val.(binaryMarshaler); ok {
bb, err := bm.MarshalBinary()
if err != nil {
return nil, err
}
b = appendBytes(b, bb)
} else {
err := fmt.Errorf(
"redis: can't marshal %T (consider implementing BinaryMarshaler)", val)
return nil, err
}
}
return b, nil
}
func appendArgs(b []byte, args []interface{}) ([]byte, error) {
b = append(b, '*')
b = strconv.AppendUint(b, uint64(len(args)), 10)
b = append(b, '\r', '\n')
for _, arg := range args {
var err error
b, err = appendArg(b, arg)
if err != nil {
return nil, err
}
}
return b, nil
}
func scan(b []byte, val interface{}) error {
switch v := val.(type) {
case nil:
return errorf("redis: Scan(nil)")
case *string:
*v = bytesToString(b)
return nil
case *[]byte:
*v = b
return nil
case *int:
var err error
*v, err = strconv.Atoi(bytesToString(b))
return err
case *int8:
n, err := strconv.ParseInt(bytesToString(b), 10, 8)
if err != nil {
return err
}
*v = int8(n)
return nil
case *int16:
n, err := strconv.ParseInt(bytesToString(b), 10, 16)
if err != nil {
return err
}
*v = int16(n)
return nil
case *int32:
n, err := strconv.ParseInt(bytesToString(b), 10, 16)
if err != nil {
return err
}
*v = int32(n)
return nil
case *int64:
n, err := strconv.ParseInt(bytesToString(b), 10, 64)
if err != nil {
return err
}
*v = n
return nil
case *uint:
n, err := strconv.ParseUint(bytesToString(b), 10, 64)
if err != nil {
return err
}
*v = uint(n)
return nil
case *uint8:
n, err := strconv.ParseUint(bytesToString(b), 10, 8)
if err != nil {
return err
}
*v = uint8(n)
return nil
case *uint16:
n, err := strconv.ParseUint(bytesToString(b), 10, 16)
if err != nil {
return err
}
*v = uint16(n)
return nil
case *uint32:
n, err := strconv.ParseUint(bytesToString(b), 10, 32)
if err != nil {
return err
}
*v = uint32(n)
return nil
case *uint64:
n, err := strconv.ParseUint(bytesToString(b), 10, 64)
if err != nil {
return err
}
*v = n
return nil
case *float32:
n, err := strconv.ParseFloat(bytesToString(b), 32)
if err != nil {
return err
}
*v = float32(n)
return err
case *float64:
var err error
*v, err = strconv.ParseFloat(bytesToString(b), 64)
return err
case *bool:
*v = len(b) == 1 && b[0] == '1'
return nil
default:
if bu, ok := val.(binaryUnmarshaler); ok {
return bu.UnmarshalBinary(b)
}
err := fmt.Errorf(
"redis: can't unmarshal %T (consider implementing BinaryUnmarshaler)", val)
return err
}
}
//------------------------------------------------------------------------------
2015-09-03 17:55:31 +03:00
func readLine(cn *conn) ([]byte, error) {
line, isPrefix, err := cn.rd.ReadLine()
if err != nil {
return line, err
}
if isPrefix {
return line, errReaderTooSmall
}
return line, nil
}
2015-09-03 17:55:31 +03:00
func readN(cn *conn, n int) ([]byte, error) {
var b []byte
if cap(cn.buf) < n {
b = make([]byte, n)
} else {
b = cn.buf[:n]
}
2015-09-03 17:55:31 +03:00
_, err := io.ReadFull(cn.rd, b)
return b, err
}
//------------------------------------------------------------------------------
2015-10-07 17:09:20 +03:00
func parseErrorReply(cn *conn, line []byte) error {
return errorf(string(line[1:]))
}
func parseIntReply(cn *conn, line []byte) (int64, error) {
n, err := strconv.ParseInt(bytesToString(line[1:]), 10, 64)
if err != nil {
return 0, err
}
return n, nil
}
func readIntReply(cn *conn) (int64, error) {
2015-09-03 17:55:31 +03:00
line, err := readLine(cn)
2015-10-07 17:09:20 +03:00
if err != nil {
return 0, err
}
switch line[0] {
case errorReply:
return 0, parseErrorReply(cn, line)
case intReply:
return parseIntReply(cn, line)
default:
return 0, fmt.Errorf("readIntReply: can't parse %.100q", line)
}
}
func parseBytesReply(cn *conn, line []byte) ([]byte, error) {
if len(line) == 3 && line[1] == '-' && line[2] == '1' {
return nil, Nil
}
replyLen, err := strconv.Atoi(bytesToString(line[1:]))
if err != nil {
return nil, err
}
b, err := readN(cn, replyLen+2)
if err != nil {
2014-05-11 11:42:40 +04:00
return nil, err
}
2015-10-07 17:09:20 +03:00
return b[:replyLen], nil
}
func readBytesReply(cn *conn) ([]byte, error) {
line, err := readLine(cn)
if err != nil {
return nil, err
}
switch line[0] {
2015-10-07 17:09:20 +03:00
case errorReply:
return nil, parseErrorReply(cn, line)
case stringReply:
return parseBytesReply(cn, line)
default:
return nil, fmt.Errorf("readBytesReply: can't parse %.100q", line)
}
}
2015-10-07 17:09:20 +03:00
func readStringReply(cn *conn) (string, error) {
b, err := readBytesReply(cn)
if err != nil {
return "", err
}
return string(b), nil
}
2015-10-07 17:09:20 +03:00
func readFloatReply(cn *conn) (float64, error) {
b, err := readBytesReply(cn)
if err != nil {
return 0, err
}
return strconv.ParseFloat(bytesToString(b), 64)
}
2015-10-07 17:09:20 +03:00
func parseArrayHeader(cn *conn, line []byte) (int64, error) {
if len(line) == 3 && line[1] == '-' && line[2] == '1' {
return 0, Nil
}
2012-08-26 13:18:42 +04:00
2015-10-07 17:09:20 +03:00
n, err := strconv.ParseInt(bytesToString(line[1:]), 10, 64)
if err != nil {
return 0, err
}
return n, nil
}
func parseArrayReply(cn *conn, p multiBulkParser, line []byte) (interface{}, error) {
n, err := parseArrayHeader(cn, line)
if err != nil {
return nil, err
}
return p(cn, n)
}
func readArrayHeader(cn *conn) (int64, error) {
line, err := readLine(cn)
if err != nil {
return 0, err
}
switch line[0] {
case errorReply:
return 0, parseErrorReply(cn, line)
case arrayReply:
return parseArrayHeader(cn, line)
default:
return 0, fmt.Errorf("readArrayReply: can't parse %.100q", line)
}
}
func readArrayReply(cn *conn, p multiBulkParser) (interface{}, error) {
line, err := readLine(cn)
if err != nil {
return nil, err
}
switch line[0] {
case errorReply:
return nil, parseErrorReply(cn, line)
case arrayReply:
return parseArrayReply(cn, p, line)
default:
return nil, fmt.Errorf("readArrayReply: can't parse %.100q", line)
}
}
func readReply(cn *conn, p multiBulkParser) (interface{}, error) {
line, err := readLine(cn)
if err != nil {
return nil, err
}
switch line[0] {
case errorReply:
return nil, parseErrorReply(cn, line)
case statusReply:
return line[1:], nil
case intReply:
return parseIntReply(cn, line)
case stringReply:
return parseBytesReply(cn, line)
case arrayReply:
return parseArrayReply(cn, p, line)
2014-05-11 11:42:40 +04:00
}
return nil, fmt.Errorf("redis: can't parse %q", line)
}
2015-10-07 17:09:20 +03:00
func sliceParser(cn *conn, n int64) (interface{}, error) {
2014-05-11 11:42:40 +04:00
vals := make([]interface{}, 0, n)
for i := int64(0); i < n; i++ {
2015-10-07 17:09:20 +03:00
v, err := readReply(cn, sliceParser)
2014-05-11 11:42:40 +04:00
if err == Nil {
vals = append(vals, nil)
} else if err != nil {
return nil, err
} else {
switch vv := v.(type) {
case []byte:
vals = append(vals, string(vv))
default:
vals = append(vals, v)
}
2014-05-11 11:42:40 +04:00
}
}
return vals, nil
}
2015-10-07 17:09:20 +03:00
func intSliceParser(cn *conn, n int64) (interface{}, error) {
ints := make([]int64, 0, n)
2014-05-11 11:42:40 +04:00
for i := int64(0); i < n; i++ {
2015-10-07 17:09:20 +03:00
n, err := readIntReply(cn)
2014-05-11 11:42:40 +04:00
if err != nil {
return nil, err
}
2015-10-07 17:09:20 +03:00
ints = append(ints, n)
}
return ints, nil
}
func boolSliceParser(cn *conn, n int64) (interface{}, error) {
bools := make([]bool, 0, n)
for i := int64(0); i < n; i++ {
n, err := readIntReply(cn)
if err != nil {
return nil, err
2014-05-11 11:42:40 +04:00
}
2015-10-07 17:09:20 +03:00
bools = append(bools, n == 1)
2014-05-11 11:42:40 +04:00
}
2015-10-07 17:09:20 +03:00
return bools, nil
2014-05-11 11:42:40 +04:00
}
2015-10-07 17:09:20 +03:00
func stringSliceParser(cn *conn, n int64) (interface{}, error) {
ss := make([]string, 0, n)
2014-05-11 11:42:40 +04:00
for i := int64(0); i < n; i++ {
2015-10-07 17:09:20 +03:00
s, err := readStringReply(cn)
2014-05-11 11:42:40 +04:00
if err != nil {
return nil, err
}
2015-10-07 17:09:20 +03:00
ss = append(ss, s)
}
return ss, nil
}
func floatSliceParser(cn *conn, n int64) (interface{}, error) {
nn := make([]float64, 0, n)
for i := int64(0); i < n; i++ {
n, err := readFloatReply(cn)
if err != nil {
return nil, err
2014-05-11 11:42:40 +04:00
}
2015-10-07 17:09:20 +03:00
nn = append(nn, n)
2014-05-11 11:42:40 +04:00
}
2015-10-07 17:09:20 +03:00
return nn, nil
2014-05-11 11:42:40 +04:00
}
2015-10-07 17:09:20 +03:00
func stringStringMapParser(cn *conn, n int64) (interface{}, error) {
2014-05-11 11:42:40 +04:00
m := make(map[string]string, n/2)
for i := int64(0); i < n; i += 2 {
2015-10-07 17:09:20 +03:00
key, err := readStringReply(cn)
2014-05-11 11:42:40 +04:00
if err != nil {
return nil, err
}
2015-10-07 17:09:20 +03:00
value, err := readStringReply(cn)
2014-05-11 11:42:40 +04:00
if err != nil {
return nil, err
}
2015-10-07 17:09:20 +03:00
m[key] = value
2014-05-11 11:42:40 +04:00
}
return m, nil
}
2015-10-07 17:09:20 +03:00
func stringIntMapParser(cn *conn, n int64) (interface{}, error) {
2015-01-25 15:05:19 +03:00
m := make(map[string]int64, n/2)
for i := int64(0); i < n; i += 2 {
2015-10-07 17:09:20 +03:00
key, err := readStringReply(cn)
2015-01-25 15:05:19 +03:00
if err != nil {
return nil, err
}
2015-10-07 17:09:20 +03:00
n, err := readIntReply(cn)
2015-01-25 15:05:19 +03:00
if err != nil {
return nil, err
}
2015-10-07 17:09:20 +03:00
m[key] = n
2015-01-25 15:05:19 +03:00
}
return m, nil
}
2015-10-07 17:09:20 +03:00
func zSliceParser(cn *conn, n int64) (interface{}, error) {
2014-07-05 14:46:27 +04:00
zz := make([]Z, n/2)
2014-05-11 11:42:40 +04:00
for i := int64(0); i < n; i += 2 {
2015-10-07 17:09:20 +03:00
var err error
2014-07-05 14:46:27 +04:00
z := &zz[i/2]
2015-10-07 17:09:20 +03:00
z.Member, err = readStringReply(cn)
2014-05-11 11:42:40 +04:00
if err != nil {
return nil, err
}
2015-10-07 17:09:20 +03:00
z.Score, err = readFloatReply(cn)
2014-05-11 11:42:40 +04:00
if err != nil {
return nil, err
}
}
2014-07-05 14:46:27 +04:00
return zz, nil
}
2015-01-24 15:12:48 +03:00
2015-10-07 17:09:20 +03:00
func clusterSlotInfoSliceParser(cn *conn, n int64) (interface{}, error) {
2015-01-24 15:12:48 +03:00
infos := make([]ClusterSlotInfo, 0, n)
for i := int64(0); i < n; i++ {
2015-10-07 17:09:20 +03:00
n, err := readArrayHeader(cn)
2015-01-24 15:12:48 +03:00
if err != nil {
return nil, err
}
2015-10-07 17:09:20 +03:00
if n < 2 {
return nil, fmt.Errorf("got %d elements in cluster info, expected at least 2", n)
}
2015-01-24 15:12:48 +03:00
2015-10-07 17:09:20 +03:00
start, err := readIntReply(cn)
if err != nil {
return nil, err
2015-01-24 15:12:48 +03:00
}
2015-10-07 17:09:20 +03:00
end, err := readIntReply(cn)
if err != nil {
return nil, err
2015-01-24 15:12:48 +03:00
}
2015-10-07 17:09:20 +03:00
addrsn := n - 2
info := ClusterSlotInfo{
Start: int(start),
End: int(end),
Addrs: make([]string, addrsn),
2015-01-24 15:12:48 +03:00
}
2015-10-07 17:09:20 +03:00
for i := int64(0); i < addrsn; i++ {
n, err := readArrayHeader(cn)
if err != nil {
return nil, err
}
if n != 2 {
return nil, fmt.Errorf("got %d elements in cluster info address, expected 2", n)
2015-01-24 15:12:48 +03:00
}
2015-10-07 17:09:20 +03:00
ip, err := readStringReply(cn)
if err != nil {
return nil, err
2015-01-24 15:12:48 +03:00
}
2015-10-07 17:09:20 +03:00
port, err := readIntReply(cn)
if err != nil {
return nil, err
2015-01-24 15:12:48 +03:00
}
2015-10-07 17:09:20 +03:00
info.Addrs[i] = net.JoinHostPort(ip, strconv.FormatInt(port, 10))
2015-01-24 15:12:48 +03:00
}
2015-10-07 17:09:20 +03:00
2015-01-24 15:12:48 +03:00
infos = append(infos, info)
}
return infos, nil
}
2015-10-07 17:09:20 +03:00
func geoLocationParser(cn *conn, n int64) (interface{}, error) {
loc := &GeoLocation{}
var err error
loc.Name, err = readStringReply(cn)
if err != nil {
return nil, err
}
if n >= 2 {
loc.Distance, err = readFloatReply(cn)
if err != nil {
return nil, err
}
}
if n >= 3 {
loc.GeoHash, err = readIntReply(cn)
if err != nil {
return nil, err
}
}
if n >= 4 {
n, err := readArrayHeader(cn)
if err != nil {
return nil, err
}
if n != 2 {
return nil, fmt.Errorf("got %d coordinates, expected 2", n)
}
loc.Longitude, err = readFloatReply(cn)
if err != nil {
return nil, err
}
loc.Latitude, err = readFloatReply(cn)
if err != nil {
return nil, err
}
}
return loc, nil
}
func geoLocationSliceParser(cn *conn, n int64) (interface{}, error) {
locs := make([]GeoLocation, 0, n)
for i := int64(0); i < n; i++ {
v, err := readReply(cn, geoLocationParser)
if err != nil {
return nil, err
}
switch vv := v.(type) {
case []byte:
locs = append(locs, GeoLocation{
Name: string(vv),
})
case *GeoLocation:
locs = append(locs, *vv)
default:
return nil, fmt.Errorf("got %T, expected string or *GeoLocation", v)
}
}
return locs, nil
}