mirror of
https://github.com/CommunitySolidServer/CommunitySolidServer.git
synced 2024-10-03 14:55:10 +00:00
Add optional path and url suffixes to FixedContentTypeMapper
This commit is contained in:
committed by
Joachim Van Herwegen
parent
cf6270d161
commit
4ac0167c8d
@@ -8,89 +8,289 @@ jest.mock('fs');
|
||||
describe('An FixedContentTypeMapper', (): void => {
|
||||
const base = 'http://test.com/';
|
||||
const rootFilepath = 'uploads/';
|
||||
const mapper = new FixedContentTypeMapper(base, rootFilepath, 'text/turtle');
|
||||
describe('without suffixes', (): void => {
|
||||
const mapper = new FixedContentTypeMapper(base, rootFilepath, 'text/turtle');
|
||||
|
||||
describe('mapUrlToFilePath', (): void => {
|
||||
it('throws 404 if the input path does not contain the base.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: 'invalid' })).rejects.toThrow(NotFoundHttpError);
|
||||
});
|
||||
describe('mapUrlToFilePath', (): void => {
|
||||
it('throws 404 if the input path does not contain the base.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: 'invalid' })).rejects.toThrow(NotFoundHttpError);
|
||||
});
|
||||
|
||||
it('throws 404 if the relative path does not start with a slash.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${trimTrailingSlashes(base)}test` }))
|
||||
.rejects.toThrow(new BadRequestHttpError('URL needs a / after the base'));
|
||||
});
|
||||
it('throws 404 if the relative path does not start with a slash.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${trimTrailingSlashes(base)}test` }))
|
||||
.rejects.toThrow(new BadRequestHttpError('URL needs a / after the base'));
|
||||
});
|
||||
|
||||
it('throws 400 if the input path contains relative parts.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test/../test2` }))
|
||||
.rejects.toThrow(new BadRequestHttpError('Disallowed /.. segment in URL'));
|
||||
});
|
||||
it('throws 400 if the input path contains relative parts.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test/../test2` }))
|
||||
.rejects.toThrow(new BadRequestHttpError('Disallowed /.. segment in URL'));
|
||||
});
|
||||
|
||||
it('returns the corresponding file path for container identifiers.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}container/` })).resolves.toEqual({
|
||||
identifier: { path: `${base}container/` },
|
||||
filePath: `${rootFilepath}container/`,
|
||||
it('returns the corresponding file path for container identifiers.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}container/` })).resolves.toEqual({
|
||||
identifier: { path: `${base}container/` },
|
||||
filePath: `${rootFilepath}container/`,
|
||||
});
|
||||
});
|
||||
|
||||
it('always returns the configured content-type.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test` })).resolves.toEqual({
|
||||
identifier: { path: `${base}test` },
|
||||
filePath: `${rootFilepath}test`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` })).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test.ttl`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.txt` })).resolves.toEqual({
|
||||
identifier: { path: `${base}test.txt` },
|
||||
filePath: `${rootFilepath}test.txt`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
});
|
||||
|
||||
it('generates a file path if supported content-type was provided.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` }, 'text/turtle')).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test.ttl`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
});
|
||||
|
||||
it('throws 400 if the given content-type is not supported.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` }, 'application/n-quads')).rejects
|
||||
.toThrow(
|
||||
new BadRequestHttpError(`Unsupported content type application/n-quads, only text/turtle is allowed`),
|
||||
);
|
||||
});
|
||||
});
|
||||
|
||||
it('always returns the configured content-type.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test` })).resolves.toEqual({
|
||||
identifier: { path: `${base}test` },
|
||||
filePath: `${rootFilepath}test`,
|
||||
contentType: 'text/turtle',
|
||||
describe('mapFilePathToUrl', (): void => {
|
||||
it('throws an error if the input path does not contain the root file path.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl('invalid', true)).rejects.toThrow(Error);
|
||||
});
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` })).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test.ttl`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.txt` })).resolves.toEqual({
|
||||
identifier: { path: `${base}test.txt` },
|
||||
filePath: `${rootFilepath}test.txt`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
});
|
||||
|
||||
it('generates a file path if supported content-type was provided.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` }, 'text/turtle')).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test.ttl`,
|
||||
contentType: 'text/turtle',
|
||||
it('returns a generated identifier for directories.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}container/`, true)).resolves.toEqual({
|
||||
identifier: { path: `${base}container/` },
|
||||
filePath: `${rootFilepath}container/`,
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
it('throws 400 if the given content-type is not supported.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` }, 'application/n-quads')).rejects
|
||||
.toThrow(new BadRequestHttpError(`Unsupported content type application/n-quads, only text/turtle is allowed`));
|
||||
it('returns files with the configured content-type.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test`, false)).resolves.toEqual({
|
||||
identifier: { path: `${base}test` },
|
||||
filePath: `${rootFilepath}test`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test.ttl`, false)).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test.ttl`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test.txt`, false)).resolves.toEqual({
|
||||
identifier: { path: `${base}test.txt` },
|
||||
filePath: `${rootFilepath}test.txt`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
describe('mapFilePathToUrl', (): void => {
|
||||
it('throws an error if the input path does not contain the root file path.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl('invalid', true)).rejects.toThrow(Error);
|
||||
});
|
||||
describe('with path suffix', (): void => {
|
||||
// Internally, everything uses the .ttl extension, but it's exposed without.
|
||||
const mapper = new FixedContentTypeMapper(base, rootFilepath, 'text/turtle', '.ttl');
|
||||
|
||||
it('returns a generated identifier for directories.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}container/`, true)).resolves.toEqual({
|
||||
identifier: { path: `${base}container/` },
|
||||
filePath: `${rootFilepath}container/`,
|
||||
describe('mapUrlToFilePath', (): void => {
|
||||
it('returns the corresponding file path for container identifiers.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}container/` })).resolves.toEqual({
|
||||
identifier: { path: `${base}container/` },
|
||||
filePath: `${rootFilepath}container/`,
|
||||
});
|
||||
});
|
||||
|
||||
it('always returns the configured content-type.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test` })).resolves.toEqual({
|
||||
identifier: { path: `${base}test` },
|
||||
filePath: `${rootFilepath}test.ttl`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` })).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test.ttl.ttl`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.txt` })).resolves.toEqual({
|
||||
identifier: { path: `${base}test.txt` },
|
||||
filePath: `${rootFilepath}test.txt.ttl`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
});
|
||||
|
||||
it('generates a file path if supported content-type was provided.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` }, 'text/turtle')).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test.ttl.ttl`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
});
|
||||
|
||||
it('throws 400 if the given content-type is not supported.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` }, 'application/n-quads')).rejects
|
||||
.toThrow(
|
||||
new BadRequestHttpError(`Unsupported content type application/n-quads, only text/turtle is allowed`),
|
||||
);
|
||||
});
|
||||
});
|
||||
|
||||
it('returns files with the configured content-type.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test`, false)).resolves.toEqual({
|
||||
identifier: { path: `${base}test` },
|
||||
filePath: `${rootFilepath}test`,
|
||||
contentType: 'text/turtle',
|
||||
describe('mapFilePathToUrl', (): void => {
|
||||
it('returns a generated identifier for directories.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}container/`, true)).resolves.toEqual({
|
||||
identifier: { path: `${base}container/` },
|
||||
filePath: `${rootFilepath}container/`,
|
||||
});
|
||||
});
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test.ttl`, false)).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test.ttl`,
|
||||
contentType: 'text/turtle',
|
||||
|
||||
it('returns files with the configured content-type.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test.ttl`, false)).resolves.toEqual({
|
||||
identifier: { path: `${base}test` },
|
||||
filePath: `${rootFilepath}test.ttl`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
});
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test.txt`, false)).resolves.toEqual({
|
||||
identifier: { path: `${base}test.txt` },
|
||||
filePath: `${rootFilepath}test.txt`,
|
||||
contentType: 'text/turtle',
|
||||
|
||||
it('throws 404 if the input path does not end with the suffix.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test`, false)).rejects.toThrow(NotFoundHttpError);
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test.txt`, false)).rejects.toThrow(NotFoundHttpError);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
describe('with url suffix', (): void => {
|
||||
// Internally, no extensions are used, but everything is exposed with the .ttl extension
|
||||
const mapper = new FixedContentTypeMapper(base, rootFilepath, 'text/turtle', undefined, '.ttl');
|
||||
|
||||
describe('mapUrlToFilePath', (): void => {
|
||||
it('returns the corresponding file path for container identifiers.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}container/` })).resolves.toEqual({
|
||||
identifier: { path: `${base}container/` },
|
||||
filePath: `${rootFilepath}container/`,
|
||||
});
|
||||
});
|
||||
|
||||
it('always returns the configured content-type.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` })).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.txt.ttl` })).resolves.toEqual({
|
||||
identifier: { path: `${base}test.txt.ttl` },
|
||||
filePath: `${rootFilepath}test.txt`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
});
|
||||
|
||||
it('generates a file path if supported content-type was provided.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` }, 'text/turtle')).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
});
|
||||
|
||||
it('throws 404 if the url does not end with the suffix.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.nq` }, 'text/turtle')).rejects
|
||||
.toThrow(NotFoundHttpError);
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test` }, 'text/turtle')).rejects
|
||||
.toThrow(NotFoundHttpError);
|
||||
});
|
||||
|
||||
it('throws 400 if the given content-type is not supported.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` }, 'application/n-quads')).rejects
|
||||
.toThrow(
|
||||
new BadRequestHttpError(`Unsupported content type application/n-quads, only text/turtle is allowed`),
|
||||
);
|
||||
});
|
||||
});
|
||||
|
||||
describe('mapFilePathToUrl', (): void => {
|
||||
it('returns a generated identifier for directories.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}container/`, true)).resolves.toEqual({
|
||||
identifier: { path: `${base}container/` },
|
||||
filePath: `${rootFilepath}container/`,
|
||||
});
|
||||
});
|
||||
|
||||
it('returns files with the configured content-type.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test`, false)).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test.ttl`, false)).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl.ttl` },
|
||||
filePath: `${rootFilepath}test.ttl`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test.txt`, false)).resolves.toEqual({
|
||||
identifier: { path: `${base}test.txt.ttl` },
|
||||
filePath: `${rootFilepath}test.txt`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
describe('with path and url suffix', (): void => {
|
||||
// Internally, everything uses the .nq extension, but it's exposed with the .ttl extension.
|
||||
const mapper = new FixedContentTypeMapper(base, rootFilepath, 'text/turtle', '.nq', '.ttl');
|
||||
|
||||
describe('mapUrlToFilePath', (): void => {
|
||||
it('returns the corresponding file path for container identifiers.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}container/` })).resolves.toEqual({
|
||||
identifier: { path: `${base}container/` },
|
||||
filePath: `${rootFilepath}container/`,
|
||||
});
|
||||
});
|
||||
|
||||
it('always returns the configured content-type.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.ttl` })).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test.nq`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
});
|
||||
|
||||
it('throws 404 if the url does not end with the suffix.', async(): Promise<void> => {
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test.nq` }, 'text/turtle')).rejects
|
||||
.toThrow(NotFoundHttpError);
|
||||
await expect(mapper.mapUrlToFilePath({ path: `${base}test` }, 'text/turtle')).rejects
|
||||
.toThrow(NotFoundHttpError);
|
||||
});
|
||||
});
|
||||
|
||||
describe('mapFilePathToUrl', (): void => {
|
||||
it('returns a generated identifier for directories.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}container/`, true)).resolves.toEqual({
|
||||
identifier: { path: `${base}container/` },
|
||||
filePath: `${rootFilepath}container/`,
|
||||
});
|
||||
});
|
||||
|
||||
it('returns files with the configured content-type.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test.nq`, false)).resolves.toEqual({
|
||||
identifier: { path: `${base}test.ttl` },
|
||||
filePath: `${rootFilepath}test.nq`,
|
||||
contentType: 'text/turtle',
|
||||
});
|
||||
});
|
||||
|
||||
it('throws 404 if the input path does not end with the suffix.', async(): Promise<void> => {
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test`, false)).rejects.toThrow(NotFoundHttpError);
|
||||
await expect(mapper.mapFilePathToUrl(`${rootFilepath}test.ttl`, false)).rejects.toThrow(NotFoundHttpError);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
Reference in New Issue
Block a user