-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add experimental server-side notification grouping (#29889)
- Loading branch information
1 parent
db49b0e
commit 974335e
Showing
14 changed files
with
618 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,91 @@ | ||
# 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 | ||
|
||
def index | ||
with_read_replica do | ||
@notifications = load_notifications | ||
@group_metadata = load_group_metadata | ||
@relationships = StatusRelationshipsPresenter.new(target_statuses_from_notifications, current_user&.account_id) | ||
end | ||
|
||
render json: @notifications.map { |notification| NotificationGroup.from_notification(notification) }, each_serializer: REST::NotificationGroupSerializer, relationships: @relationships, group_metadata: @group_metadata | ||
end | ||
|
||
def show | ||
@notification = current_account.notifications.without_suspended.find_by!(group_key: params[:id]) | ||
render json: NotificationGroup.from_notification(@notification), serializer: REST::NotificationGroupSerializer | ||
end | ||
|
||
def clear | ||
current_account.notifications.delete_all | ||
render_empty | ||
end | ||
|
||
def dismiss | ||
current_account.notifications.where(group_key: params[:id]).destroy_all | ||
render_empty | ||
end | ||
|
||
private | ||
|
||
def load_notifications | ||
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) | ||
) | ||
|
||
Notification.preload_cache_collection_target_statuses(notifications) do |target_statuses| | ||
preload_collection(target_statuses, Status) | ||
end | ||
end | ||
|
||
def load_group_metadata | ||
return {} if @notifications.empty? | ||
|
||
browserable_account_notifications | ||
.where(group_key: @notifications.filter_map(&:group_key)) | ||
.where(id: (@notifications.last.id)..(@notifications.first.id)) | ||
.group(:group_key) | ||
.pluck(:group_key, 'min(notifications.id) as min_id', 'max(notifications.id) as max_id', 'max(notifications.created_at) as latest_notification_at') | ||
.to_h { |group_key, min_id, max_id, latest_notification_at| [group_key, { min_id: min_id, max_id: max_id, latest_notification_at: latest_notification_at }] } | ||
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 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 | ||
params.permit(:include_filtered, types: [], exclude_types: []) | ||
end | ||
|
||
def pagination_params(core_params) | ||
params.slice(:limit, :types, :exclude_types, :include_filtered).permit(:limit, :include_filtered, types: [], exclude_types: []).merge(core_params) | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,29 @@ | ||
# frozen_string_literal: true | ||
|
||
class NotificationGroup < ActiveModelSerializers::Model | ||
attributes :group_key, :sample_accounts, :notifications_count, :notification | ||
|
||
def self.from_notification(notification) | ||
if notification.group_key.present? | ||
# TODO: caching and preloading | ||
sample_accounts = notification.account.notifications.where(group_key: notification.group_key).order(id: :desc).limit(3).map(&:from_account) | ||
notifications_count = notification.account.notifications.where(group_key: notification.group_key).count | ||
else | ||
sample_accounts = [notification.from_account] | ||
notifications_count = 1 | ||
end | ||
|
||
NotificationGroup.new( | ||
notification: notification, | ||
group_key: notification.group_key || "ungrouped-#{notification.id}", | ||
sample_accounts: sample_accounts, | ||
notifications_count: notifications_count | ||
) | ||
end | ||
|
||
delegate :type, | ||
:target_status, | ||
:report, | ||
:account_relationship_severance_event, | ||
to: :notification, prefix: false | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,45 @@ | ||
# frozen_string_literal: true | ||
|
||
class REST::NotificationGroupSerializer < ActiveModel::Serializer | ||
attributes :group_key, :notifications_count, :type | ||
|
||
attribute :page_min_id, if: :paginated? | ||
attribute :page_max_id, if: :paginated? | ||
attribute :latest_page_notification_at, if: :paginated? | ||
|
||
has_many :sample_accounts, serializer: REST::AccountSerializer | ||
belongs_to :target_status, key: :status, if: :status_type?, serializer: REST::StatusSerializer | ||
belongs_to :report, if: :report_type?, serializer: REST::ReportSerializer | ||
belongs_to :account_relationship_severance_event, key: :event, if: :relationship_severance_event?, serializer: REST::AccountRelationshipSeveranceEventSerializer | ||
|
||
def status_type? | ||
[:favourite, :reblog, :status, :mention, :poll, :update].include?(object.type) | ||
end | ||
|
||
def report_type? | ||
object.type == :'admin.report' | ||
end | ||
|
||
def relationship_severance_event? | ||
object.type == :severed_relationships | ||
end | ||
|
||
def page_min_id | ||
range = instance_options[:group_metadata][object.group_key] | ||
range.present? ? range[:min_id].to_s : object.notification.id.to_s | ||
end | ||
|
||
def page_max_id | ||
range = instance_options[:group_metadata][object.group_key] | ||
range.present? ? range[:max_id].to_s : object.notification.id.to_s | ||
end | ||
|
||
def latest_page_notification_at | ||
range = instance_options[:group_metadata][object.group_key] | ||
range.present? ? range[:latest_notification_at] : object.notification.created_at | ||
end | ||
|
||
def paginated? | ||
instance_options[:group_metadata].present? | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
# frozen_string_literal: true | ||
|
||
class AddGroupKeyToNotifications < ActiveRecord::Migration[7.1] | ||
def change | ||
add_column :notifications, :group_key, :string | ||
end | ||
end |
9 changes: 9 additions & 0 deletions
9
db/migrate/20240513123807_add_index_notifications_on_account_id_and_group_key.rb
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
# frozen_string_literal: true | ||
|
||
class AddIndexNotificationsOnAccountIdAndGroupKey < ActiveRecord::Migration[7.1] | ||
disable_ddl_transaction! | ||
|
||
def change | ||
add_index :notifications, [:account_id, :group_key], algorithm: :concurrently, where: 'group_key IS NOT NULL' | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.