Skip to content

Commit

Permalink
[6.8] configurable global socket timeouts (#31603) (#40906)
Browse files Browse the repository at this point in the history
* configurable global socket timeouts (#31603)

* configurable global socket timeouts

* update snapshots

* update tests

* add test

* add test

* add test

* happy path

* test happy path

* docs

* stop server after

* fix/lint import ordering

* use older new platform apis

* prettier

* config => HttpConfig
  • Loading branch information
jbudz authored Jul 15, 2019
1 parent a213ec3 commit 836bb2d
Show file tree
Hide file tree
Showing 13 changed files with 127 additions and 13 deletions.
6 changes: 6 additions & 0 deletions docs/setup/settings.asciidoc
Original file line number Diff line number Diff line change
Expand Up @@ -209,6 +209,9 @@ Specify the position of the subdomain the URL with the token `{s}`.

`server.host:`:: *Default: "localhost"* This setting specifies the host of the back end server.

`server.keepaliveTimeout:`:: *Default: "120000"* The number of milliseconds to wait for additional data before restarting
the `server.socketTimeout` counter.

`server.maxPayloadBytes:`:: *Default: 1048576* The maximum payload size in bytes for incoming server requests.

`server.name:`:: *Default: "your-hostname"* A human-readable display name that identifies this Kibana instance.
Expand All @@ -217,6 +220,9 @@ Specify the position of the subdomain the URL with the token `{s}`.

`server.ssl.enabled:`:: *Default: "false"* Enables SSL for outgoing requests from the Kibana server to the browser. When set to `true`, `server.ssl.certificate` and `server.ssl.key` are required

`server.socketTimeout:`:: *Default: "120000"* The number of milliseconds to wait before closing an
inactive socket.

`server.ssl.cert:`:: Path to the PEM-format SSL certificate. This file enables
SSL for outgoing requests from the Kibana server to the browser.
deprecated[5.3.0,Replaced by `server.ssl.certificate`]
Expand Down
2 changes: 2 additions & 0 deletions src/core/server/http/__snapshots__/http_config.test.ts.snap
Original file line number Diff line number Diff line change
Expand Up @@ -14,11 +14,13 @@ Object {
"autoListen": true,
"cors": false,
"host": "localhost",
"keepaliveTimeout": 120000,
"maxPayload": ByteSizeValue {
"valueInBytes": 1048576,
},
"port": 5601,
"rewriteBasePath": false,
"socketTimeout": 120000,
"ssl": Object {
"cipherSuites": Array [
"ECDHE-RSA-AES128-GCM-SHA256",
Expand Down
5 changes: 3 additions & 2 deletions src/core/server/http/base_path_proxy_server.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ import { sample } from 'lodash';
import { DevConfig } from '../dev';
import { Logger } from '../logging';
import { HttpConfig } from './http_config';
import { createServer, getServerOptions } from './http_tools';
import { createServer, getListenerOptions, getServerOptions } from './http_tools';

const alphabet = 'abcdefghijklmnopqrztuvwxyz'.split('');

Expand Down Expand Up @@ -62,7 +62,8 @@ export class BasePathProxyServer {
this.log.debug('starting basepath proxy server');

const serverOptions = getServerOptions(this.httpConfig);
this.server = createServer(serverOptions);
const listenerOptions = getListenerOptions(this.httpConfig);
this.server = createServer(serverOptions, listenerOptions);

// Register hapi plugin that adds proxying functionality. It can be configured
// through the route configuration object (see { handler: { proxy: ... } }).
Expand Down
10 changes: 10 additions & 0 deletions src/core/server/http/http_config.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -127,6 +127,16 @@ describe('with TLS', () => {
expect(config.ssl.certificateAuthorities).toBe('/authority/');
});

test('can specify socket timeouts', () => {
const obj = {
keepaliveTimeout: 1e5,
socketTimeout: 5e5,
};
const { keepaliveTimeout, socketTimeout } = HttpConfig.schema.validate(obj);
expect(keepaliveTimeout).toBe(1e5);
expect(socketTimeout).toBe(5e5);
});

test('can specify several `certificateAuthorities`', () => {
const httpSchema = HttpConfig.schema;
const obj = {
Expand Down
10 changes: 10 additions & 0 deletions src/core/server/http/http_config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,12 @@ const createHttpSchema = schema.object(
}),
rewriteBasePath: schema.boolean({ defaultValue: false }),
ssl: SslConfig.schema,
keepaliveTimeout: schema.number({
defaultValue: 120000,
}),
socketTimeout: schema.number({
defaultValue: 120000,
}),
},
{
validate: config => {
Expand Down Expand Up @@ -93,6 +99,8 @@ export class HttpConfig {

public autoListen: boolean;
public host: string;
public keepaliveTimeout: number;
public socketTimeout: number;
public port: number;
public cors: boolean | { origin: string[] };
public maxPayload: ByteSizeValue;
Expand All @@ -113,6 +121,8 @@ export class HttpConfig {
this.basePath = config.basePath;
this.rewriteBasePath = config.rewriteBasePath;
this.publicDir = env.staticFilesDir;
this.keepaliveTimeout = config.keepaliveTimeout;
this.socketTimeout = config.socketTimeout;
this.ssl = new SslConfig(config.ssl);
}
}
5 changes: 3 additions & 2 deletions src/core/server/http/http_server.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ import { Server, ServerOptions } from 'hapi';
import { modifyUrl } from '../../utils';
import { Logger } from '../logging';
import { HttpConfig } from './http_config';
import { createServer, getServerOptions } from './http_tools';
import { createServer, getListenerOptions, getServerOptions } from './http_tools';
import { Router } from './router';

export interface HttpServerInfo {
Expand Down Expand Up @@ -52,7 +52,8 @@ export class HttpServer {
this.log.debug('starting http server');

const serverOptions = getServerOptions(config);
this.server = createServer(serverOptions);
const listenerOptions = getListenerOptions(config);
this.server = createServer(serverOptions, listenerOptions);

this.setupBasePathRewrite(this.server, config);

Expand Down
42 changes: 42 additions & 0 deletions src/core/server/http/http_tools.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,15 @@

import { Request, ResponseToolkit } from 'hapi';
import Joi from 'joi';
import supertest from 'supertest';

import { logger } from '../logging/__mocks__';

import { ByteSizeValue } from '@kbn/config-schema';
import { HttpConfig } from './http_config';
import { HttpServer } from './http_server';
import { defaultValidationErrorHandler, HapiValidationError } from './http_tools';
import { Router } from './router';

const emptyOutput = {
statusCode: 400,
Expand Down Expand Up @@ -57,3 +65,37 @@ describe('defaultValidationErrorHandler', () => {
}
});
});

describe('timeouts', () => {
const server = new HttpServer(logger.get());

test('returns 408 on timeout error', async () => {
const router = new Router('');
router.get({ path: '/a', validate: false }, async (req, res) => {
await new Promise(resolve => setTimeout(resolve, 2000));
return res.ok({});
});
router.get({ path: '/b', validate: false }, async (req, res) => res.ok({}));
server.registerRouter(router);

const config = {
socketTimeout: 1000,
host: '127.0.0.1',
maxPayload: new ByteSizeValue(1024),
ssl: {},
} as HttpConfig;

const { server: innerServer } = await server.start(config);
await supertest(innerServer.listener)
.get('/a')
.expect(408);
await supertest(innerServer.listener)
.get('/b')
.expect(200);
});

afterAll(async () => {
await server.stop();
logger.mockClear();
});
});
27 changes: 23 additions & 4 deletions src/core/server/http/http_tools.ts
Original file line number Diff line number Diff line change
Expand Up @@ -78,11 +78,30 @@ export function getServerOptions(config: HttpConfig, { configureTLS = true } = {
return options;
}

export function createServer(options: ServerOptions) {
const server = new Server(options);
export function getListenerOptions(config: HttpConfig) {
return {
keepaliveTimeout: config.keepaliveTimeout,
socketTimeout: config.socketTimeout,
};
}

interface ListenerOptions {
keepaliveTimeout: number;
socketTimeout: number;
}

export function createServer(serverOptions: ServerOptions, listenerOptions: ListenerOptions) {
const server = new Server(serverOptions);

// Revert to previous 120 seconds keep-alive timeout in Node < 8.
server.listener.keepAliveTimeout = 120e3;
server.listener.keepAliveTimeout = listenerOptions.keepaliveTimeout;
server.listener.setTimeout(listenerOptions.socketTimeout);
server.listener.on('timeout', socket => {
if (socket.writable) {
socket.end(Buffer.from('HTTP/1.1 408 Request Timeout\r\n\r\n', 'ascii'));
} else {
socket.destroy();
}
});
server.listener.on('clientError', (err, socket) => {
if (socket.writable) {
socket.end(Buffer.from('HTTP/1.1 400 Bad Request\r\n\r\n', 'ascii'));
Expand Down
13 changes: 8 additions & 5 deletions src/core/server/http/https_redirect_server.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ import { format as formatUrl } from 'url';

import { Logger } from '../logging';
import { HttpConfig } from './http_config';
import { createServer, getServerOptions } from './http_tools';
import { createServer, getListenerOptions, getServerOptions } from './http_tools';

export class HttpsRedirectServer {
private server?: Server;
Expand All @@ -42,10 +42,13 @@ export class HttpsRedirectServer {
// Redirect server is configured in the same way as any other HTTP server
// within the platform with the only exception that it should always be a
// plain HTTP server, so we just ignore `tls` part of options.
this.server = createServer({
...getServerOptions(config, { configureTLS: false }),
port: config.ssl.redirectHttpFromPort,
});
this.server = createServer(
{
...getServerOptions(config, { configureTLS: false }),
port: config.ssl.redirectHttpFromPort,
},
getListenerOptions(config)
);

this.server.ext('onRequest', (request: Request, responseToolkit: ResponseToolkit) => {
return responseToolkit
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,11 @@ Object {
"basePath": "/abc",
"cors": false,
"host": "host",
"keepaliveTimeout": 5000,
"maxPayload": 1000,
"port": 1234,
"rewriteBasePath": false,
"socketTimeout": 2000,
"ssl": Object {
"enabled": true,
"keyPassphrase": "some-phrase",
Expand All @@ -23,9 +25,11 @@ Object {
"basePath": "/abc",
"cors": false,
"host": "host",
"keepaliveTimeout": 5000,
"maxPayload": 1000,
"port": 1234,
"rewriteBasePath": false,
"socketTimeout": 2000,
"ssl": Object {
"certificate": "cert",
"enabled": true,
Expand All @@ -40,9 +44,11 @@ Object {
"basePath": "/abc",
"cors": false,
"host": "host",
"keepaliveTimeout": 5000,
"maxPayload": 1000,
"port": 1234,
"rewriteBasePath": false,
"socketTimeout": 2000,
"ssl": Object {
"certificate": "deprecated-cert",
"enabled": true,
Expand All @@ -57,9 +63,11 @@ Object {
"basePath": "/abc",
"cors": false,
"host": "host",
"keepaliveTimeout": 5000,
"maxPayload": 1000,
"port": 1234,
"rewriteBasePath": false,
"socketTimeout": 2000,
"ssl": Object {
"certificate": "cert",
"enabled": false,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,8 @@ describe('#get', () => {
host: 'host',
maxPayloadBytes: 1000,
port: 1234,
keepaliveTimeout: 5000,
socketTimeout: 2000,
rewriteBasePath: false,
ssl: { enabled: true, keyPassphrase: 'some-phrase', someNewValue: 'new' },
someNotSupportedValue: 'val',
Expand All @@ -84,6 +86,8 @@ describe('#get', () => {
host: 'host',
maxPayloadBytes: 1000,
port: 1234,
keepaliveTimeout: 5000,
socketTimeout: 2000,
rewriteBasePath: false,
ssl: { enabled: false, certificate: 'cert', key: 'key' },
someNotSupportedValue: 'val',
Expand All @@ -98,6 +102,8 @@ describe('#get', () => {
host: 'host',
maxPayloadBytes: 1000,
port: 1234,
keepaliveTimeout: 5000,
socketTimeout: 2000,
rewriteBasePath: false,
ssl: { enabled: true, cert: 'deprecated-cert', key: 'key' },
someNotSupportedValue: 'val',
Expand All @@ -112,6 +118,8 @@ describe('#get', () => {
host: 'host',
maxPayloadBytes: 1000,
port: 1234,
keepaliveTimeout: 5000,
socketTimeout: 2000,
rewriteBasePath: false,
ssl: { certificate: 'cert', key: 'key' },
someNotSupportedValue: 'val',
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,8 @@ export class LegacyObjectToConfigAdapter extends ObjectToConfigAdapter {
host: configValue.host,
maxPayload: configValue.maxPayloadBytes,
port: configValue.port,
keepaliveTimeout: configValue.keepaliveTimeout,
socketTimeout: configValue.socketTimeout,
rewriteBasePath: configValue.rewriteBasePath,
ssl: configValue.ssl && LegacyObjectToConfigAdapter.transformSSL(configValue.ssl),
};
Expand Down
2 changes: 2 additions & 0 deletions src/server/config/schema.js
Original file line number Diff line number Diff line change
Expand Up @@ -119,6 +119,8 @@ export default () => Joi.object({
name: Joi.string().default(os.hostname()),
host: Joi.string().hostname().default('localhost'),
port: Joi.number().default(5601),
keepaliveTimeout: Joi.number().default(120000),
socketTimeout: Joi.number().default(120000),
maxPayloadBytes: Joi.number().default(1048576),
autoListen: Joi.boolean().default(true),
defaultRoute: Joi.string().default('/app/kibana').regex(/^\//, `start with a slash`),
Expand Down

0 comments on commit 836bb2d

Please sign in to comment.