Fix account counters being overwritten by parallel writes (#12045)
parent
9e3e3fa5ee
commit
62f60e86c2
@ -0,0 +1,15 @@ |
|||||||
|
require Rails.root.join('lib', 'mastodon', 'migration_helpers') |
||||||
|
|
||||||
|
class AddLockVersionToAccountStats < ActiveRecord::Migration[5.2] |
||||||
|
include Mastodon::MigrationHelpers |
||||||
|
|
||||||
|
disable_ddl_transaction! |
||||||
|
|
||||||
|
def up |
||||||
|
safety_assured { add_column_with_default :account_stats, :lock_version, :integer, allow_null: false, default: 0 } |
||||||
|
end |
||||||
|
|
||||||
|
def down |
||||||
|
remove_column :account_stats, :lock_version |
||||||
|
end |
||||||
|
end |
@ -1,4 +1,57 @@ |
|||||||
require 'rails_helper' |
require 'rails_helper' |
||||||
|
|
||||||
RSpec.describe AccountStat, type: :model do |
RSpec.describe AccountStat, type: :model do |
||||||
|
describe '#increment_count!' do |
||||||
|
it 'increments the count' do |
||||||
|
account_stat = AccountStat.create(account: Fabricate(:account)) |
||||||
|
expect(account_stat.followers_count).to eq 0 |
||||||
|
account_stat.increment_count!(:followers_count) |
||||||
|
expect(account_stat.followers_count).to eq 1 |
||||||
|
end |
||||||
|
|
||||||
|
it 'increments the count in multi-threaded an environment' do |
||||||
|
account_stat = AccountStat.create(account: Fabricate(:account), statuses_count: 0) |
||||||
|
increment_by = 15 |
||||||
|
wait_for_start = true |
||||||
|
|
||||||
|
threads = Array.new(increment_by) do |
||||||
|
Thread.new do |
||||||
|
true while wait_for_start |
||||||
|
AccountStat.find(account_stat.id).increment_count!(:statuses_count) |
||||||
|
end |
||||||
|
end |
||||||
|
|
||||||
|
wait_for_start = false |
||||||
|
threads.each(&:join) |
||||||
|
|
||||||
|
expect(account_stat.reload.statuses_count).to eq increment_by |
||||||
|
end |
||||||
|
end |
||||||
|
|
||||||
|
describe '#decrement_count!' do |
||||||
|
it 'decrements the count' do |
||||||
|
account_stat = AccountStat.create(account: Fabricate(:account), followers_count: 15) |
||||||
|
expect(account_stat.followers_count).to eq 15 |
||||||
|
account_stat.decrement_count!(:followers_count) |
||||||
|
expect(account_stat.followers_count).to eq 14 |
||||||
|
end |
||||||
|
|
||||||
|
it 'decrements the count in multi-threaded an environment' do |
||||||
|
account_stat = AccountStat.create(account: Fabricate(:account), statuses_count: 15) |
||||||
|
decrement_by = 10 |
||||||
|
wait_for_start = true |
||||||
|
|
||||||
|
threads = Array.new(decrement_by) do |
||||||
|
Thread.new do |
||||||
|
true while wait_for_start |
||||||
|
AccountStat.find(account_stat.id).decrement_count!(:statuses_count) |
||||||
|
end |
||||||
|
end |
||||||
|
|
||||||
|
wait_for_start = false |
||||||
|
threads.each(&:join) |
||||||
|
|
||||||
|
expect(account_stat.reload.statuses_count).to eq 5 |
||||||
|
end |
||||||
|
end |
||||||
end |
end |
||||||
|
Loading…
Reference in new issue