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
为了扩展 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
您可以编写一个自定义验证器,例如
# 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
早期的答案确实很好,虽然它们是一个很好的起点,但已经过去几年了!
这是最新的选项!
# 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 可以是您需要的任何内容!