# frozen_string_literal: true class UnEmojiReactService < BaseService include Redisable include Payloadable def call(account, status, emoji_reaction = nil) if emoji_reaction emoji_reaction.destroy create_notification(emoji_reaction) if !status.account.local? && status.account.activitypub? notify_to_followers(emoji_reaction) if status.account.local? write_stream(emoji_reaction) else bulk(account, status) end emoji_reaction end private def bulk(account, status) EmojiReaction.where(account: account).where(status: status).tap do |emoji_reaction| call(account, status, emoji_reaction) end end def create_notification(emoji_reaction) status = emoji_reaction.status ActivityPub::DeliveryWorker.perform_async(build_json(emoji_reaction), status.account_id, status.account.inbox_url) end def notify_to_followers(emoji_reaction) status = emoji_reaction.status ActivityPub::RawDistributionWorker.perform_async(build_json(emoji_reaction), status.account_id) end def write_stream(emoji_reaction) emoji_group = emoji_reaction.status.emoji_reactions_grouped_by_name .find { |reaction_group| reaction_group['name'] == emoji_reaction.name && (!reaction_group.key?(:domain) || reaction_group['domain'] == emoji_reaction.domain) } if emoji_group emoji_group['status_id'] = emoji_reaction.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' => emoji_reaction.status_id.to_s } emoji_group['domain'] = emoji_reaction.custom_emoji.domain if emoji_reaction.custom_emoji end redis.publish("timeline:#{emoji_reaction.status.account_id}", render_emoji_reaction(emoji_reaction, emoji_group)) end def build_json(emoji_reaction) Oj.dump(serialize_payload(emoji_reaction, ActivityPub::UndoEmojiReactionSerializer)) end def render_emoji_reaction(_emoji_reaction, emoji_group) # @rendered_emoji_reaction ||= InlineRenderer.render(emoji_group, nil, :emoji_reaction) Oj.dump(event: :emoji_reaction, payload: emoji_group.to_json) end end