我有两个模型类如下:
class FileInfo < ActiveRecord::Base
STATUS = {:UNAPPROVED => 1, :APPROVED => 2, :PROCESSED => 3 }
attr_accessible :id, :status
validates :status, :inclusion => {:in => STATUS.values}
end
FileInfo
有id和status字段。每个文件都可以有多个FileEntry
类型的文件条目(行)。 FileEntry
具有id,file_info_id(FileInfo
的外键)和状态字段。
class FileEntry < ActiveRecord::Base
STATUS = {:UNAPPROVED => 1, :READY => 2 , :SENT => 3 }
attr_accessible :id, :file_info_id, :status, :reason
validates :status, :inclusion => {:in => STATUS.values}
end
我想写一个worker来异步处理状态字段为APPROVED
(FileInfo
model)的所有文件。每个线程都应处理状态为READY
的特定文件的所有文件条目。一旦为该文件处理了所有条目,该线程就应该完成。假设状态为UNAPPROVED
的文件条目也将变为READY
。
处理完每个条目后,其状态应更新为SENT
。所有文件条目都具有特定文件的SENT
状态后,请将该文件的状态更新为PROCESSED
。
到目前为止,我的代码很多。无法弄清楚如何编写工作人员的代码:
class FileInfoObserver < ActiveRecord::Observer
def after_save(file_info)
if file_info.status.eql? 2
FileProcessingJob.perform_async(file_info.id)
end
end
end
工人如下:
class FileProcessingJob
include Sidekiq::Worker
def perform(file_id)
puts "job"
flag =1
while flag==1
count = 0
FileEntry.where("file_info_id = #{file_id}").find_each do |file_entry|
if(file_entry.status == 2)
puts "update" //Some PUT request, instead wrote a puts statement
FileEntry.update(file_entry.id, :status => 3)
elsif(file_entry.status == 0 || file_entry.status ==1)
count = count + 1
end
end
if(count == 0)
flag = 0
end
end
end
end
这是正确的做法吗?如何启用重试机制?这段代码是否是线程安全的?
答案 0 :(得分:1)
对于重试机制,
只需在“include Sidekiq::Worker
”
sidekiq_options queue: :default, retry: 1
似乎它不是一个多线程的Web服务器环境。所以不要担心这里的线程安全。
多线程Web服务器环境的示例:
How get best performance rails requests parallel sidekiq worker