diff options
author | Aaron Patterson <aaron.patterson@gmail.com> | 2013-09-20 11:31:00 -0700 |
---|---|---|
committer | Aaron Patterson <aaron.patterson@gmail.com> | 2013-09-20 11:31:00 -0700 |
commit | 5751689a68a122a1ef7892289c8683aeda4b2482 (patch) | |
tree | 59b6beb27ff8f1c922d753191f5380efbc7239b6 | |
parent | 1ea4a892a835ada735ba2943b20bdbfd39120e8c (diff) | |
download | rails-5751689a68a122a1ef7892289c8683aeda4b2482.tar.gz rails-5751689a68a122a1ef7892289c8683aeda4b2482.tar.bz2 rails-5751689a68a122a1ef7892289c8683aeda4b2482.zip |
exposing target records on the preloader
-rw-r--r-- | activerecord/lib/active_record/associations/preloader/association.rb | 41 | ||||
-rw-r--r-- | activerecord/lib/active_record/associations/preloader/has_and_belongs_to_many.rb | 4 |
2 files changed, 29 insertions, 16 deletions
diff --git a/activerecord/lib/active_record/associations/preloader/association.rb b/activerecord/lib/active_record/associations/preloader/association.rb index 928da71eed..051bc3a8da 100644 --- a/activerecord/lib/active_record/associations/preloader/association.rb +++ b/activerecord/lib/active_record/associations/preloader/association.rb @@ -3,6 +3,9 @@ module ActiveRecord class Preloader class Association #:nodoc: attr_reader :owners, :reflection, :preload_scope, :model, :klass + attr_reader :type_caster + + IDENTITY_CASTER = lambda { |value| value } # :nodoc: def initialize(klass, owners, reflection, preload_scope) @klass = klass @@ -12,6 +15,8 @@ module ActiveRecord @model = owners.first && owners.first.class @scope = nil @owners_by_key = nil + @type_caster = IDENTITY_CASTER + @target_records = nil end def run @@ -66,6 +71,20 @@ module ActiveRecord reflection.options end + def target_records + return @target_records if @target_records + + owners_map = owners_by_key + owner_keys = owners_map.keys.compact + + sliced = owner_keys.each_slice(klass.connection.in_clause_length || owner_keys.size) + @target_records = sliced.flat_map { |slice| + records = records_for(slice) + set_type_caster records, association_key_name + records + } + end + private def associated_records_by_owner @@ -78,26 +97,20 @@ module ActiveRecord if klass && owner_keys.any? # Some databases impose a limit on the number of ids in a list (in Oracle it's 1000) # Make several smaller queries if necessary or make one query if the adapter supports it - sliced = owner_keys.each_slice(klass.connection.in_clause_length || owner_keys.size) - sliced.each { |slice| - records = records_for(slice) - caster = type_caster(records, association_key_name) - records.each do |record| - owner_key = caster.call record[association_key_name] - - owners_map[owner_key].each do |owner| - records_by_owner[owner] << record - end + caster = type_caster + target_records.each do |record| + owner_key = caster.call record[association_key_name] + + owners_map[owner_key].each do |owner| + records_by_owner[owner] << record end - } + end end records_by_owner end - IDENTITY = lambda { |value| value } - def type_caster(results, name) - IDENTITY + def set_type_caster(results, name) end def reflection_scope diff --git a/activerecord/lib/active_record/associations/preloader/has_and_belongs_to_many.rb b/activerecord/lib/active_record/associations/preloader/has_and_belongs_to_many.rb index c042a44b21..87992a70e8 100644 --- a/activerecord/lib/active_record/associations/preloader/has_and_belongs_to_many.rb +++ b/activerecord/lib/active_record/associations/preloader/has_and_belongs_to_many.rb @@ -40,9 +40,9 @@ module ActiveRecord end end - def type_caster(results, name) + def set_type_caster(results, name) caster = results.column_types.fetch(name, results.identity_type) - lambda { |value| caster.type_cast value } + @type_caster = lambda { |value| caster.type_cast value } end def build_scope |