DEV: Convert allow_uploaded_avatars to groups (#24810)

This change converts the allow_uploaded_avatars site setting to uploaded_avatars_allowed_groups.

See: https://meta.discourse.org/t/283408

Hides the old setting
Adds the new site setting
Adds a deprecation warning
Updates to use the new setting
Adds a migration to fill in the new setting if the old setting was changed
Adds an entry to the site_setting.keywords section
Updates tests to account for the new change
After a couple of months, we will remove the allow_uploaded_avatars setting entirely.

Internal ref: /t/117248
This commit is contained in:
Krzysztof Kotlarek 2023-12-13 10:53:19 +11:00 committed by GitHub
parent 8355664c82
commit 1017820012
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
13 changed files with 85 additions and 43 deletions

View File

@ -98,21 +98,11 @@ export default class AvatarSelectorModal extends Component {
} }
get siteSettingMatches() { get siteSettingMatches() {
const allowUploadedAvatars = this.siteSettings.allow_uploaded_avatars; return this.siteSettings.userInAnyGroups(
switch (allowUploadedAvatars) { "uploaded_avatars_allowed_groups",
case "disabled": this.currentUser
return false;
case "staff":
return this.currentUser.staff;
case "admin":
return this.currentUser.admin;
default:
return (
this.currentUser.trust_level >= parseInt(allowUploadedAvatars, 10) ||
this.currentUser.staff
); );
} }
}
@action @action
onSelectedChanged(value) { onSelectedChanged(value) {

View File

@ -30,10 +30,10 @@ class UploadsController < ApplicationController
type = type =
(params[:upload_type].presence || params[:type].presence).parameterize(separator: "_")[0..50] (params[:upload_type].presence || params[:type].presence).parameterize(separator: "_")[0..50]
if type == "avatar" && !me.admin? && if type == "avatar" &&
( (
SiteSetting.discourse_connect_overrides_avatar || SiteSetting.discourse_connect_overrides_avatar ||
!TrustLevelAndStaffAndDisabledSetting.matches?(SiteSetting.allow_uploaded_avatars, me) !me.in_any_groups?(SiteSetting.uploaded_avatars_allowed_groups_map)
) )
return render json: failed_json, status: 422 return render json: failed_json, status: 422
end end

View File

@ -1294,7 +1294,8 @@ class UsersController < ApplicationController
if type.blank? || type == "system" if type.blank? || type == "system"
upload_id = nil upload_id = nil
elsif !TrustLevelAndStaffAndDisabledSetting.matches?(SiteSetting.allow_uploaded_avatars, user) elsif !user.in_any_groups?(SiteSetting.uploaded_avatars_allowed_groups_map) &&
!user.is_system_user?
return render json: failed_json, status: 422 return render json: failed_json, status: 422
else else
upload_id = params[:upload_id] upload_id = params[:upload_id]

View File

@ -2172,6 +2172,7 @@ en:
logout_redirect: "Location to redirect browser to after logout (eg: https://example.com/logout)" logout_redirect: "Location to redirect browser to after logout (eg: https://example.com/logout)"
allow_uploaded_avatars: "Allow users to upload custom profile pictures." allow_uploaded_avatars: "Allow users to upload custom profile pictures."
uploaded_avatars_allowed_groups: "Specify groups allowed to upload custom profile pictures."
default_avatars: "URLs to avatars that will be used by default for new users until they change them." default_avatars: "URLs to avatars that will be used by default for new users until they change them."
automatically_download_gravatars: "Download Gravatars for users upon account creation or email change." automatically_download_gravatars: "Download Gravatars for users upon account creation or email change."
digest_topics: "The maximum number of popular topics to display in the email summary." digest_topics: "The maximum number of popular topics to display in the email summary."
@ -2553,6 +2554,7 @@ en:
approve_new_topics_unless_allowed_groups: "approve_new_topics_unless_trust_level" approve_new_topics_unless_allowed_groups: "approve_new_topics_unless_trust_level"
email_in_allowed_groups: "email_in_min_trust" email_in_allowed_groups: "email_in_min_trust"
edit_wiki_post_allowed_groups: "minmin_trust_to_edit_wiki_post" edit_wiki_post_allowed_groups: "minmin_trust_to_edit_wiki_post"
uploaded_avatars_allowed_groups: "allow_uploaded_avatars"
placeholder: placeholder:
discourse_connect_provider_secrets: discourse_connect_provider_secrets:

View File

@ -1550,6 +1550,13 @@ files:
client: true client: true
default: "0" default: "0"
enum: "TrustLevelAndStaffAndDisabledSetting" enum: "TrustLevelAndStaffAndDisabledSetting"
hidden: true
uploaded_avatars_allowed_groups:
client: true
default: 10
type: group_list
allow_any: false
refresh: true
default_avatars: default_avatars:
default: "" default: ""
type: url_list type: url_list

View File

@ -0,0 +1,34 @@
# frozen_string_literal: true
class FillUploadedAvatarsAllowedGroupsBasedOnDeprecatedSettings < ActiveRecord::Migration[7.0]
def up
old_setting_trust_level =
DB.query_single(
"SELECT value FROM site_settings WHERE name = 'allow_uploaded_avatars' LIMIT 1",
).first
if old_setting_trust_level.present?
group_id =
case old_setting_trust_level
when "disabled"
""
when "admin"
"1"
when "staff"
"3"
else
"1#{old_setting_trust_level}"
end
DB.exec(
"INSERT INTO site_settings(name, value, data_type, created_at, updated_at)
VALUES('uploaded_avatars_allowed_groups', :setting, '20', NOW(), NOW())",
setting: group_id,
)
end
end
def down
raise ActiveRecord::IrreversibleMigration
end
end

View File

@ -90,7 +90,7 @@ class ComposerMessagesFinder
# - "allow uploaded avatars" is disabled # - "allow uploaded avatars" is disabled
if SiteSetting.disable_avatar_education_message || if SiteSetting.disable_avatar_education_message ||
SiteSetting.discourse_connect_overrides_avatar || SiteSetting.discourse_connect_overrides_avatar ||
!TrustLevelAndStaffAndDisabledSetting.matches?(SiteSetting.allow_uploaded_avatars, @user) !@user.in_any_groups?(SiteSetting.uploaded_avatars_allowed_groups_map)
return return
end end

View File

@ -20,6 +20,7 @@ module SiteSettings::DeprecatedSettings
], ],
["email_in_min_trust", "email_in_allowed_groups", false, "3.3"], ["email_in_min_trust", "email_in_allowed_groups", false, "3.3"],
["min_trust_to_edit_wiki_post", "edit_wiki_post_allowed_groups", false, "3.3"], ["min_trust_to_edit_wiki_post", "edit_wiki_post_allowed_groups", false, "3.3"],
["allow_uploaded_avatars", "uploaded_avatars_allowed_groups", false, "3.3"],
] ]
def setup_deprecated_methods def setup_deprecated_methods

