FIX: Badge and user title interaction fixes (#8282)

* Fix user title logic when badge name customized
* Fix an issue where a user's title was not considered a badge granted title when the user used a badge for their title and the badge name was customized. this affected the effectiveness of revoke_ungranted_titles! which only operates on badge_granted_titles.
* When a user's title is set now it is considered a badge_granted_title if the badge name OR the badge custom name from TranslationOverride is the same as the title
* When a user's badge is revoked we now also revoke their title if the user's title matches the badge name OR the badge custom name from TranslationOverride
* Add a user history log when the title is revoked to remove confusion about why titles are revoked
* Add granted_title_badge_id to user_profile, now when we set badge_granted_title on a user profile when updating a user's title based on a badge, we also remember which badge matched the title
* When badge name (or custom text) changes update titles of users in a background job
* When the name of a badge changes, or in the case of system badges when their custom translation text changes, then we need to update the title of all corresponding users who have a badge_granted_title and matching granted_title_badge_id. In the case of system badges we need to first get the proper badge ID based on the translation key e.g. badges.regular.name
* Add migration to backfill all granted_title_badge_ids for both normal badge name titles and titles using custom badge text.
This commit is contained in:
Martin Brennan 2019-11-08 15:34:24 +10:00 committed by GitHub
parent 64b4a7ba45
commit 56d3e29a69
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
19 changed files with 397 additions and 14 deletions

View File

@ -125,6 +125,15 @@ class Admin::BadgesController < Admin::AdminController
badge.save! badge.save!
end end
if opts[:new].blank?
Jobs.enqueue(
:bulk_user_title_update,
new_title: badge.name,
granted_badge_id: badge.id,
action: Jobs::BulkUserTitleUpdate::UPDATE_ACTION
)
end
errors errors
rescue ActiveRecord::RecordInvalid rescue ActiveRecord::RecordInvalid
errors.push(*badge.errors.full_messages) errors.push(*badge.errors.full_messages)

View File

@ -59,6 +59,15 @@ class Admin::SiteTextsController < Admin::AdminController
if translation_override.errors.empty? if translation_override.errors.empty?
StaffActionLogger.new(current_user).log_site_text_change(id, value, old_value) StaffActionLogger.new(current_user).log_site_text_change(id, value, old_value)
system_badge_id = Badge.find_system_badge_id_from_translation_key(id)
if system_badge_id.present?
Jobs.enqueue(
:bulk_user_title_update,
new_title: value,
granted_badge_id: system_badge_id,
action: Jobs::BulkUserTitleUpdate::UPDATE_ACTION
)
end
render_serialized(site_text, SiteTextSerializer, root: 'site_text', rest_serializer: true) render_serialized(site_text, SiteTextSerializer, root: 'site_text', rest_serializer: true)
else else
render json: failed_json.merge( render json: failed_json.merge(
@ -69,10 +78,19 @@ class Admin::SiteTextsController < Admin::AdminController
def revert def revert
site_text = find_site_text site_text = find_site_text
old_text = I18n.t(site_text[:id]) id = site_text[:id]
TranslationOverride.revert!(I18n.locale, site_text[:id]) old_text = I18n.t(id)
TranslationOverride.revert!(I18n.locale, id)
site_text = find_site_text site_text = find_site_text
StaffActionLogger.new(current_user).log_site_text_change(site_text[:id], site_text[:value], old_text) StaffActionLogger.new(current_user).log_site_text_change(id, site_text[:value], old_text)
system_badge_id = Badge.find_system_badge_id_from_translation_key(id)
if system_badge_id.present?
Jobs.enqueue(
:bulk_user_title_update,
granted_badge_id: system_badge_id,
action: Jobs::BulkUserTitleUpdate::RESET_ACTION
)
end
render_serialized(site_text, SiteTextSerializer, root: 'site_text', rest_serializer: true) render_serialized(site_text, SiteTextSerializer, root: 'site_text', rest_serializer: true)
end end

View File

@ -195,14 +195,36 @@ class UsersController < ApplicationController
guardian.ensure_can_edit!(user) guardian.ensure_can_edit!(user)
user_badge = UserBadge.find_by(id: params[:user_badge_id]) user_badge = UserBadge.find_by(id: params[:user_badge_id])
previous_title = user.title
if user_badge && user_badge.user == user && user_badge.badge.allow_title? if user_badge && user_badge.user == user && user_badge.badge.allow_title?
user.title = user_badge.badge.display_name user.title = user_badge.badge.display_name
user.user_profile.badge_granted_title = true
user.save! user.save!
user.user_profile.save!
log_params = {
details: "title matching badge id #{user_badge.badge.id}",
previous_value: previous_title,
new_value: user.title
}
if current_user.staff? && current_user != user
StaffActionLogger.new(current_user).log_title_change(user, log_params)
else
UserHistory.create!(log_params.merge(target_user_id: user.id, action: UserHistory.actions[:change_title]))
end
else else
user.title = '' user.title = ''
user.save! user.save!
log_params = {
revoke_reason: 'user title was same as revoked badge name or custom badge name',
previous_value: previous_title
}
if current_user.staff? && current_user != user
StaffActionLogger.new(current_user).log_title_revoke(user, log_params)
else
UserHistory.create!(log_params.merge(target_user_id: user.id, action: UserHistory.actions[:revoke_title]))
end
end end
render body: nil render body: nil

View File

@ -0,0 +1,51 @@
# frozen_string_literal: true
module Jobs
class BulkUserTitleUpdate < ::Jobs::Base
UPDATE_ACTION = 'update'.freeze
RESET_ACTION = 'reset'.freeze
def execute(args)
new_title = args[:new_title]
granted_badge_id = args[:granted_badge_id]
action = args[:action]
case action
when UPDATE_ACTION
update_titles_for_granted_badge(new_title, granted_badge_id)
when RESET_ACTION
reset_titles_for_granted_badge(granted_badge_id)
end
end
private
##
# If a badge name or a system badge TranslationOverride changes
# then we need to set all titles granted based on that badge to
# the new name or custom translation
def update_titles_for_granted_badge(new_title, granted_badge_id)
DB.exec(<<~SQL, granted_title_badge_id: granted_badge_id, title: new_title, updated_at: Time.now)
UPDATE users AS u
SET title = :title, updated_at = :updated_at
FROM user_profiles AS up
WHERE up.user_id = u.id AND up.granted_title_badge_id = :granted_title_badge_id
SQL
end
##
# Reset granted titles for a badge back to the original
# badge name. When a system badge has its TranslationOverride
# revoked we want to have all titles based on that translation
# for the badge reset.
def reset_titles_for_granted_badge(granted_badge_id)
DB.exec(<<~SQL, granted_title_badge_id: granted_badge_id, updated_at: Time.now)
UPDATE users AS u
SET title = badges.name, updated_at = :updated_at
FROM user_profiles AS up
INNER JOIN badges ON badges.id = up.granted_title_badge_id
WHERE up.user_id = u.id AND up.granted_title_badge_id = :granted_title_badge_id
SQL
end
end
end

View File

@ -169,8 +169,17 @@ class Badge < ActiveRecord::Base
end end
def self.display_name(name) def self.display_name(name)
key = "badges.#{i18n_name(name)}.name" I18n.t(i18n_key(name), default: name)
I18n.t(key, default: name) end
def self.i18n_key(name)
"badges.#{i18n_name(name)}.name"
end
def self.find_system_badge_id_from_translation_key(translation_key)
return unless translation_key.starts_with?('badges.')
badge_name_klass = translation_key.split('.').second.camelize
"Badge::#{badge_name_klass}".constantize
end end
def awarded_for_trust_level? def awarded_for_trust_level?
@ -208,6 +217,10 @@ class Badge < ActiveRecord::Base
self.class.display_name(name) self.class.display_name(name)
end end
def translation_key
self.class.i18n_key(name)
end
def long_description def long_description
key = "badges.#{i18n_name}.long_description" key = "badges.#{i18n_name}.long_description"
I18n.t(key, default: self[:long_description] || '', base_uri: Discourse.base_uri) I18n.t(key, default: self[:long_description] || '', base_uri: Discourse.base_uri)

View File

@ -1478,8 +1478,13 @@ class User < ActiveRecord::Base
def check_if_title_is_badged_granted def check_if_title_is_badged_granted
if title_changed? && !new_record? && user_profile if title_changed? && !new_record? && user_profile
badge_granted_title = title.present? && badges.where(allow_title: true, name: title).exists? badge_matching_title = title && badges.find do |badge|
user_profile.update_column(:badge_granted_title, badge_granted_title) badge.allow_title? && (badge.display_name == title || badge.name == title)
end
user_profile.update(
badge_granted_title: badge_matching_title.present?,
granted_title_badge_id: badge_matching_title&.id
)
end end
end end

View File

@ -101,6 +101,8 @@ class UserHistory < ActiveRecord::Base
api_key_create: 80, api_key_create: 80,
api_key_update: 81, api_key_update: 81,
api_key_destroy: 82, api_key_destroy: 82,
revoke_title: 83,
change_title: 84
) )
end end
@ -175,9 +177,11 @@ class UserHistory < ActiveRecord::Base
:change_theme_setting, :change_theme_setting,
:disable_theme_component, :disable_theme_component,
:enable_theme_component, :enable_theme_component,
:revoke_title,
:change_title,
:api_key_create, :api_key_create,
:api_key_update, :api_key_update,
:api_key_destroy, :api_key_destroy
] ]
end end

