如何将自定义 delayed_job 作业与 ActiveJob 一起使用?
How do I use custom delayed_job jobs with ActiveJob?
我正在使用 DelayedJob,我想更新我的 Rails 4.2 应用程序以使用 ActiveJob。问题是我有一堆看起来像这样的自定义作业:
AssetDeleteJob = Struct.new(:user_id, :params) do
def perform
# code
end
# more methods n' stuff
end
然后在某处的控制器中,作业使用以下语法排队:
@asset_delete_job = AssetDeleteJob.new(current_admin_user.id, params)
Delayed::Job.enqueue @asset_delete_job
我想找到 ActiveJob 的等效项。以上内容基本上直接来自 DJ 文档。使用它来排队单个调用就像调用作业的执行方法一样简单,就像使用 DJ 一样。但我的更复杂,需要 DJ 的结构语法和传递给它的参数。
这是我到目前为止尝试过的方法:
class AssetDeleteJob < ActiveJob::Base
queue_as :default
def initialize(user_id, params)
@user_id = user_id
@params = params
end
def perform
#code
end
# more methods n' stuff
end
job = AssetDeleteJob.new(1, {ids: [1,2,3]})
不幸的是,实例化对象没有我期望的#perform_later 方法。它确实有#enqueue,但我得到一个奇怪的错误:
Could not log "enqueue.active_job" event. NoMethodError: undefined method `any?' for nil:NilClass
...后跟数组中的堆栈跟踪,以
结尾
NoMethodError: undefined method `map' for nil:NilClass
一些奇怪的错误,但我可能不应该直接访问#enqueue。就 ActiveJob 正在寻找的内容而言,上面的内容看起来很漂亮。我错过了什么?
# in jobs/asset_delete_job.rb
class AssetDeleteJob < ActiveJob::Base
queue_as :default
def perform(user_id, params)
AssetService.new(user_id, params).delete # for example
end
end
# somewhere in your controllers/assets_controller.rb
AssetDeleteJob.perform_later(1, {ids: [1,2,3]})
我可以看到两个问题。首先,您在自定义作业 class 中定义了 initialize
方法,从而覆盖了 ActiveJob
class 中的 initialize
方法。这就是导致抛出异常的原因。我不确定您是否可以 fiddle 关于活动作业的实例化。其次,您试图在 class 的实例上调用 class 方法 perform_later
,ruby 不允许。
修复起来很简单,只需将参数添加到自定义 class 中的 perform
方法即可:
class AssetDeleteJob < ActiveJob::Base
#code
def perform(user_id, params)
#do a thing
end
#more code
end
然后通过调用类似的方式将作业排队:
AssetDeleteJob.perform_later(1, {ids: [1,2,3]})
我正在使用 DelayedJob,我想更新我的 Rails 4.2 应用程序以使用 ActiveJob。问题是我有一堆看起来像这样的自定义作业:
AssetDeleteJob = Struct.new(:user_id, :params) do
def perform
# code
end
# more methods n' stuff
end
然后在某处的控制器中,作业使用以下语法排队:
@asset_delete_job = AssetDeleteJob.new(current_admin_user.id, params)
Delayed::Job.enqueue @asset_delete_job
我想找到 ActiveJob 的等效项。以上内容基本上直接来自 DJ 文档。使用它来排队单个调用就像调用作业的执行方法一样简单,就像使用 DJ 一样。但我的更复杂,需要 DJ 的结构语法和传递给它的参数。
这是我到目前为止尝试过的方法:
class AssetDeleteJob < ActiveJob::Base
queue_as :default
def initialize(user_id, params)
@user_id = user_id
@params = params
end
def perform
#code
end
# more methods n' stuff
end
job = AssetDeleteJob.new(1, {ids: [1,2,3]})
不幸的是,实例化对象没有我期望的#perform_later 方法。它确实有#enqueue,但我得到一个奇怪的错误:
Could not log "enqueue.active_job" event. NoMethodError: undefined method `any?' for nil:NilClass
...后跟数组中的堆栈跟踪,以
结尾NoMethodError: undefined method `map' for nil:NilClass
一些奇怪的错误,但我可能不应该直接访问#enqueue。就 ActiveJob 正在寻找的内容而言,上面的内容看起来很漂亮。我错过了什么?
# in jobs/asset_delete_job.rb
class AssetDeleteJob < ActiveJob::Base
queue_as :default
def perform(user_id, params)
AssetService.new(user_id, params).delete # for example
end
end
# somewhere in your controllers/assets_controller.rb
AssetDeleteJob.perform_later(1, {ids: [1,2,3]})
我可以看到两个问题。首先,您在自定义作业 class 中定义了 initialize
方法,从而覆盖了 ActiveJob
class 中的 initialize
方法。这就是导致抛出异常的原因。我不确定您是否可以 fiddle 关于活动作业的实例化。其次,您试图在 class 的实例上调用 class 方法 perform_later
,ruby 不允许。
修复起来很简单,只需将参数添加到自定义 class 中的 perform
方法即可:
class AssetDeleteJob < ActiveJob::Base
#code
def perform(user_id, params)
#do a thing
end
#more code
end
然后通过调用类似的方式将作业排队:
AssetDeleteJob.perform_later(1, {ids: [1,2,3]})