Module: EventMachine::Protocols::Couchbase::Packet

Defined in:
lib/em-couchbase/packet.rb

Constant Summary collapse

REQUEST_HEADER_FMT =

uint8_t magic

uint8_t   opcode
uint16_t  keylen
uint8_t   extlen
uint8_t   datatype
uint16_t  vbucket
uint32_t  bodylen
uint32_t  opaque
uint64_t  cas
"CCnCCnNNQ"
REQUEST_HEADER_SIZE =

bytes

24
RESPONSE_HEADER_FMT =

Sum of lengths of fields below

uint8_t   magic
uint8_t   opcode
uint16_t  keylen
uint8_t   extlen
uint8_t   datatype
uint16_t  status
uint32_t  bodylen
uint32_t  opaque
uint64_t  cas
"CCnCCnNNQ"
RESPONSE_HEADER_SIZE =

bytes

24
CMD_GET =
0x00
CMD_SET =
0x01
CMD_ADD =
0x02
CMD_REPLACE =
0x03
CMD_DELETE =
0x04
CMD_INCREMENT =
0x05
CMD_DECREMENT =
0x06
CMD_QUIT =
0x07
CMD_FLUSH =
0x08
CMD_GETQ =
0x09
CMD_NOOP =
0x0a
CMD_VERSION =
0x0b
CMD_GETK =
0x0c
CMD_GETKQ =
0x0d
CMD_APPEND =
0x0e
CMD_PREPEND =
0x0f
CMD_STAT =
0x10
CMD_SETQ =
0x11
CMD_ADDQ =
0x12
CMD_REPLACEQ =
0x13
CMD_DELETEQ =
0x14
CMD_INCREMENTQ =
0x15
CMD_DECREMENTQ =
0x16
CMD_QUITQ =
0x17
CMD_FLUSHQ =
0x18
CMD_APPENDQ =
0x19
CMD_PREPENDQ =
0x1a
CMD_VERBOSITY =
0x1b
CMD_TOUCH =
0x1c
CMD_GAT =
0x1d
CMD_GATQ =
0x1e
CMD_SASL_LIST_MECHS =
0x20
CMD_SASL_AUTH =
0x21
CMD_SASL_STEP =
0x22

Class Method Summary collapse

Class Method Details

.build(opaque, vbucket, opcode, *args) ⇒ Object

[View source]

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
158
159
160
161
162
163
# File 'lib/em-couchbase/packet.rb', line 85

def self.build(opaque, vbucket, opcode, *args)
  case opcode
  when :set
    key, value, flags, expiration, cas = args.shift(5)
    bodylen = key.size + value.size + 8
    [
      0x80,             # uint8_t   magic
      CMD_SET,          # uint8_t   opcode
      key.size,         # uint16_t  keylen
      8,                # uint8_t   extlen (flags + expiration)
      0,                # uint8_t   datatype
      vbucket,          # uint16_t  vbucket
      bodylen,          # uint32_t  bodylen
      opaque || 0,      # uint32_t  opaque
      cas || 0,         # uint64_t  cas
      flags || 0,       # uint32_t  flags
      expiration || 0,  # uint32_t  expiration
      key,
      value
    ].pack("#{REQUEST_HEADER_FMT}NNa*a*")
  when :get
    key = args.shift
    bodylen = key.size
    [
      0x80,             # uint8_t   magic
      CMD_GET,          # uint8_t   opcode
      key.size,         # uint16_t  keylen
      0,                # uint8_t   extlen
      0,                # uint8_t   datatype
      vbucket,          # uint16_t  vbucket
      bodylen,          # uint32_t  bodylen
      opaque || 0,      # uint32_t  opaque
      0,                # uint64_t  cas
      key
    ].pack("#{REQUEST_HEADER_FMT}a*")
  when :incr, :decr
    cmd_id = opcode == :incr ? CMD_INCREMENT : CMD_DECREMENT
    key, delta, initial, expiration, cas = args.shift(5)
    delta ||= 1
    initial ||= 0
    bodylen = key.size + 20
    [
      0x80,                   # uint8_t   magic
      cmd_id,                 # uint8_t   opcode
      key.size,               # uint16_t  keylen
      20,                     # uint8_t   extlen (delta + initial + expiration)
      0,                      # uint8_t   datatype
      vbucket,                # uint16_t  vbucket
      bodylen,                # uint32_t  bodylen
      opaque || 0,            # uint32_t  opaque
      cas || 0,               # uint64_t  cas
      delta >> 32,            # uint64_t
      delta & 0xffffffff,     #
      initial >> 32,          # uint64_t
      initial & 0xffffffff,   #
      expiration || 0,        # uint32_t
      key
    ].pack("#{REQUEST_HEADER_FMT}NNNNNa*")
  when :sasl_auth
    mech, username, password = args.shift(3)
    value = "\0#{username}\0#{password}"
    bodylen = mech.size + value.size
    [
      0x80,             # uint8_t   magic
      CMD_SASL_AUTH,    # uint8_t   opcode
      mech.size,        # uint16_t  keylen
      0,                # uint8_t   extlen
      0,                # uint8_t   datatype
      0,                # uint16_t  vbucket
      bodylen,          # uint32_t  bodylen
      0,                # uint32_t  opaque
      0,                # uint64_t  cas
      mech,
      value
    ].pack("#{REQUEST_HEADER_FMT}a*a*")
  else
    raise Couchbase::Error::UnknownCommand, [opcode, *args].inspect
  end
end

.parse(data) ⇒ Object

[View source]

165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
# File 'lib/em-couchbase/packet.rb', line 165

def self.parse(data)
  while data.length >= RESPONSE_HEADER_SIZE
    header = data[0...RESPONSE_HEADER_SIZE]
    ( magic,
      opcode,
      keylen,
      extlen,
      datatype,
      status,
      bodylen,
      opaque,
      cas ) = header.unpack(RESPONSE_HEADER_FMT)

    if magic != 0x81
      fail Couchbase::Error::Protocol.new "Broken packet: #{header.inspect}"
    end

    if data.size < bodylen + RESPONSE_HEADER_SIZE
      return  # need moar data
    else
      data[0...RESPONSE_HEADER_SIZE] = ""
    end

    ext = data[0...extlen]
    data[0...extlen] = ""

    key = data[0...keylen]
    data[0...keylen] = ""

    vallen = bodylen - extlen - keylen
    body = data[0...vallen]
    data[0...vallen] = ""

    result = Result.new(:key => key,
                        :value => body,
                        :status => status,
                        :cas => cas)

    case opcode
    when CMD_SET
      result.operation = :set
    when CMD_SASL_AUTH
      result.operation = :sasl_auth
    when CMD_GET
      result.operation = :get
      result.flags, _ = ext.unpack("N")
    when CMD_INCREMENT
      result.operation = :incr
      hi, lo = result.value.unpack("NN")
      result.value = hi << 32 | lo
    when CMD_DECREMENT
      result.operation = :decr
      hi, lo = result.value.unpack("NN")
      result.value = hi << 32 | lo
    else
      raise Couchbase::Error::UnknownCommand, header.inspect
    end

    if error_class = Couchbase::Error.map_error_code(status)
      result.error = error_class.new(body)
      result.error.error = status
      result.error.key = key
      result.error.cas = cas
      result.error.operation = result.operation
    end
    yield(result.operation, opaque, result)
  end
end