2019-05-03 06:17:27 +08:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
#
|
|
|
|
# Check whether a user is ready for a new trust level.
|
|
|
|
#
|
|
|
|
class Promotion
|
|
|
|
|
|
|
|
def initialize(user)
|
2013-02-26 00:42:20 +08:00
|
|
|
@user = user
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
# Review a user for a promotion. Delegates work to a review_#{trust_level} method.
|
|
|
|
# Returns true if the user was promoted, false otherwise.
|
|
|
|
def review
|
|
|
|
# nil users are never promoted
|
2017-11-24 04:55:44 +08:00
|
|
|
return false if @user.blank? || !@user.manual_locked_trust_level.nil?
|
2013-02-06 03:16:51 +08:00
|
|
|
|
2014-06-28 00:26:03 +08:00
|
|
|
# Promotion beyond basic requires some expensive queries, so don't do that here.
|
2014-09-05 13:20:39 +08:00
|
|
|
return false if @user.trust_level >= TrustLevel[2]
|
2014-06-28 00:26:03 +08:00
|
|
|
|
2014-09-05 13:20:39 +08:00
|
|
|
review_method = :"review_tl#{@user.trust_level}"
|
2019-05-07 10:22:37 +08:00
|
|
|
return public_send(review_method) if respond_to?(review_method)
|
2013-02-06 03:16:51 +08:00
|
|
|
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
2014-09-05 13:20:39 +08:00
|
|
|
def review_tl0
|
2018-06-23 00:51:07 +08:00
|
|
|
if Promotion.tl1_met?(@user) && change_trust_level!(TrustLevel[1])
|
2018-09-20 03:52:45 +08:00
|
|
|
@user.enqueue_member_welcome_message unless @user.badges.where(id: Badge::BasicUser).count > 0
|
2018-06-23 00:51:07 +08:00
|
|
|
return true
|
|
|
|
end
|
|
|
|
false
|
2014-06-17 08:46:30 +08:00
|
|
|
end
|
|
|
|
|
2014-09-05 13:20:39 +08:00
|
|
|
def review_tl1
|
|
|
|
Promotion.tl2_met?(@user) && change_trust_level!(TrustLevel[2])
|
2014-06-17 08:46:30 +08:00
|
|
|
end
|
|
|
|
|
2014-09-05 13:20:39 +08:00
|
|
|
def review_tl2
|
|
|
|
Promotion.tl3_met?(@user) && change_trust_level!(TrustLevel[3])
|
2014-06-28 00:26:03 +08:00
|
|
|
end
|
|
|
|
|
2014-06-17 08:46:30 +08:00
|
|
|
def change_trust_level!(level, opts = {})
|
2014-09-05 13:20:39 +08:00
|
|
|
raise "Invalid trust level #{level}" unless TrustLevel.valid?(level)
|
2014-06-17 08:46:30 +08:00
|
|
|
|
|
|
|
old_level = @user.trust_level
|
2014-09-05 13:20:39 +08:00
|
|
|
new_level = level
|
2014-06-17 08:46:30 +08:00
|
|
|
|
2017-11-24 04:55:44 +08:00
|
|
|
if new_level < old_level && @user.manual_locked_trust_level.nil?
|
2017-07-28 09:20:09 +08:00
|
|
|
next_up = new_level + 1
|
2014-09-05 13:20:39 +08:00
|
|
|
key = "tl#{next_up}_met?"
|
2019-05-07 09:27:05 +08:00
|
|
|
if self.class.respond_to?(key) && self.class.public_send(key, @user)
|
2014-06-17 08:46:30 +08:00
|
|
|
raise Discourse::InvalidAccess.new, I18n.t('trust_levels.change_failed_explanation',
|
|
|
|
user_name: @user.name,
|
|
|
|
new_trust_level: new_level,
|
|
|
|
current_trust_level: old_level)
|
|
|
|
end
|
|
|
|
end
|
2013-02-06 03:16:51 +08:00
|
|
|
|
2014-06-17 08:46:30 +08:00
|
|
|
admin = opts && opts[:log_action_for]
|
|
|
|
|
|
|
|
@user.trust_level = new_level
|
2014-06-17 08:58:21 +08:00
|
|
|
@user.user_profile.bio_raw_will_change! # So it can get re-cooked based on the new trust level
|
2014-06-17 08:46:30 +08:00
|
|
|
|
|
|
|
@user.transaction do
|
|
|
|
if admin
|
|
|
|
StaffActionLogger.new(admin).log_trust_level_change(@user, old_level, new_level)
|
2014-07-09 05:39:36 +08:00
|
|
|
else
|
2017-07-28 09:20:09 +08:00
|
|
|
UserHistory.create!(action: UserHistory.actions[:auto_trust_level_change],
|
|
|
|
target_user_id: @user.id,
|
|
|
|
previous_value: old_level,
|
|
|
|
new_value: new_level)
|
2014-06-17 08:46:30 +08:00
|
|
|
end
|
|
|
|
@user.save!
|
|
|
|
@user.user_profile.recook_bio
|
|
|
|
@user.user_profile.save!
|
2019-11-14 05:31:49 +08:00
|
|
|
DiscourseEvent.trigger(:user_promoted, user_id: @user.id, new_trust_level: new_level, old_trust_level: old_level)
|
2014-06-17 08:46:30 +08:00
|
|
|
Group.user_trust_level_change!(@user.id, @user.trust_level)
|
2014-07-23 09:42:24 +08:00
|
|
|
BadgeGranter.queue_badge_grant(Badge::Trigger::TrustLevelChange, user: @user)
|
2014-06-17 08:46:30 +08:00
|
|
|
end
|
2013-02-26 00:42:20 +08:00
|
|
|
|
2013-02-06 03:16:51 +08:00
|
|
|
true
|
|
|
|
end
|
|
|
|
|
2014-09-05 13:20:39 +08:00
|
|
|
def self.tl2_met?(user)
|
2014-06-17 08:46:30 +08:00
|
|
|
stat = user.user_stat
|
2014-09-05 04:16:46 +08:00
|
|
|
return false if stat.topics_entered < SiteSetting.tl2_requires_topics_entered
|
|
|
|
return false if stat.posts_read_count < SiteSetting.tl2_requires_read_posts
|
|
|
|
return false if (stat.time_read / 60) < SiteSetting.tl2_requires_time_spent_mins
|
2015-04-15 00:05:09 +08:00
|
|
|
return false if ((Time.now - user.created_at) / 60) < SiteSetting.tl2_requires_time_spent_mins
|
2014-09-05 04:16:46 +08:00
|
|
|
return false if stat.days_visited < SiteSetting.tl2_requires_days_visited
|
|
|
|
return false if stat.likes_received < SiteSetting.tl2_requires_likes_received
|
|
|
|
return false if stat.likes_given < SiteSetting.tl2_requires_likes_given
|
2020-05-15 06:42:00 +08:00
|
|
|
return false if stat.calc_topic_reply_count!(SiteSetting.tl2_requires_topic_reply_count) < SiteSetting.tl2_requires_topic_reply_count
|
2013-04-05 12:29:46 +08:00
|
|
|
|
2014-06-17 08:46:30 +08:00
|
|
|
true
|
|
|
|
end
|
|
|
|
|
2014-09-05 13:20:39 +08:00
|
|
|
def self.tl1_met?(user)
|
2014-06-17 08:46:30 +08:00
|
|
|
stat = user.user_stat
|
2014-09-05 04:16:46 +08:00
|
|
|
return false if stat.topics_entered < SiteSetting.tl1_requires_topics_entered
|
|
|
|
return false if stat.posts_read_count < SiteSetting.tl1_requires_read_posts
|
|
|
|
return false if (stat.time_read / 60) < SiteSetting.tl1_requires_time_spent_mins
|
2015-04-15 00:05:09 +08:00
|
|
|
return false if ((Time.now - user.created_at) / 60) < SiteSetting.tl1_requires_time_spent_mins
|
|
|
|
|
2019-11-15 04:10:51 +08:00
|
|
|
true
|
2013-04-05 12:29:46 +08:00
|
|
|
end
|
|
|
|
|
2014-09-05 13:20:39 +08:00
|
|
|
def self.tl3_met?(user)
|
|
|
|
TrustLevel3Requirements.new(user).requirements_met?
|
2014-06-28 00:26:03 +08:00
|
|
|
end
|
|
|
|
|
2014-09-05 13:20:39 +08:00
|
|
|
def self.tl3_lost?(user)
|
|
|
|
TrustLevel3Requirements.new(user).requirements_lost?
|
2014-07-28 14:42:38 +08:00
|
|
|
end
|
|
|
|
|
2017-11-24 04:55:44 +08:00
|
|
|
# Figure out what a user's trust level should be from scratch
|
|
|
|
def self.recalculate(user, performed_by = nil)
|
|
|
|
# First, use the manual locked level
|
|
|
|
unless user.manual_locked_trust_level.nil?
|
2018-04-10 14:19:08 +08:00
|
|
|
return user.update!(
|
|
|
|
trust_level: user.manual_locked_trust_level
|
|
|
|
)
|
2017-11-24 04:55:44 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
# Then consider the group locked level
|
2018-08-25 06:41:03 +08:00
|
|
|
user_group_granted_trust_level = user.group_granted_trust_level
|
|
|
|
|
|
|
|
unless user_group_granted_trust_level.blank?
|
2018-04-10 14:19:08 +08:00
|
|
|
return user.update!(
|
2018-08-25 06:41:03 +08:00
|
|
|
trust_level: user_group_granted_trust_level
|
2018-04-10 14:19:08 +08:00
|
|
|
)
|
2017-11-24 04:55:44 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
user.update_column(:trust_level, TrustLevel[0])
|
|
|
|
|
|
|
|
p = Promotion.new(user)
|
|
|
|
p.review_tl0
|
|
|
|
p.review_tl1
|
|
|
|
p.review_tl2
|
|
|
|
if user.trust_level == 3 && Promotion.tl3_lost?(user)
|
|
|
|
user.change_trust_level!(2, log_action_for: performed_by || Discourse.system_user)
|
|
|
|
end
|
|
|
|
end
|
2013-02-06 03:16:51 +08:00
|
|
|
end
|