185 lines
5.9 KiB
Ruby
185 lines
5.9 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module Chat
|
|
# Gets a list of threads for a channel to be shown in an index.
|
|
# In future pagination and filtering will be added -- for now
|
|
# we just want to return N threads ordered by the latest
|
|
# message that a user has sent in a thread.
|
|
#
|
|
# Only threads that the user is a member of with a notification level
|
|
# of normal or tracking will be returned.
|
|
#
|
|
# @example
|
|
# Chat::LookupChannelThreads.call(channel_id: 2, guardian: guardian)
|
|
#
|
|
class LookupChannelThreads
|
|
include Service::Base
|
|
|
|
MAX_THREADS = 50
|
|
|
|
# @!method call(channel_id:, guardian:)
|
|
# @param [Integer] channel_id
|
|
# @param [Guardian] guardian
|
|
# @return [Service::Base::Context]
|
|
|
|
policy :threaded_discussions_enabled
|
|
contract
|
|
model :channel
|
|
policy :threading_enabled_for_channel
|
|
policy :can_view_channel
|
|
model :threads
|
|
step :fetch_tracking
|
|
step :fetch_memberships
|
|
|
|
# @!visibility private
|
|
class Contract
|
|
attribute :channel_id, :integer
|
|
validates :channel_id, presence: true
|
|
end
|
|
|
|
private
|
|
|
|
def threaded_discussions_enabled
|
|
SiteSetting.enable_experimental_chat_threaded_discussions
|
|
end
|
|
|
|
def fetch_channel(contract:, **)
|
|
Chat::Channel.find_by(id: contract.channel_id)
|
|
end
|
|
|
|
def threading_enabled_for_channel(channel:, **)
|
|
channel.threading_enabled
|
|
end
|
|
|
|
def can_view_channel(guardian:, channel:, **)
|
|
guardian.can_preview_chat_channel?(channel)
|
|
end
|
|
|
|
def fetch_threads(guardian:, channel:, **)
|
|
read_threads = []
|
|
|
|
unread_threads =
|
|
threads_query(guardian, channel)
|
|
.where(<<~SQL)
|
|
user_chat_thread_memberships_chat_threads.last_read_message_id IS NULL
|
|
OR tracked_threads_subquery.latest_message_id > user_chat_thread_memberships_chat_threads.last_read_message_id
|
|
SQL
|
|
.order("tracked_threads_subquery.latest_message_created_at DESC")
|
|
.limit(MAX_THREADS)
|
|
.to_a
|
|
|
|
# We do this to avoid having to query additional threads if the user
|
|
# already has a lot of unread threads.
|
|
if unread_threads.length < MAX_THREADS
|
|
final_limit = MAX_THREADS - unread_threads.length
|
|
read_threads =
|
|
threads_query(guardian, channel)
|
|
.where(<<~SQL)
|
|
tracked_threads_subquery.latest_message_id <= user_chat_thread_memberships_chat_threads.last_read_message_id
|
|
SQL
|
|
.order("tracked_threads_subquery.latest_message_created_at DESC")
|
|
.limit(final_limit)
|
|
.to_a
|
|
end
|
|
|
|
threads = unread_threads + read_threads
|
|
|
|
if threads.present?
|
|
last_replies =
|
|
Chat::Message
|
|
.strict_loading
|
|
.includes(:user, :uploads)
|
|
.from(<<~SQL)
|
|
(
|
|
SELECT thread_id, MAX(created_at) AS latest_created_at, MAX(id) AS latest_message_id
|
|
FROM chat_messages
|
|
WHERE thread_id IN (#{threads.map(&:id).join(",")})
|
|
GROUP BY thread_id
|
|
) AS last_replies_subquery
|
|
SQL
|
|
.joins(
|
|
"INNER JOIN chat_messages ON chat_messages.id = last_replies_subquery.latest_message_id",
|
|
)
|
|
.index_by(&:thread_id)
|
|
|
|
threads.each { |thread| thread.last_reply = last_replies[thread.id] }
|
|
end
|
|
|
|
threads
|
|
end
|
|
|
|
def fetch_tracking(guardian:, threads:, **)
|
|
context.tracking =
|
|
::Chat::TrackingStateReportQuery.call(
|
|
guardian: guardian,
|
|
thread_ids: threads.map(&:id),
|
|
include_threads: true,
|
|
).thread_tracking
|
|
end
|
|
|
|
def fetch_memberships(guardian:, threads:, **)
|
|
context.memberships =
|
|
::Chat::UserChatThreadMembership.where(
|
|
thread_id: threads.map(&:id),
|
|
user_id: guardian.user.id,
|
|
)
|
|
end
|
|
|
|
def threads_query(guardian, channel)
|
|
Chat::Thread
|
|
.strict_loading
|
|
.includes(
|
|
:channel,
|
|
:user_chat_thread_memberships,
|
|
original_message_user: :user_status,
|
|
original_message: [
|
|
:chat_webhook_event,
|
|
:chat_channel,
|
|
chat_mentions: {
|
|
user: :user_status,
|
|
},
|
|
user: :user_status,
|
|
],
|
|
)
|
|
.joins(
|
|
"JOIN (#{tracked_threads_subquery(guardian, channel)}) tracked_threads_subquery
|
|
ON tracked_threads_subquery.thread_id = chat_threads.id",
|
|
)
|
|
.joins(:user_chat_thread_memberships)
|
|
.joins(
|
|
"LEFT JOIN chat_messages original_messages ON chat_threads.original_message_id = original_messages.id",
|
|
)
|
|
.where("original_messages.deleted_at IS NULL")
|
|
.where(user_chat_thread_memberships_chat_threads: { user_id: guardian.user.id })
|
|
end
|
|
|
|
def tracked_threads_subquery(guardian, channel)
|
|
Chat::Thread
|
|
.joins(:chat_messages, :user_chat_thread_memberships)
|
|
.joins(
|
|
"LEFT JOIN chat_messages original_messages ON chat_threads.original_message_id = original_messages.id",
|
|
)
|
|
.where(user_chat_thread_memberships: { user_id: guardian.user.id })
|
|
.where(
|
|
"chat_threads.channel_id = :channel_id AND chat_messages.chat_channel_id = :channel_id",
|
|
channel_id: channel.id,
|
|
)
|
|
.where(
|
|
"user_chat_thread_memberships.notification_level IN (?)",
|
|
[
|
|
Chat::UserChatThreadMembership.notification_levels[:normal],
|
|
Chat::UserChatThreadMembership.notification_levels[:tracking],
|
|
],
|
|
)
|
|
.where(
|
|
"original_messages.deleted_at IS NULL AND chat_messages.deleted_at IS NULL AND original_messages.id IS NOT NULL",
|
|
)
|
|
.group("chat_threads.id")
|
|
.select(
|
|
"chat_threads.id AS thread_id, MAX(chat_messages.created_at) AS latest_message_created_at, MAX(chat_messages.id) AS latest_message_id",
|
|
)
|
|
.to_sql
|
|
end
|
|
end
|
|
end
|