2018-07-18 12:03:33 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2017-06-13 05:32:21 -04:00
|
|
|
module Users
|
|
|
|
class UpdateService < BaseService
|
2017-08-25 19:15:43 -04:00
|
|
|
include NewUserNotifier
|
2019-06-19 16:27:34 -04:00
|
|
|
attr_reader :user, :identity_params
|
2017-08-25 19:15:43 -04:00
|
|
|
|
2021-09-29 08:11:22 -04:00
|
|
|
ATTRS_REQUIRING_PASSWORD_CHECK = %w[email].freeze
|
|
|
|
|
2017-09-27 05:48:33 -04:00
|
|
|
def initialize(current_user, params = {})
|
2017-09-20 06:00:33 -04:00
|
|
|
@current_user = current_user
|
2021-09-29 08:11:22 -04:00
|
|
|
@validation_password = params.delete(:validation_password)
|
2017-09-27 06:08:10 -04:00
|
|
|
@user = params.delete(:user)
|
2018-07-13 11:52:31 -04:00
|
|
|
@status_params = params.delete(:status)
|
2019-06-19 16:27:34 -04:00
|
|
|
@identity_params = params.slice(*identity_attributes)
|
2017-06-13 05:32:21 -04:00
|
|
|
@params = params.dup
|
|
|
|
end
|
|
|
|
|
2021-09-29 08:11:22 -04:00
|
|
|
def execute(validate: true, check_password: false, &block)
|
2022-08-03 08:11:38 -04:00
|
|
|
yield(@user) if block
|
2017-06-23 11:11:31 -04:00
|
|
|
|
2017-09-20 06:31:35 -04:00
|
|
|
user_exists = @user.persisted?
|
2021-06-16 14:10:35 -04:00
|
|
|
@user.user_detail # prevent assignment
|
2020-01-13 01:08:10 -05:00
|
|
|
|
|
|
|
discard_read_only_attributes
|
2018-12-14 17:58:57 -05:00
|
|
|
assign_attributes
|
2021-09-29 08:11:22 -04:00
|
|
|
|
|
|
|
if check_password && require_password_check? && !@user.valid_password?(@validation_password)
|
|
|
|
return error(s_("Profiles|Invalid password"))
|
|
|
|
end
|
|
|
|
|
2019-06-19 16:27:34 -04:00
|
|
|
assign_identity
|
2020-03-24 05:09:25 -04:00
|
|
|
build_canonical_email
|
2017-06-13 05:32:21 -04:00
|
|
|
|
2018-07-13 11:52:31 -04:00
|
|
|
if @user.save(validate: validate) && update_status
|
2017-09-20 06:31:35 -04:00
|
|
|
notify_success(user_exists)
|
2017-06-13 05:32:21 -04:00
|
|
|
else
|
2018-07-13 11:52:31 -04:00
|
|
|
messages = @user.errors.full_messages + Array(@user.status&.errors&.full_messages)
|
|
|
|
error(messages.uniq.join('. '))
|
2017-06-13 05:32:21 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-09-29 08:11:22 -04:00
|
|
|
def execute!(*args, **kargs, &block)
|
|
|
|
result = execute(*args, **kargs, &block)
|
2017-06-14 05:35:58 -04:00
|
|
|
|
2021-05-04 11:10:36 -04:00
|
|
|
raise ActiveRecord::RecordInvalid, @user unless result[:status] == :success
|
2017-06-22 06:10:28 -04:00
|
|
|
|
|
|
|
true
|
2017-06-14 05:35:58 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
2021-09-29 08:11:22 -04:00
|
|
|
def require_password_check?
|
|
|
|
return false unless @user.persisted?
|
|
|
|
return false if @user.password_automatically_set?
|
|
|
|
|
|
|
|
changes = @user.changed
|
|
|
|
ATTRS_REQUIRING_PASSWORD_CHECK.any? { |param| changes.include?(param) }
|
|
|
|
end
|
|
|
|
|
2020-03-24 05:09:25 -04:00
|
|
|
def build_canonical_email
|
|
|
|
return unless @user.email_changed?
|
|
|
|
|
|
|
|
Users::UpdateCanonicalEmailService.new(user: @user).execute
|
|
|
|
end
|
|
|
|
|
2018-07-13 11:52:31 -04:00
|
|
|
def update_status
|
|
|
|
return true unless @status_params
|
|
|
|
|
|
|
|
Users::SetStatusService.new(current_user, @status_params.merge(user: @user)).execute
|
|
|
|
end
|
|
|
|
|
2017-09-20 06:31:35 -04:00
|
|
|
def notify_success(user_exists)
|
|
|
|
notify_new_user(@user, nil) unless user_exists
|
2017-09-20 05:00:06 -04:00
|
|
|
|
|
|
|
success
|
|
|
|
end
|
|
|
|
|
2020-01-13 01:08:10 -05:00
|
|
|
def discard_read_only_attributes
|
2020-04-01 20:08:11 -04:00
|
|
|
discard_synced_attributes
|
2020-01-13 01:08:10 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def discard_synced_attributes
|
2018-12-14 17:58:57 -05:00
|
|
|
if (metadata = @user.user_synced_attributes_metadata)
|
|
|
|
read_only = metadata.read_only_attributes
|
|
|
|
|
|
|
|
params.reject! { |key, _| read_only.include?(key.to_sym) }
|
2017-08-29 04:57:41 -04:00
|
|
|
end
|
2020-01-13 01:08:10 -05:00
|
|
|
end
|
2017-08-29 04:57:41 -04:00
|
|
|
|
2020-01-13 01:08:10 -05:00
|
|
|
def assign_attributes
|
2019-09-18 10:02:45 -04:00
|
|
|
@user.assign_attributes(params.except(*identity_attributes)) unless params.empty?
|
2019-06-19 16:27:34 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def assign_identity
|
|
|
|
return unless identity_params.present?
|
|
|
|
|
2019-06-26 16:11:30 -04:00
|
|
|
identity = user.identities.find_or_create_by(provider_params) # rubocop: disable CodeReuse/ActiveRecord
|
2019-06-19 16:27:34 -04:00
|
|
|
identity.update(identity_params)
|
|
|
|
end
|
|
|
|
|
|
|
|
def identity_attributes
|
|
|
|
[:provider, :extern_uid]
|
2017-06-14 05:35:58 -04:00
|
|
|
end
|
2019-06-26 16:11:30 -04:00
|
|
|
|
|
|
|
def provider_attributes
|
|
|
|
[:provider]
|
|
|
|
end
|
|
|
|
|
|
|
|
def provider_params
|
|
|
|
identity_params.slice(*provider_attributes)
|
|
|
|
end
|
2017-06-13 05:32:21 -04:00
|
|
|
end
|
|
|
|
end
|
2019-09-13 09:26:31 -04:00
|
|
|
|
2021-05-11 17:10:21 -04:00
|
|
|
Users::UpdateService.prepend_mod_with('Users::UpdateService')
|