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:
commit
f6a94aff2e
3 changed files with 26 additions and 0 deletions
|
@ -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
|
||||||
|
|
|
@ -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?
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in a new issue