2024-06-03 18:35:59 +10:00
# frozen_string_literal: true
class Api :: V2Alpha :: NotificationsController < Api :: BaseController
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 18:39:11 +10:00
DEFAULT_NOTIFICATIONS_COUNT_LIMIT = 100
MAX_NOTIFICATIONS_COUNT_LIMIT = 1_000
2024-06-03 18:35:59 +10:00
def index
with_read_replica do
@notifications = load_notifications
2024-07-19 00:36:09 +10:00
@grouped_notifications = load_grouped_notifications
2024-06-03 18:35:59 +10:00
@relationships = StatusRelationshipsPresenter . new ( target_statuses_from_notifications , current_user & . account_id )
2024-08-06 22:09:35 +10:00
@presenter = GroupedNotificationsPresenter . new ( @grouped_notifications , expand_accounts : expand_accounts_param )
2024-07-19 00:36:09 +10:00
# Preload associations to avoid N+1s
2024-08-06 22:09:35 +10:00
ActiveRecord :: Associations :: Preloader . new ( records : @presenter . accounts , associations : [ :account_stat , { user : :role } ] ) . call
2024-06-03 18:35:59 +10:00
end
2024-07-19 00:36:09 +10:00
MastodonOTELTracer . in_span ( 'Api::V2Alpha::NotificationsController#index rendering' ) do | span |
statuses = @grouped_notifications . filter_map { | group | group . target_status & . id }
span . add_attributes (
'app.notification_grouping.count' = > @grouped_notifications . size ,
2024-08-06 22:09:35 +10:00
'app.notification_grouping.account.count' = > @presenter . accounts . size ,
'app.notification_grouping.partial_account.count' = > @presenter . partial_accounts . size ,
2024-07-19 00:36:09 +10:00
'app.notification_grouping.status.count' = > statuses . size ,
2024-08-06 22:09:35 +10:00
'app.notification_grouping.status.unique_count' = > statuses . uniq . size ,
'app.notification_grouping.expand_accounts_param' = > expand_accounts_param
2024-07-19 00:36:09 +10:00
)
2024-09-02 19:56:00 +10:00
render json : @presenter , serializer : REST :: DedupNotificationGroupSerializer , relationships : @relationships , expand_accounts : expand_accounts_param
2024-07-19 00:36:09 +10:00
end
2024-06-03 18:35:59 +10:00
end
2024-07-30 18:39:11 +10:00
def unread_count
limit = limit_param ( DEFAULT_NOTIFICATIONS_COUNT_LIMIT , MAX_NOTIFICATIONS_COUNT_LIMIT )
with_read_replica do
2024-08-29 22:39:07 +10: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 18:39:11 +10:00
end
end
2024-06-03 18:35:59 +10:00
def show
2024-09-04 00:32:26 +10:00
@notification = current_account . notifications . without_suspended . find_by! ( group_key : params [ :group_key ] )
2024-09-02 19:56:00 +10:00
presenter = GroupedNotificationsPresenter . new ( NotificationGroup . from_notifications ( [ @notification ] ) )
2024-07-31 20:50:13 +10:00
render json : presenter , serializer : REST :: DedupNotificationGroupSerializer
2024-06-03 18:35:59 +10:00
end
def clear
current_account . notifications . delete_all
render_empty
end
def dismiss
2024-09-04 00:32:26 +10:00
current_account . notifications . where ( group_key : params [ :group_key ] ) . destroy_all
2024-06-03 18:35:59 +10:00
render_empty
end
private
def load_notifications
2024-07-19 00:36:09 +10:00
MastodonOTELTracer . in_span ( 'Api::V2Alpha::NotificationsController#load_notifications' ) do
notifications = browserable_account_notifications . includes ( from_account : [ :account_stat , :user ] ) . to_a_grouped_paginated_by_id (
limit_param ( DEFAULT_NOTIFICATIONS_LIMIT ) ,
2024-08-29 22:39:07 +10:00
params . slice ( :max_id , :since_id , :min_id , :grouped_types ) . permit ( :max_id , :since_id , :min_id , grouped_types : [ ] )
2024-07-19 00:36:09 +10:00
)
Notification . preload_cache_collection_target_statuses ( notifications ) do | target_statuses |
preload_collection ( target_statuses , Status )
end
2024-06-03 18:35:59 +10:00
end
end
2024-07-19 00:36:09 +10:00
def load_grouped_notifications
2024-09-04 22:54:15 +10:00
return [ ] if @notifications . empty?
2024-07-19 00:36:09 +10:00
MastodonOTELTracer . in_span ( 'Api::V2Alpha::NotificationsController#load_grouped_notifications' ) do
2024-09-02 19:56:00 +10:00
NotificationGroup . from_notifications ( @notifications , pagination_range : ( @notifications . last . id ) .. ( @notifications . first . id ) , grouped_types : params [ :grouped_types ] )
2024-07-19 00:36:09 +10:00
end
2024-06-03 18:35:59 +10:00
end
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 18:39:11 +10:00
def notification_marker
current_user . markers . find_by ( timeline : 'notifications' )
end
2024-06-03 18:35:59 +10:00
def target_statuses_from_notifications
@notifications . filter_map ( & :target_status )
end
def next_path
api_v2_alpha_notifications_url pagination_params ( max_id : pagination_max_id ) unless @notifications . empty?
end
def prev_path
api_v2_alpha_notifications_url pagination_params ( min_id : pagination_since_id ) unless @notifications . empty?
end
def pagination_collection
@notifications
end
def browserable_params
2024-08-29 22:39:07 +10:00
params . slice ( :include_filtered , :types , :exclude_types , :grouped_types ) . permit ( :include_filtered , types : [ ] , exclude_types : [ ] , grouped_types : [ ] )
2024-06-03 18:35:59 +10:00
end
def pagination_params ( core_params )
2024-08-29 22:39:07 +10: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 18:35:59 +10:00
end
2024-08-06 22:09:35 +10: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 18:35:59 +10:00
end