DEV: Replace User.unstage and User#unstage API with User#unstage! (#8906)

* DEV: Replace User.unstage and User#unstage API with User#unstage!

Quoting @SamSaffron:

> User.unstage mixes concerns of both unstaging users and updating params which is fragile/surprising.
> u.unstage destroys notifications and raises a user_unstaged event prior to the user becoming unstaged and the user object being saved.

User#unstage! no longer updates user attributes and saves the object before triggering the `user_unstaged` event.

* Update one more spec

* Assign attributes after unstaging
This commit is contained in:
Jarek Radosz 2020-03-17 16:48:24 +01:00 committed by GitHub
parent 43b38dbbc2
commit e950471c0f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
9 changed files with 41 additions and 62 deletions

View File

@ -113,8 +113,7 @@ class Users::OmniauthCallbacksController < ApplicationController
# automatically activate/unstage any account if a provider marked the email valid # automatically activate/unstage any account if a provider marked the email valid
if @auth_result.email_valid && @auth_result.email == user.email if @auth_result.email_valid && @auth_result.email == user.email
user.unstage user.unstage!
user.save
if !user.active || !user.email_confirmed? if !user.active || !user.email_confirmed?
user.update!(password: SecureRandom.hex) user.update!(password: SecureRandom.hex)

View File

@ -436,8 +436,16 @@ class UsersController < ApplicationController
params[:locale] ||= I18n.locale unless current_user params[:locale] ||= I18n.locale unless current_user
new_user_params = user_params.except(:timezone) new_user_params = user_params.except(:timezone)
user = User.unstage(new_user_params)
user = User.new(new_user_params) if user.nil? user = User.where(staged: true).with_email(new_user_params[:email].strip.downcase).first
if user
user.active = false
user.unstage!
end
user ||= User.new
user.attributes = new_user_params
# Handle API approval # Handle API approval
ReviewableUser.set_approved_fields!(user, current_user) if user.approved? ReviewableUser.set_approved_fields!(user, current_user) if user.approved?

View File

@ -74,8 +74,7 @@ class DiscourseSingleSignOn < SingleSignOn
end end
# ensure it's not staged anymore # ensure it's not staged anymore
user.unstage user.unstage!
user.save
change_external_attributes_and_override(sso_record, user) change_external_attributes_and_override(sso_record, user)

View File

@ -15,26 +15,27 @@ InviteRedeemer = Struct.new(:invite, :username, :name, :password, :user_custom_f
# extracted from User cause it is very specific to invites # extracted from User cause it is very specific to invites
def self.create_user_from_invite(invite, username, name, password = nil, user_custom_fields = nil, ip_address = nil) def self.create_user_from_invite(invite, username, name, password = nil, user_custom_fields = nil, ip_address = nil)
user = User.where(staged: true).with_email(invite.email.strip.downcase).first
user.unstage! if user
user ||= User.new
if username && UsernameValidator.new(username).valid_format? && User.username_available?(username) if username && UsernameValidator.new(username).valid_format? && User.username_available?(username)
available_username = username available_username = username
else else
available_username = UserNameSuggester.suggest(invite.email) available_username = UserNameSuggester.suggest(invite.email)
end end
available_name = name || available_username
user_params = { user.attributes = {
email: invite.email, email: invite.email,
username: available_username, username: available_username,
name: available_name, name: name || available_username,
active: false, active: false,
trust_level: SiteSetting.default_invitee_trust_level, trust_level: SiteSetting.default_invitee_trust_level,
ip_address: ip_address, ip_address: ip_address,
registration_ip_address: ip_address registration_ip_address: ip_address
} }
user = User.unstage(user_params)
user = User.new(user_params) if user.nil?
if !SiteSetting.must_approve_users? || (SiteSetting.must_approve_users? && invite.invited_by.staff?) if !SiteSetting.must_approve_users? || (SiteSetting.must_approve_users? && invite.invited_by.staff?)
ReviewableUser.set_approved_fields!(user, invite.invited_by) ReviewableUser.set_approved_fields!(user, invite.invited_by)
end end

View File

@ -363,22 +363,17 @@ class User < ActiveRecord::Base
user user
end end
def unstage def unstage!
if self.staged if self.staged
ActiveRecord::Base.transaction do
self.staged = false self.staged = false
self.custom_fields[FROM_STAGED] = true self.custom_fields[FROM_STAGED] = true
self.notifications.destroy_all self.notifications.destroy_all
DiscourseEvent.trigger(:user_unstaged, self) save!
end
end end
def self.unstage(params) DiscourseEvent.trigger(:user_unstaged, self)
if user = User.where(staged: true).with_email(params[:email].strip.downcase).first
params.each { |k, v| user.public_send("#{k}=", v) }
user.active = false
user.unstage
end end
user
end end
def self.suggest_name(string) def self.suggest_name(string)

View File

@ -167,7 +167,7 @@ class Auth::DefaultCurrentUserProvider
impersonate: opts[:impersonate]) impersonate: opts[:impersonate])
cookies[TOKEN_COOKIE] = cookie_hash(@user_token.unhashed_auth_token) cookies[TOKEN_COOKIE] = cookie_hash(@user_token.unhashed_auth_token)
unstage_user(user) user.unstage!
make_developer_admin(user) make_developer_admin(user)
enable_bootstrap_mode(user) enable_bootstrap_mode(user)
@ -191,13 +191,6 @@ class Auth::DefaultCurrentUserProvider
hash hash
end end
def unstage_user(user)
if user.staged
user.unstage
user.save
end
end
def make_developer_admin(user) def make_developer_admin(user)
if user.active? && if user.active? &&
!user.admin && !user.admin &&

