2018-07-03 03:20:27 -04:00
|
|
|
require 'spec_helper'
|
|
|
|
|
|
|
|
describe Gitlab::ExclusiveLeaseHelpers, :clean_gitlab_redis_shared_state do
|
|
|
|
include ::ExclusiveLeaseHelpers
|
|
|
|
|
|
|
|
let(:class_instance) { (Class.new { include ::Gitlab::ExclusiveLeaseHelpers }).new }
|
|
|
|
let(:unique_key) { SecureRandom.hex(10) }
|
|
|
|
|
|
|
|
describe '#in_lock' do
|
|
|
|
subject { class_instance.in_lock(unique_key, **options) { } }
|
|
|
|
|
2018-07-04 00:28:18 -04:00
|
|
|
let(:options) { {} }
|
2018-07-03 03:20:27 -04:00
|
|
|
|
|
|
|
context 'when the lease is not obtained yet' do
|
|
|
|
before do
|
|
|
|
stub_exclusive_lease(unique_key, 'uuid')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'calls the given block' do
|
|
|
|
expect { |b| class_instance.in_lock(unique_key, &b) }.to yield_control.once
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'calls the given block continuously' do
|
|
|
|
expect { |b| class_instance.in_lock(unique_key, &b) }.to yield_control.once
|
|
|
|
expect { |b| class_instance.in_lock(unique_key, &b) }.to yield_control.once
|
|
|
|
expect { |b| class_instance.in_lock(unique_key, &b) }.to yield_control.once
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'cancels the exclusive lease after the block' do
|
|
|
|
expect_to_cancel_exclusive_lease(unique_key, 'uuid')
|
|
|
|
|
|
|
|
subject
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the lease is obtained already' do
|
|
|
|
let!(:lease) { stub_exclusive_lease_taken(unique_key) }
|
|
|
|
|
|
|
|
it 'retries to obtain a lease and raises an error' do
|
|
|
|
expect(lease).to receive(:try_obtain).exactly(11).times
|
|
|
|
|
|
|
|
expect { subject }.to raise_error('Failed to obtain a lock')
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when ttl is specified' do
|
2018-07-04 00:28:18 -04:00
|
|
|
let(:options) { { ttl: 10.minutes } }
|
2018-07-03 03:20:27 -04:00
|
|
|
|
|
|
|
it 'receives the specified argument' do
|
2018-07-04 00:28:18 -04:00
|
|
|
expect(Gitlab::ExclusiveLease).to receive(:new).with(unique_key, { timeout: 10.minutes } )
|
|
|
|
|
2018-07-03 03:20:27 -04:00
|
|
|
expect { subject }.to raise_error('Failed to obtain a lock')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when retry count is specified' do
|
|
|
|
let(:options) { { retries: 3 } }
|
|
|
|
|
|
|
|
it 'retries for the specified times' do
|
|
|
|
expect(lease).to receive(:try_obtain).exactly(4).times
|
2018-07-04 00:28:18 -04:00
|
|
|
|
2018-07-03 03:20:27 -04:00
|
|
|
expect { subject }.to raise_error('Failed to obtain a lock')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when sleep second is specified' do
|
2018-07-04 00:28:18 -04:00
|
|
|
let(:options) { { retries: 0, sleep_sec: 0.05.seconds } }
|
2018-07-03 03:20:27 -04:00
|
|
|
|
|
|
|
it 'receives the specified argument' do
|
2018-07-04 00:28:18 -04:00
|
|
|
expect(class_instance).to receive(:sleep).with(0.05.seconds).once
|
|
|
|
|
2018-07-03 03:20:27 -04:00
|
|
|
expect { subject }.to raise_error('Failed to obtain a lock')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|