View File

@ -9,6 +9,7 @@ class UserProfile < ActiveRecord::Base
belongs_to :user, inverse_of: :user_profile belongs_to :user, inverse_of: :user_profile
belongs_to :card_background_upload, class_name: "Upload" belongs_to :card_background_upload, class_name: "Upload"
belongs_to :profile_background_upload, class_name: "Upload" belongs_to :profile_background_upload, class_name: "Upload"
belongs_to :granted_title_badge, class_name: "Badge"
validates :bio_raw, length: { maximum: 3000 } validates :bio_raw, length: { maximum: 3000 }
validates :website, url: true, allow_blank: true, if: Proc.new { |c| c.new_record? || c.website_changed? } validates :website, url: true, allow_blank: true, if: Proc.new { |c| c.new_record? || c.website_changed? }
@ -161,15 +162,18 @@ end
# views :integer default(0), not null # views :integer default(0), not null
# profile_background_upload_id :integer # profile_background_upload_id :integer
# card_background_upload_id :integer # card_background_upload_id :integer
# granted_title_badge_id :integer
# #
# Indexes # Indexes
# #
# index_user_profiles_on_bio_cooked_version (bio_cooked_version) # index_user_profiles_on_bio_cooked_version (bio_cooked_version)
# index_user_profiles_on_card_background (card_background) # index_user_profiles_on_card_background (card_background)
# index_user_profiles_on_profile_background (profile_background) # index_user_profiles_on_profile_background (profile_background)
# index_user_profiles_on_granted_title_badge_id (granted_title_badge)
# #
# Foreign Keys # Foreign Keys
# #
# fk_rails_... (card_background_upload_id => uploads.id) # fk_rails_... (card_background_upload_id => uploads.id)
# fk_rails_... (profile_background_upload_id => uploads.id) # fk_rails_... (profile_background_upload_id => uploads.id)
# fk_rails_... (granted_title_badge_id => badges.id)
# #

