You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
154 lines
5.7 KiB
154 lines
5.7 KiB
# frozen_string_literal: true |
|
|
|
class Api::V2::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 |
|
DEFAULT_NOTIFICATIONS_COUNT_LIMIT = 100 |
|
MAX_NOTIFICATIONS_COUNT_LIMIT = 1_000 |
|
|
|
def index |
|
with_read_replica do |
|
@notifications = load_notifications |
|
@grouped_notifications = load_grouped_notifications |
|
@relationships = StatusRelationshipsPresenter.new(target_statuses_from_notifications, current_user&.account_id) |
|
@presenter = GroupedNotificationsPresenter.new(@grouped_notifications, expand_accounts: expand_accounts_param) |
|
|
|
# Preload associations to avoid N+1s |
|
ActiveRecord::Associations::Preloader.new(records: @presenter.accounts, associations: [:account_stat, { user: :role }]).call |
|
end |
|
|
|
MastodonOTELTracer.in_span('Api::V2::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, |
|
'app.notification_grouping.account.count' => @presenter.accounts.size, |
|
'app.notification_grouping.partial_account.count' => @presenter.partial_accounts.size, |
|
'app.notification_grouping.status.count' => statuses.size, |
|
'app.notification_grouping.status.unique_count' => statuses.uniq.size, |
|
'app.notification_grouping.expand_accounts_param' => expand_accounts_param |
|
) |
|
|
|
render json: @presenter, serializer: REST::DedupNotificationGroupSerializer, relationships: @relationships, expand_accounts: expand_accounts_param |
|
end |
|
end |
|
|
|
def unread_count |
|
limit = limit_param(DEFAULT_NOTIFICATIONS_COUNT_LIMIT, MAX_NOTIFICATIONS_COUNT_LIMIT) |
|
|
|
with_read_replica do |
|
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 } |
|
end |
|
end |
|
|
|
def show |
|
@notification = current_account.notifications.without_suspended.by_group_key(params[:group_key]).take! |
|
presenter = GroupedNotificationsPresenter.new(NotificationGroup.from_notifications([@notification])) |
|
render json: presenter, serializer: REST::DedupNotificationGroupSerializer |
|
end |
|
|
|
def clear |
|
current_account.notifications.delete_all |
|
render_empty |
|
end |
|
|
|
def dismiss |
|
current_account.notifications.by_group_key(params[:group_key]).destroy_all |
|
render_empty |
|
end |
|
|
|
private |
|
|
|
def load_notifications |
|
MastodonOTELTracer.in_span('Api::V2::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), |
|
params.slice(:max_id, :since_id, :min_id, :grouped_types).permit(:max_id, :since_id, :min_id, grouped_types: []) |
|
) |
|
|
|
Notification.preload_cache_collection_target_statuses(notifications) do |target_statuses| |
|
preload_collection(target_statuses, Status) |
|
end |
|
end |
|
end |
|
|
|
def load_grouped_notifications |
|
return [] if @notifications.empty? |
|
|
|
MastodonOTELTracer.in_span('Api::V2::NotificationsController#load_grouped_notifications') do |
|
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]) |
|
end |
|
end |
|
|
|
def incomplete_page? |
|
@notifications.size < limit_param(DEFAULT_NOTIFICATIONS_LIMIT) |
|
end |
|
|
|
def paginating_up? |
|
params[:min_id].present? |
|
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 |
|
|
|
def notification_marker |
|
current_user.markers.find_by(timeline: 'notifications') |
|
end |
|
|
|
def target_statuses_from_notifications |
|
@notifications.filter_map(&:target_status) |
|
end |
|
|
|
def next_path |
|
api_v2_notifications_url pagination_params(max_id: pagination_max_id) unless @notifications.empty? |
|
end |
|
|
|
def prev_path |
|
api_v2_notifications_url pagination_params(min_id: pagination_since_id) unless @notifications.empty? |
|
end |
|
|
|
def pagination_collection |
|
@notifications |
|
end |
|
|
|
def browserable_params |
|
params.slice(:include_filtered, :types, :exclude_types, :grouped_types).permit(:include_filtered, types: [], exclude_types: [], grouped_types: []) |
|
end |
|
|
|
def pagination_params(core_params) |
|
params.slice(:limit, :include_filtered, :types, :exclude_types, :grouped_types).permit(:limit, :include_filtered, types: [], exclude_types: [], grouped_types: []).merge(core_params) |
|
end |
|
|
|
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 |
|
end
|
|
|