feat: add simple request parser

This commit is contained in:
Joachim Van Herwegen 2020-06-10 11:47:43 +02:00
parent 09eb665c12
commit cf258d0317
16 changed files with 166 additions and 27 deletions

View File

@ -7,5 +7,3 @@ coverage
**/*.js
**/*.d.ts
**/*.js.map
!external-types/*.d.ts

1
.gitignore vendored
View File

@ -9,4 +9,3 @@ coverage
!.eslintrc.js
!test/eslintrc.js
!jest.config.js
!external-types/*.d.ts

View File

@ -1,6 +0,0 @@
declare module 'arrayify-stream' {
import { Readable } from 'stream';
function arrayifyStream(input: Readable): Promise<any[]>;
export = arrayifyStream;
}

View File

@ -1,6 +0,0 @@
declare module 'streamify-array' {
import { Readable } from 'stream';
function streamifyArray(input: any[]): Readable;
export = streamifyArray;
}

18
package-lock.json generated
View File

@ -699,6 +699,15 @@
"@sinonjs/commons": "^1.7.0"
}
},
"@types/arrayify-stream": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/@types/arrayify-stream/-/arrayify-stream-1.0.0.tgz",
"integrity": "sha512-rTx+OXoOJxoscixxecOqaEABaN6Qh/BNImDBnytLVrDI+glba4LJ7RS9JGRcc7auLtdFcM4s8+fdN6pb/K24OA==",
"dev": true,
"requires": {
"@types/node": "*"
}
},
"@types/babel__core": {
"version": "7.1.7",
"resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.1.7.tgz",
@ -908,6 +917,15 @@
"integrity": "sha512-l42BggppR6zLmpfU6fq9HEa2oGPEI8yrSPL3GITjfRInppYFahObbIQOQK3UGxEnyQpltZLaPe75046NOZQikw==",
"dev": true
},
"@types/streamify-array": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/@types/streamify-array/-/streamify-array-1.0.0.tgz",
"integrity": "sha512-qBRnXKNEF8ejRM7TODp3bXIFnHjDfrUM3cTpCU8hnkrI5FHH708wGTo4jc/2VnyNDd73sNYtt3un2pT+9E1y1A==",
"dev": true,
"requires": {
"@types/node": "*"
}
},
"@types/yargs": {
"version": "15.0.4",
"resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-15.0.4.tgz",

View File

@ -28,7 +28,9 @@
"n3": "^1.3.7"
},
"devDependencies": {
"@types/arrayify-stream": "^1.0.0",
"@types/jest": "^25.2.1",
"@types/streamify-array": "^1.0.0",
"@typescript-eslint/eslint-plugin": "^2.33.0",
"@typescript-eslint/parser": "^2.33.0",
"arrayify-stream": "^1.0.0",

View File

@ -6,7 +6,6 @@ import { RepresentationMetadata } from '../representation/RepresentationMetadata
import { StreamParser } from 'n3';
import { TypedReadable } from '../../util/TypedReadable';
import { UnsupportedMediaTypeHttpError } from '../../util/errors/UnsupportedMediaTypeHttpError';
import 'jest-rdf';
export class SimpleBodyParser extends BodyParser {
private static readonly contentTypes = [
@ -32,12 +31,12 @@ export class SimpleBodyParser extends BodyParser {
return undefined;
}
const specificType = contentType.split(';')[0];
const mediaType = contentType.split(';')[0];
const metadata: RepresentationMetadata = {
raw: [],
profiles: [],
contentType: specificType,
contentType: mediaType,
};
// StreamParser is a Readable but typings are incorrect at time of writing

View File

@ -20,7 +20,7 @@ export class SimplePreferenceParser extends PreferenceParser {
// Datetime can have commas so requires separate rules
let datetime;
if (input.headers['accept-datetime']) {
datetime = [{ value: input.headers['accept-datetime'] as string }];
datetime = [{ value: input.headers['accept-datetime'] as string, weight: 1 }];
}
return { type, charset, datetime, language };
@ -34,7 +34,7 @@ export class SimplePreferenceParser extends PreferenceParser {
return header.split(',').map((preference): RepresentationPreference => {
const parts = preference.split(';');
if (parts.length === 1) {
return { value: parts[0].trim() };
return { value: parts[0].trim(), weight: 1 };
}
return { value: parts[0].trim(), weight: parseFloat(parts[1].trim().slice('q='.length)) };
});

View File

@ -0,0 +1,43 @@
import { BodyParser } from './BodyParser';
import { HttpRequest } from '../../server/HttpRequest';
import { Operation } from '../operations/Operation';
import { PreferenceParser } from './PreferenceParser';
import { RequestParser } from './RequestParser';
import { TargetExtractor } from './TargetExtractor';
/**
* Input parsers required for a {@link SimpleRequestParser}.
*/
export interface SimpleRequestParserArgs {
targetExtractor: TargetExtractor;
preferenceParser: PreferenceParser;
bodyParser: BodyParser;
}
export class SimpleRequestParser extends RequestParser {
private readonly targetExtractor: TargetExtractor;
private readonly preferenceParser: PreferenceParser;
private readonly bodyParser: BodyParser;
public constructor(args: SimpleRequestParserArgs) {
super();
Object.assign(this, args);
}
public async canHandle(input: HttpRequest): Promise<void> {
if (!input.url) {
throw new Error('Missing URL.');
}
if (!input.method) {
throw new Error('Missing method.');
}
}
public async handle(input: HttpRequest): Promise<Operation> {
const target = await this.targetExtractor.handleSafe(input);
const preferences = await this.preferenceParser.handleSafe(input);
const body = await this.bodyParser.handleSafe(input);
return { method: input.method, target, preferences, body };
}
}

