2013-01-31 12:42:19 -05:00
|
|
|
module Sidetiq
|
|
|
|
configure do |config|
|
2013-01-31 12:50:37 -05:00
|
|
|
config.priority = Thread.main.priority
|
2013-02-01 11:53:34 -05:00
|
|
|
config.resolution = 1
|
|
|
|
config.lock_expire = 1000
|
2013-02-01 13:27:27 -05:00
|
|
|
config.utc = false
|
2013-01-31 12:42:19 -05:00
|
|
|
end
|
|
|
|
|
2013-02-04 07:07:54 -05:00
|
|
|
# Public: The Sidetiq clock.
|
2013-01-31 12:42:19 -05:00
|
|
|
class Clock
|
|
|
|
include Singleton
|
|
|
|
include MonitorMixin
|
|
|
|
|
2013-02-04 07:07:54 -05:00
|
|
|
# Public: Start time offset from epoch used for calculating run
|
|
|
|
# times in the Sidetiq schedules.
|
2013-02-01 13:27:27 -05:00
|
|
|
START_TIME = Sidetiq.config.utc ? Time.utc(2010, 1, 1) : Time.local(2010, 1, 1)
|
2013-02-01 11:53:34 -05:00
|
|
|
|
2013-02-04 07:07:54 -05:00
|
|
|
# Public: Returns a hash of Sidetiq::Schedule instances.
|
|
|
|
attr_reader :schedules
|
|
|
|
|
|
|
|
# Public: Returns the clock thread.
|
|
|
|
attr_reader :thread
|
2013-01-31 12:42:19 -05:00
|
|
|
|
2013-02-04 05:10:53 -05:00
|
|
|
def self.method_missing(meth, *args, &block)
|
|
|
|
instance.__send__(meth, *args, &block)
|
|
|
|
end
|
|
|
|
|
2013-02-04 07:07:54 -05:00
|
|
|
def initialize # :nodoc:
|
2013-01-31 12:42:19 -05:00
|
|
|
super
|
|
|
|
@schedules = {}
|
|
|
|
end
|
|
|
|
|
2013-02-04 07:07:54 -05:00
|
|
|
# Public: Get the schedule for `worker`.
|
|
|
|
#
|
|
|
|
# worker - A Sidekiq::Worker class
|
|
|
|
#
|
|
|
|
# Examples
|
|
|
|
#
|
|
|
|
# schedule_for(MyWorker)
|
|
|
|
# # => Sidetiq::Schedule
|
|
|
|
#
|
|
|
|
# Returns a Sidetiq::Schedule instances.
|
2013-01-31 12:42:19 -05:00
|
|
|
def schedule_for(worker)
|
2013-02-01 11:53:34 -05:00
|
|
|
schedules[worker] ||= Sidetiq::Schedule.new(START_TIME)
|
2013-01-31 12:42:19 -05:00
|
|
|
end
|
|
|
|
|
2013-02-04 07:07:54 -05:00
|
|
|
# Public: Issue a single clock tick.
|
|
|
|
#
|
|
|
|
# Examples
|
|
|
|
#
|
|
|
|
# tick
|
|
|
|
# # => Hash of Sidetiq::Schedule objects
|
|
|
|
#
|
|
|
|
# Returns a hash of Sidetiq::Schedule instances.
|
2013-01-31 12:42:19 -05:00
|
|
|
def tick
|
|
|
|
@tick = gettime
|
|
|
|
synchronize do
|
|
|
|
schedules.each do |worker, schedule|
|
|
|
|
if schedule.schedule_next?(@tick)
|
2013-02-01 11:53:34 -05:00
|
|
|
enqueue(worker, schedule.next_occurrence(@tick))
|
2013-01-31 12:42:19 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-02-04 07:07:54 -05:00
|
|
|
# Public: Returns the current time used by the clock.
|
|
|
|
#
|
|
|
|
# Sidetiq::Clock uses `clock_gettime()` on UNIX systems and
|
|
|
|
# `mach_absolute_time()` on Mac OS X.
|
|
|
|
#
|
|
|
|
# Examples
|
|
|
|
#
|
|
|
|
# gettime
|
|
|
|
# # => 2013-02-04 12:00:45 +0000
|
|
|
|
#
|
|
|
|
# Returns a Time instance.
|
2013-02-01 13:27:27 -05:00
|
|
|
def gettime
|
|
|
|
Sidetiq.config.utc ? clock_gettime.utc : clock_gettime
|
|
|
|
end
|
|
|
|
|
2013-02-04 07:07:54 -05:00
|
|
|
# Public: Starts the clock unless it is already running.
|
|
|
|
#
|
|
|
|
# Examples
|
|
|
|
#
|
|
|
|
# start!
|
|
|
|
# # => Thread
|
|
|
|
#
|
|
|
|
# Returns the Thread instance of the clock thread.
|
2013-02-04 05:51:43 -05:00
|
|
|
def start!
|
|
|
|
return if ticking?
|
|
|
|
|
|
|
|
Sidekiq.logger.info "Sidetiq::Clock start"
|
|
|
|
@thread = Thread.start { clock { tick } }
|
|
|
|
@thread.abort_on_exception = true
|
|
|
|
@thread.priority = Sidetiq.config.resolution
|
2013-02-04 07:07:54 -05:00
|
|
|
@thread
|
2013-02-04 05:51:43 -05:00
|
|
|
end
|
|
|
|
|
2013-02-04 07:07:54 -05:00
|
|
|
# Public: Stops the clock if it is running.
|
|
|
|
#
|
|
|
|
# Examples
|
|
|
|
#
|
|
|
|
# stop!
|
|
|
|
# # => nil
|
|
|
|
#
|
|
|
|
# Returns nil.
|
2013-02-04 05:51:43 -05:00
|
|
|
def stop!
|
|
|
|
if ticking?
|
|
|
|
@thread.kill
|
|
|
|
Sidekiq.logger.info "Sidetiq::Clock stop"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-02-04 07:07:54 -05:00
|
|
|
# Public: Returns the status of the clock.
|
|
|
|
#
|
|
|
|
# Examples
|
|
|
|
#
|
|
|
|
# ticking?
|
|
|
|
# # => false
|
|
|
|
#
|
|
|
|
# start!
|
|
|
|
# ticking?
|
|
|
|
# # => true
|
|
|
|
#
|
|
|
|
# Returns true or false.
|
2013-02-04 05:51:43 -05:00
|
|
|
def ticking?
|
|
|
|
@thread && @thread.alive?
|
|
|
|
end
|
|
|
|
|
2013-01-31 12:42:19 -05:00
|
|
|
private
|
|
|
|
|
2013-02-01 11:53:34 -05:00
|
|
|
def enqueue(worker, time)
|
|
|
|
key = "sidetiq:#{worker.name}"
|
|
|
|
|
|
|
|
synchronize_clockworks("#{key}:lock") do |redis|
|
|
|
|
status = redis.get(key)
|
|
|
|
|
|
|
|
if status.nil? || status.to_f < time.to_f
|
|
|
|
time_f = time.to_f
|
|
|
|
Sidekiq.logger.info "Sidetiq::Clock enqueue #{worker.name} (at: #{time_f})"
|
|
|
|
redis.set(key, time_f)
|
|
|
|
worker.perform_at(time)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def synchronize_clockworks(lock)
|
|
|
|
Sidekiq.redis do |redis|
|
|
|
|
if redis.setnx(lock, 1)
|
|
|
|
Sidekiq.logger.debug "Sidetiq::Clock lock #{lock} #{Thread.current.inspect}"
|
|
|
|
|
|
|
|
redis.pexpire(lock, Sidetiq.config.lock_expire)
|
|
|
|
yield redis
|
|
|
|
redis.del(lock)
|
|
|
|
|
|
|
|
Sidekiq.logger.debug "Sidetiq::Clock unlock #{lock} #{Thread.current.inspect}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-01-31 12:42:19 -05:00
|
|
|
def clock
|
|
|
|
loop do
|
|
|
|
yield
|
|
|
|
Thread.pass
|
|
|
|
sleep Sidetiq.config.resolution
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|