class ActiveRecord::Associations::HasManyThroughAssociation
:nodoc:
= Active Record Has Many Through Association
def build_record(attributes)
def build_record(attributes) ensure_not_nested @through_scope = scope record = super inverse = if source_reflection.polymorphic? source_reflection.polymorphic_inverse_of(record.class) else source_reflection.inverse_of end if inverse if inverse.collection? record.send(inverse.name) << build_through_record(record) elsif inverse.has_one? record.send("#{inverse.name}=", build_through_record(record)) end end record ensure @through_scope = nil end
def build_through_record(record)
However, after insert_record has been called, the cache is cleared in
so that it may be reused if insert_record is subsequently called.
The through record (built with build_record) is temporarily cached
def build_through_record(record) @through_records[record] ||= begin ensure_mutable attributes = through_scope_attributes attributes[source_reflection.name] = record through_association.build(attributes).tap do |new_record| new_record.send("#{source_reflection.foreign_type}=", options[:source_type]) if options[:source_type] end end end
def concat(*records)
def concat(*records) unless owner.new_record? records.flatten.each do |record| raise_on_type_mismatch!(record) end end super end
def concat_records(records)
def concat_records(records) ensure_not_nested records = super(records, true) if owner.new_record? && records records.flatten.each do |record| build_through_record(record) end end records end
def delete_or_nullify_all_records(method)
def delete_or_nullify_all_records(method) delete_records(load_target, method) end
def delete_records(records, method)
def delete_records(records, method) ensure_not_nested scope = through_association.scope scope.where! construct_join_attributes(*records) scope = scope.where(through_scope_attributes) case method when :destroy if scope.model.primary_key count = scope.destroy_all.count(&:destroyed?) else scope.each(&:_run_destroy_callbacks) count = scope.delete_all end when :nullify count = scope.update_all(source_reflection.foreign_key => nil) else count = scope.delete_all end delete_through_records(records) if source_reflection.options[:counter_cache] && method != :destroy counter = source_reflection.counter_cache_column klass.decrement_counter counter, records.map(&:id) end if through_reflection.collection? && update_through_counter?(method) update_counter(-count, through_reflection) else update_counter(-count) end count end
def delete_through_records(records)
def delete_through_records(records) records.each do |record| through_records = through_records_for(record) if through_reflection.collection? through_records.each { |r| through_association.target.delete(r) } else if through_records.include?(through_association.target) through_association.target = nil end end @through_records.delete(record) end end
def difference(a, b)
def difference(a, b) distribution = distribution(b) a.reject { |record| mark_occurrence(distribution, record) } end
def distribution(array)
def distribution(array) array.each_with_object(Hash.new(0)) do |record, distribution| distribution[record] += 1 end end
def find_target(async: false)
def find_target(async: false) raise NotImplementedError, "No async loading for HasManyThroughAssociation yet" if async return [] unless target_reflection_has_associated_record? return scope.to_a if disable_joins super end
def initialize(owner, reflection)
def initialize(owner, reflection) super @through_records = {}.compare_by_identity end
def insert_record(record, validate = true, raise = false)
def insert_record(record, validate = true, raise = false) ensure_not_nested if record.new_record? || record.has_changes_to_save? return unless super end save_through_record(record) record end
def intersection(a, b)
def intersection(a, b) distribution = distribution(b) a.select { |record| mark_occurrence(distribution, record) } end
def invertible_for?(record)
def invertible_for?(record) false end
def mark_occurrence(distribution, record)
def mark_occurrence(distribution, record) distribution[record] > 0 && distribution[record] -= 1 end
def remove_records(existing_records, records, method)
def remove_records(existing_records, records, method) super delete_through_records(records) end
def save_through_record(record)
def save_through_record(record) association = build_through_record(record) if association.changed? association.save! end ensure @through_records.delete(record) end
def target_reflection_has_associated_record?
def target_reflection_has_associated_record? !(through_reflection.belongs_to? && Array(through_reflection.foreign_key).all? { |foreign_key_column| owner[foreign_key_column].blank? }) end
def through_records_for(record)
def through_records_for(record) attributes = construct_join_attributes(record) candidates = Array.wrap(through_association.target) candidates.find_all do |c| attributes.all? do |key, value| c.public_send(key) == value end end end
def through_scope_attributes
def through_scope_attributes scope = through_scope || self.scope attributes = scope.where_values_hash(through_association.reflection.klass.table_name) except_keys = [ *Array(through_association.reflection.foreign_key), through_association.reflection.klass.inheritance_column ] attributes.except!(*except_keys) end
def update_through_counter?(method)
def update_through_counter?(method) case method when :destroy !through_reflection.inverse_updates_counter_cache? when :nullify false else true end end