FIX: Serialize uploaded_avatars_allowed_groups check on current user (#25515)

Checking group permissions on the client does not work,
since not all groups are serialized to the client all
the time. We can check `uploaded_avatars_allowed_groups`
on the server side and serialize to the current user
instead.
This commit is contained in:
Martin Brennan 2024-02-02 09:32:45 +10:00 committed by GitHub
parent 7c8a56e116
commit 9563d02054
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
10 changed files with 32 additions and 19 deletions

View File

@ -17,11 +17,11 @@
</a> </a>
{{/each}} {{/each}}
</div> </div>
{{#if this.showAvatarUploader}} {{#if this.showCustomAvatarSelector}}
<h4>{{i18n "user.change_avatar.use_custom"}}</h4> <h4>{{i18n "user.change_avatar.use_custom"}}</h4>
{{/if}} {{/if}}
{{/if}} {{/if}}
{{#if this.showAvatarUploader}} {{#if this.showCustomAvatarSelector}}
{{#if this.user.use_logo_small_as_avatar}} {{#if this.user.use_logo_small_as_avatar}}
<div class="avatar-choice"> <div class="avatar-choice">
<RadioButton <RadioButton
@ -128,7 +128,7 @@
</:body> </:body>
<:footer> <:footer>
{{#if this.showAvatarUploader}} {{#if this.showCustomAvatarSelector}}
<DButton <DButton
@action={{this.saveAvatarSelection}} @action={{this.saveAvatarSelection}}
@disabled={{this.submitDisabled}} @disabled={{this.submitDisabled}}

View File

@ -41,7 +41,7 @@ export default class AvatarSelectorModal extends Component {
return this.siteSettings.selectable_avatars_mode !== "disabled"; return this.siteSettings.selectable_avatars_mode !== "disabled";
} }
get showAvatarUploader() { get showCustomAvatarSelector() {
const mode = this.siteSettings.selectable_avatars_mode; const mode = this.siteSettings.selectable_avatars_mode;
switch (mode) { switch (mode) {
case "no_one": case "no_one":
@ -92,18 +92,11 @@ export default class AvatarSelectorModal extends Component {
get allowAvatarUpload() { get allowAvatarUpload() {
return ( return (
this.siteSettingMatches && this.currentUser.can_upload_avatar &&
allowsImages(this.currentUser.staff, this.siteSettings) allowsImages(this.currentUser.staff, this.siteSettings)
); );
} }
get siteSettingMatches() {
return this.siteSettings.userInAnyGroups(
"uploaded_avatars_allowed_groups",
this.currentUser
);
}
@action @action
onSelectedChanged(value) { onSelectedChanged(value) {
this.selected = value; this.selected = value;

View File

@ -15,7 +15,7 @@ import { cloneJSON } from "discourse-common/lib/object";
import I18n from "discourse-i18n"; import I18n from "discourse-i18n";
acceptance("User Preferences - Account", function (needs) { acceptance("User Preferences - Account", function (needs) {
needs.user(); needs.user({ can_upload_avatar: true });
let customUserProps = {}; let customUserProps = {};
let pickAvatarRequestData = null; let pickAvatarRequestData = null;

View File

@ -19,6 +19,7 @@ class CurrentUserSerializer < BasicUserSerializer
:trust_level, :trust_level,
:can_send_private_email_messages, :can_send_private_email_messages,
:can_send_private_messages, :can_send_private_messages,
:can_upload_avatar,
:can_edit, :can_edit,
:can_invite_to_forum, :can_invite_to_forum,
:no_password, :no_password,
@ -120,6 +121,10 @@ class CurrentUserSerializer < BasicUserSerializer
scope.can_send_private_messages? scope.can_send_private_messages?
end end
def can_upload_avatar
!is_anonymous && object.in_any_groups?(SiteSetting.uploaded_avatars_allowed_groups_map)
end
def can_edit def can_edit
true true
end end

View File

@ -1573,7 +1573,6 @@ files:
enum: "TrustLevelAndStaffAndDisabledSetting" enum: "TrustLevelAndStaffAndDisabledSetting"
hidden: true hidden: true
uploaded_avatars_allowed_groups: uploaded_avatars_allowed_groups:
client: true
default: 10 default: 10
type: group_list type: group_list
allow_any: false allow_any: false

View File

@ -4,6 +4,7 @@ module PageObjects
class AvatarSelector < PageObjects::Modals::Base class AvatarSelector < PageObjects::Modals::Base
BODY_SELECTOR = ".avatar-selector" BODY_SELECTOR = ".avatar-selector"
MODAL_SELECTOR = ".avatar-selector-modal" MODAL_SELECTOR = ".avatar-selector-modal"
AVATAR_UPLOAD_BUTTON_SELECTOR = ".avatar-uploader__button"
def select_avatar_upload_option def select_avatar_upload_option
body.choose("avatar", option: "custom") body.choose("avatar", option: "custom")
@ -14,7 +15,15 @@ module PageObjects
end end
def click_avatar_upload_button def click_avatar_upload_button
body.find_button(I18n.t("js.user.change_avatar.upload_title")).click body.find(AVATAR_UPLOAD_BUTTON_SELECTOR).click
end
def has_avatar_upload_button?
has_css?(AVATAR_UPLOAD_BUTTON_SELECTOR)
end
def has_no_avatar_upload_button?
has_no_css?(AVATAR_UPLOAD_BUTTON_SELECTOR)
end end
def has_user_avatar_image_uploaded? def has_user_avatar_image_uploaded?

View File

@ -1,6 +1,6 @@
# frozen_string_literal: true # frozen_string_literal: true
describe "User preferences for Account", type: :system do describe "User preferences | Avatar", type: :system do
fab!(:user) { Fabricate(:user, refresh_auto_groups: true) } 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 }
@ -25,5 +25,12 @@ describe "User preferences for Account", type: :system do
expect(avatar_selector_modal).to be_closed expect(avatar_selector_modal).to be_closed
expect(user_account_preferences_page).to have_system_avatar_image expect(user_account_preferences_page).to have_system_avatar_image
end end
it "does not allow for custom pictures when the user is not in uploaded_avatars_allowed_groups" do
SiteSetting.uploaded_avatars_allowed_groups = Group::AUTO_GROUPS[:admins]
user_account_preferences_page.open_avatar_selector_modal(user)
expect(avatar_selector_modal).to be_open
expect(avatar_selector_modal).to have_no_avatar_upload_button
end
end end
end end

View File

@ -1,6 +1,6 @@
# frozen_string_literal: true # frozen_string_literal: true
describe "User preferences for Interface", type: :system do describe "User preferences | Interface", type: :system do
fab!(:user) fab!(:user)
let(:user_preferences_page) { PageObjects::Pages::UserPreferences.new } let(:user_preferences_page) { PageObjects::Pages::UserPreferences.new }
let(:user_preferences_interface_page) { PageObjects::Pages::UserPreferencesInterface.new } let(:user_preferences_interface_page) { PageObjects::Pages::UserPreferencesInterface.new }

View File

@ -1,6 +1,6 @@
# frozen_string_literal: true # frozen_string_literal: true
describe "User page navigation menu", type: :system do describe "User preferences | Navigation menu", type: :system do
fab!(:user) fab!(:user)
let(:everyone_group) { Group[:everyone] } let(:everyone_group) { Group[:everyone] }
let(:user_preferences_page) { PageObjects::Pages::UserPreferences.new } let(:user_preferences_page) { PageObjects::Pages::UserPreferences.new }

View File

@ -1,6 +1,6 @@
# frozen_string_literal: true # frozen_string_literal: true
describe "User preferences for Security", type: :system do describe "User preferences | Security", type: :system do
fab!(:password) { "kungfukenny" } fab!(:password) { "kungfukenny" }
fab!(:email) { "email@user.com" } fab!(:email) { "email@user.com" }
fab!(:user) { Fabricate(:user, email: email, password: password) } fab!(:user) { Fabricate(:user, email: email, password: password) }