association.rb 4.0 KB
Newer Older
1 2
# frozen_string_literal: true

J
Jon Leighton 已提交
3 4 5 6
module ActiveRecord
  module Associations
    class Preloader
      class Association #:nodoc:
7 8 9 10 11 12
        def initialize(klass, owners, reflection, preload_scope)
          @klass         = klass
          @owners        = owners
          @reflection    = reflection
          @preload_scope = preload_scope
          @model         = owners.first && owners.first.class
J
Jon Leighton 已提交
13 14
        end

15 16 17 18 19 20 21 22 23 24
        def run
          if !preload_scope || preload_scope.empty_scope?
            owners.each do |owner|
              associate_records_to_owner(owner, records_by_owner[owner] || [])
            end
          else
            # Custom preload scope is used and
            # the association can not be marked as loaded
            # Loading into a Hash instead
            records_by_owner
25
          end
26 27
          self
        end
28

29 30 31 32 33 34 35 36 37 38
        def records_by_owner
          @records_by_owner ||= preloaded_records.each_with_object({}) do |record, result|
            owners_by_key[convert_key(record[association_key_name])].each do |owner|
              (result[owner] ||= []) << record
            end
          end
        end

        def preloaded_records
          return @preloaded_records if defined?(@preloaded_records)
39
          @preloaded_records = owner_keys.empty? ? [] : records_for(owner_keys)
J
Jon Leighton 已提交
40 41
        end

R
Ryuta Kamizono 已提交
42
        private
43 44
          attr_reader :owners, :reflection, :preload_scope, :model, :klass

45 46 47 48 49
          # The name of the key on the associated records
          def association_key_name
            reflection.join_primary_key(klass)
          end

50 51 52 53 54
          # The name of the key on the model which declares the association
          def owner_key_name
            reflection.join_foreign_key
          end

55
          def associate_records_to_owner(owner, records)
R
Ryuta Kamizono 已提交
56 57
            association = owner.association(reflection.name)
            if reflection.collection?
58
              association.target = records
R
Ryuta Kamizono 已提交
59
            else
60
              association.target = records.first
R
Ryuta Kamizono 已提交
61
            end
62 63
          end

64
          def owner_keys
65
            @owner_keys ||= owners_by_key.keys
J
Jon Leighton 已提交
66 67
          end

68
          def owners_by_key
69 70 71
            @owners_by_key ||= owners.each_with_object({}) do |owner, result|
              key = convert_key(owner[owner_key_name])
              (result[key] ||= []) << owner if key
72 73 74
            end
          end

75
          def key_conversion_required?
76 77 78 79 80
            unless defined?(@key_conversion_required)
              @key_conversion_required = (association_key_type != owner_key_type)
            end

            @key_conversion_required
81
          end
82

83 84 85 86 87 88
          def convert_key(key)
            if key_conversion_required?
              key.to_s
            else
              key
            end
89
          end
90

91
          def association_key_type
D
Daniel Colson 已提交
92
            @klass.type_for_attribute(association_key_name).type
93
          end
94

95
          def owner_key_type
D
Daniel Colson 已提交
96
            @model.type_for_attribute(owner_key_name).type
97
          end
98

99 100 101 102 103 104 105
          def records_for(ids)
            scope.where(association_key_name => ids).load do |record|
              # Processing only the first owner
              # because the record is modified but not an owner
              owner = owners_by_key[convert_key(record[association_key_name])].first
              association = owner.association(reflection.name)
              association.set_inverse_instance(record)
106
            end
107
          end
108

109 110 111 112
          def scope
            @scope ||= build_scope
          end

113
          def reflection_scope
114
            @reflection_scope ||= reflection.scope ? reflection.scope_for(klass.unscoped) : klass.unscoped
115
          end
116

117
          def build_scope
118
            scope = klass.scope_for_association
119

120
            if reflection.type && !reflection.through_reflection?
121
              scope.where!(reflection.type => model.polymorphic_name)
122
            end
123

124
            scope.merge!(reflection_scope) if reflection.scope
125 126
            scope.merge!(preload_scope) if preload_scope
            scope
J
Jon Leighton 已提交
127 128 129 130 131
          end
      end
    end
  end
end