FEATURE: Trust level 3 promotion and demotion. Job is disabled for now.

This commit is contained in:
Neil Lalonde 2014-06-27 12:26:03 -04:00
parent f49b661a0d
commit 807bfbd9bb
13 changed files with 345 additions and 33 deletions

View File

@ -13,11 +13,13 @@ Discourse.LeaderRequirements = Discourse.Model.extend({
topics_replied_to: this.get('num_topics_replied_to') >= this.get('min_topics_replied_to'),
topics_viewed: this.get('topics_viewed') >= this.get('min_topics_viewed'),
posts_read: this.get('posts_read') >= this.get('min_posts_read'),
flagged_posts: this.get('num_flagged_posts') < this.get('max_flagged_posts')
flagged_posts: this.get('num_flagged_posts') <= this.get('max_flagged_posts'),
flagged_by_users: this.get('num_flagged_by_users') <= this.get('max_flagged_by_users')
};
}.property('days_visited', 'min_days_visited',
'num_topics_replied_to', 'min_topics_replied_to',
'topics_viewed', 'min_topics_viewed',
'posts_read', 'min_posts_read',
'num_flagged_posts', 'max_flagged_posts')
'num_flagged_posts', 'max_flagged_posts',
'num_flagged_by_users', 'max_flagged_by_users')
});

View File

