mirror of
https://github.com/discourse/discourse.git
synced 2025-01-19 08:22:46 +08:00
f66007ec83
There are certain design decisions that were made in this commit. Private messages implements its own version of topic tracking state because there are significant differences between regular and private_message topics. Regular topics have to track categories and tags while private messages do not. It is much easier to design the new topic tracking state if we maintain two different classes, instead of trying to mash this two worlds together. One MessageBus channel per user and one MessageBus channel per group. This allows each user and each group to have their own channel backlog instead of having one global channel which requires the client to filter away unrelated messages.
182 lines
5.6 KiB
Ruby
182 lines
5.6 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
# This class is used to mirror unread and new status for private messages between
|
|
# server and client.
|
|
#
|
|
# On the server side, this class has two main responsibilities. The first is to
|
|
# query the database for the initial state of a user's unread and new private
|
|
# messages. The second is to publish message_bus messages to notify the client
|
|
# of various topic events.
|
|
#
|
|
# On the client side, we have a `PrivateMessageTopicTrackingState` class as well
|
|
# which will load the initial state into memory and subscribes to the relevant
|
|
# message_bus messages. When a message is received, it modifies the in-memory
|
|
# state based on the message type. The filtering for new and unread topics is
|
|
# done on the client side based on the in-memory state in order to derive the
|
|
# count of new and unread topics efficiently.
|
|
class PrivateMessageTopicTrackingState
|
|
CHANNEL_PREFIX = "/private-message-topic-tracking-state"
|
|
NEW_MESSAGE_TYPE = "new_topic"
|
|
UNREAD_MESSAGE_TYPE = "unread"
|
|
ARCHIVE_MESSAGE_TYPE = "archive"
|
|
GROUP_ARCHIVE_MESSAGE_TYPE = "group_archive"
|
|
|
|
def self.report(user)
|
|
sql = new_and_unread_sql(user)
|
|
|
|
DB.query(
|
|
sql + "\n\n LIMIT :max_topics",
|
|
{
|
|
max_topics: TopicTrackingState::MAX_TOPICS,
|
|
min_new_topic_date: Time.at(SiteSetting.min_new_topics_time).to_datetime
|
|
}
|
|
)
|
|
end
|
|
|
|
def self.new_and_unread_sql(user)
|
|
sql = report_raw_sql(user, skip_unread: true)
|
|
sql << "\nUNION ALL\n\n"
|
|
sql << report_raw_sql(user, skip_new: true)
|
|
end
|
|
|
|
def self.report_raw_sql(user, skip_unread: false,
|
|
skip_new: false)
|
|
|
|
unread =
|
|
if skip_unread
|
|
"1=0"
|
|
else
|
|
TopicTrackingState.unread_filter_sql(staff: user.staff?)
|
|
end
|
|
|
|
new =
|
|
if skip_new
|
|
"1=0"
|
|
else
|
|
TopicTrackingState.new_filter_sql
|
|
end
|
|
|
|
sql = +<<~SQL
|
|
SELECT
|
|
DISTINCT topics.id AS topic_id,
|
|
u.id AS user_id,
|
|
last_read_post_number,
|
|
tu.notification_level,
|
|
#{TopicTrackingState.highest_post_number_column_select(user.staff?)},
|
|
ARRAY(SELECT group_id FROM topic_allowed_groups WHERE topic_allowed_groups.topic_id = topics.id) AS group_ids
|
|
FROM topics
|
|
JOIN users u on u.id = #{user.id.to_i}
|
|
JOIN user_stats AS us ON us.user_id = u.id
|
|
JOIN user_options AS uo ON uo.user_id = u.id
|
|
LEFT JOIN group_users gu ON gu.user_id = u.id
|
|
LEFT JOIN topic_allowed_groups tag ON tag.topic_id = topics.id AND tag.group_id = gu.group_id
|
|
LEFT JOIN topic_users tu ON tu.topic_id = topics.id AND tu.user_id = u.id
|
|
LEFT JOIN topic_allowed_users tau ON tau.topic_id = topics.id AND tau.user_id = u.id
|
|
#{skip_new ? "" : "LEFT JOIN dismissed_topic_users ON dismissed_topic_users.topic_id = topics.id AND dismissed_topic_users.user_id = #{user.id.to_i}"}
|
|
WHERE (tau.topic_id IS NOT NULL OR tag.topic_id IS NOT NULL) AND
|
|
#{skip_unread ? "" : "topics.updated_at >= LEAST(us.first_unread_pm_at, gu.first_unread_pm_at) AND"}
|
|
topics.archetype = 'private_message' AND
|
|
((#{unread}) OR (#{new})) AND
|
|
topics.deleted_at IS NULL
|
|
SQL
|
|
end
|
|
|
|
def self.publish_unread(post)
|
|
return unless post.topic.private_message?
|
|
|
|
scope = TopicUser
|
|
.tracking(post.topic_id)
|
|
.includes(user: :user_stat)
|
|
|
|
allowed_group_ids = post.topic.allowed_groups.pluck(:id)
|
|
|
|
group_ids =
|
|
if post.post_type == Post.types[:whisper]
|
|
[Group::AUTO_GROUPS[:staff]]
|
|
else
|
|
allowed_group_ids
|
|
end
|
|
|
|
if group_ids.present?
|
|
scope = scope
|
|
.joins("INNER JOIN group_users gu ON gu.user_id = topic_users.user_id")
|
|
.where("gu.group_id IN (?)", group_ids)
|
|
end
|
|
|
|
scope
|
|
.select([:user_id, :last_read_post_number, :notification_level])
|
|
.each do |tu|
|
|
|
|
message = {
|
|
topic_id: post.topic_id,
|
|
message_type: UNREAD_MESSAGE_TYPE,
|
|
payload: {
|
|
last_read_post_number: tu.last_read_post_number,
|
|
highest_post_number: post.post_number,
|
|
notification_level: tu.notification_level,
|
|
group_ids: allowed_group_ids
|
|
}
|
|
}
|
|
|
|
MessageBus.publish(self.user_channel(tu.user_id), message.as_json,
|
|
user_ids: [tu.user_id]
|
|
)
|
|
end
|
|
end
|
|
|
|
def self.publish_new(topic)
|
|
return unless topic.private_message?
|
|
|
|
message = {
|
|
message_type: NEW_MESSAGE_TYPE,
|
|
topic_id: topic.id,
|
|
payload: {
|
|
last_read_post_number: nil,
|
|
highest_post_number: 1,
|
|
group_ids: topic.allowed_groups.pluck(:id)
|
|
}
|
|
}.as_json
|
|
|
|
topic.allowed_users.pluck(:id).each do |user_id|
|
|
MessageBus.publish(self.user_channel(user_id), message, user_ids: [user_id])
|
|
end
|
|
|
|
topic.allowed_groups.pluck(:id).each do |group_id|
|
|
MessageBus.publish(self.group_channel(group_id), message, group_ids: [group_id])
|
|
end
|
|
end
|
|
|
|
def self.publish_group_archived(topic, group_id)
|
|
return unless topic.private_message?
|
|
|
|
message = {
|
|
message_type: GROUP_ARCHIVE_MESSAGE_TYPE,
|
|
topic_id: topic.id,
|
|
payload: {
|
|
group_ids: [group_id]
|
|
}
|
|
}.as_json
|
|
|
|
MessageBus.publish(self.group_channel(group_id), message, group_ids: [group_id])
|
|
end
|
|
|
|
def self.publish_user_archived(topic, user_id)
|
|
return unless topic.private_message?
|
|
|
|
message = {
|
|
message_type: ARCHIVE_MESSAGE_TYPE,
|
|
topic_id: topic.id,
|
|
}.as_json
|
|
|
|
MessageBus.publish(self.user_channel(user_id), message, user_ids: [user_id])
|
|
end
|
|
|
|
def self.user_channel(user_id)
|
|
"#{CHANNEL_PREFIX}/user/#{user_id}"
|
|
end
|
|
|
|
def self.group_channel(group_id)
|
|
"#{CHANNEL_PREFIX}/group/#{group_id}"
|
|
end
|
|
end
|