mirror of
https://github.com/discourse/discourse.git
synced 2025-04-01 05:55:57 +08:00
FIX: simplify channel threads lookup for pagination (#22596)
The previous query with subqueries was complicated to make right with pagination, this new query should be working correctly and is passing tests.
This commit is contained in:
parent
110393e438
commit
d7ef7b9c03
@ -72,37 +72,46 @@ module Chat
|
|||||||
end
|
end
|
||||||
|
|
||||||
def fetch_threads(guardian:, channel:, **)
|
def fetch_threads(guardian:, channel:, **)
|
||||||
read_threads = []
|
::Chat::Thread
|
||||||
|
.strict_loading
|
||||||
unread_threads =
|
.includes(
|
||||||
threads_query(guardian, channel)
|
:channel,
|
||||||
.where(<<~SQL)
|
:user_chat_thread_memberships,
|
||||||
user_chat_thread_memberships_chat_threads.last_read_message_id IS NULL
|
original_message_user: :user_status,
|
||||||
OR tracked_threads_subquery.latest_message_id > user_chat_thread_memberships_chat_threads.last_read_message_id
|
last_message: [
|
||||||
SQL
|
:uploads,
|
||||||
.order("tracked_threads_subquery.latest_message_created_at DESC")
|
:chat_webhook_event,
|
||||||
.limit(context.limit)
|
:chat_channel,
|
||||||
.offset(context.offset)
|
chat_mentions: {
|
||||||
.to_a
|
user: :user_status,
|
||||||
|
},
|
||||||
# We do this to avoid having to query additional threads if the user
|
user: :user_status,
|
||||||
# already has a lot of unread threads.
|
],
|
||||||
if unread_threads.length < context.limit
|
original_message: [
|
||||||
final_limit = context.limit - unread_threads.length
|
:uploads,
|
||||||
final_offset = context.offset + unread_threads.length
|
:chat_webhook_event,
|
||||||
|
:chat_channel,
|
||||||
read_threads =
|
chat_mentions: {
|
||||||
threads_query(guardian, channel)
|
user: :user_status,
|
||||||
.where(<<~SQL)
|
},
|
||||||
tracked_threads_subquery.latest_message_id <= user_chat_thread_memberships_chat_threads.last_read_message_id
|
user: :user_status,
|
||||||
SQL
|
],
|
||||||
.order("tracked_threads_subquery.latest_message_created_at DESC")
|
)
|
||||||
.limit(final_limit)
|
.joins(:user_chat_thread_memberships, :last_message, :original_message)
|
||||||
.offset(final_offset)
|
.where("user_chat_thread_memberships.user_id = ?", guardian.user.id)
|
||||||
.to_a
|
.where(
|
||||||
end
|
"user_chat_thread_memberships.notification_level IN (?)",
|
||||||
|
[
|
||||||
unread_threads + read_threads
|
::Chat::UserChatThreadMembership.notification_levels[:normal],
|
||||||
|
::Chat::UserChatThreadMembership.notification_levels[:tracking],
|
||||||
|
],
|
||||||
|
)
|
||||||
|
.where("chat_threads.channel_id = ?", channel.id)
|
||||||
|
.limit(context.limit)
|
||||||
|
.offset(context.offset)
|
||||||
|
.order(
|
||||||
|
"CASE WHEN chat_threads.last_message_id > user_chat_thread_memberships.last_read_message_id THEN 0 ELSE 1 END, chat_messages.created_at DESC",
|
||||||
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
def fetch_tracking(guardian:, threads:, **)
|
def fetch_tracking(guardian:, threads:, **)
|
||||||
@ -122,74 +131,6 @@ module Chat
|
|||||||
)
|
)
|
||||||
end
|
end
|
||||||
|
|
||||||
def threads_query(guardian, channel)
|
|
||||||
::Chat::Thread
|
|
||||||
.strict_loading
|
|
||||||
.includes(
|
|
||||||
:channel,
|
|
||||||
:user_chat_thread_memberships,
|
|
||||||
original_message_user: :user_status,
|
|
||||||
last_message: [
|
|
||||||
:chat_webhook_event,
|
|
||||||
:chat_channel,
|
|
||||||
chat_mentions: {
|
|
||||||
user: :user_status,
|
|
||||||
},
|
|
||||||
user: :user_status,
|
|
||||||
],
|
|
||||||
original_message: [
|
|
||||||
:uploads,
|
|
||||||
: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
|
|
||||||
.strict_loading
|
|
||||||
.joins(:chat_messages, :user_chat_thread_memberships)
|
|
||||||
.joins(
|
|
||||||
"LEFT JOIN chat_messages original_messages ON chat_threads.original_message_id = original_messages.id",
|
|
||||||
)
|
|
||||||
.joins(
|
|
||||||
"LEFT JOIN chat_messages last_message ON chat_threads.last_message_id = last_message.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 AND last_message.deleted_at IS NULL",
|
|
||||||
)
|
|
||||||
.select(
|
|
||||||
"chat_threads.id AS thread_id, last_message.created_at AS latest_message_created_at, last_message.id AS latest_message_id",
|
|
||||||
)
|
|
||||||
.to_sql
|
|
||||||
end
|
|
||||||
|
|
||||||
def build_load_more_url(contract:, **)
|
def build_load_more_url(contract:, **)
|
||||||
load_more_params = { offset: context.offset + context.limit }.to_query
|
load_more_params = { offset: context.offset + context.limit }.to_query
|
||||||
context.load_more_url =
|
context.load_more_url =
|
||||||
|
@ -36,7 +36,7 @@ export default class ChatChannelPaneSubscriptionsManager extends ChatPaneBaseSub
|
|||||||
|
|
||||||
handleThreadOriginalMessageUpdate(data) {
|
handleThreadOriginalMessageUpdate(data) {
|
||||||
const message = this.messagesManager.findMessage(data.original_message_id);
|
const message = this.messagesManager.findMessage(data.original_message_id);
|
||||||
if (message.thread) {
|
if (message?.thread) {
|
||||||
message.thread.preview = ChatThreadPreview.create(data.preview);
|
message.thread.preview = ChatThreadPreview.create(data.preview);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user