2014-03-05 06:52:20 -06:00
|
|
|
class BadgeGranter
|
|
|
|
|
|
|
|
def initialize(badge, user, opts={})
|
|
|
|
@badge, @user, @opts = badge, user, opts
|
|
|
|
@granted_by = opts[:granted_by] || Discourse.system_user
|
2014-06-17 01:29:49 -05:00
|
|
|
@post_id = opts[:post_id]
|
2014-03-05 06:52:20 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.grant(badge, user, opts={})
|
|
|
|
BadgeGranter.new(badge, user, opts).grant
|
|
|
|
end
|
|
|
|
|
|
|
|
def grant
|
|
|
|
return if @granted_by and !Guardian.new(@granted_by).can_grant_badges?(@user)
|
|
|
|
|
2014-06-17 01:29:49 -05:00
|
|
|
user_badge = UserBadge.find_by(badge_id: @badge.id, user_id: @user.id, post_id: @post_id)
|
2014-03-05 06:52:20 -06:00
|
|
|
|
2014-06-27 14:02:09 -05:00
|
|
|
if user_badge.nil? || (@badge.multiple_grant? && @post_id.nil?)
|
2014-04-14 00:58:27 -05:00
|
|
|
UserBadge.transaction do
|
|
|
|
user_badge = UserBadge.create!(badge: @badge, user: @user,
|
2014-06-17 01:29:49 -05:00
|
|
|
granted_by: @granted_by,
|
|
|
|
granted_at: Time.now,
|
|
|
|
post_id: @post_id)
|
2014-03-05 06:52:20 -06:00
|
|
|
|
2014-04-14 00:58:27 -05:00
|
|
|
if @granted_by != Discourse.system_user
|
|
|
|
StaffActionLogger.new(@granted_by).log_badge_grant(user_badge)
|
|
|
|
end
|
2014-04-16 14:59:45 -05:00
|
|
|
|
2014-05-04 13:15:38 -05:00
|
|
|
if SiteSetting.enable_badges?
|
2014-06-17 00:59:28 -05:00
|
|
|
notification = @user.notifications.create(notification_type: Notification.types[:granted_badge], data: { badge_id: @badge.id, badge_name: @badge.name }.to_json)
|
|
|
|
user_badge.update_attributes notification_id: notification.id
|
2014-05-04 13:15:38 -05:00
|
|
|
end
|
2014-03-19 14:30:12 -05:00
|
|
|
end
|
2014-03-05 06:52:20 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
user_badge
|
|
|
|
end
|
|
|
|
|
2014-03-19 14:30:12 -05:00
|
|
|
def self.revoke(user_badge, options={})
|
2014-03-05 06:52:20 -06:00
|
|
|
UserBadge.transaction do
|
|
|
|
user_badge.destroy!
|
2014-03-19 14:30:12 -05:00
|
|
|
if options[:revoked_by]
|
|
|
|
StaffActionLogger.new(options[:revoked_by]).log_badge_revoke(user_badge)
|
|
|
|
end
|
2014-04-17 22:10:53 -05:00
|
|
|
|
|
|
|
# If the user's title is the same as the badge name, remove their title.
|
|
|
|
if user_badge.user.title == user_badge.badge.name
|
|
|
|
user_badge.user.title = nil
|
|
|
|
user_badge.user.save!
|
|
|
|
end
|
2014-03-05 06:52:20 -06:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-06-17 02:07:46 -05:00
|
|
|
def self.update_badges(args)
|
|
|
|
Jobs.enqueue(:update_badges, args)
|
2014-05-04 13:15:38 -05:00
|
|
|
end
|
|
|
|
|
2014-07-03 02:29:44 -05:00
|
|
|
def self.backfill(badge)
|
|
|
|
return unless badge.query.present?
|
|
|
|
|
|
|
|
post_clause = badge.target_posts ? "AND q.post_id = ub.post_id" : ""
|
|
|
|
post_id_field = badge.target_posts ? "q.post_id" : "NULL"
|
|
|
|
|
|
|
|
sql = "DELETE FROM user_badges
|
|
|
|
WHERE id in (
|
|
|
|
SELECT ub.id
|
|
|
|
FROM user_badges ub
|
|
|
|
LEFT JOIN ( #{badge.query} ) q
|
|
|
|
ON q.user_id = ub.user_id
|
|
|
|
#{post_clause}
|
|
|
|
WHERE ub.id = :id AND q.user_id IS NULL
|
|
|
|
)"
|
|
|
|
|
|
|
|
Badge.exec_sql(sql, id: badge.id)
|
|
|
|
|
|
|
|
sql = "INSERT INTO user_badges(badge_id, user_id, granted_at, granted_by_id, post_id)
|
|
|
|
SELECT :id, q.user_id, q.granted_at, -1, #{post_id_field}
|
|
|
|
FROM ( #{badge.query} ) q
|
|
|
|
LEFT JOIN user_badges ub ON
|
|
|
|
ub.id = :id AND ub.user_id = q.user_id
|
|
|
|
#{post_clause}
|
|
|
|
WHERE ub.id IS NULL"
|
|
|
|
|
|
|
|
Badge.exec_sql(sql, id: badge.id)
|
|
|
|
|
|
|
|
badge.reset_grant_count!
|
2014-07-01 07:00:31 -05:00
|
|
|
|
|
|
|
end
|
|
|
|
|
2014-03-05 06:52:20 -06:00
|
|
|
end
|