Project:
go memcached
Code Location:
git://github.com/dustin/gomemcached.gitmaster
mc_req.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
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
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
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
package gomemcached import ( "encoding/binary" "fmt" "io" ) // The maximum reasonable body length to expect. // Anything larger than this will result in an error. var MaxBodyLen = int(1e6) // A Memcached Request type MCRequest struct { // The command being issued Opcode CommandCode // The CAS (if applicable, or 0) Cas uint64 // An opaque value to be returned with this request Opaque uint32 // The vbucket to which this command belongs VBucket uint16 // Command extras, key, and body Extras, Key, Body []byte } // The number of bytes this request requires. func (req *MCRequest) Size() int { return HDR_LEN + len(req.Extras) + len(req.Key) + len(req.Body) } // A debugging string representation of this request func (req MCRequest) String() string { return fmt.Sprintf("{MCRequest opcode=%s, bodylen=%d, key='%s'}", req.Opcode, len(req.Body), req.Key) } func (req *MCRequest) fillHeaderBytes(data []byte) int { pos := 0 data[pos] = REQ_MAGIC pos++ data[pos] = byte(req.Opcode) pos++ binary.BigEndian.PutUint16(data[pos:pos+2], uint16(len(req.Key))) pos += 2 // 4 data[pos] = byte(len(req.Extras)) pos++ // data[pos] = 0 pos++ binary.BigEndian.PutUint16(data[pos:pos+2], req.VBucket) pos += 2 // 8 binary.BigEndian.PutUint32(data[pos:pos+4], uint32(len(req.Body)+len(req.Key)+len(req.Extras))) pos += 4 // 12 binary.BigEndian.PutUint32(data[pos:pos+4], req.Opaque) pos += 4 // 16 if req.Cas != 0 { binary.BigEndian.PutUint64(data[pos:pos+8], req.Cas) } pos += 8 if len(req.Extras) > 0 { copy(data[pos:pos+len(req.Extras)], req.Extras) pos += len(req.Extras) } if len(req.Key) > 0 { copy(data[pos:pos+len(req.Key)], req.Key) pos += len(req.Key) } return pos } // The wire representation of the header (with the extras and key) func (req *MCRequest) HeaderBytes() []byte { data := make([]byte, HDR_LEN+len(req.Extras)+len(req.Key)) req.fillHeaderBytes(data) return data } // The wire representation of this request. func (req *MCRequest) Bytes() []byte { data := make([]byte, req.Size()) pos := req.fillHeaderBytes(data) if len(req.Body) > 0 { copy(data[pos:pos+len(req.Body)], req.Body) } return data } // Send this request message across a writer. func (req *MCRequest) Transmit(w io.Writer) (err error) { if len(req.Body) < 128 { _, err = w.Write(req.Bytes()) } else { _, err = w.Write(req.HeaderBytes()) if err == nil { _, err = w.Write(req.Body) } } return } // Fill this MCRequest with the data from this reader. func (req *MCRequest) Receive(r io.Reader, hdrBytes []byte) error { if len(hdrBytes) < HDR_LEN { hdrBytes = make([]byte, HDR_LEN) } _, err := io.ReadFull(r, hdrBytes) if err != nil { return err } if hdrBytes[0] != RES_MAGIC && hdrBytes[0] != REQ_MAGIC { return fmt.Errorf("Bad magic: 0x%02x", hdrBytes[0]) } klen := int(binary.BigEndian.Uint16(hdrBytes[2:])) elen := int(hdrBytes[4]) req.Opcode = CommandCode(hdrBytes[1]) // Vbucket at 6:7 req.VBucket = binary.BigEndian.Uint16(hdrBytes[6:]) bodyLen := int(binary.BigEndian.Uint32(hdrBytes[8:]) - uint32(klen) - uint32(elen)) if bodyLen > MaxBodyLen { return fmt.Errorf("%d is too big (max %s)", bodyLen, MaxBodyLen) } req.Opaque = binary.BigEndian.Uint32(hdrBytes[12:]) req.Cas = binary.BigEndian.Uint64(hdrBytes[16:]) buf := make([]byte, klen+elen+bodyLen) _, err = io.ReadFull(r, buf) if err == nil { req.Extras = buf[0:elen] req.Key = buf[elen : klen+elen] req.Body = buf[klen+elen:] } return err }
