# frozen_string_literal: true
RSpec.describe UploadRecovery do
fab!(:user)
let(:upload) do
UploadCreator.new(file_from_fixtures("smallest.png"), "logo.png").create_for(user.id)
end
let(:upload2) do
UploadCreator.new(file_from_fixtures("small.pdf", "pdf"), "some.pdf").create_for(user.id)
end
let(:post) do
Fabricate(:post, raw: "![logo.png](#{upload.short_url})", user: user).tap(&:link_post_uploads)
end
let(:upload_recovery) { UploadRecovery.new }
before do
SiteSetting.authorized_extensions = "png|pdf"
Jobs.run_immediately!
end
after do
[upload, upload2].each do |u|
next if u
public_path = "#{Discourse.store.public_dir}#{u.url}"
[public_path, public_path.sub("uploads", "uploads/tombstone")].each do |path|
File.delete(path) if File.exist?(path)
end
end
end
describe "#recover" do
describe "when given an invalid sha1" do
it "does nothing" do
upload_recovery.expects(:recover_from_local).never
post.update!(raw: "![logo.png](upload://#{"a" * 28}.png)")
upload_recovery.recover
post.update!(
raw: "test",
)
upload_recovery.recover
end
end
it "accepts a custom ActiveRecord relation" do
post.update!(updated_at: 2.days.ago)
upload.destroy!
upload_recovery.expects(:recover_from_local).never
upload_recovery.recover(Post.where("updated_at >= ?", 1.day.ago))
end
describe "for a missing attachment" do
let(:post) { Fabricate(:post, raw: <<~SQL, user: user).tap(&:link_post_uploads) }
some.pdf
blank
SQL
it "recovers the attachment" do
expect do upload2.destroy! end.to change { post.reload.uploads.count }.from(1).to(0)
expect do upload_recovery.recover end.to change { post.reload.uploads.count }.from(0).to(1)
expect(File.read(Discourse.store.path_for(post.uploads.first))).to eq(
File.read(file_from_fixtures("small.pdf", "pdf")),
)
end
end
it "recovers uploads and attachments" do
stub_request(:get, "http://test.localhost#{upload.url}").to_return(status: 200)
expect do upload.destroy! end.to change { post.reload.uploads.count }.from(1).to(0)
expect do upload_recovery.recover end.to change { post.reload.uploads.count }.from(0).to(1)
expect(File.read(Discourse.store.path_for(post.uploads.first))).to eq(
File.read(file_from_fixtures("smallest.png")),
)
end
describe "S3 store" do
before do
setup_s3
stub_s3_store
end
it "recovers the upload" do
expect do upload.destroy! end.to change { post.reload.uploads.count }.from(1).to(0)
original_key = Discourse.store.get_path_for_upload(upload)
tombstone_key = original_key.sub("original", "tombstone/original")
tombstone_copy = stub
tombstone_copy.expects(:key).returns(tombstone_key)
Discourse.store.s3_helper.expects(:list).with("original").returns([])
Discourse
.store
.s3_helper
.expects(:list)
.with("#{FileStore::S3Store::TOMBSTONE_PREFIX}original")
.returns([tombstone_copy])
Discourse
.store
.s3_helper
.expects(:copy)
.with(tombstone_key, original_key, options: { acl: "public-read" })
FileHelper.expects(:download).returns(file_from_fixtures("smallest.png"))
stub_request(:get, upload.url).to_return(body: file_from_fixtures("smallest.png"))
expect do upload_recovery.recover end.to change { post.reload.uploads.count }.from(0).to(1)
end
describe "when the upload exists but its file is missing" do
it "recovers the file" do
upload.verification_status = Upload.verification_statuses[:invalid_etag]
upload.save!
original_key = Discourse.store.get_path_for_upload(upload)
tombstone_key = original_key.sub("original", "tombstone/original")
tombstone_copy = stub
tombstone_copy.expects(:key).returns(tombstone_key)
Discourse.store.s3_helper.expects(:list).with("original").returns([])
Discourse
.store
.s3_helper
.expects(:list)
.with("#{FileStore::S3Store::TOMBSTONE_PREFIX}original")
.returns([tombstone_copy])
Discourse
.store
.s3_helper
.expects(:copy)
.with(tombstone_key, original_key, options: { acl: "public-read" })
expect do upload_recovery.recover end.to_not change {
[post.reload.uploads.count, Upload.count]
}
end
it "does not create a duplicate upload when secure uploads are enabled" do
SiteSetting.secure_uploads = true
upload.verification_status = Upload.verification_statuses[:invalid_etag]
upload.save!
original_key = Discourse.store.get_path_for_upload(upload)
tombstone_key = original_key.sub("original", "tombstone/original")
tombstone_copy = stub
tombstone_copy.expects(:key).returns(tombstone_key)
Discourse.store.s3_helper.expects(:list).with("original").returns([])
Discourse
.store
.s3_helper
.expects(:list)
.with("#{FileStore::S3Store::TOMBSTONE_PREFIX}original")
.returns([tombstone_copy])
Discourse
.store
.s3_helper
.expects(:copy)
.with(tombstone_key, original_key, options: { acl: "public-read" })
expect do upload_recovery.recover end.to_not change {
[post.reload.uploads.count, Upload.count]
}
end
end
end
describe "image tag" do
let(:post) { Fabricate(:post, raw: <<~SQL, user: user).tap(&:link_post_uploads) }
SQL
it "recovers the upload" do
stub_request(:get, "http://test.localhost#{upload.url}").to_return(status: 200)
expect do upload.destroy! end.to change { post.reload.uploads.count }.from(1).to(0)
expect do upload_recovery.recover end.to change { post.reload.uploads.count }.from(0).to(1)
expect(File.read(Discourse.store.path_for(post.uploads.first))).to eq(
File.read(file_from_fixtures("smallest.png")),
)
end
end
describe "image markdown" do
let(:post) { Fabricate(:post, raw: <<~SQL, user: user).tap(&:link_post_uploads) }
![image](#{upload.url})
SQL
it "recovers the upload" do
stub_request(:get, "http://test.localhost#{upload.url}").to_return(status: 200)
expect do upload.destroy! end.to change { post.reload.uploads.count }.from(1).to(0)
expect do upload_recovery.recover end.to change { post.reload.uploads.count }.from(0).to(1)
expect(File.read(Discourse.store.path_for(post.uploads.first))).to eq(
File.read(file_from_fixtures("smallest.png")),
)
end
end
describe "bbcode" do
let(:post) { Fabricate(:post, raw: <<~SQL, user: user).tap(&:link_post_uploads) }
[img]#{upload.url}[/img]
SQL
it "recovers the upload" do
stub_request(:get, "http://test.localhost#{upload.url}").to_return(status: 200)
expect do upload.destroy! end.to change { post.reload.uploads.count }.from(1).to(0)
expect do upload_recovery.recover end.to change { post.reload.uploads.count }.from(0).to(1)
expect(File.read(Discourse.store.path_for(post.uploads.first))).to eq(
File.read(file_from_fixtures("smallest.png")),
)
end
end
end
end