test: http2 client settings errors
Refs: https://github.com/nodejs/node/issues/14985 PR-URL: https://github.com/nodejs/node/pull/16096 Reviewed-By: James M Snell <jasnell@gmail.com> Reviewed-By: Matteo Collina <matteo.collina@gmail.com>
This commit is contained in:
parent
d64e94ba5b
commit
c3eeb28c6d
84
test/parallel/test-http2-client-settings-errors.js
Normal file
84
test/parallel/test-http2-client-settings-errors.js
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
const {
|
||||||
|
constants,
|
||||||
|
Http2Session,
|
||||||
|
nghttp2ErrorString
|
||||||
|
} = process.binding('http2');
|
||||||
|
const common = require('../common');
|
||||||
|
if (!common.hasCrypto)
|
||||||
|
common.skip('missing crypto');
|
||||||
|
const http2 = require('http2');
|
||||||
|
|
||||||
|
// tests error handling within requestOnConnect
|
||||||
|
// - NGHTTP2_ERR_NOMEM (should emit session error)
|
||||||
|
// - every other NGHTTP2 error from binding (should emit session error)
|
||||||
|
|
||||||
|
const specificTestKeys = [
|
||||||
|
'NGHTTP2_ERR_NOMEM'
|
||||||
|
];
|
||||||
|
|
||||||
|
const specificTests = [
|
||||||
|
{
|
||||||
|
ngError: constants.NGHTTP2_ERR_NOMEM,
|
||||||
|
error: {
|
||||||
|
code: 'ERR_OUTOFMEMORY',
|
||||||
|
type: Error,
|
||||||
|
message: 'Out of memory'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
const genericTests = Object.getOwnPropertyNames(constants)
|
||||||
|
.filter((key) => (
|
||||||
|
key.indexOf('NGHTTP2_ERR') === 0 && specificTestKeys.indexOf(key) < 0
|
||||||
|
))
|
||||||
|
.map((key) => ({
|
||||||
|
ngError: constants[key],
|
||||||
|
error: {
|
||||||
|
code: 'ERR_HTTP2_ERROR',
|
||||||
|
type: Error,
|
||||||
|
message: nghttp2ErrorString(constants[key])
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
|
||||||
|
const tests = specificTests.concat(genericTests);
|
||||||
|
|
||||||
|
const server = http2.createServer(common.mustNotCall());
|
||||||
|
server.on('sessionError', () => {}); // not being tested
|
||||||
|
|
||||||
|
server.listen(0, common.mustCall(() => runTest(tests.shift())));
|
||||||
|
|
||||||
|
function runTest(test) {
|
||||||
|
// mock submitSettings because we only care about testing error handling
|
||||||
|
Http2Session.prototype.submitSettings = () => test.ngError;
|
||||||
|
|
||||||
|
const errorMustCall = common.expectsError(test.error);
|
||||||
|
const errorMustNotCall = common.mustNotCall(
|
||||||
|
`${test.error.code} should emit on session`
|
||||||
|
);
|
||||||
|
|
||||||
|
const url = `http://localhost:${server.address().port}`;
|
||||||
|
|
||||||
|
const client = http2.connect(url, {
|
||||||
|
settings: {
|
||||||
|
maxHeaderListSize: 1
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const req = client.request();
|
||||||
|
req.resume();
|
||||||
|
req.end();
|
||||||
|
|
||||||
|
client.on('error', errorMustCall);
|
||||||
|
req.on('error', errorMustNotCall);
|
||||||
|
|
||||||
|
req.on('end', common.mustCall(() => {
|
||||||
|
client.destroy();
|
||||||
|
if (!tests.length) {
|
||||||
|
server.close();
|
||||||
|
} else {
|
||||||
|
runTest(tests.shift());
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
}
|
@ -8,110 +8,131 @@ const h2 = require('http2');
|
|||||||
|
|
||||||
const server = h2.createServer();
|
const server = h2.createServer();
|
||||||
|
|
||||||
server.on('stream', common.mustCall(onStream));
|
server.on(
|
||||||
|
'stream',
|
||||||
|
common.mustCall((stream) => {
|
||||||
|
const assertSettings = (settings) => {
|
||||||
|
assert.strictEqual(typeof settings, 'object');
|
||||||
|
assert.strictEqual(typeof settings.headerTableSize, 'number');
|
||||||
|
assert.strictEqual(typeof settings.enablePush, 'boolean');
|
||||||
|
assert.strictEqual(typeof settings.initialWindowSize, 'number');
|
||||||
|
assert.strictEqual(typeof settings.maxFrameSize, 'number');
|
||||||
|
assert.strictEqual(typeof settings.maxConcurrentStreams, 'number');
|
||||||
|
assert.strictEqual(typeof settings.maxHeaderListSize, 'number');
|
||||||
|
};
|
||||||
|
|
||||||
function assertSettings(settings) {
|
const localSettings = stream.session.localSettings;
|
||||||
assert.strictEqual(typeof settings, 'object');
|
const remoteSettings = stream.session.remoteSettings;
|
||||||
assert.strictEqual(typeof settings.headerTableSize, 'number');
|
assertSettings(localSettings);
|
||||||
assert.strictEqual(typeof settings.enablePush, 'boolean');
|
assertSettings(remoteSettings);
|
||||||
assert.strictEqual(typeof settings.initialWindowSize, 'number');
|
|
||||||
assert.strictEqual(typeof settings.maxFrameSize, 'number');
|
|
||||||
assert.strictEqual(typeof settings.maxConcurrentStreams, 'number');
|
|
||||||
assert.strictEqual(typeof settings.maxHeaderListSize, 'number');
|
|
||||||
}
|
|
||||||
|
|
||||||
function onStream(stream, headers, flags) {
|
// Test that stored settings are returned when called for second time
|
||||||
|
assert.strictEqual(stream.session.localSettings, localSettings);
|
||||||
|
assert.strictEqual(stream.session.remoteSettings, remoteSettings);
|
||||||
|
|
||||||
const localSettings = stream.session.localSettings;
|
stream.respond({
|
||||||
const remoteSettings = stream.session.remoteSettings;
|
'content-type': 'text/html',
|
||||||
assertSettings(localSettings);
|
':status': 200
|
||||||
assertSettings(remoteSettings);
|
});
|
||||||
|
stream.end('hello world');
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
// Test that stored settings are returned when called for second time
|
server.listen(
|
||||||
assert.strictEqual(stream.session.localSettings, localSettings);
|
0,
|
||||||
assert.strictEqual(stream.session.remoteSettings, remoteSettings);
|
common.mustCall(() => {
|
||||||
|
const client = h2.connect(`http://localhost:${server.address().port}`, {
|
||||||
stream.respond({
|
settings: {
|
||||||
'content-type': 'text/html',
|
enablePush: false,
|
||||||
':status': 200
|
initialWindowSize: 123456
|
||||||
});
|
}
|
||||||
stream.end('hello world');
|
|
||||||
}
|
|
||||||
|
|
||||||
server.listen(0);
|
|
||||||
|
|
||||||
server.on('listening', common.mustCall(() => {
|
|
||||||
|
|
||||||
const client = h2.connect(`http://localhost:${server.address().port}`, {
|
|
||||||
settings: {
|
|
||||||
enablePush: false,
|
|
||||||
initialWindowSize: 123456
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
client.on('localSettings', common.mustCall((settings) => {
|
|
||||||
assert(settings);
|
|
||||||
assert.strictEqual(settings.enablePush, false);
|
|
||||||
assert.strictEqual(settings.initialWindowSize, 123456);
|
|
||||||
assert.strictEqual(settings.maxFrameSize, 16384);
|
|
||||||
}, 2));
|
|
||||||
client.on('remoteSettings', common.mustCall((settings) => {
|
|
||||||
assert(settings);
|
|
||||||
}));
|
|
||||||
|
|
||||||
const headers = { ':path': '/' };
|
|
||||||
|
|
||||||
const req = client.request(headers);
|
|
||||||
|
|
||||||
req.on('connect', common.mustCall(() => {
|
|
||||||
// pendingSettingsAck will be true if a SETTINGS frame
|
|
||||||
// has been sent but we are still waiting for an acknowledgement
|
|
||||||
assert(client.pendingSettingsAck);
|
|
||||||
}));
|
|
||||||
|
|
||||||
// State will only be valid after connect event is emitted
|
|
||||||
req.on('ready', common.mustCall(() => {
|
|
||||||
assert.doesNotThrow(() => {
|
|
||||||
client.settings({
|
|
||||||
maxHeaderListSize: 1
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Verify valid error ranges
|
client.on(
|
||||||
[
|
'localSettings',
|
||||||
['headerTableSize', -1],
|
common.mustCall((settings) => {
|
||||||
['headerTableSize', 2 ** 32],
|
assert(settings);
|
||||||
['initialWindowSize', -1],
|
assert.strictEqual(settings.enablePush, false);
|
||||||
['initialWindowSize', 2 ** 32],
|
assert.strictEqual(settings.initialWindowSize, 123456);
|
||||||
['maxFrameSize', 16383],
|
assert.strictEqual(settings.maxFrameSize, 16384);
|
||||||
['maxFrameSize', 2 ** 24],
|
}, 2)
|
||||||
['maxHeaderListSize', -1],
|
);
|
||||||
['maxHeaderListSize', 2 ** 32]
|
client.on(
|
||||||
].forEach((i) => {
|
'remoteSettings',
|
||||||
const settings = {};
|
common.mustCall((settings) => {
|
||||||
settings[i[0]] = i[1];
|
assert(settings);
|
||||||
assert.throws(() => client.settings(settings),
|
})
|
||||||
common.expectsError({
|
);
|
||||||
code: 'ERR_HTTP2_INVALID_SETTING_VALUE',
|
|
||||||
type: RangeError
|
|
||||||
}));
|
|
||||||
});
|
|
||||||
[1, {}, 'test', [], null, Infinity, NaN].forEach((i) => {
|
|
||||||
assert.throws(() => client.settings({ enablePush: i }),
|
|
||||||
common.expectsError({
|
|
||||||
code: 'ERR_HTTP2_INVALID_SETTING_VALUE',
|
|
||||||
type: TypeError
|
|
||||||
}));
|
|
||||||
});
|
|
||||||
|
|
||||||
}));
|
const headers = { ':path': '/' };
|
||||||
|
|
||||||
req.on('response', common.mustCall());
|
const req = client.request(headers);
|
||||||
req.resume();
|
|
||||||
req.on('end', common.mustCall(() => {
|
|
||||||
server.close();
|
|
||||||
client.destroy();
|
|
||||||
}));
|
|
||||||
req.end();
|
|
||||||
|
|
||||||
}));
|
req.on(
|
||||||
|
'connect',
|
||||||
|
common.mustCall(() => {
|
||||||
|
// pendingSettingsAck will be true if a SETTINGS frame
|
||||||
|
// has been sent but we are still waiting for an acknowledgement
|
||||||
|
assert(client.pendingSettingsAck);
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
// State will only be valid after connect event is emitted
|
||||||
|
req.on(
|
||||||
|
'ready',
|
||||||
|
common.mustCall(() => {
|
||||||
|
assert.doesNotThrow(() => {
|
||||||
|
client.settings({
|
||||||
|
maxHeaderListSize: 1
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// Verify valid error ranges
|
||||||
|
[
|
||||||
|
['headerTableSize', -1],
|
||||||
|
['headerTableSize', 2 ** 32],
|
||||||
|
['initialWindowSize', -1],
|
||||||
|
['initialWindowSize', 2 ** 32],
|
||||||
|
['maxFrameSize', 16383],
|
||||||
|
['maxFrameSize', 2 ** 24],
|
||||||
|
['maxHeaderListSize', -1],
|
||||||
|
['maxHeaderListSize', 2 ** 32]
|
||||||
|
].forEach((i) => {
|
||||||
|
const settings = {};
|
||||||
|
settings[i[0]] = i[1];
|
||||||
|
common.expectsError(
|
||||||
|
() => client.settings(settings),
|
||||||
|
{
|
||||||
|
type: RangeError,
|
||||||
|
code: 'ERR_HTTP2_INVALID_SETTING_VALUE',
|
||||||
|
message: `Invalid value for setting "${i[0]}": ${i[1]}`
|
||||||
|
}
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
// error checks for enablePush
|
||||||
|
[1, {}, 'test', [], null, Infinity, NaN].forEach((i) => {
|
||||||
|
common.expectsError(
|
||||||
|
() => client.settings({ enablePush: i }),
|
||||||
|
{
|
||||||
|
type: TypeError,
|
||||||
|
code: 'ERR_HTTP2_INVALID_SETTING_VALUE',
|
||||||
|
message: `Invalid value for setting "enablePush": ${i}`
|
||||||
|
}
|
||||||
|
);
|
||||||
|
});
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
req.on('response', common.mustCall());
|
||||||
|
req.resume();
|
||||||
|
req.on(
|
||||||
|
'end',
|
||||||
|
common.mustCall(() => {
|
||||||
|
server.close();
|
||||||
|
client.destroy();
|
||||||
|
})
|
||||||
|
);
|
||||||
|
req.end();
|
||||||
|
})
|
||||||
|
);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user