2020-01-12 18:12:27 -05:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require_relative "shared_context_for_backup_restore"
|
|
|
|
|
2022-07-27 22:27:38 -04:00
|
|
|
RSpec.describe BackupRestore::SystemInterface do
|
2023-06-21 10:00:19 -04:00
|
|
|
subject(:system_interface) { BackupRestore::SystemInterface.new(logger) }
|
2020-01-12 18:12:27 -05:00
|
|
|
|
2024-02-29 19:07:35 -05:00
|
|
|
include_context "with shared backup restore context"
|
2020-01-12 18:12:27 -05:00
|
|
|
|
2022-07-27 12:14:14 -04:00
|
|
|
describe "readonly mode" do
|
2020-10-16 09:19:02 -04:00
|
|
|
after { Discourse::READONLY_KEYS.each { |key| Discourse.redis.del(key) } }
|
2020-01-12 18:12:27 -05:00
|
|
|
|
|
|
|
describe "#enable_readonly_mode" do
|
|
|
|
it "enables readonly mode" do
|
|
|
|
Discourse.expects(:enable_readonly_mode).once
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.enable_readonly_mode
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "does not enable readonly mode when it is already in readonly mode" do
|
|
|
|
Discourse.enable_readonly_mode
|
|
|
|
Discourse.expects(:enable_readonly_mode).never
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.enable_readonly_mode
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "#disable_readonly_mode" do
|
|
|
|
it "disables readonly mode" do
|
|
|
|
Discourse.expects(:disable_readonly_mode).once
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.disable_readonly_mode
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "does not disable readonly mode when readonly mode was explicitly enabled" do
|
|
|
|
Discourse.enable_readonly_mode
|
|
|
|
Discourse.expects(:disable_readonly_mode).never
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.disable_readonly_mode
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "#mark_restore_as_running" do
|
|
|
|
it "calls mark_restore_as_running" do
|
|
|
|
BackupRestore.expects(:mark_as_running!).once
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.mark_restore_as_running
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "#mark_restore_as_not_running" do
|
|
|
|
it "calls mark_restore_as_not_running" do
|
|
|
|
BackupRestore.expects(:mark_as_not_running!).once
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.mark_restore_as_not_running
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "#listen_for_shutdown_signal" do
|
|
|
|
before { BackupRestore.mark_as_running! }
|
|
|
|
|
|
|
|
after do
|
|
|
|
BackupRestore.clear_shutdown_signal!
|
|
|
|
BackupRestore.mark_as_not_running!
|
|
|
|
end
|
|
|
|
|
|
|
|
it "exits the process when shutdown signal is set" do
|
|
|
|
expect do
|
2023-06-21 10:00:19 -04:00
|
|
|
thread = system_interface.listen_for_shutdown_signal
|
2020-01-12 18:12:27 -05:00
|
|
|
BackupRestore.set_shutdown_signal!
|
|
|
|
thread.join
|
|
|
|
end.to raise_error(SystemExit)
|
|
|
|
end
|
2020-10-13 10:41:43 -04:00
|
|
|
|
|
|
|
it "clears an existing shutdown signal before it starts to listen" do
|
|
|
|
BackupRestore.set_shutdown_signal!
|
|
|
|
expect(BackupRestore.should_shutdown?).to eq(true)
|
|
|
|
|
2023-06-21 10:00:19 -04:00
|
|
|
thread = system_interface.listen_for_shutdown_signal
|
2020-10-13 10:41:43 -04:00
|
|
|
expect(BackupRestore.should_shutdown?).to eq(false)
|
|
|
|
Thread.kill(thread)
|
|
|
|
end
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
describe "#pause_sidekiq" do
|
2020-10-16 09:19:02 -04:00
|
|
|
after { Sidekiq.unpause! }
|
|
|
|
|
2020-01-12 18:12:27 -05:00
|
|
|
it "calls pause!" do
|
2020-10-16 09:19:02 -04:00
|
|
|
expect(Sidekiq.paused?).to eq(false)
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.pause_sidekiq("my reason")
|
2020-10-16 09:19:02 -04:00
|
|
|
expect(Sidekiq.paused?).to eq(true)
|
|
|
|
expect(Discourse.redis.get(SidekiqPauser::PAUSED_KEY)).to eq("my reason")
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "#unpause_sidekiq" do
|
|
|
|
it "calls unpause!" do
|
2020-10-16 09:19:02 -04:00
|
|
|
Sidekiq.pause!
|
|
|
|
expect(Sidekiq.paused?).to eq(true)
|
|
|
|
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.unpause_sidekiq
|
2020-10-16 09:19:02 -04:00
|
|
|
expect(Sidekiq.paused?).to eq(false)
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe "#wait_for_sidekiq" do
|
|
|
|
it "waits 6 seconds even when there are no running Sidekiq jobs" do
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.expects(:sleep).with(6).once
|
|
|
|
system_interface.wait_for_sidekiq
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
context "with Sidekiq workers" do
|
2022-04-20 03:36:19 -04:00
|
|
|
after { flush_sidekiq_redis_namespace }
|
2020-01-12 18:12:27 -05:00
|
|
|
|
2020-10-16 09:19:02 -04:00
|
|
|
def flush_sidekiq_redis_namespace
|
2022-04-20 03:36:19 -04:00
|
|
|
Sidekiq.redis { |redis| redis.scan_each { |key| redis.del(key) } }
|
2020-10-16 09:19:02 -04:00
|
|
|
end
|
2020-01-12 18:12:27 -05:00
|
|
|
|
2020-10-16 09:19:02 -04:00
|
|
|
def create_workers(site_id: nil, all_sites: false)
|
2020-01-12 18:12:27 -05:00
|
|
|
payload =
|
|
|
|
Sidekiq::Testing.fake! do
|
|
|
|
data = { post_id: 1 }
|
2023-01-09 06:18:21 -05:00
|
|
|
|
2020-01-12 18:12:27 -05:00
|
|
|
if all_sites
|
|
|
|
data[:all_sites] = true
|
|
|
|
else
|
|
|
|
data[:current_site_id] = site_id || RailsMultisite::ConnectionManagement.current_db
|
2023-01-09 06:18:21 -05:00
|
|
|
end
|
|
|
|
|
2020-01-12 18:12:27 -05:00
|
|
|
Jobs.enqueue(:process_post, data)
|
|
|
|
Jobs::ProcessPost.jobs.last
|
|
|
|
end
|
|
|
|
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
hostname = "localhost"
|
|
|
|
pid = 7890
|
|
|
|
key = "#{hostname}:#{pid}"
|
|
|
|
process = { pid: pid, hostname: hostname }
|
|
|
|
|
2023-08-28 00:58:47 -04:00
|
|
|
conn.sadd?("processes", key)
|
2020-01-12 18:12:27 -05:00
|
|
|
conn.hmset(key, "info", Sidekiq.dump_json(process))
|
|
|
|
|
|
|
|
data =
|
|
|
|
Sidekiq.dump_json(
|
|
|
|
queue: "default",
|
|
|
|
run_at: Time.now.to_i,
|
|
|
|
payload: Sidekiq.dump_json(payload),
|
|
|
|
)
|
2022-04-20 03:36:19 -04:00
|
|
|
conn.hmset("#{key}:work", "444", data)
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it "waits up to 60 seconds for jobs running for the current site to finish" do
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.expects(:sleep).with(6).times(10)
|
2020-01-12 18:12:27 -05:00
|
|
|
create_workers
|
2023-06-21 10:00:19 -04:00
|
|
|
expect { system_interface.wait_for_sidekiq }.to raise_error(
|
|
|
|
BackupRestore::RunningSidekiqJobsError,
|
|
|
|
)
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "waits up to 60 seconds for jobs running on all sites to finish" do
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.expects(:sleep).with(6).times(10)
|
2020-01-12 18:12:27 -05:00
|
|
|
create_workers(all_sites: true)
|
2023-06-21 10:00:19 -04:00
|
|
|
expect { system_interface.wait_for_sidekiq }.to raise_error(
|
|
|
|
BackupRestore::RunningSidekiqJobsError,
|
|
|
|
)
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
it "ignores jobs of other sites" do
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.expects(:sleep).with(6).once
|
2020-01-12 18:12:27 -05:00
|
|
|
create_workers(site_id: "another_site")
|
|
|
|
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.wait_for_sidekiq
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
end
|
2021-07-07 12:57:42 -04:00
|
|
|
end
|
2020-10-16 09:19:02 -04:00
|
|
|
|
2021-07-07 12:57:42 -04:00
|
|
|
describe "#flush_redis" do
|
2022-07-27 12:14:14 -04:00
|
|
|
context "with Sidekiq" do
|
2021-07-07 12:57:42 -04:00
|
|
|
after { Sidekiq.unpause! }
|
2020-10-16 09:19:02 -04:00
|
|
|
|
2021-07-07 12:57:42 -04:00
|
|
|
it "doesn't unpause Sidekiq" do
|
|
|
|
Sidekiq.pause!
|
2023-06-21 10:00:19 -04:00
|
|
|
system_interface.flush_redis
|
2020-10-16 09:19:02 -04:00
|
|
|
|
2021-07-07 12:57:42 -04:00
|
|
|
expect(Sidekiq.paused?).to eq(true)
|
2020-10-16 09:19:02 -04:00
|
|
|
end
|
|
|
|
end
|
2020-01-12 18:12:27 -05:00
|
|
|
end
|
|
|
|
end
|