View File

@ -232,8 +232,7 @@ describe Group do
expect(GroupUser.where(user_id: staged.id).count).to eq(0) expect(GroupUser.where(user_id: staged.id).count).to eq(0)
staged.unstage staged.unstage!
staged.save!
expect(GroupUser.where(user_id: staged.id).count).to eq(2) expect(GroupUser.where(user_id: staged.id).count).to eq(2)
end end

View File

@ -1123,7 +1123,7 @@ describe Post do
SiteSetting.newuser_max_links = 3 SiteSetting.newuser_max_links = 3
user = Fabricate(:user, staged: true, trust_level: 0) user = Fabricate(:user, staged: true, trust_level: 0)
user.created_at = 1.hour.ago user.created_at = 1.hour.ago
user.unstage user.unstage!
post = Fabricate(:post, raw: raw, user: user) post = Fabricate(:post, raw: raw, user: user)
expect(post.has_host_spam?).to eq(true) expect(post.has_host_spam?).to eq(true)
end end
@ -1133,7 +1133,7 @@ describe Post do
SiteSetting.newuser_max_links = 3 SiteSetting.newuser_max_links = 3
user = Fabricate(:user, staged: true, trust_level: 0) user = Fabricate(:user, staged: true, trust_level: 0)
user.created_at = 2.days.ago user.created_at = 2.days.ago
user.unstage user.unstage!
post = Fabricate(:post, raw: raw, user: user) post = Fabricate(:post, raw: raw, user: user)
expect(post.has_host_spam?).to eq(false) expect(post.has_host_spam?).to eq(false)
end end

View File

@ -1840,44 +1840,29 @@ describe User do
end end
end end
describe "#unstage" do describe "#unstage!" do
let!(:staged_user) { Fabricate(:staged, email: 'staged@account.com', active: true, username: 'staged1', name: 'Stage Name') } let!(:user) { Fabricate(:staged, email: 'staged@account.com', active: true, username: 'staged1', name: 'Stage Name') }
let(:params) { { email: 'staged@account.com', active: true, username: 'unstaged1', name: 'Foo Bar' } }
it "correctly unstages a user" do it "correctly unstages a user" do
user = User.unstage(params) user.unstage!
expect(user.id).to eq(staged_user.id)
expect(user.username).to eq('unstaged1')
expect(user.name).to eq('Foo Bar')
expect(user.active).to eq(false)
expect(user.email).to eq('staged@account.com')
expect(user.staged).to eq(false) expect(user.staged).to eq(false)
end end
it "returns nil when the user cannot be unstaged" do
Fabricate(:coding_horror)
expect(User.unstage(email: 'jeff@somewhere.com')).to be_nil
expect(User.unstage(email: 'no@account.com')).to be_nil
end
it "removes all previous notifications during unstaging" do it "removes all previous notifications during unstaging" do
Fabricate(:notification, user: staged_user) Fabricate(:notification, user: user)
Fabricate(:private_message_notification, user: staged_user) Fabricate(:private_message_notification, user: user)
staged_user.reload expect(user.total_unread_notifications).to eq(2)
expect(staged_user.total_unread_notifications).to eq(2) user.unstage!
user = User.unstage(params) user.reload
expect(user.total_unread_notifications).to eq(0) expect(user.total_unread_notifications).to eq(0)
expect(user.staged).to eq(false) expect(user.staged).to eq(false)
end end
it "triggers an event" do it "triggers an event" do
unstaged_user = nil event = DiscourseEvent.track_events { user.unstage! }.first
event = DiscourseEvent.track_events { unstaged_user = User.unstage(params) }.first
expect(event[:event_name]).to eq(:user_unstaged) expect(event[:event_name]).to eq(:user_unstaged)
expect(event[:params].first).to eq(unstaged_user) expect(event[:params].first).to eq(user)
end end
end end