mirror of
https://github.com/CommunitySolidServer/CommunitySolidServer.git
synced 2024-10-03 14:55:10 +00:00
refactor: Rename Agent to PodSettings
This commit is contained in:
parent
0f00a8dffd
commit
38afd72098
@ -34,8 +34,8 @@
|
|||||||
"PodManagerHttpHandler:_args_requestParser": {
|
"PodManagerHttpHandler:_args_requestParser": {
|
||||||
"@id": "urn:solid-server:default:RequestParser"
|
"@id": "urn:solid-server:default:RequestParser"
|
||||||
},
|
},
|
||||||
"PodManagerHttpHandler:_args_agentParser": {
|
"PodManagerHttpHandler:_args_podSettingsParser": {
|
||||||
"@type": "AgentJsonParser"
|
"@type": "PodSettingsJsonParser"
|
||||||
},
|
},
|
||||||
"PodManagerHttpHandler:_args_manager": {
|
"PodManagerHttpHandler:_args_manager": {
|
||||||
"@id": "urn:solid-server:default:PodManager"
|
"@id": "urn:solid-server:default:PodManager"
|
||||||
|
10
src/index.ts
10
src/index.ts
@ -108,11 +108,6 @@ export * from './logging/LogUtil';
|
|||||||
export * from './logging/VoidLoggerFactory';
|
export * from './logging/VoidLoggerFactory';
|
||||||
export * from './logging/WinstonLoggerFactory';
|
export * from './logging/WinstonLoggerFactory';
|
||||||
|
|
||||||
// Pods/Agent
|
|
||||||
export * from './pods/agent/Agent';
|
|
||||||
export * from './pods/agent/AgentJsonParser';
|
|
||||||
export * from './pods/agent/AgentParser';
|
|
||||||
|
|
||||||
// Pods/Generate
|
// Pods/Generate
|
||||||
export * from './pods/generate/HandlebarsTemplateEngine';
|
export * from './pods/generate/HandlebarsTemplateEngine';
|
||||||
export * from './pods/generate/IdentifierGenerator';
|
export * from './pods/generate/IdentifierGenerator';
|
||||||
@ -122,6 +117,11 @@ export * from './pods/generate/SuffixIdentifierGenerator';
|
|||||||
export * from './pods/generate/TemplateEngine';
|
export * from './pods/generate/TemplateEngine';
|
||||||
export * from './pods/generate/TemplatedResourcesGenerator';
|
export * from './pods/generate/TemplatedResourcesGenerator';
|
||||||
|
|
||||||
|
// Pods/Settings
|
||||||
|
export * from './pods/settings/PodSettings';
|
||||||
|
export * from './pods/settings/PodSettingsJsonParser';
|
||||||
|
export * from './pods/settings/PodSettingsParser';
|
||||||
|
|
||||||
// Pods
|
// Pods
|
||||||
export * from './pods/GeneratedPodManager';
|
export * from './pods/GeneratedPodManager';
|
||||||
export * from './pods/PodManager';
|
export * from './pods/PodManager';
|
||||||
|
@ -2,10 +2,10 @@ import type { ResourceIdentifier } from '../ldp/representation/ResourceIdentifie
|
|||||||
import { getLoggerFor } from '../logging/LogUtil';
|
import { getLoggerFor } from '../logging/LogUtil';
|
||||||
import type { ResourceStore } from '../storage/ResourceStore';
|
import type { ResourceStore } from '../storage/ResourceStore';
|
||||||
import { ConflictHttpError } from '../util/errors/ConflictHttpError';
|
import { ConflictHttpError } from '../util/errors/ConflictHttpError';
|
||||||
import type { Agent } from './agent/Agent';
|
|
||||||
import type { IdentifierGenerator } from './generate/IdentifierGenerator';
|
import type { IdentifierGenerator } from './generate/IdentifierGenerator';
|
||||||
import type { ResourcesGenerator } from './generate/ResourcesGenerator';
|
import type { ResourcesGenerator } from './generate/ResourcesGenerator';
|
||||||
import type { PodManager } from './PodManager';
|
import type { PodManager } from './PodManager';
|
||||||
|
import type { PodSettings } from './settings/PodSettings';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Pod manager that uses an {@link IdentifierGenerator} and {@link ResourcesGenerator}
|
* Pod manager that uses an {@link IdentifierGenerator} and {@link ResourcesGenerator}
|
||||||
@ -29,14 +29,14 @@ export class GeneratedPodManager implements PodManager {
|
|||||||
* Pod identifiers are created based on the identifier generator.
|
* Pod identifiers are created based on the identifier generator.
|
||||||
* Will throw an error if the given identifier already has a resource.
|
* Will throw an error if the given identifier already has a resource.
|
||||||
*/
|
*/
|
||||||
public async createPod(agent: Agent): Promise<ResourceIdentifier> {
|
public async createPod(settings: PodSettings): Promise<ResourceIdentifier> {
|
||||||
const podIdentifier = this.idGenerator.generate(agent.login);
|
const podIdentifier = this.idGenerator.generate(settings.login);
|
||||||
this.logger.info(`Creating pod ${podIdentifier.path}`);
|
this.logger.info(`Creating pod ${podIdentifier.path}`);
|
||||||
if (await this.store.resourceExists(podIdentifier)) {
|
if (await this.store.resourceExists(podIdentifier)) {
|
||||||
throw new ConflictHttpError(`There already is a resource at ${podIdentifier.path}`);
|
throw new ConflictHttpError(`There already is a resource at ${podIdentifier.path}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const resources = this.resourcesGenerator.generate(podIdentifier, agent);
|
const resources = this.resourcesGenerator.generate(podIdentifier, settings);
|
||||||
let count = 0;
|
let count = 0;
|
||||||
for await (const { identifier, representation } of resources) {
|
for await (const { identifier, representation } of resources) {
|
||||||
await this.store.setRepresentation(identifier, representation);
|
await this.store.setRepresentation(identifier, representation);
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import type { ResourceIdentifier } from '../ldp/representation/ResourceIdentifier';
|
import type { ResourceIdentifier } from '../ldp/representation/ResourceIdentifier';
|
||||||
import type { Agent } from './agent/Agent';
|
import type { PodSettings } from './settings/PodSettings';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Covers all functions related to pod management.
|
* Covers all functions related to pod management.
|
||||||
@ -7,9 +7,9 @@ import type { Agent } from './agent/Agent';
|
|||||||
*/
|
*/
|
||||||
export interface PodManager {
|
export interface PodManager {
|
||||||
/**
|
/**
|
||||||
* Creates a pod for the given agent data.
|
* Creates a pod for the given settings.
|
||||||
* @param agent - Data of the agent that needs a pod.
|
* @param settings - Settings describing the pod.
|
||||||
* @returns {@link ResourceIdentifier} of the newly created pod.
|
* @returns {@link ResourceIdentifier} of the newly created pod.
|
||||||
*/
|
*/
|
||||||
createPod: (agent: Agent) => Promise<ResourceIdentifier>;
|
createPod: (settings: PodSettings) => Promise<ResourceIdentifier>;
|
||||||
}
|
}
|
||||||
|
@ -8,16 +8,16 @@ import { BadRequestHttpError } from '../util/errors/BadRequestHttpError';
|
|||||||
import { isNativeError } from '../util/errors/ErrorUtil';
|
import { isNativeError } from '../util/errors/ErrorUtil';
|
||||||
import { InternalServerError } from '../util/errors/InternalServerError';
|
import { InternalServerError } from '../util/errors/InternalServerError';
|
||||||
import { NotImplementedHttpError } from '../util/errors/NotImplementedHttpError';
|
import { NotImplementedHttpError } from '../util/errors/NotImplementedHttpError';
|
||||||
import type { AgentParser } from './agent/AgentParser';
|
|
||||||
import type { PodManager } from './PodManager';
|
import type { PodManager } from './PodManager';
|
||||||
|
import type { PodSettingsParser } from './settings/PodSettingsParser';
|
||||||
|
|
||||||
export interface PodHttpHandlerArgs {
|
export interface PodHttpHandlerArgs {
|
||||||
/** The path on which this handler should intercept requests. Should start with a slash. */
|
/** The path on which this handler should intercept requests. Should start with a slash. */
|
||||||
requestPath: string;
|
requestPath: string;
|
||||||
/** Parses the incoming request. */
|
/** Parses the incoming request. */
|
||||||
requestParser: RequestParser;
|
requestParser: RequestParser;
|
||||||
/** Parses the data stream to an Agent. */
|
/** Parses the data stream to PodSettings. */
|
||||||
agentParser: AgentParser;
|
podSettingsParser: PodSettingsParser;
|
||||||
/** Handles the pod management. */
|
/** Handles the pod management. */
|
||||||
manager: PodManager;
|
manager: PodManager;
|
||||||
/** Writes the outgoing response. */
|
/** Writes the outgoing response. */
|
||||||
@ -31,7 +31,7 @@ export interface PodHttpHandlerArgs {
|
|||||||
export class PodManagerHttpHandler extends HttpHandler {
|
export class PodManagerHttpHandler extends HttpHandler {
|
||||||
private readonly requestPath!: string;
|
private readonly requestPath!: string;
|
||||||
private readonly requestParser!: RequestParser;
|
private readonly requestParser!: RequestParser;
|
||||||
private readonly agentParser!: AgentParser;
|
private readonly podSettingsParser!: PodSettingsParser;
|
||||||
private readonly manager!: PodManager;
|
private readonly manager!: PodManager;
|
||||||
private readonly responseWriter!: ResponseWriter;
|
private readonly responseWriter!: ResponseWriter;
|
||||||
|
|
||||||
@ -55,8 +55,8 @@ export class PodManagerHttpHandler extends HttpHandler {
|
|||||||
if (!op.body) {
|
if (!op.body) {
|
||||||
throw new BadRequestHttpError('A body is required to create a pod');
|
throw new BadRequestHttpError('A body is required to create a pod');
|
||||||
}
|
}
|
||||||
const agent = await this.agentParser.handleSafe(op.body);
|
const settings = await this.podSettingsParser.handleSafe(op.body);
|
||||||
const id = await this.manager.createPod(agent);
|
const id = await this.manager.createPod(settings);
|
||||||
|
|
||||||
await this.responseWriter.handleSafe({ response, result: new CreatedResponseDescription(id) });
|
await this.responseWriter.handleSafe({ response, result: new CreatedResponseDescription(id) });
|
||||||
} catch (error: unknown) {
|
} catch (error: unknown) {
|
||||||
|
@ -1,9 +0,0 @@
|
|||||||
/**
|
|
||||||
* Agent metadata related to pod generation.
|
|
||||||
*/
|
|
||||||
export type Agent = {
|
|
||||||
login: string;
|
|
||||||
webId: string;
|
|
||||||
name?: string;
|
|
||||||
email?: string;
|
|
||||||
};
|
|
@ -1,48 +0,0 @@
|
|||||||
import type { Representation } from '../../ldp/representation/Representation';
|
|
||||||
import { BadRequestHttpError } from '../../util/errors/BadRequestHttpError';
|
|
||||||
import { NotImplementedHttpError } from '../../util/errors/NotImplementedHttpError';
|
|
||||||
import { readableToString } from '../../util/StreamUtil';
|
|
||||||
import type { Agent } from './Agent';
|
|
||||||
import { AgentParser } from './AgentParser';
|
|
||||||
import Dict = NodeJS.Dict;
|
|
||||||
|
|
||||||
const requiredKeys: (keyof Agent)[] = [ 'login', 'webId' ];
|
|
||||||
const optionalKeys: (keyof Agent)[] = [ 'name', 'email' ];
|
|
||||||
const agentKeys: Set<keyof Agent> = new Set(requiredKeys.concat(optionalKeys));
|
|
||||||
|
|
||||||
/**
|
|
||||||
* A parser that extracts Agent data from a JSON body.
|
|
||||||
*/
|
|
||||||
export class AgentJsonParser extends AgentParser {
|
|
||||||
public async canHandle(input: Representation): Promise<void> {
|
|
||||||
if (!input.metadata.contentType || !this.isJSON(input.metadata.contentType)) {
|
|
||||||
throw new NotImplementedHttpError('Only JSON data is supported');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public async handle(input: Representation): Promise<Agent> {
|
|
||||||
const result = JSON.parse(await readableToString(input.data));
|
|
||||||
this.isValidAgent(result);
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
private isJSON(mediaType: string): boolean {
|
|
||||||
return mediaType === 'application/json' || mediaType.endsWith('+json');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if all keys in the object are valid Agent keys and if all required keys are there.
|
|
||||||
*/
|
|
||||||
private isValidAgent(data: Dict<string>): asserts data is Agent {
|
|
||||||
for (const key of Object.keys(data)) {
|
|
||||||
if (!agentKeys.has(key as keyof Agent)) {
|
|
||||||
throw new BadRequestHttpError(`${key} is not a valid Agent key`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (const key of requiredKeys) {
|
|
||||||
if (!data[key]) {
|
|
||||||
throw new BadRequestHttpError(`Input data is missing Agent key ${key}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,8 +0,0 @@
|
|||||||
import type { Representation } from '../../ldp/representation/Representation';
|
|
||||||
import { AsyncHandler } from '../../util/handlers/AsyncHandler';
|
|
||||||
import type { Agent } from './Agent';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Parser that generates a {@link Agent} from the data in the given {@link Representation}.
|
|
||||||
*/
|
|
||||||
export abstract class AgentParser extends AsyncHandler<Representation, Agent> { }
|
|
@ -9,6 +9,7 @@ export interface Resource {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Generator used to create resources relative to a given base identifier.
|
* Generator used to create resources relative to a given base identifier.
|
||||||
|
* Note that this is not an AsyncHandler since it returns an AsyncIterable instead of a promise.
|
||||||
*/
|
*/
|
||||||
export interface ResourcesGenerator {
|
export interface ResourcesGenerator {
|
||||||
/**
|
/**
|
||||||
|
28
src/pods/settings/PodSettings.ts
Normal file
28
src/pods/settings/PodSettings.ts
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
/**
|
||||||
|
* Metadata related to pod generation.
|
||||||
|
* Although the optional fields are not that relevant since this extends Dict,
|
||||||
|
* they give an indication of what is sometimes expected.
|
||||||
|
*/
|
||||||
|
export interface PodSettings extends NodeJS.Dict<string> {
|
||||||
|
/**
|
||||||
|
* The name of the pod. Will be used for generating the identifier.
|
||||||
|
*/
|
||||||
|
login: string;
|
||||||
|
/**
|
||||||
|
* The WebId of the owner of this pod.
|
||||||
|
*/
|
||||||
|
webId: string;
|
||||||
|
/**
|
||||||
|
* Required for dynamic pod configuration.
|
||||||
|
* Indicates the name of the config to use for the pod.
|
||||||
|
*/
|
||||||
|
template?: string;
|
||||||
|
/**
|
||||||
|
* Name of the owner. Used in provisioning templates.
|
||||||
|
*/
|
||||||
|
name?: string;
|
||||||
|
/**
|
||||||
|
* E-mail of the owner. Used in provisioning templates.
|
||||||
|
*/
|
||||||
|
email?: string;
|
||||||
|
}
|
41
src/pods/settings/PodSettingsJsonParser.ts
Normal file
41
src/pods/settings/PodSettingsJsonParser.ts
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
import type { Representation } from '../../ldp/representation/Representation';
|
||||||
|
import { BadRequestHttpError } from '../../util/errors/BadRequestHttpError';
|
||||||
|
import { NotImplementedHttpError } from '../../util/errors/NotImplementedHttpError';
|
||||||
|
import { readableToString } from '../../util/StreamUtil';
|
||||||
|
import type { PodSettings } from './PodSettings';
|
||||||
|
import { PodSettingsParser } from './PodSettingsParser';
|
||||||
|
import Dict = NodeJS.Dict;
|
||||||
|
|
||||||
|
const requiredKeys: (keyof PodSettings)[] = [ 'login', 'webId' ];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A parser that extracts PodSettings data from a JSON body.
|
||||||
|
*/
|
||||||
|
export class PodSettingsJsonParser extends PodSettingsParser {
|
||||||
|
public async canHandle(input: Representation): Promise<void> {
|
||||||
|
if (!input.metadata.contentType || !this.isJSON(input.metadata.contentType)) {
|
||||||
|
throw new NotImplementedHttpError('Only JSON data is supported');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async handle(input: Representation): Promise<PodSettings> {
|
||||||
|
const result = JSON.parse(await readableToString(input.data));
|
||||||
|
this.isValid(result);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private isJSON(mediaType: string): boolean {
|
||||||
|
return mediaType === 'application/json' || mediaType.endsWith('+json');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if all the required PodSettings keys are there.
|
||||||
|
*/
|
||||||
|
private isValid(data: Dict<string>): asserts data is PodSettings {
|
||||||
|
for (const key of requiredKeys) {
|
||||||
|
if (!data[key]) {
|
||||||
|
throw new BadRequestHttpError(`Input data is missing key ${key}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
8
src/pods/settings/PodSettingsParser.ts
Normal file
8
src/pods/settings/PodSettingsParser.ts
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
import type { Representation } from '../../ldp/representation/Representation';
|
||||||
|
import { AsyncHandler } from '../../util/handlers/AsyncHandler';
|
||||||
|
import type { PodSettings } from './PodSettings';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parser that generates a {@link PodSettings} from the data in the given {@link Representation}.
|
||||||
|
*/
|
||||||
|
export abstract class PodSettingsParser extends AsyncHandler<Representation, PodSettings> { }
|
@ -10,7 +10,7 @@ const baseUrl = `http://localhost:${port}/`;
|
|||||||
|
|
||||||
describe('A server with a pod handler', (): void => {
|
describe('A server with a pod handler', (): void => {
|
||||||
let server: Server;
|
let server: Server;
|
||||||
const agent = { login: 'alice', webId: 'http://test.com/#alice', name: 'Alice Bob' };
|
const settings = { login: 'alice', webId: 'http://test.com/#alice', name: 'Alice Bob' };
|
||||||
|
|
||||||
beforeAll(async(): Promise<void> => {
|
beforeAll(async(): Promise<void> => {
|
||||||
const factory = await instantiateFromConfig(
|
const factory = await instantiateFromConfig(
|
||||||
@ -29,8 +29,8 @@ describe('A server with a pod handler', (): void => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('creates a pod when posting an Agent to /pods.', async(): Promise<void> => {
|
it('creates a pod when posting PodSettings to /pods.', async(): Promise<void> => {
|
||||||
const pod = `${baseUrl}${agent.login}/`;
|
const pod = `${baseUrl}${settings.login}/`;
|
||||||
|
|
||||||
// Pod should not exist yet
|
// Pod should not exist yet
|
||||||
let res = await fetch(pod);
|
let res = await fetch(pod);
|
||||||
@ -42,16 +42,16 @@ describe('A server with a pod handler', (): void => {
|
|||||||
headers: {
|
headers: {
|
||||||
'content-type': 'application/json',
|
'content-type': 'application/json',
|
||||||
},
|
},
|
||||||
body: JSON.stringify(agent),
|
body: JSON.stringify(settings),
|
||||||
});
|
});
|
||||||
expect(res.status).toBe(201);
|
expect(res.status).toBe(201);
|
||||||
expect(res.headers.get('location')).toBe(`${baseUrl}${agent.login}/`);
|
expect(res.headers.get('location')).toBe(`${baseUrl}${settings.login}/`);
|
||||||
|
|
||||||
// Check if all resources are created
|
// Check if all resources are created
|
||||||
res = await fetch(`${pod}.acl`);
|
res = await fetch(`${pod}.acl`);
|
||||||
expect(res.status).toBe(200);
|
expect(res.status).toBe(200);
|
||||||
let body = await readableToString(res.body as any);
|
let body = await readableToString(res.body as any);
|
||||||
expect(body).toContain(`acl:agent <${agent.webId}>`);
|
expect(body).toContain(`acl:agent <${settings.webId}>`);
|
||||||
|
|
||||||
res = await fetch(`${pod}profile/card`);
|
res = await fetch(`${pod}profile/card`);
|
||||||
expect(res.status).toBe(200);
|
expect(res.status).toBe(200);
|
||||||
@ -59,9 +59,9 @@ describe('A server with a pod handler', (): void => {
|
|||||||
body = await readableToString(res.body as any);
|
body = await readableToString(res.body as any);
|
||||||
expect(body).toBe(`@prefix foaf: <http://xmlns.com/foaf/0.1/>.
|
expect(body).toBe(`@prefix foaf: <http://xmlns.com/foaf/0.1/>.
|
||||||
|
|
||||||
<${agent.webId}>
|
<${settings.webId}>
|
||||||
a foaf:Person ;
|
a foaf:Person ;
|
||||||
foaf:name "${agent.name}".
|
foaf:name "${settings.name}".
|
||||||
`);
|
`);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -26,7 +26,7 @@ describe.each(stores)('A subdomain server with %s', (name, { storeUrn, teardown
|
|||||||
let server: Server;
|
let server: Server;
|
||||||
let initializer: Initializer;
|
let initializer: Initializer;
|
||||||
let factory: HttpServerFactory;
|
let factory: HttpServerFactory;
|
||||||
const agent = { login: 'alice', webId: 'http://test.com/#alice', name: 'Alice Bob' };
|
const settings = { login: 'alice', webId: 'http://test.com/#alice', name: 'Alice Bob' };
|
||||||
const podHost = `alice.localhost:${port}`;
|
const podHost = `alice.localhost:${port}`;
|
||||||
const podUrl = `http://${podHost}/`;
|
const podUrl = `http://${podHost}/`;
|
||||||
|
|
||||||
@ -75,7 +75,7 @@ describe.each(stores)('A subdomain server with %s', (name, { storeUrn, teardown
|
|||||||
let res = await fetch(`${baseUrl}alice`, {
|
let res = await fetch(`${baseUrl}alice`, {
|
||||||
method: 'PUT',
|
method: 'PUT',
|
||||||
headers: {
|
headers: {
|
||||||
authorization: `WebID ${agent.webId}`,
|
authorization: `WebID ${settings.webId}`,
|
||||||
'content-type': 'text/plain',
|
'content-type': 'text/plain',
|
||||||
},
|
},
|
||||||
body: 'this is new data!',
|
body: 'this is new data!',
|
||||||
@ -95,7 +95,7 @@ describe.each(stores)('A subdomain server with %s', (name, { storeUrn, teardown
|
|||||||
headers: {
|
headers: {
|
||||||
'content-type': 'application/json',
|
'content-type': 'application/json',
|
||||||
},
|
},
|
||||||
body: JSON.stringify(agent),
|
body: JSON.stringify(settings),
|
||||||
});
|
});
|
||||||
expect(res.status).toBe(201);
|
expect(res.status).toBe(201);
|
||||||
expect(res.headers.get('location')).toBe(podUrl);
|
expect(res.headers.get('location')).toBe(podUrl);
|
||||||
@ -115,7 +115,7 @@ describe.each(stores)('A subdomain server with %s', (name, { storeUrn, teardown
|
|||||||
const res = await fetch(`${baseUrl}.acl`, {
|
const res = await fetch(`${baseUrl}.acl`, {
|
||||||
headers: {
|
headers: {
|
||||||
forwarded: `host=${podHost}`,
|
forwarded: `host=${podHost}`,
|
||||||
authorization: `WebID ${agent.webId}`,
|
authorization: `WebID ${settings.webId}`,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
expect(res.status).toBe(200);
|
expect(res.status).toBe(200);
|
||||||
@ -125,7 +125,7 @@ describe.each(stores)('A subdomain server with %s', (name, { storeUrn, teardown
|
|||||||
let res = await fetch(`${baseUrl}alice`, {
|
let res = await fetch(`${baseUrl}alice`, {
|
||||||
headers: {
|
headers: {
|
||||||
forwarded: `host=${podHost}`,
|
forwarded: `host=${podHost}`,
|
||||||
authorization: `WebID ${agent.webId}`,
|
authorization: `WebID ${settings.webId}`,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
expect(res.status).toBe(404);
|
expect(res.status).toBe(404);
|
||||||
@ -134,7 +134,7 @@ describe.each(stores)('A subdomain server with %s', (name, { storeUrn, teardown
|
|||||||
method: 'PUT',
|
method: 'PUT',
|
||||||
headers: {
|
headers: {
|
||||||
forwarded: `host=${podHost}`,
|
forwarded: `host=${podHost}`,
|
||||||
authorization: `WebID ${agent.webId}`,
|
authorization: `WebID ${settings.webId}`,
|
||||||
'content-type': 'text/plain',
|
'content-type': 'text/plain',
|
||||||
},
|
},
|
||||||
body: 'this is new data!',
|
body: 'this is new data!',
|
||||||
@ -144,7 +144,7 @@ describe.each(stores)('A subdomain server with %s', (name, { storeUrn, teardown
|
|||||||
res = await fetch(`${baseUrl}alice`, {
|
res = await fetch(`${baseUrl}alice`, {
|
||||||
headers: {
|
headers: {
|
||||||
forwarded: `host=${podHost}`,
|
forwarded: `host=${podHost}`,
|
||||||
authorization: `WebID ${agent.webId}`,
|
authorization: `WebID ${settings.webId}`,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
expect(res.status).toBe(200);
|
expect(res.status).toBe(200);
|
||||||
@ -157,7 +157,7 @@ describe.each(stores)('A subdomain server with %s', (name, { storeUrn, teardown
|
|||||||
headers: {
|
headers: {
|
||||||
'content-type': 'application/json',
|
'content-type': 'application/json',
|
||||||
},
|
},
|
||||||
body: JSON.stringify(agent),
|
body: JSON.stringify(settings),
|
||||||
});
|
});
|
||||||
expect(res.status).toBe(409);
|
expect(res.status).toBe(409);
|
||||||
});
|
});
|
||||||
|
@ -1,14 +1,14 @@
|
|||||||
import type { ResourceIdentifier } from '../../../src/ldp/representation/ResourceIdentifier';
|
import type { ResourceIdentifier } from '../../../src/ldp/representation/ResourceIdentifier';
|
||||||
import type { Agent } from '../../../src/pods/agent/Agent';
|
|
||||||
import type { IdentifierGenerator } from '../../../src/pods/generate/IdentifierGenerator';
|
import type { IdentifierGenerator } from '../../../src/pods/generate/IdentifierGenerator';
|
||||||
import type { Resource, ResourcesGenerator } from '../../../src/pods/generate/ResourcesGenerator';
|
import type { Resource, ResourcesGenerator } from '../../../src/pods/generate/ResourcesGenerator';
|
||||||
import { GeneratedPodManager } from '../../../src/pods/GeneratedPodManager';
|
import { GeneratedPodManager } from '../../../src/pods/GeneratedPodManager';
|
||||||
|
import type { PodSettings } from '../../../src/pods/settings/PodSettings';
|
||||||
import type { ResourceStore } from '../../../src/storage/ResourceStore';
|
import type { ResourceStore } from '../../../src/storage/ResourceStore';
|
||||||
import { ConflictHttpError } from '../../../src/util/errors/ConflictHttpError';
|
import { ConflictHttpError } from '../../../src/util/errors/ConflictHttpError';
|
||||||
|
|
||||||
describe('A GeneratedPodManager', (): void => {
|
describe('A GeneratedPodManager', (): void => {
|
||||||
const base = 'http://test.com/';
|
const base = 'http://test.com/';
|
||||||
let agent: Agent;
|
let settings: PodSettings;
|
||||||
let store: jest.Mocked<ResourceStore>;
|
let store: jest.Mocked<ResourceStore>;
|
||||||
let generatorData: Resource[];
|
let generatorData: Resource[];
|
||||||
const idGenerator: IdentifierGenerator = {
|
const idGenerator: IdentifierGenerator = {
|
||||||
@ -18,7 +18,7 @@ describe('A GeneratedPodManager', (): void => {
|
|||||||
let manager: GeneratedPodManager;
|
let manager: GeneratedPodManager;
|
||||||
|
|
||||||
beforeEach(async(): Promise<void> => {
|
beforeEach(async(): Promise<void> => {
|
||||||
agent = {
|
settings = {
|
||||||
login: 'user',
|
login: 'user',
|
||||||
name: 'first last',
|
name: 'first last',
|
||||||
webId: 'http://secure/webId',
|
webId: 'http://secure/webId',
|
||||||
@ -42,13 +42,13 @@ describe('A GeneratedPodManager', (): void => {
|
|||||||
|
|
||||||
it('throws an error if the generate identifier is not available.', async(): Promise<void> => {
|
it('throws an error if the generate identifier is not available.', async(): Promise<void> => {
|
||||||
store.resourceExists.mockResolvedValueOnce(true);
|
store.resourceExists.mockResolvedValueOnce(true);
|
||||||
const result = manager.createPod(agent);
|
const result = manager.createPod(settings);
|
||||||
await expect(result).rejects.toThrow(`There already is a resource at ${base}user/`);
|
await expect(result).rejects.toThrow(`There already is a resource at ${base}user/`);
|
||||||
await expect(result).rejects.toThrow(ConflictHttpError);
|
await expect(result).rejects.toThrow(ConflictHttpError);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('generates an identifier and writes containers before writing the resources in them.', async(): Promise<void> => {
|
it('generates an identifier and writes containers before writing the resources in them.', async(): Promise<void> => {
|
||||||
await expect(manager.createPod(agent)).resolves.toEqual({ path: `${base}${agent.login}/` });
|
await expect(manager.createPod(settings)).resolves.toEqual({ path: `${base}${settings.login}/` });
|
||||||
|
|
||||||
expect(store.setRepresentation).toHaveBeenCalledTimes(3);
|
expect(store.setRepresentation).toHaveBeenCalledTimes(3);
|
||||||
expect(store.setRepresentation).toHaveBeenNthCalledWith(1, { path: '/path/' }, '/');
|
expect(store.setRepresentation).toHaveBeenNthCalledWith(1, { path: '/path/' }, '/');
|
||||||
|
@ -2,9 +2,9 @@ import type { RequestParser } from '../../../src/ldp/http/RequestParser';
|
|||||||
import { CreatedResponseDescription } from '../../../src/ldp/http/response/CreatedResponseDescription';
|
import { CreatedResponseDescription } from '../../../src/ldp/http/response/CreatedResponseDescription';
|
||||||
import type { ResponseWriter } from '../../../src/ldp/http/ResponseWriter';
|
import type { ResponseWriter } from '../../../src/ldp/http/ResponseWriter';
|
||||||
import type { ResourceIdentifier } from '../../../src/ldp/representation/ResourceIdentifier';
|
import type { ResourceIdentifier } from '../../../src/ldp/representation/ResourceIdentifier';
|
||||||
import type { AgentParser } from '../../../src/pods/agent/AgentParser';
|
|
||||||
import type { PodManager } from '../../../src/pods/PodManager';
|
import type { PodManager } from '../../../src/pods/PodManager';
|
||||||
import { PodManagerHttpHandler } from '../../../src/pods/PodManagerHttpHandler';
|
import { PodManagerHttpHandler } from '../../../src/pods/PodManagerHttpHandler';
|
||||||
|
import type { PodSettingsParser } from '../../../src/pods/settings/PodSettingsParser';
|
||||||
import type { HttpRequest } from '../../../src/server/HttpRequest';
|
import type { HttpRequest } from '../../../src/server/HttpRequest';
|
||||||
import type { HttpResponse } from '../../../src/server/HttpResponse';
|
import type { HttpResponse } from '../../../src/server/HttpResponse';
|
||||||
import { BadRequestHttpError } from '../../../src/util/errors/BadRequestHttpError';
|
import { BadRequestHttpError } from '../../../src/util/errors/BadRequestHttpError';
|
||||||
@ -15,19 +15,19 @@ import { StaticAsyncHandler } from '../../util/StaticAsyncHandler';
|
|||||||
describe('A PodManagerHttpHandler', (): void => {
|
describe('A PodManagerHttpHandler', (): void => {
|
||||||
const requestPath = '/pods';
|
const requestPath = '/pods';
|
||||||
let requestParser: RequestParser;
|
let requestParser: RequestParser;
|
||||||
let agentParser: AgentParser;
|
let podSettingsParser: PodSettingsParser;
|
||||||
let manager: PodManager;
|
let manager: PodManager;
|
||||||
let responseWriter: ResponseWriter;
|
let responseWriter: ResponseWriter;
|
||||||
let handler: PodManagerHttpHandler;
|
let handler: PodManagerHttpHandler;
|
||||||
|
|
||||||
beforeEach(async(): Promise<void> => {
|
beforeEach(async(): Promise<void> => {
|
||||||
requestParser = { handleSafe: jest.fn((): any => 'requestParser') } as any;
|
requestParser = { handleSafe: jest.fn((): any => 'requestParser') } as any;
|
||||||
agentParser = new StaticAsyncHandler(true, 'agentParser' as any);
|
podSettingsParser = new StaticAsyncHandler(true, 'podSettingsParser' as any);
|
||||||
manager = {
|
manager = {
|
||||||
createPod: jest.fn(),
|
createPod: jest.fn(),
|
||||||
};
|
};
|
||||||
responseWriter = { handleSafe: jest.fn((): any => 'response') } as any;
|
responseWriter = { handleSafe: jest.fn((): any => 'response') } as any;
|
||||||
handler = new PodManagerHttpHandler({ requestPath, requestParser, agentParser, manager, responseWriter });
|
handler = new PodManagerHttpHandler({ requestPath, requestParser, podSettingsParser, manager, responseWriter });
|
||||||
});
|
});
|
||||||
|
|
||||||
it('only supports requests to /pods.', async(): Promise<void> => {
|
it('only supports requests to /pods.', async(): Promise<void> => {
|
||||||
|
@ -1,15 +1,15 @@
|
|||||||
import { BasicRepresentation } from '../../../../src/ldp/representation/BasicRepresentation';
|
import { BasicRepresentation } from '../../../../src/ldp/representation/BasicRepresentation';
|
||||||
import type { Representation } from '../../../../src/ldp/representation/Representation';
|
import type { Representation } from '../../../../src/ldp/representation/Representation';
|
||||||
import { RepresentationMetadata } from '../../../../src/ldp/representation/RepresentationMetadata';
|
import { RepresentationMetadata } from '../../../../src/ldp/representation/RepresentationMetadata';
|
||||||
import { AgentJsonParser } from '../../../../src/pods/agent/AgentJsonParser';
|
import { PodSettingsJsonParser } from '../../../../src/pods/settings/PodSettingsJsonParser';
|
||||||
import { BadRequestHttpError } from '../../../../src/util/errors/BadRequestHttpError';
|
import { BadRequestHttpError } from '../../../../src/util/errors/BadRequestHttpError';
|
||||||
import { NotImplementedHttpError } from '../../../../src/util/errors/NotImplementedHttpError';
|
import { NotImplementedHttpError } from '../../../../src/util/errors/NotImplementedHttpError';
|
||||||
import { guardedStreamFrom } from '../../../../src/util/StreamUtil';
|
import { guardedStreamFrom } from '../../../../src/util/StreamUtil';
|
||||||
|
|
||||||
describe('An AgentJsonParser', (): void => {
|
describe('An PodSettingsJsonParser', (): void => {
|
||||||
let metadata: RepresentationMetadata;
|
let metadata: RepresentationMetadata;
|
||||||
let representation: Representation;
|
let representation: Representation;
|
||||||
const parser = new AgentJsonParser();
|
const parser = new PodSettingsJsonParser();
|
||||||
|
|
||||||
beforeEach(async(): Promise<void> => {
|
beforeEach(async(): Promise<void> => {
|
||||||
metadata = new RepresentationMetadata('application/json');
|
metadata = new RepresentationMetadata('application/json');
|
||||||
@ -31,19 +31,7 @@ describe('An AgentJsonParser', (): void => {
|
|||||||
representation.data = guardedStreamFrom([ JSON.stringify({ login: 'login' }) ]);
|
representation.data = guardedStreamFrom([ JSON.stringify({ login: 'login' }) ]);
|
||||||
const result = parser.handle(representation);
|
const result = parser.handle(representation);
|
||||||
await expect(result).rejects.toThrow(BadRequestHttpError);
|
await expect(result).rejects.toThrow(BadRequestHttpError);
|
||||||
await expect(result).rejects.toThrow('Input data is missing Agent key webId');
|
await expect(result).rejects.toThrow('Input data is missing key webId');
|
||||||
});
|
|
||||||
|
|
||||||
it('errors if unknown keys are present.', async(): Promise<void> => {
|
|
||||||
representation.data = guardedStreamFrom([ JSON.stringify({
|
|
||||||
login: 'login',
|
|
||||||
webId: 'webId',
|
|
||||||
name: 'name',
|
|
||||||
unknown: 'unknown',
|
|
||||||
}) ]);
|
|
||||||
const result = parser.handle(representation);
|
|
||||||
await expect(result).rejects.toThrow(BadRequestHttpError);
|
|
||||||
await expect(result).rejects.toThrow('unknown is not a valid Agent key');
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it('generates a User object.', async(): Promise<void> => {
|
it('generates a User object.', async(): Promise<void> => {
|
Loading…
x
Reference in New Issue
Block a user