View File

@ -9,5 +9,5 @@ export interface RepresentationPreference {
/**
* How important this preference is in a value going from 0 to 1.
*/
weight?: number;
weight: number;
}

View File

@ -3,7 +3,7 @@
*/
export interface ResourceIdentifier {
/**
* Path to the relevant resource. Usually this would be an URL.
* Path to the relevant resource.
*/
path: string;
}

View File

@ -0,0 +1,52 @@
import arrayifyStream from 'arrayify-stream';
import { HttpRequest } from '../../src/server/HttpRequest';
import { SimpleBodyParser } from '../../src/ldp/http/SimpleBodyParser';
import { SimplePreferenceParser } from '../../src/ldp/http/SimplePreferenceParser';
import { SimpleRequestParser } from '../../src/ldp/http/SimpleRequestParser';
import { SimpleTargetExtractor } from '../../src/ldp/http/SimpleTargetExtractor';
import streamifyArray from 'streamify-array';
import { StreamParser } from 'n3';
import { namedNode, triple } from '@rdfjs/data-model';
describe('A SimpleRequestParser with simple input parsers', (): void => {
const targetExtractor = new SimpleTargetExtractor();
const bodyParser = new SimpleBodyParser();
const preferenceParser = new SimplePreferenceParser();
const requestParser = new SimpleRequestParser({ targetExtractor, bodyParser, preferenceParser });
it('can parse an incoming request.', async(): Promise<void> => {
const request = streamifyArray([ '<http://test.com/s> <http://test.com/p> <http://test.com/o>.' ]) as HttpRequest;
request.method = 'POST';
request.url = 'http://test.com/';
request.headers = {
accept: 'text/turtle; q=0.8',
'accept-language': 'en-gb, en;q=0.5',
'content-type': 'text/turtle',
};
const result = await requestParser.handle(request);
expect(result).toEqual({
method: 'POST',
target: { path: 'http://test.com/' },
preferences: {
type: [{ value: 'text/turtle', weight: 0.8 }],
language: [{ value: 'en-gb', weight: 1 }, { value: 'en', weight: 0.5 }],
},
body: {
data: expect.any(StreamParser),
dataType: 'quad',
metadata: {
contentType: 'text/turtle',
profiles: [],
raw: [],
},
},
});
await expect(arrayifyStream(result.body.data)).resolves.toEqualRdfQuadArray([ triple(
namedNode('http://test.com/s'),
namedNode('http://test.com/p'),
namedNode('http://test.com/o'),
) ]);
});
});

View File