View File

@ -100,7 +100,7 @@ RSpec.describe ComposerMessagesFinder do
describe ".check_avatar_notification" do describe ".check_avatar_notification" do
let(:finder) { ComposerMessagesFinder.new(user, composer_action: "createTopic") } let(:finder) { ComposerMessagesFinder.new(user, composer_action: "createTopic") }
fab!(:user) fab!(:user) { Fabricate(:user, refresh_auto_groups: true) }
context "with success" do context "with success" do
let!(:message) { finder.check_avatar_notification } let!(:message) { finder.check_avatar_notification }
@ -143,8 +143,15 @@ RSpec.describe ComposerMessagesFinder do
end end
it "doesn't notify users if 'allow_uploaded_avatars' setting is disabled" do it "doesn't notify users if 'allow_uploaded_avatars' setting is disabled" do
SiteSetting.allow_uploaded_avatars = "disabled" user.update!(trust_level: 3)
Group.refresh_automatic_groups!
user.reload
SiteSetting.uploaded_avatars_allowed_groups = ""
expect(finder.check_avatar_notification).to be_blank expect(finder.check_avatar_notification).to be_blank
SiteSetting.uploaded_avatars_allowed_groups = "13"
expect(finder.check_avatar_notification).to be_present
end end
end end

View File

