FIX: Tracking Topic State know about category_seen_at (#8351)

If category got last_seen_at is set TrackingTopicState should know about it and exclude those topics from marking them as new
This commit is contained in:
Krzysztof Kotlarek 2019-11-14 16:11:34 +11:00 committed by GitHub
parent b69450bee2
commit f434de2536
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 33 additions and 11 deletions

View File

@ -159,7 +159,6 @@ class TopicTrackingState
end end
def self.report(user, topic_id = nil) def self.report(user, topic_id = nil)
# Sam: this is a hairy report, in particular I need custom joins and fancy conditions # Sam: this is a hairy report, in particular I need custom joins and fancy conditions
# Dropping to sql_builder so I can make sense of it. # Dropping to sql_builder so I can make sense of it.
# #
@ -173,7 +172,8 @@ class TopicTrackingState
skip_unread: true, skip_unread: true,
skip_order: true, skip_order: true,
staff: user.staff?, staff: user.staff?,
admin: user.admin? admin: user.admin?,
user: user
) )
sql << "\nUNION ALL\n\n" sql << "\nUNION ALL\n\n"
@ -184,7 +184,8 @@ class TopicTrackingState
skip_order: true, skip_order: true,
staff: user.staff?, staff: user.staff?,
filter_old_unread: true, filter_old_unread: true,
admin: user.admin? admin: user.admin?,
user: user
) )
DB.query( DB.query(
@ -221,7 +222,8 @@ class TopicTrackingState
"1=0" "1=0"
else else
TopicQuery.new_filter(Topic, "xxx").where_clause.send(:predicates).join(" AND ").gsub!("'xxx'", treat_as_new_topic_clause) + TopicQuery.new_filter(Topic, "xxx").where_clause.send(:predicates).join(" AND ").gsub!("'xxx'", treat_as_new_topic_clause) +
" AND topics.created_at > :min_new_topic_date" " AND topics.created_at > :min_new_topic_date" +
" AND (category_users.last_seen_at IS NULL OR topics.created_at > category_users.last_seen_at)"
end end
select = (opts[:select]) || " select = (opts[:select]) || "
@ -240,7 +242,7 @@ class TopicTrackingState
append = "OR u.admin" if !opts.key?(:admin) append = "OR u.admin" if !opts.key?(:admin)
<<~SQL <<~SQL
( (
NOT c.read_restricted #{append} OR category_id IN ( NOT c.read_restricted #{append} OR c.id IN (
SELECT c2.id FROM categories c2 SELECT c2.id FROM categories c2
JOIN category_groups cg ON cg.category_id = c2.id JOIN category_groups cg ON cg.category_id = c2.id
JOIN group_users gu ON gu.user_id = :user_id AND cg.group_id = gu.group_id JOIN group_users gu ON gu.user_id = :user_id AND cg.group_id = gu.group_id
@ -265,6 +267,7 @@ class TopicTrackingState
JOIN user_options AS uo ON uo.user_id = u.id JOIN user_options AS uo ON uo.user_id = u.id
JOIN categories c ON c.id = topics.category_id JOIN categories c ON c.id = topics.category_id
LEFT JOIN topic_users tu ON tu.topic_id = topics.id AND tu.user_id = u.id LEFT JOIN topic_users tu ON tu.topic_id = topics.id AND tu.user_id = u.id
LEFT JOIN category_users ON category_users.category_id = topics.category_id AND category_users.user_id = #{opts[:user].id}
WHERE u.id = :user_id AND WHERE u.id = :user_id AND
#{filter_old_unread} #{filter_old_unread}
topics.archetype <> 'private_message' AND topics.archetype <> 'private_message' AND
@ -272,12 +275,9 @@ class TopicTrackingState
#{visibility_filter} #{visibility_filter}
topics.deleted_at IS NULL AND topics.deleted_at IS NULL AND
#{category_filter} #{category_filter}
NOT EXISTS( SELECT 1 FROM category_users cu (category_users.id IS NULL OR
WHERE last_read_post_number IS NULL AND last_read_post_number IS NOT NULL OR
cu.user_id = :user_id AND category_users.notification_level <> #{CategoryUser.notification_levels[:muted]})
cu.category_id = topics.category_id AND
cu.notification_level = #{CategoryUser.notification_levels[:muted]})
SQL SQL
if opts[:topic_id] if opts[:topic_id]

View File

@ -353,6 +353,28 @@ describe TopicTrackingState do
expect(report.length).to eq(1) expect(report.length).to eq(1)
end end
it "correctly handles seen categories" do
user = Fabricate(:user)
post
report = TopicTrackingState.report(user)
expect(report.length).to eq(1)
CategoryUser.create!(user_id: user.id,
notification_level: CategoryUser.notification_levels[:regular],
category_id: post.topic.category_id,
last_seen_at: post.topic.created_at
)
report = TopicTrackingState.report(user)
expect(report.length).to eq(0)
post.topic.touch(:created_at)
report = TopicTrackingState.report(user)
expect(report.length).to eq(1)
end
it "correctly handles capping" do it "correctly handles capping" do
user = Fabricate(:user) user = Fabricate(:user)