/
transaction_wrapper.rb
119 lines (101 loc) · 2.24 KB
/
transaction_wrapper.rb
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
require 'mock_redis/undef_redis_methods'
class MockRedis
class TransactionWrapper
include UndefRedisMethods
def respond_to?(method, include_private = false)
super || @db.respond_to?(method)
end
def initialize(db)
@db = db
@transaction_futures = []
@multi_stack = []
@multi_block_given = false
end
def method_missing(method, *args, &block)
if in_multi?
future = MockRedis::Future.new([method, *args], block)
@transaction_futures << future
if @multi_block_given
future
else
'QUEUED'
end
else
@db.expire_keys
@db.send(method, *args, &block)
end
end
def initialize_copy(source)
super
@db = @db.clone
@transaction_futures = @transaction_futures.clone
@multi_stack = @multi_stack.clone
end
def discard
unless in_multi?
raise Redis::CommandError, 'ERR DISCARD without MULTI'
end
pop_multi
@transaction_futures = []
'OK'
end
def exec
unless in_multi?
raise Redis::CommandError, 'ERR EXEC without MULTI'
end
pop_multi
return if in_multi?
@multi_block_given = false
responses = @transaction_futures.map do |future|
begin
result = send(*future.command)
future.store_result(result)
future.value
rescue StandardError => e
e
end
end
@transaction_futures = []
responses
end
def in_multi?
@multi_stack.any?
end
def push_multi
@multi_stack.push(@multi_stack.size + 1)
end
def pop_multi
@multi_stack.pop
end
def multi
if block_given?
push_multi
@multi_block_given = true
begin
yield(self)
exec
rescue StandardError => e
discard
raise e
end
else
raise Redis::CommandError, 'ERR MULTI calls can not be nested' if in_multi?
push_multi
'OK'
end
end
def pipelined
yield(self) if block_given?
end
def unwatch
'OK'
end
def watch(*_)
if block_given?
yield self
else
'OK'
end
end
end
end