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/manager.rb

126 lines
2.7 KiB
Ruby
Raw Normal View History

2012-01-22 19:01:46 -05:00
require 'celluloid'
require 'redis'
require 'multi_json'
require 'sidekiq/util'
require 'sidekiq/processor'
2012-01-22 19:01:46 -05:00
2012-01-16 19:14:47 -05:00
module Sidekiq
##
# The main router in the system. This
# manages the processor state and fetches messages
# from Redis to be dispatched to an idle processor.
2012-01-16 19:14:47 -05:00
#
class Manager
2012-01-22 14:32:38 -05:00
include Util
2012-01-16 23:02:58 -05:00
include Celluloid
2012-01-16 19:18:36 -05:00
trap_exit :processor_died
2012-01-22 14:32:38 -05:00
class << self
attr_accessor :redis
end
def initialize(options={})
log "Booting sidekiq #{Sidekiq::VERSION} with Redis at #{redis.client.location}"
2012-01-22 19:01:46 -05:00
verbose options.inspect
2012-02-12 02:45:55 -05:00
@count = (options[:processor_count] || 25).to_i
2012-01-22 14:32:38 -05:00
@queues = options[:queues]
2012-02-05 16:22:57 -05:00
@done_callback = nil
2012-01-22 14:32:38 -05:00
2012-01-22 19:01:46 -05:00
@done = false
@busy = []
@ready = @count.times.map { Processor.new_link(current_actor) }
2012-01-22 14:32:38 -05:00
end
def stop
@done = true
@ready.each(&:terminate)
@ready.clear
after(5) do
signal(:shutdown)
2012-01-22 14:32:38 -05:00
end
redis.with_connection do |conn|
conn.smembers('workers').each do |name|
conn.srem('workers', name) if name =~ /:#{Process.pid}:/
end
end
2012-01-22 14:32:38 -05:00
end
def start
2012-01-24 01:08:38 -05:00
dispatch(true)
2012-01-22 14:32:38 -05:00
end
2012-01-16 19:18:36 -05:00
def when_done(&blk)
@done_callback = blk
end
def processor_done(processor)
watchdog('sidekiq processor_done crashed!') do
@done_callback.call(processor) if @done_callback
@busy.delete(processor)
if stopped?
processor.terminate
else
@ready << processor
end
dispatch
2012-01-16 23:02:58 -05:00
end
2012-01-16 19:18:36 -05:00
end
2012-01-16 23:02:58 -05:00
def processor_died(processor, reason)
@busy.delete(processor)
if reason
err "Processor death: #{reason}"
err reason.backtrace.join("\n")
end
2012-01-22 19:01:46 -05:00
unless stopped?
@ready << Processor.new_link(current_actor)
2012-01-22 19:01:46 -05:00
dispatch
end
2012-01-22 14:32:38 -05:00
end
2012-01-22 19:01:46 -05:00
private
def find_work(queue)
msg = redis.lpop("queue:#{queue}")
2012-01-24 01:08:38 -05:00
if msg
processor = @ready.pop
@busy << processor
2012-02-11 02:21:03 -05:00
processor.process!(MultiJson.decode(msg), queue)
2012-01-24 01:08:38 -05:00
end
!!msg
2012-01-24 01:08:38 -05:00
end
def dispatch(schedule = false)
2012-01-22 14:32:38 -05:00
watchdog("Fatal error in sidekiq, dispatch loop died") do
return if stopped?
# Dispatch loop
2012-01-22 14:32:38 -05:00
loop do
break verbose('no processors') if @ready.empty?
found = false
@ready.size.times do
found ||= find_work(@queues.sample)
2012-01-22 14:32:38 -05:00
end
break verbose('nothing to process') unless found
2012-01-22 14:32:38 -05:00
end
# This is the polling loop that ensures we check Redis every
# second for work, even if there was nothing to do this time
# around.
2012-01-24 01:08:38 -05:00
after(1) { verbose('ping'); dispatch(schedule) } if schedule
2012-01-22 14:32:38 -05:00
end
end
def stopped?
@done
end
2012-01-16 19:14:47 -05:00
end
end