Skip to content

Commit 160af45

Browse files
committedFeb 22, 2019
[fix] Abort the handshake if the Sec-WebSocket-Key header is invalid
1 parent 1d93fb2 commit 160af45

File tree

4 files changed

+59
-23
lines changed

4 files changed

+59
-23
lines changed
 

‎lib/websocket-server.js

+16-8
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,10 @@ const http = require('http');
66

77
const PerMessageDeflate = require('./permessage-deflate');
88
const extension = require('./extension');
9-
const constants = require('./constants');
109
const WebSocket = require('./websocket');
10+
const { GUID } = require('./constants');
11+
12+
const keyRegex = /^[+/0-9A-Za-z]{22}==$/;
1113

1214
/**
1315
* Class representing a WebSocket server.
@@ -176,13 +178,18 @@ class WebSocketServer extends EventEmitter {
176178
handleUpgrade(req, socket, head, cb) {
177179
socket.on('error', socketOnError);
178180

181+
const key =
182+
req.headers['sec-websocket-key'] !== undefined
183+
? req.headers['sec-websocket-key'].trim()
184+
: false;
179185
const version = +req.headers['sec-websocket-version'];
180186
const extensions = {};
181187

182188
if (
183189
req.method !== 'GET' ||
184190
req.headers.upgrade.toLowerCase() !== 'websocket' ||
185-
!req.headers['sec-websocket-key'] ||
191+
!key ||
192+
!keyRegex.test(key) ||
186193
(version !== 8 && version !== 13) ||
187194
!this.shouldHandle(req)
188195
) {
@@ -225,43 +232,44 @@ class WebSocketServer extends EventEmitter {
225232
return abortHandshake(socket, code || 401, message, headers);
226233
}
227234

228-
this.completeUpgrade(extensions, req, socket, head, cb);
235+
this.completeUpgrade(key, extensions, req, socket, head, cb);
229236
});
230237
return;
231238
}
232239

233240
if (!this.options.verifyClient(info)) return abortHandshake(socket, 401);
234241
}
235242

236-
this.completeUpgrade(extensions, req, socket, head, cb);
243+
this.completeUpgrade(key, extensions, req, socket, head, cb);
237244
}
238245

239246
/**
240247
* Upgrade the connection to WebSocket.
241248
*
249+
* @param {String} key The value of the `Sec-WebSocket-Key` header
242250
* @param {Object} extensions The accepted extensions
243251
* @param {http.IncomingMessage} req The request object
244252
* @param {net.Socket} socket The network socket between the server and client
245253
* @param {Buffer} head The first packet of the upgraded stream
246254
* @param {Function} cb Callback
247255
* @private
248256
*/
249-
completeUpgrade(extensions, req, socket, head, cb) {
257+
completeUpgrade(key, extensions, req, socket, head, cb) {
250258
//
251259
// Destroy the socket if the client has already sent a FIN packet.
252260
//
253261
if (!socket.readable || !socket.writable) return socket.destroy();
254262

255-
const key = crypto
263+
const digest = crypto
256264
.createHash('sha1')
257-
.update(req.headers['sec-websocket-key'] + constants.GUID, 'binary')
265+
.update(key + GUID)
258266
.digest('base64');
259267

260268
const headers = [
261269
'HTTP/1.1 101 Switching Protocols',
262270
'Upgrade: websocket',
263271
'Connection: Upgrade',
264-
`Sec-WebSocket-Accept: ${key}`
272+
`Sec-WebSocket-Accept: ${digest}`
265273
];
266274

267275
const ws = new WebSocket(null);

‎lib/websocket.js

+17-11
Original file line numberDiff line numberDiff line change
@@ -11,12 +11,18 @@ const url = require('url');
1111
const PerMessageDeflate = require('./permessage-deflate');
1212
const EventTarget = require('./event-target');
1313
const extension = require('./extension');
14-
const constants = require('./constants');
1514
const Receiver = require('./receiver');
1615
const Sender = require('./sender');
16+
const {
17+
BINARY_TYPES,
18+
EMPTY_BUFFER,
19+
GUID,
20+
kStatusCode,
21+
kWebSocket,
22+
NOOP
23+
} = require('./constants');
1724

1825
const readyStates = ['CONNECTING', 'OPEN', 'CLOSING', 'CLOSED'];
19-
const kWebSocket = constants.kWebSocket;
2026
const protocolVersions = [8, 13];
2127
const closeTimeout = 30 * 1000; // Allow 30 seconds to terminate the connection cleanly.
2228

@@ -39,7 +45,7 @@ class WebSocket extends EventEmitter {
3945
this.readyState = WebSocket.CONNECTING;
4046
this.protocol = '';
4147

42-
this._binaryType = constants.BINARY_TYPES[0];
48+
this._binaryType = BINARY_TYPES[0];
4349
this._closeFrameReceived = false;
4450
this._closeFrameSent = false;
4551
this._closeMessage = '';
@@ -87,7 +93,7 @@ class WebSocket extends EventEmitter {
8793
}
8894

8995
set binaryType(type) {
90-
if (!constants.BINARY_TYPES.includes(type)) return;
96+
if (!BINARY_TYPES.includes(type)) return;
9197

9298
this._binaryType = type;
9399

@@ -265,7 +271,7 @@ class WebSocket extends EventEmitter {
265271

266272
if (typeof data === 'number') data = data.toString();
267273
if (mask === undefined) mask = !this._isServer;
268-
this._sender.ping(data || constants.EMPTY_BUFFER, mask, cb);
274+
this._sender.ping(data || EMPTY_BUFFER, mask, cb);
269275
}
270276

271277
/**
@@ -297,7 +303,7 @@ class WebSocket extends EventEmitter {
297303

298304
if (typeof data === 'number') data = data.toString();
299305
if (mask === undefined) mask = !this._isServer;
300-
this._sender.pong(data || constants.EMPTY_BUFFER, mask, cb);
306+
this._sender.pong(data || EMPTY_BUFFER, mask, cb);
301307
}
302308

303309
/**
@@ -344,7 +350,7 @@ class WebSocket extends EventEmitter {
344350
opts.compress = false;
345351
}
346352

347-
this._sender.send(data || constants.EMPTY_BUFFER, opts, cb);
353+
this._sender.send(data || EMPTY_BUFFER, opts, cb);
348354
}
349355

350356
/**
@@ -574,7 +580,7 @@ function initAsClient(address, protocols, options) {
574580

575581
const digest = crypto
576582
.createHash('sha1')
577-
.update(key + constants.GUID, 'binary')
583+
.update(key + GUID)
578584
.digest('base64');
579585

580586
if (res.headers['sec-websocket-accept'] !== digest) {
@@ -720,7 +726,7 @@ function receiverOnError(err) {
720726
websocket._socket.removeListener('data', socketOnData);
721727

722728
websocket.readyState = WebSocket.CLOSING;
723-
websocket._closeCode = err[constants.kStatusCode];
729+
websocket._closeCode = err[kStatusCode];
724730
websocket.emit('error', err);
725731
websocket._socket.destroy();
726732
}
@@ -753,7 +759,7 @@ function receiverOnMessage(data) {
753759
function receiverOnPing(data) {
754760
const websocket = this[kWebSocket];
755761

756-
websocket.pong(data, !websocket._isServer, constants.NOOP);
762+
websocket.pong(data, !websocket._isServer, NOOP);
757763
websocket.emit('ping', data);
758764
}
759765

@@ -843,7 +849,7 @@ function socketOnError() {
843849
const websocket = this[kWebSocket];
844850

845851
this.removeListener('error', socketOnError);
846-
this.on('error', constants.NOOP);
852+
this.on('error', NOOP);
847853

848854
if (websocket) {
849855
websocket.readyState = WebSocket.CLOSING;

‎test/websocket-server.test.js

+23-1
Original file line numberDiff line numberDiff line change
@@ -383,7 +383,7 @@ describe('WebSocketServer', function() {
383383
});
384384

385385
describe('Connection establishing', function() {
386-
it('fails if the Sec-WebSocket-Key header is invalid', function(done) {
386+
it('fails if the Sec-WebSocket-Key header is invalid (1/2)', function(done) {
387387
const wss = new WebSocket.Server({ port: 0 }, () => {
388388
const req = http.get({
389389
port: wss.address().port,
@@ -404,6 +404,28 @@ describe('WebSocketServer', function() {
404404
});
405405
});
406406

407+
it('fails if the Sec-WebSocket-Key header is invalid (2/2)', function(done) {
408+
const wss = new WebSocket.Server({ port: 0 }, () => {
409+
const req = http.get({
410+
port: wss.address().port,
411+
headers: {
412+
Connection: 'Upgrade',
413+
Upgrade: 'websocket',
414+
'Sec-WebSocket-Key': 'P5l8BJcZwRc='
415+
}
416+
});
417+
418+
req.on('response', (res) => {
419+
assert.strictEqual(res.statusCode, 400);
420+
wss.close(done);
421+
});
422+
});
423+
424+
wss.on('connection', () => {
425+
done(new Error("Unexpected 'connection' event"));
426+
});
427+
});
428+
407429
it('fails is the Sec-WebSocket-Version header is invalid (1/2)', function(done) {
408430
const wss = new WebSocket.Server({ port: 0 }, () => {
409431
const req = http.get({

‎test/websocket.test.js

+3-3
Original file line numberDiff line numberDiff line change
@@ -9,8 +9,8 @@ const http = require('http');
99
const url = require('url');
1010
const fs = require('fs');
1111

12-
const constants = require('../lib/constants');
1312
const WebSocket = require('..');
13+
const { GUID } = require('../lib/constants');
1414

1515
class CustomAgent extends http.Agent {
1616
addRequest() {}
@@ -463,7 +463,7 @@ describe('WebSocket', function() {
463463
server.once('upgrade', (req, socket) => {
464464
const key = crypto
465465
.createHash('sha1')
466-
.update(req.headers['sec-websocket-key'] + constants.GUID, 'binary')
466+
.update(req.headers['sec-websocket-key'] + GUID)
467467
.digest('base64');
468468

469469
socket.end(
@@ -581,7 +581,7 @@ describe('WebSocket', function() {
581581
server.once('upgrade', (req, socket) => {
582582
const key = crypto
583583
.createHash('sha1')
584-
.update(req.headers['sec-websocket-key'] + constants.GUID, 'binary')
584+
.update(req.headers['sec-websocket-key'] + GUID)
585585
.digest('base64');
586586

587587
socket.end(

0 commit comments

Comments
 (0)
Please sign in to comment.