Merge pull request #5807 from discourse/min-flags-by-topic
FEATURE: New site setting `min_flags_staff_visibility`
This commit is contained in:
commit
8262fc5d15
|
@ -52,13 +52,22 @@ class PostAction < ActiveRecord::Base
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.update_flagged_posts_count
|
def self.update_flagged_posts_count
|
||||||
posts_flagged_count = PostAction.active
|
flagged_relation = PostAction.active
|
||||||
.flags
|
.flags
|
||||||
.joins(post: :topic)
|
.joins(post: :topic)
|
||||||
.where('posts.deleted_at' => nil)
|
.where('posts.deleted_at' => nil)
|
||||||
.where('topics.deleted_at' => nil)
|
.where('topics.deleted_at' => nil)
|
||||||
.where('posts.user_id > 0')
|
.where('posts.user_id > 0')
|
||||||
.count('DISTINCT posts.id')
|
.group("posts.id")
|
||||||
|
|
||||||
|
if SiteSetting.min_flags_staff_visibility > 1
|
||||||
|
flagged_relation = flagged_relation
|
||||||
|
.having("count(*) >= ?", SiteSetting.min_flags_staff_visibility)
|
||||||
|
end
|
||||||
|
|
||||||
|
posts_flagged_count = flagged_relation
|
||||||
|
.pluck("posts.id")
|
||||||
|
.count
|
||||||
|
|
||||||
$redis.set('posts_flagged_count', posts_flagged_count)
|
$redis.set('posts_flagged_count', posts_flagged_count)
|
||||||
user_ids = User.staff.pluck(:id)
|
user_ids = User.staff.pluck(:id)
|
||||||
|
|
|
@ -1475,6 +1475,7 @@ en:
|
||||||
auto_silence_fast_typers_max_trust_level: "Maximum trust level to auto silence fast typers"
|
auto_silence_fast_typers_max_trust_level: "Maximum trust level to auto silence fast typers"
|
||||||
auto_silence_first_post_regex: "Case insensitive regex that if passed will cause first post by user to be silenced and sent to approval queue. Example: raging|a[bc]a , will cause all posts containing raging or aba or aca to be silenced on first. Only applies to first post."
|
auto_silence_first_post_regex: "Case insensitive regex that if passed will cause first post by user to be silenced and sent to approval queue. Example: raging|a[bc]a , will cause all posts containing raging or aba or aca to be silenced on first. Only applies to first post."
|
||||||
flags_default_topics: "Show flagged topics by default in the admin section"
|
flags_default_topics: "Show flagged topics by default in the admin section"
|
||||||
|
min_flags_staff_visibility: "The minimum amount of flags on a post must have before staff can see it in the admin section"
|
||||||
|
|
||||||
reply_by_email_enabled: "Enable replying to topics via email."
|
reply_by_email_enabled: "Enable replying to topics via email."
|
||||||
reply_by_email_address: "Template for reply by email incoming email address, for example: %{reply_key}@reply.example.com or replies+%{reply_key}@example.com"
|
reply_by_email_address: "Template for reply by email incoming email address, for example: %{reply_key}@reply.example.com or replies+%{reply_key}@example.com"
|
||||||
|
|
|
@ -1113,6 +1113,7 @@ spam:
|
||||||
flags_default_topics:
|
flags_default_topics:
|
||||||
default: false
|
default: false
|
||||||
client: true
|
client: true
|
||||||
|
min_flags_staff_visibility: 1
|
||||||
|
|
||||||
rate_limits:
|
rate_limits:
|
||||||
unique_posts_mins: 5
|
unique_posts_mins: 5
|
||||||
|
|
|
@ -21,12 +21,16 @@ module FlagQuery
|
||||||
|
|
||||||
total_rows = actions.count
|
total_rows = actions.count
|
||||||
|
|
||||||
post_ids = actions.limit(per_page)
|
post_ids_relation = actions.limit(per_page)
|
||||||
.offset(offset)
|
.offset(offset)
|
||||||
.group(:post_id)
|
.group(:post_id)
|
||||||
.order('MIN(post_actions.created_at) DESC')
|
.order('MIN(post_actions.created_at) DESC')
|
||||||
.pluck(:post_id)
|
|
||||||
.uniq
|
if opts[:filter] != "old" && SiteSetting.min_flags_staff_visibility > 1
|
||||||
|
post_ids_relation = post_ids_relation.having("count(*) >= ?", SiteSetting.min_flags_staff_visibility)
|
||||||
|
end
|
||||||
|
|
||||||
|
post_ids = post_ids_relation.pluck(:post_id).uniq
|
||||||
|
|
||||||
posts = SqlBuilder.new("
|
posts = SqlBuilder.new("
|
||||||
SELECT p.id,
|
SELECT p.id,
|
||||||
|
@ -182,18 +186,25 @@ module FlagQuery
|
||||||
ft_by_id = {}
|
ft_by_id = {}
|
||||||
users_by_id = {}
|
users_by_id = {}
|
||||||
topics_by_id = {}
|
topics_by_id = {}
|
||||||
|
counts_by_post = {}
|
||||||
|
|
||||||
results.each do |pa|
|
results.each do |pa|
|
||||||
if pa.post.present? && pa.post.topic.present?
|
if pa.post.present? && pa.post.topic.present?
|
||||||
ft = ft_by_id[pa.post.topic.id] ||= OpenStruct.new(
|
topic_id = pa.post.topic.id
|
||||||
|
|
||||||
|
ft = ft_by_id[topic_id] ||= OpenStruct.new(
|
||||||
topic: pa.post.topic,
|
topic: pa.post.topic,
|
||||||
flag_counts: {},
|
flag_counts: {},
|
||||||
user_ids: [],
|
user_ids: [],
|
||||||
last_flag_at: pa.created_at
|
last_flag_at: pa.created_at,
|
||||||
|
meets_minimum: false
|
||||||
)
|
)
|
||||||
|
|
||||||
topics_by_id[pa.post.topic.id] = pa.post.topic
|
counts_by_post[pa.post.id] ||= 0
|
||||||
|
sum = counts_by_post[pa.post.id] += 1
|
||||||
|
ft.meets_minimum = true if sum >= SiteSetting.min_flags_staff_visibility
|
||||||
|
|
||||||
|
topics_by_id[topic_id] = pa.post.topic
|
||||||
ft.flag_counts[pa.post_action_type_id] ||= 0
|
ft.flag_counts[pa.post_action_type_id] ||= 0
|
||||||
ft.flag_counts[pa.post_action_type_id] += 1
|
ft.flag_counts[pa.post_action_type_id] += 1
|
||||||
|
|
||||||
|
@ -204,9 +215,11 @@ module FlagQuery
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
flagged_topics = ft_by_id.values.select { |ft| ft.meets_minimum }
|
||||||
|
|
||||||
Topic.preload_custom_fields(topics_by_id.values, TopicList.preloaded_custom_fields)
|
Topic.preload_custom_fields(topics_by_id.values, TopicList.preloaded_custom_fields)
|
||||||
|
|
||||||
{ flagged_topics: ft_by_id.values, users: users_by_id.values }
|
{ flagged_topics: flagged_topics, users: users_by_id.values }
|
||||||
end
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
|
@ -5,6 +5,41 @@ describe FlagQuery do
|
||||||
|
|
||||||
let(:codinghorror) { Fabricate(:coding_horror) }
|
let(:codinghorror) { Fabricate(:coding_horror) }
|
||||||
|
|
||||||
|
describe "flagged_topics" do
|
||||||
|
it "respects `min_flags_staff_visibility`" do
|
||||||
|
admin = Fabricate(:admin)
|
||||||
|
moderator = Fabricate(:moderator)
|
||||||
|
|
||||||
|
post = create_post
|
||||||
|
|
||||||
|
PostAction.act(moderator, post, PostActionType.types[:spam])
|
||||||
|
|
||||||
|
SiteSetting.min_flags_staff_visibility = 1
|
||||||
|
|
||||||
|
result = FlagQuery.flagged_topics
|
||||||
|
expect(result[:flagged_topics]).to be_present
|
||||||
|
ft = result[:flagged_topics].first
|
||||||
|
expect(ft.topic).to eq(post.topic)
|
||||||
|
expect(ft.flag_counts).to eq(PostActionType.types[:spam] => 1)
|
||||||
|
|
||||||
|
SiteSetting.min_flags_staff_visibility = 2
|
||||||
|
|
||||||
|
result = FlagQuery.flagged_topics
|
||||||
|
expect(result[:flagged_topics]).to be_blank
|
||||||
|
|
||||||
|
PostAction.act(admin, post, PostActionType.types[:inappropriate])
|
||||||
|
result = FlagQuery.flagged_topics
|
||||||
|
expect(result[:flagged_topics]).to be_present
|
||||||
|
ft = result[:flagged_topics].first
|
||||||
|
expect(ft.topic).to eq(post.topic)
|
||||||
|
expect(ft.flag_counts).to eq(
|
||||||
|
PostActionType.types[:spam] => 1,
|
||||||
|
PostActionType.types[:inappropriate] => 1
|
||||||
|
)
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
|
|
||||||
describe "flagged_posts_report" do
|
describe "flagged_posts_report" do
|
||||||
it "does not return flags on system posts" do
|
it "does not return flags on system posts" do
|
||||||
admin = Fabricate(:admin)
|
admin = Fabricate(:admin)
|
||||||
|
@ -75,7 +110,27 @@ describe FlagQuery do
|
||||||
posts, users = FlagQuery.flagged_posts_report(moderator)
|
posts, users = FlagQuery.flagged_posts_report(moderator)
|
||||||
|
|
||||||
expect(posts.count).to eq(1)
|
expect(posts.count).to eq(1)
|
||||||
|
end
|
||||||
|
|
||||||
|
it "respects `min_flags_staff_visibility`" do
|
||||||
|
admin = Fabricate(:admin)
|
||||||
|
moderator = Fabricate(:moderator)
|
||||||
|
|
||||||
|
post = create_post
|
||||||
|
|
||||||
|
PostAction.act(moderator, post, PostActionType.types[:spam])
|
||||||
|
|
||||||
|
SiteSetting.min_flags_staff_visibility = 2
|
||||||
|
posts, topics, users = FlagQuery.flagged_posts_report(admin)
|
||||||
|
expect(posts).to be_blank
|
||||||
|
expect(topics).to be_blank
|
||||||
|
expect(users).to be_blank
|
||||||
|
|
||||||
|
PostAction.act(admin, post, PostActionType.types[:inappropriate])
|
||||||
|
posts, topics, users = FlagQuery.flagged_posts_report(admin)
|
||||||
|
expect(posts).to be_present
|
||||||
|
expect(topics).to be_present
|
||||||
|
expect(users).to be_present
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
|
@ -141,6 +141,17 @@ describe PostAction do
|
||||||
expect(PostAction.flagged_posts_count).to eq(0)
|
expect(PostAction.flagged_posts_count).to eq(0)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
it "respects min_flags_staff_visibility" do
|
||||||
|
SiteSetting.min_flags_staff_visibility = 2
|
||||||
|
expect(PostAction.flagged_posts_count).to eq(0)
|
||||||
|
|
||||||
|
PostAction.act(codinghorror, post, PostActionType.types[:off_topic])
|
||||||
|
expect(PostAction.flagged_posts_count).to eq(0)
|
||||||
|
|
||||||
|
PostAction.act(eviltrout, post, PostActionType.types[:off_topic])
|
||||||
|
expect(PostAction.flagged_posts_count).to eq(1)
|
||||||
|
end
|
||||||
|
|
||||||
it "should reset counts when a topic is deleted" do
|
it "should reset counts when a topic is deleted" do
|
||||||
PostAction.act(codinghorror, post, PostActionType.types[:off_topic])
|
PostAction.act(codinghorror, post, PostActionType.types[:off_topic])
|
||||||
post.topic.trash!
|
post.topic.trash!
|
||||||
|
|
Loading…
Reference in New Issue