123456789_123456789_123456789_123456789_123456789_

Class: ActiveRecord::Reflection::AssociationReflection

Do not use. This class is for internal use only.
Relationships & Source Files
Extension / Inclusion / Inheritance Descendants
Subclasses:
Super Chains via Extension / Inclusion / Inheritance
Class Chain:
Instance Chain:
Inherits: ActiveRecord::Reflection::MacroReflection
Defined in: activerecord/lib/active_record/reflection.rb

Overview

Holds all the metadata about an association as it was specified in the Active Record class.

Class Method Summary

Instance Attribute Summary

MacroReflection - Inherited

#active_record, #autosave=,
#name

Returns the name of the macro.

#options

Returns the hash of options used for the macro.

#scope, #plural_name

AbstractReflection - Inherited

Instance Method Summary

MacroReflection - Inherited

#==

Returns true if self and other_aggregation have the same name attribute, active_record attribute, and other_aggregation has an options hash assigned to it.

#compute_class,
#klass

Returns the class for the macro.

#scope_for, #derive_class_name, #validate_reflection!

AbstractReflection - Inherited

#alias_candidate,
#build_association

Returns a new, unsaved instance of the associated class.

#build_scope, #chain, #check_validity_of_inverse!,
#class_name

Returns the class name for the macro.

#constraints, #counter_cache_column, #inverse_of,
#inverse_which_updates_counter_cache

We need to avoid the following situation:

#join_scope, #join_scopes, #klass_join_scope,
#scopes

Returns a list of scopes that should be applied for this ::ActiveRecord::Reflection object when querying the database.

#strict_loading_violation_message, #table_name,
#actual_source_reflection

FIXME: this is a horrible name.

#ensure_option_not_given_as_class!, #predicate_builder, #primary_key

Constructor Details

.new(name, scope, options, active_record) ⇒ AssociationReflection

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 489

def initialize(name, scope, options, active_record)
  super
  @type = -(options[:foreign_type]&.to_s || "#{options[:as]}_type") if options[:as]
  @foreign_type = -(options[:foreign_type]&.to_s || "#{name}_type") if options[:polymorphic]
  @join_table = nil
  @foreign_key = nil
  @association_foreign_key = nil
  @association_primary_key = nil

  ensure_option_not_given_as_class!(:class_name)
end

Instance Attribute Details

#belongs_to?Boolean (readonly)

Returns true if self is a belongs_to reflection.

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 664

def belongs_to?; false; end

#collection?Boolean (readonly)

Returns whether or not this association reflection is for a collection association. Returns true if the #macro is either has_many or has_and_belongs_to_many, false otherwise.

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 646

def collection?
  false
end

#foreign_type (readonly)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 486

attr_reader :type, :foreign_type

#has_inverse?Boolean (readonly)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 624

def has_inverse?
  inverse_name
end

#has_one?Boolean (readonly)

Returns true if self is a has_one reflection.

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 667

def has_one?; false; end

#has_scope?Boolean (readonly)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 620

def has_scope?
  scope
end

#nested?Boolean (readonly)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 616

def nested?
  false
end

#parent_reflection (rw)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 487

attr_accessor :parent_reflection # Reflection

#polymorphic?Boolean (readonly)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 671

def polymorphic?
  options[:polymorphic]
end

#type (readonly)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 486

attr_reader :type, :foreign_type

#validate?Boolean (readonly)

Returns whether or not the association should be validated as part of the parent’s validation.

Unless you explicitly disable validation with validate: false, validation will take place when:

  • you explicitly enable validation; validate: true

  • you use autosave; autosave: true

  • the association is a has_many association

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 659

def validate?
  !options[:validate].nil? ? options[:validate] : (options[:autosave] == true || collection?)
end

Instance Method Details

#active_record_primary_key

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 537

def active_record_primary_key
  custom_primary_key = options[:primary_key]
  @active_record_primary_key ||= if custom_primary_key
    if custom_primary_key.is_a?(Array)
      custom_primary_key.map { |pk| pk.to_s.freeze }.freeze
    else
      custom_primary_key.to_s.freeze
    end
  elsif active_record.has_query_constraints? || options[:query_constraints]
    active_record.query_constraints_list
  elsif active_record.composite_primary_key?
    # If active_record has composite primary key of shape [:<tenant_key>, :id], infer primary_key as :id
    primary_key = primary_key(active_record)
    primary_key.include?("id") ? "id" : primary_key.freeze
  else
    primary_key(active_record).freeze
  end
end

#add_as_polymorphic_through(reflection, seed)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 683

def add_as_polymorphic_through(reflection, seed)
  seed + [PolymorphicReflection.new(self, reflection)]
end

#add_as_source(seed)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 679

def add_as_source(seed)
  seed