View File

@ -72,8 +72,19 @@ class BadgeGranter
StaffActionLogger.new(options[:revoked_by]).log_badge_revoke(user_badge) StaffActionLogger.new(options[:revoked_by]).log_badge_revoke(user_badge)
end end
# If the user's title is the same as the badge name, remove their title. # If the user's title is the same as the badge name OR the custom badge name, remove their title.
if user_badge.user.title == user_badge.badge.name custom_badge_name = TranslationOverride.find_by(translation_key: user_badge.badge.translation_key)&.value
user_title_is_badge_name = user_badge.user.title == user_badge.badge.name
user_title_is_custom_badge_name = custom_badge_name.present? && user_badge.user.title == custom_badge_name
if user_title_is_badge_name || user_title_is_custom_badge_name
if options[:revoked_by]
StaffActionLogger.new(options[:revoked_by]).log_title_revoke(
user_badge.user,
revoke_reason: 'user title was same as revoked badge name or custom badge name',
previous_value: user_badge.user.title
)
end
user_badge.user.title = nil user_badge.user.title = nil
user_badge.user.save! user_badge.user.save!
end end

View File

@ -352,6 +352,27 @@ class StaffActionLogger
)) ))
end end
def log_title_revoke(user, opts = {})
raise Discourse::InvalidParameters.new(:user) unless user
UserHistory.create!(params(opts).merge(
action: UserHistory.actions[:revoke_title],
target_user_id: user.id,
details: opts[:revoke_reason],
previous_value: opts[:previous_value]
))
end
def log_title_change(user, opts = {})
raise Discourse::InvalidParameters.new(:user) unless user
UserHistory.create!(params(opts).merge(
action: UserHistory.actions[:change_title],
target_user_id: user.id,
details: opts[:details],
new_value: opts[:new_value],
previous_value: opts[:previous_value]
))
end
def log_check_email(user, opts = {}) def log_check_email(user, opts = {})
raise Discourse::InvalidParameters.new(:user) unless user raise Discourse::InvalidParameters.new(:user) unless user
UserHistory.create!(params(opts).merge( UserHistory.create!(params(opts).merge(

View File

@ -3930,6 +3930,8 @@ en:
change_theme_setting: "change theme setting" change_theme_setting: "change theme setting"
disable_theme_component: "disable theme component" disable_theme_component: "disable theme component"
enable_theme_component: "enable theme component" enable_theme_component: "enable theme component"
revoke_title: "revoke title"
change_title: "change title"
api_key_create: "api key create" api_key_create: "api key create"
api_key_update: "api key update" api_key_update: "api key update"
api_key_destroy: "api key destroy" api_key_destroy: "api key destroy"

View File

@ -0,0 +1,35 @@
# frozen_string_literal: true
class AddGrantedTitleBadgeIdToUserProfile < ActiveRecord::Migration[6.0]
def up
add_reference :user_profiles, :granted_title_badge, foreign_key: { to_table: :badges }, index: true, null: true
# update all the regular badge derived titles based
# on the normal badge name
ActiveRecord::Base.connection.execute <<-SQL
UPDATE user_profiles
SET granted_title_badge_id = b.id
FROM users
INNER JOIN badges b ON users.title = b.name
WHERE users.id = user_profiles.user_id
AND user_profiles.granted_title_badge_id IS NULL
SQL
# update all of the system badge derived titles where the
# badge has had custom text set for it via TranslationOverride
ActiveRecord::Base.connection.execute <<-SQL
UPDATE user_profiles
SET granted_title_badge_id = badges.id
FROM users
JOIN translation_overrides ON translation_overrides.value = users.title
JOIN badges ON ('badges.' || LOWER(REPLACE(badges.name, ' ', '_')) || '.name') = translation_overrides.translation_key
JOIN user_badges ON user_badges.user_id = users.id AND user_badges.badge_id = badges.id
WHERE users.id = user_profiles.user_id
AND user_profiles.granted_title_badge_id IS NULL
SQL
end
def down
remove_column :user_profiles, :granted_title_badge_id
end
end

View File

@ -0,0 +1,50 @@
# frozen_string_literal: true
require 'rails_helper'
describe Jobs::BulkUserTitleUpdate do
fab!(:badge) { Fabricate(:badge, name: 'Protector of the Realm', allow_title: true) }
fab!(:user) { Fabricate(:user) }
fab!(:other_user) { Fabricate(:user) }
describe 'update action' do
before do
BadgeGranter.grant(badge, user)
user.update(title: badge.name)
end
it 'updates the title of all users with the attached granted title badge id on their profile' do
execute_update
expect(user.reload.title).to eq('King of the Forum')
end
it 'does not set the title for any other users' do
execute_update
expect(other_user.reload.title).not_to eq('King of the Forum')
end
def execute_update
described_class.new.execute(new_title: 'King of the Forum', granted_badge_id: badge.id, action: described_class::UPDATE_ACTION)
end
end
describe 'reset action' do
let(:customized_badge_name) { 'Merit Badge' }
before do
TranslationOverride.upsert!(I18n.locale, Badge.i18n_key(badge.name), customized_badge_name)
BadgeGranter.grant(badge, user)
user.update(title: customized_badge_name)
end
it 'updates the title of all users back to the original badge name' do
expect(user.reload.title).to eq(customized_badge_name)
described_class.new.execute(granted_badge_id: badge.id, action: described_class::RESET_ACTION)
expect(user.reload.title).to eq('Protector of the Realm')
end
after do
TranslationOverride.revert!(I18n.locale, Badge.i18n_key(badge.name))
end
end
end

View File

@ -95,6 +95,33 @@ describe Badge do
end end
end end
describe '.find_system_badge_id_from_translation_key' do
let(:translation_key) { 'badges.regular.name' }
it 'uses a translation key to get a system badge id, mainly to find which badge a translation override corresponds to' do
expect(Badge.find_system_badge_id_from_translation_key(translation_key)).to eq(
Badge::Regular
)
end
context 'when the translation key is snake case' do
let(:translation_key) { 'badges.crazy_in_love.name' }
it 'works to get the badge' do
expect(Badge.find_system_badge_id_from_translation_key(translation_key)).to eq(
Badge::CrazyInLove
)
end
end
context 'when a translation key not for a badge is provided' do
let(:translation_key) { 'reports.flags.title' }
it 'returns nil' do
expect(Badge.find_system_badge_id_from_translation_key(translation_key)).to eq(nil)
end
end
end
context "First Quote" do context "First Quote" do
let(:quoted_post_badge) do let(:quoted_post_badge) do
Badge.find(Badge::FirstQuote) Badge.find(Badge::FirstQuote)

View File

@ -1957,11 +1957,35 @@ describe User do
expect(user.user_profile.reload.badge_granted_title).to eq(false) expect(user.user_profile.reload.badge_granted_title).to eq(false)
badge.update!(allow_title: true) badge.update!(allow_title: true)
user.badges.reload
user.update!(title: badge.name) user.update!(title: badge.name)
expect(user.user_profile.reload.badge_granted_title).to eq(true) expect(user.user_profile.reload.badge_granted_title).to eq(true)
expect(user.user_profile.reload.granted_title_badge_id).to eq(badge.id)
user.update!(title: nil) user.update!(title: nil)
expect(user.user_profile.reload.badge_granted_title).to eq(false) expect(user.user_profile.reload.badge_granted_title).to eq(false)
expect(user.user_profile.granted_title_badge_id).to eq(nil)
end
context 'when a custom badge name has been set and it matches the title' do
let(:customized_badge_name) { 'Merit Badge' }
before do
TranslationOverride.upsert!(I18n.locale, Badge.i18n_key(badge.name), customized_badge_name)
end
it 'sets badge_granted_title correctly' do
BadgeGranter.grant(badge, user)
badge.update!(allow_title: true)
user.update!(title: customized_badge_name)
expect(user.user_profile.reload.badge_granted_title).to eq(true)
expect(user.user_profile.reload.granted_title_badge_id).to eq(badge.id)
end
after do
TranslationOverride.revert!(I18n.locale, Badge.i18n_key(badge.name))
end
end end
end end

View File

@ -153,6 +153,29 @@ describe Admin::BadgesController do
expect(badge.name).to eq('123456') expect(badge.name).to eq('123456')
expect(badge.query).to eq(sql) expect(badge.query).to eq(sql)
end end
context 'when there is a user with a title granted using the badge' do
fab!(:user_with_badge_title) { Fabricate(:active_user) }
fab!(:badge) { Fabricate(:badge, name: 'Oathbreaker', allow_title: true) }
before do
BadgeGranter.grant(badge, user_with_badge_title)
user_with_badge_title.update(title: 'Oathbreaker')
end
it 'updates the user title in a job' do
Jobs.expects(:enqueue).with(
:bulk_user_title_update,
new_title: 'Shieldbearer',
granted_badge_id: badge.id,
action: Jobs::BulkUserTitleUpdate::UPDATE_ACTION
)
put "/admin/badges/#{badge.id}.json", params: {
name: "Shieldbearer"
}
end
end
end end
end end
end end

View File

@ -415,6 +415,37 @@ RSpec.describe Admin::SiteTextsController do
json = ::JSON.parse(response.body) json = ::JSON.parse(response.body)
expect(json['site_text']['value']).to_not eq(ru_mf_text) expect(json['site_text']['value']).to_not eq(ru_mf_text)
end end
context 'when updating a translation override for a system badge' do
fab!(:user_with_badge_title) { Fabricate(:active_user) }
let(:badge) { Badge.find(Badge::Regular) }
before do
BadgeGranter.grant(badge, user_with_badge_title)
user_with_badge_title.update(title: 'Regular')
end
it 'updates matching user titles to the override text in a job' do
Jobs.expects(:enqueue).with(
:bulk_user_title_update,
new_title: 'Terminator',
granted_badge_id: badge.id,
action: Jobs::BulkUserTitleUpdate::UPDATE_ACTION
)
put '/admin/customize/site_texts/badges.regular.name.json', params: {
site_text: { value: 'Terminator' }
}
Jobs.expects(:enqueue).with(
:bulk_user_title_update,
granted_badge_id: badge.id,
action: Jobs::BulkUserTitleUpdate::RESET_ACTION
)
# Revert
delete "/admin/customize/site_texts/badges.regular.name.json"
end
end
end end
context "reseeding" do context "reseeding" do

View File

@ -1911,11 +1911,17 @@ describe UsersController do
expect(user.reload.title).to eq(badge.display_name) expect(user.reload.title).to eq(badge.display_name)
expect(user.user_profile.badge_granted_title).to eq(true) expect(user.user_profile.badge_granted_title).to eq(true)
expect(user.user_profile.granted_title_badge_id).to eq(badge.id)
user.title = "testing" badge.update allow_title: false
user.save
put "/u/#{user.username}/preferences/badge_title.json", params: { user_badge_id: user_badge.id }
user.reload
user.user_profile.reload user.user_profile.reload
expect(user.title).to eq('')
expect(user.user_profile.badge_granted_title).to eq(false) expect(user.user_profile.badge_granted_title).to eq(false)
expect(user.user_profile.granted_title_badge_id).to eq(nil)
end end
context "with overrided name" do context "with overrided name" do

View File

@ -196,6 +196,33 @@ describe BadgeGranter do
expect(user.reload.title).to eq(nil) expect(user.reload.title).to eq(nil)
end end
context 'when the badge name is customized, and the customized name is the same as the user title' do
let(:customized_badge_name) { 'Merit Badge' }
before do
TranslationOverride.upsert!(I18n.locale, Badge.i18n_key(badge.name), customized_badge_name)
end
it 'revokes the badge and title and does necessary cleanup' do
user.title = customized_badge_name; user.save!
expect(badge.reload.grant_count).to eq(1)
StaffActionLogger.any_instance.expects(:log_badge_revoke).with(user_badge)
StaffActionLogger.any_instance.expects(:log_title_revoke).with(
user,
revoke_reason: 'user title was same as revoked badge name or custom badge name',
previous_value: user_badge.user.title
)
BadgeGranter.revoke(user_badge, revoked_by: admin)
expect(UserBadge.find_by(user: user, badge: badge)).not_to be_present
expect(badge.reload.grant_count).to eq(0)
expect(user.notifications.where(notification_type: Notification.types[:granted_badge])).to be_empty
expect(user.reload.title).to eq(nil)
end
after do
TranslationOverride.revert!(I18n.locale, Badge.i18n_key(badge.name))
end
end
end end
context "update_badges" do context "update_badges" do