Skip to content

Commit

Permalink
test: Divide current tests into integration and unit tests for Lockin…
Browse files Browse the repository at this point in the history
…gResourceStore
  • Loading branch information
smessie committed Nov 17, 2020
1 parent 71d4305 commit 68a29cc
Show file tree
Hide file tree
Showing 3 changed files with 164 additions and 56 deletions.
100 changes: 100 additions & 0 deletions test/integration/LockingResourceStore.test.ts
@@ -0,0 +1,100 @@
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 instance
const acquireSpy = jest.spyOn(expiringLocker, 'acquire');

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
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: 'path' });
expect(source.getRepresentation).toHaveBeenCalledTimes(1);
});

it('destroys the stream when pauses between reads exceed 1000ms.', async(): Promise<void> => {
jest.useFakeTimers();

// Spy on a real ResourceLocker instance
const acquireSpy = jest.spyOn(expiringLocker, 'acquire');

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
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: 'path' });
expect(source.getRepresentation).toHaveBeenCalledTimes(1);
});
});
63 changes: 7 additions & 56 deletions test/unit/storage/LockingResourceStore.test.ts
Expand Up @@ -6,7 +6,6 @@ import { LockingResourceStore } from '../../../src/storage/LockingResourceStore'
import type { ResourceStore } from '../../../src/storage/ResourceStore';
import type { ExpiringLock } from '../../../src/util/locking/ExpiringLock';
import type { ExpiringResourceLocker } from '../../../src/util/locking/ExpiringResourceLocker';
import { WrappedExpiringResourceLocker } from '../../../src/util/locking/WrappedExpiringResourceLocker';

describe('A LockingResourceStore', (): void => {
let store: LockingResourceStore;
Expand Down Expand Up @@ -190,65 +189,17 @@ describe('A LockingResourceStore', (): void => {
expect(order).toEqual([ 'acquire', 'getRepresentation', 'end', 'close', 'release' ]);
});

it('destroys the stream when nothing is read after 1000ms.', async(): Promise<void> => {
jest.useFakeTimers();

// Spy on a real ResourceLocker instance
const expiringLocker = new WrappedExpiringResourceLocker(locker, 1000);
store = new LockingResourceStore(source, expiringLocker);
const acquireSpy = jest.spyOn(expiringLocker, 'acquire');

it('releases the lock on the resource when readable times out.', async(): Promise<void> => {
// Make the representation time out
const representation = await store.getRepresentation({ path: 'path' }, {});
const errorCallback = jest.fn();
representation.data.on('error', errorCallback);

// Wait 1000ms and read
jest.advanceTimersByTime(1000);
expect(representation.data.read()).toBeNull();
lock.emit('expired');
await registerEventOrder(representation.data, 'close');

// Verify a timeout error was thrown
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: 'path' });
expect(source.getRepresentation).toHaveBeenCalledTimes(1);
});

it('destroys the stream when pauses between reads exceed 1000ms.', async(): Promise<void> => {
jest.useFakeTimers();

// Spy on a real ResourceLocker instance
const expiringLocker = new WrappedExpiringResourceLocker(locker, 1000);
store = new LockingResourceStore(source, expiringLocker);
const acquireSpy = jest.spyOn(expiringLocker, 'acquire');

const representation = await store.getRepresentation({ path: '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();
await registerEventOrder(representation.data, 'close');

// Verify a timeout error was thrown
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: 'path' });
expect(locker.acquire).toHaveBeenCalledTimes(1);
expect(locker.acquire).toHaveBeenLastCalledWith({ path: 'path' });
expect(source.getRepresentation).toHaveBeenCalledTimes(1);
expect(lock.release).toHaveBeenCalledTimes(1);
expect(order).toEqual([ 'acquire', 'getRepresentation', 'close', 'release' ]);
});
});
57 changes: 57 additions & 0 deletions test/unit/util/locking/WrappedExpiringResourceLocker.test.ts
@@ -1,6 +1,23 @@
import type { EventEmitter } from 'events';
import streamifyArray from 'streamify-array';
import { WrappedExpiringResourceLocker } from '../../../../src/util/locking/WrappedExpiringResourceLocker';

describe('A WrappedExpiringResourceLocker', (): void => {
let order: string[];

beforeEach(async(): Promise<void> => {
order = [];
});

const registerEventOrder = async(eventSource: EventEmitter, event: string): Promise<void> => {
await new Promise((resolve): any => {
eventSource.prependListener(event, (): any => {
order.push(event);
resolve();
});
});
};

it('emits an error event when releasing the lock errors.', async(): Promise<void> => {
jest.useFakeTimers();

Expand All @@ -27,4 +44,44 @@ describe('A WrappedExpiringResourceLocker', (): void => {
expect(errorCallback).toHaveBeenCalledTimes(1);
expect(errorCallback).toHaveBeenLastCalledWith(new Error('Release error'));
});

it('releases the lock on the resource when data has been read.', async(): Promise<void> => {
// Mock the inner ResourceLocker.
const release = jest.fn(async(): Promise<any> => order.push('release'));
const lock = { release };
const locker = {
acquire: jest.fn(async(): Promise<any> => {
order.push('acquire');
return lock;
}),
};

const expiringLocker = new WrappedExpiringResourceLocker(locker, 1000);
const expiringLock = await expiringLocker.acquire({} as any);

// Mimic the behavior of a LockingResourceStore to test the expiringLock methods called.
const source = streamifyArray([ 1, 2, 3 ]);
// eslint-disable-next-line jest/valid-expect-in-promise
new Promise((resolve): void => {
source.on('end', resolve);
source.on('close', resolve);
}).then((): any => expiringLock.release(), null);
const readable = Object.create(source, {
read: {
value(size: number): any {
expiringLock.renew();
return source.read(size);
},
},
});

// Read all data from the "representation"
readable.on('data', (): any => true);
await registerEventOrder(readable, 'end');

// Verify the lock was acquired and released at the right time
expect(locker.acquire).toHaveBeenCalledTimes(1);
expect(lock.release).toHaveBeenCalledTimes(1);
expect(order).toEqual([ 'acquire', 'end', 'release' ]);
});
});

0 comments on commit 68a29cc

Please sign in to comment.