FIX: Better error message when username change fails.

https://meta.discourse.org/t/500-error-on-username-edit/64064
This commit is contained in:
Guo Xiang Tan 2017-06-07 10:45:27 +09:00
parent d4d0aa8ca7
commit 2cad739262
5 changed files with 35 additions and 34 deletions

View File

@ -1,11 +1,12 @@
import { default as computed, observes } from 'ember-addons/ember-computed-decorators';
import { setting, propertyEqual } from 'discourse/lib/computed'; import { setting, propertyEqual } from 'discourse/lib/computed';
import DiscourseURL from 'discourse/lib/url'; import DiscourseURL from 'discourse/lib/url';
import { userPath } from 'discourse/lib/url'; import { userPath } from 'discourse/lib/url';
import { popupAjaxError } from 'discourse/lib/ajax-error';
export default Ember.Controller.extend({ export default Ember.Controller.extend({
taken: false, taken: false,
saving: false, saving: false,
error: false,
errorMessage: null, errorMessage: null,
newUsername: null, newUsername: null,
@ -15,29 +16,34 @@ export default Ember.Controller.extend({
saveDisabled: Em.computed.or('saving', 'newUsernameEmpty', 'taken', 'unchanged', 'errorMessage'), saveDisabled: Em.computed.or('saving', 'newUsernameEmpty', 'taken', 'unchanged', 'errorMessage'),
unchanged: propertyEqual('newUsername', 'username'), unchanged: propertyEqual('newUsername', 'username'),
checkTaken: function() { @observes("newUsername")
if( this.get('newUsername') && this.get('newUsername').length < this.get('minLength') ) { checkTaken() {
let newUsername = this.get('newUsername');
if (newUsername && newUsername.length < this.get('minLength')) {
this.set('errorMessage', I18n.t('user.name.too_short')); this.set('errorMessage', I18n.t('user.name.too_short'));
} else { } else {
var self = this;
this.set('taken', false); this.set('taken', false);
this.set('errorMessage', null); this.set('errorMessage', null);
if (Ember.isEmpty(this.get('newUsername'))) return; if (Ember.isEmpty(this.get('newUsername'))) return;
if (this.get('unchanged')) return; if (this.get('unchanged')) return;
Discourse.User.checkUsername(this.get('newUsername'), undefined, this.get('content.id')).then(function(result) {
Discourse.User.checkUsername(newUsername, undefined, this.get('content.id')).then(result => {
if (result.errors) { if (result.errors) {
self.set('errorMessage', result.errors.join(' ')); this.set('errorMessage', result.errors.join(' '));
} else if (result.available === false) { } else if (result.available === false) {
self.set('taken', true); this.set('taken', true);
} }
}); });
} }
}.observes('newUsername'), },
saveButtonText: function() { @computed('saving')
if (this.get('saving')) return I18n.t("saving"); saveButtonText(saving) {
if (saving) return I18n.t("saving");
return I18n.t("user.change"); return I18n.t("user.change");
}.property('saving'), },
actions: { actions: {
changeUsername() { changeUsername() {
@ -51,7 +57,7 @@ export default Ember.Controller.extend({
this.get('content').changeUsername(this.get('newUsername')).then(() => { this.get('content').changeUsername(this.get('newUsername')).then(() => {
DiscourseURL.redirectTo(userPath(this.get('newUsername').toLowerCase() + "/preferences")); DiscourseURL.redirectTo(userPath(this.get('newUsername').toLowerCase() + "/preferences"));
}) })
.catch(() => this.set('error', true)) .catch(popupAjaxError)
.finally(() => this.set('saving', false)); .finally(() => this.set('saving', false));
} }
}); });
@ -59,5 +65,3 @@ export default Ember.Controller.extend({
} }
}); });

View File

@ -7,14 +7,6 @@
</div> </div>
</div> </div>
{{#if error}}
<div class="control-group">
<div class="instructions">
<div class='alert alert-error'>{{i18n 'user.change_username.error'}}</div>
</div>
</div>
{{/if}}
<div class="control-group"> <div class="control-group">
<label class="control-label">{{i18n 'user.username.title'}}</label> <label class="control-label">{{i18n 'user.username.title'}}</label>
<div class="controls"> <div class="controls">

View File

@ -127,14 +127,13 @@ class UsersController < ApplicationController
user = fetch_user_from_params user = fetch_user_from_params
guardian.ensure_can_edit_username!(user) guardian.ensure_can_edit_username!(user)
# TODO proper error surfacing (result is a Model#save call)
result = UsernameChanger.change(user, params[:new_username], current_user) result = UsernameChanger.change(user, params[:new_username], current_user)
raise Discourse::InvalidParameters.new(:new_username) unless result
render json: { if result
id: user.id, render json: { id: user.id, username: user.username }
username: user.username else
} render_json_error(user.errors.full_messages.join(','))
end
end end
def check_emails def check_emails

View File

@ -689,7 +689,6 @@ en:
title: "Change Username" title: "Change Username"
confirm: "If you change your username, all prior quotes of your posts and @name mentions will be broken. Are you absolutely sure you want to?" confirm: "If you change your username, all prior quotes of your posts and @name mentions will be broken. Are you absolutely sure you want to?"
taken: "Sorry, that username is taken." taken: "Sorry, that username is taken."
error: "There was an error changing your username."
invalid: "That username is invalid. It must only include numbers and letters" invalid: "That username is invalid. It must only include numbers and letters"
change_email: change_email:

View File

@ -867,7 +867,7 @@ describe UsersController do
end end
context '.username' do context '#username' do
it 'raises an error when not logged in' do it 'raises an error when not logged in' do
expect { xhr :put, :username, username: 'somename' }.to raise_error(Discourse::NotLoggedIn) expect { xhr :put, :username, username: 'somename' }.to raise_error(Discourse::NotLoggedIn)
end end
@ -894,10 +894,17 @@ describe UsersController do
expect(user.reload.username).to eq(old_username) expect(user.reload.username).to eq(old_username)
end end
# Bad behavior, this should give a real JSON error, not an InvalidParameters
it 'raises an error when change_username fails' do it 'raises an error when change_username fails' do
User.any_instance.expects(:save).returns(false) xhr :put, :username, username: user.username, new_username: '@'
expect { xhr :put, :username, username: user.username, new_username: new_username }.to raise_error(Discourse::InvalidParameters)
expect(response).to_not be_success
body = JSON.parse(response.body)
expect(body['errors'].first).to include(I18n.t(
'user.username.short', min: User.username_length.begin
))
expect(user.reload.username).to eq(old_username) expect(user.reload.username).to eq(old_username)
end end