@ -55,18 +55,30 @@
<td>{{num_flagged_posts}}</td>
<td>{{max_flagged_posts}} {{i18n max}}</td>
</tr>
<tr>
<th>{{i18n admin.user.tl3_requirements.flagged_by_users}}</th>
<td><i {{bindAttr class=":fa met.flagged_by_users:fa-check:fa-times"}}></i></td>
<td>{{num_flagged_by_users}}</td>
<td>{{max_flagged_by_users}} {{i18n max}}</td>
</tr>
</tbody>
</table>
<br/>
<p>
{{#if requirements_met}}
<i class="fa fa-check"></i> Qualifies for trust level 3
{{else}}
<i class="fa fa-times"></i> Doesn't qualify for trust level 3
{{/if}}
</p>
{{/with}}
<br/>
<p>
{{#if leaderRequirements.requirements_met}}
<i class="fa fa-check"></i> {{i18n admin.user.tl3_requirements.qualifies}}
{{#unless isLeader}}
{{i18n admin.user.tl3_requirements.will_be_promoted}}
{{/unless}}
{{else}}
<i class="fa fa-times"></i> {{i18n admin.user.tl3_requirements.does_not_qualify}}
{{#if isLeader}}
{{i18n admin.user.tl3_requirements.will_be_demoted}}
{{/if}}
{{/if}}
</p>
</div>

View File

@ -129,6 +129,7 @@ Discourse.User = Discourse.Model.extend({
return Discourse.Site.currentProp('trustLevels').findProperty('id', parseInt(this.get('trust_level'), 10));
}.property('trust_level'),
isLeader: Em.computed.equal('trust_level', 3),
isElder: Em.computed.equal('trust_level', 4),
canManageTopic: Em.computed.or('staff', 'isElder'),

View File

@ -0,0 +1,25 @@
module Jobs
class LeaderPromotions < Jobs::Scheduled
daily at: 4.hours
def execute(args)
return unless Rails.env.test? # do nothing for now
# Demotions
demoted_user_ids = []
User.real.where(trust_level: TrustLevel.levels[:leader]).find_each do |u|
unless Promotion.leader_met?(u)
demoted_user_ids << u.id
Promotion.new(u).change_trust_level!(:regular)
end
end
# Promotions
User.real.where(trust_level: TrustLevel.levels[:regular]).where.not(id: demoted_user_ids).find_each do |u|
Promotion.new(u).review_regular
end
end
end
end

View File

@ -4,8 +4,9 @@ class LeaderRequirements
include ActiveModel::Serialization
attr_accessor :time_period,
:days_visited, :min_days_visited,
TIME_PERIOD = 100 # days
attr_accessor :days_visited, :min_days_visited,
:num_topics_replied_to, :min_topics_replied_to,
:topics_viewed, :min_topics_viewed,
:posts_read, :min_posts_read,
@ -20,54 +21,86 @@ class LeaderRequirements
num_topics_replied_to >= min_topics_replied_to &&
topics_viewed >= min_topics_viewed &&
posts_read >= min_posts_read &&
num_flagged_posts <= max_flagged_posts
end
def time_period
100 # days
num_flagged_posts <= max_flagged_posts &&
num_flagged_by_users <= max_flagged_by_users
end
def days_visited
@user.user_visits.where("visited_at > ? and posts_read > 0", time_period.days.ago).count
@user.user_visits.where("visited_at > ? and posts_read > 0", TIME_PERIOD.days.ago).count
end
def min_days_visited
time_period * 0.5
(TIME_PERIOD * (SiteSetting.leader_requires_days_visited.to_f / 100.0)).to_i
end
def num_topics_replied_to
@user.posts.select('distinct topic_id').where('created_at > ? AND post_number > 1', time_period.days.ago).count
@user.posts.select('distinct topic_id').where('created_at > ? AND post_number > 1', TIME_PERIOD.days.ago).count
end
def min_topics_replied_to
10
SiteSetting.leader_requires_topics_replied_to
end
def topics_viewed
View.where('viewed_at > ?', time_period.days.ago).where(user_id: @user.id, parent_type: 'Topic').select('distinct(parent_id)').count
View.where('viewed_at > ?', TIME_PERIOD.days.ago).where(user_id: @user.id, parent_type: 'Topic').select('distinct(parent_id)').count
end
def min_topics_viewed
(Topic.listable_topics.visible.created_since(time_period.days.ago).count * 0.25).round
(LeaderRequirements.num_topics_in_time_period.to_i * (SiteSetting.leader_requires_topics_viewed.to_f / 100.0)).round
end
def posts_read
@user.user_visits.where('visited_at > ?', time_period.days.ago).pluck(:posts_read).sum
@user.user_visits.where('visited_at > ?', TIME_PERIOD.days.ago).pluck(:posts_read).sum
end
def min_posts_read
(Post.public_posts.visible.created_since(time_period.days.ago).count * 0.25).round
(LeaderRequirements.num_posts_in_time_period.to_i * (SiteSetting.leader_requires_posts_read.to_f / 100.0)).round
end
def num_flagged_posts
# Count the number of posts that were flagged, and moderators explicitly agreed with the flags
# by clicking the "Agree (hide post + send PM)" or "Defer" (on an automatically hidden post) buttons.
# In both cases, the defer flag is set to true.
post_ids = @user.posts.with_deleted.where('created_at > ? AND (spam_count > 0 OR inappropriate_count > 0)', time_period.days.ago).pluck(:id)
PostAction.with_deleted.where(post_id: post_ids).where(defer: true).pluck(:post_id).uniq.count
PostAction.with_deleted.where(post_id: flagged_post_ids).where.not(user_id: @user.id).pluck(:post_id).uniq.count
end
def max_flagged_posts
5
SiteSetting.leader_requires_max_flagged
end
def num_flagged_by_users
PostAction.with_deleted.where(post_id: flagged_post_ids).where.not(user_id: @user.id).pluck(:user_id).uniq.count
end
def max_flagged_by_users
SiteSetting.leader_requires_max_flagged
end
def self.clear_cache
$redis.del NUM_TOPICS_KEY
$redis.del NUM_POSTS_KEY
end
CACHE_DURATION = 1.day.seconds - 60
NUM_TOPICS_KEY = "tl3_num_topics"
NUM_POSTS_KEY = "tl3_num_posts"
def self.num_topics_in_time_period
$redis.get(NUM_TOPICS_KEY) || begin
count = Topic.listable_topics.visible.created_since(TIME_PERIOD.days.ago).count
$redis.setex NUM_TOPICS_KEY, CACHE_DURATION, count
count
end
end
def self.num_posts_in_time_period
$redis.get(NUM_POSTS_KEY) || begin
count = Post.public_posts.visible.created_since(TIME_PERIOD.days.ago).count
$redis.setex NUM_POSTS_KEY, CACHE_DURATION, count
count
end
end
def flagged_post_ids
# (TODO? and moderators explicitly agreed with the flags)
@user.posts.with_deleted.where('created_at > ? AND (spam_count > 0 OR inappropriate_count > 0)', TIME_PERIOD.days.ago).pluck(:id)
end
end

View File

@ -5,7 +5,12 @@ class LeaderRequirementsSerializer < ApplicationSerializer
:num_topics_replied_to, :min_topics_replied_to,
:topics_viewed, :min_topics_viewed,
:posts_read, :min_posts_read,
:num_flagged_posts, :max_flagged_posts
:num_flagged_posts, :max_flagged_posts,
:num_flagged_by_users, :max_flagged_by_users
def time_period
LeaderRequirements::TIME_PERIOD
end
def requirements_met
object.requirements_met?

View File

@ -1800,6 +1800,11 @@ en:
topics_viewed: "Topics Viewed"
posts_read: "Posts Read"
flagged_posts: "Flagged Posts"
flagged_by_users: "Users Who Flagged"
qualifies: "Qualifies for trust level 3."
will_be_promoted: "Will be promoted within 24 hours."
does_not_qualify: "Doesn't qualify for trust level 3."
will_be_demoted: "Will be demoted within 24 hours."
site_content:
none: "Choose a type of content to begin editing."

View File

@ -802,6 +802,12 @@ en:
regular_requires_likes_given: "How many likes a basic user must cast before promotion to regular (2) trust level"
regular_requires_topic_reply_count: "How many topics a basic user must reply to before promotion to regular (2) trust level"
leader_requires_days_visited: "Minimum number of days that a user needs to have visited the site in the last 100 days to qualify for promotion to leader (3) trust level. (0 to 100)"
leader_requires_topics_replied_to: "Minimum number of topics a user needs to have replied to in the last 100 days to qualify for promotion to leader (3) trust level. (0 or higher)"
leader_requires_topics_viewed: "The percentage of topics created in the last 100 days that a user needs to have viewed to qualify for promotion to leader (3) trust level. (0 to 100)"
leader_requires_posts_read: "The percentage of posts created in the last 100 days that a user needs to have viewed to qualify for promotion to leader (3) trust level. (0 to 100)"
leader_requires_max_flagged: "User must not have had more than x posts flagged by x different users in the last 100 days to qualify for promotion to leader (3) trust level, where x is this setting's value. (0 or higher)"
min_trust_to_create_topic: "The minimum trust level required to create a new topic."
min_trust_to_edit_wiki_post: "The minimum trust level required to edit post marked as wiki."

View File

@ -452,6 +452,24 @@ trust:
regular_requires_likes_received: 1
regular_requires_likes_given: 1
regular_requires_topic_reply_count: 3
leader_requires_days_visited:
default: 50
min: 1
max: 100
leader_requires_topics_replied_to:
default: 10
min: 0
leader_requires_topics_viewed:
default: 25
min: 0
max: 100
leader_requires_posts_read:
default: 25
min: 0
max: 100
leader_requires_max_flagged:
default: 5
min: 0
security:
use_https: false

View File

@ -14,6 +14,9 @@ class Promotion
# nil users are never promoted
return false if @user.blank?
# Promotion beyond basic requires some expensive queries, so don't do that here.
return false if @user.trust_level >= TrustLevel.levels[:regular]
trust_key = TrustLevel.levels[@user.trust_level]
review_method = :"review_#{trust_key.to_s}"
@ -30,6 +33,10 @@ class Promotion
Promotion.regular_met?(@user) && change_trust_level!(:regular)
end
def review_regular
Promotion.leader_met?(@user) && change_trust_level!(:leader)
end
def change_trust_level!(level, opts = {})
raise "Invalid trust level #{level}" unless TrustLevel.valid_level?(level)
@ -88,4 +95,8 @@ class Promotion
return true
end
def self.leader_met?(user)
LeaderRequirements.new(user).requirements_met?
end
end

View File

@ -3,6 +3,16 @@ require 'promotion'
describe Promotion do
describe "review" do
it "skips regular users" do
# Reviewing users at higher trust levels is expensive, so trigger those reviews in a background job.
regular = Fabricate.build(:user, trust_level: TrustLevel.levels[:regular])
promotion = described_class.new(regular)
promotion.expects(:review_regular).never
promotion.review
end
end
context "newuser" do
let(:user) { Fabricate(:user, trust_level: TrustLevel.levels[:newuser])}
@ -88,4 +98,42 @@ describe Promotion do
end
context "regular" do
let(:user) { Fabricate(:user, trust_level: TrustLevel.levels[:regular])}
let(:promotion) { Promotion.new(user) }
context "doesn't qualify for promotion" do
before do
LeaderRequirements.any_instance.expects(:requirements_met?).at_least_once.returns(false)
end
it "review_regular returns false" do
expect {
promotion.review_regular.should == false
}.to_not change { user.reload.trust_level }
end
it "doesn't promote" do
expect {
promotion.review_regular
}.to_not change { user.reload.trust_level }
end
end
context "qualifies for promotion" do
before do
LeaderRequirements.any_instance.expects(:requirements_met?).at_least_once.returns(true)
end
it "review_regular returns true" do
promotion.review_regular.should == true
end
it "promotes to leader" do
promotion.review_regular.should == true
user.reload.trust_level.should == TrustLevel.levels[:leader]
end
end
end
end

View File

@ -0,0 +1,28 @@
require 'spec_helper'
describe Jobs::LeaderPromotions do
subject(:run_job) { described_class.new.execute({}) }
it "promotes tl2 user who qualifies for tl3" do
tl2_user = Fabricate(:user, trust_level: TrustLevel.levels[:regular])
LeaderRequirements.any_instance.stubs(:requirements_met?).returns(true)
Promotion.any_instance.expects(:change_trust_level!).with(:leader, anything).once
run_job
end
it "doesn't promote tl1 and tl0 users who have met tl3 requirements" do
tl1_user = Fabricate(:user, trust_level: TrustLevel.levels[:basic])
tl0_user = Fabricate(:user, trust_level: TrustLevel.levels[:newuser])
LeaderRequirements.any_instance.expects(:requirements_met?).never
Promotion.any_instance.expects(:change_trust_level!).never
run_job
end
it "demotes tl3 user who doesn't qualify for tl3 anymore" do
tl3_user = Fabricate(:user, trust_level: TrustLevel.levels[:leader])
LeaderRequirements.any_instance.stubs(:requirements_met?).returns(false)
Promotion.any_instance.expects(:change_trust_level!).with(:regular, anything).once
run_job
end
end

View File

@ -0,0 +1,118 @@
require 'spec_helper'
describe LeaderRequirements do
let(:user) { Fabricate.build(:user) }
subject(:leader_requirements) { described_class.new(user) }
before do
described_class.clear_cache
end
describe "requirements" do
it "min_days_visited uses site setting" do
SiteSetting.stubs(:leader_requires_days_visited).returns(66)
leader_requirements.min_days_visited.should == 66
end
it "min_topics_replied_to uses site setting" do
SiteSetting.stubs(:leader_requires_topics_replied_to).returns(12)
leader_requirements.min_topics_replied_to.should == 12
end
it "min_topics_viewed depends on site setting and number of topics created" do
SiteSetting.stubs(:leader_requires_topics_viewed).returns(75)
described_class.stubs(:num_topics_in_time_period).returns(31)
leader_requirements.min_topics_viewed.should == 23
end
it "min_posts_read depends on site setting and number of posts created" do
SiteSetting.stubs(:leader_requires_posts_read).returns(66)
described_class.stubs(:num_posts_in_time_period).returns(1234)
leader_requirements.min_posts_read.should == 814
end
it "max_flagged_posts depends on site setting" do
SiteSetting.stubs(:leader_requires_max_flagged).returns(3)
leader_requirements.max_flagged_posts.should == 3
end
end
describe "days_visited" do
it "counts visits when posts were read no further back than 100 days ago" do
user.save
user.update_posts_read!(1, 2.days.ago)
user.update_posts_read!(1, 3.days.ago)
user.update_posts_read!(0, 4.days.ago)
user.update_posts_read!(3, 101.days.ago)
leader_requirements.days_visited.should == 2
end
end
describe "num_topics_replied_to" do
it "counts topics in which user replied in last 100 days" do
user.save
not_a_reply = create_post(user: user) # user created the topic, so it doesn't count
topic1 = create_post.topic
reply1 = create_post(topic: topic1, user: user)
reply_again = create_post(topic: topic1, user: user) # two replies in one topic
topic2 = create_post(created_at: 101.days.ago).topic
reply2 = create_post(topic: topic2, user: user, created_at: 101.days.ago) # topic is over 100 days old
leader_requirements.num_topics_replied_to.should == 1
end
end
describe "topics_viewed" do
def make_view(id, at, user_id)
View.create!(parent_id: id, parent_type: 'Topic', ip_address: '11.22.33.44', viewed_at: at, user_id: user_id)
end
it "counts topics views within last 100 days, not counting a topic more than once" do
user.save
make_view(9, 1.day.ago, user.id)
make_view(9, 3.days.ago, user.id) # same topic, different day
make_view(3, 4.days.ago, user.id)
make_view(2, 101.days.ago, user.id) # too long ago
leader_requirements.topics_viewed.should == 2
end
end
describe "posts_read" do
it "counts posts read within the last 100 days" do
user.save
user.update_posts_read!(3, 2.days.ago)
user.update_posts_read!(1, 3.days.ago)
user.update_posts_read!(0, 4.days.ago)
user.update_posts_read!(5, 101.days.ago)
leader_requirements.posts_read.should == 4
end
end
context "with flagged posts" do
before do
user.save
flags = [:off_topic, :inappropriate, :notify_user, :notify_moderators, :spam].map do |t|
Fabricate(:flag, post: Fabricate(:post, user: user), post_action_type_id: PostActionType.types[t])
end
# Same post, different user:
Fabricate(:flag, post: flags[1].post, post_action_type_id: PostActionType.types[:spam])
# Flagged their own post:
Fabricate(:flag, user: user, post: Fabricate(:post, user: user), post_action_type_id: PostActionType.types[:spam])
# More than 100 days ago:
Fabricate(:flag, post: Fabricate(:post, user: user, created_at: 101.days.ago), post_action_type_id: PostActionType.types[:spam], created_at: 101.days.ago)
end
it "num_flagged_posts and num_flagged_by_users count spam and inappropriate flags in the last 100 days" do
leader_requirements.num_flagged_posts.should == 2
leader_requirements.num_flagged_by_users.should == 3
end
end
end