@ -185,7 +185,7 @@ RSpec.describe "users" do
response "200", "avatar updated" do response "200", "avatar updated" do
expected_response_schema = load_spec_schema("success_ok_response") expected_response_schema = load_spec_schema("success_ok_response")
let(:user) { Fabricate(:user) } let(:user) { Fabricate(:user, refresh_auto_groups: true) }
let(:username) { user.username } let(:username) { user.username }
let(:upload) { Fabricate(:upload, user: user) } let(:upload) { Fabricate(:upload, user: user) }
let(:params) { { "upload_id" => upload.id, "type" => "uploaded" } } let(:params) { { "upload_id" => upload.id, "type" => "uploaded" } }

View File

@ -1,7 +1,7 @@
# frozen_string_literal: true # frozen_string_literal: true
RSpec.describe UploadsController do RSpec.describe UploadsController do
fab!(:user) fab!(:user) { Fabricate(:user, refresh_auto_groups: true) }
describe "#create" do describe "#create" do
it "requires you to be logged in" do it "requires you to be logged in" do
@ -132,10 +132,15 @@ RSpec.describe UploadsController do
) )
end end
it "ensures allow_uploaded_avatars is enabled when uploading an avatar" do it "ensures user belongs to uploaded_avatars_allowed_groups when uploading an avatar" do
SiteSetting.allow_uploaded_avatars = "disabled" SiteSetting.uploaded_avatars_allowed_groups = "13"
post "/uploads.json", params: { file: logo, type: "avatar" } post "/uploads.json", params: { file: logo, type: "avatar" }
expect(response.status).to eq(422) expect(response.status).to eq(422)
user.update!(trust_level: 3)
Group.refresh_automatic_groups!
post "/uploads.json", params: { file: logo, type: "avatar" }
expect(response.status).to eq(200)
end end
it "ensures discourse_connect_overrides_avatar is not enabled when uploading an avatar" do it "ensures discourse_connect_overrides_avatar is not enabled when uploading an avatar" do
@ -144,14 +149,6 @@ RSpec.describe UploadsController do
expect(response.status).to eq(422) expect(response.status).to eq(422)
end end
it "always allows admins to upload avatars" do
sign_in(Fabricate(:admin))
SiteSetting.allow_uploaded_avatars = "disabled"
post "/uploads.json", params: { file: logo, type: "avatar" }
expect(response.status).to eq(200)
end
it "allows staff to upload any file in PM" do it "allows staff to upload any file in PM" do
SiteSetting.authorized_extensions = "jpg" SiteSetting.authorized_extensions = "jpg"
SiteSetting.allow_staff_to_upload_any_file_in_pm = true SiteSetting.allow_staff_to_upload_any_file_in_pm = true

View File

