whenever/lib/outputs/cron.rb

126 lines
3.8 KiB
Ruby
Raw Normal View History

2009-02-16 03:24:10 +00:00
module Whenever
module Output
2009-02-16 03:24:10 +00:00
class Cron
2009-02-16 03:24:10 +00:00
attr_accessor :time, :task
2009-02-16 03:24:10 +00:00
def initialize(time = nil, task = nil, at = nil)
@time = time
@task = task
@at = at.is_a?(String) ? (Chronic.parse(at) || 0) : (at || 0)
end
2009-02-16 03:24:10 +00:00
def self.output(time, job)
if job.at and job.at.is_a?(String) and !job.at.empty?
ats = job.at.split(',')
else
ats = [job.at]
end
ats.each do |at|
out = new(time, job.output, at)
yield "#{out.time_in_cron_syntax} #{out.task}"
end
2009-02-16 03:24:10 +00:00
end
2009-02-16 03:24:10 +00:00
def time_in_cron_syntax
case @time
when Symbol then parse_symbol
when String then parse_as_string
else parse_time
end
end
2009-02-16 03:24:10 +00:00
protected
2009-02-16 03:24:10 +00:00
def parse_symbol
shortcut = case @time
2009-02-16 03:24:10 +00:00
when :reboot then '@reboot'
when :year, :yearly then '@annually'
when :day, :daily then '@daily'
when :midnight then '@midnight'
when :month, :monthly then '@monthly'
when :week, :weekly then '@weekly'
when :hour, :hourly then '@hourly'
end
if shortcut
if @at > 0
raise ArgumentError, "You cannot specify an ':at' when using the shortcuts for times."
else
return shortcut
end
else
parse_as_string
2009-02-16 03:24:10 +00:00
end
end
2009-02-16 03:24:10 +00:00
def parse_time
timing = Array.new(5, '*')
case @time
when 0.seconds...1.minute
raise ArgumentError, "Time must be in minutes or higher"
when 1.minute...1.hour
minute_frequency = @time / 60
timing[0] = comma_separated_timing(minute_frequency, 59)
when 1.hour...1.day
hour_frequency = (@time / 60 / 60).round
timing[0] = @at.is_a?(Time) ? @at.min : @at
2009-02-16 03:24:10 +00:00
timing[1] = comma_separated_timing(hour_frequency, 23)
when 1.day...1.month
day_frequency = (@time / 24 / 60 / 60).round
timing[0] = @at.is_a?(Time) ? @at.min : 0
timing[1] = @at.is_a?(Time) ? @at.hour : @at
timing[2] = comma_separated_timing(day_frequency, 31, 1)
when 1.month..12.months
month_frequency = (@time / 30 / 24 / 60 / 60).round
timing[0] = @at.is_a?(Time) ? @at.min : 0
timing[1] = @at.is_a?(Time) ? @at.hour : 0
timing[2] = @at.is_a?(Time) ? @at.day : (@at.zero? ? 1 : @at)
timing[3] = comma_separated_timing(month_frequency, 12, 1)
else
return parse_as_string
end
timing.join(' ')
end
2009-02-16 03:24:10 +00:00
def parse_as_string
return unless @time
string = @time.to_s
timing = Array.new(4, '*')
timing[0] = @at.is_a?(Time) ? @at.min : 0
timing[1] = @at.is_a?(Time) ? @at.hour : 0
return (timing << '1-5') * " " if string.downcase.index('weekday')
return (timing << '6,0') * " " if string.downcase.index('weekend')
%w(sun mon tue wed thu fri sat).each_with_index do |day, i|
return (timing << i) * " " if string.downcase.index(day)
2009-02-16 03:24:10 +00:00
end
2009-02-16 03:24:10 +00:00
raise ArgumentError, "Couldn't parse: #{@time}"
end
2009-02-16 03:24:10 +00:00
def comma_separated_timing(frequency, max, start = 0)
return start if frequency.blank? || frequency.zero?
return '*' if frequency == 1
return frequency if frequency > (max * 0.5).ceil
2009-02-16 03:24:10 +00:00
original_start = start
2009-02-16 03:24:10 +00:00
start += frequency unless (max + 1).modulo(frequency).zero? || start > 0
output = (start..max).step(frequency).to_a
2009-02-16 03:24:10 +00:00
max_occurances = (max.to_f / (frequency.to_f)).round
max_occurances += 1 if original_start.zero?
2009-02-16 03:24:10 +00:00
output[0, max_occurances].join(',')
end
end
2009-02-16 03:24:10 +00:00
end
end