2012-03-24 16:28:18 -04:00
|
|
|
require 'sidekiq'
|
|
|
|
require 'celluloid'
|
|
|
|
|
|
|
|
module Sidekiq
|
|
|
|
##
|
|
|
|
# The Fetcher blocks on Redis, waiting for a message to process
|
|
|
|
# from the queues. It gets the message and hands it to the Manager
|
|
|
|
# to assign to a ready Processor.
|
|
|
|
class Fetcher
|
|
|
|
include Celluloid
|
|
|
|
include Sidekiq::Util
|
|
|
|
|
2012-03-25 22:52:15 -04:00
|
|
|
TIMEOUT = 1
|
|
|
|
|
2013-01-06 00:17:08 -05:00
|
|
|
def initialize(mgr, options)
|
2012-03-24 16:28:18 -04:00
|
|
|
@mgr = mgr
|
2013-02-05 18:11:41 -05:00
|
|
|
@strategy = Fetcher.strategy.new(options)
|
2012-03-24 16:28:18 -04:00
|
|
|
end
|
|
|
|
|
2012-03-25 22:52:15 -04:00
|
|
|
# Fetching is straightforward: the Manager makes a fetch
|
|
|
|
# request for each idle processor when Sidekiq starts and
|
|
|
|
# then issues a new fetch request every time a Processor
|
|
|
|
# finishes a message.
|
|
|
|
#
|
|
|
|
# Because we have to shut down cleanly, we can't block
|
|
|
|
# forever and we can't loop forever. Instead we reschedule
|
|
|
|
# a new fetch if the current fetch turned up nothing.
|
2012-03-24 16:28:18 -04:00
|
|
|
def fetch
|
|
|
|
watchdog('Fetcher#fetch died') do
|
2012-05-12 00:25:38 -04:00
|
|
|
return if Sidekiq::Fetcher.done?
|
|
|
|
|
2012-04-18 19:31:17 -04:00
|
|
|
begin
|
2013-01-06 00:17:08 -05:00
|
|
|
work = @strategy.retrieve_work
|
2012-03-24 16:28:18 -04:00
|
|
|
|
2013-01-06 00:17:08 -05:00
|
|
|
if work
|
|
|
|
@mgr.async.assign(work)
|
2012-04-18 19:31:17 -04:00
|
|
|
else
|
|
|
|
after(0) { fetch }
|
|
|
|
end
|
|
|
|
rescue => ex
|
2012-04-18 23:13:10 -04:00
|
|
|
logger.error("Error fetching message: #{ex}")
|
|
|
|
logger.error(ex.backtrace.first)
|
2012-04-18 20:06:46 -04:00
|
|
|
sleep(TIMEOUT)
|
|
|
|
after(0) { fetch }
|
2012-03-30 23:59:08 -04:00
|
|
|
end
|
2012-03-24 16:28:18 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2012-05-12 00:25:38 -04:00
|
|
|
# Ugh. Say hello to a bloody hack.
|
|
|
|
# Can't find a clean way to get the fetcher to just stop processing
|
|
|
|
# its mailbox when shutdown starts.
|
|
|
|
def self.done!
|
|
|
|
@done = true
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.done?
|
|
|
|
@done
|
|
|
|
end
|
2013-02-05 18:11:41 -05:00
|
|
|
|
|
|
|
def self.strategy
|
|
|
|
Sidekiq.options[:fetch] || BasicFetch
|
|
|
|
end
|
2013-01-06 00:17:08 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
class BasicFetch
|
|
|
|
def initialize(options)
|
|
|
|
@strictly_ordered_queues = !!options[:strict]
|
|
|
|
@queues = options[:queues].map { |q| "queue:#{q}" }
|
|
|
|
@unique_queues = @queues.uniq
|
|
|
|
end
|
|
|
|
|
|
|
|
def retrieve_work
|
2013-01-08 11:43:34 -05:00
|
|
|
work = Sidekiq.redis { |conn| conn.brpop(*queues_cmd) }
|
|
|
|
UnitOfWork.new(*work) if work
|
2013-01-06 00:17:08 -05:00
|
|
|
end
|
2012-05-12 00:25:38 -04:00
|
|
|
|
2013-01-17 00:48:21 -05:00
|
|
|
def self.bulk_requeue(inprogress)
|
2013-01-17 00:53:48 -05:00
|
|
|
Sidekiq.logger.debug { "Re-queueing terminated jobs" }
|
2013-01-17 00:48:21 -05:00
|
|
|
jobs_to_requeue = {}
|
|
|
|
inprogress.each do |unit_of_work|
|
|
|
|
jobs_to_requeue[unit_of_work.queue] ||= []
|
|
|
|
jobs_to_requeue[unit_of_work.queue] << unit_of_work.message
|
|
|
|
end
|
|
|
|
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
jobs_to_requeue.each do |queue, jobs|
|
|
|
|
conn.rpush(queue, jobs)
|
|
|
|
end
|
|
|
|
end
|
2013-01-17 00:53:48 -05:00
|
|
|
Sidekiq.logger.info("Pushed #{inprogress.size} messages back to Redis")
|
2013-01-17 00:48:21 -05:00
|
|
|
end
|
|
|
|
|
2013-01-06 00:17:08 -05:00
|
|
|
UnitOfWork = Struct.new(:queue, :message) do
|
|
|
|
def acknowledge
|
|
|
|
# nothing to do
|
|
|
|
end
|
|
|
|
|
|
|
|
def queue_name
|
|
|
|
queue.gsub(/.*queue:/, '')
|
|
|
|
end
|
|
|
|
|
|
|
|
def requeue
|
|
|
|
Sidekiq.redis do |conn|
|
|
|
|
conn.rpush(queue, message)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2012-04-03 23:19:29 -04:00
|
|
|
|
|
|
|
# Creating the Redis#blpop command takes into account any
|
|
|
|
# configured queue weights. By default Redis#blpop returns
|
|
|
|
# data from the first queue that has pending elements. We
|
|
|
|
# recreate the queue command each time we invoke Redis#blpop
|
|
|
|
# to honor weights and avoid queue starvation.
|
|
|
|
def queues_cmd
|
2013-01-11 00:59:11 -05:00
|
|
|
queues = @strictly_ordered_queues ? @unique_queues.dup : @queues.shuffle.uniq
|
2013-01-06 00:17:08 -05:00
|
|
|
queues << Sidekiq::Fetcher::TIMEOUT
|
2012-04-03 23:19:29 -04:00
|
|
|
end
|
2012-03-24 16:28:18 -04:00
|
|
|
end
|
|
|
|
end
|