# async-observer - Rails plugin for asynchronous job execution # Copyright (C) 2007 Philotic Inc. # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # You should have received a copy of the GNU General Public License # along with this program. If not, see . require 'beanstalker/queue' module Beanstalker; end class Beanstalker::Worker SLEEP_TIME = 60 if !defined?(SLEEP_TIME) # rails loads this file twice class << self attr_accessor :finish attr_accessor :custom_error_handler attr_accessor :before_filter attr_writer :handle def handle @handle or raise 'no custom handler is defined' end def error_handler(&block) self.custom_error_handler = block end def before_reserves @before_reserves ||= [] end def before_reserve(&block) before_reserves << block end def run_before_reserve before_reserves.each {|b| b.call} end end def logger $logger or RAILS_DEFAULT_LOGGER end def initialize(top_binding, options = {}) @top_binding = top_binding @stop = false @options = options if @options && @options[:servers] Beanstalker::Queue.queue = Beanstalk::Pool.new(@options[:servers]) end end def main_loop trap('TERM') { @stop = true } loop do break if @stop safe_dispatch(get_job) end end def startup tube = @options[:tube] || "default" logger.info "Using tube #{tube}" Beanstalker::Queue.queue.watch(tube) flush_logger end def shutdown do_all_work end def run startup main_loop rescue Interrupt shutdown end def q_hint @q_hint || Beanstalker::Queue.queue end # This heuristic is to help prevent one queue from starving. The idea is that # if the connection returns a job right away, it probably has more available. # But if it takes time, then it's probably empty. So reuse the same # connection as long as it stays fast. Otherwise, have no preference. def reserve_and_set_hint t1 = Time.now.utc return job = q_hint.reserve ensure t2 = Time.now.utc @q_hint = if brief?(t1, t2) and job then job.conn else nil end end def brief?(t1, t2) ((t2 - t1) * 100).to_i.abs < 10 end def get_job loop do begin Beanstalker::Queue.queue.connect self.class.run_before_reserve return reserve_and_set_hint rescue Interrupt => ex raise ex rescue SignalException => ex raise ex rescue Beanstalk::DeadlineSoonError # Do nothing; immediately try again, giving the user a chance to # clean up in the before_reserve hook. logger.info 'Job deadline soon; you should clean up.' rescue Exception => ex @q_hint = nil # in case there's something wrong with this conn logger.info( "#{ex.class}: #{ex}\n" + ex.backtrace.join("\n")) logger.info 'something is wrong. We failed to get a job.' logger.info "sleeping for #{SLEEP_TIME}s..." sleep(SLEEP_TIME) end end end def dispatch(job) ActiveRecord::Base.verify_active_connections! return run_ao_job(job) if beanstalker_job?(job) return run_other(job) end def safe_dispatch(job) logger.info "got #{job.inspect}:\n" + job.body job.stats.each do |k,v| logger.debug "#{k}=#{v}" end begin return dispatch(job) rescue Interrupt => ex begin job.release rescue :ok end raise ex rescue Exception => ex handle_error(job, ex) ensure Daemonizer.flush_logger end end def handle_error(job, ex) if self.class.custom_error_handler self.class.custom_error_handler.call(job, ex) else self.class.default_handle_error(job, ex) end end def self.default_handle_error(job, ex) logger.info "Job failed: #{job.server}/#{job.id}" logger.info("#{ex.class}: #{ex}\n" + ex.backtrace.join("\n")) if job.stats['releases'] > 10 job.bury logger.info "BURY job due to many releases" else job.decay end rescue Beanstalk::UnexpectedResponse end def run_ao_job(job) logger.info 'running as async observer job' f = self.class.before_filter f.call(job) if f job.delete if job.ybody[:delete_first] run_code(job) job.delete unless job.ybody[:delete_first] rescue ActiveRecord::RecordNotFound => ex logger.warn "Record not found. Doing decay" unless job.ybody[:delete_first] if job.age > 60 job.delete # it's old; this error is most likely permanent else job.decay # it could be replication delay so retry quietly end end end def run_code(job) eval(job.ybody[:code], @top_binding, "(beanstalk job #{job.id})", 1) end def beanstalker_job?(job) begin job.ybody[:type] == :rails rescue false end end def run_other(job) logger.info 'trying custom handler' self.class.handle.call(job) end def do_all_work logger.info 'finishing all running jobs. interrupt again to kill them.' f = self.class.finish f.call if f end end