Rails 3:嵌套 fields_for 的唯一性验证

问题描述 投票:0回答:4

A 有两个模型,“shop”和“product”,通过 has_many :through 链接。

在商店表单中有多个产品的嵌套属性,我在产品的唯一性验证方面遇到了一些麻烦。如果我输入一个产品,保存它,然后尝试为新产品输入相同的名称,则唯一性验证会成功触发。

但是,如果我在同一嵌套表单的两行中输入相同的产品名称,则该表单将被接受 - 不会触发唯一性验证。

我猜这是一个相当常见的问题,但我找不到任何简单的解决方案。有人对确保在相同嵌套表单中遵守唯一性验证的最简单方法有任何建议吗?

编辑:产品型号如下

class Product < ActiveRecord::Base

  has_many :shop_products
  has_many :shops, :through => :shop_products

  validates_presence_of :name
  validates_uniqueness_of :name
end
ruby-on-rails ruby-on-rails-3 validates-uniqueness-of
4个回答
18
投票

为了扩展 Alberto 的解决方案,以下自定义验证器接受要验证的字段(属性),并向嵌套资源添加错误。

# config/initializers/nested_attributes_uniqueness_validator.rb
class NestedAttributesUniquenessValidator < ActiveModel::EachValidator
  def validate_each(record, attribute, value)
    unless value.map(&options[:field]).uniq.size == value.size
      record.errors[attribute] << "must be unique"
      duplicates = value - Hash[value.map{|obj| [obj[options[:field]], obj]}].values
      duplicates.each { |obj| obj.errors[options[:field]] << "has already been taken" }
    end
  end
end

# app/models/shop.rb
class Shop < ActiveRecord::Base
  validates :products, :nested_attributes_uniqueness => {:field => :name}
end

14
投票

您可以编写一个自定义验证器,例如

# app/validators/products_name_uniqueness_validator.rb
class ProductsNameUniquenessValidator < ActiveModel::EachValidator
  def validate_each(record, attribute, value)
    record.errors[attribute] << "Products names must be unique" unless value.map(&:name).uniq.size == value.size
  end
end

# app/models/shop.rb
class Shop < ActiveRecord::Base
  validates :products, :products_name_uniqueness => true
end


0
投票

早期的答案确实很好,虽然它们是一个很好的起点,但已经过去几年了!

这是最新的选项!

# config/initializers/nested_attributes_uniqueness_validator.rb
class NestedAttributesUniquenessValidator < ActiveModel::EachValidator
  def validate_each(record, attribute, items)
    field = options[:field]
    return if items.map(&field).uniq.size == items.size

    values = items.map { |item| item.send(field) }
    duplicate_values = values.select{ |value| values.count(value)>1 }.uniq.join(', ')
    record.errors.add(attribute, "are not unique. Duplicate #{field}s detected. Duplicate values: #{duplicate_values}" )
    duplicates = items.find_all { |item| values.count(item.send(field)) > 1 && item.id.nil? }
    duplicates.each { |obj| obj.errors.add(field, :taken) }
  end
end
# app/models/shop.rb
class Shop < ActiveRecord::Base
  validates :products, nested_attributes_uniqueness: { field: :name }
end
# app/controllers/shops_controller.rb
class ShopsController < ApplicationController
  def create
    @shop = Shop.new(shop_params)
    return success_response if @shop.save
    
    failure_response(@shop.errors.full_messages)
  end
  
  private
  
  def shop_params
    params.require(:shop).permit(:name, :address, products_attributes: %i[name price])
  end
end

success_response 和 failure_response 可以是您需要的任何内容!

© www.soinside.com 2019 - 2024. All rights reserved.