mirror of
https://github.com/CommunitySolidServer/CommunitySolidServer.git
synced 2024-10-03 14:55:10 +00:00
218 lines
7.5 KiB
TypeScript
218 lines
7.5 KiB
TypeScript
import { mkdirSync } from 'fs';
|
|
import type { Server } from 'http';
|
|
import request from 'supertest';
|
|
import type { BaseHttpServerOptions } from '../../../src/server/BaseHttpServerFactory';
|
|
import { BaseHttpServerFactory } from '../../../src/server/BaseHttpServerFactory';
|
|
import type { HttpHandler } from '../../../src/server/HttpHandler';
|
|
import type { HttpResponse } from '../../../src/server/HttpResponse';
|
|
import { joinFilePath } from '../../../src/util/PathUtil';
|
|
import { getTestFolder, removeFolder } from '../../integration/Config';
|
|
import { getPort, getSocket } from '../../util/Util';
|
|
|
|
const port = getPort('BaseHttpServerFactory');
|
|
|
|
const handler: jest.Mocked<HttpHandler> = {
|
|
handleSafe: jest.fn(async(input: { response: HttpResponse }): Promise<void> => {
|
|
input.response.writeHead(200);
|
|
input.response.end();
|
|
}),
|
|
} as any;
|
|
|
|
describe('A BaseHttpServerFactory', (): void => {
|
|
let server: Server;
|
|
|
|
const options: [string, BaseHttpServerOptions | undefined][] = [
|
|
[ 'http', undefined ],
|
|
[ 'https', {
|
|
https: true,
|
|
key: joinFilePath(__dirname, '../../assets/https/server.key'),
|
|
cert: joinFilePath(__dirname, '../../assets/https/server.cert'),
|
|
}],
|
|
];
|
|
|
|
describe.each(options)('with %s', (protocol, httpOptions): void => {
|
|
let rejectTls: string | undefined;
|
|
beforeAll(async(): Promise<void> => {
|
|
// Allow self-signed certificate
|
|
rejectTls = process.env.NODE_TLS_REJECT_UNAUTHORIZED;
|
|
process.env.NODE_TLS_REJECT_UNAUTHORIZED = '0';
|
|
|
|
const factory = new BaseHttpServerFactory(handler, httpOptions);
|
|
server = factory.startServer(port);
|
|
});
|
|
|
|
beforeEach(async(): Promise<void> => {
|
|
jest.clearAllMocks();
|
|
});
|
|
|
|
afterAll(async(): Promise<void> => {
|
|
process.env.NODE_TLS_REJECT_UNAUTHORIZED = rejectTls;
|
|
server.close();
|
|
});
|
|
|
|
it('sends incoming requests to the handler.', async(): Promise<void> => {
|
|
await request(server).get('/').set('Host', 'test.com').expect(200);
|
|
|
|
expect(handler.handleSafe).toHaveBeenCalledTimes(1);
|
|
expect(handler.handleSafe).toHaveBeenLastCalledWith({
|
|
request: expect.objectContaining({
|
|
headers: expect.objectContaining({ host: 'test.com' }),
|
|
}),
|
|
response: expect.objectContaining({}),
|
|
});
|
|
});
|
|
|
|
it('returns a 404 when the handler does not do anything.', async(): Promise<void> => {
|
|
handler.handleSafe.mockResolvedValueOnce(undefined);
|
|
|
|
await expect(request(server).get('/').expect(404)).resolves.toBeDefined();
|
|
});
|
|
|
|
it('writes an error to the HTTP response without the stack trace.', async(): Promise<void> => {
|
|
handler.handleSafe.mockRejectedValueOnce(new Error('dummyError'));
|
|
|
|
const res = await request(server).get('/').expect(500);
|
|
expect(res.headers['content-type']).toBe('text/plain; charset=utf-8');
|
|
expect(res.text).toBe('Error: dummyError\n');
|
|
});
|
|
|
|
it('does not write an error if the response had been started.', async(): Promise<void> => {
|
|
handler.handleSafe.mockImplementationOnce(async(input: { response: HttpResponse }): Promise<void> => {
|
|
input.response.write('content');
|
|
throw new Error('dummyError');
|
|
});
|
|
|
|
const res = await request(server).get('/');
|
|
expect(res.text).not.toContain('dummyError');
|
|
});
|
|
|
|
it('throws unknown errors if its handler throw non-Error objects.', async(): Promise<void> => {
|
|
handler.handleSafe.mockRejectedValueOnce('apple');
|
|
|
|
const res = await request(server).get('/').expect(500);
|
|
expect(res.text).toContain('Unknown error: apple.');
|
|
});
|
|
|
|
it('can handle errors on the HttpResponse.', async(): Promise<void> => {
|
|
// This just makes sure the logging line is covered.
|
|
// Actually destroying the request to trigger an error causes issues for supertest
|
|
handler.handleSafe.mockImplementationOnce(async(input): Promise<void> => {
|
|
input.request.emit('error', new Error('bad request'));
|
|
});
|
|
await request(server).get('/').expect(404);
|
|
});
|
|
});
|
|
|
|
describe('with showStackTrace enabled', (): void => {
|
|
const httpOptions = {
|
|
http: true,
|
|
showStackTrace: true,
|
|
};
|
|
|
|
beforeAll(async(): Promise<void> => {
|
|
const factory = new BaseHttpServerFactory(handler, httpOptions);
|
|
server = factory.startServer(port);
|
|
});
|
|
|
|
afterAll(async(): Promise<void> => {
|
|
server.close();
|
|
});
|
|
|
|
it('does not print the stack if that option is disabled.', async(): Promise<void> => {
|
|
const error = new Error('dummyError');
|
|
handler.handleSafe.mockRejectedValueOnce(error);
|
|
|
|
const res = await request(server).get('/').expect(500);
|
|
expect(res.headers['content-type']).toBe('text/plain; charset=utf-8');
|
|
expect(res.text).toBe(`${error.stack}\n`);
|
|
});
|
|
});
|
|
|
|
describe('A Base HttpServerFactory (With Unix Sockets)', (): void => {
|
|
const socketFolder = getTestFolder('sockets');
|
|
const socket = joinFilePath(socketFolder, getSocket('BaseHttpServerFactory'));
|
|
const httpOptions = {
|
|
http: true,
|
|
showStackTrace: true,
|
|
};
|
|
|
|
beforeAll(async(): Promise<void> => {
|
|
mkdirSync(socketFolder, { recursive: true });
|
|
});
|
|
|
|
afterAll(async(): Promise<void> => {
|
|
server.close();
|
|
await removeFolder(socketFolder);
|
|
});
|
|
|
|
beforeEach(async(): Promise<void> => {
|
|
jest.clearAllMocks();
|
|
});
|
|
describe('On linux', (): void => {
|
|
if (process.platform === 'win32') {
|
|
return;
|
|
}
|
|
it('sends incoming requests to the handler.', async(): Promise<void> => {
|
|
const factory = new BaseHttpServerFactory(handler, httpOptions);
|
|
server = factory.startServer(socket);
|
|
await request(`http+unix://${socket.replace(/\//gui, '%2F')}`).get('/').set('Host', 'test.com').expect(200);
|
|
|
|
expect(handler.handleSafe).toHaveBeenCalledTimes(1);
|
|
expect(handler.handleSafe).toHaveBeenLastCalledWith({
|
|
request: expect.objectContaining({
|
|
headers: expect.objectContaining({ host: 'test.com' }),
|
|
}),
|
|
response: expect.objectContaining({}),
|
|
});
|
|
});
|
|
|
|
it('throws an error on windows.', async(): Promise<void> => {
|
|
const prevPlatform = process.platform;
|
|
Object.defineProperty(process, 'platform', {
|
|
value: 'win32',
|
|
});
|
|
|
|
const factory = new BaseHttpServerFactory(handler, httpOptions);
|
|
expect((): void => {
|
|
factory.startServer(socket);
|
|
}).toThrow();
|
|
|
|
Object.defineProperty(process, 'platform', {
|
|
value: prevPlatform,
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('On Windows', (): void => {
|
|
if (process.platform !== 'win32') {
|
|
return;
|
|
}
|
|
it('throws an error when trying to start the server on windows.', async(): Promise<void> => {
|
|
const factory = new BaseHttpServerFactory(handler, httpOptions);
|
|
expect((): void => {
|
|
factory.startServer(socket);
|
|
}).toThrow();
|
|
});
|
|
});
|
|
|
|
describe('On any platform', (): void => {
|
|
it('throws an error when trying to start with an invalid socket Path.', async(): Promise<void> => {
|
|
const prevPlatform = process.platform;
|
|
Object.defineProperty(process, 'platform', {
|
|
value: 'linux',
|
|
});
|
|
|
|
const factory = new BaseHttpServerFactory(handler, httpOptions);
|
|
factory.startServer('/fake/path')
|
|
.on('error', (error): void => {
|
|
expect(error).toHaveProperty('code', 'EACCES');
|
|
});
|
|
|
|
Object.defineProperty(process, 'platform', {
|
|
value: prevPlatform,
|
|
});
|
|
});
|
|
});
|
|
});
|
|
});
|