Don’t use default_scope. Ever.
When you would like a scope to be applied across all queries on a model, you can use default_scope
. See more in the ActiveRecord Query Guide and Rails docs.
Where you have a blog system with posts that can be set to be hidden, for when you are writing a draft.
Instead of…
…default_scope
.
# app/models/post.rb
class Post < ActiveRecord::Base
default_scope { where(hidden: false) }
end
Use…
…explicit scopes.
# app/models/post.rb
class Post < ActiveRecord::Base
scope, :published -> { where(hidden: false) }
end
…then you can do…
Post.published
But why?
Two reasons. Both to do with avoiding later confusion and bug hunting.
Adding a default scope affects your model initialization. In the example, Post.new
is defaulted to hidden = false
whether you are expecting it or not.
Trying not to use your defined default scope is a pain. To remove the default_scope
when you don’t need it you have to use the unscoped
scope (!) which removes all applied conditions including associations.
i.e. Post.first.comments.unscoped
would return every comment in your database, not just those for the first Post
.
The explicit use of named scopes is a clearer solution. Using default_scope
will lead to many hours of bug hunting. Don’t do it to yourself.
Where might I use it?
Seriously. Trust me on this one. It will bite you.
Sign up to get a nugget of Ruby knowledge every couple of weeks or so.
Last updated on October 1st, 2017