用多态反转has_many

时间:2012-05-04 23:06:15

标签: ruby-on-rails associations polymorphism

我有两个模型:用户和项目。这个想法是用户可以关注项目和其他用户。当然,用户和项目是多态“可跟随”类型的一部分。现在,使用用户模型,我想得到三件事:

user.followed_users
user.followed_projects
user.followers

前两个工作正常;这是我遇到麻烦的第三个问题。这是一种反向查找,其中外键成为下表中的“followable_id”列,但无论我如何建模,我都无法使查询正确运行。

用户模型

has_many :follows, :dependent => :destroy
has_many :followed_projects, :through => :follows, :source => :followable, :source_type => "Project"
has_many :followed_users, :through => :follows, :source => :followable, :source_type => "User"
has_many :followers, :through => :follows, :as => :followable, :foreign_key => "followable", :source => :user, :class_name => "User"

关注模型

class Follow < ActiveRecord::Base
  belongs_to :followable, :polymorphic => true
  belongs_to :user
end

我的关注表有:  用户身份  followable_id  followable_type

每当我运行查询时,我得到:

SELECT `users`.* FROM `users` INNER JOIN `follows` ON `users`.`id` = `follows`.`user_id` WHERE `follows`.`user_id` = 7

应该是“followable_id = 7 AND followable_type ='User”,而不是“user_id = 7”

有什么想法吗?

2 个答案:

答案 0 :(得分:5)

想出来。看看sample project Michael Hartl made,并注意到正确的方法是不仅指定一个关系表(在本例中是后面的),而且还指定一个反向关系表(我称之为反向关系)。

    has_many    :follows,
                        :dependent => :destroy

    has_many    :followed_projects,
                        :through => :follows,
                        :source => :followable,
                        :source_type => "Project"

    has_many    :followed_users,
                        :through => :follows,
                        :source => :followable,
                        :source_type => "User"

    has_many    :reverse_follows,
                        :as => :followable,
                        :foreign_key => :followable_id,
                        :class_name => "Follow"

    has_many    :followers,
                        :through => :reverse_follows,
                        :source => :user

希望这可以帮助一些人走出困境!

答案 1 :(得分:0)

我认为您需要明确拼出外键。你有:

:foreign_key => "followable"

你需要:

:foreign_key => "followable_id"

完整代码:

has_many :followers, :through => :follows, :as => :followable, :foreign_key => "followable_id", :source => :user, :class_name => "User"