# frozen_string_literal: true # == Schema Information # # Table name: friend_domains # # id :bigint(8) not null, primary key # domain :string default(""), not null # inbox_url :string default(""), not null # active_state :integer default("idle"), not null # passive_state :integer default("idle"), not null # active_follow_activity_id :string # passive_follow_activity_id :string # available :boolean default(TRUE), not null # pseudo_relay :boolean default(FALSE), not null # allow_all_posts :boolean default(TRUE), not null # created_at :datetime not null # updated_at :datetime not null # delivery_local :boolean default(TRUE), not null # class FriendDomain < ApplicationRecord validates :domain, presence: true, uniqueness: true, if: :will_save_change_to_domain? validates :inbox_url, presence: true, uniqueness: true, if: :will_save_change_to_inbox_url? enum :active_state, { idle: 0, pending: 1, accepted: 2, rejected: 3 }, prefix: :i_am enum :passive_state, { idle: 0, pending: 1, accepted: 2, rejected: 3 }, prefix: :they_are scope :by_domain_and_subdomains, ->(domain) { where(domain: Instance.by_domain_and_subdomains(domain).select(:domain)) } scope :enabled, -> { where(active_state: :accepted).or(FriendDomain.where(passive_state: :accepted)).where(available: true) } scope :distributables, -> { enabled.where(pseudo_relay: true) } scope :deliver_locals, -> { enabled.where(delivery_local: true) } scope :free_receivings, -> { enabled.where(allow_all_posts: true) } before_destroy :ensure_disabled after_commit :set_default_inbox_url def accepted? i_am_accepted? || they_are_accepted? end def pending? !accepted? && (i_am_pending? || they_are_pending?) end def idle? (i_am_idle? || i_am_rejected?) && (they_are_idle? || they_are_rejected?) end def follow! activity_id = ActivityPub::TagManager.instance.generate_uri_for(nil) payload = Oj.dump(follow_activity(activity_id)) update!(active_state: :pending, passive_state: :idle, active_follow_activity_id: activity_id) DeliveryFailureTracker.reset!(inbox_url) ActivityPub::DeliveryWorker.perform_async(payload, some_local_account.id, inbox_url) end def unfollow! activity_id = ActivityPub::TagManager.instance.generate_uri_for(nil) payload = Oj.dump(unfollow_activity(activity_id)) update!(active_state: :idle, passive_state: :idle, active_follow_activity_id: nil) DeliveryFailureTracker.reset!(inbox_url) ActivityPub::DeliveryWorker.perform_async(payload, some_local_account.id, inbox_url) end def accept! return if they_are_idle? activity_id = passive_follow_activity_id payload = Oj.dump(accept_follow_activity(activity_id)) update!(passive_state: :accepted, active_state: :idle) DeliveryFailureTracker.reset!(inbox_url) ActivityPub::DeliveryWorker.perform_async(payload, some_local_account.id, inbox_url) end def reject! return if they_are_idle? activity_id = passive_follow_activity_id payload = Oj.dump(reject_follow_activity(activity_id)) update!(passive_state: :rejected, active_state: :idle, passive_follow_activity_id: nil) DeliveryFailureTracker.reset!(inbox_url) ActivityPub::DeliveryWorker.perform_async(payload, some_local_account.id, inbox_url) end def destroy_without_signal! self.active_state = :idle self.passive_state = :idle destroy! end def default_inbox_url "https://#{domain}/inbox" end private def delete_for_friend! activity_id = ActivityPub::TagManager.instance.generate_uri_for(nil) payload = Oj.dump(delete_follow_activity(activity_id)) DeliveryFailureTracker.reset!(inbox_url) ActivityPub::DeliveryWorker.perform_async(payload, some_local_account.id, inbox_url) end def follow_activity(activity_id) { '@context': ActivityPub::TagManager::CONTEXT, id: activity_id, type: 'Follow', actor: ActivityPub::TagManager.instance.uri_for(some_local_account), object: ActivityPub::TagManager::COLLECTIONS[:public], # Cannot use inbox_url method because this model also has inbox_url column # This is deprecated property. Newer version's kmyblue will ignore it. inboxUrl: "https://#{Rails.configuration.x.web_domain}/inbox", } end def unfollow_activity(activity_id) { '@context': ActivityPub::TagManager::CONTEXT, id: activity_id, type: 'Undo', actor: ActivityPub::TagManager.instance.uri_for(some_local_account), object: { id: active_follow_activity_id, type: 'Follow', actor: ActivityPub::TagManager.instance.uri_for(some_local_account), object: ActivityPub::TagManager::COLLECTIONS[:public], }, } end def accept_follow_activity(activity_id) { '@context': ActivityPub::TagManager::CONTEXT, id: "#{activity_id}#accepts/friends", type: 'Accept', actor: ActivityPub::TagManager.instance.uri_for(some_local_account), object: activity_id, } end def reject_follow_activity(activity_id) { '@context': ActivityPub::TagManager::CONTEXT, id: "#{activity_id}#rejects/friends", type: 'Reject', actor: ActivityPub::TagManager.instance.uri_for(some_local_account), object: activity_id, } end def delete_follow_activity(activity_id) { '@context': ActivityPub::TagManager::CONTEXT, id: "#{activity_id}#delete/friends", type: 'Delete', actor: ActivityPub::TagManager.instance.uri_for(some_local_account), object: ActivityPub::TagManager::COLLECTIONS[:public], } end def some_local_account @some_local_account ||= Account.representative end def ensure_disabled delete_for_friend! unless id.nil? || (i_am_idle? && they_are_idle?) end def set_default_inbox_url self.inbox_url = default_inbox_url if inbox_url.blank? end end