Есть ли способ загружать ассоциации полиморфных ассоциаций? - PullRequest
7 голосов
/ 28 декабря 2011

художников имеют много видов деятельности (в основном это кэш взаимодействий между пользователями):

class Activity < ActiveRecord::Base

  belongs_to :receiver, :class_name => 'Artist', :foreign_key => :receiver_id #owns the stuff done "TO" him
  belongs_to :link, :polymorphic => true
  belongs_to :creator, :class_name => 'Artist', :foreign_key => :creator_id #person who initiated the activity

end

Например:

  Activity.create(:receiver_id => author_id, :creator_id => artist_id, :link_id => id, :link_type => 'ArtRating') 

Я хочу создать страницу потока действий для каждого исполнителя,состоящий из списка различных типов событий, ArtRatings (лайки, антипатии), Favoriting, Follow и т. д.

Контроллер выглядит так:объекты полиморфной ссылки:

  SELECT "activities".* FROM "activities" WHERE (("activities"."receiver_id" = 6 OR "activities"."creator_id" = 6)) ORDER BY id DESC LIMIT 30
  ArtRating Load (0.5ms)  SELECT "art_ratings".* FROM "art_ratings" WHERE "art_ratings"."id" IN (137, 136, 133, 130, 126, 125, 114, 104, 103, 95, 85, 80, 73, 64)
  SELECT "follows".* FROM "follows" WHERE "follows"."id" IN (14, 10)
  SELECT "favorites".* FROM "favorites" WHERE "favorites"."id" IN (25, 16, 14)

Но когда я отображаю каждый ArtRating, мне также нужно ссылаться на заголовок поста, который принадлежит посту.По моему мнению, если я это сделаю:

activity.link.post

Это делает отдельный вызов БД для каждого сообщения art_rating.Есть ли способ загружать пост тоже?

ОБНОВЛЕНИЕ К ВОПРОСУ:

Если нет способа добиться быстрой загрузки постов с помощью «включает»Синтаксис, есть ли способ вручную выполнить запрос на загрузку самостоятельно и вставить его в объект @activities?

Я вижу в журнале БД:

SELECT "art_ratings".* FROM "art_ratings" WHERE "art_ratings"."id" IN (137, 136, 133, 130, 126, 125, 114, 104, 103, 95, 85, 80, 73, 64)

Есть ли способ, которым яполучить доступ к этому списку идентификаторов из объекта @activities?Если это так, я мог бы сделать 2 дополнительных запроса, 1 чтобы получить art_ratings.post_id (s) в этом списке, и еще один, чтобы ВЫБРАТЬ все сообщения в этом списке post_ids.Затем каким-то образом вставьте результаты 'post' обратно в @activities, чтобы они были доступны как activity.link.post, когда я выполняю итерацию по коллекции.Возможно ли это?

Ответы [ 5 ]

4 голосов
/ 31 октября 2013

TL; DR мое решение заставляет artist.created_activities.includes(:link) стремиться загрузить все, что вы хотите

Вот моя первая попытка сделать это: https://github.com/JohnAmican/music

Несколько заметок:

  • Я полагаюсь на default_scope, так что это не оптимально.
  • Похоже, вы используете STI,Мое решение не.Это означает, что вы не можете просто позвонить activities на исполнителя;Вы должны ссылаться на created_activities или received_activities.Там может быть способ обойти это.Я обновлюсь, если найду что-нибудь.
  • Я изменил некоторые имена, потому что иначе это меня смущало.

Если вы зайдете в консоль и выполните created_activities.includes(:link)соответствующие вещи загружаются с нетерпением:

irb(main):018:0> artist.created_activities.includes(:link)
  Activity Load (0.2ms)  SELECT "activities".* FROM "activities" WHERE "activities"."creator_id" = ?  [["creator_id", 1]]
  Rating Load (0.3ms)  SELECT "ratings".* FROM "ratings" WHERE "ratings"."id" IN (1)
  RatingExplanation Load (0.3ms)  SELECT "rating_explanations".* FROM "rating_explanations" WHERE "rating_explanations"."rating_id" IN (1)
  Following Load (0.3ms)  SELECT "followings".* FROM "followings" WHERE "followings"."id" IN (1)
  Favorite Load (0.2ms)  SELECT "favorites".* FROM "favorites" WHERE "favorites"."id" IN (1)
=> #<ActiveRecord::Relation [#<Activity id: 1, receiver_id: 2, creator_id: 1, link_id: 1, link_type: "Rating", created_at: "2013-10-31 02:36:27", updated_at: "2013-10-31 02:36:27">, #<Activity id: 2, receiver_id: 2, creator_id: 1, link_id: 1, link_type: "Following", created_at: "2013-10-31 02:36:41", updated_at: "2013-10-31 02:36:41">, #<Activity id: 3, receiver_id: 2, creator_id: 1, link_id: 1, link_type: "Favorite", created_at: "2013-10-31 02:37:04", updated_at: "2013-10-31 02:37:04">]>

