dalli 3.0.6 → 3.1.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of dalli might be problematic. Click here for more details.
- checksums.yaml +4 -4
- data/History.md +29 -0
- data/lib/dalli/client.rb +163 -261
- data/lib/dalli/options.rb +3 -3
- data/lib/dalli/pipelined_getter.rb +177 -0
- data/lib/dalli/protocol/base.rb +238 -0
- data/lib/dalli/protocol/binary/request_formatter.rb +11 -3
- data/lib/dalli/protocol/binary/response_header.rb +36 -0
- data/lib/dalli/protocol/binary/response_processor.rb +101 -42
- data/lib/dalli/protocol/binary/sasl_authentication.rb +1 -1
- data/lib/dalli/protocol/binary.rb +81 -445
- data/lib/dalli/protocol/connection_manager.rb +242 -0
- data/lib/dalli/protocol/response_buffer.rb +53 -0
- data/lib/dalli/protocol/server_config_parser.rb +7 -7
- data/lib/dalli/ring.rb +5 -1
- data/lib/dalli/socket.rb +8 -6
- data/lib/dalli/version.rb +1 -1
- data/lib/dalli.rb +9 -0
- data/lib/rack/session/dalli.rb +83 -74
- metadata +15 -4
@@ -1,14 +1,9 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
require 'English'
|
4
3
|
require 'forwardable'
|
5
4
|
require 'socket'
|
6
5
|
require 'timeout'
|
7
6
|
|
8
|
-
require_relative 'binary/request_formatter'
|
9
|
-
require_relative 'binary/response_processor'
|
10
|
-
require_relative 'binary/sasl_authentication'
|
11
|
-
|
12
7
|
module Dalli
|
13
8
|
module Protocol
|
14
9
|
##
|
@@ -16,383 +11,70 @@ module Dalli
|
|
16
11
|
# protocol. Contains logic for managing connection state to the server (retries, etc),
|
17
12
|
# formatting requests to the server, and unpacking responses.
|
18
13
|
##
|
19
|
-
class Binary
|
20
|
-
|
21
|
-
|
22
|
-
attr_accessor :hostname, :port, :weight, :options
|
23
|
-
attr_reader :sock, :socket_type
|
24
|
-
|
25
|
-
def_delegators :@value_marshaller, :serializer, :compressor, :compression_min_size, :compress_by_default?
|
26
|
-
|
27
|
-
DEFAULTS = {
|
28
|
-
# seconds between trying to contact a remote server
|
29
|
-
down_retry_delay: 30,
|
30
|
-
# connect/read/write timeout for socket operations
|
31
|
-
socket_timeout: 1,
|
32
|
-
# times a socket operation may fail before considering the server dead
|
33
|
-
socket_max_failures: 2,
|
34
|
-
# amount of time to sleep between retries when a failure occurs
|
35
|
-
socket_failure_delay: 0.1,
|
36
|
-
username: nil,
|
37
|
-
password: nil
|
38
|
-
}.freeze
|
39
|
-
|
40
|
-
def initialize(attribs, options = {})
|
41
|
-
@hostname, @port, @weight, @socket_type, options = ServerConfigParser.parse(attribs, options)
|
42
|
-
@options = DEFAULTS.merge(options)
|
43
|
-
@value_marshaller = ValueMarshaller.new(@options)
|
44
|
-
@response_processor = ResponseProcessor.new(self, @value_marshaller)
|
45
|
-
|
46
|
-
reset_down_info
|
47
|
-
@sock = nil
|
48
|
-
@pid = nil
|
49
|
-
@request_in_progress = false
|
50
|
-
end
|
51
|
-
|
52
|
-
def name
|
53
|
-
if socket_type == :unix
|
54
|
-
hostname
|
55
|
-
else
|
56
|
-
"#{hostname}:#{port}"
|
57
|
-
end
|
58
|
-
end
|
59
|
-
|
60
|
-
# Chokepoint method for error handling and ensuring liveness
|
61
|
-
def request(opcode, *args)
|
62
|
-
verify_state
|
63
|
-
# The alive? call has the side effect of connecting the underlying
|
64
|
-
# socket if it is not connected, or there's been a disconnect
|
65
|
-
# because of timeout or other error. Method raises an error
|
66
|
-
# if it can't connect
|
67
|
-
raise_memcached_down_err unless alive?
|
68
|
-
|
69
|
-
begin
|
70
|
-
send(opcode, *args)
|
71
|
-
rescue Dalli::MarshalError => e
|
72
|
-
log_marshall_err(args.first, e)
|
73
|
-
raise
|
74
|
-
rescue Dalli::DalliError, Dalli::NetworkError, Dalli::ValueOverMaxSize, Timeout::Error
|
75
|
-
raise
|
76
|
-
rescue StandardError => e
|
77
|
-
log_unexpected_err(e)
|
78
|
-
down!
|
79
|
-
end
|
80
|
-
end
|
81
|
-
|
82
|
-
def raise_memcached_down_err
|
83
|
-
raise Dalli::NetworkError,
|
84
|
-
"#{name} is down: #{@error} #{@msg}. If you are sure it is running, "\
|
85
|
-
"ensure memcached version is > #{::Dalli::MIN_SUPPORTED_MEMCACHED_VERSION}."
|
86
|
-
end
|
87
|
-
|
88
|
-
def log_marshall_err(key, err)
|
89
|
-
Dalli.logger.error "Marshalling error for key '#{key}': #{err.message}"
|
90
|
-
Dalli.logger.error 'You are trying to cache a Ruby object which cannot be serialized to memcached.'
|
91
|
-
end
|
92
|
-
|
93
|
-
def log_unexpected_err(err)
|
94
|
-
Dalli.logger.error "Unexpected exception during Dalli request: #{err.class.name}: #{err.message}"
|
95
|
-
Dalli.logger.error err.backtrace.join("\n\t")
|
96
|
-
end
|
97
|
-
|
98
|
-
# The socket connection to the underlying server is initialized as a side
|
99
|
-
# effect of this call. In fact, this is the ONLY place where that
|
100
|
-
# socket connection is initialized.
|
101
|
-
def alive?
|
102
|
-
return true if @sock
|
103
|
-
return false unless reconnect_down_server?
|
104
|
-
|
105
|
-
connect
|
106
|
-
!!@sock
|
107
|
-
rescue Dalli::NetworkError
|
108
|
-
false
|
109
|
-
end
|
110
|
-
|
111
|
-
def reconnect_down_server?
|
112
|
-
return true unless @last_down_at
|
113
|
-
|
114
|
-
time_to_next_reconnect = @last_down_at + options[:down_retry_delay] - Time.now
|
115
|
-
return true unless time_to_next_reconnect.positive?
|
116
|
-
|
117
|
-
Dalli.logger.debug do
|
118
|
-
format('down_retry_delay not reached for %<name>s (%<time>.3f seconds left)', name: name,
|
119
|
-
time: time_to_next_reconnect)
|
120
|
-
end
|
121
|
-
false
|
122
|
-
end
|
123
|
-
|
124
|
-
# Closes the underlying socket and cleans up
|
125
|
-
# socket state.
|
126
|
-
def close
|
127
|
-
return unless @sock
|
128
|
-
|
129
|
-
begin
|
130
|
-
@sock.close
|
131
|
-
rescue StandardError
|
132
|
-
nil
|
133
|
-
end
|
134
|
-
@sock = nil
|
135
|
-
@pid = nil
|
136
|
-
abort_request!
|
137
|
-
end
|
138
|
-
|
139
|
-
def lock!; end
|
140
|
-
|
141
|
-
def unlock!; end
|
142
|
-
|
143
|
-
# Start reading key/value pairs from this connection. This is usually called
|
144
|
-
# after a series of GETKQ commands. A NOOP is sent, and the server begins
|
145
|
-
# flushing responses for kv pairs that were found.
|
146
|
-
#
|
147
|
-
# Returns nothing.
|
148
|
-
def multi_response_start
|
149
|
-
verify_state
|
150
|
-
write_noop
|
151
|
-
@multi_buffer = +''
|
152
|
-
@position = 0
|
153
|
-
start_request!
|
154
|
-
end
|
155
|
-
|
156
|
-
# Did the last call to #multi_response_start complete successfully?
|
157
|
-
def multi_response_completed?
|
158
|
-
@multi_buffer.nil?
|
159
|
-
end
|
160
|
-
|
161
|
-
# Attempt to receive and parse as many key/value pairs as possible
|
162
|
-
# from this server. After #multi_response_start, this should be invoked
|
163
|
-
# repeatedly whenever this server's socket is readable until
|
164
|
-
# #multi_response_completed?.
|
165
|
-
#
|
166
|
-
# Returns a Hash of kv pairs received.
|
167
|
-
def multi_response_nonblock
|
168
|
-
reconnect! 'multi_response has completed' if @multi_buffer.nil?
|
169
|
-
|
170
|
-
@multi_buffer << @sock.read_available
|
171
|
-
buf = @multi_buffer
|
172
|
-
pos = @position
|
173
|
-
values = {}
|
174
|
-
|
175
|
-
while buf.bytesize - pos >= ResponseProcessor::RESP_HEADER_SIZE
|
176
|
-
header = buf.slice(pos, ResponseProcessor::RESP_HEADER_SIZE)
|
177
|
-
_, extra_len, key_len, body_len, cas = @response_processor.unpack_header(header)
|
178
|
-
|
179
|
-
# We've reached the noop at the end of the pipeline
|
180
|
-
if key_len.zero?
|
181
|
-
finish_multi_response
|
182
|
-
break
|
183
|
-
end
|
184
|
-
|
185
|
-
# Break and read more unless we already have the entire response for this header
|
186
|
-
resp_size = ResponseProcessor::RESP_HEADER_SIZE + body_len
|
187
|
-
break unless buf.bytesize - pos >= resp_size
|
188
|
-
|
189
|
-
body = buf.slice(pos + ResponseProcessor::RESP_HEADER_SIZE, body_len)
|
190
|
-
begin
|
191
|
-
key, value = @response_processor.unpack_response_body(extra_len, key_len, body, true)
|
192
|
-
values[key] = [value, cas]
|
193
|
-
rescue DalliError
|
194
|
-
# TODO: Determine if we should be swallowing
|
195
|
-
# this error
|
196
|
-
end
|
197
|
-
|
198
|
-
pos = pos + ResponseProcessor::RESP_HEADER_SIZE + body_len
|
199
|
-
end
|
200
|
-
# TODO: We should be discarding the already processed buffer at this point
|
201
|
-
@position = pos
|
202
|
-
|
203
|
-
values
|
204
|
-
rescue SystemCallError, Timeout::Error, EOFError => e
|
205
|
-
failure!(e)
|
206
|
-
end
|
207
|
-
|
208
|
-
def finish_multi_response
|
209
|
-
@multi_buffer = nil
|
210
|
-
@position = nil
|
211
|
-
finish_request!
|
212
|
-
end
|
213
|
-
|
214
|
-
# Abort an earlier #multi_response_start. Used to signal an external
|
215
|
-
# timeout. The underlying socket is disconnected, and the exception is
|
216
|
-
# swallowed.
|
217
|
-
#
|
218
|
-
# Returns nothing.
|
219
|
-
def multi_response_abort
|
220
|
-
@multi_buffer = nil
|
221
|
-
@position = nil
|
222
|
-
abort_request!
|
223
|
-
return true unless @sock
|
224
|
-
|
225
|
-
failure!(RuntimeError.new('External timeout'))
|
226
|
-
rescue NetworkError
|
227
|
-
true
|
228
|
-
end
|
229
|
-
|
230
|
-
def read(count)
|
231
|
-
start_request!
|
232
|
-
data = @sock.readfull(count)
|
233
|
-
finish_request!
|
234
|
-
data
|
235
|
-
rescue SystemCallError, Timeout::Error, EOFError => e
|
236
|
-
failure!(e)
|
237
|
-
end
|
238
|
-
|
239
|
-
def write(bytes)
|
240
|
-
start_request!
|
241
|
-
result = @sock.write(bytes)
|
242
|
-
finish_request!
|
243
|
-
result
|
244
|
-
rescue SystemCallError, Timeout::Error => e
|
245
|
-
failure!(e)
|
246
|
-
end
|
247
|
-
|
248
|
-
def socket_timeout
|
249
|
-
@socket_timeout ||= @options[:socket_timeout]
|
14
|
+
class Binary < Base
|
15
|
+
def response_processor
|
16
|
+
@response_processor ||= ResponseProcessor.new(@connection_manager, @value_marshaller)
|
250
17
|
end
|
251
18
|
|
252
|
-
# NOTE: Additional public methods should be overridden in Dalli::Threadsafe
|
253
|
-
|
254
19
|
private
|
255
20
|
|
256
|
-
|
257
|
-
|
258
|
-
|
259
|
-
|
260
|
-
def start_request!
|
261
|
-
@request_in_progress = true
|
262
|
-
end
|
263
|
-
|
264
|
-
def finish_request!
|
265
|
-
@request_in_progress = false
|
266
|
-
end
|
267
|
-
|
268
|
-
def abort_request!
|
269
|
-
@request_in_progress = false
|
270
|
-
end
|
271
|
-
|
272
|
-
def verify_state
|
273
|
-
failure!(RuntimeError.new('Already writing to socket')) if request_in_progress?
|
274
|
-
reconnect_on_fork if fork_detected?
|
275
|
-
end
|
276
|
-
|
277
|
-
def fork_detected?
|
278
|
-
@pid && @pid != Process.pid
|
279
|
-
end
|
280
|
-
|
281
|
-
def reconnect_on_fork
|
282
|
-
message = 'Fork detected, re-connecting child process...'
|
283
|
-
Dalli.logger.info { message }
|
284
|
-
reconnect! message
|
285
|
-
end
|
286
|
-
|
287
|
-
# Marks the server instance as needing reconnect. Raises a
|
288
|
-
# Dalli::NetworkError with the specified message. Calls close
|
289
|
-
# to clean up socket state
|
290
|
-
def reconnect!(message)
|
291
|
-
close
|
292
|
-
sleep(options[:socket_failure_delay]) if options[:socket_failure_delay]
|
293
|
-
raise Dalli::NetworkError, message
|
294
|
-
end
|
295
|
-
|
296
|
-
# Raises Dalli::NetworkError
|
297
|
-
def failure!(exception)
|
298
|
-
message = "#{name} failed (count: #{@fail_count}) #{exception.class}: #{exception.message}"
|
299
|
-
Dalli.logger.warn { message }
|
300
|
-
|
301
|
-
@fail_count += 1
|
302
|
-
if @fail_count >= options[:socket_max_failures]
|
303
|
-
down!
|
304
|
-
else
|
305
|
-
reconnect! 'Socket operation failed, retrying...'
|
306
|
-
end
|
307
|
-
end
|
308
|
-
|
309
|
-
# Marks the server instance as down. Updates the down_at state
|
310
|
-
# and raises an Dalli::NetworkError that includes the underlying
|
311
|
-
# error in the message. Calls close to clean up socket state
|
312
|
-
def down!
|
313
|
-
close
|
314
|
-
log_down_detected
|
315
|
-
|
316
|
-
@error = $ERROR_INFO&.class&.name
|
317
|
-
@msg ||= $ERROR_INFO&.message
|
318
|
-
raise Dalli::NetworkError, "#{name} is down: #{@error} #{@msg}"
|
319
|
-
end
|
320
|
-
|
321
|
-
def log_down_detected
|
322
|
-
@last_down_at = Time.now
|
323
|
-
|
324
|
-
if @down_at
|
325
|
-
time = Time.now - @down_at
|
326
|
-
Dalli.logger.debug { format('%<name>s is still down (for %<time>.3f seconds now)', name: name, time: time) }
|
327
|
-
else
|
328
|
-
@down_at = @last_down_at
|
329
|
-
Dalli.logger.warn("#{name} is down")
|
330
|
-
end
|
331
|
-
end
|
332
|
-
|
333
|
-
def log_up_detected
|
334
|
-
return unless @down_at
|
21
|
+
ALLOWED_QUIET_OPS = %i[add replace set delete incr decr append prepend flush noop].freeze
|
22
|
+
def verify_allowed_quiet!(opkey)
|
23
|
+
return if ALLOWED_QUIET_OPS.include?(opkey)
|
335
24
|
|
336
|
-
|
337
|
-
Dalli.logger.warn { format('%<name>s is back (downtime was %<time>.3f seconds)', name: name, time: time) }
|
25
|
+
raise Dalli::NotPermittedMultiOpError, "The operation #{opkey} is not allowed in a quiet block."
|
338
26
|
end
|
339
27
|
|
340
|
-
|
341
|
-
|
342
|
-
|
28
|
+
# Retrieval Commands
|
29
|
+
def get(key, options = nil)
|
30
|
+
req = RequestFormatter.standard_request(opkey: :get, key: key)
|
31
|
+
write(req)
|
32
|
+
response_processor.generic_response(unpack: true, cache_nils: cache_nils?(options))
|
343
33
|
end
|
344
34
|
|
345
|
-
def
|
346
|
-
|
347
|
-
@down_at = nil
|
348
|
-
@last_down_at = nil
|
349
|
-
@msg = nil
|
350
|
-
@error = nil
|
35
|
+
def quiet_get_request(key)
|
36
|
+
RequestFormatter.standard_request(opkey: :getkq, key: key)
|
351
37
|
end
|
352
38
|
|
353
|
-
def
|
354
|
-
|
39
|
+
def gat(key, ttl, options = nil)
|
40
|
+
ttl = TtlSanitizer.sanitize(ttl)
|
41
|
+
req = RequestFormatter.standard_request(opkey: :gat, key: key, ttl: ttl)
|
42
|
+
write(req)
|
43
|
+
response_processor.generic_response(unpack: true, cache_nils: cache_nils?(options))
|
355
44
|
end
|
356
45
|
|
357
|
-
def
|
358
|
-
|
359
|
-
|
360
|
-
|
46
|
+
def touch(key, ttl)
|
47
|
+
ttl = TtlSanitizer.sanitize(ttl)
|
48
|
+
write(RequestFormatter.standard_request(opkey: :touch, key: key, ttl: ttl))
|
49
|
+
response_processor.generic_response
|
361
50
|
end
|
362
51
|
|
363
|
-
|
52
|
+
# TODO: This is confusing, as there's a cas command in memcached
|
53
|
+
# and this isn't it. Maybe rename? Maybe eliminate?
|
54
|
+
def cas(key)
|
364
55
|
req = RequestFormatter.standard_request(opkey: :get, key: key)
|
365
56
|
write(req)
|
366
|
-
|
367
|
-
end
|
368
|
-
|
369
|
-
def send_multiget(keys)
|
370
|
-
req = +''
|
371
|
-
keys.each do |key|
|
372
|
-
req << RequestFormatter.standard_request(opkey: :getkq, key: key)
|
373
|
-
end
|
374
|
-
# Could send noop here instead of in multi_response_start
|
375
|
-
write(req)
|
57
|
+
response_processor.data_cas_response
|
376
58
|
end
|
377
59
|
|
60
|
+
# Storage Commands
|
378
61
|
def set(key, value, ttl, cas, options)
|
379
|
-
opkey =
|
380
|
-
|
62
|
+
opkey = quiet? ? :setq : :set
|
63
|
+
storage_req(opkey, key, value, ttl, cas, options)
|
381
64
|
end
|
382
65
|
|
383
66
|
def add(key, value, ttl, options)
|
384
|
-
opkey =
|
385
|
-
|
386
|
-
process_value_req(opkey, key, value, ttl, cas, options)
|
67
|
+
opkey = quiet? ? :addq : :add
|
68
|
+
storage_req(opkey, key, value, ttl, 0, options)
|
387
69
|
end
|
388
70
|
|
389
71
|
def replace(key, value, ttl, cas, options)
|
390
|
-
opkey =
|
391
|
-
|
72
|
+
opkey = quiet? ? :replaceq : :replace
|
73
|
+
storage_req(opkey, key, value, ttl, cas, options)
|
392
74
|
end
|
393
75
|
|
394
76
|
# rubocop:disable Metrics/ParameterLists
|
395
|
-
def
|
77
|
+
def storage_req(opkey, key, value, ttl, cas, options)
|
396
78
|
(value, bitflags) = @value_marshaller.store(key, value, options)
|
397
79
|
ttl = TtlSanitizer.sanitize(ttl)
|
398
80
|
|
@@ -400,21 +82,42 @@ module Dalli
|
|
400
82
|
value: value, bitflags: bitflags,
|
401
83
|
ttl: ttl, cas: cas)
|
402
84
|
write(req)
|
403
|
-
|
85
|
+
response_processor.storage_response unless quiet?
|
404
86
|
end
|
405
87
|
# rubocop:enable Metrics/ParameterLists
|
406
88
|
|
89
|
+
def append(key, value)
|
90
|
+
opkey = quiet? ? :appendq : :append
|
91
|
+
write_append_prepend opkey, key, value
|
92
|
+
end
|
93
|
+
|
94
|
+
def prepend(key, value)
|
95
|
+
opkey = quiet? ? :prependq : :prepend
|
96
|
+
write_append_prepend opkey, key, value
|
97
|
+
end
|
98
|
+
|
99
|
+
def write_append_prepend(opkey, key, value)
|
100
|
+
write(RequestFormatter.standard_request(opkey: opkey, key: key, value: value))
|
101
|
+
response_processor.no_body_response unless quiet?
|
102
|
+
end
|
103
|
+
|
104
|
+
# Delete Commands
|
407
105
|
def delete(key, cas)
|
408
|
-
opkey =
|
106
|
+
opkey = quiet? ? :deleteq : :delete
|
409
107
|
req = RequestFormatter.standard_request(opkey: opkey, key: key, cas: cas)
|
410
108
|
write(req)
|
411
|
-
|
109
|
+
response_processor.delete_response unless quiet?
|
412
110
|
end
|
413
111
|
|
414
|
-
|
415
|
-
|
416
|
-
|
417
|
-
|
112
|
+
# Arithmetic Commands
|
113
|
+
def decr(key, count, ttl, initial)
|
114
|
+
opkey = quiet? ? :decrq : :decr
|
115
|
+
decr_incr opkey, key, count, ttl, initial
|
116
|
+
end
|
117
|
+
|
118
|
+
def incr(key, count, ttl, initial)
|
119
|
+
opkey = quiet? ? :incrq : :incr
|
120
|
+
decr_incr opkey, key, count, ttl, initial
|
418
121
|
end
|
419
122
|
|
420
123
|
# This allows us to special case a nil initial value, and
|
@@ -429,115 +132,48 @@ module Dalli
|
|
429
132
|
initial ||= 0
|
430
133
|
write(RequestFormatter.decr_incr_request(opkey: opkey, key: key,
|
431
134
|
count: count, initial: initial, expiry: expiry))
|
432
|
-
|
433
|
-
end
|
434
|
-
|
435
|
-
def decr(key, count, ttl, initial)
|
436
|
-
decr_incr :decr, key, count, ttl, initial
|
437
|
-
end
|
438
|
-
|
439
|
-
def incr(key, count, ttl, initial)
|
440
|
-
decr_incr :incr, key, count, ttl, initial
|
441
|
-
end
|
442
|
-
|
443
|
-
def write_append_prepend(opkey, key, value)
|
444
|
-
write_generic RequestFormatter.standard_request(opkey: opkey, key: key, value: value)
|
135
|
+
response_processor.decr_incr_response unless quiet?
|
445
136
|
end
|
446
137
|
|
447
|
-
|
448
|
-
|
449
|
-
|
450
|
-
|
451
|
-
|
452
|
-
def write_noop
|
453
|
-
req = RequestFormatter.standard_request(opkey: :noop)
|
454
|
-
write(req)
|
138
|
+
# Other Commands
|
139
|
+
def flush(ttl = 0)
|
140
|
+
opkey = quiet? ? :flushq : :flush
|
141
|
+
write(RequestFormatter.standard_request(opkey: opkey, ttl: ttl))
|
142
|
+
response_processor.no_body_response unless quiet?
|
455
143
|
end
|
456
144
|
|
457
145
|
# Noop is a keepalive operation but also used to demarcate the end of a set of pipelined commands.
|
458
146
|
# We need to read all the responses at once.
|
459
147
|
def noop
|
460
148
|
write_noop
|
461
|
-
|
462
|
-
end
|
463
|
-
|
464
|
-
def append(key, value)
|
465
|
-
write_append_prepend :append, key, value
|
466
|
-
end
|
467
|
-
|
468
|
-
def prepend(key, value)
|
469
|
-
write_append_prepend :prepend, key, value
|
149
|
+
response_processor.multi_with_keys_response
|
470
150
|
end
|
471
151
|
|
472
152
|
def stats(info = '')
|
473
153
|
req = RequestFormatter.standard_request(opkey: :stat, key: info)
|
474
154
|
write(req)
|
475
|
-
|
155
|
+
response_processor.multi_with_keys_response
|
476
156
|
end
|
477
157
|
|
478
158
|
def reset_stats
|
479
|
-
|
480
|
-
|
481
|
-
|
482
|
-
def cas(key)
|
483
|
-
req = RequestFormatter.standard_request(opkey: :get, key: key)
|
484
|
-
write(req)
|
485
|
-
@response_processor.data_cas_response
|
159
|
+
write(RequestFormatter.standard_request(opkey: :stat, key: 'reset'))
|
160
|
+
response_processor.generic_response
|
486
161
|
end
|
487
162
|
|
488
163
|
def version
|
489
|
-
|
164
|
+
write(RequestFormatter.standard_request(opkey: :version))
|
165
|
+
response_processor.generic_response
|
490
166
|
end
|
491
167
|
|
492
|
-
def
|
493
|
-
|
494
|
-
write_generic RequestFormatter.standard_request(opkey: :touch, key: key, ttl: ttl)
|
495
|
-
end
|
496
|
-
|
497
|
-
def gat(key, ttl, options = nil)
|
498
|
-
ttl = TtlSanitizer.sanitize(ttl)
|
499
|
-
req = RequestFormatter.standard_request(opkey: :gat, key: key, ttl: ttl)
|
168
|
+
def write_noop
|
169
|
+
req = RequestFormatter.standard_request(opkey: :noop)
|
500
170
|
write(req)
|
501
|
-
@response_processor.generic_response(unpack: true, cache_nils: cache_nils?(options))
|
502
|
-
end
|
503
|
-
|
504
|
-
def connect
|
505
|
-
Dalli.logger.debug { "Dalli::Server#connect #{name}" }
|
506
|
-
|
507
|
-
begin
|
508
|
-
@pid = Process.pid
|
509
|
-
@sock = memcached_socket
|
510
|
-
authenticate_connection if require_auth?
|
511
|
-
@version = version # Connect socket if not authed
|
512
|
-
up!
|
513
|
-
rescue Dalli::DalliError # SASL auth failure
|
514
|
-
raise
|
515
|
-
rescue SystemCallError, Timeout::Error, EOFError, SocketError => e
|
516
|
-
# SocketError = DNS resolution failure
|
517
|
-
failure!(e)
|
518
|
-
end
|
519
|
-
end
|
520
|
-
|
521
|
-
def memcached_socket
|
522
|
-
if socket_type == :unix
|
523
|
-
Dalli::Socket::UNIX.open(hostname, self, options)
|
524
|
-
else
|
525
|
-
Dalli::Socket::TCP.open(hostname, port, self, options)
|
526
|
-
end
|
527
|
-
end
|
528
|
-
|
529
|
-
def require_auth?
|
530
|
-
!username.nil?
|
531
|
-
end
|
532
|
-
|
533
|
-
def username
|
534
|
-
@options[:username] || ENV['MEMCACHE_USERNAME']
|
535
|
-
end
|
536
|
-
|
537
|
-
def password
|
538
|
-
@options[:password] || ENV['MEMCACHE_PASSWORD']
|
539
171
|
end
|
540
172
|
|
173
|
+
require_relative 'binary/request_formatter'
|
174
|
+
require_relative 'binary/response_header'
|
175
|
+
require_relative 'binary/response_processor'
|
176
|
+
require_relative 'binary/sasl_authentication'
|
541
177
|
include SaslAuthentication
|
542
178
|
end
|
543
179
|
end
|