Skip to content

Commit 609b6bc

Browse files
committed
feat(pipeline): add flag to disable raising exceptions
1 parent bce3f41 commit 609b6bc

File tree

8 files changed

+74
-12
lines changed

8 files changed

+74
-12
lines changed

CHANGELOG.md

+2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
# Unreleased
22

3+
- Add `exception` flag in `pipelined` allowing failed commands to be returned in the result array when set to `false`.
4+
35
# 5.1.0
46

57
- `multi` now accept a `watch` keyword argument like `redis-client`. See #1236.

README.md

+22
Original file line numberDiff line numberDiff line change
@@ -191,6 +191,28 @@ end
191191
# => ["OK"]
192192
```
193193

194+
### Exception management
195+
196+
The `exception` flag in the `#pipelined` is a feature that modifies the pipeline execution behavior. When set
197+
to `false`, it doesn't raise an exception when a command error occurs. Instead, it allows the pipeline to execute all
198+
commands, and any failed command will be available in the returned array. (Defaults to `true`)
199+
200+
```ruby
201+
results = redis.pipelined(exception: false) do |pipeline|
202+
pipeline.set('key1', 'value1')
203+
pipeline.lpush('key1', 'something') # This will fail
204+
pipeline.set('key2', 'value2')
205+
end
206+
# results => ["OK", #<RedisClient::WrongTypeError: WRONGTYPE Operation against a key holding the wrong kind of value>, "OK"]
207+
208+
results.each do |result|
209+
if result.is_a?(Redis::CommandError)
210+
# Do something with the failed result
211+
end
212+
end
213+
```
214+
215+
194216
### Executing commands atomically
195217

196218
You can use `MULTI/EXEC` to run a number of commands in an atomic

cluster/lib/redis/cluster/client.rb

+2-2
Original file line numberDiff line numberDiff line change
@@ -90,8 +90,8 @@ def blocking_call_v(timeout, command, &block)
9090
handle_errors { super(timeout, command, &block) }
9191
end
9292

93-
def pipelined(&block)
94-
handle_errors { super(&block) }
93+
def pipelined(exception: true, &block)
94+
handle_errors { super(exception: exception, &block) }
9595
end
9696

9797
def multi(watch: nil, &block)

lib/redis.rb

+3-3
Original file line numberDiff line numberDiff line change
@@ -99,10 +99,10 @@ def _client
9999
@client
100100
end
101101

102-
def pipelined
102+
def pipelined(exception: true)
103103
synchronize do |client|
104-
client.pipelined do |raw_pipeline|
105-
yield PipelinedConnection.new(raw_pipeline)
104+
client.pipelined(exception: exception) do |raw_pipeline|
105+
yield PipelinedConnection.new(raw_pipeline, exception: exception)
106106
end
107107
end
108108
end

lib/redis/client.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -105,7 +105,7 @@ def blocking_call_v(timeout, command, &block)
105105
Client.translate_error!(error)
106106
end
107107

108-
def pipelined
108+
def pipelined(exception: true)
109109
super
110110
rescue ::RedisClient::Error => error
111111
Client.translate_error!(error)

lib/redis/pipeline.rb

+7-5
Original file line numberDiff line numberDiff line change
@@ -6,9 +6,10 @@ class Redis
66
class PipelinedConnection
77
attr_accessor :db
88

9-
def initialize(pipeline, futures = [])
9+
def initialize(pipeline, futures = [], exception: true)
1010
@pipeline = pipeline
1111
@futures = futures
12+
@exception = exception
1213
end
1314

1415
include Commands
@@ -37,7 +38,7 @@ def synchronize
3738
end
3839

3940
def send_command(command, &block)
40-
future = Future.new(command, block)
41+
future = Future.new(command, block, @exception)
4142
@pipeline.call_v(command) do |result|
4243
future._set(result)
4344
end
@@ -46,7 +47,7 @@ def send_command(command, &block)
4647
end
4748

4849
def send_blocking_command(command, timeout, &block)
49-
future = Future.new(command, block)
50+
future = Future.new(command, block, @exception)
5051
@pipeline.blocking_call_v(timeout, command) do |result|
5152
future._set(result)
5253
end
@@ -79,10 +80,11 @@ def initialize
7980
class Future < BasicObject
8081
FutureNotReady = ::Redis::FutureNotReady.new
8182

82-
def initialize(command, coerce)
83+
def initialize(command, coerce, exception)
8384
@command = command
8485
@object = FutureNotReady
8586
@coerce = coerce
87+
@exception = exception
8688
end
8789

8890
def inspect
@@ -95,7 +97,7 @@ def _set(object)
9597
end
9698

9799
def value
98-
::Kernel.raise(@object) if @object.is_a?(::StandardError)
100+
::Kernel.raise(@object) if @exception && @object.is_a?(::StandardError)
99101
@object
100102
end
101103

redis.gemspec

+1-1
Original file line numberDiff line numberDiff line change
@@ -45,5 +45,5 @@ Gem::Specification.new do |s|
4545

4646
s.required_ruby_version = '>= 2.6.0'
4747

48-
s.add_runtime_dependency('redis-client', '>= 0.17.0')
48+
s.add_runtime_dependency('redis-client', '>= 0.22.0')
4949
end

test/redis/pipelining_commands_test.rb

+36
Original file line numberDiff line numberDiff line change
@@ -98,6 +98,18 @@ def test_assignment_of_results_inside_the_block_with_errors
9898
assert_raises(Redis::FutureNotReady) { @second.value }
9999
end
100100

101+
def test_assignment_of_results_inside_the_block_without_raising_exception
102+
r.pipelined(exception: false) do |p|
103+
@first = p.doesnt_exist
104+
@second = p.sadd?("foo", 1)
105+
@third = p.sadd?("foo", 1)
106+
end
107+
108+
assert_equal RedisClient::CommandError, @first.value.class
109+
assert_equal true, @second.value
110+
assert_equal false, @third.value
111+
end
112+
101113
def test_assignment_of_results_inside_a_nested_block
102114
r.pipelined do |p|
103115
@first = p.sadd?("foo", 1)
@@ -111,6 +123,30 @@ def test_assignment_of_results_inside_a_nested_block
111123
assert_equal false, @second.value
112124
end
113125

126+
def test_nested_pipelining_returns_without_raising_exception
127+
result = r.pipelined(exception: false) do |p1|
128+
p1.doesnt_exist
129+
p1.set("foo", "42")
130+
p1.pipelined do |p2|
131+
p2.doesnt_exist_again
132+
p2.set("bar", "99")
133+
end
134+
end
135+
136+
assert result[0].is_a?(RedisClient::CommandError)
137+
assert_equal ["doesnt_exist"], result[0].command
138+
139+
assert_equal "OK", result[1]
140+
141+
assert result[2].is_a?(RedisClient::CommandError)
142+
assert_equal ["doesnt_exist_again"], result[2].command
143+
144+
assert_equal "OK", result[3]
145+
146+
assert_equal "42", r.get("foo")
147+
assert_equal "99", r.get("bar")
148+
end
149+
114150
def test_futures_raise_when_confused_with_something_else
115151
r.pipelined do |p|
116152
@result = p.sadd?("foo", 1)

0 commit comments

Comments
 (0)