forked from cybrespace/mastodon
		
	Improve counter caches on Status and Account (#7644)
Do not touch statuses_count on accounts table when mass-destroying statuses to reduce load when removing accounts, same for reblogs_count and favourites_count Do not count statuses with direct visibility in statuses_count Fix #828
This commit is contained in:
		
							parent
							
								
									461542784b
								
							
						
					
					
						commit
						a7d726c383
					
				
					 6 changed files with 126 additions and 7 deletions
				
			
		|  | @ -16,7 +16,7 @@ class Favourite < ApplicationRecord | ||||||
|   update_index('statuses#status', :status) if Chewy.enabled? |   update_index('statuses#status', :status) if Chewy.enabled? | ||||||
| 
 | 
 | ||||||
|   belongs_to :account, inverse_of: :favourites |   belongs_to :account, inverse_of: :favourites | ||||||
|   belongs_to :status,  inverse_of: :favourites, counter_cache: true |   belongs_to :status,  inverse_of: :favourites | ||||||
| 
 | 
 | ||||||
|   has_one :notification, as: :activity, dependent: :destroy |   has_one :notification, as: :activity, dependent: :destroy | ||||||
| 
 | 
 | ||||||
|  | @ -25,4 +25,27 @@ class Favourite < ApplicationRecord | ||||||
|   before_validation do |   before_validation do | ||||||
|     self.status = status.reblog if status&.reblog? |     self.status = status.reblog if status&.reblog? | ||||||
|   end |   end | ||||||
|  | 
 | ||||||
|  |   after_create :increment_cache_counters | ||||||
|  |   after_destroy :decrement_cache_counters | ||||||
|  | 
 | ||||||
|  |   private | ||||||
|  | 
 | ||||||
|  |   def increment_cache_counters | ||||||
|  |     if association(:status).loaded? | ||||||
|  |       status.update_attribute(:favourites_count, status.favourites_count + 1) | ||||||
|  |     else | ||||||
|  |       Status.where(id: status_id).update_all('favourites_count = COALESCE(favourites_count, 0) + 1') | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   def decrement_cache_counters | ||||||
|  |     return if association(:status).loaded? && (status.marked_for_destruction? || status.marked_for_mass_destruction?) | ||||||
|  | 
 | ||||||
|  |     if association(:status).loaded? | ||||||
|  |       status.update_attribute(:favourites_count, [status.favourites_count - 1, 0].max) | ||||||
|  |     else | ||||||
|  |       Status.where(id: status_id).update_all('favourites_count = GREATEST(COALESCE(favourites_count, 0) - 1, 0)') | ||||||
|  |     end | ||||||
|  |   end | ||||||
| end | end | ||||||
|  |  | ||||||
|  | @ -41,12 +41,12 @@ class Status < ApplicationRecord | ||||||
| 
 | 
 | ||||||
|   belongs_to :application, class_name: 'Doorkeeper::Application', optional: true |   belongs_to :application, class_name: 'Doorkeeper::Application', optional: true | ||||||
| 
 | 
 | ||||||
|   belongs_to :account, inverse_of: :statuses, counter_cache: true |   belongs_to :account, inverse_of: :statuses | ||||||
|   belongs_to :in_reply_to_account, foreign_key: 'in_reply_to_account_id', class_name: 'Account', optional: true |   belongs_to :in_reply_to_account, foreign_key: 'in_reply_to_account_id', class_name: 'Account', optional: true | ||||||
|   belongs_to :conversation, optional: true |   belongs_to :conversation, optional: true | ||||||
| 
 | 
 | ||||||
|   belongs_to :thread, foreign_key: 'in_reply_to_id', class_name: 'Status', inverse_of: :replies, optional: true |   belongs_to :thread, foreign_key: 'in_reply_to_id', class_name: 'Status', inverse_of: :replies, optional: true | ||||||
|   belongs_to :reblog, foreign_key: 'reblog_of_id', class_name: 'Status', inverse_of: :reblogs, counter_cache: :reblogs_count, optional: true |   belongs_to :reblog, foreign_key: 'reblog_of_id', class_name: 'Status', inverse_of: :reblogs, optional: true | ||||||
| 
 | 
 | ||||||
|   has_many :favourites, inverse_of: :status, dependent: :destroy |   has_many :favourites, inverse_of: :status, dependent: :destroy | ||||||
|   has_many :reblogs, foreign_key: 'reblog_of_id', class_name: 'Status', inverse_of: :reblog, dependent: :destroy |   has_many :reblogs, foreign_key: 'reblog_of_id', class_name: 'Status', inverse_of: :reblog, dependent: :destroy | ||||||
|  | @ -167,6 +167,17 @@ class Status < ApplicationRecord | ||||||
|     @emojis ||= CustomEmoji.from_text([spoiler_text, text].join(' '), account.domain) |     @emojis ||= CustomEmoji.from_text([spoiler_text, text].join(' '), account.domain) | ||||||
|   end |   end | ||||||
| 
 | 
 | ||||||
|  |   def mark_for_mass_destruction! | ||||||
|  |     @marked_for_mass_destruction = true | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   def marked_for_mass_destruction? | ||||||
|  |     @marked_for_mass_destruction | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   after_create  :increment_counter_caches | ||||||
|  |   after_destroy :decrement_counter_caches | ||||||
|  | 
 | ||||||
|   after_create_commit :store_uri, if: :local? |   after_create_commit :store_uri, if: :local? | ||||||
|   after_create_commit :update_statistics, if: :local? |   after_create_commit :update_statistics, if: :local? | ||||||
| 
 | 
 | ||||||
|  | @ -388,4 +399,40 @@ class Status < ApplicationRecord | ||||||
|     return unless public_visibility? || unlisted_visibility? |     return unless public_visibility? || unlisted_visibility? | ||||||
|     ActivityTracker.increment('activity:statuses:local') |     ActivityTracker.increment('activity:statuses:local') | ||||||
|   end |   end | ||||||
|  | 
 | ||||||
|  |   def increment_counter_caches | ||||||
|  |     return if direct_visibility? | ||||||
|  | 
 | ||||||
|  |     if association(:account).loaded? | ||||||
|  |       account.update_attribute(:statuses_count, account.statuses_count + 1) | ||||||
|  |     else | ||||||
|  |       Account.where(id: account_id).update_all('statuses_count = COALESCE(statuses_count, 0) + 1') | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     return unless reblog? | ||||||
|  | 
 | ||||||
|  |     if association(:reblog).loaded? | ||||||
|  |       reblog.update_attribute(:reblogs_count, reblog.reblogs_count + 1) | ||||||
|  |     else | ||||||
|  |       Status.where(id: reblog_of_id).update_all('reblogs_count = COALESCE(reblogs_count, 0) + 1') | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|  |   def decrement_counter_caches | ||||||
|  |     return if direct_visibility? || marked_for_mass_destruction? | ||||||
|  | 
 | ||||||
|  |     if association(:account).loaded? | ||||||
|  |       account.update_attribute(:statuses_count, [account.statuses_count - 1, 0].max) | ||||||
|  |     else | ||||||
|  |       Account.where(id: account_id).update_all('statuses_count = GREATEST(COALESCE(statuses_count, 0) - 1, 0)') | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     return unless reblog? | ||||||
|  | 
 | ||||||
|  |     if association(:reblog).loaded? | ||||||
|  |       reblog.update_attribute(:reblogs_count, [reblog.reblogs_count - 1, 0].max) | ||||||
|  |     else | ||||||
|  |       Status.where(id: reblog_of_id).update_all('reblogs_count = GREATEST(COALESCE(reblogs_count, 0) - 1, 0)') | ||||||
|  |     end | ||||||
|  |   end | ||||||
| end | end | ||||||
|  |  | ||||||
|  | @ -21,7 +21,10 @@ class BatchedRemoveStatusService < BaseService | ||||||
|     @activity_xml          = {} |     @activity_xml          = {} | ||||||
| 
 | 
 | ||||||
|     # Ensure that rendered XML reflects destroyed state |     # Ensure that rendered XML reflects destroyed state | ||||||
|     statuses.each(&:destroy) |     statuses.each do |status| | ||||||
|  |       status.mark_for_mass_destruction! | ||||||
|  |       status.destroy | ||||||
|  |     end | ||||||
| 
 | 
 | ||||||
|     # Batch by source account |     # Batch by source account | ||||||
|     statuses.group_by(&:account_id).each_value do |account_statuses| |     statuses.group_by(&:account_id).each_value do |account_statuses| | ||||||
|  |  | ||||||
|  | @ -44,6 +44,7 @@ class SuspendAccountService < BaseService | ||||||
|     @account.suspended      = true |     @account.suspended      = true | ||||||
|     @account.display_name   = '' |     @account.display_name   = '' | ||||||
|     @account.note           = '' |     @account.note           = '' | ||||||
|  |     @account.statuses_count = 0 | ||||||
|     @account.avatar.destroy |     @account.avatar.destroy | ||||||
|     @account.header.destroy |     @account.header.destroy | ||||||
|     @account.save! |     @account.save! | ||||||
|  |  | ||||||
|  | @ -525,6 +525,37 @@ RSpec.describe Account, type: :model do | ||||||
|     end |     end | ||||||
|   end |   end | ||||||
| 
 | 
 | ||||||
|  |   describe '#statuses_count' do | ||||||
|  |     subject { Fabricate(:account) } | ||||||
|  | 
 | ||||||
|  |     it 'counts statuses' do | ||||||
|  |       Fabricate(:status, account: subject) | ||||||
|  |       Fabricate(:status, account: subject) | ||||||
|  |       expect(subject.statuses_count).to eq 2 | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     it 'does not count direct statuses' do | ||||||
|  |       Fabricate(:status, account: subject, visibility: :direct) | ||||||
|  |       expect(subject.statuses_count).to eq 0 | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     it 'is decremented when status is removed' do | ||||||
|  |       status = Fabricate(:status, account: subject) | ||||||
|  |       expect(subject.statuses_count).to eq 1 | ||||||
|  |       status.destroy | ||||||
|  |       expect(subject.statuses_count).to eq 0 | ||||||
|  |     end | ||||||
|  | 
 | ||||||
|  |     it 'is decremented when status is removed when account is not preloaded' do | ||||||
|  |       status = Fabricate(:status, account: subject) | ||||||
|  |       expect(subject.reload.statuses_count).to eq 1 | ||||||
|  |       clean_status = Status.find(status.id) | ||||||
|  |       expect(clean_status.association(:account).loaded?).to be false | ||||||
|  |       clean_status.destroy | ||||||
|  |       expect(subject.reload.statuses_count).to eq 0 | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  | 
 | ||||||
|   describe '.following_map' do |   describe '.following_map' do | ||||||
|     it 'returns an hash' do |     it 'returns an hash' do | ||||||
|       expect(Account.following_map([], 1)).to be_a Hash |       expect(Account.following_map([], 1)).to be_a Hash | ||||||
|  |  | ||||||
|  | @ -175,6 +175,13 @@ RSpec.describe Status, type: :model do | ||||||
| 
 | 
 | ||||||
|       expect(subject.reblogs_count).to eq 2 |       expect(subject.reblogs_count).to eq 2 | ||||||
|     end |     end | ||||||
|  | 
 | ||||||
|  |     it 'is decremented when reblog is removed' do | ||||||
|  |       reblog = Fabricate(:status, account: bob, reblog: subject) | ||||||
|  |       expect(subject.reblogs_count).to eq 1 | ||||||
|  |       reblog.destroy | ||||||
|  |       expect(subject.reblogs_count).to eq 0 | ||||||
|  |     end | ||||||
|   end |   end | ||||||
| 
 | 
 | ||||||
|   describe '#favourites_count' do |   describe '#favourites_count' do | ||||||
|  | @ -184,6 +191,13 @@ RSpec.describe Status, type: :model do | ||||||
| 
 | 
 | ||||||
|       expect(subject.favourites_count).to eq 2 |       expect(subject.favourites_count).to eq 2 | ||||||
|     end |     end | ||||||
|  | 
 | ||||||
|  |     it 'is decremented when favourite is removed' do | ||||||
|  |       favourite = Fabricate(:favourite, account: bob, status: subject) | ||||||
|  |       expect(subject.favourites_count).to eq 1 | ||||||
|  |       favourite.destroy | ||||||
|  |       expect(subject.favourites_count).to eq 0 | ||||||
|  |     end | ||||||
|   end |   end | ||||||
| 
 | 
 | ||||||
|   describe '#proper' do |   describe '#proper' do | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		
		Reference in a new issue