2024-06-03 10:35:59 +02:00
# frozen_string_literal: true
class NotificationGroup < ActiveModelSerializers :: Model
2024-09-02 11:56:00 +02:00
attributes :group_key , :sample_accounts , :notifications_count , :notification , :most_recent_notification_id , :pagination_data
2024-06-03 10:35:59 +02:00
2024-07-18 16:36:09 +02:00
# Try to keep this consistent with `app/javascript/mastodon/models/notification_group.ts`
SAMPLE_ACCOUNTS_SIZE = 8
2024-09-02 11:56:00 +02:00
def self . from_notifications ( notifications , pagination_range : nil , grouped_types : nil )
return [ ] if notifications . empty?
2024-08-29 14:39:07 +02:00
grouped_types = grouped_types . presence & . map ( & :to_sym ) || Notification :: GROUPABLE_NOTIFICATION_TYPES
2024-06-03 10:35:59 +02:00
2024-09-02 11:56:00 +02:00
grouped_notifications = notifications . filter { | notification | notification . group_key . present? && grouped_types . include? ( notification . type ) }
group_keys = grouped_notifications . pluck ( :group_key )
groups_data = load_groups_data ( notifications . first . account_id , group_keys , pagination_range : pagination_range )
accounts_map = Account . where ( id : groups_data . values . pluck ( 1 ) . flatten ) . index_by ( & :id )
notifications . map do | notification |
if notification . group_key . present? && grouped_types . include? ( notification . type )
most_recent_notification_id , sample_account_ids , count , * raw_pagination_data = groups_data [ notification . group_key ]
pagination_data = raw_pagination_data . empty? ? nil : { min_id : raw_pagination_data [ 0 ] , latest_notification_at : raw_pagination_data [ 1 ] }
NotificationGroup . new (
notification : notification ,
group_key : notification . group_key ,
sample_accounts : sample_account_ids . map { | id | accounts_map [ id ] } ,
notifications_count : count ,
most_recent_notification_id : most_recent_notification_id ,
pagination_data : pagination_data
)
else
pagination_data = pagination_range . blank? ? nil : { min_id : notification . id , latest_notification_at : notification . created_at }
NotificationGroup . new (
notification : notification ,
group_key : " ungrouped- #{ notification . id } " ,
sample_accounts : [ notification . from_account ] ,
notifications_count : 1 ,
most_recent_notification_id : notification . id ,
pagination_data : pagination_data
)
end
end
2024-06-03 10:35:59 +02:00
end
delegate :type ,
:target_status ,
:report ,
:account_relationship_severance_event ,
2024-06-20 15:44:49 +02:00
:account_warning ,
2024-11-05 15:40:07 +01:00
:generated_annual_report ,
2024-06-03 10:35:59 +02:00
to : :notification , prefix : false
2024-09-02 11:56:00 +02:00
class << self
private
def load_groups_data ( account_id , group_keys , pagination_range : nil )
return { } if group_keys . empty?
if pagination_range . present?
binds = [
account_id ,
SAMPLE_ACCOUNTS_SIZE ,
ActiveRecord :: Relation :: QueryAttribute . new ( 'group_keys' , group_keys , ActiveRecord :: ConnectionAdapters :: PostgreSQL :: OID :: Array . new ( ActiveModel :: Type :: String . new ) ) ,
2025-01-09 14:47:12 +01:00
pagination_range . begin || 0 ,
2024-09-02 11:56:00 +02:00
]
2025-01-09 14:47:12 +01:00
binds << pagination_range . end unless pagination_range . end . nil?
upper_bound_cond = begin
if pagination_range . end . nil?
''
elsif pagination_range . exclude_end?
'AND id < $5'
else
'AND id <= $5'
end
end
2024-09-02 11:56:00 +02:00
ActiveRecord :: Base . connection . select_all ( << ~ SQL . squish , 'grouped_notifications' , binds ) . cast_values . to_h { | k , * values | [ k , values ] }
SELECT
groups . group_key ,
2025-01-09 14:47:12 +01:00
( SELECT id FROM notifications WHERE notifications . account_id = $1 AND notifications . group_key = groups . group_key #{upper_bound_cond} ORDER BY id DESC LIMIT 1),
array ( SELECT from_account_id FROM notifications WHERE notifications . account_id = $1 AND notifications . group_key = groups . group_key #{upper_bound_cond} ORDER BY id DESC LIMIT $2),
( SELECT count ( * ) FROM notifications WHERE notifications . account_id = $1 AND notifications . group_key = groups . group_key #{upper_bound_cond}) AS notifications_count,
( SELECT id FROM notifications WHERE notifications . account_id = $1 AND notifications . group_key = groups . group_key AND id > = $4 ORDER BY id ASC LIMIT 1 ) AS min_id ,
( SELECT created_at FROM notifications WHERE notifications . account_id = $1 AND notifications . group_key = groups . group_key #{upper_bound_cond} ORDER BY id DESC LIMIT 1)
2024-09-02 11:56:00 +02:00
FROM
2025-01-09 14:47:12 +01:00
unnest ( $3 :: text [ ] ) AS groups ( group_key ) ;
2024-09-02 11:56:00 +02:00
SQL
else
binds = [
account_id ,
SAMPLE_ACCOUNTS_SIZE ,
ActiveRecord :: Relation :: QueryAttribute . new ( 'group_keys' , group_keys , ActiveRecord :: ConnectionAdapters :: PostgreSQL :: OID :: Array . new ( ActiveModel :: Type :: String . new ) ) ,
]
ActiveRecord :: Base . connection . select_all ( << ~ SQL . squish , 'grouped_notifications' , binds ) . cast_values . to_h { | k , * values | [ k , values ] }
SELECT
groups . group_key ,
( SELECT id FROM notifications WHERE notifications . account_id = $1 AND notifications . group_key = groups . group_key ORDER BY id DESC LIMIT 1 ) ,
array ( SELECT from_account_id FROM notifications WHERE notifications . account_id = $1 AND notifications . group_key = groups . group_key ORDER BY id DESC LIMIT $2 ) ,
( SELECT count ( * ) FROM notifications WHERE notifications . account_id = $1 AND notifications . group_key = groups . group_key ) AS notifications_count
FROM
unnest ( $3 :: text [ ] ) AS groups ( group_key ) ;
SQL
end
end
end
2024-06-03 10:35:59 +02:00
end