2022-11-02 21:41:30 +08:00
# frozen_string_literal: true
module Chat::ChatChannelFetcher
MAX_PUBLIC_CHANNEL_RESULTS = 50
def self . structured ( guardian )
memberships = Chat :: ChatChannelMembershipManager . all_for_user ( guardian . user )
{
public_channels :
secured_public_channels ( guardian , memberships , status : :open , following : true ) ,
direct_message_channels :
secured_direct_message_channels ( guardian . user . id , memberships , guardian ) ,
memberships : memberships ,
}
end
def self . all_secured_channel_ids ( guardian , following : true )
allowed_channel_ids_sql = generate_allowed_channel_ids_sql ( guardian )
return DB . query_single ( allowed_channel_ids_sql ) if ! following
DB . query_single ( << ~ SQL , user_id : guardian . user . id )
SELECT chat_channel_id
FROM user_chat_channel_memberships
WHERE user_chat_channel_memberships . user_id = :user_id
AND user_chat_channel_memberships . chat_channel_id IN (
#{allowed_channel_ids_sql}
)
SQL
end
def self . generate_allowed_channel_ids_sql ( guardian )
<< ~ SQL
- - secured category chat channels
#{
ChatChannel
. select ( :id )
. joins (
" INNER JOIN categories ON categories.id = chat_channels.chatable_id AND chat_channels.chatable_type = 'Category' " ,
)
. where (
" categories.id IN (:allowed_category_ids) " ,
allowed_category_ids : guardian . allowed_category_ids ,
)
. to_sql
}
UNION
- - secured direct message chat channels
#{
ChatChannel
. select ( :id )
. joins (
" INNER JOIN direct_message_channels ON direct_message_channels.id = chat_channels.chatable_id
2022-11-02 22:53:36 +08:00
AND chat_channels . chatable_type = 'DirectMessage'
2022-11-02 21:41:30 +08:00
INNER JOIN direct_message_users ON direct_message_users . direct_message_channel_id = direct_message_channels . id " ,
)
. where ( " direct_message_users.user_id = :user_id " , user_id : guardian . user . id )
. to_sql
}
SQL
end
def self . secured_public_channel_search ( guardian , options = { } )
channels =
ChatChannel
. includes ( :chat_channel_archive )
. includes ( chatable : [ :topic_only_relative_url ] )
. joins (
" LEFT JOIN categories ON categories.id = chat_channels.chatable_id AND chat_channels.chatable_type = 'Category' " ,
)
. where ( chatable_type : ChatChannel . public_channel_chatable_types )
. where ( " chat_channels.id IN ( #{ generate_allowed_channel_ids_sql ( guardian ) } ) " )
channels = channels . where ( status : options [ :status ] ) if options [ :status ] . present?
if options [ :filter ] . present?
sql = " chat_channels.name ILIKE :filter OR categories.name ILIKE :filter "
channels =
channels . where ( sql , filter : " % #{ options [ :filter ] . downcase } % " ) . order (
" chat_channels.name ASC, categories.name ASC " ,
)
end
if options . key? ( :following )
if options [ :following ]
channels =
channels . joins ( :user_chat_channel_memberships ) . where (
user_chat_channel_memberships : {
user_id : guardian . user . id ,
following : true ,
} ,
)
else
channels =
channels . where (
" chat_channels.id NOT IN (SELECT chat_channel_id FROM user_chat_channel_memberships uccm WHERE uccm.chat_channel_id = chat_channels.id AND following IS TRUE AND user_id = ?) " ,
guardian . user . id ,
)
end
end
options [ :limit ] = ( options [ :limit ] || MAX_PUBLIC_CHANNEL_RESULTS ) . to_i . clamp (
1 ,
MAX_PUBLIC_CHANNEL_RESULTS ,
)
options [ :offset ] = [ options [ :offset ] . to_i , 0 ] . max
channels . limit ( options [ :limit ] ) . offset ( options [ :offset ] )
end
def self . secured_public_channels ( guardian , memberships , options = { following : true } )
channels = secured_public_channel_search ( guardian , options )
decorate_memberships_with_tracking_data ( guardian , channels , memberships )
channels = channels . to_a
preload_custom_fields_for ( channels )
channels
end
def self . preload_custom_fields_for ( channels )
preload_fields = Category . instance_variable_get ( :@custom_field_types ) . keys
Category . preload_custom_fields (
channels . select { | c | c . chatable_type == " Category " } . map ( & :chatable ) ,
preload_fields ,
)
end
def self . secured_direct_message_channels ( user_id , memberships , guardian )
query = ChatChannel . includes ( chatable : [ { direct_message_users : :user } , :users ] )
query = query . includes ( chatable : [ { users : :user_status } ] ) if SiteSetting . enable_user_status
channels =
query
. joins ( :user_chat_channel_memberships )
. where ( user_chat_channel_memberships : { user_id : user_id , following : true } )
2022-11-02 22:53:36 +08:00
. where ( chatable_type : " DirectMessage " )
2022-11-02 21:41:30 +08:00
. where ( " chat_channels.id IN ( #{ generate_allowed_channel_ids_sql ( guardian ) } ) " )
. order ( last_message_sent_at : :desc )
. to_a
preload_fields =
User . allowed_user_custom_fields ( guardian ) +
UserField . all . pluck ( :id ) . map { | fid | " #{ User :: USER_FIELD_PREFIX } #{ fid } " }
User . preload_custom_fields ( channels . map { | c | c . chatable . users } . flatten , preload_fields )
decorate_memberships_with_tracking_data ( guardian , channels , memberships )
end
def self . decorate_memberships_with_tracking_data ( guardian , channels , memberships )
unread_counts_per_channel = unread_counts ( channels , guardian . user . id )
mention_notifications =
Notification . unread . where (
user_id : guardian . user . id ,
notification_type : Notification . types [ :chat_mention ] ,
)
mention_notification_data = mention_notifications . map { | m | JSON . parse ( m . data ) }
channels . each do | channel |
membership = memberships . find { | m | m . chat_channel_id == channel . id }
if membership
membership . unread_mentions =
mention_notification_data . count do | data |
data [ " chat_channel_id " ] == channel . id &&
data [ " chat_message_id " ] > ( membership . last_read_message_id || 0 )
end
membership . unread_count = unread_counts_per_channel [ channel . id ] if ! membership . muted
end
end
end
def self . unread_counts ( channels , user_id )
unread_counts = DB . query_array ( << ~ SQL , channel_ids : channels . map ( & :id ) , user_id : user_id ) . to_h
SELECT cc . id , COUNT ( * ) as count
FROM chat_messages cm
JOIN chat_channels cc ON cc . id = cm . chat_channel_id
JOIN user_chat_channel_memberships uccm ON uccm . chat_channel_id = cc . id
WHERE cc . id IN ( :channel_ids )
AND cm . user_id != :user_id
AND uccm . user_id = :user_id
AND cm . id > COALESCE ( uccm . last_read_message_id , 0 )
AND cm . deleted_at IS NULL
GROUP BY cc . id
SQL
unread_counts . default = 0
unread_counts
end
def self . find_with_access_check ( channel_id_or_name , guardian )
begin
channel_id_or_name = Integer ( channel_id_or_name )
rescue ArgumentError
end
base_channel_relation =
ChatChannel . includes ( :chatable ) . joins (
" LEFT JOIN categories ON categories.id = chat_channels.chatable_id AND chat_channels.chatable_type = 'Category' " ,
)
if guardian . user . staff?
base_channel_relation = base_channel_relation . includes ( :chat_channel_archive )
end
if channel_id_or_name . is_a? Integer
chat_channel = base_channel_relation . find_by ( id : channel_id_or_name )
else
chat_channel =
base_channel_relation . find_by (
" LOWER(categories.name) = :name OR LOWER(chat_channels.name) = :name " ,
name : channel_id_or_name . downcase ,
)
end
raise Discourse :: NotFound if chat_channel . blank?
raise Discourse :: InvalidAccess if ! guardian . can_see_chat_channel? ( chat_channel )
chat_channel
end
end