2023-07-16 17:42:42 +01:00
|
|
|
import { supportDir, setupDatabaseAndSynchronizer, switchClient, simulateReadOnlyShareEnv, expectThrow, createTempFile } from '../testing/test-utils';
|
2022-02-09 18:04:27 +00:00
|
|
|
import Folder from '../models/Folder';
|
|
|
|
import Note from '../models/Note';
|
|
|
|
import Resource from '../models/Resource';
|
|
|
|
import shim from '../shim';
|
2023-07-16 17:42:42 +01:00
|
|
|
import { ErrorCode } from '../errors';
|
|
|
|
import { remove, pathExists } from 'fs-extra';
|
2019-07-30 09:35:42 +02:00
|
|
|
|
2021-05-21 15:17:21 +02:00
|
|
|
const testImagePath = `${supportDir}/photo.jpg`;
|
2019-05-12 11:38:33 +01:00
|
|
|
|
2023-07-16 17:42:42 +01:00
|
|
|
const setupFolderNoteResourceReadOnly = async (shareId: string) => {
|
|
|
|
const cleanup = simulateReadOnlyShareEnv(shareId);
|
|
|
|
|
|
|
|
let folder = await Folder.save({ });
|
|
|
|
let note = await Note.save({ parent_id: folder.id });
|
|
|
|
await shim.attachFileToNote(note, testImagePath);
|
|
|
|
let resource = (await Resource.all())[0];
|
|
|
|
|
|
|
|
folder = await Folder.save({ id: folder.id, share_id: shareId });
|
|
|
|
note = await Note.save({ id: note.id, share_id: shareId });
|
|
|
|
resource = await Resource.save({ id: resource.id, share_id: shareId });
|
|
|
|
|
|
|
|
resource = await Resource.load(resource.id); // reload to get all properties
|
|
|
|
|
|
|
|
return { cleanup, folder, note, resource };
|
|
|
|
};
|
|
|
|
|
2023-02-20 12:02:29 -03:00
|
|
|
describe('models/Resource', () => {
|
2018-10-11 17:18:24 +01:00
|
|
|
|
2022-11-15 10:23:50 +00:00
|
|
|
beforeEach(async () => {
|
2018-10-11 17:18:24 +01:00
|
|
|
await setupDatabaseAndSynchronizer(1);
|
|
|
|
await switchClient(1);
|
|
|
|
});
|
|
|
|
|
2020-12-01 18:05:24 +00:00
|
|
|
it('should have a "done" fetch_status when created locally', (async () => {
|
2020-03-13 23:46:14 +00:00
|
|
|
const folder1 = await Folder.save({ title: 'folder1' });
|
|
|
|
const note1 = await Note.save({ title: 'ma note', parent_id: folder1.id });
|
2019-05-12 11:38:33 +01:00
|
|
|
await shim.attachFileToNote(note1, testImagePath);
|
2020-03-13 23:46:14 +00:00
|
|
|
const resource1 = (await Resource.all())[0];
|
|
|
|
const ls = await Resource.localState(resource1);
|
2018-11-13 00:45:08 +00:00
|
|
|
expect(ls.fetch_status).toBe(Resource.FETCH_STATUS_DONE);
|
|
|
|
}));
|
|
|
|
|
2020-12-01 18:05:24 +00:00
|
|
|
it('should have a default local state', (async () => {
|
2020-03-13 23:46:14 +00:00
|
|
|
const folder1 = await Folder.save({ title: 'folder1' });
|
|
|
|
const note1 = await Note.save({ title: 'ma note', parent_id: folder1.id });
|
2019-05-12 11:38:33 +01:00
|
|
|
await shim.attachFileToNote(note1, testImagePath);
|
2020-03-13 23:46:14 +00:00
|
|
|
const resource1 = (await Resource.all())[0];
|
|
|
|
const ls = await Resource.localState(resource1);
|
2018-11-13 00:45:08 +00:00
|
|
|
expect(!ls.id).toBe(true);
|
|
|
|
expect(ls.resource_id).toBe(resource1.id);
|
|
|
|
expect(ls.fetch_status).toBe(Resource.FETCH_STATUS_DONE);
|
|
|
|
}));
|
|
|
|
|
2020-12-01 18:05:24 +00:00
|
|
|
it('should save and delete local state', (async () => {
|
2020-03-13 23:46:14 +00:00
|
|
|
const folder1 = await Folder.save({ title: 'folder1' });
|
|
|
|
const note1 = await Note.save({ title: 'ma note', parent_id: folder1.id });
|
2019-05-12 11:38:33 +01:00
|
|
|
await shim.attachFileToNote(note1, testImagePath);
|
2020-03-13 23:46:14 +00:00
|
|
|
const resource1 = (await Resource.all())[0];
|
2018-11-13 00:45:08 +00:00
|
|
|
await Resource.setLocalState(resource1, { fetch_status: Resource.FETCH_STATUS_IDLE });
|
|
|
|
|
|
|
|
let ls = await Resource.localState(resource1);
|
|
|
|
expect(!!ls.id).toBe(true);
|
|
|
|
expect(ls.fetch_status).toBe(Resource.FETCH_STATUS_IDLE);
|
|
|
|
|
|
|
|
await Resource.delete(resource1.id);
|
|
|
|
ls = await Resource.localState(resource1);
|
|
|
|
expect(!ls.id).toBe(true);
|
2018-10-11 17:18:24 +01:00
|
|
|
}));
|
|
|
|
|
2020-12-01 18:05:24 +00:00
|
|
|
it('should resize the resource if the image is below the required dimensions', (async () => {
|
2020-03-13 23:46:14 +00:00
|
|
|
const folder1 = await Folder.save({ title: 'folder1' });
|
|
|
|
const note1 = await Note.save({ title: 'ma note', parent_id: folder1.id });
|
2019-05-12 11:38:33 +01:00
|
|
|
const previousMax = Resource.IMAGE_MAX_DIMENSION;
|
|
|
|
Resource.IMAGE_MAX_DIMENSION = 5;
|
|
|
|
await shim.attachFileToNote(note1, testImagePath);
|
|
|
|
Resource.IMAGE_MAX_DIMENSION = previousMax;
|
2020-03-13 23:46:14 +00:00
|
|
|
const resource1 = (await Resource.all())[0];
|
2019-05-12 11:38:33 +01:00
|
|
|
|
|
|
|
const originalStat = await shim.fsDriver().stat(testImagePath);
|
|
|
|
const newStat = await shim.fsDriver().stat(Resource.fullPath(resource1));
|
|
|
|
|
|
|
|
expect(newStat.size < originalStat.size).toBe(true);
|
|
|
|
}));
|
|
|
|
|
2020-12-01 18:05:24 +00:00
|
|
|
it('should not resize the resource if the image is below the required dimensions', (async () => {
|
2020-03-13 23:46:14 +00:00
|
|
|
const folder1 = await Folder.save({ title: 'folder1' });
|
|
|
|
const note1 = await Note.save({ title: 'ma note', parent_id: folder1.id });
|
2019-05-12 11:38:33 +01:00
|
|
|
await shim.attachFileToNote(note1, testImagePath);
|
2020-03-13 23:46:14 +00:00
|
|
|
const resource1 = (await Resource.all())[0];
|
2019-05-12 11:38:33 +01:00
|
|
|
|
|
|
|
const originalStat = await shim.fsDriver().stat(testImagePath);
|
|
|
|
const newStat = await shim.fsDriver().stat(Resource.fullPath(resource1));
|
|
|
|
|
|
|
|
expect(originalStat.size).toBe(newStat.size);
|
|
|
|
}));
|
|
|
|
|
2023-07-16 17:42:42 +01:00
|
|
|
it('should not allow modifying a read-only resource', async () => {
|
|
|
|
const { cleanup, resource } = await setupFolderNoteResourceReadOnly('123456789');
|
|
|
|
await expectThrow(async () => Resource.save({ id: resource.id, share_id: '123456789', title: 'cannot do this!' }), ErrorCode.IsReadOnly);
|
|
|
|
cleanup();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not allow modifying a read-only resource content', async () => {
|
|
|
|
const { cleanup, resource } = await setupFolderNoteResourceReadOnly('123456789');
|
|
|
|
const tempFilePath = await createTempFile('something');
|
|
|
|
await expectThrow(async () => Resource.updateResourceBlobContent(resource.id, tempFilePath), ErrorCode.IsReadOnly);
|
|
|
|
await remove(tempFilePath);
|
|
|
|
cleanup();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should not allow deleting a read-only resource', async () => {
|
|
|
|
const { cleanup, resource } = await setupFolderNoteResourceReadOnly('123456789');
|
|
|
|
expect(await pathExists(Resource.fullPath(resource))).toBe(true);
|
|
|
|
await expectThrow(async () => Resource.delete(resource.id), ErrorCode.IsReadOnly);
|
|
|
|
// Also check that the resource blob has not been deleted
|
|
|
|
expect(await pathExists(Resource.fullPath(resource))).toBe(true);
|
|
|
|
cleanup();
|
|
|
|
});
|
2022-02-09 18:04:27 +00:00
|
|
|
|
2019-07-30 09:35:42 +02:00
|
|
|
});
|