FIX: delete orphan post revisions (#12502)

I was adding specs to ensure that post actions and uploads are removed for permanently deleted posts.

I noticed that post revisions were not permanently destroyed. I added a migration to fix old data.
This commit is contained in:
Krzysztof Kotlarek 2021-03-25 12:34:53 +11:00 committed by GitHub
parent ea6f9af08b
commit c03c85e661
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 37 additions and 0 deletions

View File

@ -0,0 +1,19 @@
# frozen_string_literal: true
class DeleteOrphanPostRevisions < ActiveRecord::Migration[6.0]
def up
sql = <<~SQL
DELETE FROM post_revisions
USING post_revisions pr
LEFT JOIN posts ON posts.id = pr.post_id
WHERE posts.id IS NULL
AND post_revisions.id = pr.id
SQL
execute(sql)
end
def down
raise ActiveRecord::IrreversibleMigration
end
end

View File

@ -151,6 +151,7 @@ class PostDestroyer
Topic.reset_highest(@post.topic_id) Topic.reset_highest(@post.topic_id)
end end
trash_public_post_actions trash_public_post_actions
trash_revisions
trash_user_actions trash_user_actions
remove_associated_replies remove_associated_replies
remove_associated_notifications remove_associated_notifications
@ -291,6 +292,11 @@ class PostDestroyer
end end
end end
def trash_revisions
return unless permanent?
@post.revisions.each(&:destroy!)
end
def agree(reviewable) def agree(reviewable)
notify_deletion(reviewable) notify_deletion(reviewable)
result = reviewable.perform(@user, :agree_and_keep, post_was_deleted: true) result = reviewable.perform(@user, :agree_and_keep, post_was_deleted: true)

View File

@ -978,6 +978,9 @@ describe PostDestroyer do
fab!(:private_post) { Fabricate(:private_message_post, topic: private_message_topic) } fab!(:private_post) { Fabricate(:private_message_post, topic: private_message_topic) }
fab!(:post_action) { Fabricate(:post_action, post: private_post) } fab!(:post_action) { Fabricate(:post_action, post: private_post) }
fab!(:reply) { Fabricate(:private_message_post, topic: private_message_topic) } fab!(:reply) { Fabricate(:private_message_post, topic: private_message_topic) }
fab!(:post_revision) { Fabricate(:post_revision, post: private_post) }
fab!(:upload1) { Fabricate(:upload_s3, created_at: 5.hours.ago) }
fab!(:post_upload) { PostUpload.create(post: private_post, upload: upload1) }
it "destroys the post and topic if deleting first post" do it "destroys the post and topic if deleting first post" do
PostDestroyer.new(reply.user, reply, permanent: true).destroy PostDestroyer.new(reply.user, reply, permanent: true).destroy
@ -988,6 +991,13 @@ describe PostDestroyer do
expect { private_post.reload }.to raise_error(ActiveRecord::RecordNotFound) expect { private_post.reload }.to raise_error(ActiveRecord::RecordNotFound)
expect { private_message_topic.reload }.to raise_error(ActiveRecord::RecordNotFound) expect { private_message_topic.reload }.to raise_error(ActiveRecord::RecordNotFound)
expect { post_action.reload }.to raise_error(ActiveRecord::RecordNotFound) expect { post_action.reload }.to raise_error(ActiveRecord::RecordNotFound)
expect { post_revision.reload }.to raise_error(ActiveRecord::RecordNotFound)
expect { post_upload.reload }.to raise_error(ActiveRecord::RecordNotFound)
Jobs::CleanUpUploads.new.reset_last_cleanup!
SiteSetting.clean_orphan_uploads_grace_period_hours = 1
Jobs::CleanUpUploads.new.execute({})
expect { upload1.reload }.to raise_error(ActiveRecord::RecordNotFound)
end end
it 'soft delete if not creator of post or not private message' do it 'soft delete if not creator of post or not private message' do
@ -996,6 +1006,8 @@ describe PostDestroyer do
PostDestroyer.new(post.user, post, permanent: true).destroy PostDestroyer.new(post.user, post, permanent: true).destroy
expect(post.user_deleted).to be true expect(post.user_deleted).to be true
expect(post_revision.reload.persisted?).to be true
end end
it 'always destroy the post when the force_destroy option is passed' do it 'always destroy the post when the force_destroy option is passed' do