2004-11-23 20:04:44 -05:00
module ActiveRecord
2009-05-01 11:01:13 -04:00
class InverseOfAssociationNotFoundError < ActiveRecordError #:nodoc:
def initialize ( reflection )
super ( " Could not find the inverse association for #{ reflection . name } ( #{ reflection . options [ :inverse_of ] . inspect } in #{ reflection . class_name } ) " )
end
end
2006-03-27 22:06:40 -05:00
class HasManyThroughAssociationNotFoundError < ActiveRecordError #:nodoc:
2006-07-19 20:34:09 -04:00
def initialize ( owner_class_name , reflection )
super ( " Could not find the association #{ reflection . options [ :through ] . inspect } in model #{ owner_class_name } " )
2006-03-18 19:53:24 -05:00
end
end
2006-03-27 22:06:40 -05:00
class HasManyThroughAssociationPolymorphicError < ActiveRecordError #:nodoc:
2006-03-18 19:53:24 -05:00
def initialize ( owner_class_name , reflection , source_reflection )
2006-07-19 20:34:09 -04:00
super ( " Cannot have a has_many :through association ' #{ owner_class_name } # #{ reflection . name } ' on the polymorphic object ' #{ source_reflection . class_name } # #{ source_reflection . name } '. " )
2006-03-18 19:53:24 -05:00
end
end
2007-12-22 06:26:03 -05:00
2007-03-13 01:23:18 -04:00
class HasManyThroughAssociationPointlessSourceTypeError < ActiveRecordError #:nodoc:
def initialize ( owner_class_name , reflection , source_reflection )
super ( " Cannot have a has_many :through association ' #{ owner_class_name } # #{ reflection . name } ' with a :source_type option if the ' #{ reflection . through_reflection . class_name } # #{ source_reflection . name } ' is not polymorphic. Try removing :source_type on your association. " )
end
end
2007-12-22 06:26:03 -05:00
2006-03-27 22:06:40 -05:00
class HasManyThroughSourceAssociationNotFoundError < ActiveRecordError #:nodoc:
2006-03-24 09:46:17 -05:00
def initialize ( reflection )
2006-07-19 20:34:09 -04:00
through_reflection = reflection . through_reflection
source_reflection_names = reflection . source_reflection_names
source_associations = reflection . through_reflection . klass . reflect_on_all_associations . collect { | a | a . name . inspect }
2009-02-27 08:22:39 -05:00
super ( " Could not find the source association(s) #{ source_reflection_names . collect ( & :inspect ) . to_sentence ( :two_words_connector = > ' or ' , :last_word_connector = > ', or ' , :locale = > :en ) } in model #{ through_reflection . klass } . Try 'has_many #{ reflection . name . inspect } , :through => #{ through_reflection . name . inspect } , :source => <name>'. Is it one of #{ source_associations . to_sentence ( :two_words_connector = > ' or ' , :last_word_connector = > ', or ' , :locale = > :en ) } ? " )
2006-03-18 19:53:24 -05:00
end
end
2007-01-26 16:37:38 -05:00
class HasManyThroughSourceAssociationMacroError < ActiveRecordError #:nodoc:
2006-04-05 11:36:02 -04:00
def initialize ( reflection )
2006-07-19 20:34:09 -04:00
through_reflection = reflection . through_reflection
source_reflection = reflection . source_reflection
super ( " Invalid source reflection macro : #{ source_reflection . macro } #{ " :through " if source_reflection . options [ :through ] } for has_many #{ reflection . name . inspect } , :through => #{ through_reflection . name . inspect } . Use :source to specify the source reflection. " )
2006-04-05 11:36:02 -04:00
end
end
2008-04-05 20:27:12 -04:00
class HasManyThroughCantAssociateThroughHasManyReflection < ActiveRecordError #:nodoc:
def initialize ( owner , reflection )
super ( " Cannot modify association ' #{ owner . class . name } # #{ reflection . name } ' because the source reflection class ' #{ reflection . source_reflection . class_name } ' is associated to ' #{ reflection . through_reflection . class_name } ' via : #{ reflection . source_reflection . macro } . " )
end
end
2006-08-18 03:35:07 -04:00
class HasManyThroughCantAssociateNewRecords < ActiveRecordError #:nodoc:
def initialize ( owner , reflection )
super ( " Cannot associate new records through ' #{ owner . class . name } # #{ reflection . name } ' on ' #{ reflection . source_reflection . class_name rescue nil } # #{ reflection . source_reflection . name rescue nil } '. Both records must have an id in order to create the has_many :through record associating them. " )
end
end
2007-10-27 14:51:32 -04:00
class HasManyThroughCantDissociateNewRecords < ActiveRecordError #:nodoc:
def initialize ( owner , reflection )
super ( " Cannot dissociate new records through ' #{ owner . class . name } # #{ reflection . name } ' on ' #{ reflection . source_reflection . class_name rescue nil } # #{ reflection . source_reflection . name rescue nil } '. Both records must have an id in order to delete the has_many :through record associating them. " )
end
end
2009-03-06 13:26:34 -05:00
class HasAndBelongsToManyAssociationForeignKeyNeeded < ActiveRecordError #:nodoc:
def initialize ( reflection )
super ( " Cannot create self referential has_and_belongs_to_many association on ' #{ reflection . class_name rescue nil } # #{ reflection . name rescue nil } '. :association_foreign_key cannot be the same as the :foreign_key. " )
end
end
2006-03-27 22:06:40 -05:00
class EagerLoadPolymorphicError < ActiveRecordError #:nodoc:
2006-03-18 19:53:24 -05:00
def initialize ( reflection )
2006-07-19 20:34:09 -04:00
super ( " Can not eagerly load the polymorphic association #{ reflection . name . inspect } " )
2006-03-18 19:53:24 -05:00
end
end
2006-04-25 01:49:14 -04:00
class ReadOnlyAssociation < ActiveRecordError #:nodoc:
def initialize ( reflection )
2006-07-19 20:34:09 -04:00
super ( " Can not add to a has_many :through association. Try adding to #{ reflection . through_reflection . name . inspect } . " )
2006-04-25 01:49:14 -04:00
end
end
2008-07-28 07:26:59 -04:00
# See ActiveRecord::Associations::ClassMethods for documentation.
2004-11-23 20:04:44 -05:00
module Associations # :nodoc:
2009-03-07 07:34:56 -05:00
# These classes will be loaded when associations are created.
2008-11-24 12:14:24 -05:00
# So there is no need to eager load them.
autoload :AssociationCollection , 'active_record/associations/association_collection'
autoload :AssociationProxy , 'active_record/associations/association_proxy'
autoload :BelongsToAssociation , 'active_record/associations/belongs_to_association'
autoload :BelongsToPolymorphicAssociation , 'active_record/associations/belongs_to_polymorphic_association'
autoload :HasAndBelongsToManyAssociation , 'active_record/associations/has_and_belongs_to_many_association'
autoload :HasManyAssociation , 'active_record/associations/has_many_association'
autoload :HasManyThroughAssociation , 'active_record/associations/has_many_through_association'
autoload :HasOneAssociation , 'active_record/associations/has_one_association'
autoload :HasOneThroughAssociation , 'active_record/associations/has_one_through_association'
2006-04-29 14:10:14 -04:00
def self . included ( base )
2005-04-02 04:29:43 -05:00
base . extend ( ClassMethods )
end
2007-12-22 06:26:03 -05:00
# Clears out the association cache
2004-12-22 08:54:44 -05:00
def clear_association_cache #:nodoc:
self . class . reflect_on_all_associations . to_a . each do | assoc |
instance_variable_set " @ #{ assoc . name } " , nil
2006-09-05 14:54:24 -04:00
end unless self . new_record?
2004-12-22 08:54:44 -05:00
end
2007-12-22 06:26:03 -05:00
2009-01-31 20:44:30 -05:00
private
# Gets the specified association instance if it responds to :loaded?, nil otherwise.
def association_instance_get ( name )
association = instance_variable_get ( " @ #{ name } " )
association if association . respond_to? ( :loaded? )
end
# Set the specified association instance.
def association_instance_set ( name , association )
instance_variable_set ( " @ #{ name } " , association )
end
2007-12-22 06:26:03 -05:00
# Associations are a set of macro-like class methods for tying objects together through foreign keys. They express relationships like
# "Project has one Project Manager" or "Project belongs to a Portfolio". Each macro adds a number of methods to the class which are
# specialized according to the collection or association symbol and the options hash. It works much the same way as Ruby's own <tt>attr*</tt>
2004-11-23 20:04:44 -05:00
# methods. Example:
#
# class Project < ActiveRecord::Base
# belongs_to :portfolio
2007-12-22 06:26:03 -05:00
# has_one :project_manager
2004-11-23 20:04:44 -05:00
# has_many :milestones
# has_and_belongs_to_many :categories
# end
#
# The project class now has the following methods (and more) to ease the traversal and manipulation of its relationships:
2005-01-15 12:45:16 -05:00
# * <tt>Project#portfolio, Project#portfolio=(portfolio), Project#portfolio.nil?</tt>
2004-11-23 20:04:44 -05:00
# * <tt>Project#project_manager, Project#project_manager=(project_manager), Project#project_manager.nil?,</tt>
# * <tt>Project#milestones.empty?, Project#milestones.size, Project#milestones, Project#milestones<<(milestone),</tt>
2006-09-01 19:37:10 -04:00
# <tt>Project#milestones.delete(milestone), Project#milestones.find(milestone_id), Project#milestones.find(:all, options),</tt>
2004-11-23 20:04:44 -05:00
# <tt>Project#milestones.build, Project#milestones.create</tt>
# * <tt>Project#categories.empty?, Project#categories.size, Project#categories, Project#categories<<(category1),</tt>
# <tt>Project#categories.delete(category1)</tt>
#
2007-05-18 15:18:01 -04:00
# === A word of warning
#
# Don't create associations that have the same name as instance methods of ActiveRecord::Base. Since the association
# adds a method with that name to its model, it will override the inherited method and break things.
2008-05-09 05:38:02 -04:00
# For instance, +attributes+ and +connection+ would be bad choices for association names.
2007-05-18 15:18:01 -04:00
#
2007-05-30 03:05:14 -04:00
# == Auto-generated methods
2004-11-23 20:04:44 -05:00
#
2008-03-21 14:21:56 -04:00
# === Singular associations (one-to-one)
2007-05-30 03:05:14 -04:00
# | | belongs_to |
# generated methods | belongs_to | :polymorphic | has_one
# ----------------------------------+------------+--------------+---------
2009-02-24 07:29:25 -05:00
# other | X | X | X
# other=(other) | X | X | X
# build_other(attributes={}) | X | | X
# create_other(attributes={}) | X | | X
# other.create!(attributes={}) | | | X
2007-05-30 03:05:14 -04:00
#
# ===Collection associations (one-to-many / many-to-many)
# | | | has_many
2007-12-22 06:26:03 -05:00
# generated methods | habtm | has_many | :through
2007-05-30 03:05:14 -04:00
# ----------------------------------+-------+----------+----------
2009-02-24 07:29:25 -05:00
# others | X | X | X
# others=(other,other,...) | X | X | X
# other_ids | X | X | X
# other_ids=(id,id,...) | X | X | X
# others<< | X | X | X
# others.push | X | X | X
# others.concat | X | X | X
# others.build(attributes={}) | X | X | X
# others.create(attributes={}) | X | X | X
# others.create!(attributes={}) | X | X | X
# others.size | X | X | X
# others.length | X | X | X
# others.count | X | X | X
# others.sum(args*,&block) | X | X | X
# others.empty? | X | X | X
# others.clear | X | X | X
# others.delete(other,other,...) | X | X | X
# others.delete_all | X | X |
# others.destroy_all | X | X | X
# others.find(*args) | X | X | X
# others.find_first | X | |
# others.exists? | X | X | X
# others.uniq | X | X | X
# others.reset | X | X | X
2004-11-23 20:04:44 -05:00
#
2007-01-15 01:43:47 -05:00
# == Cardinality and associations
2007-12-22 06:26:03 -05:00
#
2008-05-25 07:29:00 -04:00
# Active Record associations can be used to describe one-to-one, one-to-many and many-to-many
2008-04-04 23:52:58 -04:00
# relationships between models. Each model uses an association to describe its role in
# the relation. The +belongs_to+ association is always used in the model that has
2007-01-23 00:10:21 -05:00
# the foreign key.
2007-01-15 01:43:47 -05:00
#
# === One-to-one
#
2007-08-28 19:18:57 -04:00
# Use +has_one+ in the base, and +belongs_to+ in the associated model.
2007-01-15 01:43:47 -05:00
#
# class Employee < ActiveRecord::Base
# has_one :office
# end
# class Office < ActiveRecord::Base
# belongs_to :employee # foreign key - employee_id
# end
#
# === One-to-many
#
2007-08-28 19:18:57 -04:00
# Use +has_many+ in the base, and +belongs_to+ in the associated model.
2007-01-15 01:43:47 -05:00
#
# class Manager < ActiveRecord::Base
# has_many :employees
# end
# class Employee < ActiveRecord::Base
2007-01-23 00:10:21 -05:00
# belongs_to :manager # foreign key - manager_id
2007-01-15 01:43:47 -05:00
# end
#
# === Many-to-many
#
# There are two ways to build a many-to-many relationship.
#
2007-08-28 19:18:57 -04:00
# The first way uses a +has_many+ association with the <tt>:through</tt> option and a join model, so
2007-01-15 01:43:47 -05:00
# there are two stages of associations.
#
# class Assignment < ActiveRecord::Base
# belongs_to :programmer # foreign key - programmer_id
# belongs_to :project # foreign key - project_id
# end
# class Programmer < ActiveRecord::Base
# has_many :assignments
# has_many :projects, :through => :assignments
# end
# class Project < ActiveRecord::Base
# has_many :assignments
# has_many :programmers, :through => :assignments
# end
#
2007-08-28 19:18:57 -04:00
# For the second way, use +has_and_belongs_to_many+ in both models. This requires a join table
2007-01-15 01:43:47 -05:00
# that has no corresponding model or primary key.
#
# class Programmer < ActiveRecord::Base
# has_and_belongs_to_many :projects # foreign keys in the join table
# end
# class Project < ActiveRecord::Base
# has_and_belongs_to_many :programmers # foreign keys in the join table
# end
#
2007-08-28 19:18:57 -04:00
# Choosing which way to build a many-to-many relationship is not always simple.
2007-12-22 06:26:03 -05:00
# If you need to work with the relationship model as its own entity,
2007-08-28 19:18:57 -04:00
# use <tt>has_many :through</tt>. Use +has_and_belongs_to_many+ when working with legacy schemas or when
2007-01-15 01:43:47 -05:00
# you never work directly with the relationship itself.
#
2007-08-28 19:18:57 -04:00
# == Is it a +belongs_to+ or +has_one+ association?
2004-11-23 20:04:44 -05:00
#
2007-08-28 19:18:57 -04:00
# Both express a 1-1 relationship. The difference is mostly where to place the foreign key, which goes on the table for the class
# declaring the +belongs_to+ relationship. Example:
2004-11-23 20:04:44 -05:00
#
2006-07-07 06:58:22 -04:00
# class User < ActiveRecord::Base
# # I reference an account.
# belongs_to :account
2004-11-23 20:04:44 -05:00
# end
#
2006-07-07 06:58:22 -04:00
# class Account < ActiveRecord::Base
# # One user references me.
# has_one :user
2004-11-23 20:04:44 -05:00
# end
#
# The tables for these classes could look something like:
#
2006-07-07 06:58:22 -04:00
# CREATE TABLE users (
2004-11-23 20:04:44 -05:00
# id int(11) NOT NULL auto_increment,
2006-07-07 06:58:22 -04:00
# account_id int(11) default NULL,
# name varchar default NULL,
2004-11-23 20:04:44 -05:00
# PRIMARY KEY (id)
# )
#
2006-07-07 06:58:22 -04:00
# CREATE TABLE accounts (
2004-11-23 20:04:44 -05:00
# id int(11) NOT NULL auto_increment,
# name varchar default NULL,
# PRIMARY KEY (id)
# )
#
2005-01-15 12:45:16 -05:00
# == Unsaved objects and associations
#
2007-11-07 22:37:16 -05:00
# You can manipulate objects and associations before they are saved to the database, but there is some special behavior you should be
2005-01-15 12:45:16 -05:00
# aware of, mostly involving the saving of associated objects.
#
2009-01-31 20:44:30 -05:00
# Unless you enable the :autosave option on a <tt>has_one</tt>, <tt>belongs_to</tt>,
# <tt>has_many</tt>, or <tt>has_and_belongs_to_many</tt> association,
# in which case the members are always saved.
#
2005-01-15 12:45:16 -05:00
# === One-to-one associations
#
2007-08-28 19:18:57 -04:00
# * Assigning an object to a +has_one+ association automatically saves that object and the object being replaced (if there is one), in
# order to update their primary keys - except if the parent object is unsaved (<tt>new_record? == true</tt>).
# * If either of these saves fail (due to one of the objects being invalid) the assignment statement returns +false+ and the assignment
2005-01-15 12:45:16 -05:00
# is cancelled.
2008-05-09 05:38:02 -04:00
# * If you wish to assign an object to a +has_one+ association without saving it, use the <tt>association.build</tt> method (documented below).
2007-12-22 06:26:03 -05:00
# * Assigning an object to a +belongs_to+ association does not save the object, since the foreign key field belongs on the parent. It
2007-08-28 19:18:57 -04:00
# does not save the parent either.
2005-01-15 12:45:16 -05:00
#
# === Collections
#
2007-08-28 19:18:57 -04:00
# * Adding an object to a collection (+has_many+ or +has_and_belongs_to_many+) automatically saves that object, except if the parent object
2005-01-15 12:45:16 -05:00
# (the owner of the collection) is not yet stored in the database.
2008-05-09 05:38:02 -04:00
# * If saving any of the objects being added to a collection (via <tt>push</tt> or similar) fails, then <tt>push</tt> returns +false+.
# * You can add an object to a collection without automatically saving it by using the <tt>collection.build</tt> method (documented below).
2007-08-28 19:18:57 -04:00
# * All unsaved (<tt>new_record? == true</tt>) members of the collection are automatically saved when the parent is saved.
2005-01-15 12:45:16 -05:00
#
2005-07-06 06:14:58 -04:00
# === Association callbacks
2005-07-04 04:43:57 -04:00
#
2007-12-05 13:54:41 -05:00
# Similar to the normal callbacks that hook into the lifecycle of an Active Record object, you can also define callbacks that get
2007-11-07 22:37:16 -05:00
# triggered when you add an object to or remove an object from an association collection. Example:
2005-07-04 04:43:57 -04:00
#
# class Project
# has_and_belongs_to_many :developers, :after_add => :evaluate_velocity
#
# def evaluate_velocity(developer)
# ...
# end
2007-12-22 06:26:03 -05:00
# end
2005-07-04 04:43:57 -04:00
#
# It's possible to stack callbacks by passing them as an array. Example:
2007-12-22 06:26:03 -05:00
#
2005-07-04 04:43:57 -04:00
# class Project
2005-07-06 06:14:58 -04:00
# has_and_belongs_to_many :developers, :after_add => [:evaluate_velocity, Proc.new { |p, d| p.shipping_date = Time.now}]
2005-07-04 04:43:57 -04:00
# end
#
2007-08-28 19:18:57 -04:00
# Possible callbacks are: +before_add+, +after_add+, +before_remove+ and +after_remove+.
2005-07-04 04:43:57 -04:00
#
2007-08-28 19:18:57 -04:00
# Should any of the +before_add+ callbacks throw an exception, the object does not get added to the collection. Same with
# the +before_remove+ callbacks; if an exception is thrown the object doesn't get removed.
2005-07-04 04:43:57 -04:00
#
2005-11-03 04:06:42 -05:00
# === Association extensions
#
2007-08-28 19:18:57 -04:00
# The proxy objects that control the access to associations can be extended through anonymous modules. This is especially
2005-12-19 22:26:17 -05:00
# beneficial for adding new finders, creators, and other factory-type methods that are only used as part of this association.
2005-11-03 04:06:42 -05:00
# Example:
#
# class Account < ActiveRecord::Base
2005-11-06 14:05:42 -05:00
# has_many :people do
2005-11-03 04:06:42 -05:00
# def find_or_create_by_name(name)
2005-12-12 19:39:51 -05:00
# first_name, last_name = name.split(" ", 2)
2005-11-06 14:07:09 -05:00
# find_or_create_by_first_name_and_last_name(first_name, last_name)
2005-11-03 04:06:42 -05:00
# end
2005-11-06 14:05:42 -05:00
# end
2005-11-03 04:06:42 -05:00
# end
#
# person = Account.find(:first).people.find_or_create_by_name("David Heinemeier Hansson")
# person.first_name # => "David"
# person.last_name # => "Heinemeier Hansson"
#
2005-11-06 14:05:42 -05:00
# If you need to share the same extensions between many associations, you can use a named extension module. Example:
#
# module FindOrCreateByNameExtension
# def find_or_create_by_name(name)
2005-12-12 19:39:51 -05:00
# first_name, last_name = name.split(" ", 2)
2005-11-06 14:07:09 -05:00
# find_or_create_by_first_name_and_last_name(first_name, last_name)
2005-11-06 14:05:42 -05:00
# end
# end
#
# class Account < ActiveRecord::Base
# has_many :people, :extend => FindOrCreateByNameExtension
# end
#
# class Company < ActiveRecord::Base
# has_many :people, :extend => FindOrCreateByNameExtension
# end
2005-11-03 04:06:42 -05:00
#
2007-08-28 19:18:57 -04:00
# If you need to use multiple named extension modules, you can specify an array of modules with the <tt>:extend</tt> option.
2006-04-10 21:10:42 -04:00
# In the case of name conflicts between methods in the modules, methods in modules later in the array supercede
# those earlier in the array. Example:
#
# class Account < ActiveRecord::Base
# has_many :people, :extend => [FindOrCreateByNameExtension, FindRecentExtension]
# end
#
2006-05-28 17:33:34 -04:00
# Some extensions can only be made to work with knowledge of the association proxy's internals.
# Extensions can access relevant state using accessors on the association proxy:
2007-12-22 06:26:03 -05:00
#
2006-05-28 17:33:34 -04:00
# * +proxy_owner+ - Returns the object the association is part of.
# * +proxy_reflection+ - Returns the reflection object that describes the association.
2007-08-28 19:18:57 -04:00
# * +proxy_target+ - Returns the associated object for +belongs_to+ and +has_one+, or the collection of associated objects for +has_many+ and +has_and_belongs_to_many+.
2006-05-28 17:33:34 -04:00
#
2006-03-27 23:01:17 -05:00
# === Association Join Models
2007-12-22 06:26:03 -05:00
#
2007-08-28 19:18:57 -04:00
# Has Many associations can be configured with the <tt>:through</tt> option to use an explicit join model to retrieve the data. This
# operates similarly to a +has_and_belongs_to_many+ association. The advantage is that you're able to add validations,
2006-03-24 09:46:17 -05:00
# callbacks, and extra attributes on the join model. Consider the following schema:
2007-12-22 06:26:03 -05:00
#
2006-03-24 09:46:17 -05:00
# class Author < ActiveRecord::Base
# has_many :authorships
# has_many :books, :through => :authorships
# end
2007-12-22 06:26:03 -05:00
#
2006-03-24 09:46:17 -05:00
# class Authorship < ActiveRecord::Base
# belongs_to :author
# belongs_to :book
# end
2007-12-22 06:26:03 -05:00
#
2006-03-24 09:46:17 -05:00
# @author = Author.find :first
# @author.authorships.collect { |a| a.book } # selects all books that the author's authorships belong to.
# @author.books # selects all books by using the Authorship join model
2007-12-22 06:26:03 -05:00
#
2007-08-28 19:18:57 -04:00
# You can also go through a +has_many+ association on the join model:
2007-12-22 06:26:03 -05:00
#
2006-03-24 09:46:17 -05:00
# class Firm < ActiveRecord::Base
# has_many :clients
# has_many :invoices, :through => :clients
# end
2007-12-22 06:26:03 -05:00
#
2006-03-24 09:46:17 -05:00
# class Client < ActiveRecord::Base
# belongs_to :firm
# has_many :invoices
# end
2007-12-22 06:26:03 -05:00
#
2006-03-24 09:46:17 -05:00
# class Invoice < ActiveRecord::Base
# belongs_to :client
# end
#
# @firm = Firm.find :first
# @firm.clients.collect { |c| c.invoices }.flatten # select all invoices for all clients of the firm
# @firm.invoices # selects all invoices by going through the Client join model.
#
2006-03-27 23:01:17 -05:00
# === Polymorphic Associations
2007-12-22 06:26:03 -05:00
#
# Polymorphic associations on models are not restricted on what types of models they can be associated with. Rather, they
2007-08-28 19:18:57 -04:00
# specify an interface that a +has_many+ association must adhere to.
2007-12-22 06:26:03 -05:00
#
2006-03-27 23:01:17 -05:00
# class Asset < ActiveRecord::Base
# belongs_to :attachable, :polymorphic => true
# end
2007-12-22 06:26:03 -05:00
#
2006-03-27 23:01:17 -05:00
# class Post < ActiveRecord::Base
2007-08-28 19:18:57 -04:00
# has_many :assets, :as => :attachable # The :as option specifies the polymorphic interface to use.
2006-03-27 23:01:17 -05:00
# end
#
# @asset.attachable = @post
2007-12-22 06:26:03 -05:00
#
2006-03-27 23:01:17 -05:00
# This works by using a type column in addition to a foreign key to specify the associated record. In the Asset example, you'd need
2007-08-28 19:18:57 -04:00
# an +attachable_id+ integer column and an +attachable_type+ string column.
2006-03-27 23:01:17 -05:00
#
2006-10-08 22:05:50 -04:00
# Using polymorphic associations in combination with single table inheritance (STI) is a little tricky. In order
2007-12-22 06:26:03 -05:00
# for the associations to work as expected, ensure that you store the base model for the STI models in the
2006-10-08 22:05:50 -04:00
# type column of the polymorphic association. To continue with the asset example above, suppose there are guest posts
2007-08-28 19:18:57 -04:00
# and member posts that use the posts table for STI. In this case, there must be a +type+ column in the posts table.
2006-10-08 22:05:50 -04:00
#
# class Asset < ActiveRecord::Base
# belongs_to :attachable, :polymorphic => true
2007-12-22 06:26:03 -05:00
#
2006-10-08 22:05:50 -04:00
# def attachable_type=(sType)
# super(sType.to_s.classify.constantize.base_class.to_s)
# end
# end
2007-12-22 06:26:03 -05:00
#
2006-10-08 22:05:50 -04:00
# class Post < ActiveRecord::Base
# # because we store "Post" in attachable_type now :dependent => :destroy will work
# has_many :assets, :as => :attachable, :dependent => :destroy
# end
#
2007-12-05 15:24:41 -05:00
# class GuestPost < Post
2006-10-08 22:05:50 -04:00
# end
#
2007-12-05 15:24:41 -05:00
# class MemberPost < Post
2006-10-08 22:05:50 -04:00
# end
#
2004-11-23 20:04:44 -05:00
# == Caching
#
# All of the methods are built on a simple caching principle that will keep the result of the last query around unless specifically
2007-12-22 06:26:03 -05:00
# instructed not to. The cache is even shared across methods to make it even cheaper to use the macro-added methods without
2004-11-23 20:04:44 -05:00
# worrying too much about performance at the first go. Example:
#
# project.milestones # fetches milestones from the database
# project.milestones.size # uses the milestone cache
# project.milestones.empty? # uses the milestone cache
# project.milestones(true).size # fetches milestones from the database
# project.milestones # uses the milestone cache
#
2005-04-18 10:39:36 -04:00
# == Eager loading of associations
#
2008-05-25 07:29:00 -04:00
# Eager loading is a way to find objects of a certain class and a number of named associations. This is
2007-08-28 19:18:57 -04:00
# one of the easiest ways of to prevent the dreaded 1+N problem in which fetching 100 posts that each need to display their author
2008-05-25 07:29:00 -04:00
# triggers 101 database queries. Through the use of eager loading, the 101 queries can be reduced to 2. Example:
2005-04-18 10:39:36 -04:00
#
# class Post < ActiveRecord::Base
# belongs_to :author
# has_many :comments
# end
#
# Consider the following loop using the class above:
#
2008-05-25 07:29:00 -04:00
# for post in Post.all
2005-04-18 10:39:36 -04:00
# puts "Post: " + post.title
# puts "Written by: " + post.author.name
# puts "Last comment on: " + post.comments.first.created_on
2007-12-22 06:26:03 -05:00
# end
2005-04-18 10:39:36 -04:00
#
# To iterate over these one hundred posts, we'll generate 201 database queries. Let's first just optimize it for retrieving the author:
#
2005-04-19 06:39:12 -04:00
# for post in Post.find(:all, :include => :author)
2005-04-18 10:39:36 -04:00
#
2008-05-25 07:29:00 -04:00
# This references the name of the +belongs_to+ association that also used the <tt>:author</tt> symbol. After loading the posts, find
# will collect the +author_id+ from each one and load all the referenced authors with one query. Doing so will cut down the number of queries from 201 to 102.
2005-04-18 10:39:36 -04:00
#
# We can improve upon the situation further by referencing both associations in the finder with:
#
2005-04-19 06:39:12 -04:00
# for post in Post.find(:all, :include => [ :author, :comments ])
2005-04-18 10:39:36 -04:00
#
2008-05-25 07:29:00 -04:00
# This will load all comments with a single query. This reduces the total number of queries to 3. More generally the number of queries
# will be 1 plus the number of associations named (except if some of the associations are polymorphic +belongs_to+ - see below).
2007-05-19 18:36:36 -04:00
#
2007-08-28 19:18:57 -04:00
# To include a deep hierarchy of associations, use a hash:
2007-05-19 18:36:36 -04:00
#
# for post in Post.find(:all, :include => [ :author, { :comments => { :author => :gravatar } } ])
#
# That'll grab not only all the comments but all their authors and gravatar pictures. You can mix and match
# symbols, arrays and hashes in any combination to describe the associations you want to load.
#
# All of this power shouldn't fool you into thinking that you can pull out huge amounts of data with no performance penalty just because you've reduced
2005-10-26 09:05:48 -04:00
# the number of queries. The database still needs to send all the data to Active Record and it still needs to be processed. So it's no
# catch-all for performance problems, but it's a great way to cut down on the number of queries in a situation as the one described above.
2007-12-22 06:26:03 -05:00
#
2008-05-25 07:29:00 -04:00
# Since only one table is loaded at a time, conditions or orders cannot reference tables other than the main one. If this is the case
# Active Record falls back to the previously used LEFT OUTER JOIN based strategy. For example
#
# Post.find(:all, :include => [ :author, :comments ], :conditions => ['comments.approved = ?', true])
2005-04-18 10:39:36 -04:00
#
2008-05-25 07:29:00 -04:00
# will result in a single SQL query with joins along the lines of: <tt>LEFT OUTER JOIN comments ON comments.post_id = posts.id</tt> and
# <tt>LEFT OUTER JOIN authors ON authors.id = posts.author_id</tt>. Note that using conditions like this can have unintended consequences.
# In the above example posts with no approved comments are not returned at all, because the conditions apply to the SQL statement as a whole
# and not just to the association. You must disambiguate column references for this fallback to happen, for example
# <tt>:order => "author.name DESC"</tt> will work but <tt>:order => "name DESC"</tt> will not.
#
# If you do want eagerload only some members of an association it is usually more natural to <tt>:include</tt> an association
# which has conditions defined on it:
#
# class Post < ActiveRecord::Base
# has_many :approved_comments, :class_name => 'Comment', :conditions => ['approved = ?', true]
# end
#
# Post.find(:all, :include => :approved_comments)
#
# will load posts and eager load the +approved_comments+ association, which contains only those comments that have been approved.
2007-12-22 06:26:03 -05:00
#
2008-11-02 07:43:54 -05:00
# If you eager load an association with a specified <tt>:limit</tt> option, it will be ignored, returning all the associated objects:
#
# class Picture < ActiveRecord::Base
# has_many :most_recent_comments, :class_name => 'Comment', :order => 'id DESC', :limit => 10
# end
#
# Picture.find(:first, :include => :most_recent_comments).most_recent_comments # => returns all associated comments.
#
2006-10-11 10:02:24 -04:00
# When eager loaded, conditions are interpolated in the context of the model class, not the model instance. Conditions are lazily interpolated
# before the actual model exists.
2007-12-22 06:26:03 -05:00
#
2008-05-25 07:29:00 -04:00
# Eager loading is supported with polymorphic associations.
2008-01-04 18:21:52 -05:00
#
# class Address < ActiveRecord::Base
# belongs_to :addressable, :polymorphic => true
# end
#
2008-05-25 07:29:00 -04:00
# A call that tries to eager load the addressable model
2008-01-04 18:21:52 -05:00
#
2008-05-25 07:29:00 -04:00
# Address.find(:all, :include => :addressable)
2008-01-23 12:06:40 -05:00
#
2008-05-25 07:29:00 -04:00
# will execute one query to load the addresses and load the addressables with one query per addressable type.
# For example if all the addressables are either of class Person or Company then a total of 3 queries will be executed. The list of
# addressable types to load is determined on the back of the addresses loaded. This is not supported if Active Record has to fallback
# to the previous implementation of eager loading and will raise ActiveRecord::EagerLoadPolymorphicError. The reason is that the parent
# model's type is a column value so its corresponding table name cannot be put in the +FROM+/+JOIN+ clauses of that query.
2008-01-04 18:21:52 -05:00
#
2006-03-23 01:01:01 -05:00
# == Table Aliasing
#
2008-05-25 07:29:00 -04:00
# Active Record uses table aliasing in the case that a table is referenced multiple times in a join. If a table is referenced only once,
2007-08-28 19:18:57 -04:00
# the standard table name is used. The second time, the table is aliased as <tt>#{reflection_name}_#{parent_table_name}</tt>. Indexes are appended
2006-03-23 01:01:01 -05:00
# for any more successive uses of the table name.
2007-12-22 06:26:03 -05:00
#
2008-05-25 07:29:00 -04:00
# Post.find :all, :joins => :comments
# # => SELECT ... FROM posts INNER JOIN comments ON ...
# Post.find :all, :joins => :special_comments # STI
# # => SELECT ... FROM posts INNER JOIN comments ON ... AND comments.type = 'SpecialComment'
# Post.find :all, :joins => [:comments, :special_comments] # special_comments is the reflection name, posts is the parent table name
# # => SELECT ... FROM posts INNER JOIN comments ON ... INNER JOIN comments special_comments_posts
2007-12-22 06:26:03 -05:00
#
2006-03-23 01:01:01 -05:00
# Acts as tree example:
2007-12-22 06:26:03 -05:00
#
2008-05-25 07:29:00 -04:00
# TreeMixin.find :all, :joins => :children
# # => SELECT ... FROM mixins INNER JOIN mixins childrens_mixins ...
# TreeMixin.find :all, :joins => {:children => :parent}
# # => SELECT ... FROM mixins INNER JOIN mixins childrens_mixins ...
# INNER JOIN parents_mixins ...
# TreeMixin.find :all, :joins => {:children => {:parent => :children}}
# # => SELECT ... FROM mixins INNER JOIN mixins childrens_mixins ...
# INNER JOIN parents_mixins ...
# INNER JOIN mixins childrens_mixins_2
2007-12-22 06:26:03 -05:00
#
2007-08-28 19:18:57 -04:00
# Has and Belongs to Many join tables use the same idea, but add a <tt>_join</tt> suffix:
2007-12-22 06:26:03 -05:00
#
2008-05-25 07:29:00 -04:00
# Post.find :all, :joins => :categories
# # => SELECT ... FROM posts INNER JOIN categories_posts ... INNER JOIN categories ...
# Post.find :all, :joins => {:categories => :posts}
# # => SELECT ... FROM posts INNER JOIN categories_posts ... INNER JOIN categories ...
# INNER JOIN categories_posts posts_categories_join INNER JOIN posts posts_categories
# Post.find :all, :joins => {:categories => {:posts => :categories}}
# # => SELECT ... FROM posts INNER JOIN categories_posts ... INNER JOIN categories ...
# INNER JOIN categories_posts posts_categories_join INNER JOIN posts posts_categories
# INNER JOIN categories_posts categories_posts_join INNER JOIN categories categories_posts_2
2007-12-22 06:26:03 -05:00
#
2007-08-28 19:18:57 -04:00
# If you wish to specify your own custom joins using a <tt>:joins</tt> option, those table names will take precedence over the eager associations:
2007-12-22 06:26:03 -05:00
#
2008-05-25 07:29:00 -04:00
# Post.find :all, :joins => :comments, :joins => "inner join comments ..."
# # => SELECT ... FROM posts INNER JOIN comments_posts ON ... INNER JOIN comments ...
# Post.find :all, :joins => [:comments, :special_comments], :joins => "inner join comments ..."
# # => SELECT ... FROM posts INNER JOIN comments comments_posts ON ...
# INNER JOIN comments special_comments_posts ...
2006-03-23 01:01:01 -05:00
# INNER JOIN comments ...
2007-12-22 06:26:03 -05:00
#
2006-03-23 01:01:01 -05:00
# Table aliases are automatically truncated according to the maximum length of table identifiers according to the specific database.
2007-12-22 06:26:03 -05:00
#
2004-11-23 20:04:44 -05:00
# == Modules
#
# By default, associations will look for objects within the current module scope. Consider:
#
# module MyApplication
# module Business
# class Firm < ActiveRecord::Base
# has_many :clients
# end
#
2008-07-16 08:00:36 -04:00
# class Client < ActiveRecord::Base; end
2004-11-23 20:04:44 -05:00
# end
# end
#
2008-07-16 08:00:36 -04:00
# When <tt>Firm#clients</tt> is called, it will in turn call <tt>MyApplication::Business::Client.find_all_by_firm_id(firm.id)</tt>.
# If you want to associate with a class in another module scope, this can be done by specifying the complete class name.
# Example:
2004-11-23 20:04:44 -05:00
#
# module MyApplication
# module Business
# class Firm < ActiveRecord::Base; end
# end
#
# module Billing
# class Account < ActiveRecord::Base
# belongs_to :firm, :class_name => "MyApplication::Business::Firm"
# end
# end
# end
#
2007-08-28 19:18:57 -04:00
# == Type safety with <tt>ActiveRecord::AssociationTypeMismatch</tt>
2004-11-23 20:04:44 -05:00
#
# If you attempt to assign an object to an association that doesn't match the inferred or specified <tt>:class_name</tt>, you'll
2007-08-28 19:18:57 -04:00
# get an <tt>ActiveRecord::AssociationTypeMismatch</tt>.
2004-11-23 20:04:44 -05:00
#
# == Options
#
2007-08-28 19:18:57 -04:00
# All of the association macros can be specialized through options. This makes cases more complex than the simple and guessable ones
2004-11-23 20:04:44 -05:00
# possible.
module ClassMethods
2008-07-28 07:26:59 -04:00
# Specifies a one-to-many association. The following methods for retrieval and query of
# collections of associated objects will be added:
#
# [collection(force_reload = false)]
# Returns an array of all the associated objects.
2004-11-23 20:04:44 -05:00
# An empty array is returned if none are found.
2008-07-28 07:26:59 -04:00
# [collection<<(object, ...)]
# Adds one or more objects to the collection by setting their foreign keys to the collection's primary key.
# [collection.delete(object, ...)]
# Removes one or more objects from the collection by setting their foreign keys to +NULL+.
2008-10-21 13:33:40 -04:00
# Objects will be in addition destroyed if they're associated with <tt>:dependent => :destroy</tt>,
# and deleted if they're associated with <tt>:dependent => :delete_all</tt>.
2008-07-28 07:26:59 -04:00
# [collection=objects]
# Replaces the collections content by deleting and adding objects as appropriate.
# [collection_singular_ids]
# Returns an array of the associated objects' ids
# [collection_singular_ids=ids]
# Replace the collection with the objects identified by the primary keys in +ids+
# [collection.clear]
# Removes every object from the collection. This destroys the associated objects if they
# are associated with <tt>:dependent => :destroy</tt>, deletes them directly from the
# database if <tt>:dependent => :delete_all</tt>, otherwise sets their foreign keys to +NULL+.
# [collection.empty?]
# Returns +true+ if there are no associated objects.
# [collection.size]
# Returns the number of associated objects.
# [collection.find(...)]
# Finds an associated object according to the same rules as ActiveRecord::Base.find.
2008-12-19 09:27:43 -05:00
# [collection.exists?(...)]
2008-07-28 07:26:59 -04:00
# Checks whether an associated object with the given conditions exists.
# Uses the same rules as ActiveRecord::Base.exists?.
# [collection.build(attributes = {}, ...)]
# Returns one or more new objects of the collection type that have been instantiated
# with +attributes+ and linked to this object through a foreign key, but have not yet
# been saved. <b>Note:</b> This only works if an associated object already exists, not if
# it's +nil+!
# [collection.create(attributes = {})]
# Returns a new object of the collection type that has been instantiated
# with +attributes+, linked to this object through a foreign key, and that has already
# been saved (if it passed the validation). <b>Note:</b> This only works if an associated
# object already exists, not if it's +nil+!
#
# (*Note*: +collection+ is replaced with the symbol passed as the first argument, so
# <tt>has_many :clients</tt> would add among others <tt>clients.empty?</tt>.)
#
# === Example
2004-11-23 20:04:44 -05:00
#
2008-09-03 12:58:47 -04:00
# Example: A Firm class declares <tt>has_many :clients</tt>, which will add:
# * <tt>Firm#clients</tt> (similar to <tt>Clients.find :all, :conditions => ["firm_id = ?", id]</tt>)
2004-11-23 20:04:44 -05:00
# * <tt>Firm#clients<<</tt>
# * <tt>Firm#clients.delete</tt>
2005-06-16 01:35:10 -04:00
# * <tt>Firm#clients=</tt>
2006-10-08 21:32:11 -04:00
# * <tt>Firm#client_ids</tt>
2005-06-16 01:35:10 -04:00
# * <tt>Firm#client_ids=</tt>
2004-11-23 20:04:44 -05:00
# * <tt>Firm#clients.clear</tt>
# * <tt>Firm#clients.empty?</tt> (similar to <tt>firm.clients.size == 0</tt>)
# * <tt>Firm#clients.size</tt> (similar to <tt>Client.count "firm_id = #{id}"</tt>)
2005-04-18 11:31:20 -04:00
# * <tt>Firm#clients.find</tt> (similar to <tt>Client.find(id, :conditions => "firm_id = #{id}")</tt>)
2008-12-19 09:27:43 -05:00
# * <tt>Firm#clients.exists?(:name => 'ACME')</tt> (similar to <tt>Client.exists?(:name => 'ACME', :firm_id => firm.id)</tt>)
2004-11-23 20:04:44 -05:00
# * <tt>Firm#clients.build</tt> (similar to <tt>Client.new("firm_id" => id)</tt>)
2005-06-21 13:11:01 -04:00
# * <tt>Firm#clients.create</tt> (similar to <tt>c = Client.new("firm_id" => id); c.save; c</tt>)
2004-11-23 20:04:44 -05:00
# The declaration can also include an options hash to specialize the behavior of the association.
2007-12-22 06:26:03 -05:00
#
2008-07-28 07:26:59 -04:00
# === Supported options
# [:class_name]
# Specify the class name of the association. Use it only if that name can't be inferred
2008-05-09 05:38:02 -04:00
# from the association name. So <tt>has_many :products</tt> will by default be linked to the Product class, but
# if the real class name is SpecialProduct, you'll have to specify it with this option.
2008-07-28 07:26:59 -04:00
# [:conditions]
# Specify the conditions that the associated objects must meet in order to be included as a +WHERE+
2008-03-21 14:21:56 -04:00
# SQL fragment, such as <tt>price > 5 AND name LIKE 'B%'</tt>. Record creations from the association are scoped if a hash
# is used. <tt>has_many :posts, :conditions => {:published => true}</tt> will create published posts with <tt>@blog.posts.create</tt>
# or <tt>@blog.posts.build</tt>.
2008-07-28 07:26:59 -04:00
# [:order]
# Specify the order in which the associated objects are returned as an <tt>ORDER BY</tt> SQL fragment,
2008-05-09 05:38:02 -04:00
# such as <tt>last_name, first_name DESC</tt>.
2008-07-28 07:26:59 -04:00
# [:foreign_key]
# Specify the foreign key used for the association. By default this is guessed to be the name
2008-05-09 05:38:02 -04:00
# of this class in lower-case and "_id" suffixed. So a Person class that makes a +has_many+ association will use "person_id"
# as the default <tt>:foreign_key</tt>.
2008-07-28 07:26:59 -04:00
# [:primary_key]
# Specify the method that returns the primary key used for the association. By default this is +id+.
# [:dependent]
# If set to <tt>:destroy</tt> all the associated objects are destroyed
2008-05-09 05:38:02 -04:00
# alongside this object by calling their +destroy+ method. If set to <tt>:delete_all</tt> all associated
# objects are deleted *without* calling their +destroy+ method. If set to <tt>:nullify</tt> all associated
# objects' foreign keys are set to +NULL+ *without* calling their +save+ callbacks. *Warning:* This option is ignored when also using
# the <tt>:through</tt> option.
2008-07-28 07:26:59 -04:00
# [:finder_sql]
# Specify a complete SQL statement to fetch the association. This is a good way to go for complex
2005-10-26 09:05:48 -04:00
# associations that depend on multiple tables. Note: When this option is used, +find_in_collection+ is _not_ added.
2008-07-28 07:26:59 -04:00
# [:counter_sql]
# Specify a complete SQL statement to fetch the size of the association. If <tt>:finder_sql</tt> is
2007-11-07 22:37:16 -05:00
# specified but not <tt>:counter_sql</tt>, <tt>:counter_sql</tt> will be generated by replacing <tt>SELECT ... FROM</tt> with <tt>SELECT COUNT(*) FROM</tt>.
2008-07-28 07:26:59 -04:00
# [:extend]
# Specify a named module for extending the proxy. See "Association extensions".
# [:include]
# Specify second-order associations that should be eager loaded when the collection is loaded.
# [:group]
# An attribute name by which the result should be grouped. Uses the <tt>GROUP BY</tt> SQL-clause.
2008-11-21 17:20:33 -05:00
# [:having]
# Combined with +:group+ this can be used to filter the records that a <tt>GROUP BY</tt> returns. Uses the <tt>HAVING</tt> SQL-clause.
2008-07-28 07:26:59 -04:00
# [:limit]
# An integer determining the limit on the number of rows that should be returned.
# [:offset]
# An integer determining the offset from where the rows should be fetched. So at 5, it would skip the first 4 rows.
# [:select]
# By default, this is <tt>*</tt> as in <tt>SELECT * FROM</tt>, but can be changed if you, for example, want to do a join
2008-06-01 02:19:40 -04:00
# but not include the joined columns. Do not forget to include the primary and foreign keys, otherwise it will raise an error.
2008-07-28 07:26:59 -04:00
# [:as]
# Specifies a polymorphic interface (See <tt>belongs_to</tt>).
# [:through]
# Specifies a Join Model through which to perform the query. Options for <tt>:class_name</tt> and <tt>:foreign_key</tt>
2007-05-23 02:35:08 -04:00
# are ignored, as the association uses the source reflection. You can only use a <tt>:through</tt> query through a <tt>belongs_to</tt>
# or <tt>has_many</tt> association on the join model.
2008-07-28 07:26:59 -04:00
# [:source]
# Specifies the source association name used by <tt>has_many :through</tt> queries. Only use it if the name cannot be
2007-08-28 19:18:57 -04:00
# inferred from the association. <tt>has_many :subscribers, :through => :subscriptions</tt> will look for either <tt>:subscribers</tt> or
2008-05-09 05:38:02 -04:00
# <tt>:subscriber</tt> on Subscription, unless a <tt>:source</tt> is given.
2008-07-28 07:26:59 -04:00
# [:source_type]
# Specifies type of the source association used by <tt>has_many :through</tt> queries where the source
2007-08-28 19:18:57 -04:00
# association is a polymorphic +belongs_to+.
2008-07-28 07:26:59 -04:00
# [:uniq]
# If true, duplicates will be omitted from the collection. Useful in conjunction with <tt>:through</tt>.
# [:readonly]
# If true, all the associated objects are readonly through the association.
# [:validate]
# If false, don't validate the associated objects when saving the parent object. true by default.
2009-01-31 20:44:30 -05:00
# [:autosave]
# If true, always save any loaded members and destroy members marked for destruction, when saving the parent object. Off by default.
#
2004-11-23 20:04:44 -05:00
# Option examples:
# has_many :comments, :order => "posted_on"
2005-11-06 15:39:34 -05:00
# has_many :comments, :include => :author
2004-11-23 20:04:44 -05:00
# has_many :people, :class_name => "Person", :conditions => "deleted = 0", :order => "name"
2006-03-09 03:40:34 -05:00
# has_many :tracks, :order => "position", :dependent => :destroy
# has_many :comments, :dependent => :nullify
2006-03-27 23:01:17 -05:00
# has_many :tags, :as => :taggable
2008-02-13 01:32:50 -05:00
# has_many :reports, :readonly => true
2006-03-24 09:46:17 -05:00
# has_many :subscribers, :through => :subscriptions, :source => :user
2004-11-23 20:04:44 -05:00
# has_many :subscribers, :class_name => "Person", :finder_sql =>
# 'SELECT DISTINCT people.* ' +
# 'FROM people p, post_subscriptions ps ' +
# 'WHERE ps.post_id = #{id} AND ps.person_id = p.id ' +
# 'ORDER BY p.first_name'
2005-11-06 14:05:42 -05:00
def has_many ( association_id , options = { } , & extension )
2005-12-02 23:29:55 -05:00
reflection = create_has_many_reflection ( association_id , options , & extension )
configure_dependency_for_has_many ( reflection )
2008-04-05 20:27:12 -04:00
add_association_callbacks ( reflection . name , reflection . options )
2005-12-02 23:29:55 -05:00
if options [ :through ]
2008-04-05 20:27:12 -04:00
collection_accessor_methods ( reflection , HasManyThroughAssociation )
2005-12-02 23:29:55 -05:00
else
collection_accessor_methods ( reflection , HasManyAssociation )
2004-11-23 20:04:44 -05:00
end
end
2008-07-28 07:26:59 -04:00
# Specifies a one-to-one association with another class. This method should only be used
# if the other class contains the foreign key. If the current class contains the foreign key,
# then you should use +belongs_to+ instead. See also ActiveRecord::Associations::ClassMethods's overview
# on when to use has_one and when to use belongs_to.
#
# The following methods for retrieval and query of a single associated object will be added:
#
# [association(force_reload = false)]
# Returns the associated object. +nil+ is returned if none is found.
# [association=(associate)]
# Assigns the associate object, extracts the primary key, sets it as the foreign key,
2004-11-23 20:04:44 -05:00
# and saves the associate object.
2008-07-28 07:26:59 -04:00
# [build_association(attributes = {})]
# Returns a new object of the associated type that has been instantiated
# with +attributes+ and linked to this object through a foreign key, but has not
# yet been saved. <b>Note:</b> This ONLY works if an association already exists.
# It will NOT work if the association is +nil+.
# [create_association(attributes = {})]
# Returns a new object of the associated type that has been instantiated
# with +attributes+, linked to this object through a foreign key, and that
# has already been saved (if it passed the validation).
#
# (+association+ is replaced with the symbol passed as the first argument, so
# <tt>has_one :manager</tt> would add among others <tt>manager.nil?</tt>.)
2004-11-23 20:04:44 -05:00
#
2008-07-28 07:26:59 -04:00
# === Example
#
# An Account class declares <tt>has_one :beneficiary</tt>, which will add:
2005-06-26 07:25:32 -04:00
# * <tt>Account#beneficiary</tt> (similar to <tt>Beneficiary.find(:first, :conditions => "account_id = #{id}")</tt>)
2004-11-23 20:04:44 -05:00
# * <tt>Account#beneficiary=(beneficiary)</tt> (similar to <tt>beneficiary.account_id = account.id; beneficiary.save</tt>)
2005-04-12 01:34:10 -04:00
# * <tt>Account#build_beneficiary</tt> (similar to <tt>Beneficiary.new("account_id" => id)</tt>)
# * <tt>Account#create_beneficiary</tt> (similar to <tt>b = Beneficiary.new("account_id" => id); b.save; b</tt>)
#
2008-07-28 07:26:59 -04:00
# === Options
#
2004-11-23 20:04:44 -05:00
# The declaration can also include an options hash to specialize the behavior of the association.
2007-12-22 06:26:03 -05:00
#
2004-11-23 20:04:44 -05:00
# Options are:
2008-07-28 07:26:59 -04:00
# [:class_name]
# Specify the class name of the association. Use it only if that name can't be inferred
2008-05-09 05:38:02 -04:00
# from the association name. So <tt>has_one :manager</tt> will by default be linked to the Manager class, but
# if the real class name is Person, you'll have to specify it with this option.
2008-07-28 07:26:59 -04:00
# [:conditions]
# Specify the conditions that the associated object must meet in order to be included as a +WHERE+
2007-08-28 19:18:57 -04:00
# SQL fragment, such as <tt>rank = 5</tt>.
2008-07-28 07:26:59 -04:00
# [:order]
# Specify the order in which the associated objects are returned as an <tt>ORDER BY</tt> SQL fragment,
2008-05-09 05:38:02 -04:00
# such as <tt>last_name, first_name DESC</tt>.
2008-07-28 07:26:59 -04:00
# [:dependent]
# If set to <tt>:destroy</tt>, the associated object is destroyed when this object is. If set to
2007-08-28 19:18:57 -04:00
# <tt>:delete</tt>, the associated object is deleted *without* calling its destroy method. If set to <tt>:nullify</tt>, the associated
# object's foreign key is set to +NULL+. Also, association is assigned.
2008-07-28 07:26:59 -04:00
# [:foreign_key]
# Specify the foreign key used for the association. By default this is guessed to be the name
2008-05-09 05:38:02 -04:00
# of this class in lower-case and "_id" suffixed. So a Person class that makes a +has_one+ association will use "person_id"
# as the default <tt>:foreign_key</tt>.
2008-07-28 07:26:59 -04:00
# [:primary_key]
# Specify the method that returns the primary key used for the association. By default this is +id+.
# [:include]
# Specify second-order associations that should be eager loaded when this object is loaded.
# [:as]
# Specifies a polymorphic interface (See <tt>belongs_to</tt>).
# [:select]
# By default, this is <tt>*</tt> as in <tt>SELECT * FROM</tt>, but can be changed if, for example, you want to do a join
2008-05-23 05:20:13 -04:00
# but not include the joined columns. Do not forget to include the primary and foreign keys, otherwise it will raise an error.
2008-07-28 07:26:59 -04:00
# [:through]
# Specifies a Join Model through which to perform the query. Options for <tt>:class_name</tt> and <tt>:foreign_key</tt>
2008-03-21 14:09:03 -04:00
# are ignored, as the association uses the source reflection. You can only use a <tt>:through</tt> query through a
# <tt>has_one</tt> or <tt>belongs_to</tt> association on the join model.
2008-07-28 07:26:59 -04:00
# [:source]
# Specifies the source association name used by <tt>has_one :through</tt> queries. Only use it if the name cannot be
2008-03-21 14:09:03 -04:00
# inferred from the association. <tt>has_one :favorite, :through => :favorites</tt> will look for a
2009-02-27 07:50:24 -05:00
# <tt>:favorite</tt> on Favorite, unless a <tt>:source</tt> is given.
2008-07-28 07:26:59 -04:00
# [:source_type]
# Specifies type of the source association used by <tt>has_one :through</tt> queries where the source
2009-02-27 07:50:24 -05:00
# association is a polymorphic +belongs_to+.
2008-07-28 07:26:59 -04:00
# [:readonly]
# If true, the associated object is readonly through the association.
# [:validate]
# If false, don't validate the associated object when saving the parent object. +false+ by default.
2009-01-31 20:44:30 -05:00
# [:autosave]
# If true, always save the associated object or destroy it if marked for destruction, when saving the parent object. Off by default.
2008-02-13 01:32:50 -05:00
#
2004-11-23 20:04:44 -05:00
# Option examples:
2006-03-09 03:40:34 -05:00
# has_one :credit_card, :dependent => :destroy # destroys the associated credit card
2007-11-07 22:37:16 -05:00
# has_one :credit_card, :dependent => :nullify # updates the associated records foreign key value to NULL rather than destroying it
2004-11-23 20:04:44 -05:00
# has_one :last_comment, :class_name => "Comment", :order => "posted_on"
# has_one :project_manager, :class_name => "Person", :conditions => "role = 'project_manager'"
2006-05-22 04:00:05 -04:00
# has_one :attachment, :as => :attachable
2008-02-13 01:32:50 -05:00
# has_one :boss, :readonly => :true
2008-03-21 14:09:03 -04:00
# has_one :club, :through => :membership
# has_one :primary_address, :through => :addressables, :conditions => ["addressable.primary = ?", true], :source => :addressable
2004-11-23 20:04:44 -05:00
def has_one ( association_id , options = { } )
2008-03-21 14:09:03 -04:00
if options [ :through ]
reflection = create_has_one_through_reflection ( association_id , options )
association_accessor_methods ( reflection , ActiveRecord :: Associations :: HasOneThroughAssociation )
else
reflection = create_has_one_reflection ( association_id , options )
association_accessor_methods ( reflection , HasOneAssociation )
association_constructor_method ( :build , reflection , HasOneAssociation )
association_constructor_method ( :create , reflection , HasOneAssociation )
configure_dependency_for_has_one ( reflection )
end
2004-11-23 20:04:44 -05:00
end
2008-07-28 07:26:59 -04:00
# Specifies a one-to-one association with another class. This method should only be used
# if this class contains the foreign key. If the other class contains the foreign key,
# then you should use +has_one+ instead. See also ActiveRecord::Associations::ClassMethods's overview
# on when to use +has_one+ and when to use +belongs_to+.
#
# Methods will be added for retrieval and query for a single associated object, for which
# this object holds an id:
#
# [association(force_reload = false)]
# Returns the associated object. +nil+ is returned if none is found.
# [association=(associate)]
# Assigns the associate object, extracts the primary key, and sets it as the foreign key.
# [build_association(attributes = {})]
# Returns a new object of the associated type that has been instantiated
2007-08-28 19:18:57 -04:00
# with +attributes+ and linked to this object through a foreign key, but has not yet been saved.
2008-07-28 07:26:59 -04:00
# [create_association(attributes = {})]
# Returns a new object of the associated type that has been instantiated
# with +attributes+, linked to this object through a foreign key, and that
# has already been saved (if it passed the validation).
#
# (+association+ is replaced with the symbol passed as the first argument, so
# <tt>belongs_to :author</tt> would add among others <tt>author.nil?</tt>.)
2004-11-23 20:04:44 -05:00
#
2008-07-28 07:26:59 -04:00
# === Example
#
# A Post class declares <tt>belongs_to :author</tt>, which will add:
2004-11-23 20:04:44 -05:00
# * <tt>Post#author</tt> (similar to <tt>Author.find(author_id)</tt>)
# * <tt>Post#author=(author)</tt> (similar to <tt>post.author_id = author.id</tt>)
# * <tt>Post#author?</tt> (similar to <tt>post.author == some_author</tt>)
2005-06-21 13:11:01 -04:00
# * <tt>Post#build_author</tt> (similar to <tt>post.author = Author.new</tt>)
# * <tt>Post#create_author</tt> (similar to <tt>post.author = Author.new; post.author.save; post.author</tt>)
2004-11-23 20:04:44 -05:00
# The declaration can also include an options hash to specialize the behavior of the association.
2007-12-22 06:26:03 -05:00
#
2008-07-28 07:26:59 -04:00
# === Options
#
# [:class_name]
# Specify the class name of the association. Use it only if that name can't be inferred
2008-05-09 05:38:02 -04:00
# from the association name. So <tt>has_one :author</tt> will by default be linked to the Author class, but
# if the real class name is Person, you'll have to specify it with this option.
2008-07-28 07:26:59 -04:00
# [:conditions]
# Specify the conditions that the associated object must meet in order to be included as a +WHERE+
2007-08-28 19:18:57 -04:00
# SQL fragment, such as <tt>authorized = 1</tt>.
2008-07-28 07:26:59 -04:00
# [:select]
# By default, this is <tt>*</tt> as in <tt>SELECT * FROM</tt>, but can be changed if, for example, you want to do a join
2008-05-23 05:20:13 -04:00
# but not include the joined columns. Do not forget to include the primary and foreign keys, otherwise it will raise an error.
2008-07-28 07:26:59 -04:00
# [:foreign_key]
# Specify the foreign key used for the association. By default this is guessed to be the name
2008-05-09 05:38:02 -04:00
# of the association with an "_id" suffix. So a class that defines a <tt>belongs_to :person</tt> association will use
# "person_id" as the default <tt>:foreign_key</tt>. Similarly, <tt>belongs_to :favorite_person, :class_name => "Person"</tt>
# will use a foreign key of "favorite_person_id".
2008-07-28 07:26:59 -04:00
# [:dependent]
# If set to <tt>:destroy</tt>, the associated object is destroyed when this object is. If set to
2008-01-19 00:30:42 -05:00
# <tt>:delete</tt>, the associated object is deleted *without* calling its destroy method. This option should not be specified when
# <tt>belongs_to</tt> is used in conjunction with a <tt>has_many</tt> relationship on another class because of the potential to leave
# orphaned records behind.
2008-07-28 07:26:59 -04:00
# [:counter_cache]
# Caches the number of belonging objects on the associate class through the use of +increment_counter+
2007-08-28 19:18:57 -04:00
# and +decrement_counter+. The counter cache is incremented when an object of this class is created and decremented when it's
2008-05-09 05:38:02 -04:00
# destroyed. This requires that a column named <tt>#{table_name}_count</tt> (such as +comments_count+ for a belonging Comment class)
# is used on the associate class (such as a Post class). You can also specify a custom counter cache column by providing
2007-08-28 19:18:57 -04:00
# a column name instead of a +true+/+false+ value to this option (e.g., <tt>:counter_cache => :my_custom_counter</tt>.)
2008-05-09 05:38:02 -04:00
# Note: Specifying a counter cache will add it to that model's list of readonly attributes using +attr_readonly+.
2008-07-28 07:26:59 -04:00
# [:include]
# Specify second-order associations that should be eager loaded when this object is loaded.
# [:polymorphic]
# Specify this association is a polymorphic association by passing +true+.
2007-12-22 06:26:03 -05:00
# Note: If you've enabled the counter cache, then you may want to add the counter cache attribute
2008-05-09 05:38:02 -04:00
# to the +attr_readonly+ list in the associated classes (e.g. <tt>class Post; attr_readonly :comments_count; end</tt>).
2008-07-28 07:26:59 -04:00
# [:readonly]
# If true, the associated object is readonly through the association.
# [:validate]
# If false, don't validate the associated objects when saving the parent object. +false+ by default.
2009-01-31 20:44:30 -05:00
# [:autosave]
# If true, always save the associated object or destroy it if marked for destruction, when saving the parent object. Off by default.
2009-04-16 18:25:55 -04:00
# [:touch]
# If true, the associated object will be touched (the updated_at/on attributes set to now) when this record is either saved or
# destroyed. If you specify a symbol, that attribute will be updated with the current time instead of the updated_at/on attribute.
2004-11-23 20:04:44 -05:00
#
# Option examples:
# belongs_to :firm, :foreign_key => "client_of"
# belongs_to :author, :class_name => "Person", :foreign_key => "author_id"
2007-12-22 06:26:03 -05:00
# belongs_to :valid_coupon, :class_name => "Coupon", :foreign_key => "coupon_id",
2004-11-23 20:04:44 -05:00
# :conditions => 'discounts > #{payments_count}'
2006-03-27 23:01:17 -05:00
# belongs_to :attachable, :polymorphic => true
2008-02-13 01:32:50 -05:00
# belongs_to :project, :readonly => true
2008-03-19 22:15:29 -04:00
# belongs_to :post, :counter_cache => true
2009-04-16 18:25:55 -04:00
# belongs_to :company, :touch => true
# belongs_to :company, :touch => :employees_last_updated_at
2005-01-15 12:45:16 -05:00
def belongs_to ( association_id , options = { } )
2005-12-02 23:29:55 -05:00
reflection = create_belongs_to_reflection ( association_id , options )
2007-12-22 06:26:03 -05:00
2005-12-02 23:29:55 -05:00
if reflection . options [ :polymorphic ]
association_accessor_methods ( reflection , BelongsToPolymorphicAssociation )
2005-12-02 01:03:43 -05:00
else
2005-12-02 23:29:55 -05:00
association_accessor_methods ( reflection , BelongsToAssociation )
association_constructor_method ( :build , reflection , BelongsToAssociation )
association_constructor_method ( :create , reflection , BelongsToAssociation )
2005-12-02 01:03:43 -05:00
end
2006-03-04 10:11:17 -05:00
2009-04-16 18:25:55 -04:00
add_counter_cache_callbacks ( reflection ) if options [ :counter_cache ]
add_touch_callbacks ( reflection , options [ :touch ] ) if options [ :touch ]
2008-01-19 00:30:42 -05:00
configure_dependency_for_belongs_to ( reflection )
2005-01-15 12:45:16 -05:00
end
2008-07-28 07:26:59 -04:00
# Specifies a many-to-many relationship with another class. This associates two classes via an
# intermediate join table. Unless the join table is explicitly specified as an option, it is
# guessed using the lexical order of the class names. So a join between Developer and Project
2008-05-09 05:38:02 -04:00
# will give the default join table name of "developers_projects" because "D" outranks "P". Note that this precedence
# is calculated using the <tt><</tt> operator for String. This means that if the strings are of different lengths,
2007-01-16 01:30:02 -05:00
# and the strings are equal when compared up to the shortest length, then the longer string is considered of higher
2008-05-09 05:38:02 -04:00
# lexical precedence than the shorter one. For example, one would expect the tables "paper_boxes" and "papers"
# to generate a join table name of "papers_paper_boxes" because of the length of the name "paper_boxes",
# but it in fact generates a join table name of "paper_boxes_papers". Be aware of this caveat, and use the
# custom <tt>:join_table</tt> option if you need to.
2004-11-23 20:04:44 -05:00
#
2009-02-05 20:57:02 -05:00
# The join table should not have a primary key or a model associated with it. You must manually generate the
# join table with a migration such as this:
#
# class CreateDevelopersProjectsJoinTable < ActiveRecord::Migration
# def self.up
# create_table :developers_projects, :id => false do |t|
# t.integer :developer_id
# t.integer :project_id
# end
# end
#
# def self.down
# drop_table :developers_projects
# end
# end
#
2006-04-01 15:03:10 -05:00
# Deprecated: Any additional fields added to the join table will be placed as attributes when pulling records out through
2007-08-28 19:18:57 -04:00
# +has_and_belongs_to_many+ associations. Records returned from join tables with additional attributes will be marked as
2008-05-25 07:29:00 -04:00
# readonly (because we can't save changes to the additional attributes). It's strongly recommended that you upgrade any
2006-04-01 15:03:10 -05:00
# associations with attributes to a real join model (see introduction).
2004-11-23 20:04:44 -05:00
#
2007-08-28 19:18:57 -04:00
# Adds the following methods for retrieval and query:
2008-07-28 07:26:59 -04:00
#
# [collection(force_reload = false)]
# Returns an array of all the associated objects.
2007-08-28 19:18:57 -04:00
# An empty array is returned if none are found.
2008-07-28 07:26:59 -04:00
# [collection<<(object, ...)]
# Adds one or more objects to the collection by creating associations in the join table
2007-08-28 19:18:57 -04:00
# (<tt>collection.push</tt> and <tt>collection.concat</tt> are aliases to this method).
2008-07-28 07:26:59 -04:00
# [collection.delete(object, ...)]
# Removes one or more objects from the collection by removing their associations from the join table.
2004-11-23 20:04:44 -05:00
# This does not destroy the objects.
2008-07-28 07:26:59 -04:00
# [collection=objects]
# Replaces the collection's content by deleting and adding objects as appropriate.
# [collection_singular_ids]
# Returns an array of the associated objects' ids.
# [collection_singular_ids=ids]
# Replace the collection by the objects identified by the primary keys in +ids+.
# [collection.clear]
# Removes every object from the collection. This does not destroy the objects.
# [collection.empty?]
# Returns +true+ if there are no associated objects.
# [collection.size]
# Returns the number of associated objects.
# [collection.find(id)]
# Finds an associated object responding to the +id+ and that
2005-01-15 12:45:16 -05:00
# meets the condition that it has to be associated with this object.
2008-07-28 07:26:59 -04:00
# Uses the same rules as ActiveRecord::Base.find.
2008-12-19 09:27:43 -05:00
# [collection.exists?(...)]
2008-07-28 07:26:59 -04:00
# Checks whether an associated object with the given conditions exists.
# Uses the same rules as ActiveRecord::Base.exists?.
# [collection.build(attributes = {})]
# Returns a new object of the collection type that has been instantiated
2007-08-28 19:18:57 -04:00
# with +attributes+ and linked to this object through the join table, but has not yet been saved.
2008-07-28 07:26:59 -04:00
# [collection.create(attributes = {})]
# Returns a new object of the collection type that has been instantiated
2007-08-28 19:18:57 -04:00
# with +attributes+, linked to this object through the join table, and that has already been saved (if it passed the validation).
2004-11-23 20:04:44 -05:00
#
2008-07-28 07:26:59 -04:00
# (+collection+ is replaced with the symbol passed as the first argument, so
# <tt>has_and_belongs_to_many :categories</tt> would add among others <tt>categories.empty?</tt>.)
#
# === Example
#
# A Developer class declares <tt>has_and_belongs_to_many :projects</tt>, which will add:
2004-11-23 20:04:44 -05:00
# * <tt>Developer#projects</tt>
# * <tt>Developer#projects<<</tt>
# * <tt>Developer#projects.delete</tt>
2005-06-16 01:35:10 -04:00
# * <tt>Developer#projects=</tt>
2006-10-08 21:32:11 -04:00
# * <tt>Developer#project_ids</tt>
2005-06-16 01:35:10 -04:00
# * <tt>Developer#project_ids=</tt>
2004-11-23 20:04:44 -05:00
# * <tt>Developer#projects.clear</tt>
# * <tt>Developer#projects.empty?</tt>
# * <tt>Developer#projects.size</tt>
# * <tt>Developer#projects.find(id)</tt>
2008-12-19 09:27:43 -05:00
# * <tt>Developer#clients.exists?(...)</tt>
2006-05-28 23:48:17 -04:00
# * <tt>Developer#projects.build</tt> (similar to <tt>Project.new("project_id" => id)</tt>)
# * <tt>Developer#projects.create</tt> (similar to <tt>c = Project.new("project_id" => id); c.save; c</tt>)
2004-11-23 20:04:44 -05:00
# The declaration may include an options hash to specialize the behavior of the association.
2007-12-22 06:26:03 -05:00
#
2008-07-28 07:26:59 -04:00
# === Options
#
# [:class_name]
# Specify the class name of the association. Use it only if that name can't be inferred
2007-12-22 06:26:03 -05:00
# from the association name. So <tt>has_and_belongs_to_many :projects</tt> will by default be linked to the
2008-05-09 05:38:02 -04:00
# Project class, but if the real class name is SuperProject, you'll have to specify it with this option.
2008-07-28 07:26:59 -04:00
# [:join_table]
# Specify the name of the join table if the default based on lexical order isn't what you want.
# <b>WARNING:</b> If you're overwriting the table name of either class, the +table_name+ method
# MUST be declared underneath any +has_and_belongs_to_many+ declaration in order to work.
# [:foreign_key]
# Specify the foreign key used for the association. By default this is guessed to be the name
2008-05-09 05:38:02 -04:00
# of this class in lower-case and "_id" suffixed. So a Person class that makes a +has_and_belongs_to_many+ association
2009-02-01 13:25:03 -05:00
# to Project will use "person_id" as the default <tt>:foreign_key</tt>.
2008-07-28 07:26:59 -04:00
# [:association_foreign_key]
2009-02-01 13:25:03 -05:00
# Specify the foreign key used for the association on the receiving side of the association.
# By default this is guessed to be the name of the associated class in lower-case and "_id" suffixed.
# So if a Person class makes a +has_and_belongs_to_many+ association to Project,
# the association will use "project_id" as the default <tt>:association_foreign_key</tt>.
2008-07-28 07:26:59 -04:00
# [:conditions]
# Specify the conditions that the associated object must meet in order to be included as a +WHERE+
2008-03-21 14:21:56 -04:00
# SQL fragment, such as <tt>authorized = 1</tt>. Record creations from the association are scoped if a hash is used.
# <tt>has_many :posts, :conditions => {:published => true}</tt> will create published posts with <tt>@blog.posts.create</tt>
# or <tt>@blog.posts.build</tt>.
2008-07-28 07:26:59 -04:00
# [:order]
# Specify the order in which the associated objects are returned as an <tt>ORDER BY</tt> SQL fragment,
2007-11-07 22:37:16 -05:00
# such as <tt>last_name, first_name DESC</tt>
2008-07-28 07:26:59 -04:00
# [:uniq]
# If true, duplicate associated objects will be ignored by accessors and query methods.
# [:finder_sql]
# Overwrite the default generated SQL statement used to fetch the association with a manual statement
2008-08-28 14:01:42 -04:00
# [:counter_sql]
# Specify a complete SQL statement to fetch the size of the association. If <tt>:finder_sql</tt> is
# specified but not <tt>:counter_sql</tt>, <tt>:counter_sql</tt> will be generated by replacing <tt>SELECT ... FROM</tt> with <tt>SELECT COUNT(*) FROM</tt>.
2008-07-28 07:26:59 -04:00
# [:delete_sql]
# Overwrite the default generated SQL statement used to remove links between the associated
2008-05-09 05:38:02 -04:00
# classes with a manual statement.
2008-07-28 07:26:59 -04:00
# [:insert_sql]
# Overwrite the default generated SQL statement used to add links between the associated classes
2008-05-09 05:38:02 -04:00
# with a manual statement.
2008-07-28 07:26:59 -04:00
# [:extend]
# Anonymous module for extending the proxy, see "Association extensions".
# [:include]
# Specify second-order associations that should be eager loaded when the collection is loaded.
# [:group]
# An attribute name by which the result should be grouped. Uses the <tt>GROUP BY</tt> SQL-clause.
2008-11-21 17:20:33 -05:00
# [:having]
# Combined with +:group+ this can be used to filter the records that a <tt>GROUP BY</tt> returns. Uses the <tt>HAVING</tt> SQL-clause.
2008-07-28 07:26:59 -04:00
# [:limit]
# An integer determining the limit on the number of rows that should be returned.
# [:offset]
# An integer determining the offset from where the rows should be fetched. So at 5, it would skip the first 4 rows.
# [:select]
# By default, this is <tt>*</tt> as in <tt>SELECT * FROM</tt>, but can be changed if, for example, you want to do a join
2008-05-23 05:20:13 -04:00
# but not include the joined columns. Do not forget to include the primary and foreign keys, otherwise it will raise an error.
2008-07-28 07:26:59 -04:00
# [:readonly]
# If true, all the associated objects are readonly through the association.
# [:validate]
# If false, don't validate the associated objects when saving the parent object. +true+ by default.
2009-01-31 20:44:30 -05:00
# [:autosave]
# If true, always save any loaded members and destroy members marked for destruction, when saving the parent object. Off by default.
2004-11-23 20:04:44 -05:00
#
# Option examples:
# has_and_belongs_to_many :projects
2005-11-06 15:39:34 -05:00
# has_and_belongs_to_many :projects, :include => [ :milestones, :manager ]
2004-11-23 20:04:44 -05:00
# has_and_belongs_to_many :nations, :class_name => "Country"
# has_and_belongs_to_many :categories, :join_table => "prods_cats"
2008-02-13 01:32:50 -05:00
# has_and_belongs_to_many :categories, :readonly => true
2007-12-22 06:26:03 -05:00
# has_and_belongs_to_many :active_projects, :join_table => 'developers_projects', :delete_sql =>
2005-05-19 13:07:56 -04:00
# 'DELETE FROM developers_projects WHERE active=1 AND developer_id = #{id} AND project_id = #{record.id}'
2005-11-06 14:05:42 -05:00
def has_and_belongs_to_many ( association_id , options = { } , & extension )
2005-12-02 23:29:55 -05:00
reflection = create_has_and_belongs_to_many_reflection ( association_id , options , & extension )
collection_accessor_methods ( reflection , HasAndBelongsToManyAssociation )
2004-11-23 20:04:44 -05:00
2005-11-08 05:19:09 -05:00
# Don't use a before_destroy callback since users' before_destroy
# callbacks will be executed after the association is wiped out.
2005-12-02 23:29:55 -05:00
old_method = " destroy_without_habtm_shim_for_ #{ reflection . name } "
2007-10-27 16:31:09 -04:00
class_eval <<-end_eval unless method_defined?(old_method)
2008-12-28 14:48:05 -05:00
alias_method : #{old_method}, :destroy_without_callbacks # alias_method :destroy_without_habtm_shim_for_posts, :destroy_without_callbacks
def destroy_without_callbacks # def destroy_without_callbacks
#{reflection.name}.clear # posts.clear
#{old_method} # destroy_without_habtm_shim_for_posts
end # end
2005-11-08 05:19:09 -05:00
end_eval
2005-12-02 23:29:55 -05:00
add_association_callbacks ( reflection . name , options )
2004-11-23 20:04:44 -05:00
end
private
2008-09-03 12:58:47 -04:00
# Generates a join table name from two provided table names.
# The names in the join table namesme end up in lexicographic order.
#
# join_table_name("members", "clubs") # => "clubs_members"
# join_table_name("members", "special_clubs") # => "members_special_clubs"
2004-11-23 20:04:44 -05:00
def join_table_name ( first_table_name , second_table_name )
if first_table_name < second_table_name
join_table = " #{ first_table_name } _ #{ second_table_name } "
else
join_table = " #{ second_table_name } _ #{ first_table_name } "
end
table_name_prefix + join_table + table_name_suffix
end
2007-12-22 06:26:03 -05:00
2005-12-02 23:29:55 -05:00
def association_accessor_methods ( reflection , association_proxy_class )
define_method ( reflection . name ) do | * params |
2005-01-18 06:07:03 -05:00
force_reload = params . first unless params . empty?
2009-01-31 20:44:30 -05:00
association = association_instance_get ( reflection . name )
2005-12-02 23:29:55 -05:00
2008-10-10 11:02:47 -04:00
if association . nil? || force_reload
2005-12-02 23:29:55 -05:00
association = association_proxy_class . new ( self , reflection )
2005-01-18 06:07:03 -05:00
retval = association . reload
2006-08-08 12:59:06 -04:00
if retval . nil? and association_proxy_class == BelongsToAssociation
2009-01-31 20:44:30 -05:00
association_instance_set ( reflection . name , nil )
2005-01-18 06:07:03 -05:00
return nil
end
2009-01-31 20:44:30 -05:00
association_instance_set ( reflection . name , association )
2005-01-18 06:07:03 -05:00
end
2006-08-08 12:59:06 -04:00
association . target . nil? ? nil : association
2005-01-18 06:07:03 -05:00
end
2008-09-26 23:28:21 -04:00
define_method ( " loaded_ #{ reflection . name } ? " ) do
2009-01-31 20:44:30 -05:00
association = association_instance_get ( reflection . name )
2008-09-26 23:28:21 -04:00
association && association . loaded?
end
2005-12-02 23:29:55 -05:00
define_method ( " #{ reflection . name } = " ) do | new_value |
2009-01-31 20:44:30 -05:00
association = association_instance_get ( reflection . name )
2007-12-22 06:26:03 -05:00
2007-05-21 23:39:36 -04:00
if association . nil? || association . target != new_value
2005-12-02 23:29:55 -05:00
association = association_proxy_class . new ( self , reflection )
2005-01-18 06:07:03 -05:00
end
2005-12-02 23:29:55 -05:00
2008-03-21 14:09:03 -04:00
if association_proxy_class == HasOneThroughAssociation
association . create_through_record ( new_value )
self . send ( reflection . name , new_value )
else
2008-08-26 12:20:24 -04:00
association . replace ( new_value )
2009-01-31 20:44:30 -05:00
association_instance_set ( reflection . name , new_value . nil? ? nil : association )
2008-03-21 14:09:03 -04:00
end
2005-01-18 06:07:03 -05:00
end
2005-04-03 13:50:11 -04:00
2005-12-02 23:29:55 -05:00
define_method ( " set_ #{ reflection . name } _target " ) do | target |
2006-08-17 19:46:55 -04:00
return if target . nil? and association_proxy_class == BelongsToAssociation
2005-12-02 23:29:55 -05:00
association = association_proxy_class . new ( self , reflection )
2005-04-03 13:50:11 -04:00
association . target = target
2009-01-31 20:44:30 -05:00
association_instance_set ( reflection . name , association )
2005-04-03 13:50:11 -04:00
end
2005-01-18 06:07:03 -05:00
end
2005-12-02 23:29:55 -05:00
def collection_reader_method ( reflection , association_proxy_class )
define_method ( reflection . name ) do | * params |
2005-01-15 12:45:16 -05:00
force_reload = params . first unless params . empty?
2009-01-31 20:44:30 -05:00
association = association_instance_get ( reflection . name )
2005-12-02 23:29:55 -05:00
2009-01-31 20:44:30 -05:00
unless association
2005-12-02 23:29:55 -05:00
association = association_proxy_class . new ( self , reflection )
2009-01-31 20:44:30 -05:00
association_instance_set ( reflection . name , association )
2004-11-23 20:04:44 -05:00
end
2005-12-02 23:29:55 -05:00
2005-01-15 12:45:16 -05:00
association . reload if force_reload
2005-12-02 23:29:55 -05:00
2005-01-15 12:45:16 -05:00
association
end
2007-12-22 06:26:03 -05:00
define_method ( " #{ reflection . name . to_s . singularize } _ids " ) do
2008-10-23 21:53:13 -04:00
if send ( reflection . name ) . loaded? || reflection . options [ :finder_sql ]
2008-08-30 18:17:29 -04:00
send ( reflection . name ) . map ( & :id )
else
2008-08-30 20:13:53 -04:00
send ( reflection . name ) . all ( :select = > " #{ reflection . quoted_table_name } . #{ reflection . klass . primary_key } " ) . map ( & :id )
2008-08-30 18:17:29 -04:00
end
2007-12-22 06:26:03 -05:00
end
2005-12-02 23:29:55 -05:00
end
2004-11-23 20:04:44 -05:00
2007-06-27 04:19:12 -04:00
def collection_accessor_methods ( reflection , association_proxy_class , writer = true )
2005-12-02 23:29:55 -05:00
collection_reader_method ( reflection , association_proxy_class )
2007-12-22 06:26:03 -05:00
if writer
define_method ( " #{ reflection . name } = " ) do | new_value |
# Loads proxy class instance (defined in collection_reader_method) if not already loaded
association = send ( reflection . name )
association . replace ( new_value )
association
end
2005-06-16 01:35:10 -04:00
2007-12-22 06:26:03 -05:00
define_method ( " #{ reflection . name . to_s . singularize } _ids= " ) do | new_value |
ids = ( new_value || [ ] ) . reject { | nid | nid . blank? }
send ( " #{ reflection . name } = " , reflection . class_name . constantize . find ( ids ) )
end
2006-10-01 15:15:51 -04:00
end
2004-11-23 20:04:44 -05:00
end
2008-09-02 04:31:49 -04:00
2005-12-02 23:29:55 -05:00
def association_constructor_method ( constructor , reflection , association_proxy_class )
define_method ( " #{ constructor } _ #{ reflection . name } " ) do | * params |
2005-06-06 17:10:59 -04:00
attributees = params . first unless params . empty?
replace_existing = params [ 1 ] . nil? ? true : params [ 1 ]
2009-01-31 20:44:30 -05:00
association = association_instance_get ( reflection . name )
2005-04-12 01:34:10 -04:00
2009-01-31 20:44:30 -05:00
unless association
2005-12-02 23:29:55 -05:00
association = association_proxy_class . new ( self , reflection )
2009-01-31 20:44:30 -05:00
association_instance_set ( reflection . name , association )
2005-04-12 01:34:10 -04:00
end
2005-06-16 01:35:10 -04:00
if association_proxy_class == HasOneAssociation
association . send ( constructor , attributees , replace_existing )
else
association . send ( constructor , attributees )
end
2005-04-12 01:34:10 -04:00
end
end
2007-12-22 06:26:03 -05:00
2009-04-16 18:25:55 -04:00
def add_counter_cache_callbacks ( reflection )
cache_column = reflection . counter_cache_column
method_name = " belongs_to_counter_cache_after_create_for_ #{ reflection . name } " . to_sym
define_method ( method_name ) do
association = send ( reflection . name )
association . class . increment_counter ( cache_column , send ( reflection . primary_key_name ) ) unless association . nil?
end
after_create ( method_name )
method_name = " belongs_to_counter_cache_before_destroy_for_ #{ reflection . name } " . to_sym
define_method ( method_name ) do
association = send ( reflection . name )
association . class . decrement_counter ( cache_column , send ( reflection . primary_key_name ) ) unless association . nil?
end
before_destroy ( method_name )
module_eval (
" #{ reflection . class_name } .send(:attr_readonly, \" #{ cache_column } \" .intern) if defined?( #{ reflection . class_name } ) && #{ reflection . class_name } .respond_to?(:attr_readonly) "
)
end
def add_touch_callbacks ( reflection , touch_attribute )
method_name = " belongs_to_touch_after_save_or_destroy_for_ #{ reflection . name } " . to_sym
define_method ( method_name ) do
association = send ( reflection . name )
if touch_attribute == true
association . touch unless association . nil?
else
association . touch ( touch_attribute ) unless association . nil?
end
end
after_save ( method_name )
after_destroy ( method_name )
end
2005-04-03 06:52:05 -04:00
def find_with_associations ( options = { } )
2006-04-06 00:16:08 -04:00
catch :invalid_query do
2007-11-07 10:07:39 -05:00
join_dependency = JoinDependency . new ( self , merge_includes ( scope ( :find , :include ) , options [ :include ] ) , options [ :joins ] )
2006-04-06 00:16:08 -04:00
rows = select_all_rows ( options , join_dependency )
return join_dependency . instantiate ( rows )
end
[ ]
2006-03-15 22:27:21 -05:00
end
2005-04-19 12:32:57 -04:00
2008-09-21 17:51:02 -04:00
# Creates before_destroy callback methods that nullify, delete or destroy
# has_many associated objects, according to the defined :dependent rule.
#
2007-10-08 20:32:36 -04:00
# See HasManyAssociation#delete_records. Dependent associations
# delete children, otherwise foreign key is set to NULL.
2008-09-21 17:51:02 -04:00
#
# The +extra_conditions+ parameter, which is not used within the main
# Active Record codebase, is meant to allow plugins to define extra
# finder conditions.
def configure_dependency_for_has_many ( reflection , extra_conditions = nil )
2007-10-08 20:32:36 -04:00
if reflection . options . include? ( :dependent )
# Add polymorphic type if the :as option is present
dependent_conditions = [ ]
dependent_conditions << " #{ reflection . primary_key_name } = \# {record.quoted_id} "
dependent_conditions << " #{ reflection . options [ :as ] } _type = ' #{ base_class . name } ' " if reflection . options [ :as ]
2009-04-20 13:12:40 -04:00
dependent_conditions << sanitize_sql ( reflection . options [ :conditions ] , reflection . quoted_table_name ) if reflection . options [ :conditions ]
2008-09-21 17:51:02 -04:00
dependent_conditions << extra_conditions if extra_conditions
2007-10-08 20:32:36 -04:00
dependent_conditions = dependent_conditions . collect { | where | " ( #{ where } ) " } . join ( " AND " )
2008-12-21 10:38:40 -05:00
dependent_conditions = dependent_conditions . gsub ( '@' , '\@' )
2007-10-08 20:32:36 -04:00
case reflection . options [ :dependent ]
when :destroy
2008-02-14 02:53:37 -05:00
method_name = " has_many_dependent_destroy_for_ #{ reflection . name } " . to_sym
define_method ( method_name ) do
2008-09-02 04:31:49 -04:00
send ( reflection . name ) . each { | o | o . destroy }
2008-02-14 02:53:37 -05:00
end
before_destroy method_name
2007-10-08 20:32:36 -04:00
when :delete_all
2008-09-21 17:51:02 -04:00
module_eval %Q{
2008-12-28 14:48:05 -05:00
before_destroy do | record | # before_destroy do |record|
delete_all_has_many_dependencies ( record , # delete_all_has_many_dependencies(record,
" #{ reflection . name } " , # "posts",
#{reflection.class_name}, # Post,
%@#{dependent_conditions}@ ) # %@...@) # this is a string literal like %(...)
end # end
2008-09-21 17:51:02 -04:00
}
2007-10-08 20:32:36 -04:00
when :nullify
2008-09-21 17:51:02 -04:00
module_eval %Q{
2008-12-28 14:48:05 -05:00
before_destroy do | record | # before_destroy do |record|
nullify_has_many_dependencies ( record , # nullify_has_many_dependencies(record,
" #{ reflection . name } " , # "posts",
#{reflection.class_name}, # Post,
" #{ reflection . primary_key_name } " , # "user_id",
%@#{dependent_conditions}@ ) # %@...@) # this is a string literal like %(...)
end # end
2008-09-21 17:51:02 -04:00
}
2007-10-08 20:32:36 -04:00
else
raise ArgumentError , " The :dependent option expects either :destroy, :delete_all, or :nullify ( #{ reflection . options [ :dependent ] . inspect } ) "
end
2005-12-02 23:29:55 -05:00
end
end
2006-09-15 03:02:05 -04:00
2008-10-05 17:16:26 -04:00
# Creates before_destroy callback methods that nullify, delete or destroy
# has_one associated objects, according to the defined :dependent rule.
2005-12-02 23:29:55 -05:00
def configure_dependency_for_has_one ( reflection )
2007-10-08 20:32:36 -04:00
if reflection . options . include? ( :dependent )
case reflection . options [ :dependent ]
when :destroy
2008-02-14 02:53:37 -05:00
method_name = " has_one_dependent_destroy_for_ #{ reflection . name } " . to_sym
define_method ( method_name ) do
2008-09-02 04:31:49 -04:00
association = send ( reflection . name )
2008-02-14 02:53:37 -05:00
association . destroy unless association . nil?
end
before_destroy method_name
2007-10-08 20:32:36 -04:00
when :delete
2008-02-14 02:53:37 -05:00
method_name = " has_one_dependent_delete_for_ #{ reflection . name } " . to_sym
define_method ( method_name ) do
2008-10-05 17:16:26 -04:00
# Retrieve the associated object and delete it. The retrieval
# is necessary because there may be multiple associated objects
# with foreign keys pointing to this object, and we only want
# to delete the correct one, not all of them.
2008-09-02 04:31:49 -04:00
association = send ( reflection . name )
2008-09-21 17:01:32 -04:00
association . delete unless association . nil?
2008-02-14 02:53:37 -05:00
end
before_destroy method_name
2007-10-08 20:32:36 -04:00
when :nullify
2008-02-14 02:53:37 -05:00
method_name = " has_one_dependent_nullify_for_ #{ reflection . name } " . to_sym
define_method ( method_name ) do
2008-09-02 04:31:49 -04:00
association = send ( reflection . name )
association . update_attribute ( reflection . primary_key_name , nil ) unless association . nil?
2008-02-14 02:53:37 -05:00
end
before_destroy method_name
2007-10-08 20:32:36 -04:00
else
raise ArgumentError , " The :dependent option expects either :destroy, :delete or :nullify ( #{ reflection . options [ :dependent ] . inspect } ) "
end
2005-12-02 23:29:55 -05:00
end
end
2008-01-19 00:30:42 -05:00
def configure_dependency_for_belongs_to ( reflection )
if reflection . options . include? ( :dependent )
case reflection . options [ :dependent ]
when :destroy
2008-02-14 02:53:37 -05:00
method_name = " belongs_to_dependent_destroy_for_ #{ reflection . name } " . to_sym
define_method ( method_name ) do
2008-09-02 04:31:49 -04:00
association = send ( reflection . name )
2008-02-14 02:53:37 -05:00
association . destroy unless association . nil?
end
2009-01-05 15:32:34 -05:00
after_destroy method_name
2008-01-19 00:30:42 -05:00
when :delete
2008-02-14 02:53:37 -05:00
method_name = " belongs_to_dependent_delete_for_ #{ reflection . name } " . to_sym
define_method ( method_name ) do
2008-09-02 04:31:49 -04:00
association = send ( reflection . name )
2008-09-21 17:01:32 -04:00
association . delete unless association . nil?
2008-02-14 02:53:37 -05:00
end
2009-01-05 15:32:34 -05:00
after_destroy method_name
2008-01-19 00:30:42 -05:00
else
raise ArgumentError , " The :dependent option expects either :destroy or :delete ( #{ reflection . options [ :dependent ] . inspect } ) "
end
end
end
2008-09-21 17:51:02 -04:00
def delete_all_has_many_dependencies ( record , reflection_name , association_class , dependent_conditions )
association_class . delete_all ( dependent_conditions )
end
def nullify_has_many_dependencies ( record , reflection_name , association_class , primary_key_name , dependent_conditions )
association_class . update_all ( " #{ primary_key_name } = NULL " , dependent_conditions )
end
2008-09-06 12:37:31 -04:00
mattr_accessor :valid_keys_for_has_many_association
@@valid_keys_for_has_many_association = [
:class_name , :table_name , :foreign_key , :primary_key ,
:dependent ,
2008-11-21 17:20:33 -05:00
:select , :conditions , :include , :order , :group , :having , :limit , :offset ,
2008-09-06 12:37:31 -04:00
:as , :through , :source , :source_type ,
:uniq ,
:finder_sql , :counter_sql ,
:before_add , :after_add , :before_remove , :after_remove ,
:extend , :readonly ,
2009-05-01 11:01:13 -04:00
:validate , :inverse_of
2008-09-06 12:37:31 -04:00
]
2005-12-02 23:29:55 -05:00
def create_has_many_reflection ( association_id , options , & extension )
2008-09-06 12:37:31 -04:00
options . assert_valid_keys ( valid_keys_for_has_many_association )
2007-12-22 06:26:03 -05:00
options [ :extend ] = create_extension_modules ( association_id , extension , options [ :extend ] )
2005-12-02 23:29:55 -05:00
2006-02-26 23:45:36 -05:00
create_reflection ( :has_many , association_id , options , self )
2005-12-02 23:29:55 -05:00
end
2008-09-06 12:37:31 -04:00
mattr_accessor :valid_keys_for_has_one_association
@@valid_keys_for_has_one_association = [
:class_name , :foreign_key , :remote , :select , :conditions , :order ,
:include , :dependent , :counter_cache , :extend , :as , :readonly ,
2009-05-01 11:01:13 -04:00
:validate , :primary_key , :inverse_of
2008-09-06 12:37:31 -04:00
]
2005-12-02 23:29:55 -05:00
def create_has_one_reflection ( association_id , options )
2008-09-06 12:37:31 -04:00
options . assert_valid_keys ( valid_keys_for_has_one_association )
2006-02-26 23:45:36 -05:00
create_reflection ( :has_one , association_id , options , self )
2005-12-02 23:29:55 -05:00
end
2008-09-02 04:31:49 -04:00
2008-03-21 14:09:03 -04:00
def create_has_one_through_reflection ( association_id , options )
options . assert_valid_keys (
2008-06-11 07:08:35 -04:00
:class_name , :foreign_key , :remote , :select , :conditions , :order , :include , :dependent , :counter_cache , :extend , :as , :through , :source , :source_type , :validate
2008-03-21 14:09:03 -04:00
)
create_reflection ( :has_one , association_id , options , self )
end
2005-12-02 23:29:55 -05:00
2008-09-06 12:37:31 -04:00
mattr_accessor :valid_keys_for_belongs_to_association
@@valid_keys_for_belongs_to_association = [
:class_name , :foreign_key , :foreign_type , :remote , :select , :conditions ,
:include , :dependent , :counter_cache , :extend , :polymorphic , :readonly ,
2009-05-01 11:01:13 -04:00
:validate , :touch , :inverse_of
2008-09-06 12:37:31 -04:00
]
2005-12-02 23:29:55 -05:00
def create_belongs_to_reflection ( association_id , options )
2008-09-06 12:37:31 -04:00
options . assert_valid_keys ( valid_keys_for_belongs_to_association )
2005-12-02 23:29:55 -05:00
reflection = create_reflection ( :belongs_to , association_id , options , self )
if options [ :polymorphic ]
reflection . options [ :foreign_type ] || = reflection . class_name . underscore + " _type "
end
reflection
end
2007-12-22 06:26:03 -05:00
2008-10-31 12:28:16 -04:00
mattr_accessor :valid_keys_for_has_and_belongs_to_many_association
@@valid_keys_for_has_and_belongs_to_many_association = [
:class_name , :table_name , :join_table , :foreign_key , :association_foreign_key ,
2008-11-21 17:20:33 -05:00
:select , :conditions , :include , :order , :group , :having , :limit , :offset ,
2008-10-31 12:28:16 -04:00
:uniq ,
2008-11-06 15:00:54 -05:00
:finder_sql , :counter_sql , :delete_sql , :insert_sql ,
2008-10-31 12:28:16 -04:00
:before_add , :after_add , :before_remove , :after_remove ,
:extend , :readonly ,
:validate
]
2005-12-02 23:29:55 -05:00
def create_has_and_belongs_to_many_reflection ( association_id , options , & extension )
2008-10-31 12:28:16 -04:00
options . assert_valid_keys ( valid_keys_for_has_and_belongs_to_many_association )
2005-12-02 23:29:55 -05:00
2007-12-22 06:26:03 -05:00
options [ :extend ] = create_extension_modules ( association_id , extension , options [ :extend ] )
2005-12-02 23:29:55 -05:00
reflection = create_reflection ( :has_and_belongs_to_many , association_id , options , self )
2009-03-06 13:26:34 -05:00
if reflection . association_foreign_key == reflection . primary_key_name
raise HasAndBelongsToManyAssociationForeignKeyNeeded . new ( reflection )
end
2005-12-02 23:29:55 -05:00
reflection . options [ :join_table ] || = join_table_name ( undecorated_table_name ( self . to_s ) , undecorated_table_name ( reflection . class_name ) )
2007-12-22 06:26:03 -05:00
2005-12-02 23:29:55 -05:00
reflection
end
2005-04-18 14:49:34 -04:00
def reflect_on_included_associations ( associations )
2005-09-18 16:41:44 -04:00
[ associations ] . flatten . collect { | association | reflect_on_association ( association . to_s . intern ) }
2005-04-18 14:49:34 -04:00
end
2005-07-11 02:09:08 -04:00
def guard_against_unlimitable_reflections ( reflections , options )
if ( options [ :offset ] || options [ :limit ] ) && ! using_limitable_reflections? ( reflections )
raise (
2007-12-22 06:26:03 -05:00
ConfigurationError ,
2005-07-11 02:09:08 -04:00
" You can not use offset and limit together with has_many or has_and_belongs_to_many associations "
)
end
end
2006-03-04 18:33:10 -05:00
def select_all_rows ( options , join_dependency )
2005-04-18 14:49:34 -04:00
connection . select_all (
2006-03-04 18:33:10 -05:00
construct_finder_sql_with_included_associations ( options , join_dependency ) ,
2005-04-18 14:49:34 -04:00
" #{ name } Load Including Associations "
)
end
2005-04-10 13:24:56 -04:00
2006-03-04 18:33:10 -05:00
def construct_finder_sql_with_included_associations ( options , join_dependency )
2006-03-27 18:28:19 -05:00
scope = scope ( :find )
2007-12-04 21:30:30 -05:00
sql = " SELECT #{ column_aliases ( join_dependency ) } FROM #{ ( scope && scope [ :from ] ) || options [ :from ] || quoted_table_name } "
2006-03-04 18:33:10 -05:00
sql << join_dependency . join_associations . collect { | join | join . association_join } . join
2007-12-22 06:26:03 -05:00
2008-08-28 12:00:18 -04:00
add_joins! ( sql , options [ :joins ] , scope )
2006-03-27 18:28:19 -05:00
add_conditions! ( sql , options [ :conditions ] , scope )
2006-06-03 17:51:57 -04:00
add_limited_ids_condition! ( sql , options , join_dependency ) if ! using_limitable_reflections? ( join_dependency . reflections ) && ( ( scope && scope [ :limit ] ) || options [ :limit ] )
2005-10-18 08:02:25 -04:00
2008-11-21 17:20:33 -05:00
add_group! ( sql , options [ :group ] , options [ :having ] , scope )
2006-11-09 14:31:31 -05:00
add_order! ( sql , options [ :order ] , scope )
2006-03-27 18:28:19 -05:00
add_limit! ( sql , options , scope ) if using_limitable_reflections? ( join_dependency . reflections )
2006-12-05 17:07:55 -05:00
add_lock! ( sql , options , scope )
2007-12-22 06:26:03 -05:00
2005-04-03 06:52:05 -04:00
return sanitize_sql ( sql )
end
2007-12-22 06:26:03 -05:00
2006-03-04 18:33:10 -05:00
def add_limited_ids_condition! ( sql , options , join_dependency )
unless ( id_list = select_limited_ids_list ( options , join_dependency ) ) . empty?
2007-11-10 16:33:13 -05:00
sql << " #{ condition_word ( sql ) } #{ connection . quote_table_name table_name } . #{ primary_key } IN ( #{ id_list } ) "
2006-04-06 00:16:08 -04:00
else
throw :invalid_query
2005-10-18 08:02:25 -04:00
end
end
2007-09-22 19:51:03 -04:00
2006-03-04 18:33:10 -05:00
def select_limited_ids_list ( options , join_dependency )
2007-09-22 19:51:03 -04:00
pk = columns_hash [ primary_key ]
2006-04-18 18:24:03 -04:00
connection . select_all (
2006-03-04 18:33:10 -05:00
construct_finder_sql_for_association_limiting ( options , join_dependency ) ,
2005-10-18 08:02:25 -04:00
" #{ name } Load IDs For Limited Eager Loading "
2007-09-22 19:51:03 -04:00
) . collect { | row | connection . quote ( row [ primary_key ] , pk ) } . join ( " , " )
2005-10-18 08:02:25 -04:00
end
2006-11-10 14:18:07 -05:00
2006-03-04 18:33:10 -05:00
def construct_finder_sql_for_association_limiting ( options , join_dependency )
2006-10-13 04:29:00 -04:00
scope = scope ( :find )
2008-03-03 01:23:38 -05:00
# Only join tables referenced in order or conditions since this is particularly slow on the pre-query.
tables_from_conditions = conditions_tables ( options )
tables_from_order = order_tables ( options )
all_tables = tables_from_conditions + tables_from_order
2008-05-03 23:49:18 -04:00
distinct_join_associations = all_tables . uniq . map { | table |
join_dependency . joins_for_table_name ( table )
} . flatten . compact . uniq
2008-07-14 22:17:06 -04:00
order = options [ :order ]
if scoped_order = ( scope && scope [ :order ] )
order = order ? " #{ order } , #{ scoped_order } " : scoped_order
end
2008-03-03 01:23:38 -05:00
is_distinct = ! options [ :joins ] . blank? || include_eager_conditions? ( options , tables_from_conditions ) || include_eager_order? ( options , tables_from_order )
2006-02-09 04:17:40 -05:00
sql = " SELECT "
2006-10-13 04:29:00 -04:00
if is_distinct
2008-07-14 22:17:06 -04:00
sql << connection . distinct ( " #{ connection . quote_table_name table_name } . #{ primary_key } " , order )
2006-10-13 04:29:00 -04:00
else
sql << primary_key
end
2007-11-10 16:33:13 -05:00
sql << " FROM #{ connection . quote_table_name table_name } "
2006-11-10 14:18:07 -05:00
2006-10-13 04:29:00 -04:00
if is_distinct
2008-06-25 22:21:53 -04:00
sql << distinct_join_associations . collect { | assoc | assoc . association_join } . join
2008-08-28 12:00:18 -04:00
add_joins! ( sql , options [ :joins ] , scope )
2006-02-09 04:17:40 -05:00
end
2006-11-10 14:18:07 -05:00
2006-03-27 18:28:19 -05:00
add_conditions! ( sql , options [ :conditions ] , scope )
2008-11-21 17:20:33 -05:00
add_group! ( sql , options [ :group ] , options [ :having ] , scope )
2007-08-21 17:29:21 -04:00
2008-07-14 22:17:06 -04:00
if order && is_distinct
connection . add_order_by_for_association_limiting! ( sql , :order = > order )
2007-10-23 14:34:01 -04:00
else
add_order! ( sql , options [ :order ] , scope )
2007-01-12 00:14:55 -05:00
end
2007-08-21 17:29:21 -04:00
2006-03-27 18:28:19 -05:00
add_limit! ( sql , options , scope )
2007-08-21 17:29:21 -04:00
2005-10-18 08:02:25 -04:00
return sanitize_sql ( sql )
end
2006-10-08 23:12:32 -04:00
2008-12-18 14:07:55 -05:00
def tables_in_string ( string )
return [ ] if string . blank?
string . scan ( / ([ \ .a-zA-Z_]+).? \ . / ) . flatten
end
2009-05-04 10:49:43 -04:00
def tables_in_hash ( hash )
return [ ] if hash . blank?
tables = hash . map do | key , value |
if value . is_a? ( Hash )
key . to_s
else
tables_in_string ( key ) if key . is_a? ( String )
end
end
tables . flatten . compact
end
2008-03-03 01:23:38 -05:00
def conditions_tables ( options )
2006-04-19 10:50:10 -04:00
# look in both sets of conditions
conditions = [ scope ( :find , :conditions ) , options [ :conditions ] ] . inject ( [ ] ) do | all , cond |
case cond
when nil then all
2009-05-04 10:49:43 -04:00
when Array then all << tables_in_string ( cond . first )
when Hash then all << tables_in_hash ( cond )
else all << tables_in_string ( cond )
2006-04-19 10:50:10 -04:00
end
end
2009-05-04 10:49:43 -04:00
conditions . flatten
2008-03-03 01:23:38 -05:00
end
def order_tables ( options )
2008-06-09 11:30:48 -04:00
order = [ options [ :order ] , scope ( :find , :order ) ] . join ( " , " )
2008-03-03 01:23:38 -05:00
return [ ] unless order && order . is_a? ( String )
2008-12-18 14:07:55 -05:00
tables_in_string ( order )
2008-03-03 01:23:38 -05:00
end
2008-05-04 15:08:19 -04:00
def selects_tables ( options )
select = options [ :select ]
return [ ] unless select && select . is_a? ( String )
2008-12-18 14:07:55 -05:00
tables_in_string ( select )
end
def joined_tables ( options )
scope = scope ( :find )
joins = options [ :joins ]
merged_joins = scope && scope [ :joins ] && joins ? merge_joins ( scope [ :joins ] , joins ) : ( joins || scope && scope [ :joins ] )
[ table_name ] + case merged_joins
when Symbol , Hash , Array
if array_of_strings? ( merged_joins )
tables_in_string ( merged_joins . join ( ' ' ) )
else
join_dependency = ActiveRecord :: Associations :: ClassMethods :: InnerJoinDependency . new ( self , merged_joins , nil )
2008-12-18 05:41:45 -05:00
join_dependency . join_associations . collect { | join_association | [ join_association . aliased_join_table_name , join_association . aliased_table_name ] } . flatten . compact
2008-12-18 14:07:55 -05:00
end
else
tables_in_string ( merged_joins )
end
2008-05-04 15:08:19 -04:00
end
2008-03-03 01:23:38 -05:00
# Checks if the conditions reference a table other than the current model table
2008-12-18 14:07:55 -05:00
def include_eager_conditions? ( options , tables = nil , joined_tables = nil )
( ( tables || conditions_tables ( options ) ) - ( joined_tables || joined_tables ( options ) ) ) . any?
2005-10-18 08:02:25 -04:00
end
2007-12-22 06:26:03 -05:00
2006-04-19 10:50:10 -04:00
# Checks if the query order references a table other than the current model's table.
2008-12-18 14:07:55 -05:00
def include_eager_order? ( options , tables = nil , joined_tables = nil )
( ( tables || order_tables ( options ) ) - ( joined_tables || joined_tables ( options ) ) ) . any?
2006-02-09 04:17:40 -05:00
end
2005-10-18 08:02:25 -04:00
2008-12-18 14:07:55 -05:00
def include_eager_select? ( options , joined_tables = nil )
( selects_tables ( options ) - ( joined_tables || joined_tables ( options ) ) ) . any?
2008-05-04 15:08:19 -04:00
end
2008-01-18 23:19:53 -05:00
def references_eager_loaded_tables? ( options )
2008-12-18 14:07:55 -05:00
joined_tables = joined_tables ( options )
include_eager_order? ( options , nil , joined_tables ) || include_eager_conditions? ( options , nil , joined_tables ) || include_eager_select? ( options , joined_tables )
2008-01-18 23:19:53 -05:00
end
2005-07-11 02:09:08 -04:00
def using_limitable_reflections? ( reflections )
reflections . reject { | r | [ :belongs_to , :has_one ] . include? ( r . macro ) } . length . zero?
end
2006-03-04 18:33:10 -05:00
def column_aliases ( join_dependency )
join_dependency . joins . collect { | join | join . column_names_with_alias . collect { | column_name , aliased_name |
2007-11-10 16:33:13 -05:00
" #{ connection . quote_table_name join . aliased_table_name } . #{ connection . quote_column_name column_name } AS #{ aliased_name } " } } . flatten . join ( " , " )
2005-04-10 13:24:56 -04:00
end
2005-07-04 04:43:57 -04:00
def add_association_callbacks ( association_name , options )
2005-10-01 22:00:50 -04:00
callbacks = %w( before_add after_add before_remove after_remove )
callbacks . each do | callback_name |
2006-04-15 19:24:59 -04:00
full_callback_name = " #{ callback_name } _for_ #{ association_name } "
2005-10-01 22:00:50 -04:00
defined_callbacks = options [ callback_name . to_sym ]
if options . has_key? ( callback_name . to_sym )
class_inheritable_reader full_callback_name . to_sym
2007-10-27 16:31:09 -04:00
write_inheritable_attribute ( full_callback_name . to_sym , [ defined_callbacks ] . flatten )
else
write_inheritable_attribute ( full_callback_name . to_sym , [ ] )
2005-10-01 22:00:50 -04:00
end
end
2005-07-04 04:43:57 -04:00
end
2005-04-10 13:24:56 -04:00
2005-10-18 08:02:25 -04:00
def condition_word ( sql )
sql =~ / where /i ? " AND " : " WHERE "
end
2005-10-01 22:00:50 -04:00
2007-09-17 17:19:44 -04:00
def create_extension_modules ( association_id , block_extension , extensions )
2007-12-22 06:26:03 -05:00
if block_extension
2008-09-12 06:02:40 -04:00
extension_module_name = " #{ self . to_s . demodulize } #{ association_id . to_s . camelize } AssociationExtension "
2005-11-06 14:05:42 -05:00
2007-12-22 06:26:03 -05:00
silence_warnings do
2008-09-12 06:02:40 -04:00
self . parent . const_set ( extension_module_name , Module . new ( & block_extension ) )
2007-12-22 06:26:03 -05:00
end
2008-09-12 06:02:40 -04:00
Array ( extensions ) . push ( " #{ self . parent } :: #{ extension_module_name } " . constantize )
2007-12-22 06:26:03 -05:00
else
Array ( extensions )
2005-11-06 14:05:42 -05:00
end
end
2006-03-04 18:33:10 -05:00
2006-07-04 21:06:51 -04:00
class JoinDependency # :nodoc:
2006-03-05 13:43:56 -05:00
attr_reader :joins , :reflections , :table_aliases
2006-03-04 18:33:10 -05:00
2007-11-07 10:07:39 -05:00
def initialize ( base , associations , joins )
2006-03-19 11:13:52 -05:00
@joins = [ JoinBase . new ( base , joins ) ]
2006-03-04 18:33:10 -05:00
@associations = associations
@reflections = [ ]
@base_records_hash = { }
@base_records_in_order = [ ]
2006-03-05 13:43:56 -05:00
@table_aliases = Hash . new { | aliases , table | aliases [ table ] = 0 }
@table_aliases [ base . table_name ] = 1
2006-03-04 18:33:10 -05:00
build ( associations )
end
def join_associations
@joins [ 1 .. - 1 ] . to_a
end
def join_base
@joins [ 0 ]
end
def instantiate ( rows )
rows . each_with_index do | row , i |
primary_id = join_base . record_id ( row )
unless @base_records_hash [ primary_id ]
@base_records_in_order << ( @base_records_hash [ primary_id ] = join_base . instantiate ( row ) )
end
2007-11-07 10:07:39 -05:00
construct ( @base_records_hash [ primary_id ] , @associations , join_associations . dup , row )
2006-03-04 18:33:10 -05:00
end
2007-11-07 10:07:39 -05:00
remove_duplicate_results! ( join_base . active_record , @base_records_in_order , @associations )
2006-03-04 18:33:10 -05:00
return @base_records_in_order
end
2007-10-28 23:02:42 -04:00
def remove_duplicate_results! ( base , records , associations )
case associations
when Symbol , String
reflection = base . reflections [ associations ]
if reflection && [ :has_many , :has_and_belongs_to_many ] . include? ( reflection . macro )
records . each { | record | record . send ( reflection . name ) . target . uniq! }
end
when Array
associations . each do | association |
remove_duplicate_results! ( base , records , association )
end
when Hash
associations . keys . each do | name |
reflection = base . reflections [ name ]
is_collection = [ :has_many , :has_and_belongs_to_many ] . include? ( reflection . macro )
parent_records = records . map do | record |
2008-02-28 15:42:01 -05:00
descendant = record . send ( reflection . name )
next unless descendant
descendant . target . uniq! if is_collection
descendant
2007-10-28 23:02:42 -04:00
end . flatten . compact
remove_duplicate_results! ( reflection . class_name . constantize , parent_records , associations [ name ] ) unless parent_records . empty?
end
end
end
2008-05-03 23:49:18 -04:00
def join_for_table_name ( table_name )
2008-06-08 12:00:56 -04:00
join = ( @joins . select { | j | j . aliased_table_name == table_name . gsub ( / ^ \ "(.*) \ "$ / ) { $1 } } . first ) rescue nil
return join unless join . nil?
@joins . select { | j | j . is_a? ( JoinAssociation ) && j . aliased_join_table_name == table_name . gsub ( / ^ \ "(.*) \ "$ / ) { $1 } } . first rescue nil
2008-05-03 23:49:18 -04:00
end
def joins_for_table_name ( table_name )
join = join_for_table_name ( table_name )
result = nil
if join && join . is_a? ( JoinAssociation )
result = [ join ]
if join . parent && join . parent . is_a? ( JoinAssociation )
result = joins_for_table_name ( join . parent . aliased_table_name ) +
result
end
end
result
end
2006-03-04 18:33:10 -05:00
protected
def build ( associations , parent = nil )
parent || = @joins . last
case associations
when Symbol , String
reflection = parent . reflections [ associations . to_s . intern ] or
raise ConfigurationError , " Association named ' #{ associations } ' was not found; perhaps you misspelled it? "
@reflections << reflection
2007-11-07 10:07:39 -05:00
@joins << build_join_association ( reflection , parent )
2006-03-04 18:33:10 -05:00
when Array
associations . each do | association |
build ( association , parent )
end
when Hash
associations . keys . sort { | a , b | a . to_s < = > b . to_s } . each do | name |
build ( name , parent )
build ( associations [ name ] )
end
else
raise ConfigurationError , associations . inspect
end
end
2007-11-07 10:07:39 -05:00
# overridden in InnerJoinDependency subclass
def build_join_association ( reflection , parent )
JoinAssociation . new ( reflection , self , parent )
end
2006-03-04 18:33:10 -05:00
def construct ( parent , associations , joins , row )
case associations
when Symbol , String
2009-03-12 09:49:16 -04:00
join = joins . detect { | j | j . reflection . name . to_s == associations . to_s && j . parent_table_name == parent . class . table_name }
raise ( ConfigurationError , " No such association " ) if join . nil?
joins . delete ( join )
2006-03-04 18:33:10 -05:00
construct_association ( parent , join , row )
when Array
associations . each do | association |
construct ( parent , association , joins , row )
end
when Hash
associations . keys . sort { | a , b | a . to_s < = > b . to_s } . each do | name |
2009-03-12 09:49:16 -04:00
join = joins . detect { | j | j . reflection . name . to_s == name . to_s && j . parent_table_name == parent . class . table_name }
raise ( ConfigurationError , " No such association " ) if join . nil?
association = construct_association ( parent , join , row )
joins . delete ( join )
2006-03-04 18:33:10 -05:00
construct ( association , associations [ name ] , joins , row ) if association
end
else
raise ConfigurationError , associations . inspect
end
end
def construct_association ( record , join , row )
case join . reflection . macro
when :has_many , :has_and_belongs_to_many
collection = record . send ( join . reflection . name )
collection . loaded
2006-08-17 19:46:55 -04:00
2006-03-04 18:33:10 -05:00
return nil if record . id . to_s != join . parent . record_id ( row ) . to_s or row [ join . aliased_primary_key ] . nil?
association = join . instantiate ( row )
2007-10-28 23:02:42 -04:00
collection . target . push ( association )
2009-05-06 20:43:15 -04:00
collection . __send__ ( :set_inverse_instance , association , record )
2006-08-17 19:46:55 -04:00
when :has_one
return if record . id . to_s != join . parent . record_id ( row ) . to_s
2008-08-25 20:01:24 -04:00
return if record . instance_variable_defined? ( " @ #{ join . reflection . name } " )
2006-08-17 19:46:55 -04:00
association = join . instantiate ( row ) unless row [ join . aliased_primary_key ] . nil?
2009-05-06 20:43:15 -04:00
set_target_and_inverse ( join , association , record )
2006-08-17 19:46:55 -04:00
when :belongs_to
2006-03-04 18:33:10 -05:00
return if record . id . to_s != join . parent . record_id ( row ) . to_s or row [ join . aliased_primary_key ] . nil?
association = join . instantiate ( row )
2009-05-06 20:43:15 -04:00
set_target_and_inverse ( join , association , record )
2006-03-04 18:33:10 -05:00
else
raise ConfigurationError , " unknown macro: #{ join . reflection . macro } "
end
return association
end
2009-05-06 20:43:15 -04:00
def set_target_and_inverse ( join , association , record )
association_proxy = record . send ( " set_ #{ join . reflection . name } _target " , association )
association_proxy . __send__ ( :set_inverse_instance , association , record )
end
2006-07-04 21:06:51 -04:00
class JoinBase # :nodoc:
2006-03-19 11:13:52 -05:00
attr_reader :active_record , :table_joins
2006-03-18 18:14:31 -05:00
delegate :table_name , :column_names , :primary_key , :reflections , :sanitize_sql , :to = > :active_record
2006-03-04 18:33:10 -05:00
2006-03-19 11:13:52 -05:00
def initialize ( active_record , joins = nil )
2006-03-04 18:33:10 -05:00
@active_record = active_record
@cached_record = { }
2006-03-19 11:13:52 -05:00
@table_joins = joins
2006-03-04 18:33:10 -05:00
end
def aliased_prefix
" t0 "
end
def aliased_primary_key
2008-09-02 04:31:49 -04:00
" #{ aliased_prefix } _r0 "
2006-03-04 18:33:10 -05:00
end
def aliased_table_name
active_record . table_name
end
def column_names_with_alias
2007-12-22 06:26:03 -05:00
unless defined? ( @column_names_with_alias )
2006-03-04 18:33:10 -05:00
@column_names_with_alias = [ ]
2007-12-22 06:26:03 -05:00
2006-03-04 18:33:10 -05:00
( [ primary_key ] + ( column_names - [ primary_key ] ) ) . each_with_index do | column_name , i |
2008-09-02 04:31:49 -04:00
@column_names_with_alias << [ column_name , " #{ aliased_prefix } _r #{ i } " ]
2006-03-04 18:33:10 -05:00
end
end
2007-12-22 06:26:03 -05:00
@column_names_with_alias
2006-03-04 18:33:10 -05:00
end
def extract_record ( row )
column_names_with_alias . inject ( { } ) { | record , ( cn , an ) | record [ cn ] = row [ an ] ; record }
end
def record_id ( row )
row [ aliased_primary_key ]
end
def instantiate ( row )
2007-03-17 23:32:48 -04:00
@cached_record [ record_id ( row ) ] || = active_record . send ( :instantiate , extract_record ( row ) )
2006-03-04 18:33:10 -05:00
end
end
2006-07-04 21:06:51 -04:00
class JoinAssociation < JoinBase # :nodoc:
2006-03-18 02:31:01 -05:00
attr_reader :reflection , :parent , :aliased_table_name , :aliased_prefix , :aliased_join_table_name , :parent_table_name
2006-03-18 22:05:21 -05:00
delegate :options , :klass , :through_reflection , :source_reflection , :to = > :reflection
2006-03-04 18:33:10 -05:00
def initialize ( reflection , join_dependency , parent = nil )
2006-03-18 19:53:24 -05:00
reflection . check_validity!
if reflection . options [ :polymorphic ]
raise EagerLoadPolymorphicError . new ( reflection )
end
2006-03-04 18:33:10 -05:00
super ( reflection . klass )
2008-03-26 11:36:17 -04:00
@join_dependency = join_dependency
2006-03-04 18:33:10 -05:00
@parent = parent
@reflection = reflection
@aliased_prefix = " t #{ join_dependency . joins . size } "
2006-03-23 01:01:01 -05:00
@parent_table_name = parent . active_record . table_name
2008-03-26 11:36:17 -04:00
@aliased_table_name = aliased_table_name_for ( table_name )
2008-09-02 04:31:49 -04:00
2008-03-26 11:36:17 -04:00
if reflection . macro == :has_and_belongs_to_many
@aliased_join_table_name = aliased_table_name_for ( reflection . options [ :join_table ] , " _join " )
2006-03-05 13:43:56 -05:00
end
2008-09-02 04:31:49 -04:00
2008-08-25 18:20:10 -04:00
if [ :has_many , :has_one ] . include? ( reflection . macro ) && reflection . options [ :through ]
2008-03-26 11:36:17 -04:00
@aliased_join_table_name = aliased_table_name_for ( reflection . through_reflection . klass . table_name , " _join " )
2006-03-15 23:18:12 -05:00
end
2006-03-04 18:33:10 -05:00
end
def association_join
2007-11-10 16:33:13 -05:00
connection = reflection . active_record . connection
2006-03-05 13:43:56 -05:00
join = case reflection . macro
2006-03-04 18:33:10 -05:00
when :has_and_belongs_to_many
2007-10-29 17:39:52 -04:00
" #{ join_type } %s ON %s.%s = %s.%s " % [
2006-03-18 02:31:01 -05:00
table_alias_for ( options [ :join_table ] , aliased_join_table_name ) ,
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( aliased_join_table_name ) ,
2007-01-23 00:07:35 -05:00
options [ :foreign_key ] || reflection . active_record . to_s . foreign_key ,
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( parent . aliased_table_name ) ,
2007-12-22 06:26:03 -05:00
reflection . active_record . primary_key ] +
2007-10-29 17:39:52 -04:00
" #{ join_type } %s ON %s.%s = %s.%s " % [
2007-11-10 16:33:13 -05:00
table_name_and_alias ,
2007-12-22 06:26:03 -05:00
connection . quote_table_name ( aliased_table_name ) ,
klass . primary_key ,
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( aliased_join_table_name ) ,
2007-12-22 06:26:03 -05:00
options [ :association_foreign_key ] || klass . to_s . foreign_key
2006-03-04 18:33:10 -05:00
]
when :has_many , :has_one
2006-03-05 13:43:56 -05:00
case
2008-08-25 18:20:10 -04:00
when reflection . options [ :through ]
2006-03-18 18:14:31 -05:00
through_conditions = through_reflection . options [ :conditions ] ? " AND #{ interpolate_sql ( sanitize_sql ( through_reflection . options [ :conditions ] ) ) } " : ''
2007-12-22 06:26:03 -05:00
jt_foreign_key = jt_as_extra = jt_source_extra = jt_sti_extra = nil
first_key = second_key = as_extra = nil
2006-03-05 13:43:56 -05:00
if through_reflection . options [ :as ] # has_many :through against a polymorphic join
2007-03-13 01:23:18 -04:00
jt_foreign_key = through_reflection . options [ :as ] . to_s + '_id'
jt_as_extra = " AND %s.%s = %s " % [
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( aliased_join_table_name ) ,
connection . quote_column_name ( through_reflection . options [ :as ] . to_s + '_type' ) ,
2007-03-13 01:23:18 -04:00
klass . quote_value ( parent . active_record . base_class . name )
2006-03-05 13:43:56 -05:00
]
2006-03-19 14:32:07 -05:00
else
2007-12-22 06:26:03 -05:00
jt_foreign_key = through_reflection . primary_key_name
2007-03-13 01:23:18 -04:00
end
2007-12-22 06:26:03 -05:00
2007-03-13 01:23:18 -04:00
case source_reflection . macro
when :has_many
2007-12-22 06:26:03 -05:00
if source_reflection . options [ :as ]
first_key = " #{ source_reflection . options [ :as ] } _id "
second_key = options [ :foreign_key ] || primary_key
2007-03-13 01:23:18 -04:00
as_extra = " AND %s.%s = %s " % [
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( aliased_table_name ) ,
connection . quote_column_name ( " #{ source_reflection . options [ :as ] } _type " ) ,
2007-12-22 06:26:03 -05:00
klass . quote_value ( source_reflection . active_record . base_class . name )
2006-04-05 13:11:21 -04:00
]
else
2007-03-13 01:23:18 -04:00
first_key = through_reflection . klass . base_class . to_s . foreign_key
second_key = options [ :foreign_key ] || primary_key
end
2007-12-22 06:26:03 -05:00
2007-03-13 01:23:18 -04:00
unless through_reflection . klass . descends_from_active_record?
jt_sti_extra = " AND %s.%s = %s " % [
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( aliased_join_table_name ) ,
connection . quote_column_name ( through_reflection . active_record . inheritance_column ) ,
2008-07-21 15:21:13 -04:00
through_reflection . klass . quote_value ( through_reflection . klass . sti_name ) ]
2007-03-13 01:23:18 -04:00
end
when :belongs_to
first_key = primary_key
if reflection . options [ :source_type ]
second_key = source_reflection . association_foreign_key
jt_source_extra = " AND %s.%s = %s " % [
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( aliased_join_table_name ) ,
connection . quote_column_name ( reflection . source_reflection . options [ :foreign_type ] ) ,
2007-03-13 01:23:18 -04:00
klass . quote_value ( reflection . options [ :source_type ] )
2006-04-05 13:11:21 -04:00
]
2007-03-13 01:23:18 -04:00
else
2007-10-08 19:39:28 -04:00
second_key = source_reflection . primary_key_name
2006-03-19 14:32:07 -05:00
end
2006-03-05 13:43:56 -05:00
end
2007-03-13 01:23:18 -04:00
2007-10-29 17:39:52 -04:00
" #{ join_type } %s ON (%s.%s = %s.%s%s%s%s) " % [
2007-03-13 01:23:18 -04:00
table_alias_for ( through_reflection . klass . table_name , aliased_join_table_name ) ,
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( parent . aliased_table_name ) ,
2007-12-22 06:26:03 -05:00
connection . quote_column_name ( parent . primary_key ) ,
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( aliased_join_table_name ) ,
2007-12-22 06:26:03 -05:00
connection . quote_column_name ( jt_foreign_key ) ,
2007-03-13 01:23:18 -04:00
jt_as_extra , jt_source_extra , jt_sti_extra
] +
2007-10-29 17:39:52 -04:00
" #{ join_type } %s ON (%s.%s = %s.%s%s) " % [
2007-12-22 06:26:03 -05:00
table_name_and_alias ,
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( aliased_table_name ) ,
2007-12-22 06:26:03 -05:00
connection . quote_column_name ( first_key ) ,
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( aliased_join_table_name ) ,
2007-12-22 06:26:03 -05:00
connection . quote_column_name ( second_key ) ,
2007-03-13 01:23:18 -04:00
as_extra
]
2007-06-25 18:42:19 -04:00
when reflection . options [ :as ] && [ :has_many , :has_one ] . include? ( reflection . macro )
2007-10-29 17:39:52 -04:00
" #{ join_type } %s ON %s.%s = %s.%s AND %s.%s = %s " % [
2006-03-18 02:31:01 -05:00
table_name_and_alias ,
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( aliased_table_name ) ,
2007-12-22 06:26:03 -05:00
" #{ reflection . options [ :as ] } _id " ,
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( parent . aliased_table_name ) ,
2007-12-22 06:26:03 -05:00
parent . primary_key ,
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( aliased_table_name ) ,
2007-12-22 06:26:03 -05:00
" #{ reflection . options [ :as ] } _type " ,
2006-09-05 14:54:24 -04:00
klass . quote_value ( parent . active_record . base_class . name )
2006-03-05 13:43:56 -05:00
]
else
2006-04-05 13:01:56 -04:00
foreign_key = options [ :foreign_key ] || reflection . active_record . name . foreign_key
2007-10-29 17:39:52 -04:00
" #{ join_type } %s ON %s.%s = %s.%s " % [
2006-03-18 02:31:01 -05:00
table_name_and_alias ,
2007-11-10 16:33:13 -05:00
aliased_table_name ,
2007-12-22 06:26:03 -05:00
foreign_key ,
2007-11-10 16:33:13 -05:00
parent . aliased_table_name ,
2008-12-26 17:53:07 -05:00
reflection . options [ :primary_key ] || parent . primary_key
2006-03-05 13:43:56 -05:00
]
end
2006-03-04 18:33:10 -05:00
when :belongs_to
2007-10-29 17:39:52 -04:00
" #{ join_type } %s ON %s.%s = %s.%s " % [
2007-11-10 16:33:13 -05:00
table_name_and_alias ,
2007-12-22 06:26:03 -05:00
connection . quote_table_name ( aliased_table_name ) ,
reflection . klass . primary_key ,
2007-11-10 16:33:13 -05:00
connection . quote_table_name ( parent . aliased_table_name ) ,
2007-12-22 06:26:03 -05:00
options [ :foreign_key ] || reflection . primary_key_name
2006-03-04 18:33:10 -05:00
]
else
" "
2006-03-16 01:17:54 -05:00
end || ''
2008-08-15 19:24:29 -04:00
join << %( AND %s ) % [
klass . send ( :type_condition , aliased_table_name ) ] unless klass . descends_from_active_record?
2006-10-16 11:42:55 -04:00
[ through_reflection , reflection ] . each do | ref |
2009-04-20 08:51:11 -04:00
join << " AND #{ interpolate_sql ( sanitize_sql ( ref . options [ :conditions ] , aliased_table_name ) ) } " if ref && ref . options [ :conditions ]
2006-10-16 11:42:55 -04:00
end
2006-03-05 13:43:56 -05:00
join
2006-03-04 18:33:10 -05:00
end
2007-03-13 01:23:18 -04:00
2007-12-22 06:26:03 -05:00
protected
2008-09-02 04:31:49 -04:00
2008-03-26 11:36:17 -04:00
def aliased_table_name_for ( name , suffix = nil )
2008-12-20 15:52:48 -05:00
if ! parent . table_joins . blank? && parent . table_joins . to_s . downcase =~ %r{ join( \ s+ \ w+)? \ s+ #{ active_record . connection . quote_table_name name . downcase } \ son }
2008-03-26 11:36:17 -04:00
@join_dependency . table_aliases [ name ] += 1
end
unless @join_dependency . table_aliases [ name ] . zero?
# if the table name has been used, then use an alias
name = active_record . connection . table_alias_for " #{ pluralize ( reflection . name ) } _ #{ parent_table_name } #{ suffix } "
table_index = @join_dependency . table_aliases [ name ]
@join_dependency . table_aliases [ name ] += 1
name = name [ 0 .. active_record . connection . table_alias_length - 3 ] + " _ #{ table_index + 1 } " if table_index > 0
else
@join_dependency . table_aliases [ name ] += 1
end
name
end
2008-09-02 04:31:49 -04:00
2006-03-18 02:31:01 -05:00
def pluralize ( table_name )
ActiveRecord :: Base . pluralize_table_names ? table_name . to_s . pluralize : table_name
end
2007-12-22 06:26:03 -05:00
2006-03-18 02:31:01 -05:00
def table_alias_for ( table_name , table_alias )
2007-12-22 06:26:03 -05:00
" #{ reflection . active_record . connection . quote_table_name ( table_name ) } #{ table_alias if table_name != table_alias } " . strip
2006-03-18 02:31:01 -05:00
end
def table_name_and_alias
table_alias_for table_name , @aliased_table_name
end
2006-10-11 10:02:24 -04:00
def interpolate_sql ( sql )
2007-12-22 06:26:03 -05:00
instance_eval ( " %@ #{ sql . gsub ( '@' , '\@' ) } @ " )
end
2007-11-07 10:07:39 -05:00
private
2007-10-29 17:39:52 -04:00
def join_type
" LEFT OUTER JOIN "
end
end
2007-11-07 10:07:39 -05:00
end
class InnerJoinDependency < JoinDependency # :nodoc:
protected
def build_join_association ( reflection , parent )
InnerJoinAssociation . new ( reflection , self , parent )
end
class InnerJoinAssociation < JoinAssociation
2007-10-29 17:39:52 -04:00
private
def join_type
" INNER JOIN "
end
2006-03-04 18:33:10 -05:00
end
end
2007-11-07 10:07:39 -05:00
2005-11-06 14:05:42 -05:00
end
2004-11-23 20:04:44 -05:00
end
2005-01-15 12:45:16 -05:00
end