CommunitySolidServer/src/authentication/DPoPWebIdExtractor.ts
2021-12-03 12:07:00 +01:00

65 lines
2.6 KiB
TypeScript

import type { RequestMethod } from '@solid/access-token-verifier';
import { createSolidTokenVerifier } from '@solid/access-token-verifier';
import type { TargetExtractor } from '../http/input/identifier/TargetExtractor';
import { getLoggerFor } from '../logging/LogUtil';
import type { HttpRequest } from '../server/HttpRequest';
import { BadRequestHttpError } from '../util/errors/BadRequestHttpError';
import { NotImplementedHttpError } from '../util/errors/NotImplementedHttpError';
import { CredentialGroup } from './Credentials';
import type { CredentialSet } from './Credentials';
import { CredentialsExtractor } from './CredentialsExtractor';
/**
* Credentials extractor that extracts a WebID from a DPoP-bound access token.
*/
export class DPoPWebIdExtractor extends CredentialsExtractor {
private readonly originalUrlExtractor: TargetExtractor;
private readonly verify = createSolidTokenVerifier();
protected readonly logger = getLoggerFor(this);
/**
* @param originalUrlExtractor - Reconstructs the original URL as requested by the client
*/
public constructor(originalUrlExtractor: TargetExtractor) {
super();
this.originalUrlExtractor = originalUrlExtractor;
}
public async canHandle({ headers }: HttpRequest): Promise<void> {
const { authorization } = headers;
if (!authorization || !/^DPoP /ui.test(authorization)) {
throw new NotImplementedHttpError('No DPoP-bound Authorization header specified.');
}
}
public async handle(request: HttpRequest): Promise<CredentialSet> {
const { headers: { authorization, dpop }, method } = request;
if (!dpop) {
throw new BadRequestHttpError('No DPoP header specified.');
}
// Reconstruct the original URL as requested by the client,
// since this is the one it used to authorize the request
const originalUrl = await this.originalUrlExtractor.handleSafe({ request });
// Validate the Authorization and DPoP header headers
// and extract the WebID provided by the client
try {
const { webid: webId } = await this.verify(
authorization!,
{
header: dpop as string,
method: method as RequestMethod,
url: originalUrl.path,
},
);
this.logger.info(`Verified WebID via DPoP-bound access token: ${webId}`);
return { [CredentialGroup.agent]: { webId }};
} catch (error: unknown) {
const message = `Error verifying WebID via DPoP-bound access token: ${(error as Error).message}`;
this.logger.warn(message);
throw new BadRequestHttpError(message, { cause: error });
}
}
}