mirror of
https://github.com/CommunitySolidServer/CommunitySolidServer.git
synced 2024-10-03 14:55:10 +00:00
feat: Allow JSON-LD contexts to be stored locally
This commit is contained in:
parent
e050f8be93
commit
b0924bf168
1
package-lock.json
generated
1
package-lock.json
generated
@ -9,6 +9,7 @@
|
|||||||
"version": "5.0.0",
|
"version": "5.0.0",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@comunica/context-entries": "^2.2.0",
|
||||||
"@comunica/query-sparql": "^2.2.1",
|
"@comunica/query-sparql": "^2.2.1",
|
||||||
"@rdfjs/types": "^1.1.0",
|
"@rdfjs/types": "^1.1.0",
|
||||||
"@solid/access-token-verifier": "^2.0.3",
|
"@solid/access-token-verifier": "^2.0.3",
|
||||||
|
@ -99,6 +99,7 @@
|
|||||||
"templates"
|
"templates"
|
||||||
],
|
],
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@comunica/context-entries": "^2.2.0",
|
||||||
"@comunica/query-sparql": "^2.2.1",
|
"@comunica/query-sparql": "^2.2.1",
|
||||||
"@rdfjs/types": "^1.1.0",
|
"@rdfjs/types": "^1.1.0",
|
||||||
"@solid/access-token-verifier": "^2.0.3",
|
"@solid/access-token-verifier": "^2.0.3",
|
||||||
@ -193,7 +194,6 @@
|
|||||||
"nodemon": "^2.0.19",
|
"nodemon": "^2.0.19",
|
||||||
"set-cookie-parser": "^2.5.1",
|
"set-cookie-parser": "^2.5.1",
|
||||||
"simple-git": "^3.12.0",
|
"simple-git": "^3.12.0",
|
||||||
"commit-and-tag-version": "^10.1.0",
|
|
||||||
"supertest": "^6.2.4",
|
"supertest": "^6.2.4",
|
||||||
"ts-jest": "^27.1.5",
|
"ts-jest": "^27.1.5",
|
||||||
"ts-node": "^10.9.1",
|
"ts-node": "^10.9.1",
|
||||||
|
@ -1,25 +1,61 @@
|
|||||||
import { PassThrough } from 'stream';
|
import { PassThrough } from 'stream';
|
||||||
|
import { KeysRdfParseJsonLd } from '@comunica/context-entries';
|
||||||
import type { NamedNode } from '@rdfjs/types';
|
import type { NamedNode } from '@rdfjs/types';
|
||||||
|
import fetch from 'cross-fetch';
|
||||||
|
import { readJsonSync } from 'fs-extra';
|
||||||
|
import { FetchDocumentLoader } from 'jsonld-context-parser';
|
||||||
|
import type { IJsonLdContext } from 'jsonld-context-parser';
|
||||||
import rdfParser from 'rdf-parse';
|
import rdfParser from 'rdf-parse';
|
||||||
import { BasicRepresentation } from '../../http/representation/BasicRepresentation';
|
import { BasicRepresentation } from '../../http/representation/BasicRepresentation';
|
||||||
import type { Representation } from '../../http/representation/Representation';
|
import type { Representation } from '../../http/representation/Representation';
|
||||||
import { RepresentationMetadata } from '../../http/representation/RepresentationMetadata';
|
import { RepresentationMetadata } from '../../http/representation/RepresentationMetadata';
|
||||||
import { INTERNAL_QUADS } from '../../util/ContentTypes';
|
import { INTERNAL_QUADS } from '../../util/ContentTypes';
|
||||||
import { BadRequestHttpError } from '../../util/errors/BadRequestHttpError';
|
import { BadRequestHttpError } from '../../util/errors/BadRequestHttpError';
|
||||||
|
import { resolveAssetPath } from '../../util/PathUtil';
|
||||||
import { pipeSafely } from '../../util/StreamUtil';
|
import { pipeSafely } from '../../util/StreamUtil';
|
||||||
import { PREFERRED_PREFIX_TERM, SOLID_META } from '../../util/Vocabularies';
|
import { PREFERRED_PREFIX_TERM, SOLID_META } from '../../util/Vocabularies';
|
||||||
import { BaseTypedRepresentationConverter } from './BaseTypedRepresentationConverter';
|
import { BaseTypedRepresentationConverter } from './BaseTypedRepresentationConverter';
|
||||||
import type { RepresentationConverterArgs } from './RepresentationConverter';
|
import type { RepresentationConverterArgs } from './RepresentationConverter';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* First checks if a context is stored locally before letting the super class do a fetch.
|
||||||
|
*/
|
||||||
|
class ContextDocumentLoader extends FetchDocumentLoader {
|
||||||
|
private readonly contexts: Record<string, IJsonLdContext>;
|
||||||
|
|
||||||
|
public constructor(contexts: Record<string, string>) {
|
||||||
|
super(fetch);
|
||||||
|
this.contexts = {};
|
||||||
|
for (const [ key, path ] of Object.entries(contexts)) {
|
||||||
|
this.contexts[key] = readJsonSync(resolveAssetPath(path));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async load(url: string): Promise<IJsonLdContext> {
|
||||||
|
if (url in this.contexts) {
|
||||||
|
return this.contexts[url];
|
||||||
|
}
|
||||||
|
return super.load(url);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Converts most major RDF serializations to `internal/quads`.
|
* Converts most major RDF serializations to `internal/quads`.
|
||||||
|
*
|
||||||
|
* Custom contexts can be defined to be used when parsing JSON-LD.
|
||||||
|
* The keys of the object should be the URL of the context,
|
||||||
|
* and the values the file path of the contexts to use when the JSON-LD parser would fetch the given context.
|
||||||
|
* We use filepaths because embedding them directly into the configurations breaks Components.js.
|
||||||
*/
|
*/
|
||||||
export class RdfToQuadConverter extends BaseTypedRepresentationConverter {
|
export class RdfToQuadConverter extends BaseTypedRepresentationConverter {
|
||||||
public constructor() {
|
private readonly documentLoader: ContextDocumentLoader;
|
||||||
|
|
||||||
|
public constructor(contexts: Record<string, string> = {}) {
|
||||||
const inputTypes = rdfParser.getContentTypes()
|
const inputTypes = rdfParser.getContentTypes()
|
||||||
// ContentType application/json MAY NOT be converted to Quad.
|
// ContentType application/json MAY NOT be converted to Quad.
|
||||||
.then((types): string[] => types.filter((type): boolean => type !== 'application/json'));
|
.then((types): string[] => types.filter((type): boolean => type !== 'application/json'));
|
||||||
super(inputTypes, INTERNAL_QUADS);
|
super(inputTypes, INTERNAL_QUADS);
|
||||||
|
this.documentLoader = new ContextDocumentLoader(contexts);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async handle({ representation, identifier }: RepresentationConverterArgs): Promise<Representation> {
|
public async handle({ representation, identifier }: RepresentationConverterArgs): Promise<Representation> {
|
||||||
@ -27,7 +63,12 @@ export class RdfToQuadConverter extends BaseTypedRepresentationConverter {
|
|||||||
const rawQuads = rdfParser.parse(representation.data, {
|
const rawQuads = rdfParser.parse(representation.data, {
|
||||||
contentType: representation.metadata.contentType!,
|
contentType: representation.metadata.contentType!,
|
||||||
baseIRI: identifier.path,
|
baseIRI: identifier.path,
|
||||||
})
|
// All extra keys get passed in the Comunica ActionContext
|
||||||
|
// and this is the key that is used to define the document loader.
|
||||||
|
// See https://github.com/rubensworks/rdf-parse.js/blob/master/lib/RdfParser.ts
|
||||||
|
// and https://github.com/comunica/comunica/blob/master/packages/actor-rdf-parse-jsonld/lib/ActorRdfParseJsonLd.ts
|
||||||
|
[KeysRdfParseJsonLd.documentLoader.name]: this.documentLoader,
|
||||||
|
} as any)
|
||||||
// This works only for those cases where the data stream has been completely read before accessing the metadata.
|
// This works only for those cases where the data stream has been completely read before accessing the metadata.
|
||||||
// Eg. the PATCH operation, which is the main case why we store the prefixes in metadata here if there are any.
|
// Eg. the PATCH operation, which is the main case why we store the prefixes in metadata here if there are any.
|
||||||
// See also https://github.com/CommunitySolidServer/CommunitySolidServer/issues/126
|
// See also https://github.com/CommunitySolidServer/CommunitySolidServer/issues/126
|
||||||
|
7
test/assets/contexts/test.jsonld
Normal file
7
test/assets/contexts/test.jsonld
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
{
|
||||||
|
"@context": {
|
||||||
|
"@version": 1.1,
|
||||||
|
"test": "http://example.com/context#",
|
||||||
|
"testPredicate": { "@id": "test:predicate" }
|
||||||
|
}
|
||||||
|
}
|
@ -1,6 +1,7 @@
|
|||||||
import 'jest-rdf';
|
import 'jest-rdf';
|
||||||
import { Readable } from 'stream';
|
import { Readable } from 'stream';
|
||||||
import arrayifyStream from 'arrayify-stream';
|
import arrayifyStream from 'arrayify-stream';
|
||||||
|
import fetch, { Headers } from 'cross-fetch';
|
||||||
import { DataFactory } from 'n3';
|
import { DataFactory } from 'n3';
|
||||||
import rdfParser from 'rdf-parse';
|
import rdfParser from 'rdf-parse';
|
||||||
import { PREFERRED_PREFIX_TERM, SOLID_META } from '../../../../src';
|
import { PREFERRED_PREFIX_TERM, SOLID_META } from '../../../../src';
|
||||||
@ -14,10 +15,25 @@ import { INTERNAL_QUADS } from '../../../../src/util/ContentTypes';
|
|||||||
import { BadRequestHttpError } from '../../../../src/util/errors/BadRequestHttpError';
|
import { BadRequestHttpError } from '../../../../src/util/errors/BadRequestHttpError';
|
||||||
const { namedNode, triple, literal, quad } = DataFactory;
|
const { namedNode, triple, literal, quad } = DataFactory;
|
||||||
|
|
||||||
describe('A RdfToQuadConverter', (): void => {
|
// All of this is necessary to not break the cross-fetch imports that happen in `rdf-parse`
|
||||||
const converter = new RdfToQuadConverter();
|
jest.mock('cross-fetch', (): any => {
|
||||||
const identifier: ResourceIdentifier = { path: 'path' };
|
const mock = jest.fn();
|
||||||
|
// Require the original module to not be mocked...
|
||||||
|
const originalFetch = jest.requireActual('cross-fetch');
|
||||||
|
return {
|
||||||
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
||||||
|
__esModule: true,
|
||||||
|
...originalFetch,
|
||||||
|
fetch: mock,
|
||||||
|
default: mock,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
// Not mocking `fs` since this breaks the `rdf-parser` library
|
||||||
|
describe('A RdfToQuadConverter', (): void => {
|
||||||
|
const fetchMock: jest.Mock = fetch as any;
|
||||||
|
const converter = new RdfToQuadConverter();
|
||||||
|
const identifier: ResourceIdentifier = { path: 'http://example.com/resource' };
|
||||||
it('supports serializing as quads.', async(): Promise<void> => {
|
it('supports serializing as quads.', async(): Promise<void> => {
|
||||||
const types = rdfParser.getContentTypes()
|
const types = rdfParser.getContentTypes()
|
||||||
.then((inputTypes): string[] => inputTypes.filter((type): boolean => type !== 'application/json'));
|
.then((inputTypes): string[] => inputTypes.filter((type): boolean => type !== 'application/json'));
|
||||||
@ -123,4 +139,43 @@ describe('A RdfToQuadConverter', (): void => {
|
|||||||
expect(result.metadata.contentType).toEqual(INTERNAL_QUADS);
|
expect(result.metadata.contentType).toEqual(INTERNAL_QUADS);
|
||||||
await expect(arrayifyStream(result.data)).rejects.toThrow(BadRequestHttpError);
|
await expect(arrayifyStream(result.data)).rejects.toThrow(BadRequestHttpError);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('can use locally stored contexts.', async(): Promise<void> => {
|
||||||
|
const fetchedContext = {
|
||||||
|
'@context': {
|
||||||
|
'@version': 1.1,
|
||||||
|
test: 'http://example.com/context2#',
|
||||||
|
testPredicate2: { '@id': 'test:predicate2' },
|
||||||
|
},
|
||||||
|
};
|
||||||
|
// This depends on the fields needed by the `jsonld-context-parser` so could break if library changes
|
||||||
|
fetchMock.mockResolvedValueOnce({
|
||||||
|
json: (): any => fetchedContext,
|
||||||
|
status: 200,
|
||||||
|
ok: true,
|
||||||
|
headers: new Headers({ 'content-type': 'application/ld+json' }),
|
||||||
|
});
|
||||||
|
|
||||||
|
const contextConverter = new RdfToQuadConverter(
|
||||||
|
{ 'http://example.com/context.jsonld': '@css:test/assets/contexts/test.jsonld' },
|
||||||
|
);
|
||||||
|
const jsonld = {
|
||||||
|
'@context': [ 'http://example.com/context.jsonld', 'http://example.com/context2.jsonld' ],
|
||||||
|
'@id': 'http://example.com/resource',
|
||||||
|
testPredicate: 123,
|
||||||
|
testPredicate2: 456,
|
||||||
|
};
|
||||||
|
const representation = new BasicRepresentation(JSON.stringify(jsonld), 'application/ld+json');
|
||||||
|
const preferences: RepresentationPreferences = { type: { [INTERNAL_QUADS]: 1 }};
|
||||||
|
const result = await contextConverter.handle({ identifier, representation, preferences });
|
||||||
|
await expect(arrayifyStream(result.data)).resolves.toEqualRdfQuadArray([ triple(
|
||||||
|
namedNode('http://example.com/resource'),
|
||||||
|
namedNode('http://example.com/context#predicate'),
|
||||||
|
literal(123),
|
||||||
|
), triple(
|
||||||
|
namedNode('http://example.com/resource'),
|
||||||
|
namedNode('http://example.com/context2#predicate2'),
|
||||||
|
literal(456),
|
||||||
|
) ]);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
Loading…
x
Reference in New Issue
Block a user