FIX: Don't store translated trust level names in anonymous cache (#13224)

Refactors `TrustLevel` and moves translations from server to client

Additional changes:
  * "staff" and "admin" wasn't translatable in site settings
  * it replaces a concatenated string with a translation
  * uses translation for trust levels in users_by_trust_level report
  * adds a DB migration to rename keys of translation overrides affected by this commit
This commit is contained in:
Gerhard Schlager 2021-06-01 22:11:48 +02:00 committed by GitHub
parent 409c8585e4
commit 41ee5b7c86
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
17 changed files with 144 additions and 88 deletions

View File

@ -179,8 +179,10 @@ Site.reopenClass(Singleton, {
} }
if (result.trust_levels) { if (result.trust_levels) {
result.trustLevels = result.trust_levels.map((tl) => result.trustLevels = Object.entries(result.trust_levels).map(
TrustLevel.create(tl) ([key, id]) => {
return new TrustLevel(id, key);
}
); );
delete result.trust_levels; delete result.trust_levels;
} }

View File

@ -1,6 +1,22 @@
import RestModel from "discourse/models/rest"; import { computed } from "@ember/object";
import { fmt } from "discourse/lib/computed"; import I18n from "I18n";
export default RestModel.extend({ export default class TrustLevel {
detailedName: fmt("id", "name", "%@ - %@"), constructor(id, key) {
}); this.id = id;
this._key = key;
}
@computed
get name() {
return I18n.t(`trust_levels.names.${this._key}`);
}
@computed
get detailedName() {
return I18n.t("trust_levels.detailed_name", {
level: this.id,
name: this.name,
});
}
}

View File

@ -12,6 +12,13 @@ export default {
small_action: 3, small_action: 3,
whisper: 4, whisper: 4,
}, },
trust_levels: {
newuser: 0,
basic: 1,
member: 2,
regular: 3,
leader: 4,
},
groups: [ groups: [
{ id: 0, name: "everyone" }, { id: 0, name: "everyone" },
{ id: 1, name: "admins" }, { id: 1, name: "admins" },
@ -535,28 +542,6 @@ export default {
is_custom_flag: true, is_custom_flag: true,
}, },
], ],
trust_levels: [
{
id: 0,
name: "new user",
},
{
id: 1,
name: "basic user",
},
{
id: 2,
name: "member",
},
{
id: 3,
name: "regular",
},
{
id: 4,
name: "leader",
},
],
archetypes: [ archetypes: [
{ {
id: "regular", id: "regular",

View File

@ -16,6 +16,13 @@ PreloadStore.store("site", {
moved_post: 10, moved_post: 10,
}, },
post_types: { regular: 1, moderator_action: 2 }, post_types: { regular: 1, moderator_action: 2 },
trust_levels: {
newuser: 0,
basic: 1,
member: 2,
regular: 3,
leader: 4,
},
groups: [ groups: [
{ id: 0, name: "everyone" }, { id: 0, name: "everyone" },
{ id: 1, name: "admins" }, { id: 1, name: "admins" },
@ -349,12 +356,5 @@ PreloadStore.store("site", {
is_custom_flag: true, is_custom_flag: true,
}, },
], ],
trust_levels: [
{ id: 0, name: "new user" },
{ id: 1, name: "basic user" },
{ id: 2, name: "member" },
{ id: 3, name: "regular" },
{ id: 4, name: "leader" },
],
archetypes: [{ id: "regular", name: "Regular Topic", options: [] }], archetypes: [{ id: "regular", name: "Regular Topic", options: [] }],
}); });

View File

@ -24,7 +24,7 @@ module Reports::UsersByTrustLevel
] ]
User.real.group('trust_level').count.sort.each do |level, count| User.real.group('trust_level').count.sort.each do |level, count|
key = TrustLevel.levels[level.to_i] key = TrustLevel.name(level.to_i)
url = Proc.new { |k| "/admin/users/list/#{k}" } url = Proc.new { |k| "/admin/users/list/#{k}" }
report.data << { url: url.call(key), key: key, x: level.to_i, y: count } report.data << { url: url.call(key), key: key, x: level.to_i, y: count }
end end

View File

@ -21,7 +21,7 @@ class Site
end end
def trust_levels def trust_levels
TrustLevel.all TrustLevel.levels
end end
def user_fields def user_fields

View File

@ -18,4 +18,12 @@ class TrustLevelAndStaffSetting < TrustLevelSetting
def self.special_groups def self.special_groups
['staff', 'admin'] ['staff', 'admin']
end end
def self.translation(value)
if special_group?(value)
I18n.t("trust_levels.#{value}")
else
super
end
end
end end

View File

@ -8,18 +8,23 @@ class TrustLevelSetting < EnumSiteSetting
end end
def self.values def self.values
levels = TrustLevel.all valid_values.map do |value|
valid_values.map { |x| { name: translation(value), value: value }
{ end
name: x.is_a?(Integer) ? "#{x}: #{levels[x.to_i].name}" : x,
value: x
}
}
end end
def self.valid_values def self.valid_values
TrustLevel.valid_range.to_a TrustLevel.valid_range.to_a
end end
def self.translation(value)
I18n.t(
"js.trust_levels.detailed_name",
level: value,
name: TrustLevel.name(value)
)
end
private_class_method :valid_values private_class_method :valid_values
private_class_method :translation
end end

