使用 AirBrake 或 ExceptionNotifier 之类的工具从邮寄延迟的作业中获取错误报告的正确方法是什么?
我试图创建自己的延迟作业类,但由Mailer.welcome()
(或类似)创建的邮件对象未正确序列化。我还尝试向and类添加一个error(job, exception)
方法,但我相信我得到了更多通常与序列化相关的错误。我尝试了 psych 和 sych 来进行连载。PerformableMailer
PerformableMethod
使用 AirBrake 或 ExceptionNotifier 之类的工具从邮寄延迟的作业中获取错误报告的正确方法是什么?
我试图创建自己的延迟作业类,但由Mailer.welcome()
(或类似)创建的邮件对象未正确序列化。我还尝试向and类添加一个error(job, exception)
方法,但我相信我得到了更多通常与序列化相关的错误。我尝试了 psych 和 sych 来进行连载。PerformableMailer
PerformableMethod
总的来说,解决方案非常简单。如果您正在对对象(如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 查找目标对象上的所有钩子(error
、before
、after
等)。但在我们的例子中,目标对象就是类本身。因此,钩子必须定义为类方法。
处理 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:重新排序的答案
首先,一个包含在邮件程序中的模块,可能还有其他延迟的工作:
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)