Pulled in ActiveRecord synchronize functionality from the old ar-extensions code base.
https://github.com/zdennis/activerecord-import/issues#issue/6
This commit is contained in:
parent
9e6866fd29
commit
319c52d80f
3 changed files with 53 additions and 0 deletions
|
@ -25,3 +25,4 @@ end
|
|||
this_dir = Pathname.new File.dirname(__FILE__)
|
||||
require this_dir.join("import")
|
||||
require this_dir.join("active_record/adapters/abstract_adapter")
|
||||
require this_dir.join("synchronize")
|
30
lib/activerecord-import/synchronize.rb
Normal file
30
lib/activerecord-import/synchronize.rb
Normal file
|
@ -0,0 +1,30 @@
|
|||
module ActiveRecord # :nodoc:
|
||||
class Base # :nodoc:
|
||||
|
||||
# Synchronizes the passed in ActiveRecord instances with data
|
||||
# from the database. This is like calling reload
|
||||
# on an individual ActiveRecord instance but it is intended for use on
|
||||
# multiple instances.
|
||||
#
|
||||
# This uses one query for all instance updates and then updates existing
|
||||
# instances rather sending one query for each instance
|
||||
def self.synchronize(instances, key=self.primary_key)
|
||||
return if instances.empty?
|
||||
|
||||
keys = instances.map(&"#{key}".to_sym)
|
||||
klass = instances.first.class
|
||||
fresh_instances = klass.find( :all, :conditions=>{ key=>keys }, :order=>"#{key} ASC" )
|
||||
|
||||
instances.each_with_index do |instance, index|
|
||||
instance.clear_aggregation_cache
|
||||
instance.clear_association_cache
|
||||
instance.instance_variable_set '@attributes', fresh_instances[index].attributes
|
||||
end
|
||||
end
|
||||
|
||||
# See ActiveRecord::ConnectionAdapters::AbstractAdapter.synchronize
|
||||
def synchronize(instances, key=ActiveRecord::Base.primary_key)
|
||||
self.class.synchronize(instances, key)
|
||||
end
|
||||
end
|
||||
end
|
22
test/synchronize_test.rb
Normal file
22
test/synchronize_test.rb
Normal file
|
@ -0,0 +1,22 @@
|
|||
require File.expand_path('../test_helper', __FILE__)
|
||||
|
||||
describe ".synchronize" do
|
||||
let(:topics){ Generate(3, :topics) }
|
||||
let(:titles){ %w(one two three) }
|
||||
|
||||
setup do
|
||||
# update records outside of ActiveRecord knowing about it
|
||||
Topic.connection.execute( "UPDATE #{Topic.table_name} SET title='#{titles[0]}_haha' WHERE id=#{topics[0].id}", "Updating record 1 without ActiveRecord" )
|
||||
Topic.connection.execute( "UPDATE #{Topic.table_name} SET title='#{titles[1]}_haha' WHERE id=#{topics[1].id}", "Updating record 2 without ActiveRecord" )
|
||||
Topic.connection.execute( "UPDATE #{Topic.table_name} SET title='#{titles[2]}_haha' WHERE id=#{topics[2].id}", "Updating record 3 without ActiveRecord" )
|
||||
end
|
||||
|
||||
it "reloads data for the specified records" do
|
||||
Book.synchronize topics
|
||||
|
||||
actual_titles = topics.map(&:title)
|
||||
assert_equal "#{titles[0]}_haha", actual_titles[0], "the first record was not correctly updated"
|
||||
assert_equal "#{titles[1]}_haha", actual_titles[1], "the second record was not correctly updated"
|
||||
assert_equal "#{titles[2]}_haha", actual_titles[2], "the third record was not correctly updated"
|
||||
end
|
||||
end
|
Loading…
Reference in a new issue