Skip to content

Commit

Permalink
fix: Thumbnails are not deleted on room deletion (#34851)
Browse files Browse the repository at this point in the history
  • Loading branch information
matheusbsilva137 authored Dec 31, 2024
1 parent 1b17bb1 commit 7619988
Show file tree
Hide file tree
Showing 5 changed files with 122 additions and 4 deletions.
5 changes: 5 additions & 0 deletions .changeset/twelve-pumas-behave.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@rocket.chat/meteor": patch
---

Fixes thumbnails not being deleted from storage on room deletion
1 change: 1 addition & 0 deletions apps/meteor/.mocharc.js
Original file line number Diff line number Diff line change
Expand Up @@ -37,5 +37,6 @@ module.exports = {
'tests/unit/server/**/*.tests.ts',
'tests/unit/server/**/*.spec.ts',
'app/api/**/*.spec.ts',
'app/file-upload/server/**/*.spec.ts',
],
};
103 changes: 103 additions & 0 deletions apps/meteor/app/file-upload/server/lib/FileUpload.spec.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,103 @@
import { expect } from 'chai';
import { before, beforeEach, describe, it } from 'mocha';
import proxyquire from 'proxyquire';
import sinon from 'sinon';

import { createFakeMessageWithAttachment } from '../../../../tests/mocks/data';

const fakeStorageModel = { findOneById: sinon.stub(), deleteFile: sinon.stub() };
const settingsStub = { watch: sinon.stub(), get: sinon.stub() };
const settingsGetMap = new Map();
const messagesModelStub = {
find: sinon.stub(),
};

const { FileUpload, FileUploadClass } = proxyquire.noCallThru().load('./FileUpload', {
'@rocket.chat/models': {
Messages: messagesModelStub,
},
'meteor/check': sinon.stub(),
'meteor/meteor': sinon.stub(),
'meteor/ostrio:cookies': { Cookies: sinon.stub() },
'sharp': sinon.stub(),
'stream-buffers': sinon.stub(),
'./streamToBuffer': sinon.stub(),
'../../../../server/lib/i18n': sinon.stub(),
'../../../../server/lib/logger/system': sinon.stub(),
'../../../../server/lib/rooms/roomCoordinator': sinon.stub(),
'../../../../server/ufs': sinon.stub(),
'../../../../server/ufs/ufs-methods': sinon.stub(),
'../../../settings/server': { settings: settingsStub },
'../../../utils/lib/mimeTypes': sinon.stub(),
'../../../utils/server/lib/JWTHelper': sinon.stub(),
'../../../utils/server/restrictions': sinon.stub(),
});

describe('FileUpload', () => {
before(() => {
new FileUploadClass({ name: 'fakeStorage:Uploads', model: fakeStorageModel, store: {} });
settingsGetMap.set('FileUpload_Storage_Type', 'fakeStorage');
settingsStub.get.callsFake((settingName) => settingsGetMap.get(settingName));
});

beforeEach(() => {
messagesModelStub.find.reset();
fakeStorageModel.findOneById.reset();
fakeStorageModel.deleteFile.reset();
});

it('should not remove any file if no room id is provided', async () => {
expect(await FileUpload.removeFilesByRoomId()).to.be.undefined;

expect(messagesModelStub.find.called).to.be.false;
expect(fakeStorageModel.findOneById.called).to.be.false;
});

it('should not remove any file if an empty room id is provided', async () => {
expect(await FileUpload.removeFilesByRoomId('')).to.be.undefined;

expect(messagesModelStub.find.called).to.be.false;
expect(fakeStorageModel.findOneById.called).to.be.false;
});

it('should not remove any file if an invalid room id is provided', async () => {
messagesModelStub.find.returns([]);
expect(await FileUpload.removeFilesByRoomId('invalid')).to.be.undefined;

expect(messagesModelStub.find.called).to.be.true;
expect(fakeStorageModel.findOneById.called).to.be.false;
});

it('should delete file from storage if message contains a single file', async () => {
fakeStorageModel.findOneById.resolves({ _id: 'file-id', store: 'fakeStorage:Uploads' });

const fakeMessage = createFakeMessageWithAttachment();
messagesModelStub.find.returns([fakeMessage]);
expect(await FileUpload.removeFilesByRoomId('invalid')).to.be.undefined;

expect(messagesModelStub.find.called).to.be.true;
expect(fakeStorageModel.findOneById.calledOnceWith(fakeMessage.files?.[0]._id)).to.be.true;
expect(fakeStorageModel.deleteFile.calledOnceWith('file-id')).to.be.true;
});

it('should delete multiple files from storage if message contains many files (e.g. image and thumbnail)', async () => {
fakeStorageModel.findOneById.callsFake((_id) => ({ _id, store: 'fakeStorage:Uploads' }));

const fakeMessage = createFakeMessageWithAttachment({
files: [
{ _id: 'file-id', name: 'image', size: 100, type: 'image/png', format: 'png' },
{ _id: 'thumbnail-id', name: 'thumbnail-image', size: 25, type: 'image/png', format: 'png' },
],
});
messagesModelStub.find.returns([fakeMessage]);
expect(await FileUpload.removeFilesByRoomId('invalid')).to.be.undefined;

expect(messagesModelStub.find.called).to.be.true;
expect(fakeStorageModel.findOneById.calledTwice).to.be.true;
expect(fakeStorageModel.findOneById.calledWith('file-id')).to.be.true;
expect(fakeStorageModel.findOneById.calledWith('thumbnail-id')).to.be.true;
expect(fakeStorageModel.deleteFile.calledTwice).to.be.true;
expect(fakeStorageModel.deleteFile.calledWith('file-id')).to.be.true;
expect(fakeStorageModel.deleteFile.calledWith('thumbnail-id')).to.be.true;
});
});
8 changes: 4 additions & 4 deletions apps/meteor/app/file-upload/server/lib/FileUpload.ts
Original file line number Diff line number Diff line change
Expand Up @@ -638,20 +638,20 @@ export const FileUpload = {
const cursor = Messages.find(
{
rid,
'file._id': {
'files._id': {
$exists: true,
},
},
{
projection: {
'file._id': 1,
'files._id': 1,
},
},
);

for await (const document of cursor) {
if (document.file) {
await FileUpload.getStore('Uploads').deleteById(document.file._id);
if (document.files) {
await Promise.all(document.files.map((file) => FileUpload.getStore('Uploads').deleteById(file._id)));
}
}
},
Expand Down
9 changes: 9 additions & 0 deletions apps/meteor/tests/mocks/data.ts
Original file line number Diff line number Diff line change
Expand Up @@ -253,6 +253,15 @@ export function createFakeMessageWithAttachment(overrides?: Partial<IMessage>):
title_link: `/file-upload/${fileId}/${fileName}`,
},
],
files: [
{
_id: fileId,
name: fileName,
type: 'text/plain',
size: faker.number.int(),
format: faker.string.alpha(),
},
],
...overrides,
};
}
Expand Down

0 comments on commit 7619988

Please sign in to comment.