notify_service.rb 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. # frozen_string_literal: true
  2. class NotifyService < BaseService
  3. def call(recipient, activity)
  4. @recipient = recipient
  5. @activity = activity
  6. @notification = Notification.new(account: @recipient, activity: @activity)
  7. return if recipient.user.nil? || blocked?
  8. create_notification!
  9. push_notification! if @notification.browserable?
  10. push_to_conversation! if direct_message?
  11. send_email! if email_enabled?
  12. rescue ActiveRecord::RecordInvalid
  13. return
  14. end
  15. private
  16. def blocked_mention?
  17. FeedManager.instance.filter?(:mentions, @notification.mention.status, @recipient.id)
  18. end
  19. def blocked_favourite?
  20. false
  21. end
  22. def blocked_follow?
  23. false
  24. end
  25. def blocked_reblog?
  26. false
  27. end
  28. def blocked_follow_request?
  29. false
  30. end
  31. def following_sender?
  32. return @following_sender if defined?(@following_sender)
  33. @following_sender = @recipient.following?(@notification.from_account) || @recipient.requested?(@notification.from_account)
  34. end
  35. def optional_non_follower?
  36. @recipient.user.settings.interactions['must_be_follower'] && !@notification.from_account.following?(@recipient)
  37. end
  38. def optional_non_following?
  39. @recipient.user.settings.interactions['must_be_following'] && !following_sender?
  40. end
  41. def message?
  42. @notification.type == :mention
  43. end
  44. def direct_message?
  45. message? && @notification.target_status.direct_visibility?
  46. end
  47. def response_to_recipient?
  48. @notification.target_status.in_reply_to_account_id == @recipient.id && @notification.target_status.thread&.direct_visibility?
  49. end
  50. def from_staff?
  51. @notification.from_account.local? && @notification.from_account.user.present? && @notification.from_account.user.staff?
  52. end
  53. def optional_non_following_and_direct?
  54. direct_message? &&
  55. @recipient.user.settings.interactions['must_be_following_dm'] &&
  56. !following_sender? &&
  57. !response_to_recipient?
  58. end
  59. def hellbanned?
  60. @notification.from_account.silenced? && !following_sender?
  61. end
  62. def from_self?
  63. @recipient.id == @notification.from_account.id
  64. end
  65. def domain_blocking?
  66. @recipient.domain_blocking?(@notification.from_account.domain) && !following_sender?
  67. end
  68. def blocked?
  69. blocked = @recipient.suspended? # Skip if the recipient account is suspended anyway
  70. blocked ||= from_self? # Skip for interactions with self
  71. return blocked if message? && from_staff?
  72. blocked ||= domain_blocking? # Skip for domain blocked accounts
  73. blocked ||= @recipient.blocking?(@notification.from_account) # Skip for blocked accounts
  74. blocked ||= @recipient.muting_notifications?(@notification.from_account)
  75. blocked ||= hellbanned? # Hellban
  76. blocked ||= optional_non_follower? # Options
  77. blocked ||= optional_non_following? # Options
  78. blocked ||= optional_non_following_and_direct? # Options
  79. blocked ||= conversation_muted?
  80. blocked ||= send("blocked_#{@notification.type}?") # Type-dependent filters
  81. blocked
  82. end
  83. def conversation_muted?
  84. if @notification.target_status
  85. @recipient.muting_conversation?(@notification.target_status.conversation)
  86. else
  87. false
  88. end
  89. end
  90. def create_notification!
  91. @notification.save!
  92. end
  93. def push_notification!
  94. return if @notification.activity.nil?
  95. Redis.current.publish("timeline:#{@recipient.id}", Oj.dump(event: :notification, payload: InlineRenderer.render(@notification, @recipient, :notification)))
  96. send_push_notifications!
  97. end
  98. def push_to_conversation!
  99. return if @notification.activity.nil?
  100. AccountConversation.add_status(@recipient, @notification.target_status)
  101. end
  102. def send_push_notifications!
  103. subscriptions_ids = ::Web::PushSubscription.where(user_id: @recipient.user.id)
  104. .select { |subscription| subscription.pushable?(@notification) }
  105. .map(&:id)
  106. ::Web::PushNotificationWorker.push_bulk(subscriptions_ids) do |subscription_id|
  107. [subscription_id, @notification.id]
  108. end
  109. end
  110. def send_email!
  111. return if @notification.activity.nil?
  112. NotificationMailer.public_send(@notification.type, @recipient, @notification).deliver_later(wait: 2.minutes)
  113. end
  114. def email_enabled?
  115. @recipient.user.settings.notification_emails[@notification.type.to_s]
  116. end
  117. end