@ -3342,8 +3342,8 @@ RSpec.describe UsersController do
expect(response.status).to eq(422) expect(response.status).to eq(422)
end end
it "raises an error when selecting the custom/uploaded avatar and allow_uploaded_avatars is disabled" do it "raises an error when selecting the custom/uploaded avatar and uploaded_avatars_allowed_groups is disabled" do
SiteSetting.allow_uploaded_avatars = "disabled" SiteSetting.uploaded_avatars_allowed_groups = ""
put "/u/#{user1.username}/preferences/avatar/pick.json", put "/u/#{user1.username}/preferences/avatar/pick.json",
params: { params: {
upload_id: upload.id, upload_id: upload.id,
@ -3353,8 +3353,8 @@ RSpec.describe UsersController do
expect(response.status).to eq(422) expect(response.status).to eq(422)
end end
it "raises an error when selecting the custom/uploaded avatar and allow_uploaded_avatars is admin" do it "raises an error when selecting the custom/uploaded avatar and uploaded_avatars_allowed_groups is admin" do
SiteSetting.allow_uploaded_avatars = "admin" SiteSetting.uploaded_avatars_allowed_groups = "1"
put "/u/#{user1.username}/preferences/avatar/pick.json", put "/u/#{user1.username}/preferences/avatar/pick.json",
params: { params: {
upload_id: upload.id, upload_id: upload.id,
@ -3363,6 +3363,7 @@ RSpec.describe UsersController do
expect(response.status).to eq(422) expect(response.status).to eq(422)
user1.update!(admin: true) user1.update!(admin: true)
Group.refresh_automatic_groups!
put "/u/#{user1.username}/preferences/avatar/pick.json", put "/u/#{user1.username}/preferences/avatar/pick.json",
params: { params: {
upload_id: upload.id, upload_id: upload.id,
@ -3371,8 +3372,8 @@ RSpec.describe UsersController do
expect(response.status).to eq(200) expect(response.status).to eq(200)
end end
it "raises an error when selecting the custom/uploaded avatar and allow_uploaded_avatars is staff" do it "raises an error when selecting the custom/uploaded avatar and uploaded_avatars_allowed_groups is staff" do
SiteSetting.allow_uploaded_avatars = "staff" SiteSetting.uploaded_avatars_allowed_groups = "3"
put "/u/#{user1.username}/preferences/avatar/pick.json", put "/u/#{user1.username}/preferences/avatar/pick.json",
params: { params: {
upload_id: upload.id, upload_id: upload.id,
@ -3381,6 +3382,7 @@ RSpec.describe UsersController do
expect(response.status).to eq(422) expect(response.status).to eq(422)
user1.update!(moderator: true) user1.update!(moderator: true)
Group.refresh_automatic_groups!
put "/u/#{user1.username}/preferences/avatar/pick.json", put "/u/#{user1.username}/preferences/avatar/pick.json",
params: { params: {
upload_id: upload.id, upload_id: upload.id,
@ -3389,8 +3391,8 @@ RSpec.describe UsersController do
expect(response.status).to eq(200) expect(response.status).to eq(200)
end end
it "raises an error when selecting the custom/uploaded avatar and allow_uploaded_avatars is a trust level" do it "raises an error when selecting the custom/uploaded avatar and uploaded_avatars_allowed_groups is a trust level" do
SiteSetting.allow_uploaded_avatars = "3" SiteSetting.uploaded_avatars_allowed_groups = "13"
put "/u/#{user1.username}/preferences/avatar/pick.json", put "/u/#{user1.username}/preferences/avatar/pick.json",
params: { params: {
upload_id: upload.id, upload_id: upload.id,
@ -3399,6 +3401,7 @@ RSpec.describe UsersController do
expect(response.status).to eq(422) expect(response.status).to eq(422)
user1.update!(trust_level: 3) user1.update!(trust_level: 3)
Group.refresh_automatic_groups!
put "/u/#{user1.username}/preferences/avatar/pick.json", put "/u/#{user1.username}/preferences/avatar/pick.json",
params: { params: {
upload_id: upload.id, upload_id: upload.id,
@ -3408,7 +3411,7 @@ RSpec.describe UsersController do
end end
it "ignores the upload if picking a system avatar" do it "ignores the upload if picking a system avatar" do
SiteSetting.allow_uploaded_avatars = "disabled" SiteSetting.uploaded_avatars_allowed_groups = ""
another_upload = Fabricate(:upload) another_upload = Fabricate(:upload)
put "/u/#{user1.username}/preferences/avatar/pick.json", put "/u/#{user1.username}/preferences/avatar/pick.json",
@ -3422,7 +3425,7 @@ RSpec.describe UsersController do
end end
it "raises an error if the type is invalid" do it "raises an error if the type is invalid" do
SiteSetting.allow_uploaded_avatars = "disabled" SiteSetting.uploaded_avatars_allowed_groups = ""
another_upload = Fabricate(:upload) another_upload = Fabricate(:upload)
put "/u/#{user1.username}/preferences/avatar/pick.json", put "/u/#{user1.username}/preferences/avatar/pick.json",

View File

@ -1,7 +1,7 @@
# frozen_string_literal: true # frozen_string_literal: true
describe "User preferences for Account", type: :system do describe "User preferences for Account", type: :system do
fab!(:user) fab!(:user) { Fabricate(:user, refresh_auto_groups: true) }
let(:user_account_preferences_page) { PageObjects::Pages::UserPreferencesAccount.new } let(:user_account_preferences_page) { PageObjects::Pages::UserPreferencesAccount.new }
let(:avatar_selector_modal) { PageObjects::Modals::AvatarSelector.new } let(:avatar_selector_modal) { PageObjects::Modals::AvatarSelector.new }
before { sign_in(user) } before { sign_in(user) }