Optimize MuteService and AfterBlockService (#2836)
This commit is contained in:
		| @@ -76,6 +76,14 @@ class FeedManager | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def clear_from_timeline(account, target_account) | ||||
|     timeline_key = key(:home, account.id) | ||||
|     timeline_status_ids = redis.zrange(timeline_key, 0, -1) | ||||
|     target_status_ids = Status.where(id: timeline_status_ids, account: target_account).ids | ||||
|  | ||||
|     redis.zrem(timeline_key, target_status_ids) if target_status_ids.present? | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def redis | ||||
|   | ||||
| @@ -2,30 +2,16 @@ | ||||
|  | ||||
| class AfterBlockService < BaseService | ||||
|   def call(account, target_account) | ||||
|     clear_timelines(account, target_account) | ||||
|     FeedManager.instance.clear_from_timeline(account, target_account) | ||||
|     clear_notifications(account, target_account) | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def clear_timelines(account, target_account) | ||||
|     home_key = FeedManager.instance.key(:home, account.id) | ||||
|  | ||||
|     redis.pipelined do | ||||
|       target_account.statuses.select('id').reorder(nil).find_each do |status| | ||||
|         redis.zrem(home_key, status.id) | ||||
|       end | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def clear_notifications(account, target_account) | ||||
|     Notification.where(account: account).joins(:follow).where(activity_type: 'Follow', follows: { account_id: target_account.id }).delete_all | ||||
|     Notification.where(account: account).joins(mention: :status).where(activity_type: 'Mention', statuses: { account_id: target_account.id }).delete_all | ||||
|     Notification.where(account: account).joins(:favourite).where(activity_type: 'Favourite', favourites: { account_id: target_account.id }).delete_all | ||||
|     Notification.where(account: account).joins(:status).where(activity_type: 'Status', statuses: { account_id: target_account.id }).delete_all | ||||
|   end | ||||
|  | ||||
|   def redis | ||||
|     Redis.current | ||||
|   end | ||||
| end | ||||
|   | ||||
| @@ -3,21 +3,7 @@ | ||||
| class MuteService < BaseService | ||||
|   def call(account, target_account) | ||||
|     return if account.id == target_account.id | ||||
|     clear_home_timeline(account, target_account) | ||||
|     FeedManager.instance.clear_from_timeline(account, target_account) | ||||
|     account.mute!(target_account) | ||||
|   end | ||||
|  | ||||
|   private | ||||
|  | ||||
|   def clear_home_timeline(account, target_account) | ||||
|     home_key = FeedManager.instance.key(:home, account.id) | ||||
|  | ||||
|     target_account.statuses.select('id').reorder(nil).find_each do |status| | ||||
|       redis.zrem(home_key, status.id) | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   def redis | ||||
|     Redis.current | ||||
|   end | ||||
| end | ||||
|   | ||||
							
								
								
									
										29
									
								
								spec/services/after_block_service_spec.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								spec/services/after_block_service_spec.rb
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| require 'rails_helper' | ||||
|  | ||||
| RSpec.describe AfterBlockService do | ||||
|   subject do | ||||
|     -> { described_class.new.call(account, target_account) } | ||||
|   end | ||||
|  | ||||
|   let(:account) { Fabricate(:account) } | ||||
|   let(:target_account) { Fabricate(:account) } | ||||
|  | ||||
|   describe 'home timeline' do | ||||
|     let(:status) { Fabricate(:status, account: target_account) } | ||||
|     let(:other_account_status) { Fabricate(:status) } | ||||
|     let(:home_timeline_key) { FeedManager.instance.key(:home, account.id) } | ||||
|  | ||||
|     before do | ||||
|       Redis.current.del(home_timeline_key) | ||||
|     end | ||||
|  | ||||
|     it "clears account's statuses" do | ||||
|       FeedManager.instance.push(:home, account, status) | ||||
|       FeedManager.instance.push(:home, account, other_account_status) | ||||
|  | ||||
|       is_expected.to change { | ||||
|         Redis.current.zrange(home_timeline_key, 0, -1) | ||||
|       }.from([status.id.to_s, other_account_status.id.to_s]).to([other_account_status.id.to_s]) | ||||
|     end | ||||
|   end | ||||
| end | ||||
| @@ -1,5 +1,35 @@ | ||||
| require 'rails_helper' | ||||
|  | ||||
| RSpec.describe MuteService do | ||||
|   subject { MuteService.new } | ||||
|   subject do | ||||
|     -> { described_class.new.call(account, target_account) } | ||||
|   end | ||||
|  | ||||
|   let(:account) { Fabricate(:account) } | ||||
|   let(:target_account) { Fabricate(:account) } | ||||
|  | ||||
|   describe 'home timeline' do | ||||
|     let(:status) { Fabricate(:status, account: target_account) } | ||||
|     let(:other_account_status) { Fabricate(:status) } | ||||
|     let(:home_timeline_key) { FeedManager.instance.key(:home, account.id) } | ||||
|  | ||||
|     before do | ||||
|       Redis.current.del(home_timeline_key) | ||||
|     end | ||||
|  | ||||
|     it "clears account's statuses" do | ||||
|       FeedManager.instance.push(:home, account, status) | ||||
|       FeedManager.instance.push(:home, account, other_account_status) | ||||
|  | ||||
|       is_expected.to change { | ||||
|         Redis.current.zrange(home_timeline_key, 0, -1) | ||||
|       }.from([status.id.to_s, other_account_status.id.to_s]).to([other_account_status.id.to_s]) | ||||
|     end | ||||
|   end | ||||
|  | ||||
|   it 'mutes account' do | ||||
|     is_expected.to change { | ||||
|       account.muting?(target_account) | ||||
|     }.from(false).to(true) | ||||
|   end | ||||
| end | ||||
|   | ||||
		Reference in New Issue
	
	Block a user