* Add buttons to block and unblock domain * Relationship API now returns "domain_blocking" status for accounts, rename "block entire domain" to "hide entire domain", fix unblocking domain, do not block notifications from domain-blocked-but-followed people, do not send Salmons to domain blocked users * Add test * Personal domain blocks shouldn't affect Salmon after all, since in this direction of communication the control is very thin when it comes to public stuff. Best stay consistent and not affect federation in this way * Ignore followers and follow request from domain blocked folks, ensure account domain blocks are not created for empty domain, and avoid duplicates in validation * Purge followers when blocking domain (without soft-blocks, since they are useless here) * Add tests, fix local timeline being empty when having any domain blocks
		
			
				
	
	
		
			138 lines
		
	
	
	
		
			4.6 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			138 lines
		
	
	
	
		
			4.6 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # frozen_string_literal: true
 | |
| 
 | |
| class ProcessInteractionService < BaseService
 | |
|   include AuthorExtractor
 | |
| 
 | |
|   # Record locally the remote interaction with our user
 | |
|   # @param [String] envelope Salmon envelope
 | |
|   # @param [Account] target_account Account the Salmon was addressed to
 | |
|   def call(envelope, target_account)
 | |
|     body = salmon.unpack(envelope)
 | |
| 
 | |
|     xml = Nokogiri::XML(body)
 | |
|     xml.encoding = 'utf-8'
 | |
| 
 | |
|     account = author_from_xml(xml.at_xpath('/xmlns:entry', xmlns: TagManager::XMLNS))
 | |
| 
 | |
|     return if account.nil? || account.suspended?
 | |
| 
 | |
|     if salmon.verify(envelope, account.keypair)
 | |
|       RemoteProfileUpdateWorker.perform_async(account.id, body.force_encoding('UTF-8'), true)
 | |
| 
 | |
|       case verb(xml)
 | |
|       when :follow
 | |
|         follow!(account, target_account) unless target_account.locked? || target_account.blocking?(account) || target_account.domain_blocking?(account.domain)
 | |
|       when :request_friend
 | |
|         follow_request!(account, target_account) unless !target_account.locked? || target_account.blocking?(account) || target_account.domain_blocking?(account.domain)
 | |
|       when :authorize
 | |
|         authorize_follow_request!(account, target_account)
 | |
|       when :reject
 | |
|         reject_follow_request!(account, target_account)
 | |
|       when :unfollow
 | |
|         unfollow!(account, target_account)
 | |
|       when :favorite
 | |
|         favourite!(xml, account)
 | |
|       when :unfavorite
 | |
|         unfavourite!(xml, account)
 | |
|       when :post
 | |
|         add_post!(body, account) if mentions_account?(xml, target_account)
 | |
|       when :share
 | |
|         add_post!(body, account) unless status(xml).nil?
 | |
|       when :delete
 | |
|         delete_post!(xml, account)
 | |
|       when :block
 | |
|         reflect_block!(account, target_account)
 | |
|       when :unblock
 | |
|         reflect_unblock!(account, target_account)
 | |
|       end
 | |
|     end
 | |
|   rescue Goldfinger::Error, HTTP::Error, OStatus2::BadSalmonError
 | |
|     nil
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def mentions_account?(xml, account)
 | |
|     xml.xpath('/xmlns:entry/xmlns:link[@rel="mentioned"]', xmlns: TagManager::XMLNS).each { |mention_link| return true if [TagManager.instance.uri_for(account), TagManager.instance.url_for(account)].include?(mention_link.attribute('href').value) }
 | |
|     false
 | |
|   end
 | |
| 
 | |
|   def verb(xml)
 | |
|     raw = xml.at_xpath('//activity:verb', activity: TagManager::AS_XMLNS).content
 | |
|     TagManager::VERBS.key(raw)
 | |
|   rescue
 | |
|     :post
 | |
|   end
 | |
| 
 | |
|   def follow!(account, target_account)
 | |
|     follow = account.follow!(target_account)
 | |
|     NotifyService.new.call(target_account, follow)
 | |
|   end
 | |
| 
 | |
|   def follow_request!(account, target_account)
 | |
|     follow_request = FollowRequest.create!(account: account, target_account: target_account)
 | |
|     NotifyService.new.call(target_account, follow_request)
 | |
|   end
 | |
| 
 | |
|   def authorize_follow_request!(account, target_account)
 | |
|     follow_request = FollowRequest.find_by(account: target_account, target_account: account)
 | |
|     follow_request&.authorize!
 | |
|     Pubsubhubbub::SubscribeWorker.perform_async(account.id) unless account.subscribed?
 | |
|   end
 | |
| 
 | |
|   def reject_follow_request!(account, target_account)
 | |
|     follow_request = FollowRequest.find_by(account: target_account, target_account: account)
 | |
|     follow_request&.reject!
 | |
|   end
 | |
| 
 | |
|   def unfollow!(account, target_account)
 | |
|     account.unfollow!(target_account)
 | |
|   end
 | |
| 
 | |
|   def reflect_block!(account, target_account)
 | |
|     UnfollowService.new.call(target_account, account) if target_account.following?(account)
 | |
|     account.block!(target_account)
 | |
|   end
 | |
| 
 | |
|   def reflect_unblock!(account, target_account)
 | |
|     UnblockService.new.call(account, target_account)
 | |
|   end
 | |
| 
 | |
|   def delete_post!(xml, account)
 | |
|     status = Status.find(xml.at_xpath('//xmlns:id', xmlns: TagManager::XMLNS).content)
 | |
| 
 | |
|     return if status.nil?
 | |
| 
 | |
|     RemovalWorker.perform_async(status.id) if account.id == status.account_id
 | |
|   end
 | |
| 
 | |
|   def favourite!(xml, from_account)
 | |
|     current_status = status(xml)
 | |
|     favourite = current_status.favourites.where(account: from_account).first_or_create!(account: from_account)
 | |
|     NotifyService.new.call(current_status.account, favourite)
 | |
|   end
 | |
| 
 | |
|   def unfavourite!(xml, from_account)
 | |
|     current_status = status(xml)
 | |
|     favourite = current_status.favourites.where(account: from_account).first
 | |
|     favourite&.destroy
 | |
|   end
 | |
| 
 | |
|   def add_post!(body, account)
 | |
|     ProcessingWorker.perform_async(account.id, body.force_encoding('UTF-8'))
 | |
|   end
 | |
| 
 | |
|   def status(xml)
 | |
|     uri = activity_id(xml)
 | |
|     return nil unless TagManager.instance.local_id?(uri)
 | |
|     Status.find(TagManager.instance.unique_tag_to_local_id(uri, 'Status'))
 | |
|   end
 | |
| 
 | |
|   def activity_id(xml)
 | |
|     xml.at_xpath('//activity:object', activity: TagManager::AS_XMLNS).at_xpath('./xmlns:id', xmlns: TagManager::XMLNS).content
 | |
|   end
 | |
| 
 | |
|   def salmon
 | |
|     @salmon ||= OStatus2::Salmon.new
 | |
|   end
 | |
| end
 |