mirror of
https://github.com/CommunitySolidServer/CommunitySolidServer.git
synced 2024-10-03 14:55:10 +00:00
test: Divide current tests into integration and unit tests for LockingResourceStore
This commit is contained in:
committed by
Joachim Van Herwegen
parent
dacfb74a6a
commit
f4afc3ce16
104
test/integration/LockingResourceStore.test.ts
Normal file
104
test/integration/LockingResourceStore.test.ts
Normal file
@@ -0,0 +1,104 @@
|
||||
import streamifyArray from 'streamify-array';
|
||||
import { RepresentationMetadata } from '../../src/ldp/representation/RepresentationMetadata';
|
||||
import { InMemoryDataAccessor } from '../../src/storage/accessors/InMemoryDataAccessor';
|
||||
import { DataAccessorBasedStore } from '../../src/storage/DataAccessorBasedStore';
|
||||
import { LockingResourceStore } from '../../src/storage/LockingResourceStore';
|
||||
import type { ResourceStore } from '../../src/storage/ResourceStore';
|
||||
import { UrlContainerManager } from '../../src/storage/UrlContainerManager';
|
||||
import { APPLICATION_OCTET_STREAM } from '../../src/util/ContentTypes';
|
||||
import type { ExpiringResourceLocker } from '../../src/util/locking/ExpiringResourceLocker';
|
||||
import type { ResourceLocker } from '../../src/util/locking/ResourceLocker';
|
||||
import { SingleThreadedResourceLocker } from '../../src/util/locking/SingleThreadedResourceLocker';
|
||||
import { WrappedExpiringResourceLocker } from '../../src/util/locking/WrappedExpiringResourceLocker';
|
||||
import { MetadataController } from '../../src/util/MetadataController';
|
||||
import { CONTENT_TYPE } from '../../src/util/UriConstants';
|
||||
|
||||
describe('A LockingResourceStore', (): void => {
|
||||
let path: string;
|
||||
let store: LockingResourceStore;
|
||||
let locker: ResourceLocker;
|
||||
let expiringLocker: ExpiringResourceLocker;
|
||||
let source: ResourceStore;
|
||||
|
||||
beforeEach(async(): Promise<void> => {
|
||||
jest.clearAllMocks();
|
||||
|
||||
const base = 'http://test.com/';
|
||||
path = `${base}path`;
|
||||
const metadataController = new MetadataController();
|
||||
const containerManager = new UrlContainerManager(base);
|
||||
source = new DataAccessorBasedStore(
|
||||
new InMemoryDataAccessor(base, metadataController), base, metadataController, containerManager,
|
||||
);
|
||||
|
||||
locker = new SingleThreadedResourceLocker();
|
||||
expiringLocker = new WrappedExpiringResourceLocker(locker, 1000);
|
||||
|
||||
store = new LockingResourceStore(source, expiringLocker);
|
||||
|
||||
// Make sure something is in the store before we read from it in our tests.
|
||||
const metadata = new RepresentationMetadata({ [CONTENT_TYPE]: APPLICATION_OCTET_STREAM });
|
||||
const data = streamifyArray([ 1, 2, 3 ]);
|
||||
await store.setRepresentation({ path }, { metadata, data, binary: true });
|
||||
});
|
||||
|
||||
it('destroys the stream when nothing is read after 1000ms.', async(): Promise<void> => {
|
||||
jest.useFakeTimers();
|
||||
|
||||
// Spy on a real ResourceLocker and ResourceStore instance
|
||||
const acquireSpy = jest.spyOn(expiringLocker, 'acquire');
|
||||
const getRepresentationSpy = jest.spyOn(source, 'getRepresentation');
|
||||
|
||||
const representation = await store.getRepresentation({ path }, {});
|
||||
const errorCallback = jest.fn();
|
||||
representation.data.on('error', errorCallback);
|
||||
|
||||
// Wait 1000ms and read
|
||||
jest.advanceTimersByTime(1000);
|
||||
expect(representation.data.read()).toBeNull();
|
||||
|
||||
// Verify a timeout error was thrown
|
||||
await new Promise((resolve): any => setImmediate(resolve));
|
||||
expect(errorCallback).toHaveBeenCalledTimes(1);
|
||||
expect(errorCallback).toHaveBeenLastCalledWith(new Error('Stream reading timout exceeded'));
|
||||
|
||||
// Verify the lock was acquired and released at the right time
|
||||
expect(acquireSpy).toHaveBeenCalledTimes(1);
|
||||
expect(acquireSpy).toHaveBeenLastCalledWith({ path });
|
||||
expect(getRepresentationSpy).toHaveBeenCalledTimes(1);
|
||||
});
|
||||
|
||||
it('destroys the stream when pauses between reads exceed 1000ms.', async(): Promise<void> => {
|
||||
jest.useFakeTimers();
|
||||
|
||||
// Spy on a real ResourceLocker and ResourceStore instance
|
||||
const acquireSpy = jest.spyOn(expiringLocker, 'acquire');
|
||||
const getRepresentationSpy = jest.spyOn(source, 'getRepresentation');
|
||||
|
||||
const representation = await store.getRepresentation({ path }, {});
|
||||
const errorCallback = jest.fn();
|
||||
representation.data.on('error', errorCallback);
|
||||
|
||||
// Wait 750ms and read
|
||||
jest.advanceTimersByTime(750);
|
||||
expect(representation.data.read()).toBe(1);
|
||||
|
||||
// Wait 750ms and read
|
||||
jest.advanceTimersByTime(750);
|
||||
expect(representation.data.read()).toBe(2);
|
||||
|
||||
// Wait 1000ms and watch the stream be destroyed
|
||||
jest.advanceTimersByTime(1000);
|
||||
expect(representation.data.read()).toBeNull();
|
||||
|
||||
// Verify a timeout error was thrown
|
||||
await new Promise((resolve): any => setImmediate(resolve));
|
||||
expect(errorCallback).toHaveBeenCalledTimes(1);
|
||||
expect(errorCallback).toHaveBeenLastCalledWith(new Error('Stream reading timout exceeded'));
|
||||
|
||||
// Verify the lock was acquired and released at the right time
|
||||
expect(acquireSpy).toHaveBeenCalledTimes(1);
|
||||
expect(acquireSpy).toHaveBeenLastCalledWith({ path });
|
||||
expect(getRepresentationSpy).toHaveBeenCalledTimes(1);
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user