forked from docker-archive/go-redis-server
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathparser.go
115 lines (97 loc) · 2.6 KB
/
parser.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
package redis
import (
"bufio"
"fmt"
"io"
"io/ioutil"
"strings"
)
func parseRequest(conn io.ReadCloser) (*Request, error) {
r := bufio.NewReader(conn)
// first line of redis request should be:
// *<number of arguments>CRLF
line, err := r.ReadString('\n')
if err != nil {
return nil, err
}
// note that this line also protects us from negative integers
var argsCount int
// Multiline request:
if line[0] == '*' {
if _, err := fmt.Sscanf(line, "*%d\r", &argsCount); err != nil {
return nil, malformed("*<numberOfArguments>", line)
}
// All next lines are pairs of:
//$<number of bytes of argument 1> CR LF
//<argument data> CR LF
// first argument is a command name, so just convert
firstArg, err := readArgument(r)
if err != nil {
return nil, err
}
args := make([][]byte, argsCount-1)
for i := 0; i < argsCount-1; i += 1 {
if args[i], err = readArgument(r); err != nil {
return nil, err
}
}
return &Request{
Name: strings.ToLower(string(firstArg)),
Args: args,
Body: conn,
}, nil
}
// Inline request:
fields := strings.Split(strings.Trim(line, "\r\n"), " ")
var args [][]byte
if len(fields) > 1 {
for _, arg := range fields[1:] {
args = append(args, []byte(arg))
}
}
return &Request{
Name: strings.ToLower(string(fields[0])),
Args: args,
Body: conn,
}, nil
}
func readArgument(r *bufio.Reader) ([]byte, error) {
line, err := r.ReadString('\n')
if err != nil {
return nil, malformed("$<argumentLength>", line)
}
var argSize int
if _, err := fmt.Sscanf(line, "$%d\r", &argSize); err != nil {
return nil, malformed("$<argumentSize>", line)
}
// I think int is safe here as the max length of request
// should be less then max int value?
data, err := ioutil.ReadAll(io.LimitReader(r, int64(argSize)))
if err != nil {
return nil, err
}
if len(data) != argSize {
return nil, malformedLength(argSize, len(data))
}
// Now check for trailing CR
if b, err := r.ReadByte(); err != nil || b != '\r' {
return nil, malformedMissingCRLF()
}
// And LF
if b, err := r.ReadByte(); err != nil || b != '\n' {
return nil, malformedMissingCRLF()
}
return data, nil
}
func malformed(expected string, got string) error {
Debugf("Mailformed request:'%s does not match %s\\r\\n'", got, expected)
return fmt.Errorf("Mailformed request:'%s does not match %s\\r\\n'", got, expected)
}
func malformedLength(expected int, got int) error {
return fmt.Errorf(
"Mailformed request: argument length '%d does not match %d\\r\\n'",
got, expected)
}
func malformedMissingCRLF() error {
return fmt.Errorf("Mailformed request: line should end with \\r\\n")
}