CommunitySolidServer/test/util/TestHelpers.ts
2021-01-11 14:23:52 +01:00

198 lines
6.2 KiB
TypeScript

import { EventEmitter } from 'events';
import { promises as fs } from 'fs';
import type { IncomingHttpHeaders } from 'http';
import { Readable } from 'stream';
import * as url from 'url';
import type { MockResponse } from 'node-mocks-http';
import { createResponse } from 'node-mocks-http';
import type { ResourceStore, PermissionSet, HttpHandler, HttpRequest } from '../../src/';
import { BasicRepresentation, joinFilePath, ensureTrailingSlash } from '../../src/';
import { performRequest } from './Util';
/* eslint-disable jest/no-standalone-expect */
export class AclHelper {
public readonly store: ResourceStore;
public id: string;
public constructor(store: ResourceStore, id: string) {
this.store = store;
this.id = ensureTrailingSlash(id);
}
public async setSimpleAcl(
permissions: PermissionSet,
agentClass: 'agent' | 'authenticated',
): Promise<void> {
const acl: string[] = [
'@prefix acl: <http://www.w3.org/ns/auth/acl#>.\n',
'@prefix foaf: <http://xmlns.com/foaf/0.1/>.\n',
'<http://test.com/#auth> a acl:Authorization',
];
for (const perm of [ 'Read', 'Append', 'Write', 'Delete' ]) {
if (permissions[perm.toLowerCase() as keyof PermissionSet]) {
acl.push(`;\n acl:mode acl:${perm}`);
}
}
acl.push(';\n acl:mode acl:Control');
acl.push(`;\n acl:accessTo <${this.id}>`);
acl.push(`;\n acl:default <${this.id}>`);
acl.push(
`;\n acl:agentClass ${
agentClass === 'agent' ? 'foaf:Agent' : 'foaf:AuthenticatedAgent'
}`,
);
acl.push('.');
await this.store.setRepresentation({ path: `${this.id}.acl` }, new BasicRepresentation(acl, 'text/turtle'));
}
}
export class ResourceHelper {
public readonly handler: HttpHandler;
public readonly baseUrl: URL;
public constructor(handler: HttpHandler, baseUrl: string) {
this.handler = handler;
this.baseUrl = new URL(baseUrl);
}
public async performRequest(
requestUrl: URL,
method: string,
headers: IncomingHttpHeaders,
): Promise<MockResponse<any>> {
return performRequest(this.handler, requestUrl, method, headers, []);
}
public async performRequestWithBody(
requestUrl: URL,
method: string,
headers: IncomingHttpHeaders,
data: Buffer,
): Promise<MockResponse<any>> {
const request = Readable.from([ data ]) as HttpRequest;
request.url = requestUrl.pathname;
request.method = method;
request.headers = headers;
request.headers.host = requestUrl.host;
const response: MockResponse<any> = createResponse({
eventEmitter: EventEmitter,
});
const endPromise = new Promise((resolve): void => {
response.on('end', (): void => {
expect(response._isEndCalled()).toBeTruthy();
resolve();
});
});
await this.handler.handleSafe({ request, response });
await endPromise;
return response;
}
public async createResource(fileLocation: string, slug: string, contentType: string, mayFail = false):
Promise<MockResponse<any>> {
const fileData = await fs.readFile(
joinFilePath(__dirname, fileLocation),
);
const response: MockResponse<any> = await this.performRequestWithBody(
this.baseUrl,
'POST',
{ 'content-type': contentType,
slug,
'transfer-encoding': 'chunked' },
fileData,
);
if (!mayFail) {
expect(response.statusCode).toBe(201);
expect(response._getData()).toHaveLength(0);
expect(response._getHeaders().location).toContain(url.format(this.baseUrl));
}
return response;
}
public async replaceResource(fileLocation: string, requestUrl: string, contentType: string):
Promise<MockResponse<any>> {
const fileData = await fs.readFile(
joinFilePath(__dirname, fileLocation),
);
const putUrl = new URL(requestUrl);
const response: MockResponse<any> = await this.performRequestWithBody(
putUrl,
'PUT',
{ 'content-type': contentType, 'transfer-encoding': 'chunked' },
fileData,
);
expect(response.statusCode).toBe(205);
expect(response._getData()).toHaveLength(0);
return response;
}
public async getResource(requestUrl: string): Promise<MockResponse<any>> {
const getUrl = new URL(requestUrl);
const response = await this.performRequest(getUrl, 'GET', { accept: '*/*' });
expect(response.statusCode).toBe(200);
return response;
}
public async deleteResource(requestUrl: string, mayFail = false): Promise<MockResponse<any>> {
const deleteUrl = new URL(requestUrl);
const response = await this.performRequest(deleteUrl, 'DELETE', {});
if (!mayFail) {
expect(response.statusCode).toBe(205);
expect(response._getData()).toHaveLength(0);
}
return response;
}
public async createContainer(slug: string): Promise<MockResponse<any>> {
const response: MockResponse<any> = await this.performRequest(
this.baseUrl,
'POST',
{
slug,
link: '<http://www.w3.org/ns/ldp#Container>; rel="type"',
'content-type': 'text/turtle',
'transfer-encoding': 'chunked',
},
);
expect(response.statusCode).toBe(201);
expect(response._getData()).toHaveLength(0);
expect(response._getHeaders().location).toContain(url.format(this.baseUrl));
return response;
}
public async getContainer(requestUrl: string): Promise<MockResponse<any>> {
const getUrl = new URL(requestUrl);
// `n-quads` allow for easy testing if a triple is present
return await this.performRequest(getUrl, 'GET', { accept: 'application/n-quads' });
}
public async shouldNotExist(requestUrl: string): Promise<MockResponse<any>> {
const getUrl = new URL(requestUrl);
const response = await this.performRequest(getUrl, 'GET', { accept: '*/*' });
expect(response.statusCode).toBe(404);
expect(response._getData()).toContain('NotFoundHttpError');
return response;
}
}
export function describeIf(envFlag: string, name: string, fn: () => void): void {
const flag = `TEST_${envFlag.toUpperCase()}`;
const enabled = !/^(|0|false)$/iu.test(process.env[flag] ?? '');
// eslint-disable-next-line jest/valid-describe, jest/valid-title, jest/no-disabled-tests
return enabled ? describe(name, fn) : describe.skip(name, fn);
}