我正在研究一种解决方案,以显示延迟作业的完成百分比(使用delayed_job gem)。目前,我有一个数据库迁移,如下所示:delayed_jobs表:
class CreateDelayedJobs < ActiveRecord::Migration
def self.up
create_table :delayed_jobs, :force => true do |table|
table.integer :priority, :default => 0 # Allows some jobs to jump to the front of the queue
table.integer :attempts, :default => 0 # Provides for retries, but still fail eventually.
table.text :handler # YAML-encoded string of the object that will do work
table.text :last_error # reason for last failure (See Note below)
table.datetime :run_at # When to run. Could be Time.zone.now for immediately, or sometime in the future.
table.datetime :locked_at # Set when a client is working on this object
table.datetime :failed_at # Set when all retries have failed (actually, by default, the record is deleted instead)
table.string :locked_by # Who is working on this object (if locked)
table.string :queue # The name of the queue this job is in
table.integer :progress
table.timestamps
end
add_index :delayed_jobs, [:priority, :run_at], :name => 'delayed_jobs_priority'
end
def self.down
drop_table :delayed_jobs
end
end
我在一个控制器方法中使用一个enqueue进程来处理延迟的作业,并在lib / build_detail.rb中引用一个类:
Delayed::Job.enqueue(BuildDetail.new(@object, @com))
lib / build_detail.rb文件如下:
class BuildDetail < Struct.new(:object, :com)
def perform
total_count = object.person_ids.length
progress_count = 0
people = com.person object.person_ids do |abc|
progress_count += abc.size
Delayed::Job.current.update_attribute :progress, (progress_count/total_count)
end
end
end
延迟:: Job.current不起作用。我在this posting上看到了提出的Delayed :: Job.current方法,但是看起来这个方法从未包含在主delayed_jobs github项目中。
如何在每次作业完成循环时访问当前作业(从实际作业中),更新进度字段?
答案 0 :(得分:8)
要回答迟到但我已经面临同样的要求,所以可能会对某人有所帮助。
您需要做的就是实现自定义作业和before
- 钩子,您将在当前作业中存储参考:
class MyTestJob
def before(job)
@job = job
end
def perform
...
@job.update_attributes({ progress: your_progress_var }, without_protection: true)
end
end
答案 1 :(得分:0)
延迟工作的一个很酷的事情也是在这里挫败你的。排队大量延迟作业时,如果有多个运行工作程序的实例,则可以并行处理它们 - 例如您可以拥有10台机器,每台机器运行您的应用程序实例访问同一个数据库,并获得10倍的处理速度提升。因此,可能会有多个“当前”工作。一次只运行一个作业有点特殊情况。
这是一种查看所有活动作业状态的方法。如果您只运行一个实例,它将只返回一个作业,这样就可以满足您的需求:
active_jobs = Delayed::Job.where("progress > 0")
progress_of_first_job = active_jobs.first.progress if active_jobs.present?
progress_of_all_jobs = active_jobs.map{|job| job.progress}
progress_of_first_job是其中一个作业的进度(使用命令子句是安全的)。 progress_of_all_jobs是每个活动作业的进度值(可能为空)。