Подтвердить что ты не робот

Подтвердить уникальность области в моделях полиморфных ассоциаций

Правильно, поэтому у меня есть полиморфная ассоциация, которая позволяет использовать разные типы объектов. Таким образом, человек может любить продукт, или человек, или что-то еще. То, что я хочу сделать, - это оградить кого-то, дублирующего любимого, с помощью проверки подлинности уникальности в избранной модели.

class Favorite < ActiveRecord::Base
 belongs_to :favoritable, :polymorphic => true
 belongs_to :user

 attr_accessible :user

 validates_presence_of :user
 validates :user_id, :uniqueness => { :scope => [:favoritable_type, :favoritable_id] }
end

Валидация, похоже, работает, но по какой-либо причине новая избранная строка все еще создается с помощью user_id, когда делается попытка дублировать запись.

enter image description here

Есть ли способ остановить это первоначальное сохранение?

Кажется, что Rails создает запись БД, а затем обновляет ее с помощью favoritable_id и favoritable_type следующим образом:

  SQL (28.3ms)  INSERT INTO "favorites" ("created_at", "favoritable_id", "favoritable_type", "updated_at", "user_id") VALUES ($1, $2, $3, $4, $5) RETURNING "id"  [["created_at", Tue, 14 Aug 2012 10:26:31 UTC +00:00], ["favoritable_id", nil], ["favoritable_type", nil], ["updated_at", Tue, 14 Aug 2012 10:26:31 UTC +00:00], ["user_id", 23]]
   (7.8ms)  COMMIT
   (0.1ms)  BEGIN
  Favorite Exists (0.3ms)  SELECT 1 AS one FROM "favorites" WHERE ("favorites"."user_id" = 23 AND "favorites"."id" != 123 AND "favorites"."favoritable_type" = 'Style' AND "favorites"."favoritable_id" = 29) LIMIT 1
   (0.2ms)  UPDATE "favorites" SET "favoritable_id" = 29, "favoritable_type" = 'Style', "updated_at" = '2012-08-14 10:26:31.943937' WHERE "favorites"."id" = 123
   (6.7ms)  COMMIT
   (0.1ms)  BEGIN
4b9b3361

Ответ 1

Если вы внимательно наблюдаете, вы можете обнаружить, что проверка единственности работает нормально:)

validates :user_id, :uniqueness => { :scope => [:favoritable_type, :favoritable_id] }

Посмотрите на добавленный образ данных. внутри изображения вы можете узнать, что вторая запись не имеет favouritable, тогда как первая имеет то, что отличается от этого, поэтому 2 записи являются uniq и ее не вызывают с uniqueness, но это ваш логический разрыв.

Если вы строго хотели избежать второй записи, сохраните favouritable в качестве обязательного поля

validates :favoritable_type, :favoritable_id, :presence => true

Ответ 2

class Favorite < ActiveRecord::Base

  belongs_to :user
  belongs_to :favoritable, polymorphic: true

  validates :user_id, :favoritable_id, presence: true,
    numericality: { only_integer: true }

  validates :favoritable_type, presence: true,
    inclusion: { 
      in: %w(FirstModelName SecondModelName),
      message: "%{value} is not a valid" 
    }

  validates :user_id, uniqueness: { scope: [ :favoritable_type, :favoritable_id ] }
end