View File

@ -6,6 +6,7 @@ class SiteSerializer < ApplicationSerializer
:default_archetype, :default_archetype,
:notification_types, :notification_types,
:post_types, :post_types,
:trust_levels,
:groups, :groups,
:filters, :filters,
:periods, :periods,
@ -32,7 +33,6 @@ class SiteSerializer < ApplicationSerializer
) )
has_many :categories, serializer: SiteCategorySerializer, embed: :objects has_many :categories, serializer: SiteCategorySerializer, embed: :objects
has_many :trust_levels, embed: :objects
has_many :archetypes, embed: :objects, serializer: ArchetypeSerializer has_many :archetypes, embed: :objects, serializer: ArchetypeSerializer
has_many :user_fields, embed: :objects, serializer: UserFieldSerializer has_many :user_fields, embed: :objects, serializer: UserFieldSerializer
has_many :auth_providers, embed: :objects, serializer: AuthProviderSerializer has_many :auth_providers, embed: :objects, serializer: AuthProviderSerializer

View File

@ -1,7 +0,0 @@
# frozen_string_literal: true
class TrustLevelSerializer < ApplicationSerializer
attributes :id, :name
end

View File

@ -702,7 +702,7 @@ en:
smtp_instructions: "When you enable SMTP for the group, all outbound emails sent from the group's inbox will be sent via the SMTP settings specified here instead of the mail server configured for other emails sent by your forum." smtp_instructions: "When you enable SMTP for the group, all outbound emails sent from the group's inbox will be sent via the SMTP settings specified here instead of the mail server configured for other emails sent by your forum."
imap_title: "IMAP" imap_title: "IMAP"
imap_additional_settings: "Additional Settings" imap_additional_settings: "Additional Settings"
imap_instructions: "When you enable IMAP for the group, emails are synced between the group inbox and the provided IMAP server and mailbox. SMTP must be enabled with valid and tested credentials before IMAP can be enabled. The email username and password used for SMTP will be used for IMAP. For more information see <a target=\"_blank\" href=\"https://meta.discourse.org/t/imap-support-for-group-inboxes/160588\">feature announcement on Discourse Meta</a>." imap_instructions: 'When you enable IMAP for the group, emails are synced between the group inbox and the provided IMAP server and mailbox. SMTP must be enabled with valid and tested credentials before IMAP can be enabled. The email username and password used for SMTP will be used for IMAP. For more information see <a target="_blank" href="https://meta.discourse.org/t/imap-support-for-group-inboxes/160588">feature announcement on Discourse Meta</a>.'
imap_alpha_warning: "Warning: This is an alpha-stage feature. Only Gmail is officially supported. Use at your own risk!" imap_alpha_warning: "Warning: This is an alpha-stage feature. Only Gmail is officially supported. Use at your own risk!"
imap_settings_valid: "IMAP settings valid." imap_settings_valid: "IMAP settings valid."
smtp_disable_confirm: "If you disable SMTP, all SMTP and IMAP settings will be reset and the associated functionality will be disabled. Are you sure you want to continue?" smtp_disable_confirm: "If you disable SMTP, all SMTP and IMAP settings will be reset and the associated functionality will be disabled. Are you sure you want to continue?"
@ -3764,6 +3764,15 @@ en:
custom: "Custom" custom: "Custom"
set_schedule: "Set a notification schedule" set_schedule: "Set a notification schedule"
trust_levels:
names:
newuser: "new user"
basic: "basic user"
member: "member"
regular: "regular"
leader: "leader"
detailed_name: "%{level}: %{name}"
# This section is exported to the javascript for i18n in the admin section # This section is exported to the javascript for i18n in the admin section
admin_js: admin_js:
type_to_filter: "type to filter..." type_to_filter: "type to filter..."

View File

