1
0
Fork 0
mirror of https://github.com/rails/rails.git synced 2022-11-09 12:12:34 -05:00

Merge pull request #39316 from alipman88/resolve_counter_cache_lock_version_conflict

Resolve conflict between counter cache and optimistic locking
This commit is contained in:
Eugene Kenny 2020-05-20 02:12:41 +01:00 committed by GitHub
commit f6a94aff2e
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 26 additions and 0 deletions

View file

@ -1,3 +1,14 @@
* Resolve conflict between counter cache and optimistic locking.
Bump an Active Record instance's lock version after updating its counter
cache. This avoids raising an unnecessary `ActiveRecord::StaleObjectError`
upon subsequent transactions by maintaining parity with the corresponding
database record's `lock_version` column.
Fixes #16449.
*Aaron Lipman*
* Support merging option `:rewhere` to allow mergee side condition to be replaced exactly. * Support merging option `:rewhere` to allow mergee side condition to be replaced exactly.
```ruby ```ruby

View file

@ -60,6 +60,15 @@ module ActiveRecord
self.class.locking_enabled? self.class.locking_enabled?
end end
def increment!(*, **) #:nodoc:
super.tap do
if locking_enabled?
self[self.class.locking_column] += 1
clear_attribute_change(self.class.locking_column)
end
end
end
private private
def _create_record(attribute_names = self.attribute_names) def _create_record(attribute_names = self.attribute_names)
if locking_enabled? if locking_enabled?

View file

@ -509,6 +509,12 @@ class OptimisticLockingTest < ActiveRecord::TestCase
assert_equal 3, car.lock_version assert_equal 3, car.lock_version
assert_operator previously_updated_at, :<, car.updated_at assert_operator previously_updated_at, :<, car.updated_at
assert_operator previously_wheels_owned_at, :<, car.wheels_owned_at assert_operator previously_wheels_owned_at, :<, car.wheels_owned_at
car.wheels << Wheel.create!
assert_equal 1, car.wheels_count
assert_equal 4, car.lock_version
assert_not car.lock_version_changed?
assert_nothing_raised { car.update(name: "herbie") }
end end
def test_polymorphic_destroy_with_dependencies_and_lock_version def test_polymorphic_destroy_with_dependencies_and_lock_version