Skip to content

Commit 4abd8f6

Browse files
committedJun 16, 2024
[security] Fix crash when the Upgrade header cannot be read (#2231)
It is possible that the Upgrade header is correctly received and handled (the `'upgrade'` event is emitted) without its value being returned to the user. This can happen if the number of received headers exceed the `server.maxHeadersCount` or `request.maxHeadersCount` threshold. In this case `incomingMessage.headers.upgrade` may not be set. Handle the case correctly and abort the handshake. Fixes #2230
1 parent 36a3f4d commit 4abd8f6

File tree

2 files changed

+44
-1
lines changed

2 files changed

+44
-1
lines changed
 

‎lib/websocket-server.js

+3-1
Original file line numberDiff line numberDiff line change
@@ -161,11 +161,13 @@ class WebSocketServer extends EventEmitter {
161161
handleUpgrade (req, socket, head, cb) {
162162
socket.on('error', socketOnError);
163163

164+
const upgrade = req.headers.upgrade;
164165
const version = +req.headers['sec-websocket-version'];
165166
const extensions = {};
166167

167168
if (
168-
req.method !== 'GET' || req.headers.upgrade.toLowerCase() !== 'websocket' ||
169+
req.method !== 'GET' || upgrade === undefined ||
170+
upgrade.toLowerCase() !== 'websocket' ||
169171
!req.headers['sec-websocket-key'] || (version !== 8 && version !== 13) ||
170172
!this.shouldHandle(req)
171173
) {

‎test/websocket-server.test.js

+41
Original file line numberDiff line numberDiff line change
@@ -364,6 +364,47 @@ describe('WebSocketServer', function () {
364364
});
365365

366366
describe('Connection establishing', function () {
367+
it('fails if the Upgrade header field value cannot be read', (done) => {
368+
const server = http.createServer();
369+
const wss = new WebSocket.Server({ noServer: true });
370+
371+
server.maxHeadersCount = 1;
372+
373+
server.on('upgrade', (req, socket, head) => {
374+
assert.deepStrictEqual(req.headers, { foo: 'bar' });
375+
wss.handleUpgrade(req, socket, head, () => {
376+
done(new Error('Unexpected callback invocation'));
377+
});
378+
});
379+
380+
server.listen(() => {
381+
const req = http.get({
382+
port: server.address().port,
383+
headers: {
384+
foo: 'bar',
385+
bar: 'baz',
386+
Connection: 'Upgrade',
387+
Upgrade: 'websocket'
388+
}
389+
});
390+
391+
req.on('response', (res) => {
392+
assert.strictEqual(res.statusCode, 400);
393+
394+
const chunks = [];
395+
396+
res.on('data', (chunk) => {
397+
chunks.push(chunk);
398+
});
399+
400+
res.on('end', () => {
401+
assert.strictEqual(Buffer.concat(chunks).toString(), 'Bad Request');
402+
server.close(done);
403+
});
404+
});
405+
});
406+
});
407+
367408
it('fails if the Sec-WebSocket-Key header is invalid', function (done) {
368409
const wss = new WebSocket.Server({ port: 0 }, () => {
369410
const req = http.get({

0 commit comments

Comments
 (0)
Please sign in to comment.