2021-06-18 17:17:25 +01:00
|
|
|
import { FileApi } from '../../file-api';
|
|
|
|
import BaseItem from '../../models/BaseItem';
|
|
|
|
import Note from '../../models/Note';
|
|
|
|
import { expectNotThrow, expectThrow, setupDatabaseAndSynchronizer, switchClient } from '../../testing/test-utils';
|
|
|
|
import time from '../../time';
|
|
|
|
import ItemUploader, { ApiCallFunction } from './ItemUploader';
|
|
|
|
|
|
|
|
interface ApiCall {
|
|
|
|
name: string;
|
|
|
|
args: any[];
|
|
|
|
}
|
|
|
|
|
|
|
|
function clearArray(a: any[]) {
|
|
|
|
a.splice(0, a.length);
|
|
|
|
}
|
|
|
|
|
|
|
|
function newFakeApi(): FileApi {
|
|
|
|
return { supportsMultiPut: true } as any;
|
|
|
|
}
|
|
|
|
|
|
|
|
function newFakeApiCall(callRecorder: ApiCall[], itemBodyCallback: Function = null): ApiCallFunction {
|
|
|
|
const apiCall = async (callName: string, ...args: any[]): Promise<any> => {
|
|
|
|
callRecorder.push({ name: callName, args });
|
|
|
|
|
|
|
|
if (callName === 'multiPut') {
|
|
|
|
const [batch] = args;
|
|
|
|
const output: any = { items: {} };
|
|
|
|
for (const item of batch) {
|
|
|
|
if (itemBodyCallback) {
|
|
|
|
output.items[item.name] = itemBodyCallback(item);
|
|
|
|
} else {
|
|
|
|
output.items[item.name] = {
|
|
|
|
item: item.body,
|
|
|
|
error: null,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
return apiCall;
|
|
|
|
}
|
|
|
|
|
2021-08-12 16:54:10 +01:00
|
|
|
describe('synchronizer/ItemUploader', function() {
|
2021-06-18 17:17:25 +01:00
|
|
|
|
2022-11-15 10:23:50 +00:00
|
|
|
beforeEach(async () => {
|
2021-06-18 17:17:25 +01:00
|
|
|
await setupDatabaseAndSynchronizer(1);
|
|
|
|
await setupDatabaseAndSynchronizer(2);
|
|
|
|
await switchClient(1);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should batch uploads and use the cache afterwards', (async () => {
|
|
|
|
const callRecorder: ApiCall[] = [];
|
|
|
|
const itemUploader = new ItemUploader(newFakeApi(), newFakeApiCall(callRecorder));
|
|
|
|
|
|
|
|
const notes = [
|
|
|
|
await Note.save({ title: '1' }),
|
|
|
|
await Note.save({ title: '2' }),
|
|
|
|
];
|
|
|
|
|
|
|
|
await itemUploader.preUploadItems(notes);
|
|
|
|
|
|
|
|
// There should be only one call to "multiPut" because the items have
|
|
|
|
// been batched.
|
|
|
|
expect(callRecorder.length).toBe(1);
|
|
|
|
expect(callRecorder[0].name).toBe('multiPut');
|
|
|
|
|
|
|
|
clearArray(callRecorder);
|
|
|
|
|
|
|
|
// Now if we try to upload the item it shouldn't call the API because it
|
|
|
|
// will use the cached item.
|
|
|
|
await itemUploader.serializeAndUploadItem(Note, BaseItem.systemPath(notes[0]), notes[0]);
|
|
|
|
expect(callRecorder.length).toBe(0);
|
|
|
|
|
|
|
|
// Now try to process a note that hasn't been cached. In that case, it
|
|
|
|
// should call "PUT" directly.
|
|
|
|
const note3 = await Note.save({ title: '3' });
|
|
|
|
await itemUploader.serializeAndUploadItem(Note, BaseItem.systemPath(note3), note3);
|
|
|
|
expect(callRecorder.length).toBe(1);
|
|
|
|
expect(callRecorder[0].name).toBe('put');
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should not batch upload if the items are over the batch size limit', (async () => {
|
|
|
|
const callRecorder: ApiCall[] = [];
|
|
|
|
const itemUploader = new ItemUploader(newFakeApi(), newFakeApiCall(callRecorder));
|
|
|
|
itemUploader.maxBatchSize = 1;
|
|
|
|
|
|
|
|
const notes = [
|
|
|
|
await Note.save({ title: '1' }),
|
|
|
|
await Note.save({ title: '2' }),
|
|
|
|
];
|
|
|
|
|
|
|
|
await itemUploader.preUploadItems(notes);
|
|
|
|
expect(callRecorder.length).toBe(0);
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should not use the cache if the note has changed since the pre-upload', (async () => {
|
|
|
|
const callRecorder: ApiCall[] = [];
|
|
|
|
const itemUploader = new ItemUploader(newFakeApi(), newFakeApiCall(callRecorder));
|
|
|
|
|
|
|
|
const notes = [
|
|
|
|
await Note.save({ title: '1' }),
|
|
|
|
await Note.save({ title: '2' }),
|
|
|
|
];
|
|
|
|
|
|
|
|
await itemUploader.preUploadItems(notes);
|
|
|
|
clearArray(callRecorder);
|
|
|
|
|
|
|
|
await itemUploader.serializeAndUploadItem(Note, BaseItem.systemPath(notes[0]), notes[0]);
|
|
|
|
expect(callRecorder.length).toBe(0);
|
|
|
|
|
|
|
|
await time.msleep(1);
|
|
|
|
notes[1] = await Note.save({ title: '22' }),
|
|
|
|
await itemUploader.serializeAndUploadItem(Note, BaseItem.systemPath(notes[1]), notes[1]);
|
|
|
|
expect(callRecorder.length).toBe(1);
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should respect the max batch size', (async () => {
|
|
|
|
const callRecorder: ApiCall[] = [];
|
|
|
|
const itemUploader = new ItemUploader(newFakeApi(), newFakeApiCall(callRecorder));
|
|
|
|
|
|
|
|
const notes = [
|
|
|
|
await Note.save({ title: '1' }),
|
|
|
|
await Note.save({ title: '2' }),
|
|
|
|
await Note.save({ title: '3' }),
|
|
|
|
];
|
|
|
|
|
|
|
|
const noteSize = BaseItem.systemPath(notes[0]).length + (await Note.serializeForSync(notes[0])).length;
|
|
|
|
itemUploader.maxBatchSize = noteSize * 2;
|
|
|
|
|
|
|
|
// It should send two batches - one with two notes, and the second with
|
|
|
|
// only one note.
|
|
|
|
await itemUploader.preUploadItems(notes);
|
|
|
|
expect(callRecorder.length).toBe(2);
|
|
|
|
expect(callRecorder[0].args[0].length).toBe(2);
|
|
|
|
expect(callRecorder[1].args[0].length).toBe(1);
|
|
|
|
}));
|
|
|
|
|
|
|
|
it('should rethrow error for items within the batch', (async () => {
|
|
|
|
const callRecorder: ApiCall[] = [];
|
|
|
|
|
|
|
|
const notes = [
|
|
|
|
await Note.save({ title: '1' }),
|
|
|
|
await Note.save({ title: '2' }),
|
|
|
|
await Note.save({ title: '3' }),
|
|
|
|
];
|
|
|
|
|
|
|
|
// Simulates throwing an error on note 2
|
|
|
|
const itemBodyCallback = (item: any): any => {
|
|
|
|
if (item.name === BaseItem.systemPath(notes[1])) {
|
|
|
|
return { error: new Error('Could not save item'), item: null };
|
|
|
|
} else {
|
|
|
|
return { error: null, item: item.body };
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const itemUploader = new ItemUploader(newFakeApi(), newFakeApiCall(callRecorder, itemBodyCallback));
|
|
|
|
|
|
|
|
await itemUploader.preUploadItems(notes);
|
|
|
|
|
|
|
|
await expectNotThrow(async () => itemUploader.serializeAndUploadItem(Note, BaseItem.systemPath(notes[0]), notes[0]));
|
|
|
|
await expectThrow(async () => itemUploader.serializeAndUploadItem(Note, BaseItem.systemPath(notes[1]), notes[1]));
|
|
|
|
await expectNotThrow(async () => itemUploader.serializeAndUploadItem(Note, BaseItem.systemPath(notes[2]), notes[2]));
|
|
|
|
}));
|
|
|
|
|
|
|
|
});
|