FIX: Destroy all posts when hard deleting topic (#17359)
Hard deleting topics that contained soft deleted posts or small actions used to create orphan posts because only the first post was hard deleted. This commit adds an error message if there are still posts left in the topic that must be hard deleted first or hard deletes all small actions too immediately (there is no other way of hard deleting a small action because there is no wrench menu).
This commit is contained in:
parent
590a13377b
commit
a0537816fb
|
@ -321,14 +321,12 @@ class PostsController < ApplicationController
|
||||||
|
|
||||||
def destroy
|
def destroy
|
||||||
post = find_post_from_params
|
post = find_post_from_params
|
||||||
|
force_destroy = ActiveModel::Type::Boolean.new.cast(params[:force_destroy])
|
||||||
|
|
||||||
force_destroy = false
|
if force_destroy
|
||||||
if params[:force_destroy].present?
|
|
||||||
if !guardian.can_permanently_delete?(post)
|
if !guardian.can_permanently_delete?(post)
|
||||||
return render_json_error post.cannot_permanently_delete_reason(current_user), status: 403
|
return render_json_error post.cannot_permanently_delete_reason(current_user), status: 403
|
||||||
end
|
end
|
||||||
|
|
||||||
force_destroy = true
|
|
||||||
else
|
else
|
||||||
guardian.ensure_can_delete!(post)
|
guardian.ensure_can_delete!(post)
|
||||||
end
|
end
|
||||||
|
@ -338,8 +336,12 @@ class PostsController < ApplicationController
|
||||||
RateLimiter.new(current_user, "delete_post_per_day", SiteSetting.max_post_deletions_per_day, 1.day).performed!
|
RateLimiter.new(current_user, "delete_post_per_day", SiteSetting.max_post_deletions_per_day, 1.day).performed!
|
||||||
end
|
end
|
||||||
|
|
||||||
destroyer = PostDestroyer.new(current_user, post, context: params[:context], force_destroy: force_destroy)
|
PostDestroyer.new(
|
||||||
destroyer.destroy
|
current_user,
|
||||||
|
post,
|
||||||
|
context: params[:context],
|
||||||
|
force_destroy: force_destroy
|
||||||
|
).destroy
|
||||||
|
|
||||||
render body: nil
|
render body: nil
|
||||||
end
|
end
|
||||||
|
|
|
@ -621,20 +621,22 @@ class TopicsController < ApplicationController
|
||||||
|
|
||||||
def destroy
|
def destroy
|
||||||
topic = Topic.with_deleted.find_by(id: params[:id])
|
topic = Topic.with_deleted.find_by(id: params[:id])
|
||||||
|
force_destroy = ActiveModel::Type::Boolean.new.cast(params[:force_destroy])
|
||||||
|
|
||||||
force_destroy = false
|
if force_destroy
|
||||||
if params[:force_destroy].present?
|
|
||||||
if !guardian.can_permanently_delete?(topic)
|
if !guardian.can_permanently_delete?(topic)
|
||||||
return render_json_error topic.cannot_permanently_delete_reason(current_user), status: 403
|
return render_json_error topic.cannot_permanently_delete_reason(current_user), status: 403
|
||||||
end
|
end
|
||||||
|
|
||||||
force_destroy = true
|
|
||||||
else
|
else
|
||||||
guardian.ensure_can_delete!(topic)
|
guardian.ensure_can_delete!(topic)
|
||||||
end
|
end
|
||||||
|
|
||||||
first_post = topic.posts.with_deleted.order(:post_number).first
|
PostDestroyer.new(
|
||||||
PostDestroyer.new(current_user, first_post, context: params[:context], force_destroy: force_destroy).destroy
|
current_user,
|
||||||
|
topic.ordered_posts.with_deleted.first,
|
||||||
|
context: params[:context],
|
||||||
|
force_destroy: force_destroy
|
||||||
|
).destroy
|
||||||
|
|
||||||
render body: nil
|
render body: nil
|
||||||
rescue Discourse::InvalidAccess
|
rescue Discourse::InvalidAccess
|
||||||
|
|
|
@ -1786,7 +1786,12 @@ class Topic < ActiveRecord::Base
|
||||||
end
|
end
|
||||||
|
|
||||||
def cannot_permanently_delete_reason(user)
|
def cannot_permanently_delete_reason(user)
|
||||||
if self.posts_count > 0
|
all_posts_count = Post.with_deleted
|
||||||
|
.where(topic_id: self.id)
|
||||||
|
.where(post_type: [Post.types[:regular], Post.types[:moderator_action], Post.types[:whisper]])
|
||||||
|
.count
|
||||||
|
|
||||||
|
if posts_count > 0 || all_posts_count > 1
|
||||||
I18n.t('post.cannot_permanently_delete.many_posts')
|
I18n.t('post.cannot_permanently_delete.many_posts')
|
||||||
elsif self.deleted_by_id == user&.id && self.deleted_at >= Post::PERMANENT_DELETE_TIMER.ago
|
elsif self.deleted_by_id == user&.id && self.deleted_at >= Post::PERMANENT_DELETE_TIMER.ago
|
||||||
time_left = RateLimiter.time_left(Post::PERMANENT_DELETE_TIMER.to_i - Time.zone.now.to_i + self.deleted_at.to_i)
|
time_left = RateLimiter.time_left(Post::PERMANENT_DELETE_TIMER.to_i - Time.zone.now.to_i + self.deleted_at.to_i)
|
||||||
|
|
|
@ -158,7 +158,20 @@ module TopicGuardian
|
||||||
def can_permanently_delete_topic?(topic)
|
def can_permanently_delete_topic?(topic)
|
||||||
return false if !SiteSetting.can_permanently_delete
|
return false if !SiteSetting.can_permanently_delete
|
||||||
return false if !topic
|
return false if !topic
|
||||||
|
|
||||||
|
# Ensure that all posts (including small actions) are at least soft
|
||||||
|
# deleted.
|
||||||
return false if topic.posts_count > 0
|
return false if topic.posts_count > 0
|
||||||
|
|
||||||
|
# All other posts that were deleted still must be permanently deleted
|
||||||
|
# before the topic can be deleted with the exception of small action
|
||||||
|
# posts that will be deleted right before the topic is.
|
||||||
|
all_posts_count = Post.with_deleted
|
||||||
|
.where(topic_id: topic.id)
|
||||||
|
.where(post_type: [Post.types[:regular], Post.types[:moderator_action], Post.types[:whisper]])
|
||||||
|
.count
|
||||||
|
return false if all_posts_count > 1
|
||||||
|
|
||||||
return false if !is_admin? || !can_see_topic?(topic)
|
return false if !is_admin? || !can_see_topic?(topic)
|
||||||
return false if !topic.deleted_at
|
return false if !topic.deleted_at
|
||||||
return false if topic.deleted_by_id == @user.id && topic.deleted_at >= Post::PERMANENT_DELETE_TIMER.ago
|
return false if topic.deleted_by_id == @user.id && topic.deleted_at >= Post::PERMANENT_DELETE_TIMER.ago
|
||||||
|
|
|
@ -145,6 +145,14 @@ class PostDestroyer
|
||||||
# show up in the topic
|
# show up in the topic
|
||||||
# Permanent option allows to hard delete.
|
# Permanent option allows to hard delete.
|
||||||
def perform_delete
|
def perform_delete
|
||||||
|
# All posts in the topic must be force deleted if the first is force
|
||||||
|
# deleted (except @post which is destroyed by current instance).
|
||||||
|
if @topic && @post.is_first_post? && permanent?
|
||||||
|
@topic.ordered_posts.with_deleted.reverse_order.find_each do |post|
|
||||||
|
PostDestroyer.new(@user, post, @opts).destroy if post.id != @post.id
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
Post.transaction do
|
Post.transaction do
|
||||||
permanent? ? @post.destroy! : @post.trash!(@user)
|
permanent? ? @post.destroy! : @post.trash!(@user)
|
||||||
if @post.topic
|
if @post.topic
|
||||||
|
|
|
@ -3030,6 +3030,9 @@ RSpec.describe Topic do
|
||||||
expect(topic.reload.cannot_permanently_delete_reason(Fabricate(:admin))).to eq(I18n.t('post.cannot_permanently_delete.many_posts'))
|
expect(topic.reload.cannot_permanently_delete_reason(Fabricate(:admin))).to eq(I18n.t('post.cannot_permanently_delete.many_posts'))
|
||||||
|
|
||||||
PostDestroyer.new(admin, post_2.reload).destroy
|
PostDestroyer.new(admin, post_2.reload).destroy
|
||||||
|
expect(topic.reload.cannot_permanently_delete_reason(Fabricate(:admin))).to eq(I18n.t('post.cannot_permanently_delete.many_posts'))
|
||||||
|
|
||||||
|
PostDestroyer.new(admin, post_2.reload, force_destroy: true).destroy
|
||||||
expect(topic.reload.cannot_permanently_delete_reason(Fabricate(:admin))).to eq(nil)
|
expect(topic.reload.cannot_permanently_delete_reason(Fabricate(:admin))).to eq(nil)
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -1188,6 +1188,48 @@ RSpec.describe TopicsController do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
describe 'force destroy' do
|
||||||
|
fab!(:post) { Fabricate(:post, topic: topic, post_number: 1) }
|
||||||
|
|
||||||
|
before do
|
||||||
|
SiteSetting.can_permanently_delete = true
|
||||||
|
|
||||||
|
sign_in(admin)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'force destroys all deleted small actions in topic too' do
|
||||||
|
small_action_post = Fabricate(:small_action, topic: topic)
|
||||||
|
PostDestroyer.new(Discourse.system_user, post).destroy
|
||||||
|
PostDestroyer.new(Discourse.system_user, small_action_post).destroy
|
||||||
|
|
||||||
|
delete "/t/#{topic.id}.json", params: { force_destroy: true }
|
||||||
|
|
||||||
|
expect(response.status).to eq(200)
|
||||||
|
|
||||||
|
expect(Topic.find_by(id: topic.id)).to eq(nil)
|
||||||
|
expect(Post.find_by(id: post.id)).to eq(nil)
|
||||||
|
expect(Post.find_by(id: small_action_post.id)).to eq(nil)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'does not allow to destroy topic if not all posts were force destroyed' do
|
||||||
|
other_post = Fabricate(:post, topic: topic, post_number: 2)
|
||||||
|
PostDestroyer.new(Discourse.system_user, post).destroy
|
||||||
|
|
||||||
|
delete "/t/#{topic.id}.json", params: { force_destroy: true }
|
||||||
|
|
||||||
|
expect(response.status).to eq(403)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'does not allow to destroy topic if not all small action posts were deleted' do
|
||||||
|
small_action_post = Fabricate(:small_action, topic: topic)
|
||||||
|
PostDestroyer.new(Discourse.system_user, small_action_post).destroy
|
||||||
|
|
||||||
|
delete "/t/#{topic.id}.json", params: { force_destroy: true }
|
||||||
|
|
||||||
|
expect(response.status).to eq(403)
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
describe '#id_for_slug' do
|
describe '#id_for_slug' do
|
||||||
|
|
Loading…
Reference in New Issue