10

使用 AirBrake 或 ExceptionNotifier 之类的工具从邮寄延迟的作业中获取错误报告的正确方法是什么?

我试图创建自己的延迟作业类,但由Mailer.welcome()(或类似)创建的邮件对象未正确序列化。我还尝试向and类添加一个error(job, exception)方法,但我相信我得到了更多通常与序列化相关的错误。我尝试了 psych 和 sych 来进行连载。PerformableMailerPerformableMethod

4

2 回答 2

7

更新的解决方案

总的来说,解决方案非常简单。如果您正在对对象(如MyClass.new.delay.some_method)执行延迟作业,那么您需要将错误处理定义为对象方法。如果您在Class (like MyTestMailer.test_email ...) 上执行延迟作业,那么您需要将错误处理定义为类方法。

假设您有一个名为TestMailer. 解决方案是将错误处理定义为类方法,而不是对象方法

# Your rails mailer
class TestMailer

  # Whoa! error has to be a class method!
  def self.error(job, e)
    puts "I can now handle test mailer errors in delayed job!!!!"
  end

end

现在上述def self.error方法将用作延迟作业中的错误回调!

或者,如果您希望能够处理所有操作邮件错误,

class ActionMailer::Base
  def self.error(job, e)
    puts "I can now handle all mailer errors in delayed job!!!"
  end
end

原因在于 DelayedJob 的内部PerformableMethod处理错误的方式。APerformableMethod有两个东西:一个目标对象和一个目标方法。在 Action Mailer 的例子中,目标对象不是一个对象,而是你的 mailer 类TestMailer。目标方法是您使用的邮件方法,例如test_mail. DelayedJob 查找目标对象上的所有钩子(errorbeforeafter等)。但在我们的例子中,目标对象就是类本身。因此,钩子必须定义为类方法。

处理 ActionMailer 邮件的方式DelayedJob有点笨拙。如果添加对象方法而不是类方法,则会引发不需要的异常。例如,这里是代码:

# In <delayed-job-gem>/lib/delayed/performable_method.rb
module Delayed
  class PerformableMethod

    # line #7
    delegate :method, :to => :object

ruby 中的每个对象都有一个method函数,用于获取对该类中方法的原始引用。但是在 DelayedJob - 这个原始method函数已经被委托给其他一些目标对象。这种 hack 使我们无法正常使用该def error功能来处理作业错误。

编辑:添加脚注,次要澄清

编辑2:重新排序的答案

于 2013-06-10T17:19:20.563 回答
2

首先,一个包含在邮件程序中的模块,可能还有其他延迟的工作:

module Delayed
  module Airbrake
    # Send error via Airbrake
    def error(job, e)
      ::Airbrake.notify(e, :component => job.name, :action => 'perform', :parameters => {:job => job.inspect})
    end
 end

结尾

然后包括它:

Delayed::PerformableMailer.send(:include, Delayed::Airbrake)
于 2012-10-06T16:43:33.723 回答