I am using polymorphic associations to track Comments in my project. All very straight forward stuff.
The problem I have is in querying based on the polymorphic association and joining from the Comment model back to it's owner.
So ...
I have a Comment model
class Comment < ActiveRecord::Base
belongs_to :commentable, :polymorphic => true
end
And a ForumTopics mode:
class ForumTopic < ActiveRecord::Base
has_many :comments, :as => :commentable
end
I have several other "commentable" models that aren't important right now. All of this works.
What I am trying to do is find all of the Comments that belong to a ForumTopic with a specified condition (in this case, 'featured' == true).
When I try and use a finder to join the models:
@comments = Comment.find(:all
:joins => :commentable
:conditions => ["forum_topics.featured = ? ", true]
)
I receive the following error:
Can not eagerly load the polymorphic association :commentable
Using the AR "include syntax":
@comments = Comment.find(:all
:include => :forum_topics
:conditions => ["forum_topics.featured = ? ", true]
)
returns:
Association named 'forum_topics' was not found; perhaps you misspelled it?
If I try and join with a table name instead of the association name (string instead of symbol):
@comments = Comment.find(:all,
:joins => "forum_topics",
:conditions => ["forum_topics.featured = ? ", true]
)
I see:
Mysql::Error: Unknown table 'comments': SELECT comments. FROM comments forum_topics WHERE (forum_topics.featured = 1 )*
(You can see here that the syntax of the underlying query is totally off and the join is missing altogether).
Not sure if what I am doing is even possible, and there are other ways to achieve the required result but it seems like it should be doable.
Any ideas? Anything I am missing?
Argh!
I think I found the problem.
When joining via:
You need the whole join!
See the ever-awesome: http://guides.rubyonrails.org/active_record_querying.html#joining-tables
I came across this post and it lead me to my solution. Using the commentable_type as one of my conditions but using a LEFT OUTER JOIN instead. That way forum topics without comments will be included.
You Need a Conditional, Plus Rails 3+
A lot of people alluded to it in the answers and comments but I felt that people, including myself, would get tripped up if I landed here and didn't read thoroughly enough.
So, here's the proper answer, including the conditional that is absolutely necessary.
Thanks to all, especially @Jits, @Peter, and @prograils for their comments.
Checked to work under Rails 5:
Solution 1:
or
Solution 2:
Pay attention to the raw SQL syntax: no backticks are allowed. See http://guides.rubyonrails.org/active_record_querying.html#joining-tables .
I personally prefer Solution 1 as it contains fewer raw SQL syntax.
An old question, but there is a cleaner way of achieving this by setting up a direct association for the specific type along with the polymorphic:
You are then able to pass
:forum_topics
toincludes
getting rid of the need for a messy join:You could then further clean this up by moving the query into a scope:
Leaving you to be able to simply do
The accepted solution does not work once you introduce another model that has an association using "commentable". commentable_id is not unique and therefore you'll start retrieving the wrong comments.
For example:
You decide to add a news model that accepts comments...
Now you may get two records back if you made a comment on a forum_topic with an id of 1 and a news article with an id of 1 using your query:
You could probably solve the problem by supplying a commentable_type as one of your conditions, but I don't think that's the best way to approach this issue.