mirror of
https://github.com/CommunitySolidServer/CommunitySolidServer.git
synced 2024-10-03 14:55:10 +00:00
feat: add simple body parser
This commit is contained in:
parent
70af46933b
commit
d4f70d9c59
@ -7,3 +7,5 @@ coverage
|
||||
**/*.js
|
||||
**/*.d.ts
|
||||
**/*.js.map
|
||||
|
||||
!external-types/*.d.ts
|
@ -16,6 +16,7 @@ module.exports = {
|
||||
'@typescript-eslint/space-before-function-paren': [ 'error', 'never' ],
|
||||
'class-methods-use-this': 'off',
|
||||
'comma-dangle': ['error', 'always-multiline'],
|
||||
'dot-location': ['error', 'property'],
|
||||
'lines-between-class-members': ['error', 'always', { exceptAfterSingleLine: true }],
|
||||
'padding-line-between-statements': 'off',
|
||||
'tsdoc/syntax': 'error',
|
||||
|
1
.gitignore
vendored
1
.gitignore
vendored
@ -9,3 +9,4 @@ coverage
|
||||
!.eslintrc.js
|
||||
!test/eslintrc.js
|
||||
!jest.config.js
|
||||
!external-types/*.d.ts
|
||||
|
6
external-types/arrayifyStream.d.ts
vendored
Normal file
6
external-types/arrayifyStream.d.ts
vendored
Normal file
@ -0,0 +1,6 @@
|
||||
declare module 'arrayify-stream' {
|
||||
import { Readable } from 'stream';
|
||||
|
||||
function arrayifyStream(input: Readable): Promise<any[]>;
|
||||
export = arrayifyStream;
|
||||
}
|
6
external-types/streamifyArray.d.ts
vendored
Normal file
6
external-types/streamifyArray.d.ts
vendored
Normal file
@ -0,0 +1,6 @@
|
||||
declare module 'streamify-array' {
|
||||
import { Readable } from 'stream';
|
||||
|
||||
function streamifyArray(input: any[]): Readable;
|
||||
export = streamifyArray;
|
||||
}
|
@ -13,6 +13,7 @@ module.exports = {
|
||||
"js"
|
||||
],
|
||||
"testEnvironment": "node",
|
||||
"setupFilesAfterEnv": ["jest-rdf"],
|
||||
"collectCoverage": true,
|
||||
"coveragePathIgnorePatterns": [
|
||||
"/node_modules/"
|
||||
|
71
package-lock.json
generated
71
package-lock.json
generated
@ -862,6 +862,15 @@
|
||||
"integrity": "sha512-8+KAKzEvSUdeo+kmqnKrqgeE+LcA0tjYWFY7RPProVYwnqDjukzO+3b6dLD56rYX5TdWejnEOLJYOIeh4CXKuA==",
|
||||
"dev": true
|
||||
},
|
||||
"@types/n3": {
|
||||
"version": "1.1.6",
|
||||
"resolved": "https://registry.npmjs.org/@types/n3/-/n3-1.1.6.tgz",
|
||||
"integrity": "sha512-ZUZsoA13IkJOLZCXG9bgZGuzvGqhPQJ3CisPvZJ5Yfi+RxVtDCcezyTRw7olArfmxpD0UuEOtrIP/1PUCfkEBw==",
|
||||
"requires": {
|
||||
"@types/node": "*",
|
||||
"@types/rdf-js": "*"
|
||||
}
|
||||
},
|
||||
"@types/node": {
|
||||
"version": "14.0.1",
|
||||
"resolved": "https://registry.npmjs.org/@types/node/-/node-14.0.1.tgz",
|
||||
@ -1106,6 +1115,12 @@
|
||||
"integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=",
|
||||
"dev": true
|
||||
},
|
||||
"arrayify-stream": {
|
||||
"version": "1.0.0",
|
||||
"resolved": "https://registry.npmjs.org/arrayify-stream/-/arrayify-stream-1.0.0.tgz",
|
||||
"integrity": "sha512-RP80ep76Lbew2wWN5ogrl2NluTnBVYYh2K3NNCcWfcmmUB7nBcNBctiJeEZAixp3I1vQ9H88iHZ9MbHSdkuupQ==",
|
||||
"dev": true
|
||||
},
|
||||
"asn1": {
|
||||
"version": "0.2.4",
|
||||
"resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz",
|
||||
@ -3603,6 +3618,16 @@
|
||||
"integrity": "sha512-pgFw2tm54fzgYvc/OHrnysABEObZCUNFnhjoRjaVOCN8NYc032/gVjPaHD4Aq6ApkSieWtfKAFQtmDKAmhupnQ==",
|
||||
"dev": true
|
||||
},
|
||||
"jest-rdf": {
|
||||
"version": "1.5.0",
|
||||
"resolved": "https://registry.npmjs.org/jest-rdf/-/jest-rdf-1.5.0.tgz",
|
||||
"integrity": "sha512-B7kaGHC/YZ+KHRM0W94FWLaIOAPqsdVEE2Ni3U84aGZnBlF4U+1lmJvuvDABlAm3E9Smh7HdaQgEQJRj29biew==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"rdf-isomorphic": "^1.1.0",
|
||||
"rdf-string": "^1.3.1"
|
||||
}
|
||||
},
|
||||
"jest-regex-util": {
|
||||
"version": "26.0.0",
|
||||
"resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-26.0.0.tgz",
|
||||
@ -3992,6 +4017,12 @@
|
||||
"integrity": "sha1-7dFMgk4sycHgsKG0K7UhBRakJDg=",
|
||||
"dev": true
|
||||
},
|
||||
"lodash.uniqwith": {
|
||||
"version": "4.5.0",
|
||||
"resolved": "https://registry.npmjs.org/lodash.uniqwith/-/lodash.uniqwith-4.5.0.tgz",
|
||||
"integrity": "sha1-egy/ZfQ7WShiWp1NDcVLGMrcfvM=",
|
||||
"dev": true
|
||||
},
|
||||
"lodash.zip": {
|
||||
"version": "4.2.0",
|
||||
"resolved": "https://registry.npmjs.org/lodash.zip/-/lodash.zip-4.2.0.tgz",
|
||||
@ -4146,6 +4177,11 @@
|
||||
"integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==",
|
||||
"dev": true
|
||||
},
|
||||
"n3": {
|
||||
"version": "1.3.7",
|
||||
"resolved": "https://registry.npmjs.org/n3/-/n3-1.3.7.tgz",
|
||||
"integrity": "sha512-IREbOmZyTlc34vxlp31ECT5jliRNZqHg3THhzrVd5bcSWGto5xsN8fF5xWKYXZr8TdZX+GXFkCxttTKM1N3JZg=="
|
||||
},
|
||||
"nanomatch": {
|
||||
"version": "1.2.13",
|
||||
"resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz",
|
||||
@ -4642,6 +4678,35 @@
|
||||
"integrity": "sha512-pVzZdDpWwWqEVVLshWUHjNwuVP7SfcmPraYuqocJp1yo2U1R7P+5QAfDhdItkuoGqIBnBYrtPp7rEPqDn9HlZA==",
|
||||
"dev": true
|
||||
},
|
||||
"rdf-isomorphic": {
|
||||
"version": "1.1.0",
|
||||
"resolved": "https://registry.npmjs.org/rdf-isomorphic/-/rdf-isomorphic-1.1.0.tgz",
|
||||
"integrity": "sha512-E4E3RJJ0RBBCDGJ6cx7httfnV0Z2xcdF81epe581xSvPsCe42qWYysZ6DKTkBTrmMjNeScNnDkjubLS5RSODtw==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"rdf-string": "^1.3.1",
|
||||
"rdf-terms": "^1.4.0"
|
||||
}
|
||||
},
|
||||
"rdf-string": {
|
||||
"version": "1.4.2",
|
||||
"resolved": "https://registry.npmjs.org/rdf-string/-/rdf-string-1.4.2.tgz",
|
||||
"integrity": "sha512-74yYjS0W4N3nYDGbXBZrNsqDmhBTjqChTETO9heC2G2M3iMYaIPtEfUikNsBWUj4+4bIKyqL7vAntWBTfJpFFA==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"@rdfjs/data-model": "^1.1.1"
|
||||
}
|
||||
},
|
||||
"rdf-terms": {
|
||||
"version": "1.5.1",
|
||||
"resolved": "https://registry.npmjs.org/rdf-terms/-/rdf-terms-1.5.1.tgz",
|
||||
"integrity": "sha512-dDhpUYxTAOWKT3Ln93A5k5UB5SftG8bPAzeZEjGeP4e7eboMhITUTDks8HDmUt9X1P+HpfnY/o6VSTSpf3Advw==",
|
||||
"dev": true,
|
||||
"requires": {
|
||||
"@rdfjs/data-model": "^1.1.1",
|
||||
"lodash.uniqwith": "^4.5.0"
|
||||
}
|
||||
},
|
||||
"react-is": {
|
||||
"version": "16.13.1",
|
||||
"resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz",
|
||||
@ -5489,6 +5554,12 @@
|
||||
"integrity": "sha1-NbCYdbT/SfJqd35QmzCQoyJr8ks=",
|
||||
"dev": true
|
||||
},
|
||||
"streamify-array": {
|
||||
"version": "1.0.1",
|
||||
"resolved": "https://registry.npmjs.org/streamify-array/-/streamify-array-1.0.1.tgz",
|
||||
"integrity": "sha512-ZnswaBcC6B1bhPLSQOlC6CdaDUSzU0wr2lvvHpbHNms8V7+DLd8uEAzDAWpsjxbFkijBHhuObFO/qqu52DZUMA==",
|
||||
"dev": true
|
||||
},
|
||||
"string-length": {
|
||||
"version": "4.0.1",
|
||||
"resolved": "https://registry.npmjs.org/string-length/-/string-length-4.0.1.tgz",
|
||||
|
@ -22,18 +22,23 @@
|
||||
],
|
||||
"dependencies": {
|
||||
"@rdfjs/data-model": "^1.1.2",
|
||||
"@types/n3": "^1.1.6",
|
||||
"@types/node": "^14.0.1",
|
||||
"@types/rdf-js": "^3.0.0"
|
||||
"@types/rdf-js": "^3.0.0",
|
||||
"n3": "^1.3.7"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/jest": "^25.2.1",
|
||||
"@typescript-eslint/eslint-plugin": "^2.33.0",
|
||||
"@typescript-eslint/parser": "^2.33.0",
|
||||
"arrayify-stream": "^1.0.0",
|
||||
"eslint": "^7.0.0",
|
||||
"eslint-config-es": "^3.19.61",
|
||||
"eslint-plugin-tsdoc": "^0.2.4",
|
||||
"husky": "^4.2.5",
|
||||
"jest": "^26.0.1",
|
||||
"jest-rdf": "^1.5.0",
|
||||
"streamify-array": "^1.0.1",
|
||||
"ts-jest": "^26.0.0",
|
||||
"typescript": "^3.9.2"
|
||||
}
|
||||
|
52
src/ldp/http/SimpleBodyParser.ts
Normal file
52
src/ldp/http/SimpleBodyParser.ts
Normal file
@ -0,0 +1,52 @@
|
||||
import { BodyParser } from './BodyParser';
|
||||
import { HttpRequest } from '../../server/HttpRequest';
|
||||
import { Quad } from 'rdf-js';
|
||||
import { QuadRepresentation } from '../representation/QuadRepresentation';
|
||||
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 = [
|
||||
'application/n-quads',
|
||||
'application/trig',
|
||||
'application/n-triples',
|
||||
'text/turtle',
|
||||
'text/n3',
|
||||
];
|
||||
|
||||
public async canHandle(input: HttpRequest): Promise<void> {
|
||||
const contentType = input.headers['content-type'];
|
||||
|
||||
if (contentType && !SimpleBodyParser.contentTypes.some((type): boolean => contentType.includes(type))) {
|
||||
throw new UnsupportedMediaTypeHttpError('This parser only supports RDF data.');
|
||||
}
|
||||
}
|
||||
|
||||
public async handle(input: HttpRequest): Promise<QuadRepresentation> {
|
||||
const contentType = input.headers['content-type'];
|
||||
|
||||
if (!contentType) {
|
||||
return undefined;
|
||||
}
|
||||
|
||||
const specificType = contentType.split(';')[0];
|
||||
|
||||
const metadata: RepresentationMetadata = {
|
||||
raw: [],
|
||||
profiles: [],
|
||||
contentType: specificType,
|
||||
};
|
||||
|
||||
// StreamParser is a Readable but typings are incorrect at time of writing
|
||||
const quads: TypedReadable<Quad> = input.pipe(new StreamParser()) as unknown as TypedReadable<Quad>;
|
||||
|
||||
return {
|
||||
dataType: 'quad',
|
||||
data: quads,
|
||||
metadata,
|
||||
};
|
||||
}
|
||||
}
|
7
src/util/errors/UnsupportedMediaTypeHttpError.ts
Normal file
7
src/util/errors/UnsupportedMediaTypeHttpError.ts
Normal file
@ -0,0 +1,7 @@
|
||||
import { HttpError } from './HttpError';
|
||||
|
||||
export class UnsupportedMediaTypeHttpError extends HttpError {
|
||||
public constructor(message?: string) {
|
||||
super(415, 'UnsupportedHttpError', message);
|
||||
}
|
||||
}
|
58
test/unit/ldp/http/SimpleBodyParser.test.ts
Normal file
58
test/unit/ldp/http/SimpleBodyParser.test.ts
Normal file
@ -0,0 +1,58 @@
|
||||
import arrayifyStream from 'arrayify-stream';
|
||||
import { HttpRequest } from '../../../../src/server/HttpRequest';
|
||||
import { SimpleBodyParser } from '../../../../src/ldp/http/SimpleBodyParser';
|
||||
import streamifyArray from 'streamify-array';
|
||||
import { StreamParser } from 'n3';
|
||||
import { UnsupportedMediaTypeHttpError } from '../../../../src/util/errors/UnsupportedMediaTypeHttpError';
|
||||
import { namedNode, triple } from '@rdfjs/data-model';
|
||||
|
||||
const contentTypes = [
|
||||
'application/n-quads',
|
||||
'application/trig',
|
||||
'application/n-triples',
|
||||
'text/turtle',
|
||||
'text/n3',
|
||||
];
|
||||
|
||||
describe('A SimpleBodyparser', (): void => {
|
||||
const bodyParser = new SimpleBodyParser();
|
||||
|
||||
it('rejects input with unsupported content type.', async(): Promise<void> => {
|
||||
await expect(bodyParser.canHandle({ headers: { 'content-type': 'application/rdf+xml' }} as HttpRequest))
|
||||
.rejects.toThrow(new UnsupportedMediaTypeHttpError('This parser only supports RDF data.'));
|
||||
});
|
||||
|
||||
it('accepts input with no content type.', async(): Promise<void> => {
|
||||
await expect(bodyParser.canHandle({ headers: { }} as HttpRequest)).resolves.toBeUndefined();
|
||||
});
|
||||
|
||||
it('accepts turtle and similar content types.', async(): Promise<void> => {
|
||||
for (const type of contentTypes) {
|
||||
await expect(bodyParser.canHandle({ headers: { 'content-type': type }} as HttpRequest)).resolves.toBeUndefined();
|
||||
}
|
||||
});
|
||||
|
||||
it('returns empty output if there was no content-type.', async(): Promise<void> => {
|
||||
await expect(bodyParser.handle({ headers: { }} as HttpRequest)).resolves.toBeUndefined();
|
||||
});
|
||||
|
||||
it('returns a stream of quads if there was data.', async(): Promise<void> => {
|
||||
const input = streamifyArray([ '<http://test.com/s> <http://test.com/p> <http://test.com/o>.' ]) as HttpRequest;
|
||||
input.headers = { 'content-type': 'text/turtle' };
|
||||
const result = await bodyParser.handle(input);
|
||||
expect(result).toEqual({
|
||||
data: expect.any(StreamParser),
|
||||
dataType: 'quad',
|
||||
metadata: {
|
||||
contentType: 'text/turtle',
|
||||
profiles: [],
|
||||
raw: [],
|
||||
},
|
||||
});
|
||||
await expect(arrayifyStream(result.data)).resolves.toEqualRdfQuadArray([ triple(
|
||||
namedNode('http://test.com/s'),
|
||||
namedNode('http://test.com/p'),
|
||||
namedNode('http://test.com/o'),
|
||||
) ]);
|
||||
});
|
||||
});
|
@ -5,6 +5,7 @@
|
||||
"newLine": "lf",
|
||||
"alwaysStrict": true,
|
||||
"declaration": true,
|
||||
"esModuleInterop": true,
|
||||
"inlineSources": true,
|
||||
"noImplicitAny": true,
|
||||
"noImplicitThis": true,
|
||||
@ -14,6 +15,7 @@
|
||||
"stripInternal": true
|
||||
},
|
||||
"include": [
|
||||
"external-types/**/*.ts",
|
||||
"src/**/*.ts",
|
||||
"test/**/*.ts"
|
||||
],
|
||||
|
Loading…
x
Reference in New Issue
Block a user