end

#add_as_through(seed)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 687

def add_as_through(seed)
  seed + [self]
end

#association_class

Raises:

  • (NotImplementedError)
[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 669

def association_class; raise NotImplementedError; end

#association_foreign_key

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 529

def association_foreign_key
  @association_foreign_key ||= -(options[:association_foreign_key]&.to_s || class_name.foreign_key)
end

#association_primary_key(klass = nil)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 533

def association_primary_key(klass = nil)
  primary_key(klass || self.klass)
end

#association_scope_cache(klass, owner, &block)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 501

def association_scope_cache(klass, owner, &block)
  key = self
  if polymorphic?
    key = [key, owner._read_attribute(@foreign_type)]
  end
  klass.cached_find_by_statement(key, &block)
end

#automatic_inverse_of (private)

returns either nil or the inverse association name that it finds.

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 708

def automatic_inverse_of
  if can_find_inverse_of_automatically?(self)
    inverse_name = ActiveSupport::Inflector.underscore(options[:as] || active_record.name.demodulize).to_sym

    begin
      reflection = klass._reflect_on_association(inverse_name)
    rescue NameError => error
      raise unless error.name.to_s == class_name

      # Give up: we couldn't compute the klass type so we won't be able
      # to find any associations either.
      reflection = false
    end

    if valid_inverse_reflection?(reflection)
      inverse_name
    end
  end
end

#can_find_inverse_of_automatically?(reflection, inverse_reflection = false) ⇒ Boolean (private)

Checks to see if the reflection doesn’t have any options that prevent us from being able to guess the inverse automatically. First, the inverse_of option cannot be set to false. Second, we must have has_many, has_one, belongs_to associations. Third, we must not have options such as :foreign_key which prevent us from correctly guessing the inverse association.

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 746

def can_find_inverse_of_automatically?(reflection, inverse_reflection = false)
  reflection.options[:inverse_of] != false &&
    !reflection.options[:through] &&
    !reflection.options[:foreign_key] &&
    scope_allows_automatic_inverse_of?(reflection, inverse_reflection)
end

#check_eager_loadable!

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 580

def check_eager_loadable!
  return unless scope

  unless scope.arity == 0
    raise ArgumentError, <<-MSG.squish
      The association scope '#{name}' is instance dependent (the scope
      block takes an argument). Eager loading instance dependent scopes
      is not supported.
    MSG
  end
end

#check_validity!

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 568

def check_validity!
  check_validity_of_inverse!

  if !polymorphic? && (klass.composite_primary_key? || active_record.composite_primary_key?)
    if (has_one? || collection?) && Array(active_record_primary_key).length != Array(foreign_key).length
      raise CompositePrimaryKeyMismatchError.new(self)
    elsif belongs_to? && Array(association_primary_key).length != Array(foreign_key).length
      raise CompositePrimaryKeyMismatchError.new(self)
    end
  end
end

#clear_association_scope_cache

This is for clearing cache on the reflection. Useful for tests that need to compare SQL queries on associations.

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 612

def clear_association_scope_cache # :nodoc:
  klass.initialize_find_by_cache
end

#collect_join_chain

A chain of reflections from this one back to the owner. For more see the explanation in ThroughReflection.

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 606

def collect_join_chain
  [self]
end

#compute_class(name)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 462

def compute_class(name)
  if polymorphic?
    raise ArgumentError, "Polymorphic associations do not support computing the class."
  end

  begin
    klass = active_record.send(:compute_type, name)
  rescue NameError => error
    if error.name.match?(/(?:\A|::)#{name}\z/)
      message = "Missing model class #{name} for the #{active_record}##{self.name} association."
      message += " You can specify a different model class with the :class_name option." unless options[:class_name]
      raise NameError.new(message, name)
    else
      raise
    end
  end

  unless klass < ActiveRecord::Base
    raise ArgumentError, "The #{name} model class for the #{active_record}##{self.name} association is not an ActiveRecord::Base subclass."
  end

  klass
end

#derive_class_name (private)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 767

def derive_class_name
  class_name = name.to_s
  class_name = class_name.singularize if collection?
  class_name.camelize
end

#derive_fk_query_constraints(foreign_key) (private)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 785

def derive_fk_query_constraints(foreign_key)
  primary_query_constraints = active_record.query_constraints_list
  owner_pk = active_record.primary_key

  if primary_query_constraints.size > 2
    raise ArgumentError, <<~MSG.squish
      The query constraints list on the `#{active_record}` model has more than 2
      attributes. Active Record is unable to derive the query constraints
      for the association. You need to explicitly define the query constraints
      for this association.
    MSG
  end

  if !primary_query_constraints.include?(owner_pk)
    raise ArgumentError, <<~MSG.squish
      The query constraints on the `#{active_record}` model does not include the primary
      key so Active Record is unable to derive the foreign key constraints for
      the association. You need to explicitly define the query constraints for this
      association.
    MSG
  end

  return foreign_key if primary_query_constraints.include?(foreign_key)

  first_key, last_key = primary_query_constraints

  if first_key == owner_pk
    [foreign_key, last_key.to_s]
  elsif last_key == owner_pk
    [first_key.to_s, foreign_key]
  else
    raise ArgumentError, <<~MSG.squish
      Active Record couldn't correctly interpret the query constraints
      for the `#{active_record}` model. The query constraints on `#{active_record}` are
      `#{primary_query_constraints}` and the foreign key is `#{foreign_key}`.
      You need to explicitly set the query constraints for this association.
    MSG
  end
end

#derive_foreign_key(infer_from_inverse_of: true) (private)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 773

def derive_foreign_key(infer_from_inverse_of: true)
  if belongs_to?
    "#{name}_id"
  elsif options[:as]
    "#{options[:as]}_id"
  elsif options[:inverse_of] && infer_from_inverse_of
    inverse_of.foreign_key(infer_from_inverse_of: false)
  else
    active_record.model_name.to_s.foreign_key
  end
end

#derive_join_table (private)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 825

def derive_join_table
  ModelSchema.derive_join_table_name active_record.table_name, klass.table_name
end

#extensions

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 691

def extensions
  Array(options[:extend])
end

#foreign_key(infer_from_inverse_of: true)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 513

def foreign_key(infer_from_inverse_of: true)
  @foreign_key ||= if options[:query_constraints]
    options[:query_constraints].map { |fk| fk.to_s.freeze }.freeze
  elsif options[:foreign_key]
    options[:foreign_key].to_s
  else
    derived_fk = derive_foreign_key(infer_from_inverse_of: infer_from_inverse_of)

    if active_record.has_query_constraints?
      derived_fk = derive_fk_query_constraints(derived_fk)
    end

    derived_fk
  end
end

#inverse_name (private)

Attempts to find the inverse association name automatically. If it cannot find a suitable inverse association name, it returns nil.

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 699

def inverse_name
  unless defined?(@inverse_name)
    @inverse_name = options.fetch(:inverse_of) { automatic_inverse_of }
  end

  @inverse_name
end

#join_foreign_key

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 564

def join_foreign_key
  active_record_primary_key
end

#join_id_for(owner)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 592

def join_id_for(owner) # :nodoc:
  Array(join_foreign_key).map { |key| owner._read_attribute(key) }
end

#join_primary_key(klass = nil)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 556

def join_primary_key(klass = nil)
  foreign_key
end

#join_primary_type

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 560

def join_primary_type
  type
end

#join_table

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 509

def join_table
  @join_table ||= -(options[:join_table]&.to_s || derive_join_table)
end

#macro

Returns the macro type.

has_many :clients returns :has_many

Raises:

  • (NotImplementedError)
[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 641

def macro; raise NotImplementedError; end

#polymorphic_inverse_of(associated_class)

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 628

def polymorphic_inverse_of(associated_class)
  if has_inverse?
    if inverse_relationship = associated_class._reflect_on_association(options[:inverse_of])
      inverse_relationship
    else
      raise InverseOfAssociationNotFoundError.new(self, associated_class)
    end
  end
end

#polymorphic_name

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 675

def polymorphic_name
  active_record.polymorphic_name
end

#scope_allows_automatic_inverse_of?(reflection, inverse_reflection) ⇒ Boolean (private)

Scopes on the potential inverse reflection prevent automatic inverse_of, since the scope could exclude the owner record we would inverse from. Scopes on the reflection itself allow for automatic inverse_of as long as <tt>config.active_record.automatic_scope_inversing<tt> is set to true (the default for new applications).

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 759

def scope_allows_automatic_inverse_of?(reflection, inverse_reflection)
  if inverse_reflection
    !reflection.scope
  else
    !reflection.scope || reflection.klass.automatic_scope_inversing
  end
end

#source_reflection

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 600

def source_reflection
  self
end

#through_reflection

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 596

def through_reflection
  nil
end

#valid_inverse_reflection?(reflection) ⇒ Boolean (private)

Checks if the inverse reflection that is returned from the #automatic_inverse_of method is a valid reflection. We must make sure that the reflection’s active_record name matches up with the current reflection’s klass name.

[ GitHub ]

  
# File 'activerecord/lib/active_record/reflection.rb', line 732

def valid_inverse_reflection?(reflection)
  reflection &&
    reflection != self &&
    foreign_key == reflection.foreign_key &&
    klass <= reflection.active_record &&
    can_find_inverse_of_automatically?(reflection, true)
end