@ -5,6 +5,7 @@ import streamifyArray from 'streamify-array';
import { StreamParser } from 'n3';
import { UnsupportedMediaTypeHttpError } from '../../../../src/util/errors/UnsupportedMediaTypeHttpError';
import { namedNode, triple } from '@rdfjs/data-model';
import 'jest-rdf';
const contentTypes = [
'application/n-quads',

View File

@ -14,21 +14,21 @@ describe('A SimplePreferenceParser', (): void => {
it('parses accept headers.', async(): Promise<void> => {
await expect(preferenceParser.handle({ headers: { accept: 'audio/*; q=0.2, audio/basic' }} as HttpRequest))
.resolves.toEqual({ type: [{ value: 'audio/*', weight: 0.2 }, { value: 'audio/basic' }]});
.resolves.toEqual({ type: [{ value: 'audio/*', weight: 0.2 }, { value: 'audio/basic', weight: 1 }]});
});
it('parses accept-charset headers.', async(): Promise<void> => {
await expect(preferenceParser.handle({ headers: { 'accept-charset': 'iso-8859-5, unicode-1-1;q=0.8' }} as unknown as HttpRequest))
.resolves.toEqual({ charset: [{ value: 'iso-8859-5' }, { value: 'unicode-1-1', weight: 0.8 }]});
.resolves.toEqual({ charset: [{ value: 'iso-8859-5', weight: 1 }, { value: 'unicode-1-1', weight: 0.8 }]});
});
it('parses accept-datetime headers.', async(): Promise<void> => {
await expect(preferenceParser.handle({ headers: { 'accept-datetime': 'Tue, 20 Mar 2001 20:35:00 GMT' }} as unknown as HttpRequest))
.resolves.toEqual({ datetime: [{ value: 'Tue, 20 Mar 2001 20:35:00 GMT' }]});
.resolves.toEqual({ datetime: [{ value: 'Tue, 20 Mar 2001 20:35:00 GMT', weight: 1 }]});
});
it('parses accept-language headers.', async(): Promise<void> => {
await expect(preferenceParser.handle({ headers: { 'accept-language': 'da, en-gb;q=0.8, en;q=0.7' }} as HttpRequest))
.resolves.toEqual({ language: [{ value: 'da' }, { value: 'en-gb', weight: 0.8 }, { value: 'en', weight: 0.7 }]});
.resolves.toEqual({ language: [{ value: 'da', weight: 1 }, { value: 'en-gb', weight: 0.8 }, { value: 'en', weight: 0.7 }]});
});
});

View File

@ -0,0 +1,40 @@
import { BodyParser } from '../../../../src/ldp/http/BodyParser';
import { PreferenceParser } from '../../../../src/ldp/http/PreferenceParser';
import { SimpleRequestParser } from '../../../../src/ldp/http/SimpleRequestParser';
import { StaticAsyncHandler } from '../../../util/StaticAsyncHandler';
import { TargetExtractor } from '../../../../src/ldp/http/TargetExtractor';
describe('A SimpleRequestParser', (): void => {
let targetExtractor: TargetExtractor;
let bodyParser: BodyParser;
let preferenceParser: PreferenceParser;
let requestParser: SimpleRequestParser;
beforeEach(async(): Promise<void> => {
targetExtractor = new StaticAsyncHandler(true, 'target' as any);
bodyParser = new StaticAsyncHandler(true, 'body' as any);
preferenceParser = new StaticAsyncHandler(true, 'preference' as any);
requestParser = new SimpleRequestParser({ targetExtractor, bodyParser, preferenceParser });
});
it('can handle input with both a URL and a method.', async(): Promise<void> => {
await expect(requestParser.canHandle({ url: 'url', method: 'GET' } as any)).resolves.toBeUndefined();
});
it('rejects input with no URL.', async(): Promise<void> => {
await expect(requestParser.canHandle({ method: 'GET' } as any)).rejects.toThrow('Missing URL.');
});
it('rejects input with no method.', async(): Promise<void> => {
await expect(requestParser.canHandle({ url: 'url' } as any)).rejects.toThrow('Missing method.');
});
it('returns the output of all input parsers after calling handle.', async(): Promise<void> => {
await expect(requestParser.handle({ url: 'url', method: 'GET' } as any)).resolves.toEqual({
method: 'GET',
target: 'target',
preferences: 'preference',
body: 'body',
});
});
});

View File

@ -15,7 +15,6 @@
"stripInternal": true
},
"include": [
"external-types/**/*.ts",
"src/**/*.ts",
"test/**/*.ts"
],