# frozen_string_literal: true
module ActiveRecord
module Associations
# = Active Record Associations
#
# This is the root class of all associations ('+ Foo' signifies an included module Foo):
#
# Association
# SingularAssociation
# HasOneAssociation + ForeignAssociation
# HasOneThroughAssociation + ThroughAssociation
# BelongsToAssociation
# BelongsToPolymorphicAssociation
# CollectionAssociation
# HasManyAssociation + ForeignAssociation
# HasManyThroughAssociation + ThroughAssociation
#
# Associations in Active Record are middlemen between the object that
# holds the association, known as the owner, and the associated
# result set, known as the target. Association metadata is available in
# reflection, which is an instance of +ActiveRecord::Reflection::AssociationReflection+.
#
# For example, given
#
# class Blog < ActiveRecord::Base
# has_many :posts
# end
#
# blog = Blog.first
#
# The association of blog.posts has the object +blog+ as its
# owner, the collection of its posts as target, and
# the reflection object represents a :has_many macro.
class Association # :nodoc:
attr_accessor :owner
attr_reader :target, :reflection, :disable_joins
delegate :options, to: :reflection
def initialize(owner, reflection)
reflection.check_validity!
@owner, @reflection = owner, reflection
@disable_joins = @reflection.options[:disable_joins] || false
reset
reset_scope
@skip_strict_loading = nil
end
# Resets the \loaded flag to +false+ and sets the \target to +nil+.
def reset
@loaded = false
@target = nil
@stale_state = nil
end
def reset_negative_cache # :nodoc:
reset if loaded? && target.nil?
end
# Reloads the \target and returns +self+ on success.
# The QueryCache is cleared if +force+ is true.
def reload(force = false)
klass.connection.clear_query_cache if force && klass
reset
reset_scope
load_target
self unless target.nil?
end
# Has the \target been already \loaded?
def loaded?
@loaded
end
# Asserts the \target has been loaded setting the \loaded flag to +true+.
def loaded!
@loaded = true
@stale_state = stale_state
end
# The target is stale if the target no longer points to the record(s) that the
# relevant foreign_key(s) refers to. If stale, the association accessor method
# on the owner will reload the target. It's up to subclasses to implement the
# stale_state method if relevant.
#
# Note that if the target has not been loaded, it is not considered stale.
def stale_target?
loaded? && @stale_state != stale_state
end
# Sets the target of this association to \target, and the \loaded flag to +true+.
def target=(target)
@target = target
loaded!
end
def scope
if disable_joins
DisableJoinsAssociationScope.create.scope(self)
elsif (scope = klass.current_scope) && scope.try(:proxy_association) == self
scope.spawn
elsif scope = klass.global_current_scope
target_scope.merge!(association_scope).merge!(scope)
else
target_scope.merge!(association_scope)
end
end
def reset_scope
@association_scope = nil
end
# Set the inverse association, if possible
def set_inverse_instance(record)
if inverse = inverse_association_for(record)
inverse.inversed_from(owner)
end
record
end
def set_inverse_instance_from_queries(record)
if inverse = inverse_association_for(record)
inverse.inversed_from_queries(owner)
end
record
end
# Remove the inverse association, if possible
def remove_inverse_instance(record)
if inverse = inverse_association_for(record)
inverse.inversed_from(nil)
end
end
def inversed_from(record)
self.target = record
end
def inversed_from_queries(record)
if inversable?(record)
self.target = record
end
end
# Returns the class of the target. belongs_to polymorphic overrides this to look at the
# polymorphic_type field on the owner.
def klass
reflection.klass
end
def extensions
extensions = klass.default_extensions | reflection.extensions
if reflection.scope
extensions |= reflection.scope_for(klass.unscoped, owner).extensions
end
extensions
end
# Loads the \target if needed and returns it.
#
# This method is abstract in the sense that it relies on +find_target+,
# which is expected to be provided by descendants.
#
# If the \target is already \loaded it is just returned. Thus, you can call
# +load_target+ unconditionally to get the \target.
#
# ActiveRecord::RecordNotFound is rescued within the method, and it is
# not reraised. The proxy is \reset and +nil+ is the return value.
def load_target
@target = find_target if (@stale_state && stale_target?) || find_target?
loaded! unless loaded?
target
rescue ActiveRecord::RecordNotFound
reset
end
# We can't dump @reflection and @through_reflection since it contains the scope proc
def marshal_dump
ivars = (instance_variables - [:@reflection, :@through_reflection]).map { |name| [name, instance_variable_get(name)] }
[@reflection.name, ivars]
end
def marshal_load(data)
reflection_name, ivars = data
ivars.each { |name, val| instance_variable_set(name, val) }
@reflection = @owner.class._reflect_on_association(reflection_name)
end
def initialize_attributes(record, except_from_scope_attributes = nil) # :nodoc:
except_from_scope_attributes ||= {}
skip_assign = [reflection.foreign_key, reflection.type].compact
assigned_keys = record.changed_attribute_names_to_save
assigned_keys += except_from_scope_attributes.keys.map(&:to_s)
attributes = scope_for_create.except!(*(assigned_keys - skip_assign))
record.send(:_assign_attributes, attributes) if attributes.any?
set_inverse_instance(record)
end
def create(attributes = nil, &block)
_create_record(attributes, &block)
end
def create!(attributes = nil, &block)
_create_record(attributes, true, &block)
end
private
# Reader and writer methods call this so that consistent errors are presented
# when the association target class does not exist.
def ensure_klass_exists!
klass
end
def find_target
if violates_strict_loading?
Base.strict_loading_violation!(owner: owner.class, reflection: reflection)
end
scope = self.scope
return scope.to_a if skip_statement_cache?(scope)
sc = reflection.association_scope_cache(klass, owner) do |params|
as = AssociationScope.create { params.bind }
target_scope.merge!(as.scope(self))
end
binds = AssociationScope.get_bind_values(owner, reflection.chain)
sc.execute(binds, klass.connection) do |record|
set_inverse_instance(record)
if owner.strict_loading_n_plus_one_only? && reflection.macro == :has_many
record.strict_loading!
else
record.strict_loading!(false, mode: owner.strict_loading_mode)
end
end
end
def skip_strict_loading(&block)
skip_strict_loading_was = @skip_strict_loading
@skip_strict_loading = true
yield
ensure
@skip_strict_loading = skip_strict_loading_was
end
def violates_strict_loading?
return if @skip_strict_loading
return unless owner.validation_context.nil?
return reflection.strict_loading? if reflection.options.key?(:strict_loading)
owner.strict_loading? && !owner.strict_loading_n_plus_one_only?
end
# The scope for this association.
#
# Note that the association_scope is merged into the target_scope only when the
# scope method is called. This is because at that point the call may be surrounded
# by scope.scoping { ... } or unscoped { ... } etc, which affects the scope which
# actually gets built.
def association_scope
if klass
@association_scope ||= if disable_joins
DisableJoinsAssociationScope.scope(self)
else
AssociationScope.scope(self)
end
end
end
# Can be overridden (i.e. in ThroughAssociation) to merge in other scopes (i.e. the
# through association's scope)
def target_scope
AssociationRelation.create(klass, self).merge!(klass.scope_for_association)
end
def scope_for_create
scope.scope_for_create
end
def find_target?
!loaded? && (!owner.new_record? || foreign_key_present?) && klass
end
# Returns true if there is a foreign key present on the owner which
# references the target. This is used to determine whether we can load
# the target if the owner is currently a new record (and therefore
# without a key). If the owner is a new record then foreign_key must
# be present in order to load target.
#
# Currently implemented by belongs_to (vanilla and polymorphic) and
# has_one/has_many :through associations which go through a belongs_to.
def foreign_key_present?
false
end
# Raises ActiveRecord::AssociationTypeMismatch unless +record+ is of
# the kind of the class of the associated objects. Meant to be used as
# a safety check when you are about to assign an associated record.
def raise_on_type_mismatch!(record)
unless record.is_a?(reflection.klass)
fresh_class = reflection.class_name.safe_constantize
unless fresh_class && record.is_a?(fresh_class)
message = "#{reflection.class_name}(##{reflection.klass.object_id}) expected, "\
"got #{record.inspect} which is an instance of #{record.class}(##{record.class.object_id})"
raise ActiveRecord::AssociationTypeMismatch, message
end
end
end
def inverse_association_for(record)
if invertible_for?(record)
record.association(inverse_reflection_for(record).name)
end
end
# Can be redefined by subclasses, notably polymorphic belongs_to
# The record parameter is necessary to support polymorphic inverses as we must check for
# the association in the specific class of the record.
def inverse_reflection_for(record)
reflection.inverse_of
end
# Returns true if inverse association on the given record needs to be set.
# This method is redefined by subclasses.
def invertible_for?(record)
foreign_key_for?(record) && inverse_reflection_for(record)
end
# Returns true if record contains the foreign_key
def foreign_key_for?(record)
foreign_key = Array(reflection.foreign_key)
foreign_key.all? { |key| record._has_attribute?(key) }
end
# This should be implemented to return the values of the relevant key(s) on the owner,
# so that when stale_state is different from the value stored on the last find_target,
# the target is stale.
#
# This is only relevant to certain associations, which is why it returns +nil+ by default.
def stale_state
end
def build_record(attributes)
reflection.build_association(attributes) do |record|
initialize_attributes(record, attributes)
yield(record) if block_given?
end
end
# Returns true if statement cache should be skipped on the association reader.
def skip_statement_cache?(scope)
reflection.has_scope? ||
scope.eager_loading? ||
klass.scope_attributes? ||
reflection.source_reflection.active_record.default_scopes.any?
end
def enqueue_destroy_association(options)
job_class = owner.class.destroy_association_async_job
if job_class
owner._after_commit_jobs.push([job_class, options])
end
end
def inversable?(record)
record &&
((!record.persisted? || !owner.persisted?) || matches_foreign_key?(record))
end
def matches_foreign_key?(record)
if foreign_key_for?(record)
record.read_attribute(reflection.foreign_key) == owner.id ||
(foreign_key_for?(owner) && owner.read_attribute(reflection.foreign_key) == record.id)
else
owner.read_attribute(reflection.foreign_key) == record.id
end
end
end
end
end