Rails仅在有条件时才验证唯一性

时间:2014-01-03 23:54:59

标签: ruby-on-rails validates-uniqueness-of

我有一个问题类:

class Question < ActiveRecord::Base
  attr_accessible :user_id, :created_on

  validates_uniqueness_of :created_on, :scope => :user_id
end

给定用户每天只能创建一个问题,因此我想通过唯一索引强制数据库中的唯一性,并通过validates_uniqueness_of强制使用Question类。

我遇到的麻烦是我只想要非管理员用户的约束。因此,管理员可以根据需要每天创建尽可能多的问题。关于如何优雅地实现这一目标的任何想法?

3 个答案:

答案 0 :(得分:75)

您可以通过将要执行的简单Ruby字符串,Proc或方法名称作为符号作为值传递给选项中的:if:unless来进行验证条件。为了验证。以下是一些例子:

在Rails版本5.2之前,你可以传递一个字符串:

# using a string:
validates :name, uniqueness: true, if: 'name.present?'

从5.2开始,不再支持字符串,只留下以下选项:

# using a Proc:
validates :email, presence: true, if: Proc.new { |user| user.approved? }

# using a Lambda (a type of proc ... and a good replacement for deprecated strings):
validates :email, presence: true, if: -> { name.present? }

# using a symbol to call a method:
validates :address, presence: true, if: :some_complex_condition

def some_complex_condition
  true # do your checking and return true or false
end

在你的情况下,你可以这样做:

class Question < ActiveRecord::Base
  attr_accessible :user_id, :created_on

  validates_uniqueness_of :created_on, :scope => :user_id, unless: Proc.new { |question| question.user.is_admin? }
end

有关详细信息,请查看rails指南中的条件验证部分:http://edgeguides.rubyonrails.org/active_record_validations.html#conditional-validation

答案 1 :(得分:4)

我所知道的保证唯一性的唯一方法是通过数据库(例如唯一索引)。所有基于Rails的方法都涉及竞争条件。鉴于您的约束,我认为最简单的方法是建立一个单独的,唯一索引的列,其中包含您为管理员留下null的日期和用户ID的组合。

对于validates_uniqueness_of,您可以使用ifunless选项将验证限制为非管理员,如http://apidock.com/rails/ActiveRecord/Validations/ClassMethods/validates_uniqueness_of

中所述

答案 2 :(得分:1)

只需在validates_uniqueness_of来电添加条件。

validates_uniqueness_of :created_on, scope: :user_id, unless: :has_posted?
def has_posted
  exists.where(user_id: user_id).where("created_at >= ?", Time.zone.now.beginning_of_day)
end

但更好的是,只需创建一个自定义验证:

validate :has_not_posted
def has_not_posted
  posted = exists.where(user: user).where("DATE(created_at) = DATE(?)", Time.now)
  errors.add(:base, "Error message") if posted
end