mirror of
https://github.com/CommunitySolidServer/CommunitySolidServer.git
synced 2024-10-03 14:55:10 +00:00
refactor: Split off ErrorResponseWriter
This commit is contained in:
parent
f4161d406c
commit
e8fdcb0ad0
@ -8,6 +8,7 @@
|
|||||||
"files-scs:config/presets/ldp/metadata-handler.json",
|
"files-scs:config/presets/ldp/metadata-handler.json",
|
||||||
"files-scs:config/presets/ldp/operation-handler.json",
|
"files-scs:config/presets/ldp/operation-handler.json",
|
||||||
"files-scs:config/presets/ldp/permissions-extractor.json",
|
"files-scs:config/presets/ldp/permissions-extractor.json",
|
||||||
|
"files-scs:config/presets/ldp/response-writer.json",
|
||||||
"files-scs:config/presets/ldp/request-parser.json",
|
"files-scs:config/presets/ldp/request-parser.json",
|
||||||
"files-scs:config/presets/logging.json",
|
"files-scs:config/presets/logging.json",
|
||||||
"files-scs:config/presets/representation-conversion.json",
|
"files-scs:config/presets/representation-conversion.json",
|
||||||
|
@ -8,6 +8,7 @@
|
|||||||
"files-scs:config/presets/ldp/metadata-handler.json",
|
"files-scs:config/presets/ldp/metadata-handler.json",
|
||||||
"files-scs:config/presets/ldp/operation-handler.json",
|
"files-scs:config/presets/ldp/operation-handler.json",
|
||||||
"files-scs:config/presets/ldp/permissions-extractor.json",
|
"files-scs:config/presets/ldp/permissions-extractor.json",
|
||||||
|
"files-scs:config/presets/ldp/response-writer.json",
|
||||||
"files-scs:config/presets/ldp/request-parser.json",
|
"files-scs:config/presets/ldp/request-parser.json",
|
||||||
"files-scs:config/presets/logging.json",
|
"files-scs:config/presets/logging.json",
|
||||||
"files-scs:config/presets/representation-conversion.json",
|
"files-scs:config/presets/representation-conversion.json",
|
||||||
|
@ -8,6 +8,7 @@
|
|||||||
"files-scs:config/presets/ldp/metadata-handler.json",
|
"files-scs:config/presets/ldp/metadata-handler.json",
|
||||||
"files-scs:config/presets/ldp/operation-handler.json",
|
"files-scs:config/presets/ldp/operation-handler.json",
|
||||||
"files-scs:config/presets/ldp/permissions-extractor.json",
|
"files-scs:config/presets/ldp/permissions-extractor.json",
|
||||||
|
"files-scs:config/presets/ldp/response-writer.json",
|
||||||
"files-scs:config/presets/ldp/request-parser.json",
|
"files-scs:config/presets/ldp/request-parser.json",
|
||||||
"files-scs:config/presets/logging.json",
|
"files-scs:config/presets/logging.json",
|
||||||
"files-scs:config/presets/representation-conversion.json",
|
"files-scs:config/presets/representation-conversion.json",
|
||||||
|
@ -8,6 +8,7 @@
|
|||||||
"files-scs:config/presets/ldp/metadata-handler.json",
|
"files-scs:config/presets/ldp/metadata-handler.json",
|
||||||
"files-scs:config/presets/ldp/operation-handler.json",
|
"files-scs:config/presets/ldp/operation-handler.json",
|
||||||
"files-scs:config/presets/ldp/permissions-extractor.json",
|
"files-scs:config/presets/ldp/permissions-extractor.json",
|
||||||
|
"files-scs:config/presets/ldp/response-writer.json",
|
||||||
"files-scs:config/presets/ldp/request-parser.json",
|
"files-scs:config/presets/ldp/request-parser.json",
|
||||||
"files-scs:config/presets/logging.json",
|
"files-scs:config/presets/logging.json",
|
||||||
"files-scs:config/presets/representation-conversion.json",
|
"files-scs:config/presets/representation-conversion.json",
|
||||||
|
@ -8,6 +8,7 @@
|
|||||||
"files-scs:config/presets/ldp/metadata-handler.json",
|
"files-scs:config/presets/ldp/metadata-handler.json",
|
||||||
"files-scs:config/presets/ldp/operation-handler.json",
|
"files-scs:config/presets/ldp/operation-handler.json",
|
||||||
"files-scs:config/presets/ldp/permissions-extractor.json",
|
"files-scs:config/presets/ldp/permissions-extractor.json",
|
||||||
|
"files-scs:config/presets/ldp/response-writer.json",
|
||||||
"files-scs:config/presets/ldp/request-parser.json",
|
"files-scs:config/presets/ldp/request-parser.json",
|
||||||
"files-scs:config/presets/logging.json",
|
"files-scs:config/presets/logging.json",
|
||||||
"files-scs:config/presets/representation-conversion.json",
|
"files-scs:config/presets/representation-conversion.json",
|
||||||
|
@ -20,7 +20,7 @@
|
|||||||
"@id": "urn:solid-server:default:OperationHandler"
|
"@id": "urn:solid-server:default:OperationHandler"
|
||||||
},
|
},
|
||||||
"AuthenticatedLdpHandler:_responseWriter": {
|
"AuthenticatedLdpHandler:_responseWriter": {
|
||||||
"@type": "BasicResponseWriter"
|
"@id": "urn:solid-server:default:ResponseWriter"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
17
config/presets/ldp/response-writer.json
Normal file
17
config/presets/ldp/response-writer.json
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
{
|
||||||
|
"@context": "https://linkedsoftwaredependencies.org/bundles/npm/@solid/community-server/^1.0.0/components/context.jsonld",
|
||||||
|
"@graph": [
|
||||||
|
{
|
||||||
|
"@id": "urn:solid-server:default:ResponseWriter",
|
||||||
|
"@type": "CompositeAsyncHandler",
|
||||||
|
"CompositeAsyncHandler:_handlers": [
|
||||||
|
{
|
||||||
|
"@type": "ErrorResponseWriter"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"@type": "BasicResponseWriter"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
1
index.ts
1
index.ts
@ -28,6 +28,7 @@ export * from './src/ldp/http/BasicRequestParser';
|
|||||||
export * from './src/ldp/http/BasicResponseWriter';
|
export * from './src/ldp/http/BasicResponseWriter';
|
||||||
export * from './src/ldp/http/BasicTargetExtractor';
|
export * from './src/ldp/http/BasicTargetExtractor';
|
||||||
export * from './src/ldp/http/BodyParser';
|
export * from './src/ldp/http/BodyParser';
|
||||||
|
export * from './src/ldp/http/ErrorResponseWriter';
|
||||||
export * from './src/ldp/http/Patch';
|
export * from './src/ldp/http/Patch';
|
||||||
export * from './src/ldp/http/PreferenceParser';
|
export * from './src/ldp/http/PreferenceParser';
|
||||||
export * from './src/ldp/http/RawBodyParser';
|
export * from './src/ldp/http/RawBodyParser';
|
||||||
|
@ -1,49 +1,37 @@
|
|||||||
import { getLoggerFor } from '../../logging/LogUtil';
|
import { getLoggerFor } from '../../logging/LogUtil';
|
||||||
import type { HttpResponse } from '../../server/HttpResponse';
|
import type { HttpResponse } from '../../server/HttpResponse';
|
||||||
import { HttpError } from '../../util/errors/HttpError';
|
|
||||||
import { UnsupportedHttpError } from '../../util/errors/UnsupportedHttpError';
|
import { UnsupportedHttpError } from '../../util/errors/UnsupportedHttpError';
|
||||||
import type { ResponseDescription } from '../operations/ResponseDescription';
|
import type { ResponseDescription } from '../operations/ResponseDescription';
|
||||||
import { ResponseWriter } from './ResponseWriter';
|
import { ResponseWriter } from './ResponseWriter';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Writes to an {@link HttpResponse} based on the incoming {@link ResponseDescription} or error.
|
* Writes to an {@link HttpResponse} based on the incoming {@link ResponseDescription}.
|
||||||
* Still needs a way to write correct status codes for successful operations.
|
* Still needs a way to write correct status codes for successful operations.
|
||||||
*/
|
*/
|
||||||
export class BasicResponseWriter extends ResponseWriter {
|
export class BasicResponseWriter extends ResponseWriter {
|
||||||
protected readonly logger = getLoggerFor(this);
|
protected readonly logger = getLoggerFor(this);
|
||||||
|
|
||||||
public async canHandle(input: { response: HttpResponse; result: ResponseDescription | Error }): Promise<void> {
|
public async canHandle(input: { response: HttpResponse; result: ResponseDescription | Error }): Promise<void> {
|
||||||
if (!(input.result instanceof Error) && input.result.body && !input.result.body.binary) {
|
if ((input.result instanceof Error) || (input.result.body && !input.result.body.binary)) {
|
||||||
this.logger.warn('This writer can only write binary bodies and errors');
|
this.logger.warn('This writer can only write binary bodies');
|
||||||
throw new UnsupportedHttpError('Only binary results and errors are supported');
|
throw new UnsupportedHttpError('Only binary results are supported');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public async handle(input: { response: HttpResponse; result: ResponseDescription | Error }): Promise<void> {
|
public async handle(input: { response: HttpResponse; result: ResponseDescription }): Promise<void> {
|
||||||
if (input.result instanceof Error) {
|
|
||||||
let code = 500;
|
|
||||||
if (input.result instanceof HttpError) {
|
|
||||||
code = input.result.statusCode;
|
|
||||||
}
|
|
||||||
input.response.setHeader('content-type', 'text/plain');
|
|
||||||
input.response.writeHead(code);
|
|
||||||
input.response.end(typeof input.result.stack === 'string' ?
|
|
||||||
`${input.result.stack}\n` :
|
|
||||||
`${input.result.name}: ${input.result.message}\n`);
|
|
||||||
} else {
|
|
||||||
input.response.setHeader('location', input.result.identifier.path);
|
input.response.setHeader('location', input.result.identifier.path);
|
||||||
if (input.result.body) {
|
if (input.result.body) {
|
||||||
const contentType = input.result.body.metadata.contentType ?? 'text/plain';
|
const contentType = input.result.body.metadata.contentType ?? 'text/plain';
|
||||||
input.response.setHeader('content-type', contentType);
|
input.response.setHeader('content-type', contentType);
|
||||||
input.result.body.data.pipe(input.response);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
input.response.writeHead(200);
|
input.response.writeHead(200);
|
||||||
|
|
||||||
if (!input.result.body) {
|
if (input.result.body) {
|
||||||
|
input.result.body.data.pipe(input.response);
|
||||||
|
} else {
|
||||||
// If there is an input body the response will end once the input stream ends
|
// If there is an input body the response will end once the input stream ends
|
||||||
input.response.end();
|
input.response.end();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
32
src/ldp/http/ErrorResponseWriter.ts
Normal file
32
src/ldp/http/ErrorResponseWriter.ts
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
import { getLoggerFor } from '../../logging/LogUtil';
|
||||||
|
import type { HttpResponse } from '../../server/HttpResponse';
|
||||||
|
import { HttpError } from '../../util/errors/HttpError';
|
||||||
|
import { UnsupportedHttpError } from '../../util/errors/UnsupportedHttpError';
|
||||||
|
import type { ResponseDescription } from '../operations/ResponseDescription';
|
||||||
|
import { ResponseWriter } from './ResponseWriter';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Writes to an {@link HttpResponse} based on the incoming Error.
|
||||||
|
*/
|
||||||
|
export class ErrorResponseWriter extends ResponseWriter {
|
||||||
|
protected readonly logger = getLoggerFor(this);
|
||||||
|
|
||||||
|
public async canHandle(input: { response: HttpResponse; result: ResponseDescription | Error }): Promise<void> {
|
||||||
|
if (!(input.result instanceof Error)) {
|
||||||
|
this.logger.warn('This writer can only write errors');
|
||||||
|
throw new UnsupportedHttpError('Only errors are supported');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public async handle(input: { response: HttpResponse; result: Error }): Promise<void> {
|
||||||
|
let code = 500;
|
||||||
|
if (input.result instanceof HttpError) {
|
||||||
|
code = input.result.statusCode;
|
||||||
|
}
|
||||||
|
input.response.setHeader('content-type', 'text/plain');
|
||||||
|
input.response.writeHead(code);
|
||||||
|
input.response.end(typeof input.result.stack === 'string' ?
|
||||||
|
`${input.result.stack}\n` :
|
||||||
|
`${input.result.name}: ${input.result.message}\n`);
|
||||||
|
}
|
||||||
|
}
|
@ -5,7 +5,6 @@ import type {
|
|||||||
} from '../../index';
|
} from '../../index';
|
||||||
import {
|
import {
|
||||||
AuthenticatedLdpHandler,
|
AuthenticatedLdpHandler,
|
||||||
BasicResponseWriter,
|
|
||||||
CompositeAsyncHandler,
|
CompositeAsyncHandler,
|
||||||
MethodPermissionsExtractor,
|
MethodPermissionsExtractor,
|
||||||
RdfToQuadConverter,
|
RdfToQuadConverter,
|
||||||
@ -19,6 +18,7 @@ import {
|
|||||||
getOperationHandler,
|
getOperationHandler,
|
||||||
getWebAclAuthorizer,
|
getWebAclAuthorizer,
|
||||||
getDataAccessorStore,
|
getDataAccessorStore,
|
||||||
|
getResponseWriter,
|
||||||
} from './Util';
|
} from './Util';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -50,7 +50,7 @@ export class AuthenticatedDataAccessorBasedConfig implements ServerConfig {
|
|||||||
|
|
||||||
const operationHandler = getOperationHandler(this.store);
|
const operationHandler = getOperationHandler(this.store);
|
||||||
|
|
||||||
const responseWriter = new BasicResponseWriter();
|
const responseWriter = getResponseWriter();
|
||||||
const authorizer = getWebAclAuthorizer(this.store, this.base);
|
const authorizer = getWebAclAuthorizer(this.store, this.base);
|
||||||
|
|
||||||
const handler = new AuthenticatedLdpHandler({
|
const handler = new AuthenticatedLdpHandler({
|
||||||
|
@ -3,12 +3,11 @@ import type { HttpHandler,
|
|||||||
import {
|
import {
|
||||||
AllowEverythingAuthorizer,
|
AllowEverythingAuthorizer,
|
||||||
AuthenticatedLdpHandler,
|
AuthenticatedLdpHandler,
|
||||||
BasicResponseWriter,
|
|
||||||
MethodPermissionsExtractor,
|
MethodPermissionsExtractor,
|
||||||
UnsecureWebIdExtractor,
|
UnsecureWebIdExtractor,
|
||||||
} from '../../index';
|
} from '../../index';
|
||||||
import type { ServerConfig } from './ServerConfig';
|
import type { ServerConfig } from './ServerConfig';
|
||||||
import { getOperationHandler, getInMemoryResourceStore, getBasicRequestParser } from './Util';
|
import { getOperationHandler, getInMemoryResourceStore, getBasicRequestParser, getResponseWriter } from './Util';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* BasicConfig works with
|
* BasicConfig works with
|
||||||
@ -33,7 +32,7 @@ export class BasicConfig implements ServerConfig {
|
|||||||
|
|
||||||
const operationHandler = getOperationHandler(this.store);
|
const operationHandler = getOperationHandler(this.store);
|
||||||
|
|
||||||
const responseWriter = new BasicResponseWriter();
|
const responseWriter = getResponseWriter();
|
||||||
|
|
||||||
const handler = new AuthenticatedLdpHandler({
|
const handler = new AuthenticatedLdpHandler({
|
||||||
requestParser,
|
requestParser,
|
||||||
|
@ -3,7 +3,6 @@ import type { HttpHandler,
|
|||||||
import {
|
import {
|
||||||
AllowEverythingAuthorizer,
|
AllowEverythingAuthorizer,
|
||||||
AuthenticatedLdpHandler,
|
AuthenticatedLdpHandler,
|
||||||
BasicResponseWriter,
|
|
||||||
CompositeAsyncHandler,
|
CompositeAsyncHandler,
|
||||||
MethodPermissionsExtractor,
|
MethodPermissionsExtractor,
|
||||||
QuadToRdfConverter,
|
QuadToRdfConverter,
|
||||||
@ -21,6 +20,7 @@ import {
|
|||||||
getConvertingStore,
|
getConvertingStore,
|
||||||
getPatchingStore,
|
getPatchingStore,
|
||||||
getBasicRequestParser,
|
getBasicRequestParser,
|
||||||
|
getResponseWriter,
|
||||||
} from './Util';
|
} from './Util';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -56,7 +56,7 @@ export class BasicHandlersConfig implements ServerConfig {
|
|||||||
|
|
||||||
const operationHandler = getOperationHandler(this.store);
|
const operationHandler = getOperationHandler(this.store);
|
||||||
|
|
||||||
const responseWriter = new BasicResponseWriter();
|
const responseWriter = getResponseWriter();
|
||||||
|
|
||||||
const handler = new AuthenticatedLdpHandler({
|
const handler = new AuthenticatedLdpHandler({
|
||||||
requestParser,
|
requestParser,
|
||||||
|
@ -2,7 +2,6 @@ import type { HttpHandler,
|
|||||||
ResourceStore } from '../../index';
|
ResourceStore } from '../../index';
|
||||||
import {
|
import {
|
||||||
AuthenticatedLdpHandler,
|
AuthenticatedLdpHandler,
|
||||||
BasicResponseWriter,
|
|
||||||
CompositeAsyncHandler,
|
CompositeAsyncHandler,
|
||||||
MethodPermissionsExtractor,
|
MethodPermissionsExtractor,
|
||||||
RdfToQuadConverter,
|
RdfToQuadConverter,
|
||||||
@ -16,6 +15,7 @@ import {
|
|||||||
getBasicRequestParser,
|
getBasicRequestParser,
|
||||||
getOperationHandler,
|
getOperationHandler,
|
||||||
getWebAclAuthorizer,
|
getWebAclAuthorizer,
|
||||||
|
getResponseWriter,
|
||||||
} from './Util';
|
} from './Util';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -46,7 +46,7 @@ export class BasicHandlersWithAclConfig implements ServerConfig {
|
|||||||
|
|
||||||
const operationHandler = getOperationHandler(this.store);
|
const operationHandler = getOperationHandler(this.store);
|
||||||
|
|
||||||
const responseWriter = new BasicResponseWriter();
|
const responseWriter = getResponseWriter();
|
||||||
const authorizer = getWebAclAuthorizer(this.store);
|
const authorizer = getWebAclAuthorizer(this.store);
|
||||||
|
|
||||||
const handler = new AuthenticatedLdpHandler({
|
const handler = new AuthenticatedLdpHandler({
|
||||||
|
@ -6,7 +6,6 @@ import type {
|
|||||||
import {
|
import {
|
||||||
AllowEverythingAuthorizer,
|
AllowEverythingAuthorizer,
|
||||||
AuthenticatedLdpHandler,
|
AuthenticatedLdpHandler,
|
||||||
BasicResponseWriter,
|
|
||||||
CompositeAsyncHandler,
|
CompositeAsyncHandler,
|
||||||
MethodPermissionsExtractor,
|
MethodPermissionsExtractor,
|
||||||
QuadToRdfConverter,
|
QuadToRdfConverter,
|
||||||
@ -20,6 +19,7 @@ import {
|
|||||||
getConvertingStore,
|
getConvertingStore,
|
||||||
getBasicRequestParser,
|
getBasicRequestParser,
|
||||||
getDataAccessorStore,
|
getDataAccessorStore,
|
||||||
|
getResponseWriter,
|
||||||
} from './Util';
|
} from './Util';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -50,7 +50,7 @@ export class DataAccessorBasedConfig implements ServerConfig {
|
|||||||
const authorizer = new AllowEverythingAuthorizer();
|
const authorizer = new AllowEverythingAuthorizer();
|
||||||
|
|
||||||
const operationHandler = getOperationHandler(this.store);
|
const operationHandler = getOperationHandler(this.store);
|
||||||
const responseWriter = new BasicResponseWriter();
|
const responseWriter = getResponseWriter();
|
||||||
|
|
||||||
const handler = new AuthenticatedLdpHandler({
|
const handler = new AuthenticatedLdpHandler({
|
||||||
requestParser,
|
requestParser,
|
||||||
|
@ -6,16 +6,20 @@ import type {
|
|||||||
RepresentationConverter,
|
RepresentationConverter,
|
||||||
ResourceStore,
|
ResourceStore,
|
||||||
ResponseDescription,
|
ResponseDescription,
|
||||||
|
HttpResponse,
|
||||||
|
ResponseWriter,
|
||||||
} from '../../index';
|
} from '../../index';
|
||||||
import {
|
import {
|
||||||
AcceptPreferenceParser,
|
AcceptPreferenceParser,
|
||||||
BasicMetadataExtractor,
|
BasicMetadataExtractor,
|
||||||
BasicRequestParser,
|
BasicRequestParser,
|
||||||
|
BasicResponseWriter,
|
||||||
BasicTargetExtractor,
|
BasicTargetExtractor,
|
||||||
CompositeAsyncHandler,
|
CompositeAsyncHandler,
|
||||||
ContentTypeParser,
|
ContentTypeParser,
|
||||||
DataAccessorBasedStore,
|
DataAccessorBasedStore,
|
||||||
DeleteOperationHandler,
|
DeleteOperationHandler,
|
||||||
|
ErrorResponseWriter,
|
||||||
GetOperationHandler,
|
GetOperationHandler,
|
||||||
HeadOperationHandler,
|
HeadOperationHandler,
|
||||||
InMemoryDataAccessor,
|
InMemoryDataAccessor,
|
||||||
@ -113,6 +117,12 @@ export const getOperationHandler = (store: ResourceStore): CompositeAsyncHandler
|
|||||||
return new CompositeAsyncHandler<Operation, ResponseDescription>(handlers);
|
return new CompositeAsyncHandler<Operation, ResponseDescription>(handlers);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const getResponseWriter = (): ResponseWriter =>
|
||||||
|
new CompositeAsyncHandler<{ response: HttpResponse; result: ResponseDescription | Error }, void>([
|
||||||
|
new ErrorResponseWriter(),
|
||||||
|
new BasicResponseWriter(),
|
||||||
|
]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a BasicMetadataExtractor with parsers for content-type, slugs and link types.
|
* Creates a BasicMetadataExtractor with parsers for content-type, slugs and link types.
|
||||||
*/
|
*/
|
||||||
|
@ -16,7 +16,9 @@ describe('A BasicResponseWriter', (): void => {
|
|||||||
response = createResponse({ eventEmitter: EventEmitter });
|
response = createResponse({ eventEmitter: EventEmitter });
|
||||||
});
|
});
|
||||||
|
|
||||||
it('requires the description body to be a string or binary stream if present.', async(): Promise<void> => {
|
it('requires a description body, with a binary stream if there is data.', async(): Promise<void> => {
|
||||||
|
await expect(writer.canHandle({ response, result: new Error('error') }))
|
||||||
|
.rejects.toThrow(UnsupportedHttpError);
|
||||||
await expect(writer.canHandle({ response, result: { body: { binary: false }} as ResponseDescription }))
|
await expect(writer.canHandle({ response, result: { body: { binary: false }} as ResponseDescription }))
|
||||||
.rejects.toThrow(UnsupportedHttpError);
|
.rejects.toThrow(UnsupportedHttpError);
|
||||||
await expect(writer.canHandle({ response, result: { body: { binary: true }} as ResponseDescription }))
|
await expect(writer.canHandle({ response, result: { body: { binary: true }} as ResponseDescription }))
|
||||||
@ -72,34 +74,4 @@ describe('A BasicResponseWriter', (): void => {
|
|||||||
await writer.handle({ response, result: { identifier: { path: 'path' }, body }});
|
await writer.handle({ response, result: { identifier: { path: 'path' }, body }});
|
||||||
await end;
|
await end;
|
||||||
});
|
});
|
||||||
|
|
||||||
it('responds with 500 if an error if there is an error.', async(): Promise<void> => {
|
|
||||||
await writer.handle({ response, result: new Error('error') });
|
|
||||||
expect(response._isEndCalled()).toBeTruthy();
|
|
||||||
expect(response._getStatusCode()).toBe(500);
|
|
||||||
expect(response._getData()).toMatch('Error: error');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('responds with the given statuscode if there is an HttpError.', async(): Promise<void> => {
|
|
||||||
const error = new UnsupportedHttpError('error');
|
|
||||||
await writer.handle({ response, result: error });
|
|
||||||
expect(response._isEndCalled()).toBeTruthy();
|
|
||||||
expect(response._getStatusCode()).toBe(error.statusCode);
|
|
||||||
expect(response._getData()).toMatch('UnsupportedHttpError: error');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('responds with the error name and message when no stack trace is lazily generated.', async(): Promise<void> => {
|
|
||||||
const error = new Error('error');
|
|
||||||
error.stack = undefined;
|
|
||||||
await writer.handle({ response, result: error });
|
|
||||||
expect(response._isEndCalled()).toBeTruthy();
|
|
||||||
expect(response._getStatusCode()).toBe(500);
|
|
||||||
expect(response._getData()).toMatch('Error: error');
|
|
||||||
});
|
|
||||||
|
|
||||||
it('ends its response with a newline if there is an error.', async(): Promise<void> => {
|
|
||||||
await writer.handle({ response, result: new Error('error') });
|
|
||||||
expect(response._isEndCalled()).toBeTruthy();
|
|
||||||
expect(response._getData().endsWith('\n')).toBeTruthy();
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
54
test/unit/ldp/http/ErrorResponseWriter.test.ts
Normal file
54
test/unit/ldp/http/ErrorResponseWriter.test.ts
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
import { EventEmitter } from 'events';
|
||||||
|
import type { MockResponse } from 'node-mocks-http';
|
||||||
|
import { createResponse } from 'node-mocks-http';
|
||||||
|
import { ErrorResponseWriter } from '../../../../src/ldp/http/ErrorResponseWriter';
|
||||||
|
import type { ResponseDescription } from '../../../../src/ldp/operations/ResponseDescription';
|
||||||
|
import { UnsupportedHttpError } from '../../../../src/util/errors/UnsupportedHttpError';
|
||||||
|
|
||||||
|
describe('An ErrorResponseWriter', (): void => {
|
||||||
|
const writer = new ErrorResponseWriter();
|
||||||
|
let response: MockResponse<any>;
|
||||||
|
|
||||||
|
beforeEach(async(): Promise<void> => {
|
||||||
|
response = createResponse({ eventEmitter: EventEmitter });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('requires the input to be an error.', async(): Promise<void> => {
|
||||||
|
await expect(writer.canHandle({ response, result: new Error('error') }))
|
||||||
|
.resolves.toBeUndefined();
|
||||||
|
await expect(writer.canHandle({ response, result: { body: { binary: false }} as ResponseDescription }))
|
||||||
|
.rejects.toThrow(UnsupportedHttpError);
|
||||||
|
await expect(writer.canHandle({ response, result: { body: { binary: true }} as ResponseDescription }))
|
||||||
|
.rejects.toThrow(UnsupportedHttpError);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('responds with 500 if an error if there is an error.', async(): Promise<void> => {
|
||||||
|
await writer.handle({ response, result: new Error('error') });
|
||||||
|
expect(response._isEndCalled()).toBeTruthy();
|
||||||
|
expect(response._getStatusCode()).toBe(500);
|
||||||
|
expect(response._getData()).toMatch('Error: error');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('responds with the given statuscode if there is an HttpError.', async(): Promise<void> => {
|
||||||
|
const error = new UnsupportedHttpError('error');
|
||||||
|
await writer.handle({ response, result: error });
|
||||||
|
expect(response._isEndCalled()).toBeTruthy();
|
||||||
|
expect(response._getStatusCode()).toBe(error.statusCode);
|
||||||
|
expect(response._getData()).toMatch('UnsupportedHttpError: error');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('responds with the error name and message when no stack trace is lazily generated.', async(): Promise<void> => {
|
||||||
|
const error = new Error('error');
|
||||||
|
error.stack = undefined;
|
||||||
|
await writer.handle({ response, result: error });
|
||||||
|
expect(response._isEndCalled()).toBeTruthy();
|
||||||
|
expect(response._getStatusCode()).toBe(500);
|
||||||
|
expect(response._getData()).toMatch('Error: error');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('ends its response with a newline if there is an error.', async(): Promise<void> => {
|
||||||
|
await writer.handle({ response, result: new Error('error') });
|
||||||
|
expect(response._isEndCalled()).toBeTruthy();
|
||||||
|
expect(response._getData().endsWith('\n')).toBeTruthy();
|
||||||
|
});
|
||||||
|
});
|
Loading…
x
Reference in New Issue
Block a user