redis 4.2.0 → 4.6.0
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.
- checksums.yaml +4 -4
- data/CHANGELOG.md +114 -0
- data/README.md +27 -21
- data/lib/redis/client.rb +52 -13
- data/lib/redis/cluster/command.rb +4 -6
- data/lib/redis/cluster/command_loader.rb +6 -7
- data/lib/redis/cluster/node.rb +14 -1
- data/lib/redis/cluster/option.rb +5 -2
- data/lib/redis/cluster.rb +33 -13
- data/lib/redis/commands/bitmaps.rb +63 -0
- data/lib/redis/commands/cluster.rb +45 -0
- data/lib/redis/commands/connection.rb +58 -0
- data/lib/redis/commands/geo.rb +84 -0
- data/lib/redis/commands/hashes.rb +251 -0
- data/lib/redis/commands/hyper_log_log.rb +37 -0
- data/lib/redis/commands/keys.rb +411 -0
- data/lib/redis/commands/lists.rb +289 -0
- data/lib/redis/commands/pubsub.rb +72 -0
- data/lib/redis/commands/scripting.rb +114 -0
- data/lib/redis/commands/server.rb +188 -0
- data/lib/redis/commands/sets.rb +207 -0
- data/lib/redis/commands/sorted_sets.rb +804 -0
- data/lib/redis/commands/streams.rb +382 -0
- data/lib/redis/commands/strings.rb +313 -0
- data/lib/redis/commands/transactions.rb +92 -0
- data/lib/redis/commands.rb +242 -0
- data/lib/redis/connection/command_helper.rb +2 -0
- data/lib/redis/connection/hiredis.rb +3 -2
- data/lib/redis/connection/ruby.rb +65 -56
- data/lib/redis/connection/synchrony.rb +10 -8
- data/lib/redis/connection.rb +1 -1
- data/lib/redis/distributed.rb +136 -23
- data/lib/redis/pipeline.rb +95 -2
- data/lib/redis/version.rb +1 -1
- data/lib/redis.rb +145 -3369
- metadata +30 -8
@@ -0,0 +1,92 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class Redis
|
4
|
+
module Commands
|
5
|
+
module Transactions
|
6
|
+
# Watch the given keys to determine execution of the MULTI/EXEC block.
|
7
|
+
#
|
8
|
+
# Using a block is optional, but is necessary for thread-safety.
|
9
|
+
#
|
10
|
+
# An `#unwatch` is automatically issued if an exception is raised within the
|
11
|
+
# block that is a subclass of StandardError and is not a ConnectionError.
|
12
|
+
#
|
13
|
+
# @example With a block
|
14
|
+
# redis.watch("key") do
|
15
|
+
# if redis.get("key") == "some value"
|
16
|
+
# redis.multi do |multi|
|
17
|
+
# multi.set("key", "other value")
|
18
|
+
# multi.incr("counter")
|
19
|
+
# end
|
20
|
+
# else
|
21
|
+
# redis.unwatch
|
22
|
+
# end
|
23
|
+
# end
|
24
|
+
# # => ["OK", 6]
|
25
|
+
#
|
26
|
+
# @example Without a block
|
27
|
+
# redis.watch("key")
|
28
|
+
# # => "OK"
|
29
|
+
#
|
30
|
+
# @param [String, Array<String>] keys one or more keys to watch
|
31
|
+
# @return [Object] if using a block, returns the return value of the block
|
32
|
+
# @return [String] if not using a block, returns `OK`
|
33
|
+
#
|
34
|
+
# @see #unwatch
|
35
|
+
# @see #multi
|
36
|
+
def watch(*keys)
|
37
|
+
synchronize do |client|
|
38
|
+
res = client.call([:watch, *keys])
|
39
|
+
|
40
|
+
if block_given?
|
41
|
+
begin
|
42
|
+
yield(self)
|
43
|
+
rescue ConnectionError
|
44
|
+
raise
|
45
|
+
rescue StandardError
|
46
|
+
unwatch
|
47
|
+
raise
|
48
|
+
end
|
49
|
+
else
|
50
|
+
res
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
54
|
+
|
55
|
+
# Forget about all watched keys.
|
56
|
+
#
|
57
|
+
# @return [String] `OK`
|
58
|
+
#
|
59
|
+
# @see #watch
|
60
|
+
# @see #multi
|
61
|
+
def unwatch
|
62
|
+
send_command([:unwatch])
|
63
|
+
end
|
64
|
+
|
65
|
+
# Execute all commands issued after MULTI.
|
66
|
+
#
|
67
|
+
# Only call this method when `#multi` was called **without** a block.
|
68
|
+
#
|
69
|
+
# @return [nil, Array<...>]
|
70
|
+
# - when commands were not executed, `nil`
|
71
|
+
# - when commands were executed, an array with their replies
|
72
|
+
#
|
73
|
+
# @see #multi
|
74
|
+
# @see #discard
|
75
|
+
def exec
|
76
|
+
send_command([:exec])
|
77
|
+
end
|
78
|
+
|
79
|
+
# Discard all commands issued after MULTI.
|
80
|
+
#
|
81
|
+
# Only call this method when `#multi` was called **without** a block.
|
82
|
+
#
|
83
|
+
# @return [String] `"OK"`
|
84
|
+
#
|
85
|
+
# @see #multi
|
86
|
+
# @see #exec
|
87
|
+
def discard
|
88
|
+
send_command([:discard])
|
89
|
+
end
|
90
|
+
end
|
91
|
+
end
|
92
|
+
end
|
@@ -0,0 +1,242 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "redis/commands/bitmaps"
|
4
|
+
require "redis/commands/cluster"
|
5
|
+
require "redis/commands/connection"
|
6
|
+
require "redis/commands/geo"
|
7
|
+
require "redis/commands/hashes"
|
8
|
+
require "redis/commands/hyper_log_log"
|
9
|
+
require "redis/commands/keys"
|
10
|
+
require "redis/commands/lists"
|
11
|
+
require "redis/commands/pubsub"
|
12
|
+
require "redis/commands/scripting"
|
13
|
+
require "redis/commands/server"
|
14
|
+
require "redis/commands/sets"
|
15
|
+
require "redis/commands/sorted_sets"
|
16
|
+
require "redis/commands/streams"
|
17
|
+
require "redis/commands/strings"
|
18
|
+
require "redis/commands/transactions"
|
19
|
+
|
20
|
+
class Redis
|
21
|
+
module Commands
|
22
|
+
include Bitmaps
|
23
|
+
include Cluster
|
24
|
+
include Connection
|
25
|
+
include Geo
|
26
|
+
include Hashes
|
27
|
+
include HyperLogLog
|
28
|
+
include Keys
|
29
|
+
include Lists
|
30
|
+
include Pubsub
|
31
|
+
include Scripting
|
32
|
+
include Server
|
33
|
+
include Sets
|
34
|
+
include SortedSets
|
35
|
+
include Streams
|
36
|
+
include Strings
|
37
|
+
include Transactions
|
38
|
+
|
39
|
+
# Commands returning 1 for true and 0 for false may be executed in a pipeline
|
40
|
+
# where the method call will return nil. Propagate the nil instead of falsely
|
41
|
+
# returning false.
|
42
|
+
Boolify = lambda { |value|
|
43
|
+
case value
|
44
|
+
when 1
|
45
|
+
true
|
46
|
+
when 0
|
47
|
+
false
|
48
|
+
else
|
49
|
+
value
|
50
|
+
end
|
51
|
+
}
|
52
|
+
|
53
|
+
BoolifySet = lambda { |value|
|
54
|
+
case value
|
55
|
+
when "OK"
|
56
|
+
true
|
57
|
+
when nil
|
58
|
+
false
|
59
|
+
else
|
60
|
+
value
|
61
|
+
end
|
62
|
+
}
|
63
|
+
|
64
|
+
Hashify = lambda { |value|
|
65
|
+
if value.respond_to?(:each_slice)
|
66
|
+
value.each_slice(2).to_h
|
67
|
+
else
|
68
|
+
value
|
69
|
+
end
|
70
|
+
}
|
71
|
+
|
72
|
+
Pairify = lambda { |value|
|
73
|
+
if value.respond_to?(:each_slice)
|
74
|
+
value.each_slice(2).to_a
|
75
|
+
else
|
76
|
+
value
|
77
|
+
end
|
78
|
+
}
|
79
|
+
|
80
|
+
Floatify = lambda { |value|
|
81
|
+
case value
|
82
|
+
when "inf"
|
83
|
+
Float::INFINITY
|
84
|
+
when "-inf"
|
85
|
+
-Float::INFINITY
|
86
|
+
when String
|
87
|
+
Float(value)
|
88
|
+
else
|
89
|
+
value
|
90
|
+
end
|
91
|
+
}
|
92
|
+
|
93
|
+
FloatifyPairs = lambda { |value|
|
94
|
+
return value unless value.respond_to?(:each_slice)
|
95
|
+
|
96
|
+
value.each_slice(2).map do |member, score|
|
97
|
+
[member, Floatify.call(score)]
|
98
|
+
end
|
99
|
+
}
|
100
|
+
|
101
|
+
HashifyInfo = lambda { |reply|
|
102
|
+
lines = reply.split("\r\n").grep_v(/^(#|$)/)
|
103
|
+
lines.map! { |line| line.split(':', 2) }
|
104
|
+
lines.compact!
|
105
|
+
lines.to_h
|
106
|
+
}
|
107
|
+
|
108
|
+
HashifyStreams = lambda { |reply|
|
109
|
+
case reply
|
110
|
+
when nil
|
111
|
+
{}
|
112
|
+
else
|
113
|
+
reply.map { |key, entries| [key, HashifyStreamEntries.call(entries)] }.to_h
|
114
|
+
end
|
115
|
+
}
|
116
|
+
|
117
|
+
EMPTY_STREAM_RESPONSE = [nil].freeze
|
118
|
+
private_constant :EMPTY_STREAM_RESPONSE
|
119
|
+
|
120
|
+
HashifyStreamEntries = lambda { |reply|
|
121
|
+
reply.compact.map do |entry_id, values|
|
122
|
+
[entry_id, values&.each_slice(2)&.to_h]
|
123
|
+
end
|
124
|
+
}
|
125
|
+
|
126
|
+
HashifyStreamAutoclaim = lambda { |reply|
|
127
|
+
{
|
128
|
+
'next' => reply[0],
|
129
|
+
'entries' => reply[1].map { |entry| [entry[0], entry[1].each_slice(2).to_h] }
|
130
|
+
}
|
131
|
+
}
|
132
|
+
|
133
|
+
HashifyStreamAutoclaimJustId = lambda { |reply|
|
134
|
+
{
|
135
|
+
'next' => reply[0],
|
136
|
+
'entries' => reply[1]
|
137
|
+
}
|
138
|
+
}
|
139
|
+
|
140
|
+
HashifyStreamPendings = lambda { |reply|
|
141
|
+
{
|
142
|
+
'size' => reply[0],
|
143
|
+
'min_entry_id' => reply[1],
|
144
|
+
'max_entry_id' => reply[2],
|
145
|
+
'consumers' => reply[3].nil? ? {} : reply[3].to_h
|
146
|
+
}
|
147
|
+
}
|
148
|
+
|
149
|
+
HashifyStreamPendingDetails = lambda { |reply|
|
150
|
+
reply.map do |arr|
|
151
|
+
{
|
152
|
+
'entry_id' => arr[0],
|
153
|
+
'consumer' => arr[1],
|
154
|
+
'elapsed' => arr[2],
|
155
|
+
'count' => arr[3]
|
156
|
+
}
|
157
|
+
end
|
158
|
+
}
|
159
|
+
|
160
|
+
HashifyClusterNodeInfo = lambda { |str|
|
161
|
+
arr = str.split(' ')
|
162
|
+
{
|
163
|
+
'node_id' => arr[0],
|
164
|
+
'ip_port' => arr[1],
|
165
|
+
'flags' => arr[2].split(','),
|
166
|
+
'master_node_id' => arr[3],
|
167
|
+
'ping_sent' => arr[4],
|
168
|
+
'pong_recv' => arr[5],
|
169
|
+
'config_epoch' => arr[6],
|
170
|
+
'link_state' => arr[7],
|
171
|
+
'slots' => arr[8].nil? ? nil : Range.new(*arr[8].split('-'))
|
172
|
+
}
|
173
|
+
}
|
174
|
+
|
175
|
+
HashifyClusterSlots = lambda { |reply|
|
176
|
+
reply.map do |arr|
|
177
|
+
first_slot, last_slot = arr[0..1]
|
178
|
+
master = { 'ip' => arr[2][0], 'port' => arr[2][1], 'node_id' => arr[2][2] }
|
179
|
+
replicas = arr[3..-1].map { |r| { 'ip' => r[0], 'port' => r[1], 'node_id' => r[2] } }
|
180
|
+
{
|
181
|
+
'start_slot' => first_slot,
|
182
|
+
'end_slot' => last_slot,
|
183
|
+
'master' => master,
|
184
|
+
'replicas' => replicas
|
185
|
+
}
|
186
|
+
end
|
187
|
+
}
|
188
|
+
|
189
|
+
HashifyClusterNodes = lambda { |reply|
|
190
|
+
reply.split(/[\r\n]+/).map { |str| HashifyClusterNodeInfo.call(str) }
|
191
|
+
}
|
192
|
+
|
193
|
+
HashifyClusterSlaves = lambda { |reply|
|
194
|
+
reply.map { |str| HashifyClusterNodeInfo.call(str) }
|
195
|
+
}
|
196
|
+
|
197
|
+
Noop = ->(reply) { reply }
|
198
|
+
|
199
|
+
# Sends a command to Redis and returns its reply.
|
200
|
+
#
|
201
|
+
# Replies are converted to Ruby objects according to the RESP protocol, so
|
202
|
+
# you can expect a Ruby array, integer or nil when Redis sends one. Higher
|
203
|
+
# level transformations, such as converting an array of pairs into a Ruby
|
204
|
+
# hash, are up to consumers.
|
205
|
+
#
|
206
|
+
# Redis error replies are raised as Ruby exceptions.
|
207
|
+
def call(*command)
|
208
|
+
send_command(command)
|
209
|
+
end
|
210
|
+
|
211
|
+
# Interact with the sentinel command (masters, master, slaves, failover)
|
212
|
+
#
|
213
|
+
# @param [String] subcommand e.g. `masters`, `master`, `slaves`
|
214
|
+
# @param [Array<String>] args depends on subcommand
|
215
|
+
# @return [Array<String>, Hash<String, String>, String] depends on subcommand
|
216
|
+
def sentinel(subcommand, *args)
|
217
|
+
subcommand = subcommand.to_s.downcase
|
218
|
+
send_command([:sentinel, subcommand] + args) do |reply|
|
219
|
+
case subcommand
|
220
|
+
when "get-master-addr-by-name"
|
221
|
+
reply
|
222
|
+
else
|
223
|
+
if reply.is_a?(Array)
|
224
|
+
if reply[0].is_a?(Array)
|
225
|
+
reply.map(&Hashify)
|
226
|
+
else
|
227
|
+
Hashify.call(reply)
|
228
|
+
end
|
229
|
+
else
|
230
|
+
reply
|
231
|
+
end
|
232
|
+
end
|
233
|
+
end
|
234
|
+
end
|
235
|
+
|
236
|
+
private
|
237
|
+
|
238
|
+
def method_missing(*command) # rubocop:disable Style/MissingRespondToMissing
|
239
|
+
send_command(command)
|
240
|
+
end
|
241
|
+
end
|
242
|
+
end
|
@@ -12,11 +12,13 @@ class Redis
|
|
12
12
|
if i.is_a? Array
|
13
13
|
i.each do |j|
|
14
14
|
j = j.to_s
|
15
|
+
j = j.encoding == Encoding::BINARY ? j : j.b
|
15
16
|
command << "$#{j.bytesize}"
|
16
17
|
command << j
|
17
18
|
end
|
18
19
|
else
|
19
20
|
i = i.to_s
|
21
|
+
i = i.encoding == Encoding::BINARY ? i : i.b
|
20
22
|
command << "$#{i.bytesize}"
|
21
23
|
command << i
|
22
24
|
end
|
@@ -1,8 +1,9 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
|
4
|
-
|
5
|
-
|
3
|
+
require "redis/connection/registry"
|
4
|
+
require "redis/connection/command_helper"
|
5
|
+
require "redis/errors"
|
6
|
+
|
6
7
|
require "socket"
|
7
8
|
require "timeout"
|
8
9
|
|
@@ -21,7 +22,7 @@ class Redis
|
|
21
22
|
super(*args)
|
22
23
|
|
23
24
|
@timeout = @write_timeout = nil
|
24
|
-
@buffer = "".
|
25
|
+
@buffer = "".b
|
25
26
|
end
|
26
27
|
|
27
28
|
def timeout=(timeout)
|
@@ -35,7 +36,8 @@ class Redis
|
|
35
36
|
def read(nbytes)
|
36
37
|
result = @buffer.slice!(0, nbytes)
|
37
38
|
|
38
|
-
|
39
|
+
buffer = String.new(capacity: nbytes, encoding: Encoding::ASCII_8BIT)
|
40
|
+
result << _read_from_socket(nbytes - result.bytesize, buffer) while result.bytesize < nbytes
|
39
41
|
|
40
42
|
result
|
41
43
|
end
|
@@ -48,58 +50,51 @@ class Redis
|
|
48
50
|
@buffer.slice!(0, crlf + CRLF.bytesize)
|
49
51
|
end
|
50
52
|
|
51
|
-
def _read_from_socket(nbytes)
|
52
|
-
|
53
|
-
read_nonblock(nbytes)
|
54
|
-
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
end
|
67
|
-
rescue EOFError
|
68
|
-
raise Errno::ECONNRESET
|
69
|
-
end
|
70
|
-
|
71
|
-
def _write_to_socket(data)
|
72
|
-
begin
|
73
|
-
write_nonblock(data)
|
74
|
-
rescue IO::WaitWritable
|
75
|
-
if IO.select(nil, [self], nil, @write_timeout)
|
76
|
-
retry
|
77
|
-
else
|
78
|
-
raise Redis::TimeoutError
|
79
|
-
end
|
80
|
-
rescue IO::WaitReadable
|
81
|
-
if IO.select([self], nil, nil, @write_timeout)
|
82
|
-
retry
|
83
|
-
else
|
84
|
-
raise Redis::TimeoutError
|
53
|
+
def _read_from_socket(nbytes, buffer = nil)
|
54
|
+
loop do
|
55
|
+
case chunk = read_nonblock(nbytes, buffer, exception: false)
|
56
|
+
when :wait_readable
|
57
|
+
unless wait_readable(@timeout)
|
58
|
+
raise Redis::TimeoutError
|
59
|
+
end
|
60
|
+
when :wait_writable
|
61
|
+
unless wait_writable(@timeout)
|
62
|
+
raise Redis::TimeoutError
|
63
|
+
end
|
64
|
+
when nil
|
65
|
+
raise Errno::ECONNRESET
|
66
|
+
when String
|
67
|
+
return chunk
|
85
68
|
end
|
86
69
|
end
|
87
|
-
rescue EOFError
|
88
|
-
raise Errno::ECONNRESET
|
89
70
|
end
|
90
71
|
|
91
|
-
def write(
|
92
|
-
return super(
|
72
|
+
def write(buffer)
|
73
|
+
return super(buffer) unless @write_timeout
|
93
74
|
|
94
|
-
|
95
|
-
|
75
|
+
bytes_to_write = buffer.bytesize
|
76
|
+
total_bytes_written = 0
|
96
77
|
loop do
|
97
|
-
|
78
|
+
case bytes_written = write_nonblock(buffer, exception: false)
|
79
|
+
when :wait_readable
|
80
|
+
unless wait_readable(@write_timeout)
|
81
|
+
raise Redis::TimeoutError
|
82
|
+
end
|
83
|
+
when :wait_writable
|
84
|
+
unless wait_writable(@write_timeout)
|
85
|
+
raise Redis::TimeoutError
|
86
|
+
end
|
87
|
+
when nil
|
88
|
+
raise Errno::ECONNRESET
|
89
|
+
when Integer
|
90
|
+
total_bytes_written += bytes_written
|
98
91
|
|
99
|
-
|
100
|
-
|
92
|
+
if total_bytes_written >= bytes_to_write
|
93
|
+
return total_bytes_written
|
94
|
+
end
|
101
95
|
|
102
|
-
|
96
|
+
buffer = buffer.byteslice(bytes_written..-1)
|
97
|
+
end
|
103
98
|
end
|
104
99
|
end
|
105
100
|
end
|
@@ -135,11 +130,13 @@ class Redis
|
|
135
130
|
raise TimeoutError
|
136
131
|
end
|
137
132
|
|
138
|
-
# JRuby raises Errno::EAGAIN on #read_nonblock even when
|
133
|
+
# JRuby raises Errno::EAGAIN on #read_nonblock even when it
|
139
134
|
# says it is readable (1.6.6, in both 1.8 and 1.9 mode).
|
140
135
|
# Use the blocking #readpartial method instead.
|
141
136
|
|
142
|
-
def _read_from_socket(nbytes)
|
137
|
+
def _read_from_socket(nbytes, _buffer = nil)
|
138
|
+
# JRuby: Throw away the buffer as we won't need it
|
139
|
+
# but still need to support the max arity of 2
|
143
140
|
readpartial(nbytes)
|
144
141
|
rescue EOFError
|
145
142
|
raise Errno::ECONNRESET
|
@@ -160,7 +157,7 @@ class Redis
|
|
160
157
|
begin
|
161
158
|
sock.connect_nonblock(sockaddr)
|
162
159
|
rescue Errno::EINPROGRESS
|
163
|
-
raise TimeoutError
|
160
|
+
raise TimeoutError unless sock.wait_writable(timeout)
|
164
161
|
|
165
162
|
begin
|
166
163
|
sock.connect_nonblock(sockaddr)
|
@@ -215,7 +212,7 @@ class Redis
|
|
215
212
|
begin
|
216
213
|
sock.connect_nonblock(sockaddr)
|
217
214
|
rescue Errno::EINPROGRESS
|
218
|
-
raise TimeoutError
|
215
|
+
raise TimeoutError unless sock.wait_writable(timeout)
|
219
216
|
|
220
217
|
begin
|
221
218
|
sock.connect_nonblock(sockaddr)
|
@@ -233,8 +230,20 @@ class Redis
|
|
233
230
|
class SSLSocket < ::OpenSSL::SSL::SSLSocket
|
234
231
|
include SocketMixin
|
235
232
|
|
233
|
+
unless method_defined?(:wait_readable)
|
234
|
+
def wait_readable(timeout = nil)
|
235
|
+
to_io.wait_readable(timeout)
|
236
|
+
end
|
237
|
+
end
|
238
|
+
|
239
|
+
unless method_defined?(:wait_writable)
|
240
|
+
def wait_writable(timeout = nil)
|
241
|
+
to_io.wait_writable(timeout)
|
242
|
+
end
|
243
|
+
end
|
244
|
+
|
236
245
|
def self.connect(host, port, timeout, ssl_params)
|
237
|
-
#
|
246
|
+
# NOTE: this is using Redis::Connection::TCPSocket
|
238
247
|
tcp_sock = TCPSocket.connect(host, port, timeout)
|
239
248
|
|
240
249
|
ctx = OpenSSL::SSL::SSLContext.new
|
@@ -254,13 +263,13 @@ class Redis
|
|
254
263
|
# Instead, you have to retry.
|
255
264
|
ssl_sock.connect_nonblock
|
256
265
|
rescue Errno::EAGAIN, Errno::EWOULDBLOCK, IO::WaitReadable
|
257
|
-
if
|
266
|
+
if ssl_sock.wait_readable(timeout)
|
258
267
|
retry
|
259
268
|
else
|
260
269
|
raise TimeoutError
|
261
270
|
end
|
262
271
|
rescue IO::WaitWritable
|
263
|
-
if
|
272
|
+
if ssl_sock.wait_writable(timeout)
|
264
273
|
retry
|
265
274
|
else
|
266
275
|
raise TimeoutError
|
@@ -1,13 +1,14 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
|
4
|
-
|
5
|
-
|
3
|
+
require "redis/connection/registry"
|
4
|
+
require "redis/connection/command_helper"
|
5
|
+
require "redis/errors"
|
6
|
+
|
6
7
|
require "em-synchrony"
|
7
8
|
require "hiredis/reader"
|
8
9
|
|
9
|
-
|
10
|
-
"The redis synchrony driver is deprecated and will be removed in redis-rb 5.0. " \
|
10
|
+
::Redis.deprecate!(
|
11
|
+
"The redis synchrony driver is deprecated and will be removed in redis-rb 5.0.0. " \
|
11
12
|
"We're looking for people to maintain it as a separate gem, see https://github.com/redis/redis-rb/issues/915"
|
12
13
|
)
|
13
14
|
|
@@ -129,11 +130,12 @@ class Redis
|
|
129
130
|
def read
|
130
131
|
type, payload = @connection.read
|
131
132
|
|
132
|
-
|
133
|
+
case type
|
134
|
+
when :reply
|
133
135
|
payload
|
134
|
-
|
136
|
+
when :error
|
135
137
|
raise payload
|
136
|
-
|
138
|
+
when :timeout
|
137
139
|
raise TimeoutError
|
138
140
|
else
|
139
141
|
raise "Unknown type #{type.inspect}"
|
data/lib/redis/connection.rb
CHANGED