mirror of
https://github.com/discourse/discourse.git
synced 2024-11-25 10:20:58 -06:00
d4b5a81b05
Recalculating a ReviewableFlaggedPost's score after rejecting or ignoring it sets the score as 0, which means that we can't find them after reviewing. They don't surpass the minimum priority threshold and are hidden. Additionally, we only want to use agreed flags when calculating the different priority thresholds.
211 lines
7.1 KiB
Ruby
211 lines
7.1 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'rails_helper'
|
|
|
|
RSpec.describe ReviewableScore, type: :model do
|
|
|
|
context "transitions" do
|
|
fab!(:user) { Fabricate(:user, trust_level: 3) }
|
|
fab!(:post) { Fabricate(:post) }
|
|
fab!(:moderator) { Fabricate(:moderator) }
|
|
|
|
it "scores agreements properly" do
|
|
reviewable = PostActionCreator.off_topic(user, post).reviewable
|
|
rs = reviewable.reviewable_scores.find_by(user: user)
|
|
expect(rs.score).to eq(4.0)
|
|
expect(reviewable.score).to eq(4.0)
|
|
expect(reviewable.latest_score).to be_present
|
|
|
|
reviewable.perform(moderator, :agree_and_keep)
|
|
expect(rs.reload).to be_agreed
|
|
expect(rs.reviewed_by).to eq(moderator)
|
|
expect(rs.reviewed_at).to be_present
|
|
expect(reviewable.score).to eq(4.0)
|
|
end
|
|
|
|
it "scores disagreements properly" do
|
|
reviewable = PostActionCreator.spam(user, post).reviewable
|
|
rs = reviewable.reviewable_scores.find_by(user: user)
|
|
expect(rs).to be_pending
|
|
expect(rs.score).to eq(4.0)
|
|
expect(reviewable.score).to eq(4.0)
|
|
expect(reviewable.latest_score).to be_present
|
|
|
|
reviewable.perform(moderator, :disagree)
|
|
expect(rs.reload).to be_disagreed
|
|
expect(rs.reviewed_by).to eq(moderator)
|
|
expect(rs.reviewed_at).to be_present
|
|
expect(reviewable.score).to eq(4.0)
|
|
end
|
|
|
|
it "increases the score by the post action type's score bonus" do
|
|
PostActionType.where(name_key: 'spam').update_all(score_bonus: 2.25)
|
|
reviewable = PostActionCreator.spam(user, post).reviewable
|
|
score = reviewable.reviewable_scores.find_by(user: user)
|
|
expect(score).to be_pending
|
|
expect(score.score).to eq(6.25)
|
|
expect(reviewable.score).to eq(6.25)
|
|
end
|
|
end
|
|
|
|
describe "overall score" do
|
|
fab!(:user0) { Fabricate(:user, trust_level: 1) }
|
|
fab!(:user1) { Fabricate(:user, trust_level: 2) }
|
|
fab!(:user2) { Fabricate(:user, trust_level: 3) }
|
|
fab!(:moderator) { Fabricate(:moderator) }
|
|
fab!(:post) { Fabricate(:post) }
|
|
let(:topic) { post.topic }
|
|
|
|
it "gives a bonus for take_action" do
|
|
result = PostActionCreator.new(
|
|
moderator,
|
|
post,
|
|
PostActionType.types[:spam],
|
|
take_action: true
|
|
).perform
|
|
|
|
expect(result.reviewable_score.take_action_bonus).to eq(5.0)
|
|
expect(result.reviewable.score).to eq(11.0)
|
|
expect(topic.reviewable_score).to eq(11.0)
|
|
end
|
|
|
|
it "is the total of the pending reviewable scores" do
|
|
reviewable = PostActionCreator.spam(user0, post).reviewable
|
|
expect(reviewable.score).to eq(2.0)
|
|
expect(topic.reload.reviewable_score).to eq(2.0)
|
|
|
|
reviewable = PostActionCreator.inappropriate(user1, post).reviewable
|
|
expect(reviewable.score).to eq(5.0)
|
|
expect(topic.reload.reviewable_score).to eq(5.0)
|
|
|
|
reviewable.perform(Discourse.system_user, :agree_and_keep)
|
|
expect(reviewable.score).to eq(5.0)
|
|
expect(topic.reload.reviewable_score).to eq(5.0)
|
|
|
|
reviewable = PostActionCreator.off_topic(user2, post).reviewable
|
|
expect(reviewable.score).to eq(9.0)
|
|
expect(topic.reload.reviewable_score).to eq(9.0)
|
|
end
|
|
end
|
|
|
|
describe ".user_accuracy_bonus" do
|
|
fab!(:user) { Fabricate(:user) }
|
|
let(:user_stat) { user.user_stat }
|
|
|
|
it "returns 0 for a user with no flags" do
|
|
expect(ReviewableScore.user_accuracy_bonus(user)).to eq(0.0)
|
|
end
|
|
|
|
it "returns 0 for a user with no flags" do
|
|
system_user = Discourse.system_user
|
|
stats = system_user.user_stat
|
|
|
|
stats.flags_agreed = 10
|
|
stats.flags_disagreed = 42
|
|
|
|
expect(ReviewableScore.user_accuracy_bonus(system_user)).to eq(0.0)
|
|
end
|
|
|
|
it "returns 0 until the user has made more than 5 flags" do
|
|
user_stat.flags_agreed = 4
|
|
user_stat.flags_disagreed = 1
|
|
expect(ReviewableScore.user_accuracy_bonus(user)).to eq(0.0)
|
|
end
|
|
|
|
it "returns the users weighted accuracy bonus" do
|
|
user_stat.flags_agreed = 10
|
|
user_stat.flags_disagreed = 42
|
|
expect(ReviewableScore.user_accuracy_bonus(user).floor(2)).to eq(-10.34)
|
|
|
|
user_stat.flags_agreed = 2
|
|
user_stat.flags_disagreed = 12
|
|
expect(ReviewableScore.user_accuracy_bonus(user).floor(2)).to eq(-7.58)
|
|
|
|
user_stat.flags_agreed = 1
|
|
user_stat.flags_disagreed = 6
|
|
expect(ReviewableScore.user_accuracy_bonus(user).floor(2)).to eq(-5.59)
|
|
|
|
user_stat.flags_agreed = 2
|
|
user_stat.flags_disagreed = 4
|
|
expect(ReviewableScore.user_accuracy_bonus(user).floor(2)).to eq(-3.39)
|
|
|
|
user_stat.flags_agreed = 7
|
|
user_stat.flags_disagreed = 3
|
|
expect(ReviewableScore.user_accuracy_bonus(user).floor(2)).to eq(0)
|
|
|
|
user_stat.flags_agreed = 14
|
|
user_stat.flags_disagreed = 6
|
|
expect(ReviewableScore.user_accuracy_bonus(user).floor(2)).to eq(0)
|
|
|
|
# Ignored flags don't count
|
|
user_stat.flags_agreed = 121
|
|
user_stat.flags_disagreed = 44
|
|
user_stat.flags_ignored = 4
|
|
expect(ReviewableScore.user_accuracy_bonus(user).floor(2)).to eq(2.04)
|
|
|
|
user_stat.flags_agreed = 9
|
|
user_stat.flags_disagreed = 2
|
|
expect(ReviewableScore.user_accuracy_bonus(user).floor(2)).to eq(3.41)
|
|
|
|
user_stat.flags_agreed = 25
|
|
user_stat.flags_disagreed = 4
|
|
expect(ReviewableScore.user_accuracy_bonus(user).floor(2)).to eq(6.56)
|
|
|
|
user_stat.flags_agreed = 120
|
|
user_stat.flags_disagreed = 12
|
|
expect(ReviewableScore.user_accuracy_bonus(user).floor(2)).to eq(12.27)
|
|
end
|
|
end
|
|
|
|
describe ".user_flag_score" do
|
|
context "a user with no flags" do
|
|
it "returns 1.0 + trust_level" do
|
|
expect(ReviewableScore.user_flag_score(Fabricate.build(:user, trust_level: 2))).to eq(3.0)
|
|
expect(ReviewableScore.user_flag_score(Fabricate.build(:user, trust_level: 3))).to eq(4.0)
|
|
end
|
|
|
|
it "returns 6.0 for staff" do
|
|
expect(ReviewableScore.user_flag_score(Fabricate.build(:moderator, trust_level: 2))).to eq(6.0)
|
|
expect(ReviewableScore.user_flag_score(Fabricate.build(:admin, trust_level: 1))).to eq(6.0)
|
|
end
|
|
end
|
|
|
|
context "a user with some flags" do
|
|
fab!(:user) { Fabricate(:user) }
|
|
let(:user_stat) { user.user_stat }
|
|
|
|
it "returns 1.0 + trust_level + accuracy_bonus" do
|
|
user.trust_level = 2
|
|
user_stat.flags_agreed = 12
|
|
user_stat.flags_disagreed = 2
|
|
user_stat.flags_ignored = 2
|
|
expect(ReviewableScore.user_flag_score(user).floor(2)).to eq(7.99)
|
|
end
|
|
end
|
|
end
|
|
|
|
describe ".calculate_score" do
|
|
fab!(:user) { Fabricate(:user) }
|
|
let(:user_stat) { user.user_stat }
|
|
|
|
it 'never returns less than 0' do
|
|
user.trust_level = 2
|
|
user_stat.flags_agreed = 1
|
|
user_stat.flags_disagreed = 1000
|
|
flag_score = -21.88
|
|
expect(ReviewableScore.user_flag_score(user).floor(2)).to eq(flag_score)
|
|
expect(ReviewableScore.calculate_score(user, 5, 5)).to eq(0)
|
|
end
|
|
|
|
it 'returns user_flag_score + type_bonus + take_action_bonus' do
|
|
user.trust_level = 2
|
|
user_stat.flags_agreed = 12
|
|
user_stat.flags_disagreed = 2
|
|
flag_score = 7.99
|
|
expect(ReviewableScore.user_flag_score(user).floor(2)).to eq(flag_score)
|
|
expect(ReviewableScore.calculate_score(user, 2, 3)).to eq(flag_score + 2 + 3)
|
|
end
|
|
end
|
|
|
|
end
|