验证嵌套属性的存在

4
如何使用嵌套属性验证模型至少有一个关联模型?这让我很疯狂,因为我相信我可能漏掉了一些简单的东西。例如,我想要求列表始终至少具有一个任务。
class List < ActiveRecord::Base
  has_many :tasks, :dependent => :destroy
  accepts_nested_attributes_for :tasks, :allow_destroy => true
end

class Task < ActiveRecord::Base
  belongs_to :list
end

我尝试了许多不同的选项。 1- 添加列表验证:
def validate
  if self.tasks.length < 1
    self.errors[:base] << "A list must have at least one task."
  end
end

但是这仍然允许您删除一个已存在列表的所有任务,因为在删除任务时,列表的验证发生在任务被销毁之前。
2- 在before_save回调中检查是否有任何任务未标记为销毁。
before_save :check_tasks

private
#look for any task which won't be deleted
def check_tasks
  for t in self.tasks
    return true if ! t.marked_for_destruction?
  end
  false  
end

由于某种原因,我无法通过对任务列表进行迭代的任何方式来删除任务。如果我在 def validate 中执行此检查而不是回调,则结果相同。
3- 需要存在任务 validates_presence_of :tasks,但使用此选项将永远不会删除任何任务。
2个回答

8
您可以在验证方法中同时检查两个条件:
  validate :check_tasks
  def check_tasks
    if self.tasks.size < 1 || self.tasks.all?{|task| task.marked_for_destruction? }
      errors.add_to_base("A list must have at least one task.")
    end
  end

我喜欢你的风格,而且我同意这应该可以工作(我在我的测试中尝试了类似的东西),但是每当你从模型中的列表对象迭代任务时,没有任何任务会被删除。我已经尝试过2.3.8和3.0.0beta4,但都无济于事。 - Ryan Horrisberger
很奇怪,我在Rails 2.3.8上测试过了,它可以正常工作。有一件事-记录已从数据库中删除,但未从'list.tasks'缓存集合中移除。因此,这可能会导致任务似乎没有被删除。如果您需要在更新后访问对象,则可以尝试list.tasks.reload - Tatjana N.

3
我最终扩展了Magazine的保存方法来解决这个问题。它完美地解决了问题。
def save
  saved = false
  ActiveRecord::Base.transaction do
    saved = super
    if self.conditions.size < 1
      saved = false
      errors[:base] << "A rule must have at least one condition."
      raise ActiveRecord::Rollback
    end
  end
  saved
end

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接