1
0
Fork 0
mirror of https://github.com/mperham/sidekiq.git synced 2022-11-09 13:52:34 -05:00
mperham--sidekiq/lib/sidekiq/fetch.rb

77 lines
2.1 KiB
Ruby
Raw Normal View History

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
def initialize(mgr, queues)
@mgr = mgr
2012-04-03 23:38:03 -04:00
@queues = queues.map { |q| "queue:#{q}" }
@unique_queues = @queues.uniq
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.
def fetch
watchdog('Fetcher#fetch died') do
return if Sidekiq::Fetcher.done?
begin
queue = nil
msg = nil
Sidekiq.redis { |conn| queue, msg = conn.blpop(*queues_cmd) }
if msg
@mgr.assign!(msg, queue.gsub(/.*queue:/, ''))
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
end
end
# 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
2012-04-03 23:19:29 -04:00
private
# 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
queues = @queues.sample(@unique_queues.size).uniq
queues.concat(@unique_queues - queues)
queues << TIMEOUT
2012-04-03 23:19:29 -04:00
end
end
end