* Fix mastodon version * テーブル作成 * Wip: フレンドサーバーフォローの承認を受信 * Wip: フレンド申請拒否を受信 * Wip: フォローリクエストを受理 * Wip: 相手からのフォロー・アンフォローを受理 * 普通のフォローとフレンドサーバーのフォローを区別するテストを追加 * ドメインブロックによるフォロー拒否 * ドメインブロックしたあと、申請中のフォロリクを取り下げる処理 * スタブに条件を追加 * Wip: 相手からのDelete信号に対応 * DB定義が消えていたので修正 * Wip: ローカル公開投稿をフレンドに送信する処理など * Wip: 未収載+誰でもの投稿をフレンドに送る設定 * Wip: ローカル公開をそのまま送信する設定を考慮 * Fix test * Wip: 他サーバーからのローカル公開投稿の受け入れ * Wip: Web画面作成 * Fix test * Wip: ローカル公開を連合TLに流す * Wip: フレンドサーバーの削除ボタン * Wip: メール通知や設定のテストなど * Wip: 翻訳を作成 * Fix: 却下されたあとフォローボタンが表示されない問題 * Wip: 編集できない問題 * 有効にしていないフレンドサーバーをリストで無効表示
163 lines
5.5 KiB
Ruby
163 lines
5.5 KiB
Ruby
# 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
|
|
# unlocked :boolean default(FALSE), not null
|
|
# allow_all_posts :boolean default(TRUE), not null
|
|
# created_at :datetime not null
|
|
# updated_at :datetime 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(available: true) }
|
|
scope :mutuals, -> { enabled.where(active_state: :accepted, passive_state: :accepted) }
|
|
scope :distributables, -> { mutuals.where(pseudo_relay: true) }
|
|
scope :deliver_locals, -> { enabled.where(active_state: :accepted) }
|
|
scope :free_receivings, -> { mutuals.where(allow_all_posts: true) }
|
|
|
|
before_destroy :ensure_disabled
|
|
after_commit :set_default_inbox_url
|
|
|
|
def mutual?
|
|
i_am_accepted? && they_are_accepted?
|
|
end
|
|
|
|
def follow!
|
|
activity_id = ActivityPub::TagManager.instance.generate_uri_for(nil)
|
|
payload = Oj.dump(follow_activity(activity_id))
|
|
|
|
update!(active_state: :pending, 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, 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)
|
|
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, passive_follow_activity_id: nil)
|
|
DeliveryFailureTracker.reset!(inbox_url)
|
|
ActivityPub::DeliveryWorker.perform_async(payload, some_local_account.id, inbox_url)
|
|
end
|
|
|
|
private
|
|
|
|
def default_inbox_url
|
|
"https://#{domain}/inbox"
|
|
end
|
|
|
|
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],
|
|
}
|
|
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 i_am_idle? && they_are_idle?
|
|
end
|
|
|
|
def set_default_inbox_url
|
|
self.inbox_url = default_inbox_url if inbox_url.blank?
|
|
end
|
|
end
|