Rails 3 validates_uniqueness_of 两个模型属性和 created_at 年

Rails 3 validates_uniqueness_of for two model attributes AND created_at year

我有一个投影模型可以验证属性的唯一性 "ppr"。我不想要 :week、:player_id、:ppr 和 :created_at 年份都相同的多个投影记录。 这里的关键问题是我不知道如何针对 created_at 年 进行此验证。我不介意是否有多个记录,其中 :ppr、:player_id 和 :week 都是相同的,只要它们用于不同的 created_at 年。

例如:

这是投影记录。

 #<Projection id: 44, ppr: 3.5, salary: 6600, week: 1, created_at: "2014-09-04 06:05:34", player_id: 44> 

如果我想创建这个额外的投影记录,我当前的验证将不允许它,即使 created_at 年份是 2015 年而不是 2014 年。

#<Projection id: 89, ppr: 3.5, salary: 6600, week: 1, created_at: "2015-09-05 06:05:34", player_id: 44> 

我如何修改我的验证以考虑这个额外的限制并允许来自不同年份的记录?

我基本上需要结合这两个验证:

validates_uniqueness_of :ppr, :scope => [:week, :player_id], :if => :nfl?

validates_uniqueness_of :ppr, conditions: { -> { where("created_at >= ? and created_at <= ?", "#{Date.today.year}0101", "#{Date.today.year}1231") } }

下面发布的是我目前的模型设置方式:

class Projection < ActiveRecord::Base
  attr_accessible :ppr, :salary, :score, :week, :player_id, :created_at

  belongs_to :player

  validates_uniqueness_of :ppr, :scope => [:week, :player_id], :if => :nfl?

  scope :for_year, lambda {|year| where("created_at >= ? and created_at <= ?", "#{year}0101", "#{year}1231")}
end

更新

由于明显的原因,此语法不起作用,但我可以做类似的事情吗?

validates_uniqueness_of :ppr, :scope, :created_at => [:week, :player_id, created_at <= ?", "#{Date.today.year}0101", "#{Date.today.year}1231"], :if => :nfl?

我建议编写一个验证方法来查询是否存在这样的记录。由于每年的数据都是唯一的,查询当年的日期范围

注意:这确实假设所有数据输入都是当年的,因为您的条件与 created_at 属性相关联。如果您需要不同的时间段,请修改 range 变量。

class Projection < ActiveRecord::Base
  belongs_to :player
  delegate :nfl?, to: :player

  validates :player, presence: true
  validate :unique_player_projection, if: :nfl?

  private

  def unique_player_projection
    range = Time.now.beginning_of_year..Time.now.end_of_year
    return unless Projection.exists?(player_id: player_id, ppr: ppr, week: week, created_at: range)
    errors.add(:base, 'Projection not unique.')
  end
end

此示例与数据库无关。如果您正在处理巨大的记录集,则有更优化的 SQL 查询。

我为 nfl? 属性添加了委托。