FIX: Always allow us to reject users, even if they are deleted

This commit is contained in:
Robin Ward 2019-04-10 11:00:14 -04:00
parent 3d545d66df
commit 5d99346740
3 changed files with 67 additions and 48 deletions

View File

@ -12,7 +12,7 @@ class ReviewableUser < Reviewable
return unless pending? return unless pending?
actions.add(:approve) if guardian.can_approve?(target) || args[:approved_by_invite] actions.add(:approve) if guardian.can_approve?(target) || args[:approved_by_invite]
actions.add(:reject) if guardian.can_delete_user?(target) actions.add(:reject)
end end
def perform_approve(performed_by, args) def perform_approve(performed_by, args)
@ -34,13 +34,20 @@ class ReviewableUser < Reviewable
end end
def perform_reject(performed_by, args) def perform_reject(performed_by, args)
destroyer = UserDestroyer.new(performed_by) unless args[:skip_delete]
# If a user has posts, we won't delete them to preserve their content. # We'll delete the user if we can
# However the reviable record will be "rejected" and they will remain if target.present?
# unapproved in the database. A staff member can still approve them destroyer = UserDestroyer.new(performed_by)
# via the admin.
destroyer.destroy(target) rescue UserDestroyer::PostsExistError begin
destroyer.destroy(target)
rescue UserDestroyer::PostsExistError
# If a user has posts, we won't delete them to preserve their content.
# However the reviable record will be "rejected" and they will remain
# unapproved in the database. A staff member can still approve them
# via the admin.
end
end
create_result(:success, :rejected) create_result(:success, :rejected)
end end

View File

@ -23,15 +23,13 @@ class UserDestroyer
prepare_for_destroy(user) if opts[:prepare_for_destroy] == true prepare_for_destroy(user) if opts[:prepare_for_destroy] == true
result = nil
optional_transaction(open_transaction: opts[:transaction]) do optional_transaction(open_transaction: opts[:transaction]) do
Draft.where(user_id: user.id).delete_all Draft.where(user_id: user.id).delete_all
Reviewable.where(created_by_id: user.id).delete_all Reviewable.where(created_by_id: user.id).delete_all
if reviewable = Reviewable.find_by(target: user)
reviewable.perform(@actor, :reject, skip_delete: true) rescue Reviewable::InvalidAction
end
if opts[:delete_posts] if opts[:delete_posts]
user.posts.each do |post| user.posts.each do |post|
@ -75,47 +73,52 @@ class UserDestroyer
# keep track of emails used # keep track of emails used
user_emails = user.user_emails.pluck(:email) user_emails = user.user_emails.pluck(:email)
user.destroy.tap do |u| if result = user.destroy
if u if opts[:block_email]
if opts[:block_email] user_emails.each do |email|
user_emails.each do |email| ScreenedEmail.block(email, ip_address: result.ip_address)&.record_match!
ScreenedEmail.block(email, ip_address: u.ip_address)&.record_match!
end
end end
if opts[:block_ip] && u.ip_address
ScreenedIpAddress.watch(u.ip_address)&.record_match!
if u.registration_ip_address && u.ip_address != u.registration_ip_address
ScreenedIpAddress.watch(u.registration_ip_address)&.record_match!
end
end
Post.unscoped.where(user_id: u.id).update_all(user_id: nil)
# If this user created categories, fix those up:
Category.where(user_id: u.id).each do |c|
c.user_id = Discourse::SYSTEM_USER_ID
c.save!
if topic = Topic.unscoped.find_by(id: c.topic_id)
topic.recover!
topic.user_id = Discourse::SYSTEM_USER_ID
topic.save!
end
end
unless opts[:quiet]
if @actor == user
deleted_by = Discourse.system_user
opts[:context] = I18n.t("staff_action_logs.user_delete_self", url: opts[:context])
else
deleted_by = @actor
end
StaffActionLogger.new(deleted_by).log_user_deletion(user, opts.slice(:context))
end
MessageBus.publish "/file-change", ["refresh"], user_ids: [u.id]
end end
if opts[:block_ip] && result.ip_address
ScreenedIpAddress.watch(result.ip_address)&.record_match!
if result.registration_ip_address && result.ip_address != result.registration_ip_address
ScreenedIpAddress.watch(result.registration_ip_address)&.record_match!
end
end
Post.unscoped.where(user_id: result.id).update_all(user_id: nil)
# If this user created categories, fix those up:
Category.where(user_id: result.id).each do |c|
c.user_id = Discourse::SYSTEM_USER_ID
c.save!
if topic = Topic.unscoped.find_by(id: c.topic_id)
topic.recover!
topic.user_id = Discourse::SYSTEM_USER_ID
topic.save!
end
end
unless opts[:quiet]
if @actor == user
deleted_by = Discourse.system_user
opts[:context] = I18n.t("staff_action_logs.user_delete_self", url: opts[:context])
else
deleted_by = @actor
end
StaffActionLogger.new(deleted_by).log_user_deletion(user, opts.slice(:context))
end
MessageBus.publish "/file-change", ["refresh"], user_ids: [result.id]
end end
end end
# After the user is deleted, remove the reviewable
if reviewable = Reviewable.pending.find_by(target: user)
reviewable.perform(@actor, :reject)
end
result
end end
protected protected

View File

@ -87,6 +87,15 @@ RSpec.describe ReviewableUser, type: :model do
expect(reviewable.target).to be_present expect(reviewable.target).to be_present
expect(reviewable.target.approved).to eq(false) expect(reviewable.target.approved).to eq(false)
end end
it "allows us to reject a user who has been deleted" do
reviewable.target.destroy!
reviewable.reload
result = reviewable.perform(moderator, :reject)
expect(result.success?).to eq(true)
expect(reviewable.rejected?).to eq(true)
expect(reviewable.target).to be_blank
end
end end
end end