2019-07-25 01:21:37 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2017-08-07 13:13:02 -04:00
|
|
|
require 'spec_helper'
|
|
|
|
|
2020-06-24 14:09:03 -04:00
|
|
|
RSpec.describe Gitlab::Daemon do
|
2017-08-07 13:13:02 -04:00
|
|
|
subject { described_class.new }
|
|
|
|
|
|
|
|
before do
|
2019-10-16 05:07:51 -04:00
|
|
|
allow(subject).to receive(:run_thread)
|
2017-08-07 13:13:02 -04:00
|
|
|
allow(subject).to receive(:stop_working)
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '.instance' do
|
|
|
|
before do
|
|
|
|
allow(Kernel).to receive(:at_exit)
|
|
|
|
end
|
|
|
|
|
2017-08-10 18:31:42 -04:00
|
|
|
after do
|
2017-08-07 13:13:02 -04:00
|
|
|
described_class.instance_variable_set(:@instance, nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'provides instance of Daemon' do
|
|
|
|
expect(described_class.instance).to be_instance_of(described_class)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'subsequent invocations provide the same instance' do
|
|
|
|
expect(described_class.instance).to eq(described_class.instance)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'creates at_exit hook when instance is created' do
|
|
|
|
expect(described_class.instance).not_to be_nil
|
|
|
|
|
|
|
|
expect(Kernel).to have_received(:at_exit)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-06-06 11:09:05 -04:00
|
|
|
describe '.initialize_instance' do
|
|
|
|
before do
|
|
|
|
allow(Kernel).to receive(:at_exit)
|
|
|
|
end
|
|
|
|
|
|
|
|
after do
|
|
|
|
described_class.instance_variable_set(:@instance, nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'provides instance of Daemon' do
|
|
|
|
expect(described_class.instance).to be_instance_of(described_class)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when instance has already been created' do
|
|
|
|
before do
|
|
|
|
described_class.instance
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'and recreate flag is false' do
|
|
|
|
it 'raises an error' do
|
|
|
|
expect { described_class.initialize_instance }.to raise_error(/singleton instance already initialized/)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'and recreate flag is true' do
|
|
|
|
it 'calls stop on existing instance and returns new instance' do
|
|
|
|
old_instance = described_class.instance
|
|
|
|
expect(old_instance).to receive(:stop)
|
|
|
|
|
|
|
|
new_instance = described_class.initialize_instance(recreate: true)
|
|
|
|
|
|
|
|
expect(new_instance.object_id).not_to eq(old_instance.object_id)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-08-21 05:32:45 -04:00
|
|
|
context 'when Daemon is enabled' do
|
2017-08-07 13:13:02 -04:00
|
|
|
before do
|
|
|
|
allow(subject).to receive(:enabled?).and_return(true)
|
|
|
|
end
|
|
|
|
|
2019-08-21 05:32:45 -04:00
|
|
|
context 'when Daemon is stopped' do
|
2017-08-07 13:13:02 -04:00
|
|
|
describe '#start' do
|
|
|
|
it 'starts the Daemon' do
|
|
|
|
expect { subject.start.join }.to change { subject.thread? }.from(false).to(true)
|
|
|
|
|
2019-10-16 05:07:51 -04:00
|
|
|
expect(subject).to have_received(:run_thread)
|
2017-08-07 13:13:02 -04:00
|
|
|
end
|
2021-11-22 07:10:20 -05:00
|
|
|
|
|
|
|
context '@synchronous' do
|
|
|
|
context 'when @synchronous is set to true' do
|
|
|
|
subject { described_class.instance(synchronous: true) }
|
|
|
|
|
|
|
|
it 'calls join on the thread' do
|
|
|
|
# Thread has to be run in a block, expect_next_instance_of does not support this.
|
|
|
|
expect_any_instance_of(Thread).to receive(:join) # rubocop:disable RSpec/AnyInstanceOf
|
|
|
|
|
|
|
|
subject.start
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when @synchronous is not set to a truthy value' do
|
|
|
|
subject { described_class.instance }
|
|
|
|
|
|
|
|
it 'does not call join on the thread' do
|
|
|
|
# Thread has to be run in a block, expect_next_instance_of does not support this.
|
|
|
|
expect_any_instance_of(Thread).not_to receive(:join) # rubocop:disable RSpec/AnyInstanceOf
|
|
|
|
|
|
|
|
subject.start
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2017-08-07 13:13:02 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
describe '#stop' do
|
|
|
|
it "doesn't shutdown stopped Daemon" do
|
|
|
|
expect { subject.stop }.not_to change { subject.thread? }
|
|
|
|
|
2019-10-16 05:07:51 -04:00
|
|
|
expect(subject).not_to have_received(:run_thread)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#start_working' do
|
|
|
|
context 'when start_working fails' do
|
|
|
|
before do
|
|
|
|
expect(subject).to receive(:start_working) { false }
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not start thread' do
|
|
|
|
expect(subject).not_to receive(:run_thread)
|
|
|
|
|
|
|
|
expect(subject.start).to eq(nil)
|
2017-08-07 13:13:02 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-08-21 05:32:45 -04:00
|
|
|
context 'when Daemon is running' do
|
2017-08-07 13:13:02 -04:00
|
|
|
before do
|
2019-08-21 05:32:45 -04:00
|
|
|
subject.start
|
2017-08-07 13:13:02 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
describe '#start' do
|
|
|
|
it "doesn't start running Daemon" do
|
2019-08-21 05:32:45 -04:00
|
|
|
expect { subject.start.join }.not_to change { subject.thread }
|
2017-08-07 13:13:02 -04:00
|
|
|
|
2019-10-16 05:07:51 -04:00
|
|
|
expect(subject).to have_received(:run_thread).once
|
2017-08-07 13:13:02 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#stop' do
|
|
|
|
it 'shutdowns Daemon' do
|
|
|
|
expect { subject.stop }.to change { subject.thread? }.from(true).to(false)
|
|
|
|
|
|
|
|
expect(subject).to have_received(:stop_working)
|
|
|
|
end
|
2019-08-21 05:32:45 -04:00
|
|
|
|
|
|
|
context 'when stop_working raises exception' do
|
|
|
|
before do
|
2019-10-16 05:07:51 -04:00
|
|
|
allow(subject).to receive(:run_thread) do
|
2019-08-21 05:32:45 -04:00
|
|
|
sleep(1000)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'shutdowns Daemon' do
|
|
|
|
expect(subject).to receive(:stop_working) do
|
|
|
|
subject.thread.raise(Interrupt)
|
|
|
|
end
|
|
|
|
|
|
|
|
expect(subject.thread).to be_alive
|
|
|
|
expect { subject.stop }.not_to raise_error
|
|
|
|
expect(subject.thread).to be_nil
|
|
|
|
end
|
|
|
|
end
|
2017-08-07 13:13:02 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-08-21 05:32:45 -04:00
|
|
|
context 'when Daemon is disabled' do
|
2017-08-07 13:13:02 -04:00
|
|
|
before do
|
|
|
|
allow(subject).to receive(:enabled?).and_return(false)
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#start' do
|
|
|
|
it "doesn't start working" do
|
|
|
|
expect(subject.start).to be_nil
|
|
|
|
expect { subject.start }.not_to change { subject.thread? }
|
|
|
|
|
2019-10-16 05:07:51 -04:00
|
|
|
expect(subject).not_to have_received(:run_thread)
|
2017-08-07 13:13:02 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#stop' do
|
|
|
|
it "doesn't stop working" do
|
|
|
|
expect { subject.stop }.not_to change { subject.thread? }
|
|
|
|
|
|
|
|
expect(subject).not_to have_received(:stop_working)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|