aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--activerecord/CHANGELOG.md6
-rw-r--r--activerecord/lib/active_record/core.rb1
-rw-r--r--activerecord/lib/active_record/transactions.rb2
-rw-r--r--activerecord/test/cases/transactions_test.rb22
4 files changed, 11 insertions, 20 deletions
diff --git a/activerecord/CHANGELOG.md b/activerecord/CHANGELOG.md
index 4164b928bd..ae4a672ef1 100644
--- a/activerecord/CHANGELOG.md
+++ b/activerecord/CHANGELOG.md
@@ -1,3 +1,9 @@
+* Keep track of dirty attributes after transaction is rollback.
+
+ Related #13166.
+
+ *Bogdan Gusiev* *arthurnn*
+
* Add support for module-level `table_name_suffix` in models.
This makes `table_name_suffix` work the same way as `table_name_prefix` when
diff --git a/activerecord/lib/active_record/core.rb b/activerecord/lib/active_record/core.rb
index 22870fb6c5..4571cc0786 100644
--- a/activerecord/lib/active_record/core.rb
+++ b/activerecord/lib/active_record/core.rb
@@ -505,6 +505,7 @@ module ActiveRecord
@destroyed_by_association = nil
@new_record = true
@txn = nil
+ @_start_transaction_state = {}
@transaction_state = nil
@reflects_state = [false]
end
diff --git a/activerecord/lib/active_record/transactions.rb b/activerecord/lib/active_record/transactions.rb
index 33c9af3d3d..cf76d53bf7 100644
--- a/activerecord/lib/active_record/transactions.rb
+++ b/activerecord/lib/active_record/transactions.rb
@@ -341,7 +341,6 @@ module ActiveRecord
# Save the new record state and id of a record so it can be restored later if a transaction fails.
def remember_transaction_record_state #:nodoc:
- @_start_transaction_state ||= {}
@_start_transaction_state[:id] = id if has_attribute?(self.class.primary_key)
unless @_start_transaction_state.include?(:new_record)
@_start_transaction_state[:new_record] = @new_record
@@ -352,7 +351,6 @@ module ActiveRecord
@_start_transaction_state[:level] = (@_start_transaction_state[:level] || 0) + 1
@_start_transaction_state[:frozen?] = @attributes.frozen?
@_start_transaction_state[:changed_attributes] ||= changed_attributes
- @_start_transaction_state
end
# Clear the new record state and id of a record.
diff --git a/activerecord/test/cases/transactions_test.rb b/activerecord/test/cases/transactions_test.rb
index bb5548a60f..7f2e830083 100644
--- a/activerecord/test/cases/transactions_test.rb
+++ b/activerecord/test/cases/transactions_test.rb
@@ -274,7 +274,7 @@ class TransactionTest < ActiveRecord::TestCase
end
end
- def test_dirty_state_rollback
+ def test_rollback_when_changing_inside_transaction
assert !@first.approved?
Topic.transaction do
@first.approved = true
@@ -287,7 +287,7 @@ class TransactionTest < ActiveRecord::TestCase
assert @first.reload.approved
end
- def test_dirty_state_rollback2
+ def test_rollback_when_changing_outside_transaction
assert !@first.approved?
@first.approved = true
Topic.transaction do
@@ -300,22 +300,7 @@ class TransactionTest < ActiveRecord::TestCase
assert @first.reload.approved
end
- def test_dirty_state_rollback3
- assert !@first.approved?
- @first.approved = true
- @first.save!
- Topic.transaction do
- @first.approved = false
- @first.save!
- raise ActiveRecord::Rollback
- end
- assert !@first.approved
- assert @first.changes["approved"]
- @first.save!
- assert !@first.reload.approved
- end
-
- def test_dirty_state_rollback4
+ def test_rollback_when_changing_back_to_prev_stage
assert !@first.approved?
Topic.transaction do
@first.approved = true
@@ -330,6 +315,7 @@ class TransactionTest < ActiveRecord::TestCase
assert !@first.reload.approved
end
+
def test_force_savepoint_in_nested_transaction
Topic.transaction do
@first.approved = true