写作工人使用sidekiq为activerecord类

时间:2014-06-25 11:22:32

标签: ruby-on-rails activerecord sidekiq

我有两个模型类如下:

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来异步处理状态字段为APPROVEDFileInfo 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

这是正确的做法吗?如何启用重试机制?这段代码是否是线程安全的?

1 个答案:

答案 0 :(得分:1)

对于重试机制,

只需在“include Sidekiq::Worker

之后添加此行
sidekiq_options queue: :default, retry: 1

似乎它不是一个多线程的Web服务器环境。所以不要担心这里的线程安全。

多线程Web服务器环境的示例:
How get best performance rails requests parallel sidekiq worker