По крайней мере, это доказывает, что Rails имеет возможность сделать это.Обход default_scope кажется проблемой с сообщением Rails, что вы хотите сделать, а не техническим ограничением.

ОБНОВЛЕНИЕ:

Оказывается, когда выпередать блок области видимости ассоциации и вызвать эту ассоциацию, self внутри этого блока относится к отношению.Итак, вы можете подумать об этом и действовать соответствующим образом:

class Activity < ActiveRecord::Base
  belongs_to :creator,  class_name: 'Artist', foreign_key: :creator_id,  inverse_of: :created_activities
  belongs_to :receiver, class_name: 'Artist', foreign_key: :receiver_id, inverse_of: :received_activities
  belongs_to :link, -> { self.reflections[:activity].active_record == Rating ? includes(:rating_explanation) : scoped }, polymorphic: true
end

Я обновил свой код, чтобы отразить (хаха) это.

Это можно исправить.Например, может быть, вы не всегда хотите загружать rating_explanations при доступе к ссылкам активности.Есть несколько способов решить это.Я мог бы опубликовать один, если хотите.

Но, я думаю, самое важное, что это показывает, это то, что в пределах блока области ассоциации, у вас есть доступ к строящемуся ActiveRecord::Relation.Это позволит вам сделать что-то условно.

0 голосов
/ 01 ноября 2013

Это простое ActiveRecord должно работать:

@artist.activities.includes([:link => :post,:creator,:receiver]).order("id DESC").limit(30)

Если нет, если вы получаете сообщение об ошибке типа "Association named 'post' was not found; perhaps you misspelled it?", то у вас проблема с моделью, потому что хотя бы один из ваших linkассоциации не имеют ассоциации post.

Полиморфные ассоциации предназначены для использования с общим интерфейсом.Если вы попросите post для какой-либо полиморфной ассоциации, то ВСЕ ваши ассоциации должны также реализовать эту ассоциацию (post).

Убедитесь, что все ваши модели, используемые в полиморфной ассоциации, реализуют пост-ассоциацию.

0 голосов
/ 25 октября 2013

Если я правильно понял, вы хотите загрузить не только ассоциацию полиморфной ассоциации , но полиморфную ассоциацию полиморфную ассоциацию .

Это в основном означает объединение одной определенной таблицы с другой определенной таблицей через набор неопределенных таблиц, которые поступают из некоторых полей в базе данных.

Поскольку и действие, и публикация объединяются через полиморфный объект, они оба имеютsomething_id и something_type столбец.

Теперь я думаю, что активная запись не позволяет вам делать это из коробки, но в основном вам нужно что-то вроде:

class Activity
  has_one :post, :primary_key => [:link_id, :link_type], :foreign_key => [:postable_id, :postable_type]
end

(при условии, что вашполиморфная ассоциация в Post равна belongs_to :postable)

Это даст вам запрос типа прямой связи между Post и Activity, что очень странно для habtm с 'таблица полиморфных соединений »(я только что придумал этот термин).Поскольку оба разделяют один и тот же полиморфно связанный объект, они могут быть связаны.Вроде как друзья моих друзей.

ПЕРЕДАЧА : Как я уже сказал, насколько я знаю, AR не позволяет вам делать это из коробки (хотя это было бы круто) но есть некоторые драгоценные камни, которые дают вам составные внешние и / или первичные ключи.Может быть, вы найдете такой, который поможет вам решить вашу проблему таким образом.

0 голосов
/ 27 октября 2013

Учитывая ваше обновление вопроса (что это невозможно с использованием AR include), вы можете попробовать следующее, чтобы получить соответствующие сообщения с одним дополнительным запросом:

  • извлеките Действия, затем создайте Массив их связанных post_ids.

    @activities = @artist.activities.includes([:link,:creator,:receiver])
    activity_post_ids = @activities.map{|a| a.link.post_id}.compact
    
  • затем загрузите сообщения в одном запросе и сохраните их в Hash, проиндексированном по их id

    activity_posts = Hash[Post.where(id: activity_post_ids).map{|p| [p.id, p]}]
    #=> {1 => #<Post id:1>, 3 => #<Post id:3>, ... }
    
  • наконец, переберите @activities и установите атрибут post для каждой связанной ссылки

    @activities.each{|a| a.link.post = activity_posts[a.link.post_id]}
    
0 голосов
/ 28 декабря 2011

Попробуйте это:

@artist.activities.includes([{:link => :post},:creator,:receiver])...

Подробнее смотрите в документации по Rails.

...