FEATURE: add APIS for unpausing all sites

This adjusts 53d592ad by @tgxworld

- Adds Sidekiq.upause_all! to unpause all sites
- Adds Sidekiq.paused_dbs to list dbs that are currently paused
- Handles some edge cases where unpause thread could extend expiry on
sites that were unpaused from a different process
- Ensures tests always terminates background thread used for pause
keepalive
This commit is contained in:
Sam 2019-02-14 13:33:58 +11:00
parent 37666c3d18
commit 74d2d4f658
2 changed files with 57 additions and 10 deletions

View File

@ -14,7 +14,6 @@ class SidekiqPauser
@mutex.synchronize do @mutex.synchronize do
extend_lease_thread extend_lease_thread
sleep 0.001 while !paused?
end end
true true
@ -24,10 +23,29 @@ class SidekiqPauser
!!$redis.get(PAUSED_KEY) !!$redis.get(PAUSED_KEY)
end end
def unpause_all!
@mutex.synchronize do
@dbs = Set.new
stop_extend_lease_thread
end
RailsMultisite::ConnectionManagement.each_connection do
unpause! if paused?
end
end
def paused_dbs
dbs = []
RailsMultisite::ConnectionManagement.each_connection do
dbs << RailsMultisite::ConnectionManagement.current_db if paused?
end
dbs
end
def unpause! def unpause!
@mutex.synchronize do @mutex.synchronize do
@dbs.delete(RailsMultisite::ConnectionManagement.current_db) @dbs.delete(RailsMultisite::ConnectionManagement.current_db)
@extend_lease_thread = nil if @dbs.size == 0 stop_extend_lease_thread if @dbs.size == 0
end end
$redis.del(PAUSED_KEY) $redis.del(PAUSED_KEY)
@ -36,16 +54,32 @@ class SidekiqPauser
private private
def extend_lease_thread def stop_extend_lease_thread
@dbs << RailsMultisite::ConnectionManagement.current_db # should always be called from a mutex
if t = @extend_lease_thread
@extend_lease_thread = nil
while t.alive?
t.wakeup
sleep 0
end
end
end
def extend_lease_thread
# should always be called from a mutex
@dbs << RailsMultisite::ConnectionManagement.current_db
@extend_lease_thread ||= Thread.new do @extend_lease_thread ||= Thread.new do
while true do while true do
break unless @mutex.synchronize { @extend_lease_thread }
break if !@extend_lease_thread
@dbs.each do |db| @dbs.each do |db|
RailsMultisite::ConnectionManagement.with_connection(db) do RailsMultisite::ConnectionManagement.with_connection(db) do
$redis.expire PAUSED_KEY, TTL if !$redis.expire(PAUSED_KEY, TTL)
# if it was unpaused in another process we got to remove the
# bad key
@dbs.delete(db)
end
end end
end end
@ -68,6 +102,14 @@ module Sidekiq
def self.unpause! def self.unpause!
@pauser.unpause! @pauser.unpause!
end end
def self.unpause_all!
@pauser.unpause_all!
end
def self.paused_dbs
@pauser.paused_dbs
end
end end
# server middleware that will reschedule work whenever Sidekiq is paused # server middleware that will reschedule work whenever Sidekiq is paused

View File

@ -2,9 +2,6 @@ require 'rails_helper'
require_dependency 'sidekiq/pausable' require_dependency 'sidekiq/pausable'
RSpec.describe "Pausing/Unpausing Sidekiq", type: :multisite do RSpec.describe "Pausing/Unpausing Sidekiq", type: :multisite do
after do
$redis.flushall
end
describe '#pause!, #unpause! and #paused?' do describe '#pause!, #unpause! and #paused?' do
it "can pause and unpause" do it "can pause and unpause" do
@ -23,13 +20,21 @@ RSpec.describe "Pausing/Unpausing Sidekiq", type: :multisite do
Sidekiq.pause! Sidekiq.pause!
expect(Sidekiq.paused?).to eq(true) expect(Sidekiq.paused?).to eq(true)
end end
expect(Sidekiq.paused_dbs).to eq(["second"])
Sidekiq.unpause_all!
RailsMultisite::ConnectionManagement.each_connection do
expect(Sidekiq.paused?).to eq(false)
end
end end
end end
end end
RSpec.describe Sidekiq::Pausable do RSpec.describe Sidekiq::Pausable do
after do after do
$redis.flushall Sidekiq.unpause_all!
end end
it "can still run heartbeats when paused" do it "can still run heartbeats when paused" do