* 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: 編集できない問題 * 有効にしていないフレンドサーバーをリストで無効表示
214 lines
6 KiB
Ruby
214 lines
6 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
class ActivityPub::Activity::Undo < ActivityPub::Activity
|
|
def perform
|
|
case @object['type']
|
|
when 'Announce'
|
|
undo_announce
|
|
when 'Accept'
|
|
undo_accept
|
|
when 'Follow'
|
|
undo_follow
|
|
when 'Like', 'EmojiReaction', 'EmojiReact'
|
|
undo_like
|
|
when 'Block'
|
|
undo_block
|
|
when nil
|
|
handle_reference
|
|
end
|
|
end
|
|
|
|
private
|
|
|
|
def handle_reference
|
|
# Some implementations do not inline the object, and as we don't have a
|
|
# global index, we have to guess what object it is.
|
|
return if object_uri.nil?
|
|
|
|
try_undo_announce || try_undo_accept || try_undo_follow || try_undo_like || try_undo_block || delete_later!(object_uri)
|
|
end
|
|
|
|
def try_undo_announce
|
|
status = Status.where.not(reblog_of_id: nil).find_by(uri: object_uri, account: @account)
|
|
if status.present?
|
|
RemoveStatusService.new.call(status)
|
|
true
|
|
else
|
|
false
|
|
end
|
|
end
|
|
|
|
def try_undo_accept
|
|
# We can't currently handle `Undo Accept` as we don't record `Accept`'s uri
|
|
false
|
|
end
|
|
|
|
def try_undo_follow
|
|
follow = @account.follow_requests.find_by(uri: object_uri) || @account.active_relationships.find_by(uri: object_uri)
|
|
|
|
if follow.present?
|
|
follow.destroy
|
|
true
|
|
else
|
|
false
|
|
end
|
|
end
|
|
|
|
def try_undo_like
|
|
# There is an index on accounts, but an account may have *many* favs, so this may be too costly
|
|
false
|
|
end
|
|
|
|
def try_undo_block
|
|
block = @account.block_relationships.find_by(uri: object_uri)
|
|
if block.present?
|
|
UnblockService.new.call(@account, block.target_account)
|
|
true
|
|
else
|
|
false
|
|
end
|
|
end
|
|
|
|
def undo_announce
|
|
return if object_uri.nil?
|
|
|
|
status = Status.find_by(uri: object_uri, account: @account)
|
|
status ||= Status.find_by(uri: @object['atomUri'], account: @account) if @object.is_a?(Hash) && @object['atomUri'].present?
|
|
|
|
if status.nil?
|
|
delete_later!(object_uri)
|
|
else
|
|
RemoveStatusService.new.call(status)
|
|
end
|
|
end
|
|
|
|
def undo_accept
|
|
::Follow.find_by(target_account: @account, uri: target_uri)&.revoke_request!
|
|
end
|
|
|
|
def undo_follow
|
|
return remove_follow_from_friend if friend_follow?
|
|
|
|
target_account = account_from_uri(target_uri)
|
|
|
|
return if target_account.nil? || !target_account.local?
|
|
|
|
if @account.following?(target_account)
|
|
@account.unfollow!(target_account)
|
|
elsif @account.requested?(target_account)
|
|
FollowRequest.find_by(account: @account, target_account: target_account)&.destroy
|
|
else
|
|
delete_later!(object_uri)
|
|
end
|
|
end
|
|
|
|
def remove_follow_from_friend
|
|
friend.update!(passive_state: :idle, passive_follow_activity_id: nil)
|
|
end
|
|
|
|
def friend
|
|
@friend ||= FriendDomain.find_by(domain: @account.domain) if @account.domain.present? && @object['object'] == ActivityPub::TagManager::COLLECTIONS[:public]
|
|
end
|
|
|
|
def friend_follow?
|
|
friend.present?
|
|
end
|
|
|
|
def undo_like_original
|
|
status = status_from_uri(target_uri)
|
|
|
|
return if status.nil? || !status.account.local?
|
|
|
|
if @account.favourited?(status)
|
|
favourite = status.favourites.where(account: @account).first
|
|
favourite&.destroy
|
|
else
|
|
delete_later!(object_uri)
|
|
end
|
|
end
|
|
|
|
def undo_like
|
|
@original_status = status_from_uri(target_uri)
|
|
|
|
return if @original_status.nil?
|
|
|
|
if shortcode.present?
|
|
emoji_tag = @object['tag'].is_a?(Array) ? @object['tag']&.first : @object['tag']
|
|
|
|
emoji = CustomEmoji.find_by(shortcode: shortcode, domain: @account.domain) if emoji_tag.present? && emoji_tag['id'].present?
|
|
|
|
emoji_reaction = @original_status.emoji_reactions.where(account: @account, name: shortcode, custom_emoji: emoji).first
|
|
|
|
if emoji_reaction
|
|
emoji_reaction.destroy
|
|
write_stream(emoji_reaction)
|
|
|
|
if @original_status.account.local?
|
|
forward_for_undo_emoji_reaction
|
|
relay_for_undo_emoji_reaction
|
|
end
|
|
end
|
|
else
|
|
undo_like_original
|
|
end
|
|
end
|
|
|
|
def write_stream(emoji_reaction)
|
|
emoji_group = @original_status.emoji_reactions_grouped_by_name
|
|
.find { |reaction_group| reaction_group['name'] == emoji_reaction.name && (!reaction_group.key?(:domain) || reaction_group['domain'] == emoji_reaction.custom_emoji&.domain) }
|
|
if emoji_group
|
|
emoji_group['status_id'] = @original_status.id.to_s
|
|
else
|
|
# name: emoji_reaction.name, count: 0, domain: emoji_reaction.domain
|
|
emoji_group = { 'name' => emoji_reaction.name, 'count' => 0, 'account_ids' => [], 'status_id' => @original_status.id.to_s }
|
|
emoji_group['domain'] = emoji_reaction.custom_emoji.domain if emoji_reaction.custom_emoji
|
|
end
|
|
DeliveryEmojiReactionWorker.perform_async(render_emoji_reaction(emoji_group), @original_status.id, emoji_reaction.account_id) if @original_status.local? || Setting.streaming_other_servers_emoji_reaction
|
|
end
|
|
|
|
def render_emoji_reaction(emoji_group)
|
|
@render_emoji_reaction ||= Oj.dump(event: :emoji_reaction, payload: emoji_group.to_json)
|
|
end
|
|
|
|
def forward_for_undo_emoji_reaction
|
|
return if @json['signature'].blank?
|
|
|
|
ActivityPub::RawDistributionWorker.perform_async(Oj.dump(@json), @original_status.account.id, [@account.preferred_inbox_url])
|
|
end
|
|
|
|
def relay_for_undo_emoji_reaction
|
|
return unless @json['signature'].present? && @original_status.public_visibility?
|
|
|
|
ActivityPub::DeliveryWorker.push_bulk(Relay.enabled.pluck(:inbox_url)) do |inbox_url|
|
|
[Oj.dump(@json), @original_status.account.id, inbox_url]
|
|
end
|
|
end
|
|
|
|
def shortcode
|
|
return @shortcode if defined?(@shortcode)
|
|
|
|
@shortcode = begin
|
|
if @object['_misskey_reaction'] == '⭐'
|
|
nil
|
|
else
|
|
@object['content']&.delete(':')
|
|
end
|
|
end
|
|
end
|
|
|
|
def undo_block
|
|
target_account = account_from_uri(target_uri)
|
|
|
|
return if target_account.nil? || !target_account.local?
|
|
|
|
if @account.blocking?(target_account)
|
|
UnblockService.new.call(@account, target_account)
|
|
else
|
|
delete_later!(object_uri)
|
|
end
|
|
end
|
|
|
|
def target_uri
|
|
@target_uri ||= value_or_id(@object['object'])
|
|
end
|
|
end
|