aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorRick Olson <technoweenie@gmail.com>2007-01-02 05:36:30 +0000
committerRick Olson <technoweenie@gmail.com>2007-01-02 05:36:30 +0000
commit1af2022cc32b604529a5ac3a85a3bd92a3c42936 (patch)
tree494c54bce8f7446f066afea2675425233342fdfa
parentb75f28edb4675250e5f18cf9760e4a11a5f4c926 (diff)
downloadrails-1af2022cc32b604529a5ac3a85a3bd92a3c42936.tar.gz
rails-1af2022cc32b604529a5ac3a85a3bd92a3c42936.tar.bz2
rails-1af2022cc32b604529a5ac3a85a3bd92a3c42936.zip
Rollback #new_record? and #id values for created records that rollback in an after_save callback. Closes #6910 [Ben Curren]
git-svn-id: http://svn-commit.rubyonrails.org/rails/trunk@5830 5ecf4fe2-1ee6-0310-87b1-e25e094e27de
-rw-r--r--activerecord/CHANGELOG2
-rw-r--r--activerecord/lib/active_record/transactions.rb21
-rw-r--r--activerecord/test/transactions_test.rb38
3 files changed, 59 insertions, 2 deletions
diff --git a/activerecord/CHANGELOG b/activerecord/CHANGELOG
index c5c6925e24..f6d3bc2ae9 100644
--- a/activerecord/CHANGELOG
+++ b/activerecord/CHANGELOG
@@ -1,5 +1,7 @@
*SVN*
+* Rollback #new_record? and #id values for created records that rollback in an after_save callback. Closes #6910 [Ben Curren]
+
* Pushing a record on an association collection doesn't unnecessarily load all the associated records. [Obie Fernandez, Jeremy Kemper]
* Oracle: fix connection reset failure. #6846 [leonlleslie]
diff --git a/activerecord/lib/active_record/transactions.rb b/activerecord/lib/active_record/transactions.rb
index 6a776966bb..5ee644c0b4 100644
--- a/activerecord/lib/active_record/transactions.rb
+++ b/activerecord/lib/active_record/transactions.rb
@@ -126,11 +126,28 @@ module ActiveRecord
end
def save_with_transactions(perform_validation = true) #:nodoc:
- transaction { save_without_transactions(perform_validation) }
+ rollback_active_record_state { transaction { save_without_transactions(perform_validation) } }
end
def save_with_transactions! #:nodoc:
- transaction { save_without_transactions! }
+ rollback_active_record_state { transaction { save_without_transactions! } }
+ end
+
+ # stores the current id and @new_record values so that they are reset
+ # after rolling the transaction back.
+ def rollback_active_record_state
+ previous_new_record = @new_record
+ previous_id = self.id
+ response = yield
+ rescue
+ response = false
+ raise
+ ensure
+ unless response
+ @new_record = previous_new_record
+ self.id = previous_id
+ end
+ response
end
end
end
diff --git a/activerecord/test/transactions_test.rb b/activerecord/test/transactions_test.rb
index b0d1eb4eb8..ecc0da425d 100644
--- a/activerecord/test/transactions_test.rb
+++ b/activerecord/test/transactions_test.rb
@@ -121,6 +121,36 @@ class TransactionTest < Test::Unit::TestCase
remove_exception_raising_after_save_callback_to_topic
end
end
+
+ def test_callback_rollback_in_create
+ new_topic = Topic.new(
+ :title => "A new topic",
+ :author_name => "Ben",
+ :author_email_address => "ben@example.com",
+ :written_on => "2003-07-16t15:28:11.2233+01:00",
+ :last_read => "2004-04-15",
+ :bonus_time => "2005-01-30t15:28:00.00+01:00",
+ :content => "Have a nice day",
+ :approved => false)
+ new_record_snapshot = new_topic.new_record?
+ id_snapshot = new_topic.id
+
+ # Make sure the second save gets the after_create callback called.
+ 2.times do
+ begin
+ add_exception_raising_after_create_callback_to_topic
+ new_topic.approved = true
+ new_topic.save
+ flunk
+ rescue => e
+ assert_equal "Make the transaction rollback", e.message
+ assert_equal new_record_snapshot, new_topic.new_record?, "The topic should have its old new_record value"
+ assert_equal id_snapshot, new_topic.id, "The topic should have its old id"
+ ensure
+ remove_exception_raising_after_create_callback_to_topic
+ end
+ end
+ end
def test_nested_explicit_transactions
Topic.transaction do
@@ -144,6 +174,14 @@ class TransactionTest < Test::Unit::TestCase
def remove_exception_raising_after_save_callback_to_topic
Topic.class_eval { remove_method :after_save }
end
+
+ def add_exception_raising_after_create_callback_to_topic
+ Topic.class_eval { def after_create() raise "Make the transaction rollback" end }
+ end
+
+ def remove_exception_raising_after_create_callback_to_topic
+ Topic.class_eval { remove_method :after_create }
+ end
end
if current_adapter?(:PostgreSQLAdapter)