require "active_support/concern" module RocketJob module Plugins # Automatically retry the job on failure. # # Example: # # class MyJob < RocketJob::Job # include RocketJob::Plugins::Retry # # # Set the maximum number of times a job should be retried before giving up. # self.retry_limit = 3 # # def perform # puts "DONE" # end # end # # # Queue the job for processing using the default cron_schedule specified above # MyJob.create! # # # Override the default retry_limit for a specific job instance. # MyCronJob.create!(retry_limit: 10) # # # Disable retries for this job instance. # MyCronJob.create!(retry_limit: 0) # module Retry extend ActiveSupport::Concern included do after_fail :rocket_job_retry # Maximum number of times to retry this job # 25 is approximately 3 weeks of retries field :retry_limit, type: Integer, default: 25, class_attribute: true, user_editable: true, copy_on_restart: true # List of times when this job failed field :failed_at_list, type: Array, default: [] validates_presence_of :retry_limit end # Returns [true|false] whether this job should be retried on failure. def rocket_job_retry_on_fail? rocket_job_failure_count < retry_limit end def rocket_job_failure_count failed_at_list.size end private def rocket_job_retry # Failure count is incremented during before_fail return if expired? || !rocket_job_retry_on_fail? delay_seconds = rocket_job_retry_seconds_to_delay logger.info "Job failed, automatically retrying in #{delay_seconds} seconds. Retry count: #{failure_count}" now = Time.now self.run_at = now + delay_seconds failed_at_list << now new_record? ? self.retry : retry! end # Prevent exception from being cleared on retry def rocket_job_clear_exception self.completed_at = nil self.exception = nil unless rocket_job_retry_on_fail? self.worker_name = nil end # Returns [Time] when to retry this job at # Same basic formula as Delayed Job def rocket_job_retry_seconds_to_delay (rocket_job_failure_count**4) + 15 + (rand(30) * (rocket_job_failure_count + 1)) end end end end