/
semaphore_spec.rb
183 lines (156 loc) · 4.59 KB
/
semaphore_spec.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
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
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
RSpec.shared_examples :semaphore do
let(:semaphore) { described_class.new(3) }
describe '#initialize' do
it 'raises an exception if the initial count is not an integer' do
expect {
described_class.new('foo')
}.to raise_error(ArgumentError)
end
context 'when initializing with 0' do
let(:semaphore) { described_class.new(0) }
it do
expect(semaphore).to_not be nil
end
end
context 'when initializing with -1' do
let(:semaphore) { described_class.new(-1) }
it do
semaphore.release
expect(semaphore.available_permits).to eq 0
end
end
end
describe '#acquire' do
context 'permits available' do
it 'should return true immediately' do
result = semaphore.acquire
expect(result).to be_nil
end
end
context 'not enough permits available' do
it 'should block thread until permits are available' do
semaphore.drain_permits
Thread.new { sleep(0.2); semaphore.release }
result = semaphore.acquire
expect(result).to be_nil
expect(semaphore.available_permits).to eq 0
end
end
context 'when acquiring negative permits' do
it do
expect {
semaphore.acquire(-1)
}.to raise_error(ArgumentError)
end
end
end
describe '#drain_permits' do
it 'drains all available permits' do
drained = semaphore.drain_permits
expect(drained).to eq 3
expect(semaphore.available_permits).to eq 0
end
it 'drains nothing in no permits are available' do
semaphore.reduce_permits 3
drained = semaphore.drain_permits
expect(drained).to eq 0
end
end
describe '#try_acquire' do
context 'without timeout' do
it 'acquires immediately if permits are available' do
result = semaphore.try_acquire(1)
expect(result).to be_truthy
end
it 'returns false immediately in no permits are available' do
result = semaphore.try_acquire(20)
expect(result).to be_falsey
end
context 'when trying to acquire negative permits' do
it do
expect {
semaphore.try_acquire(-1)
}.to raise_error(ArgumentError)
end
end
end
context 'with timeout' do
it 'acquires immediately if permits are available' do
result = semaphore.try_acquire(1, 5)
expect(result).to be_truthy
end
it 'acquires when permits are available within timeout' do
semaphore.drain_permits
Thread.new { sleep 0.1; semaphore.release }
result = semaphore.try_acquire(1, 1)
expect(result).to be_truthy
end
it 'returns false on timeout' do
semaphore.drain_permits
result = semaphore.try_acquire(1, 0.1)
expect(result).to be_falsey
end
end
end
describe '#reduce_permits' do
it 'raises ArgumentError if reducing by negative number' do
expect {
semaphore.reduce_permits(-1)
}.to raise_error(ArgumentError)
end
it 'reduces permits below zero' do
semaphore.reduce_permits 1003
expect(semaphore.available_permits).to eq(-1000)
end
it 'reduces permits' do
semaphore.reduce_permits 1
expect(semaphore.available_permits).to eq 2
semaphore.reduce_permits 2
expect(semaphore.available_permits).to eq 0
end
it 'reduces zero permits' do
semaphore.reduce_permits 0
expect(semaphore.available_permits).to eq 3
end
end
describe '#release' do
it 'increases the number of available permits by one' do
semaphore.release
expect(semaphore.available_permits).to eq 4
end
context 'when a number of permits is specified' do
it 'increases the number of available permits by the specified value' do
semaphore.release(2)
expect(semaphore.available_permits).to eq 5
end
context 'when permits is set to negative number' do
it do
expect {
semaphore.release(-1)
}.to raise_error(ArgumentError)
end
end
end
end
end
module Concurrent
RSpec.describe MutexSemaphore do
it_should_behave_like :semaphore
end
if Concurrent.on_jruby?
RSpec.describe JavaSemaphore do
it_should_behave_like :semaphore
end
end
RSpec.describe Semaphore do
if Concurrent.on_jruby?
it 'inherits from JavaSemaphore' do
expect(Semaphore.ancestors).to include(JavaSemaphore)
end
else
it 'inherits from MutexSemaphore' do
expect(Semaphore.ancestors).to include(MutexSemaphore)
end
end
end
end