@ -717,16 +717,8 @@ en:
topic_exists_no_oldest: "Can't delete this category because topic count is %{count}." topic_exists_no_oldest: "Can't delete this category because topic count is %{count}."
uncategorized_description: "Topics that don't need a category, or don't fit into any other existing category." uncategorized_description: "Topics that don't need a category, or don't fit into any other existing category."
trust_levels: trust_levels:
newuser: admin: "Admin"
title: "new user" staff: "Staff"
basic:
title: "basic user"
member:
title: "member"
regular:
title: "regular"
leader:
title: "leader"
change_failed_explanation: "You attempted to demote %{user_name} to '%{new_trust_level}'. However their trust level is already '%{current_trust_level}'. %{user_name} will remain at '%{current_trust_level}' - if you wish to demote user lock trust level first" change_failed_explanation: "You attempted to demote %{user_name} to '%{new_trust_level}'. However their trust level is already '%{current_trust_level}'. %{user_name} will remain at '%{current_trust_level}' - if you wish to demote user lock trust level first"
post: post:
@ -994,7 +986,7 @@ en:
imap_authentication_error: "There was an issue with the IMAP credentials provided, check the username and password and try again." imap_authentication_error: "There was an issue with the IMAP credentials provided, check the username and password and try again."
imap_no_response_error: "An error occurred when communicating with the IMAP server. %{message}" imap_no_response_error: "An error occurred when communicating with the IMAP server. %{message}"
smtp_authentication_error: "There was an issue with the SMTP credentials provided, check the username and password and try again." smtp_authentication_error: "There was an issue with the SMTP credentials provided, check the username and password and try again."
authentication_error_gmail_app_password: "Application-specific password required. Learn more at <a target=\"_blank\" href=\"https://support.google.com/accounts/answer/185833\">this Google Help article</a>" authentication_error_gmail_app_password: 'Application-specific password required. Learn more at <a target="_blank" href="https://support.google.com/accounts/answer/185833">this Google Help article</a>'
smtp_server_busy_error: "The SMTP server is currently busy, try again later." smtp_server_busy_error: "The SMTP server is currently busy, try again later."
smtp_unhandled_error: "There was an unhandled error when communicating with the SMTP server. %{message}" smtp_unhandled_error: "There was an unhandled error when communicating with the SMTP server. %{message}"
imap_unhandled_error: "There was an unhandled error when communicating with the IMAP server. %{message}" imap_unhandled_error: "There was an unhandled error when communicating with the IMAP server. %{message}"

View File

@ -0,0 +1,25 @@
# frozen_string_literal: true
class RenameTrustLevelTranslations < ActiveRecord::Migration[6.1]
KEYS = %w[newuser basic member regular leader]
def up
KEYS.each do |key|
execute <<~SQL
UPDATE translation_overrides
SET translation_key = 'js.trust_levels.names.#{key}'
WHERE translation_key = 'trust_levels.#{key}.title'
SQL
end
end
def down
KEYS.each do |key|
execute <<~SQL
UPDATE translation_overrides
SET translation_key = 'trust_levels.#{key}.title'
WHERE translation_key = 'js.trust_levels.names.#{key}'
SQL
end
end
end

View File

@ -4,8 +4,6 @@ class InvalidTrustLevel < StandardError; end
class TrustLevel class TrustLevel
attr_reader :id, :name
class << self class << self
def [](level) def [](level)
@ -17,12 +15,6 @@ class TrustLevel
@levels ||= Enum.new(:newuser, :basic, :member, :regular, :leader, start: 0) @levels ||= Enum.new(:newuser, :basic, :member, :regular, :leader, start: 0)
end end
def all
levels.map do |name_key, id|
TrustLevel.new(name_key, id)
end
end
def valid?(level) def valid?(level)
valid_range === level valid_range === level
end end
@ -35,15 +27,9 @@ class TrustLevel
(current_level || 0) >= level (current_level || 0) >= level
end end
end def name(level)
I18n.t("js.trust_levels.names.#{levels[level]}")
def initialize(name_key, id) end
@name = I18n.t("trust_levels.#{name_key}.title")
@id = id
end
def serializable_hash
{ id: @id, name: @name }
end end
end end

View File

@ -24,12 +24,6 @@ def is_yaml_compatible?(english, translated)
end end
describe "i18n integrity checks" do describe "i18n integrity checks" do
it 'has an i18n key for each Trust Levels' do
TrustLevel.all.each do |ts|
expect(ts.name).not_to match(/translation missing/)
end
end
it "has an i18n key for each Site Setting" do it "has an i18n key for each Site Setting" do
SiteSetting.all_settings.each do |s| SiteSetting.all_settings.each do |s|
next if s[:setting][/^test_/] next if s[:setting][/^test_/]

View File

@ -0,0 +1,26 @@
# frozen_string_literal: true
require 'rails_helper'
describe TrustLevelAndStaffSetting do
describe ".values" do
it "returns translated names" do
TranslationOverride.upsert!(I18n.locale, "js.trust_levels.names.newuser", "New Member")
TranslationOverride.upsert!(I18n.locale, "trust_levels.admin", "Hero")
values = TrustLevelAndStaffSetting.values
value = values.find { |v| v[:value] == 0 }
expect(value).to be_present
expect(value[:name]).to eq(I18n.t("js.trust_levels.detailed_name", level: 0, name: "New Member"))
value = values.find { |v| v[:value] == "admin" }
expect(value).to be_present
expect(value[:name]).to eq("Hero")
value = values.find { |v| v[:value] == "staff" }
expect(value).to be_present
expect(value[:name]).to eq(I18n.t("trust_levels.staff"))
end
end
end

View File

@ -0,0 +1,15 @@
# frozen_string_literal: true
require 'rails_helper'
describe TrustLevelSetting do
describe ".values" do
it "returns translated names" do
TranslationOverride.upsert!(I18n.locale, "js.trust_levels.names.newuser", "New Member")
value = TrustLevelSetting.values.first
expect(value[:name]).to eq(I18n.t("js.trust_levels.detailed_name", level: 0, name: "New Member"))
expect(value[:value]).to eq(0)
end
end
end