package main import ( "bufio" "fmt" "io" "strconv" ) var ( start_string byte = '+' start_error byte = '-' start_integer byte = ':' start_bulkstring byte = '$' start_array byte = '*' ) type value interface { } func readValue(r io.Reader) (value, error) { var br *bufio.Reader switch t := r.(type) { case *bufio.Reader: br = t default: br = bufio.NewReader(r) } line, err := br.ReadBytes('\n') switch err { case io.EOF: if line != nil { break } return nil, err case nil: break default: return nil, fmt.Errorf("unable to read value in redis protocol: %v") } if len(line) < 3 { return nil, fmt.Errorf("unable to read redis protocol value: input %q is too small", line) } if line[len(line)-2] != '\r' { return nil, fmt.Errorf("unable to read redis protocol value: bad line terminator") } line = line[:len(line)-2] switch line[0] { case start_string: return readString(line[1:]) case start_error: return readError(line[1:]) case start_integer: return readInteger(line[1:]) case start_bulkstring: return readBulkString(line[1:], br) case start_array: return readArray(line[1:], br) default: return nil, fmt.Errorf("unable to read redis protocol value: illegal start character: %c", line[0]) } } // ------------------------------------------------------------------------------ type String string func readString(b []byte) (value, error) { return String(b), nil } // ------------------------------------------------------------------------------ type Error string func readError(b []byte) (value, error) { return Error(b), nil } // ------------------------------------------------------------------------------ type Integer int64 func readInteger(b []byte) (value, error) { i, err := strconv.ParseInt(string(b), 10, 64) if err != nil { return nil, fmt.Errorf("unable to read integer in redis protocol format: %v", err) } return Integer(i), nil } // ----------------------------------------------------------------------------------------- type BulkString string func readBulkString(prefix []byte, r io.Reader) (value, error) { n, err := strconv.ParseInt(string(prefix), 10, 64) if err != nil { return nil, fmt.Errorf("unable to read bulkstring in redis protocol: bad prefix: %v", err) } if n == -1 { return nil, nil } n += 2 b := make([]byte, n) n_read, err := r.Read(b) switch err { case io.EOF, nil: break default: return nil, fmt.Errorf("unable to read bulkstring in redis protocol: error on read: %v", err) } if int64(n_read) != n { return nil, fmt.Errorf("unable to read bulkstring in redis protocol: read %d bytes, expected to read %d bytes", int64(n_read), n) } if len(b) < 2 { return nil, fmt.Errorf("unable to read bulkstring in redis protocol: input %q is too short", b) } return BulkString(b[:len(b)-2]), nil } // ----------------------------------------------------------------------------------------- type Array []value func readArray(prefix []byte, r *bufio.Reader) (value, error) { n, err := strconv.ParseInt(string(prefix), 10, 64) if err != nil { return nil, fmt.Errorf("unable to read array in redis protocol: bad prefix: %v", err) } a := make(Array, n) for i := int64(0); i < n; i++ { v, err := readValue(r) if err != nil { return nil, fmt.Errorf("unable to read array value in redis protocol: %v", err) } a[i] = v } return a, nil }