2024-06-03 10:35:59 +02:00
# frozen_string_literal: true
2024-09-16 14:10:02 +02:00
class Api :: V2 :: NotificationsController < Api :: BaseController
2024-06-03 10:35:59 +02:00
before_action - > { doorkeeper_authorize! :read , :'read:notifications' } , except : [ :clear , :dismiss ]
before_action - > { doorkeeper_authorize! :write , :'write:notifications' } , only : [ :clear , :dismiss ]
before_action :require_user!
after_action :insert_pagination_headers , only : :index
DEFAULT_NOTIFICATIONS_LIMIT = 40
2024-07-30 10:39:11 +02:00
DEFAULT_NOTIFICATIONS_COUNT_LIMIT = 100
MAX_NOTIFICATIONS_COUNT_LIMIT = 1_000
2024-06-03 10:35:59 +02:00
def index
with_read_replica do
@notifications = load_notifications
2024-07-18 16:36:09 +02:00
@grouped_notifications = load_grouped_notifications
2024-06-03 10:35:59 +02:00
@relationships = StatusRelationshipsPresenter . new ( target_statuses_from_notifications , current_user & . account_id )
2024-08-06 14:09:35 +02:00
@presenter = GroupedNotificationsPresenter . new ( @grouped_notifications , expand_accounts : expand_accounts_param )
2024-07-18 16:36:09 +02:00
# Preload associations to avoid N+1s
2024-08-06 14:09:35 +02:00
ActiveRecord :: Associations :: Preloader . new ( records : @presenter . accounts , associations : [ :account_stat , { user : :role } ] ) . call
2024-06-03 10:35:59 +02:00
end
2024-09-16 14:10:02 +02:00
MastodonOTELTracer . in_span ( 'Api::V2::NotificationsController#index rendering' ) do | span |
2024-07-18 16:36:09 +02:00
statuses = @grouped_notifications . filter_map { | group | group . target_status & . id }
span . add_attributes (
'app.notification_grouping.count' = > @grouped_notifications . size ,
2024-08-06 14:09:35 +02:00
'app.notification_grouping.account.count' = > @presenter . accounts . size ,
'app.notification_grouping.partial_account.count' = > @presenter . partial_accounts . size ,
2024-07-18 16:36:09 +02:00
'app.notification_grouping.status.count' = > statuses . size ,
2024-08-06 14:09:35 +02:00
'app.notification_grouping.status.unique_count' = > statuses . uniq . size ,
'app.notification_grouping.expand_accounts_param' = > expand_accounts_param
2024-07-18 16:36:09 +02:00
)
2024-09-02 11:56:00 +02:00
render json : @presenter , serializer : REST :: DedupNotificationGroupSerializer , relationships : @relationships , expand_accounts : expand_accounts_param
2024-07-18 16:36:09 +02:00
end
2024-06-03 10:35:59 +02:00
end
2024-07-30 10:39:11 +02:00
def unread_count
limit = limit_param ( DEFAULT_NOTIFICATIONS_COUNT_LIMIT , MAX_NOTIFICATIONS_COUNT_LIMIT )
with_read_replica do
2024-08-29 14:39:07 +02:00
render json : { count : browserable_account_notifications . paginate_groups_by_min_id ( limit , min_id : notification_marker & . last_read_id , grouped_types : params [ :grouped_types ] ) . count }
2024-07-30 10:39:11 +02:00
end
end
2024-06-03 10:35:59 +02:00
def show
2024-09-03 16:32:26 +02:00
@notification = current_account . notifications . without_suspended . find_by! ( group_key : params [ :group_key ] )
2024-09-02 11:56:00 +02:00
presenter = GroupedNotificationsPresenter . new ( NotificationGroup . from_notifications ( [ @notification ] ) )
2024-07-31 12:50:13 +02:00
render json : presenter , serializer : REST :: DedupNotificationGroupSerializer
2024-06-03 10:35:59 +02:00
end
def clear
current_account . notifications . delete_all
render_empty
end
def dismiss
2024-09-03 16:32:26 +02:00
current_account . notifications . where ( group_key : params [ :group_key ] ) . destroy_all
2024-06-03 10:35:59 +02:00
render_empty
end
private
def load_notifications
2024-09-16 14:10:02 +02:00
MastodonOTELTracer . in_span ( 'Api::V2::NotificationsController#load_notifications' ) do
2024-07-18 16:36:09 +02:00
notifications = browserable_account_notifications . includes ( from_account : [ :account_stat , :user ] ) . to_a_grouped_paginated_by_id (
limit_param ( DEFAULT_NOTIFICATIONS_LIMIT ) ,
2024-08-29 14:39:07 +02:00
params . slice ( :max_id , :since_id , :min_id , :grouped_types ) . permit ( :max_id , :since_id , :min_id , grouped_types : [ ] )
2024-07-18 16:36:09 +02:00
)
Notification . preload_cache_collection_target_statuses ( notifications ) do | target_statuses |
preload_collection ( target_statuses , Status )
end
2024-06-03 10:35:59 +02:00
end
end
2024-07-18 16:36:09 +02:00
def load_grouped_notifications
2024-09-04 14:54:15 +02:00
return [ ] if @notifications . empty?
2024-09-16 14:10:02 +02:00
MastodonOTELTracer . in_span ( 'Api::V2::NotificationsController#load_grouped_notifications' ) do
2025-01-09 14:47:12 +01:00
pagination_range = ( @notifications . last . id ) .. @notifications . first . id
# If the page is incomplete, we know we are on the last page
if incomplete_page?
if paginating_up?
pagination_range = @notifications . last . id ... ( params [ :max_id ] & . to_i )
else
range_start = params [ :since_id ] & . to_i
range_start += 1 unless range_start . nil?
pagination_range = range_start .. ( @notifications . first . id )
end
end
NotificationGroup . from_notifications ( @notifications , pagination_range : pagination_range , grouped_types : params [ :grouped_types ] )
2024-07-18 16:36:09 +02:00
end
2024-06-03 10:35:59 +02:00
end
2025-01-09 14:47:12 +01:00
def incomplete_page?
@notifications . size < limit_param ( DEFAULT_NOTIFICATIONS_LIMIT )
end
def paginating_up?
params [ :min_id ] . present?
end
2024-06-03 10:35:59 +02:00
def browserable_account_notifications
current_account . notifications . without_suspended . browserable (
types : Array ( browserable_params [ :types ] ) ,
exclude_types : Array ( browserable_params [ :exclude_types ] ) ,
include_filtered : truthy_param? ( :include_filtered )
)
end
2024-07-30 10:39:11 +02:00
def notification_marker
current_user . markers . find_by ( timeline : 'notifications' )
end
2024-06-03 10:35:59 +02:00
def target_statuses_from_notifications
@notifications . filter_map ( & :target_status )
end
def next_path
2024-09-16 14:10:02 +02:00
api_v2_notifications_url pagination_params ( max_id : pagination_max_id ) unless @notifications . empty?
2024-06-03 10:35:59 +02:00
end
def prev_path
2024-09-16 14:10:02 +02:00
api_v2_notifications_url pagination_params ( min_id : pagination_since_id ) unless @notifications . empty?
2024-06-03 10:35:59 +02:00
end
def pagination_collection
@notifications
end
def browserable_params
2024-08-29 14:39:07 +02:00
params . slice ( :include_filtered , :types , :exclude_types , :grouped_types ) . permit ( :include_filtered , types : [ ] , exclude_types : [ ] , grouped_types : [ ] )
2024-06-03 10:35:59 +02:00
end
def pagination_params ( core_params )
2024-08-29 14:39:07 +02:00
params . slice ( :limit , :include_filtered , :types , :exclude_types , :grouped_types ) . permit ( :limit , :include_filtered , types : [ ] , exclude_types : [ ] , grouped_types : [ ] ) . merge ( core_params )
2024-06-03 10:35:59 +02:00
end
2024-08-06 14:09:35 +02:00
def expand_accounts_param
case params [ :expand_accounts ]
when nil , 'full'
'full'
when 'partial_avatars'
'partial_avatars'
else
raise Mastodon :: InvalidParameterError , " Invalid value for 'expand_accounts': ' #{ params [ :expand_accounts ] } ', allowed values are 'full' and 'partial_avatars' "
end
end
2024-06-03 10:35:59 +02:00
end