pax_global_header00006660000000000000000000000064135751235310014517gustar00rootroot0000000000000052 comment=9531cd0f20be0b0cd059443c31ea4bda2670c7f1 ws-7.2.1/000077500000000000000000000000001357512353100121575ustar00rootroot00000000000000ws-7.2.1/.eslintrc.yaml000066400000000000000000000004641357512353100147500ustar00rootroot00000000000000env: browser: true es6: true mocha: true node: true extends: - eslint:recommended - prettier parserOptions: ecmaVersion: 9 plugins: - prettier rules: no-console: off no-var: error prefer-const: error prettier/prettier: error quotes: - error - single - avoidEscape: true ws-7.2.1/.gitattributes000066400000000000000000000000231357512353100150450ustar00rootroot00000000000000* text=auto eol=lf ws-7.2.1/.gitignore000066400000000000000000000000561357512353100141500ustar00rootroot00000000000000node_modules/ .nyc_output/ coverage/ .vscode/ ws-7.2.1/.npmrc000066400000000000000000000000231357512353100132720ustar00rootroot00000000000000package-lock=false ws-7.2.1/.prettierrc.yaml000066400000000000000000000001061357512353100153010ustar00rootroot00000000000000arrowParens: always endOfLine: lf proseWrap: always singleQuote: true ws-7.2.1/.travis.yml000066400000000000000000000002371357512353100142720ustar00rootroot00000000000000language: node_js node_js: - '13' - '12' - '10' - '8' os: - linux - osx - windows after_success: - nyc report --reporter=text-lcov | coveralls ws-7.2.1/ISSUE_TEMPLATE.md000066400000000000000000000012761357512353100146720ustar00rootroot00000000000000 - [ ] I've searched for any related issues and avoided creating a duplicate issue. #### Description #### Reproducible in: - version: - Node.js version(s): - OS version(s): #### Steps to reproduce: 1. 2. 3. #### Expected result: #### Actual result: #### Attachments: ws-7.2.1/LICENSE000066400000000000000000000021221357512353100131610ustar00rootroot00000000000000The MIT License (MIT) Copyright (c) 2011 Einar Otto Stangvik Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ws-7.2.1/README.md000066400000000000000000000327111357512353100134420ustar00rootroot00000000000000# ws: a Node.js WebSocket library [![Version npm](https://img.shields.io/npm/v/ws.svg?logo=npm)](https://www.npmjs.com/package/ws) [![Linux Build](https://img.shields.io/travis/websockets/ws/master.svg?logo=travis)](https://travis-ci.com/websockets/ws) [![Windows Build](https://img.shields.io/appveyor/ci/lpinca/ws/master.svg?logo=appveyor)](https://ci.appveyor.com/project/lpinca/ws) [![Coverage Status](https://img.shields.io/coveralls/websockets/ws/master.svg)](https://coveralls.io/github/websockets/ws) ws is a simple to use, blazing fast, and thoroughly tested WebSocket client and server implementation. Passes the quite extensive Autobahn test suite: [server][server-report], [client][client-report]. **Note**: This module does not work in the browser. The client in the docs is a reference to a back end with the role of a client in the WebSocket communication. Browser clients must use the native [`WebSocket`](https://developer.mozilla.org/en-US/docs/Web/API/WebSocket) object. To make the same code work seamlessly on Node.js and the browser, you can use one of the many wrappers available on npm, like [isomorphic-ws](https://github.com/heineiuo/isomorphic-ws). ## Table of Contents - [Protocol support](#protocol-support) - [Installing](#installing) - [Opt-in for performance and spec compliance](#opt-in-for-performance-and-spec-compliance) - [API docs](#api-docs) - [WebSocket compression](#websocket-compression) - [Usage examples](#usage-examples) - [Sending and receiving text data](#sending-and-receiving-text-data) - [Sending binary data](#sending-binary-data) - [Simple server](#simple-server) - [External HTTP/S server](#external-https-server) - [Multiple servers sharing a single HTTP/S server](#multiple-servers-sharing-a-single-https-server) - [Client authentication](#client-authentication) - [Server broadcast](#server-broadcast) - [echo.websocket.org demo](#echowebsocketorg-demo) - [Use the Node.js streams API](#use-the-nodejs-streams-api) - [Other examples](#other-examples) - [FAQ](#faq) - [How to get the IP address of the client?](#how-to-get-the-ip-address-of-the-client) - [How to detect and close broken connections?](#how-to-detect-and-close-broken-connections) - [How to connect via a proxy?](#how-to-connect-via-a-proxy) - [Changelog](#changelog) - [License](#license) ## Protocol support - **HyBi drafts 07-12** (Use the option `protocolVersion: 8`) - **HyBi drafts 13-17** (Current default, alternatively option `protocolVersion: 13`) ## Installing ``` npm install ws ``` ### Opt-in for performance and spec compliance There are 2 optional modules that can be installed along side with the ws module. These modules are binary addons which improve certain operations. Prebuilt binaries are available for the most popular platforms so you don't necessarily need to have a C++ compiler installed on your machine. - `npm install --save-optional bufferutil`: Allows to efficiently perform operations such as masking and unmasking the data payload of the WebSocket frames. - `npm install --save-optional utf-8-validate`: Allows to efficiently check if a message contains valid UTF-8 as required by the spec. ## API docs See [`/doc/ws.md`](./doc/ws.md) for Node.js-like documentation of ws classes and utility functions. ## WebSocket compression ws supports the [permessage-deflate extension][permessage-deflate] which enables the client and server to negotiate a compression algorithm and its parameters, and then selectively apply it to the data payloads of each WebSocket message. The extension is disabled by default on the server and enabled by default on the client. It adds a significant overhead in terms of performance and memory consumption so we suggest to enable it only if it is really needed. Note that Node.js has a variety of issues with high-performance compression, where increased concurrency, especially on Linux, can lead to [catastrophic memory fragmentation][node-zlib-bug] and slow performance. If you intend to use permessage-deflate in production, it is worthwhile to set up a test representative of your workload and ensure Node.js/zlib will handle it with acceptable performance and memory usage. Tuning of permessage-deflate can be done via the options defined below. You can also use `zlibDeflateOptions` and `zlibInflateOptions`, which is passed directly into the creation of [raw deflate/inflate streams][node-zlib-deflaterawdocs]. See [the docs][ws-server-options] for more options. ```js const WebSocket = require('ws'); const wss = new WebSocket.Server({ port: 8080, perMessageDeflate: { zlibDeflateOptions: { // See zlib defaults. chunkSize: 1024, memLevel: 7, level: 3 }, zlibInflateOptions: { chunkSize: 10 * 1024 }, // Other options settable: clientNoContextTakeover: true, // Defaults to negotiated value. serverNoContextTakeover: true, // Defaults to negotiated value. serverMaxWindowBits: 10, // Defaults to negotiated value. // Below options specified as default values. concurrencyLimit: 10, // Limits zlib concurrency for perf. threshold: 1024 // Size (in bytes) below which messages // should not be compressed. } }); ``` The client will only use the extension if it is supported and enabled on the server. To always disable the extension on the client set the `perMessageDeflate` option to `false`. ```js const WebSocket = require('ws'); const ws = new WebSocket('ws://www.host.com/path', { perMessageDeflate: false }); ``` ## Usage examples ### Sending and receiving text data ```js const WebSocket = require('ws'); const ws = new WebSocket('ws://www.host.com/path'); ws.on('open', function open() { ws.send('something'); }); ws.on('message', function incoming(data) { console.log(data); }); ``` ### Sending binary data ```js const WebSocket = require('ws'); const ws = new WebSocket('ws://www.host.com/path'); ws.on('open', function open() { const array = new Float32Array(5); for (var i = 0; i < array.length; ++i) { array[i] = i / 2; } ws.send(array); }); ``` ### Simple server ```js const WebSocket = require('ws'); const wss = new WebSocket.Server({ port: 8080 }); wss.on('connection', function connection(ws) { ws.on('message', function incoming(message) { console.log('received: %s', message); }); ws.send('something'); }); ``` ### External HTTP/S server ```js const fs = require('fs'); const https = require('https'); const WebSocket = require('ws'); const server = https.createServer({ cert: fs.readFileSync('/path/to/cert.pem'), key: fs.readFileSync('/path/to/key.pem') }); const wss = new WebSocket.Server({ server }); wss.on('connection', function connection(ws) { ws.on('message', function incoming(message) { console.log('received: %s', message); }); ws.send('something'); }); server.listen(8080); ``` ### Multiple servers sharing a single HTTP/S server ```js const http = require('http'); const WebSocket = require('ws'); const url = require('url'); const server = http.createServer(); const wss1 = new WebSocket.Server({ noServer: true }); const wss2 = new WebSocket.Server({ noServer: true }); wss1.on('connection', function connection(ws) { // ... }); wss2.on('connection', function connection(ws) { // ... }); server.on('upgrade', function upgrade(request, socket, head) { const pathname = url.parse(request.url).pathname; if (pathname === '/foo') { wss1.handleUpgrade(request, socket, head, function done(ws) { wss1.emit('connection', ws, request); }); } else if (pathname === '/bar') { wss2.handleUpgrade(request, socket, head, function done(ws) { wss2.emit('connection', ws, request); }); } else { socket.destroy(); } }); server.listen(8080); ``` ### Client authentication ```js const http = require('http'); const WebSocket = require('ws'); const server = http.createServer(); const wss = new WebSocket.Server({ noServer: true }); wss.on('connection', function connection(ws, request, client) { ws.on('message', function message(msg) { console.log(`Received message ${msg} from user ${client}`); }); }); server.on('upgrade', function upgrade(request, socket, head) { authenticate(request, (err, client) => { if (err || !client) { socket.destroy(); return; } wss.handleUpgrade(request, socket, head, function done(ws) { wss.emit('connection', ws, request, client); }); }); }); server.listen(8080); ``` Also see the provided [example][session-parse-example] using `express-session`. ### Server broadcast A client WebSocket broadcasting to all connected WebSocket clients, including itself. ```js const WebSocket = require('ws'); const wss = new WebSocket.Server({ port: 8080 }); wss.on('connection', function connection(ws) { ws.on('message', function incoming(data) { wss.clients.forEach(function each(client) { if (client.readyState === WebSocket.OPEN) { client.send(data); } }); }); }); ``` A client WebSocket broadcasting to every other connected WebSocket clients, excluding itself. ```js const WebSocket = require('ws'); const wss = new WebSocket.Server({ port: 8080 }); wss.on('connection', function connection(ws) { ws.on('message', function incoming(data) { wss.clients.forEach(function each(client) { if (client !== ws && client.readyState === WebSocket.OPEN) { client.send(data); } }); }); }); ``` ### echo.websocket.org demo ```js const WebSocket = require('ws'); const ws = new WebSocket('wss://echo.websocket.org/', { origin: 'https://websocket.org' }); ws.on('open', function open() { console.log('connected'); ws.send(Date.now()); }); ws.on('close', function close() { console.log('disconnected'); }); ws.on('message', function incoming(data) { console.log(`Roundtrip time: ${Date.now() - data} ms`); setTimeout(function timeout() { ws.send(Date.now()); }, 500); }); ``` ### Use the Node.js streams API ```js const WebSocket = require('ws'); const ws = new WebSocket('wss://echo.websocket.org/', { origin: 'https://websocket.org' }); const duplex = WebSocket.createWebSocketStream(ws, { encoding: 'utf8' }); duplex.pipe(process.stdout); process.stdin.pipe(duplex); ``` ### Other examples For a full example with a browser client communicating with a ws server, see the examples folder. Otherwise, see the test cases. ## FAQ ### How to get the IP address of the client? The remote IP address can be obtained from the raw socket. ```js const WebSocket = require('ws'); const wss = new WebSocket.Server({ port: 8080 }); wss.on('connection', function connection(ws, req) { const ip = req.connection.remoteAddress; }); ``` When the server runs behind a proxy like NGINX, the de-facto standard is to use the `X-Forwarded-For` header. ```js wss.on('connection', function connection(ws, req) { const ip = req.headers['x-forwarded-for'].split(/\s*,\s*/)[0]; }); ``` ### How to detect and close broken connections? Sometimes the link between the server and the client can be interrupted in a way that keeps both the server and the client unaware of the broken state of the connection (e.g. when pulling the cord). In these cases ping messages can be used as a means to verify that the remote endpoint is still responsive. ```js const WebSocket = require('ws'); function noop() {} function heartbeat() { this.isAlive = true; } const wss = new WebSocket.Server({ port: 8080 }); wss.on('connection', function connection(ws) { ws.isAlive = true; ws.on('pong', heartbeat); }); const interval = setInterval(function ping() { wss.clients.forEach(function each(ws) { if (ws.isAlive === false) return ws.terminate(); ws.isAlive = false; ws.ping(noop); }); }, 30000); ``` Pong messages are automatically sent in response to ping messages as required by the spec. Just like the server example above your clients might as well lose connection without knowing it. You might want to add a ping listener on your clients to prevent that. A simple implementation would be: ```js const WebSocket = require('ws'); function heartbeat() { clearTimeout(this.pingTimeout); // Use `WebSocket#terminate()`, which immediately destroys the connection, // instead of `WebSocket#close()`, which waits for the close timer. // Delay should be equal to the interval at which your server // sends out pings plus a conservative assumption of the latency. this.pingTimeout = setTimeout(() => { this.terminate(); }, 30000 + 1000); } const client = new WebSocket('wss://echo.websocket.org/'); client.on('open', heartbeat); client.on('ping', heartbeat); client.on('close', function clear() { clearTimeout(this.pingTimeout); }); ``` ### How to connect via a proxy? Use a custom `http.Agent` implementation like [https-proxy-agent][] or [socks-proxy-agent][]. ## Changelog We're using the GitHub [releases][changelog] for changelog entries. ## License [MIT](LICENSE) [changelog]: https://github.com/websockets/ws/releases [client-report]: http://websockets.github.io/ws/autobahn/clients/ [https-proxy-agent]: https://github.com/TooTallNate/node-https-proxy-agent [node-zlib-bug]: https://github.com/nodejs/node/issues/8871 [node-zlib-deflaterawdocs]: https://nodejs.org/api/zlib.html#zlib_zlib_createdeflateraw_options [permessage-deflate]: https://tools.ietf.org/html/rfc7692 [server-report]: http://websockets.github.io/ws/autobahn/servers/ [session-parse-example]: ./examples/express-session-parse [socks-proxy-agent]: https://github.com/TooTallNate/node-socks-proxy-agent [ws-server-options]: https://github.com/websockets/ws/blob/master/doc/ws.md#new-websocketserveroptions-callback ws-7.2.1/SECURITY.md000066400000000000000000000031341357512353100137510ustar00rootroot00000000000000# Security Guidelines Please contact us directly at **security@3rd-Eden.com** for any bug that might impact the security of this project. Please prefix the subject of your email with `[security]` in lowercase and square brackets. Our email filters will automatically prevent these messages from being moved to our spam box. You will receive an acknowledgement of your report within **24 hours**. All emails that do not include security vulnerabilities will be removed and blocked instantly. ## Exceptions If you do not receive an acknowledgement within the said time frame please give us the benefit of the doubt as it's possible that we haven't seen it yet. In this case please send us a message **without details** using one of the following methods: - Contact the lead developers of this project on their personal e-mails. You can find the e-mails in the git logs, for example using the following command: `git --no-pager show -s --format='%an <%ae>' ` where `` is the SHA1 of their latest commit in the project. - Create a GitHub issue stating contact details and the severity of the issue. Once we have acknowledged receipt of your report and confirmed the bug ourselves we will work with you to fix the vulnerability and publicly acknowledge your responsible disclosure, if you wish. In addition to that we will report all vulnerabilities to the [Node Security Project](https://nodesecurity.io/). ## History - 04 Jan 2016: [Buffer vulnerability](https://github.com/websockets/ws/releases/tag/1.0.1) - 08 Nov 2017: [DoS vulnerability](https://github.com/websockets/ws/releases/tag/3.3.1) ws-7.2.1/appveyor.yml000066400000000000000000000005221357512353100145460ustar00rootroot00000000000000environment: matrix: - nodejs_version: '13' - nodejs_version: '12' - nodejs_version: '10' - nodejs_version: '8' platform: - x86 matrix: fast_finish: true install: - ps: Install-Product node $env:nodejs_version $env:platform - npm install test_script: - node --version - npm --version - npm test build: off ws-7.2.1/bench/000077500000000000000000000000001357512353100132365ustar00rootroot00000000000000ws-7.2.1/bench/parser.benchmark.js000066400000000000000000000042521357512353100170240ustar00rootroot00000000000000'use strict'; const benchmark = require('benchmark'); const crypto = require('crypto'); const WebSocket = require('..'); const Receiver = WebSocket.Receiver; const Sender = WebSocket.Sender; const options = { fin: true, rsv1: false, mask: true, readOnly: false }; function createBinaryFrame(length) { const list = Sender.frame( crypto.randomBytes(length), Object.assign({ opcode: 0x02 }, options) ); return Buffer.concat(list); } const pingFrame1 = Buffer.concat( Sender.frame(crypto.randomBytes(5), Object.assign({ opcode: 0x09 }, options)) ); const textFrame = Buffer.from('819461616161' + '61'.repeat(20), 'hex'); const pingFrame2 = Buffer.from('8900', 'hex'); const binaryFrame1 = createBinaryFrame(125); const binaryFrame2 = createBinaryFrame(65535); const binaryFrame3 = createBinaryFrame(200 * 1024); const binaryFrame4 = createBinaryFrame(1024 * 1024); const suite = new benchmark.Suite(); const receiver = new Receiver(); suite.add('ping frame (5 bytes payload)', { defer: true, fn: (deferred) => { receiver.write(pingFrame1, deferred.resolve.bind(deferred)); } }); suite.add('ping frame (no payload)', { defer: true, fn: (deferred) => { receiver.write(pingFrame2, deferred.resolve.bind(deferred)); } }); suite.add('text frame (20 bytes payload)', { defer: true, fn: (deferred) => { receiver.write(textFrame, deferred.resolve.bind(deferred)); } }); suite.add('binary frame (125 bytes payload)', { defer: true, fn: (deferred) => { receiver.write(binaryFrame1, deferred.resolve.bind(deferred)); } }); suite.add('binary frame (65535 bytes payload)', { defer: true, fn: (deferred) => { receiver.write(binaryFrame2, deferred.resolve.bind(deferred)); } }); suite.add('binary frame (200 KiB payload)', { defer: true, fn: (deferred) => { receiver.write(binaryFrame3, deferred.resolve.bind(deferred)); } }); suite.add('binary frame (1 MiB payload)', { defer: true, fn: (deferred) => { receiver.write(binaryFrame4, deferred.resolve.bind(deferred)); } }); suite.on('cycle', (e) => console.log(e.target.toString())); if (require.main === module) { suite.run({ async: true }); } else { module.exports = suite; } ws-7.2.1/bench/sender.benchmark.js000066400000000000000000000026521357512353100170120ustar00rootroot00000000000000'use strict'; const benchmark = require('benchmark'); const crypto = require('crypto'); const Sender = require('../').Sender; const data1 = crypto.randomBytes(64); const data2 = crypto.randomBytes(16 * 1024); const data3 = crypto.randomBytes(64 * 1024); const data4 = crypto.randomBytes(200 * 1024); const data5 = crypto.randomBytes(1024 * 1024); const opts1 = { readOnly: false, mask: false, rsv1: false, opcode: 2, fin: true }; const opts2 = { readOnly: true, rsv1: false, mask: true, opcode: 2, fin: true }; const suite = new benchmark.Suite(); suite.add('frame, unmasked (64 B)', () => Sender.frame(data1, opts1)); suite.add('frame, masked (64 B)', () => Sender.frame(data1, opts2)); suite.add('frame, unmasked (16 KiB)', () => Sender.frame(data2, opts1)); suite.add('frame, masked (16 KiB)', () => Sender.frame(data2, opts2)); suite.add('frame, unmasked (64 KiB)', () => Sender.frame(data3, opts1)); suite.add('frame, masked (64 KiB)', () => Sender.frame(data3, opts2)); suite.add('frame, unmasked (200 KiB)', () => Sender.frame(data4, opts1)); suite.add('frame, masked (200 KiB)', () => Sender.frame(data4, opts2)); suite.add('frame, unmasked (1 MiB)', () => Sender.frame(data5, opts1)); suite.add('frame, masked (1 MiB)', () => Sender.frame(data5, opts2)); suite.on('cycle', (e) => console.log(e.target.toString())); if (require.main === module) { suite.run({ async: true }); } else { module.exports = suite; } ws-7.2.1/bench/speed.js000066400000000000000000000055501357512353100147010ustar00rootroot00000000000000'use strict'; const cluster = require('cluster'); const http = require('http'); const WebSocket = require('..'); const port = 8181; const path = ''; // const path = '/tmp/wss.sock'; if (cluster.isMaster) { const server = http.createServer(); const wss = new WebSocket.Server({ maxPayload: 600 * 1024 * 1024, perMessageDeflate: false, clientTracking: false, server }); wss.on('connection', (ws) => { ws.on('message', (data) => ws.send(data)); }); server.listen(path ? { path } : { port }, () => cluster.fork()); cluster.on('exit', () => { wss.close(); server.close(); }); } else { const configs = [ [true, 10000, 64], [true, 5000, 16 * 1024], [true, 1000, 128 * 1024], [true, 100, 1024 * 1024], [true, 1, 500 * 1024 * 1024], [false, 10000, 64], [false, 5000, 16 * 1024], [false, 1000, 128 * 1024], [false, 100, 1024 * 1024] ]; const roundPrec = (num, prec) => { const mul = Math.pow(10, prec); return Math.round(num * mul) / mul; }; const humanSize = (bytes) => { if (bytes >= 1073741824) return roundPrec(bytes / 1073741824, 2) + ' GiB'; if (bytes >= 1048576) return roundPrec(bytes / 1048576, 2) + ' MiB'; if (bytes >= 1024) return roundPrec(bytes / 1024, 2) + ' KiB'; return roundPrec(bytes, 2) + ' B'; }; const largest = configs.reduce( (prev, curr) => (curr[2] > prev ? curr[2] : prev), 0 ); console.log('Generating %s of test data...', humanSize(largest)); const randomBytes = Buffer.allocUnsafe(largest); for (let i = 0; i < largest; ++i) { randomBytes[i] = ~~(Math.random() * 127); } console.log(`Testing ws on ${path || '[::]:' + port}`); const runConfig = (useBinary, roundtrips, size, cb) => { const data = randomBytes.slice(0, size); const url = path ? `ws+unix://${path}` : `ws://localhost:${port}`; const ws = new WebSocket(url, { maxPayload: 600 * 1024 * 1024 }); let roundtrip = 0; let time; ws.on('error', (err) => { console.error(err.stack); cluster.worker.disconnect(); }); ws.on('open', () => { time = process.hrtime(); ws.send(data, { binary: useBinary }); }); ws.on('message', () => { if (++roundtrip !== roundtrips) return ws.send(data, { binary: useBinary }); let elapsed = process.hrtime(time); elapsed = elapsed[0] * 1e9 + elapsed[1]; console.log( '%d roundtrips of %s %s data:\t%ss\t%s', roundtrips, humanSize(size), useBinary ? 'binary' : 'text', roundPrec(elapsed / 1e9, 1), humanSize(((size * 2 * roundtrips) / elapsed) * 1e9) + '/s' ); ws.close(); cb(); }); }; (function run() { if (configs.length === 0) return cluster.worker.disconnect(); const config = configs.shift(); config.push(run); runConfig.apply(null, config); })(); } ws-7.2.1/browser.js000066400000000000000000000002571357512353100142040ustar00rootroot00000000000000'use strict'; module.exports = function() { throw new Error( 'ws does not work in the browser. Browser clients must use the native ' + 'WebSocket object' ); }; ws-7.2.1/doc/000077500000000000000000000000001357512353100127245ustar00rootroot00000000000000ws-7.2.1/doc/ws.md000066400000000000000000000430341357512353100137030ustar00rootroot00000000000000# ws ## Table of Contents - [Class: WebSocket.Server](#class-websocketserver) - [new WebSocket.Server(options[, callback])](#new-websocketserveroptions-callback) - [Event: 'close'](#event-close) - [Event: 'connection'](#event-connection) - [Event: 'error'](#event-error) - [Event: 'headers'](#event-headers) - [Event: 'listening'](#event-listening) - [server.address()](#serveraddress) - [server.clients](#serverclients) - [server.close([callback])](#serverclosecallback) - [server.handleUpgrade(request, socket, head, callback)](#serverhandleupgraderequest-socket-head-callback) - [server.shouldHandle(request)](#servershouldhandlerequest) - [Class: WebSocket](#class-websocket) - [Ready state constants](#ready-state-constants) - [new WebSocket(address[, protocols][, options])](#new-websocketaddress-protocols-options) - [UNIX Domain Sockets](#unix-domain-sockets) - [Event: 'close'](#event-close-1) - [Event: 'error'](#event-error-1) - [Event: 'message'](#event-message) - [Event: 'open'](#event-open) - [Event: 'ping'](#event-ping) - [Event: 'pong'](#event-pong) - [Event: 'unexpected-response'](#event-unexpected-response) - [Event: 'upgrade'](#event-upgrade) - [websocket.addEventListener(type, listener)](#websocketaddeventlistenertype-listener) - [websocket.binaryType](#websocketbinarytype) - [websocket.bufferedAmount](#websocketbufferedamount) - [websocket.close([code[, reason]])](#websocketclosecode-reason) - [websocket.extensions](#websocketextensions) - [websocket.onclose](#websocketonclose) - [websocket.onerror](#websocketonerror) - [websocket.onmessage](#websocketonmessage) - [websocket.onopen](#websocketonopen) - [websocket.ping([data[, mask]][, callback])](#websocketpingdata-mask-callback) - [websocket.pong([data[, mask]][, callback])](#websocketpongdata-mask-callback) - [websocket.protocol](#websocketprotocol) - [websocket.readyState](#websocketreadystate) - [websocket.removeEventListener(type, listener)](#websocketremoveeventlistenertype-listener) - [websocket.send(data[, options][, callback])](#websocketsenddata-options-callback) - [websocket.terminate()](#websocketterminate) - [websocket.url](#websocketurl) - [WebSocket.createWebSocketStream(websocket[, options])](#websocketcreatewebsocketstreamwebsocket-options) ## Class: WebSocket.Server This class represents a WebSocket server. It extends the `EventEmitter`. ### new WebSocket.Server(options[, callback]) - `options` {Object} - `host` {String} The hostname where to bind the server. - `port` {Number} The port where to bind the server. - `backlog` {Number} The maximum length of the queue of pending connections. - `server` {http.Server|https.Server} A pre-created Node.js HTTP/S server. - `verifyClient` {Function} A function which can be used to validate incoming connections. See description below. (Usage is discouraged: see [Issue #337](https://github.com/websockets/ws/issues/377#issuecomment-462152231)) - `handleProtocols` {Function} A function which can be used to handle the WebSocket subprotocols. See description below. - `path` {String} Accept only connections matching this path. - `noServer` {Boolean} Enable no server mode. - `clientTracking` {Boolean} Specifies whether or not to track clients. - `perMessageDeflate` {Boolean|Object} Enable/disable permessage-deflate. - `maxPayload` {Number} The maximum allowed message size in bytes. - `callback` {Function} Create a new server instance. One of `port`, `server` or `noServer` must be provided or an error is thrown. An HTTP server is automatically created, started, and used if `port` is set. To use an external HTTP/S server instead, specify only `server` or `noServer`. In this case the HTTP/S server must be started manually. The "noServer" mode allows the WebSocket server to be completly detached from the HTTP/S server. This makes it possible, for example, to share a single HTTP/S server between multiple WebSocket servers. > **NOTE:** Use of `verifyClient` is discouraged. Rather handle client > authentication in the `upgrade` event of the HTTP server. See examples for > more details. If `verifyClient` is not set then the handshake is automatically accepted. If it is provided with a single argument then that is: - `info` {Object} - `origin` {String} The value in the Origin header indicated by the client. - `req` {http.IncomingMessage} The client HTTP GET request. - `secure` {Boolean} `true` if `req.connection.authorized` or `req.connection.encrypted` is set. The return value (Boolean) of the function determines whether or not to accept the handshake. if `verifyClient` is provided with two arguments then those are: - `info` {Object} Same as above. - `cb` {Function} A callback that must be called by the user upon inspection of the `info` fields. Arguments in this callback are: - `result` {Boolean} Whether or not to accept the handshake. - `code` {Number} When `result` is `false` this field determines the HTTP error status code to be sent to the client. - `name` {String} When `result` is `false` this field determines the HTTP reason phrase. - `headers` {Object} When `result` is `false` this field determines additional HTTP headers to be sent to the client. For example, `{ 'Retry-After': 120 }`. `handleProtocols` takes two arguments: - `protocols` {Array} The list of WebSocket subprotocols indicated by the client in the `Sec-WebSocket-Protocol` header. - `request` {http.IncomingMessage} The client HTTP GET request. The returned value sets the value of the `Sec-WebSocket-Protocol` header in the HTTP 101 response. If returned value is `false` the header is not added in the response. If `handleProtocols` is not set then the first of the client's requested subprotocols is used. `perMessageDeflate` can be used to control the behavior of [permessage-deflate extension][permessage-deflate]. The extension is disabled when `false` (default value). If an object is provided then that is extension parameters: - `serverNoContextTakeover` {Boolean} Whether to use context takeover or not. - `clientNoContextTakeover` {Boolean} Acknowledge disabling of client context takeover. - `serverMaxWindowBits` {Number} The value of `windowBits`. - `clientMaxWindowBits` {Number} Request a custom client window size. - `zlibDeflateOptions` {Object} [Additional options][zlib-options] to pass to zlib on deflate. - `zlibInflateOptions` {Object} [Additional options][zlib-options] to pass to zlib on inflate. - `threshold` {Number} Payloads smaller than this will not be compressed. Defaults to 1024 bytes. - `concurrencyLimit` {Number} The number of concurrent calls to zlib. Calls above this limit will be queued. Default 10. You usually won't need to touch this option. See [this issue][concurrency-limit] for more details. If a property is empty then either an offered configuration or a default value is used. When sending a fragmented message the length of the first fragment is compared to the threshold. This determines if compression is used for the entire message. `callback` will be added as a listener for the `listening` event on the HTTP server when not operating in "noServer" mode. ### Event: 'close' Emitted when the server closes. This event depends on the `'close'` event of HTTP server only when it is created internally. In all other cases, the event is emitted independently. ### Event: 'connection' - `socket` {WebSocket} - `request` {http.IncomingMessage} Emitted when the handshake is complete. `request` is the http GET request sent by the client. Useful for parsing authority headers, cookie headers, and other information. ### Event: 'error' - `error` {Error} Emitted when an error occurs on the underlying server. ### Event: 'headers' - `headers` {Array} - `request` {http.IncomingMessage} Emitted before the response headers are written to the socket as part of the handshake. This allows you to inspect/modify the headers before they are sent. ### Event: 'listening' Emitted when the underlying server has been bound. ### server.address() Returns an object with `port`, `family`, and `address` properties specifying the bound address, the address family name, and port of the server as reported by the operating system if listening on an IP socket. If the server is listening on a pipe or UNIX domain socket, the name is returned as a string. ### server.clients - {Set} A set that stores all connected clients. Please note that this property is only added when the `clientTracking` is truthy. ### server.close([callback]) Close the HTTP server if created internally, terminate all clients and call callback when done. If an external HTTP server is used via the `server` or `noServer` constructor options, it must be closed manually. ### server.handleUpgrade(request, socket, head, callback) - `request` {http.IncomingMessage} The client HTTP GET request. - `socket` {net.Socket} The network socket between the server and client. - `head` {Buffer} The first packet of the upgraded stream. - `callback` {Function}. Handle a HTTP upgrade request. When the HTTP server is created internally or when the HTTP server is passed via the `server` option, this method is called automatically. When operating in "noServer" mode, this method must be called manually. If the upgrade is successful, the `callback` is called with a `WebSocket` object as parameter. ### server.shouldHandle(request) - `request` {http.IncomingMessage} The client HTTP GET request. See if a given request should be handled by this server. By default this method validates the pathname of the request, matching it against the `path` option if provided. The return value, `true` or `false`, determines whether or not to accept the handshake. This method can be overridden when a custom handling logic is required. ## Class: WebSocket This class represents a WebSocket. It extends the `EventEmitter`. ### Ready state constants | Constant | Value | Description | | ---------- | ----- | ------------------------------------------------ | | CONNECTING | 0 | The connection is not yet open. | | OPEN | 1 | The connection is open and ready to communicate. | | CLOSING | 2 | The connection is in the process of closing. | | CLOSED | 3 | The connection is closed. | ### new WebSocket(address[, protocols][, options]) - `address` {String|url.URL} The URL to which to connect. - `protocols` {String|Array} The list of subprotocols. - `options` {Object} - `followRedirects` {Boolean} Whether or not to follow redirects. Defaults to `false`. - `handshakeTimeout` {Number} Timeout in milliseconds for the handshake request. This is reset after every redirection. - `maxRedirects` {Number} The maximum number of redirects allowed. Defaults to 10. - `perMessageDeflate` {Boolean|Object} Enable/disable permessage-deflate. - `protocolVersion` {Number} Value of the `Sec-WebSocket-Version` header. - `origin` {String} Value of the `Origin` or `Sec-WebSocket-Origin` header depending on the `protocolVersion`. - `maxPayload` {Number} The maximum allowed message size in bytes. - Any other option allowed in [http.request()][] or [https.request()][]. `perMessageDeflate` default value is `true`. When using an object, parameters are the same of the server. The only difference is the direction of requests. For example, `serverNoContextTakeover` can be used to ask the server to disable context takeover. Create a new WebSocket instance. #### UNIX Domain Sockets `ws` supports making requests to UNIX domain sockets. To make one, use the following URL scheme: ``` ws+unix:///absolute/path/to/uds_socket:/pathname?search_params ``` Note that `:` is the separator between the socket path and the URL path. If the URL path is omitted ``` ws+unix:///absolute/path/to/uds_socket ``` it defaults to `/`. ### Event: 'close' - `code` {Number} - `reason` {String} Emitted when the connection is closed. `code` is a numeric value indicating the status code explaining why the connection has been closed. `reason` is a human-readable string explaining why the connection has been closed. ### Event: 'error' - `error` {Error} Emitted when an error occurs. ### Event: 'message' - `data` {String|Buffer|ArrayBuffer|Buffer[]} Emitted when a message is received from the server. ### Event: 'open' Emitted when the connection is established. ### Event: 'ping' - `data` {Buffer} Emitted when a ping is received from the server. ### Event: 'pong' - `data` {Buffer} Emitted when a pong is received from the server. ### Event: 'unexpected-response' - `request` {http.ClientRequest} - `response` {http.IncomingMessage} Emitted when the server response is not the expected one, for example a 401 response. This event gives the ability to read the response in order to extract useful information. If the server sends an invalid response and there isn't a listener for this event, an error is emitted. ### Event: 'upgrade' - `response` {http.IncomingMessage} Emitted when response headers are received from the server as part of the handshake. This allows you to read headers from the server, for example 'set-cookie' headers. ### websocket.addEventListener(type, listener) - `type` {String} A string representing the event type to listen for. - `listener` {Function} The listener to add. Register an event listener emulating the `EventTarget` interface. ### websocket.binaryType - {String} A string indicating the type of binary data being transmitted by the connection. This should be one of "nodebuffer", "arraybuffer" or "fragments". Defaults to "nodebuffer". Type "fragments" will emit the array of fragments as received from the sender, without copyfull concatenation, which is useful for the performance of binary protocols transferring large messages with multiple fragments. ### websocket.bufferedAmount - {Number} The number of bytes of data that have been queued using calls to `send()` but not yet transmitted to the network. ### websocket.close([code[, reason]]) - `code` {Number} A numeric value indicating the status code explaining why the connection is being closed. - `reason` {String} A human-readable string explaining why the connection is closing. Initiate a closing handshake. ### websocket.extensions - {Object} An object containing the negotiated extensions. ### websocket.onclose - {Function} An event listener to be called when connection is closed. The listener receives a `CloseEvent` named "close". ### websocket.onerror - {Function} An event listener to be called when an error occurs. The listener receives an `ErrorEvent` named "error". ### websocket.onmessage - {Function} An event listener to be called when a message is received from the server. The listener receives a `MessageEvent` named "message". ### websocket.onopen - {Function} An event listener to be called when the connection is established. The listener receives an `OpenEvent` named "open". ### websocket.ping([data[, mask]][, callback]) - `data` {Any} The data to send in the ping frame. - `mask` {Boolean} Specifies whether `data` should be masked or not. Defaults to `true` when `websocket` is not a server client. - `callback` {Function} An optional callback which is invoked when the ping frame is written out. Send a ping. ### websocket.pong([data[, mask]][, callback]) - `data` {Any} The data to send in the pong frame. - `mask` {Boolean} Specifies whether `data` should be masked or not. Defaults to `true` when `websocket` is not a server client. - `callback` {Function} An optional callback which is invoked when the pong frame is written out. Send a pong. ### websocket.protocol - {String} The subprotocol selected by the server. ### websocket.readyState - {Number} The current state of the connection. This is one of the ready state constants. ### websocket.removeEventListener(type, listener) - `type` {String} A string representing the event type to remove. - `listener` {Function} The listener to remove. Removes an event listener emulating the `EventTarget` interface. ### websocket.send(data[, options][, callback]) - `data` {Any} The data to send. - `options` {Object} - `compress` {Boolean} Specifies whether `data` should be compressed or not. Defaults to `true` when permessage-deflate is enabled. - `binary` {Boolean} Specifies whether `data` should be sent as a binary or not. Default is autodetected. - `mask` {Boolean} Specifies whether `data` should be masked or not. Defaults to `true` when `websocket` is not a server client. - `fin` {Boolean} Specifies whether `data` is the last fragment of a message or not. Defaults to `true`. - `callback` {Function} An optional callback which is invoked when `data` is written out. Send `data` through the connection. ### websocket.terminate() Forcibly close the connection. ### websocket.url - {String} The URL of the WebSocket server. Server clients don't have this attribute. ## WebSocket.createWebSocketStream(websocket[, options]) - `websocket` {WebSocket} A `WebSocket` object. - `options` {Object} [Options][duplex-options] to pass to the `Duplex` constructor. Returns a `Duplex` stream that allows to use the Node.js streams API on top of a given `WebSocket`. [concurrency-limit]: https://github.com/websockets/ws/issues/1202 [duplex-options]: https://nodejs.org/api/stream.html#stream_new_stream_duplex_options [http.request()]: https://nodejs.org/api/http.html#http_http_request_options_callback [https.request()]: https://nodejs.org/api/https.html#https_https_request_options_callback [permessage-deflate]: https://tools.ietf.org/html/draft-ietf-hybi-permessage-compression-19 [zlib-options]: https://nodejs.org/api/zlib.html#zlib_class_options ws-7.2.1/examples/000077500000000000000000000000001357512353100137755ustar00rootroot00000000000000ws-7.2.1/examples/express-session-parse/000077500000000000000000000000001357512353100202575ustar00rootroot00000000000000ws-7.2.1/examples/express-session-parse/index.js000066400000000000000000000040651357512353100217310ustar00rootroot00000000000000'use strict'; const session = require('express-session'); const express = require('express'); const http = require('http'); const uuid = require('uuid'); const WebSocket = require('../..'); const app = express(); const map = new Map(); // // We need the same instance of the session parser in express and // WebSocket server. // const sessionParser = session({ saveUninitialized: false, secret: '$eCuRiTy', resave: false }); // // Serve static files from the 'public' folder. // app.use(express.static('public')); app.use(sessionParser); app.post('/login', function(req, res) { // // "Log in" user and set userId to session. // const id = uuid.v4(); console.log(`Updating session for user ${id}`); req.session.userId = id; res.send({ result: 'OK', message: 'Session updated' }); }); app.delete('/logout', function(request, response) { const ws = map.get(request.session.userId); console.log('Destroying session'); request.session.destroy(function() { if (ws) ws.close(); response.send({ result: 'OK', message: 'Session destroyed' }); }); }); // // Create HTTP server by ourselves. // const server = http.createServer(app); const wss = new WebSocket.Server({ clientTracking: false, noServer: true }); server.on('upgrade', function(request, socket, head) { console.log('Parsing session from request...'); sessionParser(request, {}, () => { if (!request.session.userId) { socket.destroy(); return; } console.log('Session is parsed!'); wss.handleUpgrade(request, socket, head, function(ws) { wss.emit('connection', ws, request); }); }); }); wss.on('connection', function(ws, request) { const userId = request.session.userId; map.set(userId, ws); ws.on('message', function(message) { // // Here we can now use session parameters. // console.log(`Received message ${message} from user ${userId}`); }); ws.on('close', function() { map.delete(userId); }); }); // // Start the server. // server.listen(8080, function() { console.log('Listening on http://localhost:8080'); }); ws-7.2.1/examples/express-session-parse/package.json000066400000000000000000000003271357512353100225470ustar00rootroot00000000000000{ "author": "", "name": "express-session-parse", "version": "0.0.0", "repository": "websockets/ws", "dependencies": { "express": "^4.16.4", "express-session": "^1.16.1", "uuid": "^3.3.2" } } ws-7.2.1/examples/express-session-parse/public/000077500000000000000000000000001357512353100215355ustar00rootroot00000000000000ws-7.2.1/examples/express-session-parse/public/app.js000066400000000000000000000033171357512353100226570ustar00rootroot00000000000000(function() { const messages = document.querySelector('#messages'); const wsButton = document.querySelector('#wsButton'); const wsSendButton = document.querySelector('#wsSendButton'); const logout = document.querySelector('#logout'); const login = document.querySelector('#login'); function showMessage(message) { messages.textContent += `\n${message}`; messages.scrollTop = messages.scrollHeight; } function handleResponse(response) { return response.ok ? response.json().then((data) => JSON.stringify(data, null, 2)) : Promise.reject(new Error('Unexpected response')); } login.onclick = function() { fetch('/login', { method: 'POST', credentials: 'same-origin' }) .then(handleResponse) .then(showMessage) .catch(function(err) { showMessage(err.message); }); }; logout.onclick = function() { fetch('/logout', { method: 'DELETE', credentials: 'same-origin' }) .then(handleResponse) .then(showMessage) .catch(function(err) { showMessage(err.message); }); }; let ws; wsButton.onclick = function() { if (ws) { ws.onerror = ws.onopen = ws.onclose = null; ws.close(); } ws = new WebSocket(`ws://${location.host}`); ws.onerror = function() { showMessage('WebSocket error'); }; ws.onopen = function() { showMessage('WebSocket connection established'); }; ws.onclose = function() { showMessage('WebSocket connection closed'); ws = null; }; }; wsSendButton.onclick = function() { if (!ws) { showMessage('No WebSocket connection'); return; } ws.send('Hello World!'); showMessage('Sent "Hello World!"'); }; })(); ws-7.2.1/examples/express-session-parse/public/index.html000066400000000000000000000013211357512353100235270ustar00rootroot00000000000000 Express session demo

Choose an action.


    
  

ws-7.2.1/examples/server-stats/000077500000000000000000000000001357512353100164375ustar00rootroot00000000000000ws-7.2.1/examples/server-stats/index.js000066400000000000000000000013761357512353100201130ustar00rootroot00000000000000'use strict';

const express = require('express');
const path = require('path');
const { createServer } = require('http');

const WebSocket = require('../../');

const app = express();
app.use(express.static(path.join(__dirname, '/public')));

const server = createServer(app);
const wss = new WebSocket.Server({ server });

wss.on('connection', function(ws) {
  const id = setInterval(function() {
    ws.send(JSON.stringify(process.memoryUsage()), function() {
      //
      // Ignore errors.
      //
    });
  }, 100);
  console.log('started client interval');

  ws.on('close', function() {
    console.log('stopping client interval');
    clearInterval(id);
  });
});

server.listen(8080, function() {
  console.log('Listening on http://localhost:8080');
});
ws-7.2.1/examples/server-stats/package.json000066400000000000000000000002251357512353100207240ustar00rootroot00000000000000{
  "author": "",
  "name": "serverstats",
  "version": "0.0.0",
  "repository": "websockets/ws",
  "dependencies": {
    "express": "^4.16.4"
  }
}
ws-7.2.1/examples/server-stats/public/000077500000000000000000000000001357512353100177155ustar00rootroot00000000000000ws-7.2.1/examples/server-stats/public/index.html000066400000000000000000000026761357512353100217250ustar00rootroot00000000000000

  
    
    Server stats
    
  
  
    

Server stats

Memory usage
RSS
Heap total
Heap used
External
ws-7.2.1/examples/ssl.js000066400000000000000000000017111357512353100151340ustar00rootroot00000000000000'use strict'; const https = require('https'); const fs = require('fs'); const WebSocket = require('..'); const server = https.createServer({ cert: fs.readFileSync('../test/fixtures/certificate.pem'), key: fs.readFileSync('../test/fixtures/key.pem') }); const wss = new WebSocket.Server({ server }); wss.on('connection', function connection(ws) { ws.on('message', function message(msg) { console.log(msg); }); }); server.listen(function listening() { // // If the `rejectUnauthorized` option is not `false`, the server certificate // is verified against a list of well-known CAs. An 'error' event is emitted // if verification fails. // // The certificate used in this example is self-signed so `rejectUnauthorized` // is set to `false`. // const ws = new WebSocket(`wss://localhost:${server.address().port}`, { rejectUnauthorized: false }); ws.on('open', function open() { ws.send('All glory to WebSockets!'); }); }); ws-7.2.1/index.js000066400000000000000000000004501357512353100136230ustar00rootroot00000000000000'use strict'; const WebSocket = require('./lib/websocket'); WebSocket.createWebSocketStream = require('./lib/stream'); WebSocket.Server = require('./lib/websocket-server'); WebSocket.Receiver = require('./lib/receiver'); WebSocket.Sender = require('./lib/sender'); module.exports = WebSocket; ws-7.2.1/lib/000077500000000000000000000000001357512353100127255ustar00rootroot00000000000000ws-7.2.1/lib/buffer-util.js000066400000000000000000000064751357512353100155230ustar00rootroot00000000000000'use strict'; const { EMPTY_BUFFER } = require('./constants'); /** * Merges an array of buffers into a new buffer. * * @param {Buffer[]} list The array of buffers to concat * @param {Number} totalLength The total length of buffers in the list * @return {Buffer} The resulting buffer * @public */ function concat(list, totalLength) { if (list.length === 0) return EMPTY_BUFFER; if (list.length === 1) return list[0]; const target = Buffer.allocUnsafe(totalLength); let offset = 0; for (let i = 0; i < list.length; i++) { const buf = list[i]; target.set(buf, offset); offset += buf.length; } if (offset < totalLength) return target.slice(0, offset); return target; } /** * Masks a buffer using the given mask. * * @param {Buffer} source The buffer to mask * @param {Buffer} mask The mask to use * @param {Buffer} output The buffer where to store the result * @param {Number} offset The offset at which to start writing * @param {Number} length The number of bytes to mask. * @public */ function _mask(source, mask, output, offset, length) { for (let i = 0; i < length; i++) { output[offset + i] = source[i] ^ mask[i & 3]; } } /** * Unmasks a buffer using the given mask. * * @param {Buffer} buffer The buffer to unmask * @param {Buffer} mask The mask to use * @public */ function _unmask(buffer, mask) { // Required until https://github.com/nodejs/node/issues/9006 is resolved. const length = buffer.length; for (let i = 0; i < length; i++) { buffer[i] ^= mask[i & 3]; } } /** * Converts a buffer to an `ArrayBuffer`. * * @param {Buffer} buf The buffer to convert * @return {ArrayBuffer} Converted buffer * @public */ function toArrayBuffer(buf) { if (buf.byteLength === buf.buffer.byteLength) { return buf.buffer; } return buf.buffer.slice(buf.byteOffset, buf.byteOffset + buf.byteLength); } /** * Converts `data` to a `Buffer`. * * @param {*} data The data to convert * @return {Buffer} The buffer * @throws {TypeError} * @public */ function toBuffer(data) { toBuffer.readOnly = true; if (Buffer.isBuffer(data)) return data; let buf; if (data instanceof ArrayBuffer) { buf = Buffer.from(data); } else if (ArrayBuffer.isView(data)) { buf = viewToBuffer(data); } else { buf = Buffer.from(data); toBuffer.readOnly = false; } return buf; } /** * Converts an `ArrayBuffer` view into a buffer. * * @param {(DataView|TypedArray)} view The view to convert * @return {Buffer} Converted view * @private */ function viewToBuffer(view) { const buf = Buffer.from(view.buffer); if (view.byteLength !== view.buffer.byteLength) { return buf.slice(view.byteOffset, view.byteOffset + view.byteLength); } return buf; } try { const bufferUtil = require('bufferutil'); const bu = bufferUtil.BufferUtil || bufferUtil; module.exports = { concat, mask(source, mask, output, offset, length) { if (length < 48) _mask(source, mask, output, offset, length); else bu.mask(source, mask, output, offset, length); }, toArrayBuffer, toBuffer, unmask(buffer, mask) { if (buffer.length < 32) _unmask(buffer, mask); else bu.unmask(buffer, mask); } }; } catch (e) /* istanbul ignore next */ { module.exports = { concat, mask: _mask, toArrayBuffer, toBuffer, unmask: _unmask }; } ws-7.2.1/lib/constants.js000066400000000000000000000004141357512353100152760ustar00rootroot00000000000000'use strict'; module.exports = { BINARY_TYPES: ['nodebuffer', 'arraybuffer', 'fragments'], GUID: '258EAFA5-E914-47DA-95CA-C5AB0DC85B11', kStatusCode: Symbol('status-code'), kWebSocket: Symbol('websocket'), EMPTY_BUFFER: Buffer.alloc(0), NOOP: () => {} }; ws-7.2.1/lib/event-target.js000066400000000000000000000075411357512353100156770ustar00rootroot00000000000000'use strict'; /** * Class representing an event. * * @private */ class Event { /** * Create a new `Event`. * * @param {String} type The name of the event * @param {Object} target A reference to the target to which the event was dispatched */ constructor(type, target) { this.target = target; this.type = type; } } /** * Class representing a message event. * * @extends Event * @private */ class MessageEvent extends Event { /** * Create a new `MessageEvent`. * * @param {(String|Buffer|ArrayBuffer|Buffer[])} data The received data * @param {WebSocket} target A reference to the target to which the event was dispatched */ constructor(data, target) { super('message', target); this.data = data; } } /** * Class representing a close event. * * @extends Event * @private */ class CloseEvent extends Event { /** * Create a new `CloseEvent`. * * @param {Number} code The status code explaining why the connection is being closed * @param {String} reason A human-readable string explaining why the connection is closing * @param {WebSocket} target A reference to the target to which the event was dispatched */ constructor(code, reason, target) { super('close', target); this.wasClean = target._closeFrameReceived && target._closeFrameSent; this.reason = reason; this.code = code; } } /** * Class representing an open event. * * @extends Event * @private */ class OpenEvent extends Event { /** * Create a new `OpenEvent`. * * @param {WebSocket} target A reference to the target to which the event was dispatched */ constructor(target) { super('open', target); } } /** * Class representing an error event. * * @extends Event * @private */ class ErrorEvent extends Event { /** * Create a new `ErrorEvent`. * * @param {Object} error The error that generated this event * @param {WebSocket} target A reference to the target to which the event was dispatched */ constructor(error, target) { super('error', target); this.message = error.message; this.error = error; } } /** * This provides methods for emulating the `EventTarget` interface. It's not * meant to be used directly. * * @mixin */ const EventTarget = { /** * Register an event listener. * * @param {String} method A string representing the event type to listen for * @param {Function} listener The listener to add * @public */ addEventListener(method, listener) { if (typeof listener !== 'function') return; function onMessage(data) { listener.call(this, new MessageEvent(data, this)); } function onClose(code, message) { listener.call(this, new CloseEvent(code, message, this)); } function onError(error) { listener.call(this, new ErrorEvent(error, this)); } function onOpen() { listener.call(this, new OpenEvent(this)); } if (method === 'message') { onMessage._listener = listener; this.on(method, onMessage); } else if (method === 'close') { onClose._listener = listener; this.on(method, onClose); } else if (method === 'error') { onError._listener = listener; this.on(method, onError); } else if (method === 'open') { onOpen._listener = listener; this.on(method, onOpen); } else { this.on(method, listener); } }, /** * Remove an event listener. * * @param {String} method A string representing the event type to remove * @param {Function} listener The listener to remove * @public */ removeEventListener(method, listener) { const listeners = this.listeners(method); for (let i = 0; i < listeners.length; i++) { if (listeners[i] === listener || listeners[i]._listener === listener) { this.removeListener(method, listeners[i]); } } } }; module.exports = EventTarget; ws-7.2.1/lib/extension.js000066400000000000000000000153431357512353100153050ustar00rootroot00000000000000'use strict'; // // Allowed token characters: // // '!', '#', '$', '%', '&', ''', '*', '+', '-', // '.', 0-9, A-Z, '^', '_', '`', a-z, '|', '~' // // tokenChars[32] === 0 // ' ' // tokenChars[33] === 1 // '!' // tokenChars[34] === 0 // '"' // ... // // prettier-ignore const tokenChars = [ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 0 - 15 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 16 - 31 0, 1, 0, 1, 1, 1, 1, 1, 0, 0, 1, 1, 0, 1, 1, 0, // 32 - 47 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, // 48 - 63 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 64 - 79 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 1, 1, // 80 - 95 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 96 - 111 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 0, 1, 0 // 112 - 127 ]; /** * Adds an offer to the map of extension offers or a parameter to the map of * parameters. * * @param {Object} dest The map of extension offers or parameters * @param {String} name The extension or parameter name * @param {(Object|Boolean|String)} elem The extension parameters or the * parameter value * @private */ function push(dest, name, elem) { if (dest[name] === undefined) dest[name] = [elem]; else dest[name].push(elem); } /** * Parses the `Sec-WebSocket-Extensions` header into an object. * * @param {String} header The field value of the header * @return {Object} The parsed object * @public */ function parse(header) { const offers = Object.create(null); if (header === undefined || header === '') return offers; let params = Object.create(null); let mustUnescape = false; let isEscaping = false; let inQuotes = false; let extensionName; let paramName; let start = -1; let end = -1; let i = 0; for (; i < header.length; i++) { const code = header.charCodeAt(i); if (extensionName === undefined) { if (end === -1 && tokenChars[code] === 1) { if (start === -1) start = i; } else if (code === 0x20 /* ' ' */ || code === 0x09 /* '\t' */) { if (end === -1 && start !== -1) end = i; } else if (code === 0x3b /* ';' */ || code === 0x2c /* ',' */) { if (start === -1) { throw new SyntaxError(`Unexpected character at index ${i}`); } if (end === -1) end = i; const name = header.slice(start, end); if (code === 0x2c) { push(offers, name, params); params = Object.create(null); } else { extensionName = name; } start = end = -1; } else { throw new SyntaxError(`Unexpected character at index ${i}`); } } else if (paramName === undefined) { if (end === -1 && tokenChars[code] === 1) { if (start === -1) start = i; } else if (code === 0x20 || code === 0x09) { if (end === -1 && start !== -1) end = i; } else if (code === 0x3b || code === 0x2c) { if (start === -1) { throw new SyntaxError(`Unexpected character at index ${i}`); } if (end === -1) end = i; push(params, header.slice(start, end), true); if (code === 0x2c) { push(offers, extensionName, params); params = Object.create(null); extensionName = undefined; } start = end = -1; } else if (code === 0x3d /* '=' */ && start !== -1 && end === -1) { paramName = header.slice(start, i); start = end = -1; } else { throw new SyntaxError(`Unexpected character at index ${i}`); } } else { // // The value of a quoted-string after unescaping must conform to the // token ABNF, so only token characters are valid. // Ref: https://tools.ietf.org/html/rfc6455#section-9.1 // if (isEscaping) { if (tokenChars[code] !== 1) { throw new SyntaxError(`Unexpected character at index ${i}`); } if (start === -1) start = i; else if (!mustUnescape) mustUnescape = true; isEscaping = false; } else if (inQuotes) { if (tokenChars[code] === 1) { if (start === -1) start = i; } else if (code === 0x22 /* '"' */ && start !== -1) { inQuotes = false; end = i; } else if (code === 0x5c /* '\' */) { isEscaping = true; } else { throw new SyntaxError(`Unexpected character at index ${i}`); } } else if (code === 0x22 && header.charCodeAt(i - 1) === 0x3d) { inQuotes = true; } else if (end === -1 && tokenChars[code] === 1) { if (start === -1) start = i; } else if (start !== -1 && (code === 0x20 || code === 0x09)) { if (end === -1) end = i; } else if (code === 0x3b || code === 0x2c) { if (start === -1) { throw new SyntaxError(`Unexpected character at index ${i}`); } if (end === -1) end = i; let value = header.slice(start, end); if (mustUnescape) { value = value.replace(/\\/g, ''); mustUnescape = false; } push(params, paramName, value); if (code === 0x2c) { push(offers, extensionName, params); params = Object.create(null); extensionName = undefined; } paramName = undefined; start = end = -1; } else { throw new SyntaxError(`Unexpected character at index ${i}`); } } } if (start === -1 || inQuotes) { throw new SyntaxError('Unexpected end of input'); } if (end === -1) end = i; const token = header.slice(start, end); if (extensionName === undefined) { push(offers, token, params); } else { if (paramName === undefined) { push(params, token, true); } else if (mustUnescape) { push(params, paramName, token.replace(/\\/g, '')); } else { push(params, paramName, token); } push(offers, extensionName, params); } return offers; } /** * Builds the `Sec-WebSocket-Extensions` header field value. * * @param {Object} extensions The map of extensions and parameters to format * @return {String} A string representing the given object * @public */ function format(extensions) { return Object.keys(extensions) .map((extension) => { let configurations = extensions[extension]; if (!Array.isArray(configurations)) configurations = [configurations]; return configurations .map((params) => { return [extension] .concat( Object.keys(params).map((k) => { let values = params[k]; if (!Array.isArray(values)) values = [values]; return values .map((v) => (v === true ? k : `${k}=${v}`)) .join('; '); }) ) .join('; '); }) .join(', '); }) .join(', '); } module.exports = { format, parse }; ws-7.2.1/lib/limiter.js000066400000000000000000000017251357512353100147350ustar00rootroot00000000000000'use strict'; const kDone = Symbol('kDone'); const kRun = Symbol('kRun'); /** * A very simple job queue with adjustable concurrency. Adapted from * https://github.com/STRML/async-limiter */ class Limiter { /** * Creates a new `Limiter`. * * @param {Number} concurrency The maximum number of jobs allowed to run * concurrently */ constructor(concurrency) { this[kDone] = () => { this.pending--; this[kRun](); }; this.concurrency = concurrency || Infinity; this.jobs = []; this.pending = 0; } /** * Adds a job to the queue. * * @public */ add(job) { this.jobs.push(job); this[kRun](); } /** * Removes a job from the queue and runs it if possible. * * @private */ [kRun]() { if (this.pending === this.concurrency) return; if (this.jobs.length) { const job = this.jobs.shift(); this.pending++; job(this[kDone]); } } } module.exports = Limiter; ws-7.2.1/lib/permessage-deflate.js000066400000000000000000000336371357512353100170340ustar00rootroot00000000000000'use strict'; const zlib = require('zlib'); const bufferUtil = require('./buffer-util'); const Limiter = require('./limiter'); const { kStatusCode, NOOP } = require('./constants'); const TRAILER = Buffer.from([0x00, 0x00, 0xff, 0xff]); const kPerMessageDeflate = Symbol('permessage-deflate'); const kTotalLength = Symbol('total-length'); const kCallback = Symbol('callback'); const kBuffers = Symbol('buffers'); const kError = Symbol('error'); // // We limit zlib concurrency, which prevents severe memory fragmentation // as documented in https://github.com/nodejs/node/issues/8871#issuecomment-250915913 // and https://github.com/websockets/ws/issues/1202 // // Intentionally global; it's the global thread pool that's an issue. // let zlibLimiter; /** * permessage-deflate implementation. */ class PerMessageDeflate { /** * Creates a PerMessageDeflate instance. * * @param {Object} options Configuration options * @param {Boolean} options.serverNoContextTakeover Request/accept disabling * of server context takeover * @param {Boolean} options.clientNoContextTakeover Advertise/acknowledge * disabling of client context takeover * @param {(Boolean|Number)} options.serverMaxWindowBits Request/confirm the * use of a custom server window size * @param {(Boolean|Number)} options.clientMaxWindowBits Advertise support * for, or request, a custom client window size * @param {Object} options.zlibDeflateOptions Options to pass to zlib on deflate * @param {Object} options.zlibInflateOptions Options to pass to zlib on inflate * @param {Number} options.threshold Size (in bytes) below which messages * should not be compressed * @param {Number} options.concurrencyLimit The number of concurrent calls to * zlib * @param {Boolean} isServer Create the instance in either server or client * mode * @param {Number} maxPayload The maximum allowed message length */ constructor(options, isServer, maxPayload) { this._maxPayload = maxPayload | 0; this._options = options || {}; this._threshold = this._options.threshold !== undefined ? this._options.threshold : 1024; this._isServer = !!isServer; this._deflate = null; this._inflate = null; this.params = null; if (!zlibLimiter) { const concurrency = this._options.concurrencyLimit !== undefined ? this._options.concurrencyLimit : 10; zlibLimiter = new Limiter(concurrency); } } /** * @type {String} */ static get extensionName() { return 'permessage-deflate'; } /** * Create an extension negotiation offer. * * @return {Object} Extension parameters * @public */ offer() { const params = {}; if (this._options.serverNoContextTakeover) { params.server_no_context_takeover = true; } if (this._options.clientNoContextTakeover) { params.client_no_context_takeover = true; } if (this._options.serverMaxWindowBits) { params.server_max_window_bits = this._options.serverMaxWindowBits; } if (this._options.clientMaxWindowBits) { params.client_max_window_bits = this._options.clientMaxWindowBits; } else if (this._options.clientMaxWindowBits == null) { params.client_max_window_bits = true; } return params; } /** * Accept an extension negotiation offer/response. * * @param {Array} configurations The extension negotiation offers/reponse * @return {Object} Accepted configuration * @public */ accept(configurations) { configurations = this.normalizeParams(configurations); this.params = this._isServer ? this.acceptAsServer(configurations) : this.acceptAsClient(configurations); return this.params; } /** * Releases all resources used by the extension. * * @public */ cleanup() { if (this._inflate) { this._inflate.close(); this._inflate = null; } if (this._deflate) { if (this._deflate[kCallback]) { this._deflate[kCallback](); } this._deflate.close(); this._deflate = null; } } /** * Accept an extension negotiation offer. * * @param {Array} offers The extension negotiation offers * @return {Object} Accepted configuration * @private */ acceptAsServer(offers) { const opts = this._options; const accepted = offers.find((params) => { if ( (opts.serverNoContextTakeover === false && params.server_no_context_takeover) || (params.server_max_window_bits && (opts.serverMaxWindowBits === false || (typeof opts.serverMaxWindowBits === 'number' && opts.serverMaxWindowBits > params.server_max_window_bits))) || (typeof opts.clientMaxWindowBits === 'number' && !params.client_max_window_bits) ) { return false; } return true; }); if (!accepted) { throw new Error('None of the extension offers can be accepted'); } if (opts.serverNoContextTakeover) { accepted.server_no_context_takeover = true; } if (opts.clientNoContextTakeover) { accepted.client_no_context_takeover = true; } if (typeof opts.serverMaxWindowBits === 'number') { accepted.server_max_window_bits = opts.serverMaxWindowBits; } if (typeof opts.clientMaxWindowBits === 'number') { accepted.client_max_window_bits = opts.clientMaxWindowBits; } else if ( accepted.client_max_window_bits === true || opts.clientMaxWindowBits === false ) { delete accepted.client_max_window_bits; } return accepted; } /** * Accept the extension negotiation response. * * @param {Array} response The extension negotiation response * @return {Object} Accepted configuration * @private */ acceptAsClient(response) { const params = response[0]; if ( this._options.clientNoContextTakeover === false && params.client_no_context_takeover ) { throw new Error('Unexpected parameter "client_no_context_takeover"'); } if (!params.client_max_window_bits) { if (typeof this._options.clientMaxWindowBits === 'number') { params.client_max_window_bits = this._options.clientMaxWindowBits; } } else if ( this._options.clientMaxWindowBits === false || (typeof this._options.clientMaxWindowBits === 'number' && params.client_max_window_bits > this._options.clientMaxWindowBits) ) { throw new Error( 'Unexpected or invalid parameter "client_max_window_bits"' ); } return params; } /** * Normalize parameters. * * @param {Array} configurations The extension negotiation offers/reponse * @return {Array} The offers/response with normalized parameters * @private */ normalizeParams(configurations) { configurations.forEach((params) => { Object.keys(params).forEach((key) => { let value = params[key]; if (value.length > 1) { throw new Error(`Parameter "${key}" must have only a single value`); } value = value[0]; if (key === 'client_max_window_bits') { if (value !== true) { const num = +value; if (!Number.isInteger(num) || num < 8 || num > 15) { throw new TypeError( `Invalid value for parameter "${key}": ${value}` ); } value = num; } else if (!this._isServer) { throw new TypeError( `Invalid value for parameter "${key}": ${value}` ); } } else if (key === 'server_max_window_bits') { const num = +value; if (!Number.isInteger(num) || num < 8 || num > 15) { throw new TypeError( `Invalid value for parameter "${key}": ${value}` ); } value = num; } else if ( key === 'client_no_context_takeover' || key === 'server_no_context_takeover' ) { if (value !== true) { throw new TypeError( `Invalid value for parameter "${key}": ${value}` ); } } else { throw new Error(`Unknown parameter "${key}"`); } params[key] = value; }); }); return configurations; } /** * Decompress data. Concurrency limited. * * @param {Buffer} data Compressed data * @param {Boolean} fin Specifies whether or not this is the last fragment * @param {Function} callback Callback * @public */ decompress(data, fin, callback) { zlibLimiter.add((done) => { this._decompress(data, fin, (err, result) => { done(); callback(err, result); }); }); } /** * Compress data. Concurrency limited. * * @param {Buffer} data Data to compress * @param {Boolean} fin Specifies whether or not this is the last fragment * @param {Function} callback Callback * @public */ compress(data, fin, callback) { zlibLimiter.add((done) => { this._compress(data, fin, (err, result) => { done(); if (err || result) { callback(err, result); } }); }); } /** * Decompress data. * * @param {Buffer} data Compressed data * @param {Boolean} fin Specifies whether or not this is the last fragment * @param {Function} callback Callback * @private */ _decompress(data, fin, callback) { const endpoint = this._isServer ? 'client' : 'server'; if (!this._inflate) { const key = `${endpoint}_max_window_bits`; const windowBits = typeof this.params[key] !== 'number' ? zlib.Z_DEFAULT_WINDOWBITS : this.params[key]; this._inflate = zlib.createInflateRaw({ ...this._options.zlibInflateOptions, windowBits }); this._inflate[kPerMessageDeflate] = this; this._inflate[kTotalLength] = 0; this._inflate[kBuffers] = []; this._inflate.on('error', inflateOnError); this._inflate.on('data', inflateOnData); } this._inflate[kCallback] = callback; this._inflate.write(data); if (fin) this._inflate.write(TRAILER); this._inflate.flush(() => { const err = this._inflate[kError]; if (err) { this._inflate.close(); this._inflate = null; callback(err); return; } const data = bufferUtil.concat( this._inflate[kBuffers], this._inflate[kTotalLength] ); if (fin && this.params[`${endpoint}_no_context_takeover`]) { this._inflate.close(); this._inflate = null; } else { this._inflate[kTotalLength] = 0; this._inflate[kBuffers] = []; } callback(null, data); }); } /** * Compress data. * * @param {Buffer} data Data to compress * @param {Boolean} fin Specifies whether or not this is the last fragment * @param {Function} callback Callback * @private */ _compress(data, fin, callback) { const endpoint = this._isServer ? 'server' : 'client'; if (!this._deflate) { const key = `${endpoint}_max_window_bits`; const windowBits = typeof this.params[key] !== 'number' ? zlib.Z_DEFAULT_WINDOWBITS : this.params[key]; this._deflate = zlib.createDeflateRaw({ ...this._options.zlibDeflateOptions, windowBits }); this._deflate[kTotalLength] = 0; this._deflate[kBuffers] = []; // // An `'error'` event is emitted, only on Node.js < 10.0.0, if the // `zlib.DeflateRaw` instance is closed while data is being processed. // This can happen if `PerMessageDeflate#cleanup()` is called at the wrong // time due to an abnormal WebSocket closure. // this._deflate.on('error', NOOP); this._deflate.on('data', deflateOnData); } this._deflate[kCallback] = callback; this._deflate.write(data); this._deflate.flush(zlib.Z_SYNC_FLUSH, () => { if (!this._deflate) { // // This `if` statement is only needed for Node.js < 10.0.0 because as of // commit https://github.com/nodejs/node/commit/5e3f5164, the flush // callback is no longer called if the deflate stream is closed while // data is being processed. // return; } let data = bufferUtil.concat( this._deflate[kBuffers], this._deflate[kTotalLength] ); if (fin) data = data.slice(0, data.length - 4); // // Ensure that the callback will not be called again in // `PerMessageDeflate#cleanup()`. // this._deflate[kCallback] = null; if (fin && this.params[`${endpoint}_no_context_takeover`]) { this._deflate.close(); this._deflate = null; } else { this._deflate[kTotalLength] = 0; this._deflate[kBuffers] = []; } callback(null, data); }); } } module.exports = PerMessageDeflate; /** * The listener of the `zlib.DeflateRaw` stream `'data'` event. * * @param {Buffer} chunk A chunk of data * @private */ function deflateOnData(chunk) { this[kBuffers].push(chunk); this[kTotalLength] += chunk.length; } /** * The listener of the `zlib.InflateRaw` stream `'data'` event. * * @param {Buffer} chunk A chunk of data * @private */ function inflateOnData(chunk) { this[kTotalLength] += chunk.length; if ( this[kPerMessageDeflate]._maxPayload < 1 || this[kTotalLength] <= this[kPerMessageDeflate]._maxPayload ) { this[kBuffers].push(chunk); return; } this[kError] = new RangeError('Max payload size exceeded'); this[kError][kStatusCode] = 1009; this.removeListener('data', inflateOnData); this.reset(); } /** * The listener of the `zlib.InflateRaw` stream `'error'` event. * * @param {Error} err The emitted error * @private */ function inflateOnError(err) { // // There is no need to call `Zlib#close()` as the handle is automatically // closed when an error is emitted. // this[kPerMessageDeflate]._inflate = null; err[kStatusCode] = 1007; this[kCallback](err); } ws-7.2.1/lib/receiver.js000066400000000000000000000273001357512353100150710ustar00rootroot00000000000000'use strict'; const { Writable } = require('stream'); const PerMessageDeflate = require('./permessage-deflate'); const { BINARY_TYPES, EMPTY_BUFFER, kStatusCode, kWebSocket } = require('./constants'); const { concat, toArrayBuffer, unmask } = require('./buffer-util'); const { isValidStatusCode, isValidUTF8 } = require('./validation'); const GET_INFO = 0; const GET_PAYLOAD_LENGTH_16 = 1; const GET_PAYLOAD_LENGTH_64 = 2; const GET_MASK = 3; const GET_DATA = 4; const INFLATING = 5; /** * HyBi Receiver implementation. * * @extends stream.Writable */ class Receiver extends Writable { /** * Creates a Receiver instance. * * @param {String} binaryType The type for binary data * @param {Object} extensions An object containing the negotiated extensions * @param {Number} maxPayload The maximum allowed message length */ constructor(binaryType, extensions, maxPayload) { super(); this._binaryType = binaryType || BINARY_TYPES[0]; this[kWebSocket] = undefined; this._extensions = extensions || {}; this._maxPayload = maxPayload | 0; this._bufferedBytes = 0; this._buffers = []; this._compressed = false; this._payloadLength = 0; this._mask = undefined; this._fragmented = 0; this._masked = false; this._fin = false; this._opcode = 0; this._totalPayloadLength = 0; this._messageLength = 0; this._fragments = []; this._state = GET_INFO; this._loop = false; } /** * Implements `Writable.prototype._write()`. * * @param {Buffer} chunk The chunk of data to write * @param {String} encoding The character encoding of `chunk` * @param {Function} cb Callback */ _write(chunk, encoding, cb) { if (this._opcode === 0x08 && this._state == GET_INFO) return cb(); this._bufferedBytes += chunk.length; this._buffers.push(chunk); this.startLoop(cb); } /** * Consumes `n` bytes from the buffered data. * * @param {Number} n The number of bytes to consume * @return {Buffer} The consumed bytes * @private */ consume(n) { this._bufferedBytes -= n; if (n === this._buffers[0].length) return this._buffers.shift(); if (n < this._buffers[0].length) { const buf = this._buffers[0]; this._buffers[0] = buf.slice(n); return buf.slice(0, n); } const dst = Buffer.allocUnsafe(n); do { const buf = this._buffers[0]; const offset = dst.length - n; if (n >= buf.length) { dst.set(this._buffers.shift(), offset); } else { dst.set(new Uint8Array(buf.buffer, buf.byteOffset, n), offset); this._buffers[0] = buf.slice(n); } n -= buf.length; } while (n > 0); return dst; } /** * Starts the parsing loop. * * @param {Function} cb Callback * @private */ startLoop(cb) { let err; this._loop = true; do { switch (this._state) { case GET_INFO: err = this.getInfo(); break; case GET_PAYLOAD_LENGTH_16: err = this.getPayloadLength16(); break; case GET_PAYLOAD_LENGTH_64: err = this.getPayloadLength64(); break; case GET_MASK: this.getMask(); break; case GET_DATA: err = this.getData(cb); break; default: // `INFLATING` this._loop = false; return; } } while (this._loop); cb(err); } /** * Reads the first two bytes of a frame. * * @return {(RangeError|undefined)} A possible error * @private */ getInfo() { if (this._bufferedBytes < 2) { this._loop = false; return; } const buf = this.consume(2); if ((buf[0] & 0x30) !== 0x00) { this._loop = false; return error(RangeError, 'RSV2 and RSV3 must be clear', true, 1002); } const compressed = (buf[0] & 0x40) === 0x40; if (compressed && !this._extensions[PerMessageDeflate.extensionName]) { this._loop = false; return error(RangeError, 'RSV1 must be clear', true, 1002); } this._fin = (buf[0] & 0x80) === 0x80; this._opcode = buf[0] & 0x0f; this._payloadLength = buf[1] & 0x7f; if (this._opcode === 0x00) { if (compressed) { this._loop = false; return error(RangeError, 'RSV1 must be clear', true, 1002); } if (!this._fragmented) { this._loop = false; return error(RangeError, 'invalid opcode 0', true, 1002); } this._opcode = this._fragmented; } else if (this._opcode === 0x01 || this._opcode === 0x02) { if (this._fragmented) { this._loop = false; return error(RangeError, `invalid opcode ${this._opcode}`, true, 1002); } this._compressed = compressed; } else if (this._opcode > 0x07 && this._opcode < 0x0b) { if (!this._fin) { this._loop = false; return error(RangeError, 'FIN must be set', true, 1002); } if (compressed) { this._loop = false; return error(RangeError, 'RSV1 must be clear', true, 1002); } if (this._payloadLength > 0x7d) { this._loop = false; return error( RangeError, `invalid payload length ${this._payloadLength}`, true, 1002 ); } } else { this._loop = false; return error(RangeError, `invalid opcode ${this._opcode}`, true, 1002); } if (!this._fin && !this._fragmented) this._fragmented = this._opcode; this._masked = (buf[1] & 0x80) === 0x80; if (this._payloadLength === 126) this._state = GET_PAYLOAD_LENGTH_16; else if (this._payloadLength === 127) this._state = GET_PAYLOAD_LENGTH_64; else return this.haveLength(); } /** * Gets extended payload length (7+16). * * @return {(RangeError|undefined)} A possible error * @private */ getPayloadLength16() { if (this._bufferedBytes < 2) { this._loop = false; return; } this._payloadLength = this.consume(2).readUInt16BE(0); return this.haveLength(); } /** * Gets extended payload length (7+64). * * @return {(RangeError|undefined)} A possible error * @private */ getPayloadLength64() { if (this._bufferedBytes < 8) { this._loop = false; return; } const buf = this.consume(8); const num = buf.readUInt32BE(0); // // The maximum safe integer in JavaScript is 2^53 - 1. An error is returned // if payload length is greater than this number. // if (num > Math.pow(2, 53 - 32) - 1) { this._loop = false; return error( RangeError, 'Unsupported WebSocket frame: payload length > 2^53 - 1', false, 1009 ); } this._payloadLength = num * Math.pow(2, 32) + buf.readUInt32BE(4); return this.haveLength(); } /** * Payload length has been read. * * @return {(RangeError|undefined)} A possible error * @private */ haveLength() { if (this._payloadLength && this._opcode < 0x08) { this._totalPayloadLength += this._payloadLength; if (this._totalPayloadLength > this._maxPayload && this._maxPayload > 0) { this._loop = false; return error(RangeError, 'Max payload size exceeded', false, 1009); } } if (this._masked) this._state = GET_MASK; else this._state = GET_DATA; } /** * Reads mask bytes. * * @private */ getMask() { if (this._bufferedBytes < 4) { this._loop = false; return; } this._mask = this.consume(4); this._state = GET_DATA; } /** * Reads data bytes. * * @param {Function} cb Callback * @return {(Error|RangeError|undefined)} A possible error * @private */ getData(cb) { let data = EMPTY_BUFFER; if (this._payloadLength) { if (this._bufferedBytes < this._payloadLength) { this._loop = false; return; } data = this.consume(this._payloadLength); if (this._masked) unmask(data, this._mask); } if (this._opcode > 0x07) return this.controlMessage(data); if (this._compressed) { this._state = INFLATING; this.decompress(data, cb); return; } if (data.length) { // // This message is not compressed so its lenght is the sum of the payload // length of all fragments. // this._messageLength = this._totalPayloadLength; this._fragments.push(data); } return this.dataMessage(); } /** * Decompresses data. * * @param {Buffer} data Compressed data * @param {Function} cb Callback * @private */ decompress(data, cb) { const perMessageDeflate = this._extensions[PerMessageDeflate.extensionName]; perMessageDeflate.decompress(data, this._fin, (err, buf) => { if (err) return cb(err); if (buf.length) { this._messageLength += buf.length; if (this._messageLength > this._maxPayload && this._maxPayload > 0) { return cb( error(RangeError, 'Max payload size exceeded', false, 1009) ); } this._fragments.push(buf); } const er = this.dataMessage(); if (er) return cb(er); this.startLoop(cb); }); } /** * Handles a data message. * * @return {(Error|undefined)} A possible error * @private */ dataMessage() { if (this._fin) { const messageLength = this._messageLength; const fragments = this._fragments; this._totalPayloadLength = 0; this._messageLength = 0; this._fragmented = 0; this._fragments = []; if (this._opcode === 2) { let data; if (this._binaryType === 'nodebuffer') { data = concat(fragments, messageLength); } else if (this._binaryType === 'arraybuffer') { data = toArrayBuffer(concat(fragments, messageLength)); } else { data = fragments; } this.emit('message', data); } else { const buf = concat(fragments, messageLength); if (!isValidUTF8(buf)) { this._loop = false; return error(Error, 'invalid UTF-8 sequence', true, 1007); } this.emit('message', buf.toString()); } } this._state = GET_INFO; } /** * Handles a control message. * * @param {Buffer} data Data to handle * @return {(Error|RangeError|undefined)} A possible error * @private */ controlMessage(data) { if (this._opcode === 0x08) { this._loop = false; if (data.length === 0) { this.emit('conclude', 1005, ''); this.end(); } else if (data.length === 1) { return error(RangeError, 'invalid payload length 1', true, 1002); } else { const code = data.readUInt16BE(0); if (!isValidStatusCode(code)) { return error(RangeError, `invalid status code ${code}`, true, 1002); } const buf = data.slice(2); if (!isValidUTF8(buf)) { return error(Error, 'invalid UTF-8 sequence', true, 1007); } this.emit('conclude', code, buf.toString()); this.end(); } } else if (this._opcode === 0x09) { this.emit('ping', data); } else { this.emit('pong', data); } this._state = GET_INFO; } } module.exports = Receiver; /** * Builds an error object. * * @param {(Error|RangeError)} ErrorCtor The error constructor * @param {String} message The error message * @param {Boolean} prefix Specifies whether or not to add a default prefix to * `message` * @param {Number} statusCode The status code * @return {(Error|RangeError)} The error * @private */ function error(ErrorCtor, message, prefix, statusCode) { const err = new ErrorCtor( prefix ? `Invalid WebSocket frame: ${message}` : message ); Error.captureStackTrace(err, error); err[kStatusCode] = statusCode; return err; } ws-7.2.1/lib/sender.js000066400000000000000000000224631357512353100145520ustar00rootroot00000000000000'use strict'; const { randomFillSync } = require('crypto'); const PerMessageDeflate = require('./permessage-deflate'); const { EMPTY_BUFFER } = require('./constants'); const { isValidStatusCode } = require('./validation'); const { mask: applyMask, toBuffer } = require('./buffer-util'); const mask = Buffer.alloc(4); /** * HyBi Sender implementation. */ class Sender { /** * Creates a Sender instance. * * @param {net.Socket} socket The connection socket * @param {Object} extensions An object containing the negotiated extensions */ constructor(socket, extensions) { this._extensions = extensions || {}; this._socket = socket; this._firstFragment = true; this._compress = false; this._bufferedBytes = 0; this._deflating = false; this._queue = []; } /** * Frames a piece of data according to the HyBi WebSocket protocol. * * @param {Buffer} data The data to frame * @param {Object} options Options object * @param {Number} options.opcode The opcode * @param {Boolean} options.readOnly Specifies whether `data` can be modified * @param {Boolean} options.fin Specifies whether or not to set the FIN bit * @param {Boolean} options.mask Specifies whether or not to mask `data` * @param {Boolean} options.rsv1 Specifies whether or not to set the RSV1 bit * @return {Buffer[]} The framed data as a list of `Buffer` instances * @public */ static frame(data, options) { const merge = options.mask && options.readOnly; let offset = options.mask ? 6 : 2; let payloadLength = data.length; if (data.length >= 65536) { offset += 8; payloadLength = 127; } else if (data.length > 125) { offset += 2; payloadLength = 126; } const target = Buffer.allocUnsafe(merge ? data.length + offset : offset); target[0] = options.fin ? options.opcode | 0x80 : options.opcode; if (options.rsv1) target[0] |= 0x40; target[1] = payloadLength; if (payloadLength === 126) { target.writeUInt16BE(data.length, 2); } else if (payloadLength === 127) { target.writeUInt32BE(0, 2); target.writeUInt32BE(data.length, 6); } if (!options.mask) return [target, data]; randomFillSync(mask, 0, 4); target[1] |= 0x80; target[offset - 4] = mask[0]; target[offset - 3] = mask[1]; target[offset - 2] = mask[2]; target[offset - 1] = mask[3]; if (merge) { applyMask(data, mask, target, offset, data.length); return [target]; } applyMask(data, mask, data, 0, data.length); return [target, data]; } /** * Sends a close message to the other peer. * * @param {(Number|undefined)} code The status code component of the body * @param {String} data The message component of the body * @param {Boolean} mask Specifies whether or not to mask the message * @param {Function} cb Callback * @public */ close(code, data, mask, cb) { let buf; if (code === undefined) { buf = EMPTY_BUFFER; } else if (typeof code !== 'number' || !isValidStatusCode(code)) { throw new TypeError('First argument must be a valid error code number'); } else if (data === undefined || data === '') { buf = Buffer.allocUnsafe(2); buf.writeUInt16BE(code, 0); } else { buf = Buffer.allocUnsafe(2 + Buffer.byteLength(data)); buf.writeUInt16BE(code, 0); buf.write(data, 2); } if (this._deflating) { this.enqueue([this.doClose, buf, mask, cb]); } else { this.doClose(buf, mask, cb); } } /** * Frames and sends a close message. * * @param {Buffer} data The message to send * @param {Boolean} mask Specifies whether or not to mask `data` * @param {Function} cb Callback * @private */ doClose(data, mask, cb) { this.sendFrame( Sender.frame(data, { fin: true, rsv1: false, opcode: 0x08, mask, readOnly: false }), cb ); } /** * Sends a ping message to the other peer. * * @param {*} data The message to send * @param {Boolean} mask Specifies whether or not to mask `data` * @param {Function} cb Callback * @public */ ping(data, mask, cb) { const buf = toBuffer(data); if (this._deflating) { this.enqueue([this.doPing, buf, mask, toBuffer.readOnly, cb]); } else { this.doPing(buf, mask, toBuffer.readOnly, cb); } } /** * Frames and sends a ping message. * * @param {*} data The message to send * @param {Boolean} mask Specifies whether or not to mask `data` * @param {Boolean} readOnly Specifies whether `data` can be modified * @param {Function} cb Callback * @private */ doPing(data, mask, readOnly, cb) { this.sendFrame( Sender.frame(data, { fin: true, rsv1: false, opcode: 0x09, mask, readOnly }), cb ); } /** * Sends a pong message to the other peer. * * @param {*} data The message to send * @param {Boolean} mask Specifies whether or not to mask `data` * @param {Function} cb Callback * @public */ pong(data, mask, cb) { const buf = toBuffer(data); if (this._deflating) { this.enqueue([this.doPong, buf, mask, toBuffer.readOnly, cb]); } else { this.doPong(buf, mask, toBuffer.readOnly, cb); } } /** * Frames and sends a pong message. * * @param {*} data The message to send * @param {Boolean} mask Specifies whether or not to mask `data` * @param {Boolean} readOnly Specifies whether `data` can be modified * @param {Function} cb Callback * @private */ doPong(data, mask, readOnly, cb) { this.sendFrame( Sender.frame(data, { fin: true, rsv1: false, opcode: 0x0a, mask, readOnly }), cb ); } /** * Sends a data message to the other peer. * * @param {*} data The message to send * @param {Object} options Options object * @param {Boolean} options.compress Specifies whether or not to compress `data` * @param {Boolean} options.binary Specifies whether `data` is binary or text * @param {Boolean} options.fin Specifies whether the fragment is the last one * @param {Boolean} options.mask Specifies whether or not to mask `data` * @param {Function} cb Callback * @public */ send(data, options, cb) { const buf = toBuffer(data); const perMessageDeflate = this._extensions[PerMessageDeflate.extensionName]; let opcode = options.binary ? 2 : 1; let rsv1 = options.compress; if (this._firstFragment) { this._firstFragment = false; if (rsv1 && perMessageDeflate) { rsv1 = buf.length >= perMessageDeflate._threshold; } this._compress = rsv1; } else { rsv1 = false; opcode = 0; } if (options.fin) this._firstFragment = true; if (perMessageDeflate) { const opts = { fin: options.fin, rsv1, opcode, mask: options.mask, readOnly: toBuffer.readOnly }; if (this._deflating) { this.enqueue([this.dispatch, buf, this._compress, opts, cb]); } else { this.dispatch(buf, this._compress, opts, cb); } } else { this.sendFrame( Sender.frame(buf, { fin: options.fin, rsv1: false, opcode, mask: options.mask, readOnly: toBuffer.readOnly }), cb ); } } /** * Dispatches a data message. * * @param {Buffer} data The message to send * @param {Boolean} compress Specifies whether or not to compress `data` * @param {Object} options Options object * @param {Number} options.opcode The opcode * @param {Boolean} options.readOnly Specifies whether `data` can be modified * @param {Boolean} options.fin Specifies whether or not to set the FIN bit * @param {Boolean} options.mask Specifies whether or not to mask `data` * @param {Boolean} options.rsv1 Specifies whether or not to set the RSV1 bit * @param {Function} cb Callback * @private */ dispatch(data, compress, options, cb) { if (!compress) { this.sendFrame(Sender.frame(data, options), cb); return; } const perMessageDeflate = this._extensions[PerMessageDeflate.extensionName]; this._deflating = true; perMessageDeflate.compress(data, options.fin, (_, buf) => { this._deflating = false; options.readOnly = false; this.sendFrame(Sender.frame(buf, options), cb); this.dequeue(); }); } /** * Executes queued send operations. * * @private */ dequeue() { while (!this._deflating && this._queue.length) { const params = this._queue.shift(); this._bufferedBytes -= params[1].length; Reflect.apply(params[0], this, params.slice(1)); } } /** * Enqueues a send operation. * * @param {Array} params Send operation parameters. * @private */ enqueue(params) { this._bufferedBytes += params[1].length; this._queue.push(params); } /** * Sends a frame. * * @param {Buffer[]} list The frame to send * @param {Function} cb Callback * @private */ sendFrame(list, cb) { if (list.length === 2) { this._socket.cork(); this._socket.write(list[0]); this._socket.write(list[1], cb); this._socket.uncork(); } else { this._socket.write(list[0], cb); } } } module.exports = Sender; ws-7.2.1/lib/stream.js000066400000000000000000000065221357512353100145630ustar00rootroot00000000000000'use strict'; const { Duplex } = require('stream'); /** * Emits the `'close'` event on a stream. * * @param {stream.Duplex} The stream. * @private */ function emitClose(stream) { stream.emit('close'); } /** * The listener of the `'end'` event. * * @private */ function duplexOnEnd() { if (!this.destroyed && this._writableState.finished) { this.destroy(); } } /** * The listener of the `'error'` event. * * @private */ function duplexOnError(err) { this.removeListener('error', duplexOnError); this.destroy(); if (this.listenerCount('error') === 0) { // Do not suppress the throwing behavior. this.emit('error', err); } } /** * Wraps a `WebSocket` in a duplex stream. * * @param {WebSocket} ws The `WebSocket` to wrap * @param {Object} options The options for the `Duplex` constructor * @return {stream.Duplex} The duplex stream * @public */ function createWebSocketStream(ws, options) { let resumeOnReceiverDrain = true; function receiverOnDrain() { if (resumeOnReceiverDrain) ws._socket.resume(); } if (ws.readyState === ws.CONNECTING) { ws.once('open', function open() { ws._receiver.removeAllListeners('drain'); ws._receiver.on('drain', receiverOnDrain); }); } else { ws._receiver.removeAllListeners('drain'); ws._receiver.on('drain', receiverOnDrain); } const duplex = new Duplex({ ...options, autoDestroy: false, emitClose: false, objectMode: false, writableObjectMode: false }); ws.on('message', function message(msg) { if (!duplex.push(msg)) { resumeOnReceiverDrain = false; ws._socket.pause(); } }); ws.once('error', function error(err) { duplex.destroy(err); }); ws.once('close', function close() { if (duplex.destroyed) return; duplex.push(null); }); duplex._destroy = function(err, callback) { if (ws.readyState === ws.CLOSED) { callback(err); process.nextTick(emitClose, duplex); return; } ws.once('close', function close() { callback(err); process.nextTick(emitClose, duplex); }); ws.terminate(); }; duplex._final = function(callback) { if (ws.readyState === ws.CONNECTING) { ws.once('open', function open() { duplex._final(callback); }); return; } if (ws._socket._writableState.finished) { if (duplex._readableState.endEmitted) duplex.destroy(); callback(); } else { ws._socket.once('finish', function finish() { // `duplex` is not destroyed here because the `'end'` event will be // emitted on `duplex` after this `'finish'` event. The EOF signaling // `null` chunk is, in fact, pushed when the WebSocket emits `'close'`. callback(); }); ws.close(); } }; duplex._read = function() { if (ws.readyState === ws.OPEN && !resumeOnReceiverDrain) { resumeOnReceiverDrain = true; if (!ws._receiver._writableState.needDrain) ws._socket.resume(); } }; duplex._write = function(chunk, encoding, callback) { if (ws.readyState === ws.CONNECTING) { ws.once('open', function open() { duplex._write(chunk, encoding, callback); }); return; } ws.send(chunk, callback); }; duplex.on('end', duplexOnEnd); duplex.on('error', duplexOnError); return duplex; } module.exports = createWebSocketStream; ws-7.2.1/lib/validation.js000066400000000000000000000012671357512353100154230ustar00rootroot00000000000000'use strict'; try { const isValidUTF8 = require('utf-8-validate'); exports.isValidUTF8 = typeof isValidUTF8 === 'object' ? isValidUTF8.Validation.isValidUTF8 // utf-8-validate@<3.0.0 : isValidUTF8; } catch (e) /* istanbul ignore next */ { exports.isValidUTF8 = () => true; } /** * Checks if a status code is allowed in a close frame. * * @param {Number} code The status code * @return {Boolean} `true` if the status code is valid, else `false` * @public */ exports.isValidStatusCode = (code) => { return ( (code >= 1000 && code <= 1013 && code !== 1004 && code !== 1005 && code !== 1006) || (code >= 3000 && code <= 4999) ); }; ws-7.2.1/lib/websocket-server.js000066400000000000000000000263041357512353100165620ustar00rootroot00000000000000'use strict'; const EventEmitter = require('events'); const { createHash } = require('crypto'); const { createServer, STATUS_CODES } = require('http'); const PerMessageDeflate = require('./permessage-deflate'); const WebSocket = require('./websocket'); const { format, parse } = require('./extension'); const { GUID } = require('./constants'); const keyRegex = /^[+/0-9A-Za-z]{22}==$/; const kUsedByWebSocketServer = Symbol('kUsedByWebSocketServer'); /** * Class representing a WebSocket server. * * @extends EventEmitter */ class WebSocketServer extends EventEmitter { /** * Create a `WebSocketServer` instance. * * @param {Object} options Configuration options * @param {Number} options.backlog The maximum length of the queue of pending * connections * @param {Boolean} options.clientTracking Specifies whether or not to track * clients * @param {Function} options.handleProtocols A hook to handle protocols * @param {String} options.host The hostname where to bind the server * @param {Number} options.maxPayload The maximum allowed message size * @param {Boolean} options.noServer Enable no server mode * @param {String} options.path Accept only connections matching this path * @param {(Boolean|Object)} options.perMessageDeflate Enable/disable * permessage-deflate * @param {Number} options.port The port where to bind the server * @param {http.Server} options.server A pre-created HTTP/S server to use * @param {Function} options.verifyClient A hook to reject connections * @param {Function} callback A listener for the `listening` event */ constructor(options, callback) { super(); options = { maxPayload: 100 * 1024 * 1024, perMessageDeflate: false, handleProtocols: null, clientTracking: true, verifyClient: null, noServer: false, backlog: null, // use default (511 as implemented in net.js) server: null, host: null, path: null, port: null, ...options }; if (options.port == null && !options.server && !options.noServer) { throw new TypeError( 'One of the "port", "server", or "noServer" options must be specified' ); } if (options.port != null) { this._server = createServer((req, res) => { const body = STATUS_CODES[426]; res.writeHead(426, { 'Content-Length': body.length, 'Content-Type': 'text/plain' }); res.end(body); }); this._server.listen( options.port, options.host, options.backlog, callback ); } else if (options.server) { if (options.server[kUsedByWebSocketServer]) { throw new Error( 'The HTTP/S server is already being used by another WebSocket server' ); } options.server[kUsedByWebSocketServer] = true; this._server = options.server; } if (this._server) { this._removeListeners = addListeners(this._server, { listening: this.emit.bind(this, 'listening'), error: this.emit.bind(this, 'error'), upgrade: (req, socket, head) => { this.handleUpgrade(req, socket, head, (ws) => { this.emit('connection', ws, req); }); } }); } if (options.perMessageDeflate === true) options.perMessageDeflate = {}; if (options.clientTracking) this.clients = new Set(); this.options = options; } /** * Returns the bound address, the address family name, and port of the server * as reported by the operating system if listening on an IP socket. * If the server is listening on a pipe or UNIX domain socket, the name is * returned as a string. * * @return {(Object|String|null)} The address of the server * @public */ address() { if (this.options.noServer) { throw new Error('The server is operating in "noServer" mode'); } if (!this._server) return null; return this._server.address(); } /** * Close the server. * * @param {Function} cb Callback * @public */ close(cb) { if (cb) this.once('close', cb); // // Terminate all associated clients. // if (this.clients) { for (const client of this.clients) client.terminate(); } const server = this._server; if (server) { this._removeListeners(); this._removeListeners = this._server = null; // // Close the http server if it was internally created. // if (this.options.port != null) { server.close(() => this.emit('close')); return; } delete server[kUsedByWebSocketServer]; } process.nextTick(emitClose, this); } /** * See if a given request should be handled by this server instance. * * @param {http.IncomingMessage} req Request object to inspect * @return {Boolean} `true` if the request is valid, else `false` * @public */ shouldHandle(req) { if (this.options.path) { const index = req.url.indexOf('?'); const pathname = index !== -1 ? req.url.slice(0, index) : req.url; if (pathname !== this.options.path) return false; } return true; } /** * Handle a HTTP Upgrade request. * * @param {http.IncomingMessage} req The request object * @param {net.Socket} socket The network socket between the server and client * @param {Buffer} head The first packet of the upgraded stream * @param {Function} cb Callback * @public */ handleUpgrade(req, socket, head, cb) { socket.on('error', socketOnError); const key = req.headers['sec-websocket-key'] !== undefined ? req.headers['sec-websocket-key'].trim() : false; const version = +req.headers['sec-websocket-version']; const extensions = {}; if ( req.method !== 'GET' || req.headers.upgrade.toLowerCase() !== 'websocket' || !key || !keyRegex.test(key) || (version !== 8 && version !== 13) || !this.shouldHandle(req) ) { return abortHandshake(socket, 400); } if (this.options.perMessageDeflate) { const perMessageDeflate = new PerMessageDeflate( this.options.perMessageDeflate, true, this.options.maxPayload ); try { const offers = parse(req.headers['sec-websocket-extensions']); if (offers[PerMessageDeflate.extensionName]) { perMessageDeflate.accept(offers[PerMessageDeflate.extensionName]); extensions[PerMessageDeflate.extensionName] = perMessageDeflate; } } catch (err) { return abortHandshake(socket, 400); } } // // Optionally call external client verification handler. // if (this.options.verifyClient) { const info = { origin: req.headers[`${version === 8 ? 'sec-websocket-origin' : 'origin'}`], secure: !!(req.connection.authorized || req.connection.encrypted), req }; if (this.options.verifyClient.length === 2) { this.options.verifyClient(info, (verified, code, message, headers) => { if (!verified) { return abortHandshake(socket, code || 401, message, headers); } this.completeUpgrade(key, extensions, req, socket, head, cb); }); return; } if (!this.options.verifyClient(info)) return abortHandshake(socket, 401); } this.completeUpgrade(key, extensions, req, socket, head, cb); } /** * Upgrade the connection to WebSocket. * * @param {String} key The value of the `Sec-WebSocket-Key` header * @param {Object} extensions The accepted extensions * @param {http.IncomingMessage} req The request object * @param {net.Socket} socket The network socket between the server and client * @param {Buffer} head The first packet of the upgraded stream * @param {Function} cb Callback * @private */ completeUpgrade(key, extensions, req, socket, head, cb) { // // Destroy the socket if the client has already sent a FIN packet. // if (!socket.readable || !socket.writable) return socket.destroy(); const digest = createHash('sha1') .update(key + GUID) .digest('base64'); const headers = [ 'HTTP/1.1 101 Switching Protocols', 'Upgrade: websocket', 'Connection: Upgrade', `Sec-WebSocket-Accept: ${digest}` ]; const ws = new WebSocket(null); let protocol = req.headers['sec-websocket-protocol']; if (protocol) { protocol = protocol.trim().split(/ *, */); // // Optionally call external protocol selection handler. // if (this.options.handleProtocols) { protocol = this.options.handleProtocols(protocol, req); } else { protocol = protocol[0]; } if (protocol) { headers.push(`Sec-WebSocket-Protocol: ${protocol}`); ws.protocol = protocol; } } if (extensions[PerMessageDeflate.extensionName]) { const params = extensions[PerMessageDeflate.extensionName].params; const value = format({ [PerMessageDeflate.extensionName]: [params] }); headers.push(`Sec-WebSocket-Extensions: ${value}`); ws._extensions = extensions; } // // Allow external modification/inspection of handshake headers. // this.emit('headers', headers, req); socket.write(headers.concat('\r\n').join('\r\n')); socket.removeListener('error', socketOnError); ws.setSocket(socket, head, this.options.maxPayload); if (this.clients) { this.clients.add(ws); ws.on('close', () => this.clients.delete(ws)); } cb(ws); } } module.exports = WebSocketServer; /** * Add event listeners on an `EventEmitter` using a map of * pairs. * * @param {EventEmitter} server The event emitter * @param {Object.} map The listeners to add * @return {Function} A function that will remove the added listeners when called * @private */ function addListeners(server, map) { for (const event of Object.keys(map)) server.on(event, map[event]); return function removeListeners() { for (const event of Object.keys(map)) { server.removeListener(event, map[event]); } }; } /** * Emit a `'close'` event on an `EventEmitter`. * * @param {EventEmitter} server The event emitter * @private */ function emitClose(server) { server.emit('close'); } /** * Handle premature socket errors. * * @private */ function socketOnError() { this.destroy(); } /** * Close the connection when preconditions are not fulfilled. * * @param {net.Socket} socket The socket of the upgrade request * @param {Number} code The HTTP response status code * @param {String} [message] The HTTP response body * @param {Object} [headers] Additional HTTP response headers * @private */ function abortHandshake(socket, code, message, headers) { if (socket.writable) { message = message || STATUS_CODES[code]; headers = { Connection: 'close', 'Content-type': 'text/html', 'Content-Length': Buffer.byteLength(message), ...headers }; socket.write( `HTTP/1.1 ${code} ${STATUS_CODES[code]}\r\n` + Object.keys(headers) .map((h) => `${h}: ${headers[h]}`) .join('\r\n') + '\r\n\r\n' + message ); } socket.removeListener('error', socketOnError); socket.destroy(); } ws-7.2.1/lib/websocket.js000066400000000000000000000573231357512353100152630ustar00rootroot00000000000000'use strict'; const EventEmitter = require('events'); const https = require('https'); const http = require('http'); const net = require('net'); const tls = require('tls'); const { randomBytes, createHash } = require('crypto'); const { URL } = require('url'); const PerMessageDeflate = require('./permessage-deflate'); const Receiver = require('./receiver'); const Sender = require('./sender'); const { BINARY_TYPES, EMPTY_BUFFER, GUID, kStatusCode, kWebSocket, NOOP } = require('./constants'); const { addEventListener, removeEventListener } = require('./event-target'); const { format, parse } = require('./extension'); const { toBuffer } = require('./buffer-util'); const readyStates = ['CONNECTING', 'OPEN', 'CLOSING', 'CLOSED']; const protocolVersions = [8, 13]; const closeTimeout = 30 * 1000; /** * Class representing a WebSocket. * * @extends EventEmitter */ class WebSocket extends EventEmitter { /** * Create a new `WebSocket`. * * @param {(String|url.URL)} address The URL to which to connect * @param {(String|String[])} protocols The subprotocols * @param {Object} options Connection options */ constructor(address, protocols, options) { super(); this.readyState = WebSocket.CONNECTING; this.protocol = ''; this._binaryType = BINARY_TYPES[0]; this._closeFrameReceived = false; this._closeFrameSent = false; this._closeMessage = ''; this._closeTimer = null; this._closeCode = 1006; this._extensions = {}; this._receiver = null; this._sender = null; this._socket = null; if (address !== null) { this._bufferedAmount = 0; this._isServer = false; this._redirects = 0; if (Array.isArray(protocols)) { protocols = protocols.join(', '); } else if (typeof protocols === 'object' && protocols !== null) { options = protocols; protocols = undefined; } initAsClient(this, address, protocols, options); } else { this._isServer = true; } } get CONNECTING() { return WebSocket.CONNECTING; } get CLOSING() { return WebSocket.CLOSING; } get CLOSED() { return WebSocket.CLOSED; } get OPEN() { return WebSocket.OPEN; } /** * This deviates from the WHATWG interface since ws doesn't support the * required default "blob" type (instead we define a custom "nodebuffer" * type). * * @type {String} */ get binaryType() { return this._binaryType; } set binaryType(type) { if (!BINARY_TYPES.includes(type)) return; this._binaryType = type; // // Allow to change `binaryType` on the fly. // if (this._receiver) this._receiver._binaryType = type; } /** * @type {Number} */ get bufferedAmount() { if (!this._socket) return this._bufferedAmount; // // `socket.bufferSize` is `undefined` if the socket is closed. // return (this._socket.bufferSize || 0) + this._sender._bufferedBytes; } /** * @type {String} */ get extensions() { return Object.keys(this._extensions).join(); } /** * Set up the socket and the internal resources. * * @param {net.Socket} socket The network socket between the server and client * @param {Buffer} head The first packet of the upgraded stream * @param {Number} maxPayload The maximum allowed message size * @private */ setSocket(socket, head, maxPayload) { const receiver = new Receiver( this._binaryType, this._extensions, maxPayload ); this._sender = new Sender(socket, this._extensions); this._receiver = receiver; this._socket = socket; receiver[kWebSocket] = this; socket[kWebSocket] = this; receiver.on('conclude', receiverOnConclude); receiver.on('drain', receiverOnDrain); receiver.on('error', receiverOnError); receiver.on('message', receiverOnMessage); receiver.on('ping', receiverOnPing); receiver.on('pong', receiverOnPong); socket.setTimeout(0); socket.setNoDelay(); if (head.length > 0) socket.unshift(head); socket.on('close', socketOnClose); socket.on('data', socketOnData); socket.on('end', socketOnEnd); socket.on('error', socketOnError); this.readyState = WebSocket.OPEN; this.emit('open'); } /** * Emit the `'close'` event. * * @private */ emitClose() { this.readyState = WebSocket.CLOSED; if (!this._socket) { this.emit('close', this._closeCode, this._closeMessage); return; } if (this._extensions[PerMessageDeflate.extensionName]) { this._extensions[PerMessageDeflate.extensionName].cleanup(); } this._receiver.removeAllListeners(); this.emit('close', this._closeCode, this._closeMessage); } /** * Start a closing handshake. * * +----------+ +-----------+ +----------+ * - - -|ws.close()|-->|close frame|-->|ws.close()|- - - * | +----------+ +-----------+ +----------+ | * +----------+ +-----------+ | * CLOSING |ws.close()|<--|close frame|<--+-----+ CLOSING * +----------+ +-----------+ | * | | | +---+ | * +------------------------+-->|fin| - - - - * | +---+ | +---+ * - - - - -|fin|<---------------------+ * +---+ * * @param {Number} code Status code explaining why the connection is closing * @param {String} data A string explaining why the connection is closing * @public */ close(code, data) { if (this.readyState === WebSocket.CLOSED) return; if (this.readyState === WebSocket.CONNECTING) { const msg = 'WebSocket was closed before the connection was established'; return abortHandshake(this, this._req, msg); } if (this.readyState === WebSocket.CLOSING) { if (this._closeFrameSent && this._closeFrameReceived) this._socket.end(); return; } this.readyState = WebSocket.CLOSING; this._sender.close(code, data, !this._isServer, (err) => { // // This error is handled by the `'error'` listener on the socket. We only // want to know if the close frame has been sent here. // if (err) return; this._closeFrameSent = true; if (this._closeFrameReceived) this._socket.end(); }); // // Specify a timeout for the closing handshake to complete. // this._closeTimer = setTimeout( this._socket.destroy.bind(this._socket), closeTimeout ); } /** * Send a ping. * * @param {*} data The data to send * @param {Boolean} mask Indicates whether or not to mask `data` * @param {Function} cb Callback which is executed when the ping is sent * @public */ ping(data, mask, cb) { if (this.readyState === WebSocket.CONNECTING) { throw new Error('WebSocket is not open: readyState 0 (CONNECTING)'); } if (typeof data === 'function') { cb = data; data = mask = undefined; } else if (typeof mask === 'function') { cb = mask; mask = undefined; } if (typeof data === 'number') data = data.toString(); if (this.readyState !== WebSocket.OPEN) { sendAfterClose(this, data, cb); return; } if (mask === undefined) mask = !this._isServer; this._sender.ping(data || EMPTY_BUFFER, mask, cb); } /** * Send a pong. * * @param {*} data The data to send * @param {Boolean} mask Indicates whether or not to mask `data` * @param {Function} cb Callback which is executed when the pong is sent * @public */ pong(data, mask, cb) { if (this.readyState === WebSocket.CONNECTING) { throw new Error('WebSocket is not open: readyState 0 (CONNECTING)'); } if (typeof data === 'function') { cb = data; data = mask = undefined; } else if (typeof mask === 'function') { cb = mask; mask = undefined; } if (typeof data === 'number') data = data.toString(); if (this.readyState !== WebSocket.OPEN) { sendAfterClose(this, data, cb); return; } if (mask === undefined) mask = !this._isServer; this._sender.pong(data || EMPTY_BUFFER, mask, cb); } /** * Send a data message. * * @param {*} data The message to send * @param {Object} options Options object * @param {Boolean} options.compress Specifies whether or not to compress * `data` * @param {Boolean} options.binary Specifies whether `data` is binary or text * @param {Boolean} options.fin Specifies whether the fragment is the last one * @param {Boolean} options.mask Specifies whether or not to mask `data` * @param {Function} cb Callback which is executed when data is written out * @public */ send(data, options, cb) { if (this.readyState === WebSocket.CONNECTING) { throw new Error('WebSocket is not open: readyState 0 (CONNECTING)'); } if (typeof options === 'function') { cb = options; options = {}; } if (typeof data === 'number') data = data.toString(); if (this.readyState !== WebSocket.OPEN) { sendAfterClose(this, data, cb); return; } const opts = { binary: typeof data !== 'string', mask: !this._isServer, compress: true, fin: true, ...options }; if (!this._extensions[PerMessageDeflate.extensionName]) { opts.compress = false; } this._sender.send(data || EMPTY_BUFFER, opts, cb); } /** * Forcibly close the connection. * * @public */ terminate() { if (this.readyState === WebSocket.CLOSED) return; if (this.readyState === WebSocket.CONNECTING) { const msg = 'WebSocket was closed before the connection was established'; return abortHandshake(this, this._req, msg); } if (this._socket) { this.readyState = WebSocket.CLOSING; this._socket.destroy(); } } } readyStates.forEach((readyState, i) => { WebSocket[readyState] = i; }); // // Add the `onopen`, `onerror`, `onclose`, and `onmessage` attributes. // See https://html.spec.whatwg.org/multipage/comms.html#the-websocket-interface // ['open', 'error', 'close', 'message'].forEach((method) => { Object.defineProperty(WebSocket.prototype, `on${method}`, { /** * Return the listener of the event. * * @return {(Function|undefined)} The event listener or `undefined` * @public */ get() { const listeners = this.listeners(method); for (let i = 0; i < listeners.length; i++) { if (listeners[i]._listener) return listeners[i]._listener; } return undefined; }, /** * Add a listener for the event. * * @param {Function} listener The listener to add * @public */ set(listener) { const listeners = this.listeners(method); for (let i = 0; i < listeners.length; i++) { // // Remove only the listeners added via `addEventListener`. // if (listeners[i]._listener) this.removeListener(method, listeners[i]); } this.addEventListener(method, listener); } }); }); WebSocket.prototype.addEventListener = addEventListener; WebSocket.prototype.removeEventListener = removeEventListener; module.exports = WebSocket; /** * Initialize a WebSocket client. * * @param {WebSocket} websocket The client to initialize * @param {(String|url.URL)} address The URL to which to connect * @param {String} protocols The subprotocols * @param {Object} options Connection options * @param {(Boolean|Object)} options.perMessageDeflate Enable/disable * permessage-deflate * @param {Number} options.handshakeTimeout Timeout in milliseconds for the * handshake request * @param {Number} options.protocolVersion Value of the `Sec-WebSocket-Version` * header * @param {String} options.origin Value of the `Origin` or * `Sec-WebSocket-Origin` header * @param {Number} options.maxPayload The maximum allowed message size * @param {Boolean} options.followRedirects Whether or not to follow redirects * @param {Number} options.maxRedirects The maximum number of redirects allowed * @private */ function initAsClient(websocket, address, protocols, options) { const opts = { protocolVersion: protocolVersions[1], maxPayload: 100 * 1024 * 1024, perMessageDeflate: true, followRedirects: false, maxRedirects: 10, ...options, createConnection: undefined, socketPath: undefined, hostname: undefined, protocol: undefined, timeout: undefined, method: undefined, auth: undefined, host: undefined, path: undefined, port: undefined }; if (!protocolVersions.includes(opts.protocolVersion)) { throw new RangeError( `Unsupported protocol version: ${opts.protocolVersion} ` + `(supported versions: ${protocolVersions.join(', ')})` ); } let parsedUrl; if (address instanceof URL) { parsedUrl = address; websocket.url = address.href; } else { parsedUrl = new URL(address); websocket.url = address; } const isUnixSocket = parsedUrl.protocol === 'ws+unix:'; if (!parsedUrl.host && (!isUnixSocket || !parsedUrl.pathname)) { throw new Error(`Invalid URL: ${websocket.url}`); } const isSecure = parsedUrl.protocol === 'wss:' || parsedUrl.protocol === 'https:'; const defaultPort = isSecure ? 443 : 80; const key = randomBytes(16).toString('base64'); const get = isSecure ? https.get : http.get; let perMessageDeflate; opts.createConnection = isSecure ? tlsConnect : netConnect; opts.defaultPort = opts.defaultPort || defaultPort; opts.port = parsedUrl.port || defaultPort; opts.host = parsedUrl.hostname.startsWith('[') ? parsedUrl.hostname.slice(1, -1) : parsedUrl.hostname; opts.headers = { 'Sec-WebSocket-Version': opts.protocolVersion, 'Sec-WebSocket-Key': key, Connection: 'Upgrade', Upgrade: 'websocket', ...opts.headers }; opts.path = parsedUrl.pathname + parsedUrl.search; opts.timeout = opts.handshakeTimeout; if (opts.perMessageDeflate) { perMessageDeflate = new PerMessageDeflate( opts.perMessageDeflate !== true ? opts.perMessageDeflate : {}, false, opts.maxPayload ); opts.headers['Sec-WebSocket-Extensions'] = format({ [PerMessageDeflate.extensionName]: perMessageDeflate.offer() }); } if (protocols) { opts.headers['Sec-WebSocket-Protocol'] = protocols; } if (opts.origin) { if (opts.protocolVersion < 13) { opts.headers['Sec-WebSocket-Origin'] = opts.origin; } else { opts.headers.Origin = opts.origin; } } if (parsedUrl.username || parsedUrl.password) { opts.auth = `${parsedUrl.username}:${parsedUrl.password}`; } if (isUnixSocket) { const parts = opts.path.split(':'); opts.socketPath = parts[0]; opts.path = parts[1]; } let req = (websocket._req = get(opts)); if (opts.timeout) { req.on('timeout', () => { abortHandshake(websocket, req, 'Opening handshake has timed out'); }); } req.on('error', (err) => { if (websocket._req.aborted) return; req = websocket._req = null; websocket.readyState = WebSocket.CLOSING; websocket.emit('error', err); websocket.emitClose(); }); req.on('response', (res) => { const location = res.headers.location; const statusCode = res.statusCode; if ( location && opts.followRedirects && statusCode >= 300 && statusCode < 400 ) { if (++websocket._redirects > opts.maxRedirects) { abortHandshake(websocket, req, 'Maximum redirects exceeded'); return; } req.abort(); const addr = new URL(location, address); initAsClient(websocket, addr, protocols, options); } else if (!websocket.emit('unexpected-response', req, res)) { abortHandshake( websocket, req, `Unexpected server response: ${res.statusCode}` ); } }); req.on('upgrade', (res, socket, head) => { websocket.emit('upgrade', res); // // The user may have closed the connection from a listener of the `upgrade` // event. // if (websocket.readyState !== WebSocket.CONNECTING) return; req = websocket._req = null; const digest = createHash('sha1') .update(key + GUID) .digest('base64'); if (res.headers['sec-websocket-accept'] !== digest) { abortHandshake(websocket, socket, 'Invalid Sec-WebSocket-Accept header'); return; } const serverProt = res.headers['sec-websocket-protocol']; const protList = (protocols || '').split(/, */); let protError; if (!protocols && serverProt) { protError = 'Server sent a subprotocol but none was requested'; } else if (protocols && !serverProt) { protError = 'Server sent no subprotocol'; } else if (serverProt && !protList.includes(serverProt)) { protError = 'Server sent an invalid subprotocol'; } if (protError) { abortHandshake(websocket, socket, protError); return; } if (serverProt) websocket.protocol = serverProt; if (perMessageDeflate) { try { const extensions = parse(res.headers['sec-websocket-extensions']); if (extensions[PerMessageDeflate.extensionName]) { perMessageDeflate.accept(extensions[PerMessageDeflate.extensionName]); websocket._extensions[ PerMessageDeflate.extensionName ] = perMessageDeflate; } } catch (err) { abortHandshake( websocket, socket, 'Invalid Sec-WebSocket-Extensions header' ); return; } } websocket.setSocket(socket, head, opts.maxPayload); }); } /** * Create a `net.Socket` and initiate a connection. * * @param {Object} options Connection options * @return {net.Socket} The newly created socket used to start the connection * @private */ function netConnect(options) { options.path = options.socketPath; return net.connect(options); } /** * Create a `tls.TLSSocket` and initiate a connection. * * @param {Object} options Connection options * @return {tls.TLSSocket} The newly created socket used to start the connection * @private */ function tlsConnect(options) { options.path = undefined; if (!options.servername && options.servername !== '') { options.servername = options.host; } return tls.connect(options); } /** * Abort the handshake and emit an error. * * @param {WebSocket} websocket The WebSocket instance * @param {(http.ClientRequest|net.Socket)} stream The request to abort or the * socket to destroy * @param {String} message The error message * @private */ function abortHandshake(websocket, stream, message) { websocket.readyState = WebSocket.CLOSING; const err = new Error(message); Error.captureStackTrace(err, abortHandshake); if (stream.setHeader) { stream.abort(); stream.once('abort', websocket.emitClose.bind(websocket)); websocket.emit('error', err); } else { stream.destroy(err); stream.once('error', websocket.emit.bind(websocket, 'error')); stream.once('close', websocket.emitClose.bind(websocket)); } } /** * Handle cases where the `ping()`, `pong()`, or `send()` methods are called * when the `readyState` attribute is `CLOSING` or `CLOSED`. * * @param {WebSocket} websocket The WebSocket instance * @param {*} data The data to send * @param {Function} cb Callback * @private */ function sendAfterClose(websocket, data, cb) { if (data) { const length = toBuffer(data).length; // // The `_bufferedAmount` property is used only when the peer is a client and // the opening handshake fails. Under these circumstances, in fact, the // `setSocket()` method is not called, so the `_socket` and `_sender` // properties are set to `null`. // if (websocket._socket) websocket._sender._bufferedBytes += length; else websocket._bufferedAmount += length; } if (cb) { const err = new Error( `WebSocket is not open: readyState ${websocket.readyState} ` + `(${readyStates[websocket.readyState]})` ); cb(err); } } /** * The listener of the `Receiver` `'conclude'` event. * * @param {Number} code The status code * @param {String} reason The reason for closing * @private */ function receiverOnConclude(code, reason) { const websocket = this[kWebSocket]; websocket._socket.removeListener('data', socketOnData); websocket._socket.resume(); websocket._closeFrameReceived = true; websocket._closeMessage = reason; websocket._closeCode = code; if (code === 1005) websocket.close(); else websocket.close(code, reason); } /** * The listener of the `Receiver` `'drain'` event. * * @private */ function receiverOnDrain() { this[kWebSocket]._socket.resume(); } /** * The listener of the `Receiver` `'error'` event. * * @param {(RangeError|Error)} err The emitted error * @private */ function receiverOnError(err) { const websocket = this[kWebSocket]; websocket._socket.removeListener('data', socketOnData); websocket.readyState = WebSocket.CLOSING; websocket._closeCode = err[kStatusCode]; websocket.emit('error', err); websocket._socket.destroy(); } /** * The listener of the `Receiver` `'finish'` event. * * @private */ function receiverOnFinish() { this[kWebSocket].emitClose(); } /** * The listener of the `Receiver` `'message'` event. * * @param {(String|Buffer|ArrayBuffer|Buffer[])} data The message * @private */ function receiverOnMessage(data) { this[kWebSocket].emit('message', data); } /** * The listener of the `Receiver` `'ping'` event. * * @param {Buffer} data The data included in the ping frame * @private */ function receiverOnPing(data) { const websocket = this[kWebSocket]; websocket.pong(data, !websocket._isServer, NOOP); websocket.emit('ping', data); } /** * The listener of the `Receiver` `'pong'` event. * * @param {Buffer} data The data included in the pong frame * @private */ function receiverOnPong(data) { this[kWebSocket].emit('pong', data); } /** * The listener of the `net.Socket` `'close'` event. * * @private */ function socketOnClose() { const websocket = this[kWebSocket]; this.removeListener('close', socketOnClose); this.removeListener('end', socketOnEnd); websocket.readyState = WebSocket.CLOSING; // // The close frame might not have been received or the `'end'` event emitted, // for example, if the socket was destroyed due to an error. Ensure that the // `receiver` stream is closed after writing any remaining buffered data to // it. If the readable side of the socket is in flowing mode then there is no // buffered data as everything has been already written and `readable.read()` // will return `null`. If instead, the socket is paused, any possible buffered // data will be read as a single chunk and emitted synchronously in a single // `'data'` event. // websocket._socket.read(); websocket._receiver.end(); this.removeListener('data', socketOnData); this[kWebSocket] = undefined; clearTimeout(websocket._closeTimer); if ( websocket._receiver._writableState.finished || websocket._receiver._writableState.errorEmitted ) { websocket.emitClose(); } else { websocket._receiver.on('error', receiverOnFinish); websocket._receiver.on('finish', receiverOnFinish); } } /** * The listener of the `net.Socket` `'data'` event. * * @param {Buffer} chunk A chunk of data * @private */ function socketOnData(chunk) { if (!this[kWebSocket]._receiver.write(chunk)) { this.pause(); } } /** * The listener of the `net.Socket` `'end'` event. * * @private */ function socketOnEnd() { const websocket = this[kWebSocket]; websocket.readyState = WebSocket.CLOSING; websocket._receiver.end(); this.end(); } /** * The listener of the `net.Socket` `'error'` event. * * @private */ function socketOnError() { const websocket = this[kWebSocket]; this.removeListener('error', socketOnError); this.on('error', NOOP); if (websocket) { websocket.readyState = WebSocket.CLOSING; this.destroy(); } } ws-7.2.1/package.json000066400000000000000000000033071357512353100144500ustar00rootroot00000000000000{ "name": "ws", "version": "7.2.1", "description": "Simple to use, blazing fast and thoroughly tested websocket client and server for Node.js", "keywords": [ "HyBi", "Push", "RFC-6455", "WebSocket", "WebSockets", "real-time" ], "homepage": "https://github.com/websockets/ws", "bugs": "https://github.com/websockets/ws/issues", "repository": "websockets/ws", "author": "Einar Otto Stangvik (http://2x.io)", "license": "MIT", "main": "index.js", "browser": "browser.js", "engines": { "node": ">=8.3.0" }, "files": [ "browser.js", "index.js", "lib/*.js" ], "scripts": { "test": "npm run lint && nyc --reporter=html --reporter=text mocha --throw-deprecation test/*.test.js", "integration": "npm run lint && mocha --throw-deprecation test/*.integration.js", "lint": "eslint --ignore-path .gitignore . && prettier --check --ignore-path .gitignore \"**/*.{json,md,yaml,yml}\"" }, "peerDependencies": { "bufferutil": "^4.0.1", "utf-8-validate": "^5.0.2" }, "peerDependenciesMeta": { "bufferutil": { "optional": true }, "utf-8-validate": { "optional": true } }, "devDependencies": { "benchmark": "^2.1.4", "bufferutil": "^4.0.1", "coveralls": "^3.0.3", "eslint": "^6.0.0", "eslint-config-prettier": "^6.0.0", "eslint-plugin-prettier": "^3.0.1", "mocha": "^6.1.3", "nyc": "^14.0.0", "prettier": "^1.17.0", "utf-8-validate": "^5.0.2" }, "greenkeeper": { "commitMessages": { "dependencyUpdate": "[pkg] Update ${dependency} to version ${version}", "devDependencyUpdate": "[pkg] Update ${dependency} to version ${version}" } } } ws-7.2.1/test/000077500000000000000000000000001357512353100131365ustar00rootroot00000000000000ws-7.2.1/test/autobahn-server.js000066400000000000000000000006111357512353100165770ustar00rootroot00000000000000'use strict'; const WebSocket = require('../'); const port = process.argv.length > 2 ? parseInt(process.argv[2]) : 9001; const wss = new WebSocket.Server({ port }, () => { console.log( `Listening to port ${port}. Use extra argument to define the port` ); }); wss.on('connection', (ws) => { ws.on('message', (data) => ws.send(data)); ws.on('error', (e) => console.error(e)); }); ws-7.2.1/test/autobahn.js000066400000000000000000000013731357512353100153010ustar00rootroot00000000000000'use strict'; const WebSocket = require('../'); let currentTest = 1; let testCount; function nextTest() { let ws; if (currentTest > testCount) { ws = new WebSocket('ws://localhost:9001/updateReports?agent=ws'); return; } console.log(`Running test case ${currentTest}/${testCount}`); ws = new WebSocket( `ws://localhost:9001/runCase?case=${currentTest}&agent=ws` ); ws.on('message', (data) => ws.send(data)); ws.on('close', () => { currentTest++; process.nextTick(nextTest); }); ws.on('error', (e) => console.error(e)); } const ws = new WebSocket('ws://localhost:9001/getCaseCount'); ws.on('message', (data) => { testCount = parseInt(data); }); ws.on('close', () => { if (testCount > 0) { nextTest(); } }); ws-7.2.1/test/buffer-util.test.js000066400000000000000000000005501357512353100166760ustar00rootroot00000000000000'use strict'; const assert = require('assert'); const { concat } = require('../lib/buffer-util'); describe('bufferUtil', () => { describe('concat', () => { it('never returns uninitialized data', () => { const buf = concat([Buffer.from([1, 2]), Buffer.from([3, 4])], 6); assert.ok(buf.equals(Buffer.from([1, 2, 3, 4]))); }); }); }); ws-7.2.1/test/create-websocket-stream.test.js000066400000000000000000000262251357512353100212010ustar00rootroot00000000000000'use strict'; const assert = require('assert'); const EventEmitter = require('events'); const { Duplex } = require('stream'); const { randomBytes } = require('crypto'); const createWebSocketStream = require('../lib/stream'); const Sender = require('../lib/sender'); const WebSocket = require('..'); describe('createWebSocketStream', () => { it('is exposed as a property of the `WebSocket` class', () => { assert.strictEqual(WebSocket.createWebSocketStream, createWebSocketStream); }); it('returns a `Duplex` stream', () => { const duplex = createWebSocketStream(new EventEmitter()); assert.ok(duplex instanceof Duplex); }); it('passes the options object to the `Duplex` constructor', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const duplex = createWebSocketStream(ws, { allowHalfOpen: false, encoding: 'utf8' }); duplex.on('data', (chunk) => { assert.strictEqual(chunk, 'hi'); duplex.on('close', () => { wss.close(done); }); }); }); wss.on('connection', (ws) => { ws.send(Buffer.from('hi')); ws.close(); }); }); describe('The returned stream', () => { it('buffers writes if `readyState` is `CONNECTING`', (done) => { const chunk = randomBytes(1024); const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); assert.strictEqual(ws.readyState, 0); const duplex = createWebSocketStream(ws); duplex.write(chunk); }); wss.on('connection', (ws) => { ws.on('message', (message) => { ws.on('close', (code, reason) => { assert.ok(message.equals(chunk)); assert.strictEqual(code, 1005); assert.strictEqual(reason, ''); wss.close(done); }); }); ws.close(); }); }); it('errors if a write occurs when `readyState` is `CLOSING`', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const duplex = createWebSocketStream(ws); duplex.on('error', (err) => { assert.ok(duplex.destroyed); assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket is not open: readyState 2 (CLOSING)' ); duplex.on('close', () => { wss.close(done); }); }); ws.on('open', () => { ws._receiver.on('conclude', () => { duplex.write('hi'); }); }); }); wss.on('connection', (ws) => { ws.close(); }); }); it('errors if a write occurs when `readyState` is `CLOSED`', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const duplex = createWebSocketStream(ws); duplex.on('error', (err) => { assert.ok(duplex.destroyed); assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket is not open: readyState 3 (CLOSED)' ); duplex.on('close', () => { wss.close(done); }); }); ws.on('close', () => { duplex.write('hi'); }); }); wss.on('connection', (ws) => { ws.close(); }); }); it('does not error if `_final()` is called while connecting', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); assert.strictEqual(ws.readyState, 0); const duplex = createWebSocketStream(ws); duplex.on('close', () => { wss.close(done); }); duplex.resume(); duplex.end(); }); }); it('reemits errors', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const duplex = createWebSocketStream(ws); duplex.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: invalid opcode 5' ); duplex.on('close', () => { wss.close(done); }); }); }); wss.on('connection', (ws) => { ws._socket.write(Buffer.from([0x85, 0x00])); }); }); it("does not suppress the throwing behavior of 'error' events", (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); createWebSocketStream(ws); }); wss.on('connection', (ws) => { ws._socket.write(Buffer.from([0x85, 0x00])); }); assert.strictEqual(process.listenerCount('uncaughtException'), 1); const [listener] = process.listeners('uncaughtException'); process.removeAllListeners('uncaughtException'); process.once('uncaughtException', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'Invalid WebSocket frame: invalid opcode 5' ); process.on('uncaughtException', listener); wss.close(done); }); }); it("is destroyed after 'end' and 'finish' are emitted (1/2)", (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const events = []; const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const duplex = createWebSocketStream(ws); duplex.on('end', () => { events.push('end'); assert.ok(duplex.destroyed); }); duplex.on('close', () => { assert.deepStrictEqual(events, ['finish', 'end']); wss.close(done); }); duplex.on('finish', () => { events.push('finish'); assert.ok(!duplex.destroyed); assert.ok(duplex.readable); duplex.resume(); }); ws.on('close', () => { duplex.end(); }); }); wss.on('connection', (ws) => { ws.send('foo'); ws.close(); }); }); it("is destroyed after 'end' and 'finish' are emitted (2/2)", (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const events = []; const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const duplex = createWebSocketStream(ws); duplex.on('end', () => { events.push('end'); assert.ok(!duplex.destroyed); assert.ok(duplex.writable); duplex.end(); }); duplex.on('close', () => { assert.deepStrictEqual(events, ['end', 'finish']); wss.close(done); }); duplex.on('finish', () => { events.push('finish'); assert.ok(duplex.destroyed); }); duplex.resume(); }); wss.on('connection', (ws) => { ws.close(); }); }); it('handles backpressure (1/3)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { // eslint-disable-next-line no-unused-vars const ws = new WebSocket(`ws://localhost:${wss.address().port}`); }); wss.on('connection', (ws) => { const duplex = createWebSocketStream(ws); duplex.resume(); duplex.on('drain', () => { duplex.on('close', () => { wss.close(done); }); duplex.end(); }); const chunk = randomBytes(1024); let ret; do { ret = duplex.write(chunk); } while (ret !== false); }); }); it('handles backpressure (2/3)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const called = []; const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const duplex = createWebSocketStream(ws); const read = duplex._read; duplex._read = () => { called.push('read'); assert.ok(ws._receiver._writableState.needDrain); read(); assert.ok(ws._socket.isPaused()); }; ws.on('open', () => { ws._socket.on('pause', () => { duplex.resume(); }); ws._receiver.on('drain', () => { called.push('drain'); assert.ok(!ws._socket.isPaused()); }); const list = Sender.frame(randomBytes(16 * 1024), { fin: true, rsv1: false, opcode: 0x02, mask: false, readOnly: false }); // This hack is used because there is no guarantee that more than // 16KiB will be sent as a single TCP packet. ws._socket.push(Buffer.concat(list)); }); duplex.on('resume', duplex.end); duplex.on('close', () => { assert.deepStrictEqual(called, ['read', 'drain']); wss.close(done); }); }); }); it('handles backpressure (3/3)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const called = []; const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const duplex = createWebSocketStream(ws); const read = duplex._read; duplex._read = () => { called.push('read'); assert.ok(!ws._receiver._writableState.needDrain); read(); assert.ok(!ws._socket.isPaused()); }; ws.on('open', () => { ws._receiver.on('drain', () => { called.push('drain'); assert.ok(ws._socket.isPaused()); duplex.resume(); }); const list = Sender.frame(randomBytes(16 * 1024), { fin: true, rsv1: false, opcode: 0x02, mask: false, readOnly: false }); ws._socket.push(Buffer.concat(list)); }); duplex.on('resume', duplex.end); duplex.on('close', () => { assert.deepStrictEqual(called, ['drain', 'read']); wss.close(done); }); }); }); it('can be destroyed (1/2)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const error = new Error('Oops'); const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const duplex = createWebSocketStream(ws); duplex.on('error', (err) => { assert.strictEqual(err, error); duplex.on('close', () => { wss.close(done); }); }); ws.on('open', () => { duplex.destroy(error); }); }); }); it('can be destroyed (2/2)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const duplex = createWebSocketStream(ws); duplex.on('close', () => { wss.close(done); }); ws.on('open', () => { duplex.destroy(); }); }); }); }); }); ws-7.2.1/test/extension.test.js000066400000000000000000000121011357512353100164610ustar00rootroot00000000000000'use strict'; const assert = require('assert'); const { format, parse } = require('../lib/extension'); describe('extension', () => { describe('parse', () => { it('returns an empty object if the argument is `undefined`', () => { assert.deepStrictEqual(parse(), { __proto__: null }); assert.deepStrictEqual(parse(''), { __proto__: null }); }); it('parses a single extension', () => { assert.deepStrictEqual(parse('foo'), { foo: [{ __proto__: null }], __proto__: null }); }); it('parses params', () => { assert.deepStrictEqual(parse('foo;bar;baz=1;bar=2'), { foo: [{ bar: [true, '2'], baz: ['1'], __proto__: null }], __proto__: null }); }); it('parses multiple extensions', () => { assert.deepStrictEqual(parse('foo,bar;baz,foo;baz'), { foo: [{ __proto__: null }, { baz: [true], __proto__: null }], bar: [{ baz: [true], __proto__: null }], __proto__: null }); }); it('parses quoted params', () => { assert.deepStrictEqual(parse('foo;bar="hi"'), { foo: [{ bar: ['hi'], __proto__: null }], __proto__: null }); assert.deepStrictEqual(parse('foo;bar="\\0"'), { foo: [{ bar: ['0'], __proto__: null }], __proto__: null }); assert.deepStrictEqual(parse('foo;bar="b\\a\\z"'), { foo: [{ bar: ['baz'], __proto__: null }], __proto__: null }); assert.deepStrictEqual(parse('foo;bar="b\\az";bar'), { foo: [{ bar: ['baz', true], __proto__: null }], __proto__: null }); assert.throws( () => parse('foo;bar="baz"qux'), /^SyntaxError: Unexpected character at index 13$/ ); assert.throws( () => parse('foo;bar="baz" qux'), /^SyntaxError: Unexpected character at index 14$/ ); }); it('works with names that match `Object.prototype` property names', () => { assert.deepStrictEqual(parse('hasOwnProperty, toString'), { hasOwnProperty: [{ __proto__: null }], toString: [{ __proto__: null }], __proto__: null }); assert.deepStrictEqual(parse('foo;constructor'), { foo: [{ constructor: [true], __proto__: null }], __proto__: null }); }); it('ignores the optional white spaces', () => { const header = 'foo; bar\t; \tbaz=1\t ; bar="1"\t\t, \tqux\t ;norf '; assert.deepStrictEqual(parse(header), { foo: [{ bar: [true, '1'], baz: ['1'], __proto__: null }], qux: [{ norf: [true], __proto__: null }], __proto__: null }); }); it('throws an error if a name is empty', () => { [ [',', 0], ['foo,,', 4], ['foo, ,', 6], ['foo;=', 4], ['foo; =', 5], ['foo;;', 4], ['foo; ;', 5], ['foo;bar=,', 8], ['foo;bar=""', 9] ].forEach((element) => { assert.throws( () => parse(element[0]), new RegExp( `^SyntaxError: Unexpected character at index ${element[1]}$` ) ); }); }); it('throws an error if a white space is misplaced', () => { [ ['f oo', 2], ['foo;ba r', 7], ['foo;bar =', 8], ['foo;bar= ', 8] ].forEach((element) => { assert.throws( () => parse(element[0]), new RegExp( `^SyntaxError: Unexpected character at index ${element[1]}$` ) ); }); }); it('throws an error if a token contains invalid characters', () => { [ ['f@o', 1], ['f\\oo', 1], ['"foo"', 0], ['f"oo"', 1], ['foo;b@r', 5], ['foo;b\\ar', 5], ['foo;"bar"', 4], ['foo;b"ar"', 5], ['foo;bar=b@z', 9], ['foo;bar=b\\az ', 9], ['foo;bar="b@z"', 10], ['foo;bar="baz;"', 12], ['foo;bar=b"az"', 9], ['foo;bar="\\\\"', 10] ].forEach((element) => { assert.throws( () => parse(element[0]), new RegExp( `^SyntaxError: Unexpected character at index ${element[1]}$` ) ); }); }); it('throws an error if the header value ends prematurely', () => { [ 'foo, ', 'foo;', 'foo;bar,', 'foo;bar; ', 'foo;bar=', 'foo;bar="baz', 'foo;bar="1\\' ].forEach((header) => { assert.throws( () => parse(header), /^SyntaxError: Unexpected end of input$/ ); }); }); }); describe('format', () => { it('formats a single extension', () => { const extensions = format({ foo: {} }); assert.strictEqual(extensions, 'foo'); }); it('formats params', () => { const extensions = format({ foo: { bar: [true, 2], baz: 1 } }); assert.strictEqual(extensions, 'foo; bar; bar=2; baz=1'); }); it('formats multiple extensions', () => { const extensions = format({ foo: [{}, { baz: true }], bar: { baz: true } }); assert.strictEqual(extensions, 'foo, foo; baz, bar; baz'); }); }); }); ws-7.2.1/test/fixtures/000077500000000000000000000000001357512353100150075ustar00rootroot00000000000000ws-7.2.1/test/fixtures/agent1-cert.pem000066400000000000000000000016101357512353100176220ustar00rootroot00000000000000-----BEGIN CERTIFICATE----- MIICbjCCAdcCCQCVvok5oeLpqzANBgkqhkiG9w0BAQUFADB6MQswCQYDVQQGEwJV UzELMAkGA1UECBMCQ0ExCzAJBgNVBAcTAlNGMQ8wDQYDVQQKEwZKb3llbnQxEDAO BgNVBAsTB05vZGUuanMxDDAKBgNVBAMTA2NhMTEgMB4GCSqGSIb3DQEJARYRcnlA dGlueWNsb3Vkcy5vcmcwHhcNMTMwMzA4MDAzMDIyWhcNNDAwNzIzMDAzMDIyWjB9 MQswCQYDVQQGEwJVUzELMAkGA1UECBMCQ0ExCzAJBgNVBAcTAlNGMQ8wDQYDVQQK EwZKb3llbnQxEDAOBgNVBAsTB05vZGUuanMxDzANBgNVBAMTBmFnZW50MTEgMB4G CSqGSIb3DQEJARYRcnlAdGlueWNsb3Vkcy5vcmcwgZ8wDQYJKoZIhvcNAQEBBQAD gY0AMIGJAoGBAL6GwKosYb0Yc3Qo0OtQVlCJ4208Idw11ij+t2W5sfYbCil5tyQo jnhGM1CJhEXynQpXXwjKJuIeTQCkeUibTyFKa0bs8+li2FiGoKYbb4G81ovnqkmE 2iDVb8Gw3rrM4zeZ0ZdFnjMsAZac8h6+C4sB/pS9BiMOo6qTl15RQlcJAgMBAAEw DQYJKoZIhvcNAQEFBQADgYEAOtmLo8DwTPnI4wfQbQ3hWlTS/9itww6IsxH2ODt9 ggB7wi7N3uAdIWRZ54ke0NEAO5CW1xNTwsWcxQbiHrDOqX1vfVCjIenI76jVEEap /Ay53ydHNBKdsKkib61Me14Mu0bA3lUul57VXwmH4NUEFB3w973Q60PschUhOEXj 7DY= -----END CERTIFICATE----- ws-7.2.1/test/fixtures/agent1-key.pem000066400000000000000000000015671357512353100174700ustar00rootroot00000000000000-----BEGIN RSA PRIVATE KEY----- MIICXAIBAAKBgQC+hsCqLGG9GHN0KNDrUFZQieNtPCHcNdYo/rdlubH2Gwopebck KI54RjNQiYRF8p0KV18IyibiHk0ApHlIm08hSmtG7PPpYthYhqCmG2+BvNaL56pJ hNog1W/BsN66zOM3mdGXRZ4zLAGWnPIevguLAf6UvQYjDqOqk5deUUJXCQIDAQAB AoGANu/CBA+SCyVOvRK70u4yRTzNMAUjukxnuSBhH1rg/pajYnwvG6T6F6IeT72n P0gKkh3JUE6B0bds+p9yPUZTFUXghxjcF33wlIY44H6gFE4K5WutsFJ9c450wtuu 8rXZTsIg7lAXWjTFVmdtOEPetcGlO2Hpi1O7ZzkzHgB2w9ECQQDksCCYx78or1zY ZSokm8jmpIjG3VLKdvI9HAoJRN40ldnwFoigrFa1AHwsFtWNe8bKyVRPDoLDUjpB dkPWgweVAkEA1UfgqguQ2KIkbtp9nDBionu3QaajksrRHwIa8vdfRfLxszfHk2fh NGY3dkRZF8HUAbzYLrd9poVhCBAEjWekpQJASOM6AHfpnXYHCZF01SYx6hEW5wsz kARJQODm8f1ZNTlttO/5q/xBxn7ZFNRSTD3fJlL05B2j380ddC/Vf1FT4QJAP1BC GliqnBSuGhZUWYxni3KMeTm9rzL0F29pjpzutHYlWB2D6ndY/FQnvL0XcZ0Bka58 womIDGnl3x3aLBwLXQJBAJv6h5CHbXHx7VyDJAcNfppAqZGcEaiVg8yf2F33iWy2 FLthhJucx7df7SO2aw5h06bRDRAhb9br0R9/3mLr7RE= -----END RSA PRIVATE KEY----- ws-7.2.1/test/fixtures/ca1-cert.pem000066400000000000000000000016031357512353100171110ustar00rootroot00000000000000-----BEGIN CERTIFICATE----- MIICazCCAdQCCQC9/g69HtxXRzANBgkqhkiG9w0BAQUFADB6MQswCQYDVQQGEwJV UzELMAkGA1UECBMCQ0ExCzAJBgNVBAcTAlNGMQ8wDQYDVQQKEwZKb3llbnQxEDAO BgNVBAsTB05vZGUuanMxDDAKBgNVBAMTA2NhMTEgMB4GCSqGSIb3DQEJARYRcnlA dGlueWNsb3Vkcy5vcmcwHhcNMTMwMzA4MDAzMDIyWhcNNDAwNzIzMDAzMDIyWjB6 MQswCQYDVQQGEwJVUzELMAkGA1UECBMCQ0ExCzAJBgNVBAcTAlNGMQ8wDQYDVQQK EwZKb3llbnQxEDAOBgNVBAsTB05vZGUuanMxDDAKBgNVBAMTA2NhMTEgMB4GCSqG SIb3DQEJARYRcnlAdGlueWNsb3Vkcy5vcmcwgZ8wDQYJKoZIhvcNAQEBBQADgY0A MIGJAoGBAKxr1mARUcv7zaqx5y4AxJPK6c1jdbSg7StcL4vg8klaPAlfNO6o+/Cl w5CdQD3ukaVUwUOJ4T/+b3Xf7785XcWBC33GdjVQkfbHATJYcka7j7JDw3qev5Jk 1rAbRw48hF6rYlSGcx1mccAjoLoa3I8jgxCNAYHIjUQXgdmU893rAgMBAAEwDQYJ KoZIhvcNAQEFBQADgYEAis05yxjCtJRuv8uX/DK6TX/j9C9Lzp1rKDNFTaTZ0iRw KCw1EcNx4OXSj9gNblW4PWxpDvygrt1AmH9h2cb8K859NSHa9JOBFw6MA5C2A4Sj NQfNATqUl4T6cdORlcDEZwHtT8b6D4A6Er31G/eJF4Sen0TUFpjdjd+l9RBjHlo= -----END CERTIFICATE----- ws-7.2.1/test/fixtures/ca1-key.pem000066400000000000000000000020211357512353100167370ustar00rootroot00000000000000-----BEGIN ENCRYPTED PRIVATE KEY----- MIICxjBABgkqhkiG9w0BBQ0wMzAbBgkqhkiG9w0BBQwwDgQIFeWxJE1BrRECAggA MBQGCCqGSIb3DQMHBAgu9PlMSQ+BOASCAoDEZN2tX0xWo/N+Jg+PrvCrFDk3P+3x 5xG/PEDjtMCAWPBEwbnaYHDzYmhNcAmxzGqEHGMDiWYs46LbO560VS3uMvFbEWPo KYYVb13vkxl2poXdonCb5cHZA5GUYzTIVVJFptl4LHwBczHoMHtA4FqAhKlYvlWw EOrdLB8XcwMmGPFabbbGxno0+EWWM27uNjlogfoxj35mQqSW4rOlhZ460XjOB1Zx LjXMuZeONojkGYQRG5EUMchBoctQpCOM6cAi9r1B9BvtFCBpDV1c1zEZBzTEUd8o kLn6tjLmY+QpTdylFjEWc7U3ppLY/pkoTBv4r85a2sEMWqkhSJboLaTboWzDJcU3 Ke61pMpovt/3yCUd3TKgwduVwwQtDVTlBe0p66aN9QVj3CrFy/bKAGO3vxlli24H aIjZf+OVoBY21ESlW3jLvNlBf7Ezf///2E7j4SCDLyZSFMTpFoAG/jDRyvi+wTKX Kh485Bptnip6DCSuoH4u2SkOqwz3gJS/6s02YKe4m311QT4Pzne5/FwOFaS/HhQg Xvyh2/d00OgJ0Y0PYQsHILPRgTUCKUXvj1O58opn3fxSacsPxIXwj6Z4FYAjUTaV 2B85k1lpant/JJEilDqMjqzx4pHZ/Z3Uto1lSM1JZs9SNL/0UR+6F0TXZTULVU9V w8jYzz4sPr7LEyrrTbzmjQgnQFVbhAN/eKgRZK/SpLjxpmBV5MfpbPKsPUZqT4UC 4nXa8a/NYUQ9e+QKK8enq9E599c2W442W7Z1uFRZTWReMx/lF8wwA6G8zOPG0bdj d+T5Gegzd5mvRiXMBklCo8RLxOOvgxun1n3PY4a63aH6mqBhdfhiLp5j -----END ENCRYPTED PRIVATE KEY----- ws-7.2.1/test/fixtures/certificate.pem000066400000000000000000000013651357512353100200010ustar00rootroot00000000000000-----BEGIN CERTIFICATE----- MIICATCCAWoCCQDPufXH86n2QzANBgkqhkiG9w0BAQUFADBFMQswCQYDVQQGEwJu bzETMBEGA1UECAwKU29tZS1TdGF0ZTEhMB8GA1UECgwYSW50ZXJuZXQgV2lkZ2l0 cyBQdHkgTHRkMB4XDTEyMDEwMTE0NDQwMFoXDTIwMDMxOTE0NDQwMFowRTELMAkG A1UEBhMCbm8xEzARBgNVBAgMClNvbWUtU3RhdGUxITAfBgNVBAoMGEludGVybmV0 IFdpZGdpdHMgUHR5IEx0ZDCBnzANBgkqhkiG9w0BAQEFAAOBjQAwgYkCgYEAtrQ7 +r//2iV/B6F+4boH0XqFn7alcV9lpjvAmwRXNKnxAoa0f97AjYPGNLKrjpkNXXhB JROIdbRbZnCNeC5fzX1a+JCo7KStzBXuGSZr27TtFmcV4H+9gIRIcNHtZmJLnxbJ sIhkGR8yVYdmJZe4eT5ldk1zoB1adgPF1hZhCBMCAwEAATANBgkqhkiG9w0BAQUF AAOBgQCeWBEHYJ4mCB5McwSSUox0T+/mJ4W48L/ZUE4LtRhHasU9hiW92xZkTa7E QLcoJKQiWfiLX2ysAro0NX4+V8iqLziMqvswnPzz5nezaOLE/9U/QvH3l8qqNkXu rNbsW1h/IO6FV8avWFYVFoutUwOaZ809k7iMh2F2JMgXQ5EymQ== -----END CERTIFICATE----- ws-7.2.1/test/fixtures/key.pem000066400000000000000000000015671357512353100163130ustar00rootroot00000000000000-----BEGIN RSA PRIVATE KEY----- MIICXAIBAAKBgQC2tDv6v//aJX8HoX7hugfReoWftqVxX2WmO8CbBFc0qfEChrR/ 3sCNg8Y0squOmQ1deEElE4h1tFtmcI14Ll/NfVr4kKjspK3MFe4ZJmvbtO0WZxXg f72AhEhw0e1mYkufFsmwiGQZHzJVh2Yll7h5PmV2TXOgHVp2A8XWFmEIEwIDAQAB AoGAAlVY8sHi/aE+9xT77twWX3mGHV0SzdjfDnly40fx6S1Gc7bOtVdd9DC7pk6l 3ENeJVR02IlgU8iC5lMHq4JEHPE272jtPrLlrpWLTGmHEqoVFv9AITPqUDLhB9Kk Hjl7h8NYBKbr2JHKICr3DIPKOT+RnXVb1PD4EORbJ3ooYmkCQQDfknUnVxPgxUGs ouABw1WJIOVgcCY/IFt4Ihf6VWTsxBgzTJKxn3HtgvE0oqTH7V480XoH0QxHhjLq DrgobWU9AkEA0TRJ8/ouXGnFEPAXjWr9GdPQRZ1Use2MrFjneH2+Sxc0CmYtwwqL Kr5kS6mqJrxprJeluSjBd+3/ElxURrEXjwJAUvmlN1OPEhXDmRHd92mKnlkyKEeX OkiFCiIFKih1S5Y/sRJTQ0781nyJjtJqO7UyC3pnQu1oFEePL+UEniRztQJAMfav AtnpYKDSM+1jcp7uu9BemYGtzKDTTAYfoiNF42EzSJiGrWJDQn4eLgPjY0T0aAf/ yGz3Z9ErbhMm/Ysl+QJBAL4kBxRT8gM4ByJw4sdOvSeCCANFq8fhbgm8pGWlCPb5 JGmX3/GHFM8x2tbWMGpyZP1DLtiNEFz7eCGktWK5rqE= -----END RSA PRIVATE KEY----- ws-7.2.1/test/fixtures/request.pem000066400000000000000000000011331357512353100172000ustar00rootroot00000000000000-----BEGIN CERTIFICATE REQUEST----- MIIBhDCB7gIBADBFMQswCQYDVQQGEwJubzETMBEGA1UECAwKU29tZS1TdGF0ZTEh MB8GA1UECgwYSW50ZXJuZXQgV2lkZ2l0cyBQdHkgTHRkMIGfMA0GCSqGSIb3DQEB AQUAA4GNADCBiQKBgQC2tDv6v//aJX8HoX7hugfReoWftqVxX2WmO8CbBFc0qfEC hrR/3sCNg8Y0squOmQ1deEElE4h1tFtmcI14Ll/NfVr4kKjspK3MFe4ZJmvbtO0W ZxXgf72AhEhw0e1mYkufFsmwiGQZHzJVh2Yll7h5PmV2TXOgHVp2A8XWFmEIEwID AQABoAAwDQYJKoZIhvcNAQEFBQADgYEAjsUXEARgfxZNkMjuUcudgU2w4JXS0gGI JQ0U1LmU0vMDSKwqndMlvCbKzEgPbJnGJDI8D4MeINCJHa5Ceyb8c+jaJYUcCabl lQW5Psn3+eWp8ncKlIycDRj1Qk615XuXtV0fhkrgQM2ZCm9LaQ1O1Gd/CzLihLjF W0MmgMKMMRk= -----END CERTIFICATE REQUEST----- ws-7.2.1/test/limiter.test.js000066400000000000000000000016171357512353100161240ustar00rootroot00000000000000'use strict'; const assert = require('assert'); const Limiter = require('../lib/limiter'); describe('Limiter', () => { describe('#ctor', () => { it('takes a `concurrency` argument', () => { const limiter = new Limiter(0); assert.strictEqual(limiter.concurrency, Infinity); }); }); describe('#kRun', () => { it('limits the number of jobs allowed to run concurrently', (done) => { const limiter = new Limiter(1); limiter.add((callback) => { setImmediate(() => { callback(); assert.strictEqual(limiter.jobs.length, 0); assert.strictEqual(limiter.pending, 1); }); }); limiter.add((callback) => { setImmediate(() => { callback(); assert.strictEqual(limiter.pending, 0); done(); }); }); assert.strictEqual(limiter.jobs.length, 1); }); }); }); ws-7.2.1/test/permessage-deflate.test.js000066400000000000000000000514321357512353100202140ustar00rootroot00000000000000'use strict'; const assert = require('assert'); const PerMessageDeflate = require('../lib/permessage-deflate'); const extension = require('../lib/extension'); describe('PerMessageDeflate', () => { describe('#offer', () => { it('creates an offer', () => { const perMessageDeflate = new PerMessageDeflate(); assert.deepStrictEqual(perMessageDeflate.offer(), { client_max_window_bits: true }); }); it('uses the configuration options', () => { const perMessageDeflate = new PerMessageDeflate({ serverNoContextTakeover: true, clientNoContextTakeover: true, serverMaxWindowBits: 10, clientMaxWindowBits: 11 }); assert.deepStrictEqual(perMessageDeflate.offer(), { server_no_context_takeover: true, client_no_context_takeover: true, server_max_window_bits: 10, client_max_window_bits: 11 }); }); }); describe('#accept', () => { it('throws an error if a parameter has multiple values', () => { const perMessageDeflate = new PerMessageDeflate(); const extensions = extension.parse( 'permessage-deflate; server_no_context_takeover; server_no_context_takeover' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^Error: Parameter "server_no_context_takeover" must have only a single value$/ ); }); it('throws an error if a parameter has an invalid name', () => { const perMessageDeflate = new PerMessageDeflate(); const extensions = extension.parse('permessage-deflate;foo'); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^Error: Unknown parameter "foo"$/ ); }); it('throws an error if client_no_context_takeover has a value', () => { const perMessageDeflate = new PerMessageDeflate(); const extensions = extension.parse( 'permessage-deflate; client_no_context_takeover=10' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^TypeError: Invalid value for parameter "client_no_context_takeover": 10$/ ); }); it('throws an error if server_no_context_takeover has a value', () => { const perMessageDeflate = new PerMessageDeflate(); const extensions = extension.parse( 'permessage-deflate; server_no_context_takeover=10' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^TypeError: Invalid value for parameter "server_no_context_takeover": 10$/ ); }); it('throws an error if server_max_window_bits has an invalid value', () => { const perMessageDeflate = new PerMessageDeflate(); let extensions = extension.parse( 'permessage-deflate; server_max_window_bits=7' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^TypeError: Invalid value for parameter "server_max_window_bits": 7$/ ); extensions = extension.parse( 'permessage-deflate; server_max_window_bits' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^TypeError: Invalid value for parameter "server_max_window_bits": true$/ ); }); describe('As server', () => { it('accepts an offer with no parameters', () => { const perMessageDeflate = new PerMessageDeflate({}, true); assert.deepStrictEqual(perMessageDeflate.accept([{}]), {}); }); it('accepts an offer with parameters', () => { const perMessageDeflate = new PerMessageDeflate({}, true); const extensions = extension.parse( 'permessage-deflate; server_no_context_takeover; ' + 'client_no_context_takeover; server_max_window_bits=10; ' + 'client_max_window_bits=11' ); assert.deepStrictEqual( perMessageDeflate.accept(extensions['permessage-deflate']), { server_no_context_takeover: true, client_no_context_takeover: true, server_max_window_bits: 10, client_max_window_bits: 11, __proto__: null } ); }); it('prefers the configuration options', () => { const perMessageDeflate = new PerMessageDeflate( { serverNoContextTakeover: true, clientNoContextTakeover: true, serverMaxWindowBits: 12, clientMaxWindowBits: 11 }, true ); const extensions = extension.parse( 'permessage-deflate; server_max_window_bits=14; client_max_window_bits=13' ); assert.deepStrictEqual( perMessageDeflate.accept(extensions['permessage-deflate']), { server_no_context_takeover: true, client_no_context_takeover: true, server_max_window_bits: 12, client_max_window_bits: 11, __proto__: null } ); }); it('accepts the first supported offer', () => { const perMessageDeflate = new PerMessageDeflate( { serverMaxWindowBits: 11 }, true ); const extensions = extension.parse( 'permessage-deflate; server_max_window_bits=10, permessage-deflate' ); assert.deepStrictEqual( perMessageDeflate.accept(extensions['permessage-deflate']), { server_max_window_bits: 11, __proto__: null } ); }); it('throws an error if server_no_context_takeover is unsupported', () => { const perMessageDeflate = new PerMessageDeflate( { serverNoContextTakeover: false }, true ); const extensions = extension.parse( 'permessage-deflate; server_no_context_takeover' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^Error: None of the extension offers can be accepted$/ ); }); it('throws an error if server_max_window_bits is unsupported', () => { const perMessageDeflate = new PerMessageDeflate( { serverMaxWindowBits: false }, true ); const extensions = extension.parse( 'permessage-deflate; server_max_window_bits=10' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^Error: None of the extension offers can be accepted$/ ); }); it('throws an error if server_max_window_bits is less than configuration', () => { const perMessageDeflate = new PerMessageDeflate( { serverMaxWindowBits: 11 }, true ); const extensions = extension.parse( 'permessage-deflate; server_max_window_bits=10' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^Error: None of the extension offers can be accepted$/ ); }); it('throws an error if client_max_window_bits is unsupported on client', () => { const perMessageDeflate = new PerMessageDeflate( { clientMaxWindowBits: 10 }, true ); const extensions = extension.parse('permessage-deflate'); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^Error: None of the extension offers can be accepted$/ ); }); it('throws an error if client_max_window_bits has an invalid value', () => { const perMessageDeflate = new PerMessageDeflate({}, true); const extensions = extension.parse( 'permessage-deflate; client_max_window_bits=16' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^TypeError: Invalid value for parameter "client_max_window_bits": 16$/ ); }); }); describe('As client', () => { it('accepts a response with no parameters', () => { const perMessageDeflate = new PerMessageDeflate({}); assert.deepStrictEqual(perMessageDeflate.accept([{}]), {}); }); it('accepts a response with parameters', () => { const perMessageDeflate = new PerMessageDeflate({}); const extensions = extension.parse( 'permessage-deflate; server_no_context_takeover; ' + 'client_no_context_takeover; server_max_window_bits=10; ' + 'client_max_window_bits=11' ); assert.deepStrictEqual( perMessageDeflate.accept(extensions['permessage-deflate']), { server_no_context_takeover: true, client_no_context_takeover: true, server_max_window_bits: 10, client_max_window_bits: 11, __proto__: null } ); }); it('throws an error if client_no_context_takeover is unsupported', () => { const perMessageDeflate = new PerMessageDeflate({ clientNoContextTakeover: false }); const extensions = extension.parse( 'permessage-deflate; client_no_context_takeover' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^Error: Unexpected parameter "client_no_context_takeover"$/ ); }); it('throws an error if client_max_window_bits is unsupported', () => { const perMessageDeflate = new PerMessageDeflate({ clientMaxWindowBits: false }); const extensions = extension.parse( 'permessage-deflate; client_max_window_bits=10' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^Error: Unexpected or invalid parameter "client_max_window_bits"$/ ); }); it('throws an error if client_max_window_bits is greater than configuration', () => { const perMessageDeflate = new PerMessageDeflate({ clientMaxWindowBits: 10 }); const extensions = extension.parse( 'permessage-deflate; client_max_window_bits=11' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^Error: Unexpected or invalid parameter "client_max_window_bits"$/ ); }); it('throws an error if client_max_window_bits has an invalid value', () => { const perMessageDeflate = new PerMessageDeflate(); let extensions = extension.parse( 'permessage-deflate; client_max_window_bits=16' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^TypeError: Invalid value for parameter "client_max_window_bits": 16$/ ); extensions = extension.parse( 'permessage-deflate; client_max_window_bits' ); assert.throws( () => perMessageDeflate.accept(extensions['permessage-deflate']), /^TypeError: Invalid value for parameter "client_max_window_bits": true$/ ); }); it('uses the config value if client_max_window_bits is not specified', () => { const perMessageDeflate = new PerMessageDeflate({ clientMaxWindowBits: 10 }); assert.deepStrictEqual(perMessageDeflate.accept([{}]), { client_max_window_bits: 10 }); }); }); }); describe('#compress and #decompress', () => { it('works with unfragmented messages', (done) => { const perMessageDeflate = new PerMessageDeflate({ threshold: 0 }); const buf = Buffer.from([1, 2, 3]); perMessageDeflate.accept([{}]); perMessageDeflate.compress(buf, true, (err, data) => { if (err) return done(err); perMessageDeflate.decompress(data, true, (err, data) => { if (err) return done(err); assert.ok(data.equals(buf)); done(); }); }); }); it('works with fragmented messages', (done) => { const perMessageDeflate = new PerMessageDeflate({ threshold: 0 }); const buf = Buffer.from([1, 2, 3, 4]); perMessageDeflate.accept([{}]); perMessageDeflate.compress(buf.slice(0, 2), false, (err, compressed1) => { if (err) return done(err); perMessageDeflate.compress(buf.slice(2), true, (err, compressed2) => { if (err) return done(err); perMessageDeflate.decompress(compressed1, false, (err, data1) => { if (err) return done(err); perMessageDeflate.decompress(compressed2, true, (err, data2) => { if (err) return done(err); assert.ok(Buffer.concat([data1, data2]).equals(buf)); done(); }); }); }); }); }); it('works with the negotiated parameters', (done) => { const perMessageDeflate = new PerMessageDeflate({ threshold: 0, memLevel: 5, level: 9 }); const extensions = extension.parse( 'permessage-deflate; server_no_context_takeover; ' + 'client_no_context_takeover; server_max_window_bits=10; ' + 'client_max_window_bits=11' ); const buf = Buffer.from("Some compressible data, it's compressible."); perMessageDeflate.accept(extensions['permessage-deflate']); perMessageDeflate.compress(buf, true, (err, data) => { if (err) return done(err); perMessageDeflate.decompress(data, true, (err, data) => { if (err) return done(err); assert.ok(data.equals(buf)); done(); }); }); }); it('honors the `level` option', (done) => { const lev0 = new PerMessageDeflate({ threshold: 0, zlibDeflateOptions: { level: 0 } }); const lev9 = new PerMessageDeflate({ threshold: 0, zlibDeflateOptions: { level: 9 } }); const extensionStr = 'permessage-deflate; server_no_context_takeover; ' + 'client_no_context_takeover; server_max_window_bits=10; ' + 'client_max_window_bits=11'; const buf = Buffer.from("Some compressible data, it's compressible."); lev0.accept(extension.parse(extensionStr)['permessage-deflate']); lev9.accept(extension.parse(extensionStr)['permessage-deflate']); lev0.compress(buf, true, (err, compressed1) => { if (err) return done(err); lev0.decompress(compressed1, true, (err, decompressed1) => { if (err) return done(err); lev9.compress(buf, true, (err, compressed2) => { if (err) return done(err); lev9.decompress(compressed2, true, (err, decompressed2) => { if (err) return done(err); // Level 0 compression actually adds a few bytes due to headers. assert.ok(compressed1.length > buf.length); // Level 9 should not, of course. assert.ok(compressed2.length < buf.length); // Ensure they both decompress back properly. assert.ok(decompressed1.equals(buf)); assert.ok(decompressed2.equals(buf)); done(); }); }); }); }); }); it('honors the `zlib{Deflate,Inflate}Options` option', (done) => { const lev0 = new PerMessageDeflate({ threshold: 0, zlibDeflateOptions: { level: 0, chunkSize: 256 }, zlibInflateOptions: { chunkSize: 2048 } }); const lev9 = new PerMessageDeflate({ threshold: 0, zlibDeflateOptions: { level: 9, chunkSize: 128 }, zlibInflateOptions: { chunkSize: 1024 } }); // Note no context takeover so we can get a hold of the raw streams after // we do the dance. const extensionStr = 'permessage-deflate; server_max_window_bits=10; ' + 'client_max_window_bits=11'; const buf = Buffer.from("Some compressible data, it's compressible."); lev0.accept(extension.parse(extensionStr)['permessage-deflate']); lev9.accept(extension.parse(extensionStr)['permessage-deflate']); lev0.compress(buf, true, (err, compressed1) => { if (err) return done(err); lev0.decompress(compressed1, true, (err, decompressed1) => { if (err) return done(err); lev9.compress(buf, true, (err, compressed2) => { if (err) return done(err); lev9.decompress(compressed2, true, (err, decompressed2) => { if (err) return done(err); // Level 0 compression actually adds a few bytes due to headers. assert.ok(compressed1.length > buf.length); // Level 9 should not, of course. assert.ok(compressed2.length < buf.length); // Ensure they both decompress back properly. assert.ok(decompressed1.equals(buf)); assert.ok(decompressed2.equals(buf)); // Assert options were set. assert.ok(lev0._deflate._level === 0); assert.ok(lev9._deflate._level === 9); assert.ok(lev0._deflate._chunkSize === 256); assert.ok(lev9._deflate._chunkSize === 128); assert.ok(lev0._inflate._chunkSize === 2048); assert.ok(lev9._inflate._chunkSize === 1024); done(); }); }); }); }); }); it("doesn't use contex takeover if not allowed", (done) => { const perMessageDeflate = new PerMessageDeflate({ threshold: 0 }, true); const extensions = extension.parse( 'permessage-deflate;server_no_context_takeover' ); const buf = Buffer.from('foofoo'); perMessageDeflate.accept(extensions['permessage-deflate']); perMessageDeflate.compress(buf, true, (err, compressed1) => { if (err) return done(err); perMessageDeflate.decompress(compressed1, true, (err, data) => { if (err) return done(err); assert.ok(data.equals(buf)); perMessageDeflate.compress(data, true, (err, compressed2) => { if (err) return done(err); assert.strictEqual(compressed2.length, compressed1.length); perMessageDeflate.decompress(compressed2, true, (err, data) => { if (err) return done(err); assert.ok(data.equals(buf)); done(); }); }); }); }); }); it('uses contex takeover if allowed', (done) => { const perMessageDeflate = new PerMessageDeflate({ threshold: 0 }, true); const extensions = extension.parse('permessage-deflate'); const buf = Buffer.from('foofoo'); perMessageDeflate.accept(extensions['permessage-deflate']); perMessageDeflate.compress(buf, true, (err, compressed1) => { if (err) return done(err); perMessageDeflate.decompress(compressed1, true, (err, data) => { if (err) return done(err); assert.ok(data.equals(buf)); perMessageDeflate.compress(data, true, (err, compressed2) => { if (err) return done(err); assert.ok(compressed2.length < compressed1.length); perMessageDeflate.decompress(compressed2, true, (err, data) => { if (err) return done(err); assert.ok(data.equals(buf)); done(); }); }); }); }); }); it('calls the callback when an error occurs (inflate)', (done) => { const perMessageDeflate = new PerMessageDeflate({ threshold: 0 }); const data = Buffer.from('something invalid'); perMessageDeflate.accept([{}]); perMessageDeflate.decompress(data, true, (err) => { assert.ok(err instanceof Error); assert.strictEqual(err.errno, -3); done(); }); }); it("doesn't call the callback twice when `maxPayload` is exceeded", (done) => { const perMessageDeflate = new PerMessageDeflate( { threshold: 0 }, false, 25 ); const buf = Buffer.from('A'.repeat(50)); perMessageDeflate.accept([{}]); perMessageDeflate.compress(buf, true, (err, data) => { if (err) return done(err); perMessageDeflate.decompress(data, true, (err) => { assert.ok(err instanceof RangeError); assert.strictEqual(err.message, 'Max payload size exceeded'); done(); }); }); }); it("doesn't call the callback if the deflate stream is closed prematurely", (done) => { const perMessageDeflate = new PerMessageDeflate({ threshold: 0 }); const buf = Buffer.from('A'.repeat(50)); perMessageDeflate.accept([{}]); perMessageDeflate.compress(buf, true, () => { done(new Error('Unexpected callback invocation')); }); perMessageDeflate._deflate.on('close', done); process.nextTick(() => perMessageDeflate.cleanup()); }); }); }); ws-7.2.1/test/receiver.test.js000066400000000000000000000617301357512353100162650ustar00rootroot00000000000000'use strict'; const assert = require('assert'); const crypto = require('crypto'); const PerMessageDeflate = require('../lib/permessage-deflate'); const constants = require('../lib/constants'); const Receiver = require('../lib/receiver'); const Sender = require('../lib/sender'); const kStatusCode = constants.kStatusCode; describe('Receiver', () => { it('parses an unmasked text message', (done) => { const receiver = new Receiver(); receiver.on('message', (data) => { assert.strictEqual(data, 'Hello'); done(); }); receiver.write(Buffer.from('810548656c6c6f', 'hex')); }); it('parses a close message', (done) => { const receiver = new Receiver(); receiver.on('conclude', (code, data) => { assert.strictEqual(code, 1005); assert.strictEqual(data, ''); done(); }); receiver.write(Buffer.from('8800', 'hex')); }); it('parses a close message spanning multiple writes', (done) => { const receiver = new Receiver(); receiver.on('conclude', (code, data) => { assert.strictEqual(code, 1000); assert.strictEqual(data, 'DONE'); done(); }); receiver.write(Buffer.from('8806', 'hex')); receiver.write(Buffer.from('03e8444F4E45', 'hex')); }); it('parses a masked text message', (done) => { const receiver = new Receiver(); receiver.on('message', (data) => { assert.strictEqual(data, '5:::{"name":"echo"}'); done(); }); receiver.write( Buffer.from('81933483a86801b992524fa1c60959e68a5216e6cb005ba1d5', 'hex') ); }); it('parses a masked text message longer than 125 B', (done) => { const receiver = new Receiver(); const msg = 'A'.repeat(200); const list = Sender.frame(Buffer.from(msg), { fin: true, rsv1: false, opcode: 0x01, mask: true, readOnly: false }); const frame = Buffer.concat(list); receiver.on('message', (data) => { assert.strictEqual(data, msg); done(); }); receiver.write(frame.slice(0, 2)); setImmediate(() => receiver.write(frame.slice(2))); }); it('parses a really long masked text message', (done) => { const receiver = new Receiver(); const msg = 'A'.repeat(64 * 1024); const list = Sender.frame(Buffer.from(msg), { fin: true, rsv1: false, opcode: 0x01, mask: true, readOnly: false }); const frame = Buffer.concat(list); receiver.on('message', (data) => { assert.strictEqual(data, msg); done(); }); receiver.write(frame); }); it('parses a 300 B fragmented masked text message', (done) => { const receiver = new Receiver(); const msg = 'A'.repeat(300); const fragment1 = msg.substr(0, 150); const fragment2 = msg.substr(150); const options = { rsv1: false, mask: true, readOnly: false }; const frame1 = Buffer.concat( Sender.frame( Buffer.from(fragment1), Object.assign({ fin: false, opcode: 0x01 }, options) ) ); const frame2 = Buffer.concat( Sender.frame( Buffer.from(fragment2), Object.assign({ fin: true, opcode: 0x00 }, options) ) ); receiver.on('message', (data) => { assert.strictEqual(data, msg); done(); }); receiver.write(frame1); receiver.write(frame2); }); it('parses a ping message', (done) => { const receiver = new Receiver(); const msg = 'Hello'; const list = Sender.frame(Buffer.from(msg), { fin: true, rsv1: false, opcode: 0x09, mask: true, readOnly: false }); const frame = Buffer.concat(list); receiver.on('ping', (data) => { assert.strictEqual(data.toString(), msg); done(); }); receiver.write(frame); }); it('parses a ping message with no data', (done) => { const receiver = new Receiver(); receiver.on('ping', (data) => { assert.ok(data.equals(Buffer.alloc(0))); done(); }); receiver.write(Buffer.from('8900', 'hex')); }); it('parses a 300 B fragmented masked text message with a ping in the middle (1/2)', (done) => { const receiver = new Receiver(); const msg = 'A'.repeat(300); const pingMessage = 'Hello'; const fragment1 = msg.substr(0, 150); const fragment2 = msg.substr(150); const options = { rsv1: false, mask: true, readOnly: false }; const frame1 = Buffer.concat( Sender.frame( Buffer.from(fragment1), Object.assign({ fin: false, opcode: 0x01 }, options) ) ); const frame2 = Buffer.concat( Sender.frame( Buffer.from(pingMessage), Object.assign({ fin: true, opcode: 0x09 }, options) ) ); const frame3 = Buffer.concat( Sender.frame( Buffer.from(fragment2), Object.assign({ fin: true, opcode: 0x00 }, options) ) ); let gotPing = false; receiver.on('message', (data) => { assert.strictEqual(data, msg); assert.ok(gotPing); done(); }); receiver.on('ping', (data) => { gotPing = true; assert.strictEqual(data.toString(), pingMessage); }); receiver.write(frame1); receiver.write(frame2); receiver.write(frame3); }); it('parses a 300 B fragmented masked text message with a ping in the middle (2/2)', (done) => { const receiver = new Receiver(); const msg = 'A'.repeat(300); const pingMessage = 'Hello'; const fragment1 = msg.substr(0, 150); const fragment2 = msg.substr(150); const options = { rsv1: false, mask: true, readOnly: false }; const frame1 = Buffer.concat( Sender.frame( Buffer.from(fragment1), Object.assign({ fin: false, opcode: 0x01 }, options) ) ); const frame2 = Buffer.concat( Sender.frame( Buffer.from(pingMessage), Object.assign({ fin: true, opcode: 0x09 }, options) ) ); const frame3 = Buffer.concat( Sender.frame( Buffer.from(fragment2), Object.assign({ fin: true, opcode: 0x00 }, options) ) ); let chunks = []; const splitBuffer = (buf) => { const i = Math.floor(buf.length / 2); return [buf.slice(0, i), buf.slice(i)]; }; chunks = chunks.concat(splitBuffer(frame1)); chunks = chunks.concat(splitBuffer(frame2)); chunks = chunks.concat(splitBuffer(frame3)); let gotPing = false; receiver.on('message', (data) => { assert.strictEqual(data, msg); assert.ok(gotPing); done(); }); receiver.on('ping', (data) => { gotPing = true; assert.strictEqual(data.toString(), pingMessage); }); for (let i = 0; i < chunks.length; ++i) { receiver.write(chunks[i]); } }); it('parses a 100 B masked binary message', (done) => { const receiver = new Receiver(); const msg = crypto.randomBytes(100); const list = Sender.frame(msg, { fin: true, rsv1: false, opcode: 0x02, mask: true, readOnly: true }); const frame = Buffer.concat(list); receiver.on('message', (data) => { assert.ok(data.equals(msg)); done(); }); receiver.write(frame); }); it('parses a 256 B masked binary message', (done) => { const receiver = new Receiver(); const msg = crypto.randomBytes(256); const list = Sender.frame(msg, { fin: true, rsv1: false, opcode: 0x02, mask: true, readOnly: true }); const frame = Buffer.concat(list); receiver.on('message', (data) => { assert.ok(data.equals(msg)); done(); }); receiver.write(frame); }); it('parses a 200 KiB masked binary message', (done) => { const receiver = new Receiver(); const msg = crypto.randomBytes(200 * 1024); const list = Sender.frame(msg, { fin: true, rsv1: false, opcode: 0x02, mask: true, readOnly: true }); const frame = Buffer.concat(list); receiver.on('message', (data) => { assert.ok(data.equals(msg)); done(); }); receiver.write(frame); }); it('parses a 200 KiB unmasked binary message', (done) => { const receiver = new Receiver(); const msg = crypto.randomBytes(200 * 1024); const list = Sender.frame(msg, { fin: true, rsv1: false, opcode: 0x02, mask: false, readOnly: true }); const frame = Buffer.concat(list); receiver.on('message', (data) => { assert.ok(data.equals(msg)); done(); }); receiver.write(frame); }); it('parses a compressed message', (done) => { const perMessageDeflate = new PerMessageDeflate(); perMessageDeflate.accept([{}]); const receiver = new Receiver(undefined, { 'permessage-deflate': perMessageDeflate }); const buf = Buffer.from('Hello'); receiver.on('message', (data) => { assert.strictEqual(data, 'Hello'); done(); }); perMessageDeflate.compress(buf, true, (err, data) => { if (err) return done(err); receiver.write(Buffer.from([0xc1, data.length])); receiver.write(data); }); }); it('parses a compressed and fragmented message', (done) => { const perMessageDeflate = new PerMessageDeflate(); perMessageDeflate.accept([{}]); const receiver = new Receiver(undefined, { 'permessage-deflate': perMessageDeflate }); const buf1 = Buffer.from('foo'); const buf2 = Buffer.from('bar'); receiver.on('message', (data) => { assert.strictEqual(data, 'foobar'); done(); }); perMessageDeflate.compress(buf1, false, (err, fragment1) => { if (err) return done(err); receiver.write(Buffer.from([0x41, fragment1.length])); receiver.write(fragment1); perMessageDeflate.compress(buf2, true, (err, fragment2) => { if (err) return done(err); receiver.write(Buffer.from([0x80, fragment2.length])); receiver.write(fragment2); }); }); }); it('parses a buffer with thousands of frames', (done) => { const buf = Buffer.allocUnsafe(40000); for (let i = 0; i < buf.length; i += 2) { buf[i] = 0x81; buf[i + 1] = 0x00; } const receiver = new Receiver(); let counter = 0; receiver.on('message', (data) => { assert.strictEqual(data, ''); if (++counter === 20000) done(); }); receiver.write(buf); }); it('resets `totalPayloadLength` only on final frame (unfragmented)', (done) => { const receiver = new Receiver(undefined, {}, 10); receiver.on('message', (data) => { assert.strictEqual(receiver._totalPayloadLength, 0); assert.strictEqual(data, 'Hello'); done(); }); assert.strictEqual(receiver._totalPayloadLength, 0); receiver.write(Buffer.from('810548656c6c6f', 'hex')); }); it('resets `totalPayloadLength` only on final frame (fragmented)', (done) => { const receiver = new Receiver(undefined, {}, 10); receiver.on('message', (data) => { assert.strictEqual(receiver._totalPayloadLength, 0); assert.strictEqual(data, 'Hello'); done(); }); assert.strictEqual(receiver._totalPayloadLength, 0); receiver.write(Buffer.from('01024865', 'hex')); assert.strictEqual(receiver._totalPayloadLength, 2); receiver.write(Buffer.from('80036c6c6f', 'hex')); }); it('resets `totalPayloadLength` only on final frame (fragmented + ping)', (done) => { const receiver = new Receiver(undefined, {}, 10); let data; receiver.on('ping', (buf) => { assert.strictEqual(receiver._totalPayloadLength, 2); data = buf.toString(); }); receiver.on('message', (buf) => { assert.strictEqual(receiver._totalPayloadLength, 0); assert.strictEqual(data, ''); assert.strictEqual(buf.toString(), 'Hello'); done(); }); assert.strictEqual(receiver._totalPayloadLength, 0); receiver.write(Buffer.from('02024865', 'hex')); receiver.write(Buffer.from('8900', 'hex')); receiver.write(Buffer.from('80036c6c6f', 'hex')); }); it('ignores any data after a close frame', (done) => { const perMessageDeflate = new PerMessageDeflate(); perMessageDeflate.accept([{}]); const receiver = new Receiver(undefined, { 'permessage-deflate': perMessageDeflate }); const results = []; const push = results.push.bind(results); receiver.on('conclude', push).on('message', push); receiver.on('finish', () => { assert.deepStrictEqual(results, ['', 1005, '']); done(); }); receiver.write(Buffer.from([0xc1, 0x01, 0x00])); receiver.write(Buffer.from([0x88, 0x00])); receiver.write(Buffer.from([0x81, 0x00])); }); it('emits an error if RSV1 is on and permessage-deflate is disabled', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: RSV1 must be clear' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0xc2, 0x80, 0x00, 0x00, 0x00, 0x00])); }); it('emits an error if RSV1 is on and opcode is 0', (done) => { const perMessageDeflate = new PerMessageDeflate(); perMessageDeflate.accept([{}]); const receiver = new Receiver(undefined, { 'permessage-deflate': perMessageDeflate }); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: RSV1 must be clear' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0x40, 0x00])); }); it('emits an error if RSV2 is on', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: RSV2 and RSV3 must be clear' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0xa2, 0x00])); }); it('emits an error if RSV3 is on', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: RSV2 and RSV3 must be clear' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0x92, 0x00])); }); it('emits an error if the first frame in a fragmented message has opcode 0', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: invalid opcode 0' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0x00, 0x00])); }); it('emits an error if a frame has opcode 1 in the middle of a fragmented message', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: invalid opcode 1' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0x01, 0x00])); receiver.write(Buffer.from([0x01, 0x00])); }); it('emits an error if a frame has opcode 2 in the middle of a fragmented message', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: invalid opcode 2' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0x01, 0x00])); receiver.write(Buffer.from([0x02, 0x00])); }); it('emits an error if a control frame has the FIN bit off', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: FIN must be set' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0x09, 0x00])); }); it('emits an error if a control frame has the RSV1 bit on', (done) => { const perMessageDeflate = new PerMessageDeflate(); perMessageDeflate.accept([{}]); const receiver = new Receiver(undefined, { 'permessage-deflate': perMessageDeflate }); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: RSV1 must be clear' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0xc9, 0x00])); }); it('emits an error if a control frame has the FIN bit off', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: FIN must be set' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0x09, 0x00])); }); it('emits an error if a control frame has a payload bigger than 125 B', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: invalid payload length 126' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0x89, 0x7e])); }); it('emits an error if a data frame has a payload bigger than 2^53 - 1 B', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Unsupported WebSocket frame: payload length > 2^53 - 1' ); assert.strictEqual(err[kStatusCode], 1009); done(); }); receiver.write(Buffer.from([0x82, 0x7f])); setImmediate(() => receiver.write( Buffer.from([0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00]) ) ); }); it('emits an error if a text frame contains invalid UTF-8 data (1/2)', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'Invalid WebSocket frame: invalid UTF-8 sequence' ); assert.strictEqual(err[kStatusCode], 1007); done(); }); receiver.write(Buffer.from([0x81, 0x04, 0xce, 0xba, 0xe1, 0xbd])); }); it('emits an error if a text frame contains invalid UTF-8 data (2/2)', (done) => { const perMessageDeflate = new PerMessageDeflate(); perMessageDeflate.accept([{}]); const receiver = new Receiver(undefined, { 'permessage-deflate': perMessageDeflate }); const buf = Buffer.from([0xce, 0xba, 0xe1, 0xbd]); receiver.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'Invalid WebSocket frame: invalid UTF-8 sequence' ); assert.strictEqual(err[kStatusCode], 1007); done(); }); perMessageDeflate.compress(buf, true, (err, data) => { if (err) return done(err); receiver.write(Buffer.from([0xc1, data.length])); receiver.write(data); }); }); it('emits an error if a close frame has a payload of 1 B', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: invalid payload length 1' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0x88, 0x01, 0x00])); }); it('emits an error if a close frame contains an invalid close code', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: invalid status code 0' ); assert.strictEqual(err[kStatusCode], 1002); done(); }); receiver.write(Buffer.from([0x88, 0x02, 0x00, 0x00])); }); it('emits an error if a close frame contains invalid UTF-8 data', (done) => { const receiver = new Receiver(); receiver.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'Invalid WebSocket frame: invalid UTF-8 sequence' ); assert.strictEqual(err[kStatusCode], 1007); done(); }); receiver.write( Buffer.from([0x88, 0x06, 0x03, 0xef, 0xce, 0xba, 0xe1, 0xbd]) ); }); it('emits an error if a frame payload length is bigger than `maxPayload`', (done) => { const receiver = new Receiver(undefined, {}, 20 * 1024); const msg = crypto.randomBytes(200 * 1024); const list = Sender.frame(msg, { fin: true, rsv1: false, opcode: 0x02, mask: true, readOnly: true }); const frame = Buffer.concat(list); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual(err.message, 'Max payload size exceeded'); assert.strictEqual(err[kStatusCode], 1009); done(); }); receiver.write(frame); }); it('emits an error if the message length exceeds `maxPayload`', (done) => { const perMessageDeflate = new PerMessageDeflate({}, false, 25); perMessageDeflate.accept([{}]); const receiver = new Receiver( undefined, { 'permessage-deflate': perMessageDeflate }, 25 ); const buf = Buffer.from('A'.repeat(50)); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual(err.message, 'Max payload size exceeded'); assert.strictEqual(err[kStatusCode], 1009); done(); }); perMessageDeflate.compress(buf, true, (err, data) => { if (err) return done(err); receiver.write(Buffer.from([0xc1, data.length])); receiver.write(data); }); }); it('emits an error if the sum of fragment lengths exceeds `maxPayload`', (done) => { const perMessageDeflate = new PerMessageDeflate({}, false, 25); perMessageDeflate.accept([{}]); const receiver = new Receiver( undefined, { 'permessage-deflate': perMessageDeflate }, 25 ); const buf = Buffer.from('A'.repeat(15)); receiver.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual(err.message, 'Max payload size exceeded'); assert.strictEqual(err[kStatusCode], 1009); done(); }); perMessageDeflate.compress(buf, false, (err, fragment1) => { if (err) return done(err); receiver.write(Buffer.from([0x41, fragment1.length])); receiver.write(fragment1); perMessageDeflate.compress(buf, true, (err, fragment2) => { if (err) return done(err); receiver.write(Buffer.from([0x80, fragment2.length])); receiver.write(fragment2); }); }); }); it("honors the 'nodebuffer' binary type", (done) => { const receiver = new Receiver(); const frags = [ crypto.randomBytes(7321), crypto.randomBytes(137), crypto.randomBytes(285787), crypto.randomBytes(3) ]; receiver.on('message', (data) => { assert.ok(Buffer.isBuffer(data)); assert.ok(data.equals(Buffer.concat(frags))); done(); }); frags.forEach((frag, i) => { Sender.frame(frag, { fin: i === frags.length - 1, opcode: i === 0 ? 2 : 0, readOnly: true, mask: false, rsv1: false }).forEach((buf) => receiver.write(buf)); }); }); it("honors the 'arraybuffer' binary type", (done) => { const receiver = new Receiver(); const frags = [ crypto.randomBytes(19221), crypto.randomBytes(954), crypto.randomBytes(623987) ]; receiver._binaryType = 'arraybuffer'; receiver.on('message', (data) => { assert.ok(data instanceof ArrayBuffer); assert.ok(Buffer.from(data).equals(Buffer.concat(frags))); done(); }); frags.forEach((frag, i) => { Sender.frame(frag, { fin: i === frags.length - 1, opcode: i === 0 ? 2 : 0, readOnly: true, mask: false, rsv1: false }).forEach((buf) => receiver.write(buf)); }); }); it("honors the 'fragments' binary type", (done) => { const receiver = new Receiver(); const frags = [ crypto.randomBytes(17), crypto.randomBytes(419872), crypto.randomBytes(83), crypto.randomBytes(9928), crypto.randomBytes(1) ]; receiver._binaryType = 'fragments'; receiver.on('message', (data) => { assert.deepStrictEqual(data, frags); done(); }); frags.forEach((frag, i) => { Sender.frame(frag, { fin: i === frags.length - 1, opcode: i === 0 ? 2 : 0, readOnly: true, mask: false, rsv1: false }).forEach((buf) => receiver.write(buf)); }); }); }); ws-7.2.1/test/sender.test.js000066400000000000000000000205471357512353100157420ustar00rootroot00000000000000'use strict'; const assert = require('assert'); const PerMessageDeflate = require('../lib/permessage-deflate'); const Sender = require('../lib/sender'); class MockSocket { constructor({ write } = {}) { this.readable = true; this.writable = true; if (write) this.write = write; } cork() {} write() {} uncork() {} } describe('Sender', () => { describe('.frame', () => { it('does not mutate the input buffer if data is `readOnly`', () => { const buf = Buffer.from([1, 2, 3, 4, 5]); Sender.frame(buf, { readOnly: true, rsv1: false, mask: true, opcode: 2, fin: true }); assert.ok(buf.equals(Buffer.from([1, 2, 3, 4, 5]))); }); it('sets RSV1 bit if compressed', () => { const list = Sender.frame(Buffer.from('hi'), { readOnly: false, mask: false, rsv1: true, opcode: 1, fin: true }); assert.strictEqual(list[0][0] & 0x40, 0x40); }); }); describe('#send', () => { it('compresses data if compress option is enabled', (done) => { const perMessageDeflate = new PerMessageDeflate({ threshold: 0 }); let count = 0; const mockSocket = new MockSocket({ write: (data) => { assert.strictEqual(data[0] & 0x40, 0x40); if (++count === 3) done(); } }); const sender = new Sender(mockSocket, { 'permessage-deflate': perMessageDeflate }); perMessageDeflate.accept([{}]); const options = { compress: true, fin: true }; const array = new Uint8Array([0x68, 0x69]); sender.send(array.buffer, options); sender.send(array, options); sender.send('hi', options); }); it('does not compress data for small payloads', (done) => { const perMessageDeflate = new PerMessageDeflate(); const mockSocket = new MockSocket({ write: (data) => { assert.notStrictEqual(data[0] & 0x40, 0x40); done(); } }); const sender = new Sender(mockSocket, { 'permessage-deflate': perMessageDeflate }); perMessageDeflate.accept([{}]); sender.send('hi', { compress: true, fin: true }); }); it('compresses all frames in a fragmented message', (done) => { const chunks = []; const perMessageDeflate = new PerMessageDeflate({ threshold: 3 }); const mockSocket = new MockSocket({ write: (chunk) => { chunks.push(chunk); if (chunks.length !== 4) return; assert.strictEqual(chunks[0].length, 2); assert.strictEqual(chunks[0][0] & 0x40, 0x40); assert.strictEqual(chunks[1].length, 9); assert.strictEqual(chunks[2].length, 2); assert.strictEqual(chunks[2][0] & 0x40, 0x00); assert.strictEqual(chunks[3].length, 4); done(); } }); const sender = new Sender(mockSocket, { 'permessage-deflate': perMessageDeflate }); perMessageDeflate.accept([{}]); sender.send('123', { compress: true, fin: false }); sender.send('12', { compress: true, fin: true }); }); it('compresses no frames in a fragmented message', (done) => { const chunks = []; const perMessageDeflate = new PerMessageDeflate({ threshold: 3 }); const mockSocket = new MockSocket({ write: (chunk) => { chunks.push(chunk); if (chunks.length !== 4) return; assert.strictEqual(chunks[0].length, 2); assert.strictEqual(chunks[0][0] & 0x40, 0x00); assert.strictEqual(chunks[1].length, 2); assert.strictEqual(chunks[2].length, 2); assert.strictEqual(chunks[2][0] & 0x40, 0x00); assert.strictEqual(chunks[3].length, 3); done(); } }); const sender = new Sender(mockSocket, { 'permessage-deflate': perMessageDeflate }); perMessageDeflate.accept([{}]); sender.send('12', { compress: true, fin: false }); sender.send('123', { compress: true, fin: true }); }); it('compresses empty buffer as first fragment', (done) => { const chunks = []; const perMessageDeflate = new PerMessageDeflate({ threshold: 0 }); const mockSocket = new MockSocket({ write: (chunk) => { chunks.push(chunk); if (chunks.length !== 4) return; assert.strictEqual(chunks[0].length, 2); assert.strictEqual(chunks[0][0] & 0x40, 0x40); assert.strictEqual(chunks[1].length, 5); assert.strictEqual(chunks[2].length, 2); assert.strictEqual(chunks[2][0] & 0x40, 0x00); assert.strictEqual(chunks[3].length, 6); done(); } }); const sender = new Sender(mockSocket, { 'permessage-deflate': perMessageDeflate }); perMessageDeflate.accept([{}]); sender.send(Buffer.alloc(0), { compress: true, fin: false }); sender.send('data', { compress: true, fin: true }); }); it('compresses empty buffer as last fragment', (done) => { const chunks = []; const perMessageDeflate = new PerMessageDeflate({ threshold: 0 }); const mockSocket = new MockSocket({ write: (chunk) => { chunks.push(chunk); if (chunks.length !== 4) return; assert.strictEqual(chunks[0].length, 2); assert.strictEqual(chunks[0][0] & 0x40, 0x40); assert.strictEqual(chunks[1].length, 10); assert.strictEqual(chunks[2].length, 2); assert.strictEqual(chunks[2][0] & 0x40, 0x00); assert.strictEqual(chunks[3].length, 1); done(); } }); const sender = new Sender(mockSocket, { 'permessage-deflate': perMessageDeflate }); perMessageDeflate.accept([{}]); sender.send('data', { compress: true, fin: false }); sender.send(Buffer.alloc(0), { compress: true, fin: true }); }); }); describe('#ping', () => { it('works with multiple types of data', (done) => { const perMessageDeflate = new PerMessageDeflate({ threshold: 0 }); let count = 0; const mockSocket = new MockSocket({ write: (data) => { if (++count < 3) return; if (count % 2) { assert.ok(data.equals(Buffer.from([0x89, 0x02]))); } else { assert.ok(data.equals(Buffer.from([0x68, 0x69]))); } if (count === 8) done(); } }); const sender = new Sender(mockSocket, { 'permessage-deflate': perMessageDeflate }); perMessageDeflate.accept([{}]); const array = new Uint8Array([0x68, 0x69]); sender.send('foo', { compress: true, fin: true }); sender.ping(array.buffer, false); sender.ping(array, false); sender.ping('hi', false); }); }); describe('#pong', () => { it('works with multiple types of data', (done) => { const perMessageDeflate = new PerMessageDeflate({ threshold: 0 }); let count = 0; const mockSocket = new MockSocket({ write: (data) => { if (++count < 3) return; if (count % 2) { assert.ok(data.equals(Buffer.from([0x8a, 0x02]))); } else { assert.ok(data.equals(Buffer.from([0x68, 0x69]))); } if (count === 8) done(); } }); const sender = new Sender(mockSocket, { 'permessage-deflate': perMessageDeflate }); perMessageDeflate.accept([{}]); const array = new Uint8Array([0x68, 0x69]); sender.send('foo', { compress: true, fin: true }); sender.pong(array.buffer, false); sender.pong(array, false); sender.pong('hi', false); }); }); describe('#close', () => { it('should consume all data before closing', (done) => { const perMessageDeflate = new PerMessageDeflate({ threshold: 0 }); let count = 0; const mockSocket = new MockSocket({ write: (data, cb) => { count++; if (cb) cb(); } }); const sender = new Sender(mockSocket, { 'permessage-deflate': perMessageDeflate }); perMessageDeflate.accept([{}]); sender.send('foo', { compress: true, fin: true }); sender.send('bar', { compress: true, fin: true }); sender.send('baz', { compress: true, fin: true }); sender.close(1000, undefined, false, () => { assert.strictEqual(count, 8); done(); }); }); }); }); ws-7.2.1/test/websocket-server.test.js000066400000000000000000000573341357512353100177600ustar00rootroot00000000000000/* eslint no-unused-vars: ["error", { "varsIgnorePattern": "^ws$" }] */ 'use strict'; const assert = require('assert'); const crypto = require('crypto'); const https = require('https'); const http = require('http'); const net = require('net'); const fs = require('fs'); const WebSocket = require('..'); describe('WebSocketServer', () => { describe('#ctor', () => { it('throws an error if no option object is passed', () => { assert.throws(() => new WebSocket.Server()); }); describe('options', () => { it('throws an error if no `port` or `server` option is specified', () => { assert.throws(() => new WebSocket.Server({})); }); it('throws an error if the server is already being used', () => { const server = http.createServer(); new WebSocket.Server({ server }); assert.throws( () => new WebSocket.Server({ server }), /^Error: The HTTP\/S server is already being used by another WebSocket server$/ ); }); it('exposes options passed to constructor', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { assert.strictEqual(wss.options.port, 0); wss.close(done); }); }); it('accepts the `maxPayload` option', (done) => { const maxPayload = 20480; const wss = new WebSocket.Server( { perMessageDeflate: true, maxPayload, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); } ); wss.on('connection', (ws) => { assert.strictEqual(ws._receiver._maxPayload, maxPayload); assert.strictEqual( ws._receiver._extensions['permessage-deflate']._maxPayload, maxPayload ); wss.close(done); }); }); }); it('emits an error if http server bind fails', (done) => { const wss1 = new WebSocket.Server({ port: 0 }, () => { const wss2 = new WebSocket.Server({ port: wss1.address().port }); wss2.on('error', () => wss1.close(done)); }); }); it('starts a server on a given port', (done) => { const port = 1337; const wss = new WebSocket.Server({ port }, () => { const ws = new WebSocket(`ws://localhost:${port}`); }); wss.on('connection', () => wss.close(done)); }); it('binds the server on any IPv6 address when available', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { assert.strictEqual(wss._server.address().address, '::'); wss.close(done); }); }); it('uses a precreated http server', (done) => { const server = http.createServer(); server.listen(0, () => { const wss = new WebSocket.Server({ server }); const ws = new WebSocket(`ws://localhost:${server.address().port}`); wss.on('connection', () => { wss.close(); server.close(done); }); }); }); it('426s for non-Upgrade requests', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { http.get(`http://localhost:${wss.address().port}`, (res) => { let body = ''; assert.strictEqual(res.statusCode, 426); res.on('data', (chunk) => { body += chunk; }); res.on('end', () => { assert.strictEqual(body, http.STATUS_CODES[426]); wss.close(done); }); }); }); }); it('uses a precreated http server listening on unix socket', function(done) { // // Skip this test on Windows as it throws errors for obvious reasons. // if (process.platform === 'win32') return this.skip(); const server = http.createServer(); const sockPath = `/tmp/ws.${crypto .randomBytes(16) .toString('hex')}.socket`; server.listen(sockPath, () => { const wss = new WebSocket.Server({ server }); wss.on('connection', (ws, req) => { if (wss.clients.size === 1) { assert.strictEqual(req.url, '/foo?bar=bar'); } else { assert.strictEqual(req.url, '/'); wss.close(); server.close(done); } }); const ws = new WebSocket(`ws+unix://${sockPath}:/foo?bar=bar`); ws.on('open', () => new WebSocket(`ws+unix://${sockPath}`)); }); }); }); describe('#address', () => { it('returns the address of the server', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const addr = wss.address(); assert.deepStrictEqual(addr, wss._server.address()); wss.close(done); }); }); it('throws an error when operating in "noServer" mode', () => { const wss = new WebSocket.Server({ noServer: true }); assert.throws(() => { wss.address(); }, /^Error: The server is operating in "noServer" mode$/); }); it('returns `null` if called after close', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { wss.close(() => { assert.strictEqual(wss.address(), null); done(); }); }); }); }); describe('#close', () => { it('does not throw when called twice', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { wss.close(); wss.close(); wss.close(); done(); }); }); it('closes all clients', (done) => { let closes = 0; const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('close', () => { if (++closes === 2) done(); }); }); wss.on('connection', (ws) => { ws.on('close', () => { if (++closes === 2) done(); }); wss.close(); }); }); it("doesn't close a precreated server", (done) => { const server = http.createServer(); const realClose = server.close; server.close = () => { done(new Error('Must not close pre-created server')); }; const wss = new WebSocket.Server({ server }); wss.on('connection', () => { wss.close(); server.close = realClose; server.close(done); }); server.listen(0, () => { const ws = new WebSocket(`ws://localhost:${server.address().port}`); }); }); it('invokes the callback in noServer mode', (done) => { const wss = new WebSocket.Server({ noServer: true }); wss.close(done); }); it('cleans event handlers on precreated server', (done) => { const server = http.createServer(); const wss1 = new WebSocket.Server({ server }); server.listen(0, () => { wss1.close(() => { assert.strictEqual(server.listenerCount('listening'), 0); assert.strictEqual(server.listenerCount('upgrade'), 0); assert.strictEqual(server.listenerCount('error'), 0); // Check that no error is thrown if `server` is resued now as there // are no other `WebSocketServer`s using it. const wss2 = new WebSocket.Server({ server }); wss2.close(() => { server.close(done); }); }); }); }); it("emits the 'close' event", (done) => { const wss = new WebSocket.Server({ noServer: true }); wss.on('close', done); wss.close(); }); }); describe('#clients', () => { it('returns a list of connected clients', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { assert.strictEqual(wss.clients.size, 0); const ws = new WebSocket(`ws://localhost:${wss.address().port}`); }); wss.on('connection', () => { assert.strictEqual(wss.clients.size, 1); wss.close(done); }); }); it('can be disabled', (done) => { const wss = new WebSocket.Server( { port: 0, clientTracking: false }, () => { assert.strictEqual(wss.clients, undefined); const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.close()); } ); wss.on('connection', (ws) => { assert.strictEqual(wss.clients, undefined); ws.on('close', () => wss.close(done)); }); }); it('is updated when client terminates the connection', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.terminate()); }); wss.on('connection', (ws) => { ws.on('close', () => { assert.strictEqual(wss.clients.size, 0); wss.close(done); }); }); }); it('is updated when client closes the connection', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.close()); }); wss.on('connection', (ws) => { ws.on('close', () => { assert.strictEqual(wss.clients.size, 0); wss.close(done); }); }); }); }); describe('#shouldHandle', () => { it('returns true when the path matches', () => { const wss = new WebSocket.Server({ noServer: true, path: '/foo' }); assert.strictEqual(wss.shouldHandle({ url: '/foo' }), true); }); it("returns false when the path doesn't match", () => { const wss = new WebSocket.Server({ noServer: true, path: '/foo' }); assert.strictEqual(wss.shouldHandle({ url: '/bar' }), false); }); }); describe('#handleUpgrade', () => { it('can be used for a pre-existing server', (done) => { const server = http.createServer(); server.listen(0, () => { const wss = new WebSocket.Server({ noServer: true }); server.on('upgrade', (req, socket, head) => { wss.handleUpgrade(req, socket, head, (client) => client.send('hello') ); }); const ws = new WebSocket(`ws://localhost:${server.address().port}`); ws.on('message', (message) => { assert.strictEqual(message, 'hello'); wss.close(); server.close(done); }); }); }); it("closes the connection when path doesn't match", (done) => { const wss = new WebSocket.Server({ port: 0, path: '/ws' }, () => { const req = http.get({ port: wss.address().port, headers: { Connection: 'Upgrade', Upgrade: 'websocket' } }); req.on('response', (res) => { assert.strictEqual(res.statusCode, 400); wss.close(done); }); }); }); it('closes the connection when protocol version is Hixie-76', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const req = http.get({ port: wss.address().port, headers: { Connection: 'Upgrade', Upgrade: 'WebSocket', 'Sec-WebSocket-Key1': '4 @1 46546xW%0l 1 5', 'Sec-WebSocket-Key2': '12998 5 Y3 1 .P00', 'Sec-WebSocket-Protocol': 'sample' } }); req.on('response', (res) => { assert.strictEqual(res.statusCode, 400); wss.close(done); }); }); }); }); describe('Connection establishing', () => { it('fails if the Sec-WebSocket-Key header is invalid (1/2)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const req = http.get({ port: wss.address().port, headers: { Connection: 'Upgrade', Upgrade: 'websocket' } }); req.on('response', (res) => { assert.strictEqual(res.statusCode, 400); wss.close(done); }); }); wss.on('connection', () => { done(new Error("Unexpected 'connection' event")); }); }); it('fails if the Sec-WebSocket-Key header is invalid (2/2)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const req = http.get({ port: wss.address().port, headers: { Connection: 'Upgrade', Upgrade: 'websocket', 'Sec-WebSocket-Key': 'P5l8BJcZwRc=' } }); req.on('response', (res) => { assert.strictEqual(res.statusCode, 400); wss.close(done); }); }); wss.on('connection', () => { done(new Error("Unexpected 'connection' event")); }); }); it('fails is the Sec-WebSocket-Version header is invalid (1/2)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const req = http.get({ port: wss.address().port, headers: { Connection: 'Upgrade', Upgrade: 'websocket', 'Sec-WebSocket-Key': 'dGhlIHNhbXBsZSBub25jZQ==' } }); req.on('response', (res) => { assert.strictEqual(res.statusCode, 400); wss.close(done); }); }); wss.on('connection', () => { done(new Error("Unexpected 'connection' event")); }); }); it('fails is the Sec-WebSocket-Version header is invalid (2/2)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const req = http.get({ port: wss.address().port, headers: { Connection: 'Upgrade', Upgrade: 'websocket', 'Sec-WebSocket-Key': 'dGhlIHNhbXBsZSBub25jZQ==', 'Sec-WebSocket-Version': 12 } }); req.on('response', (res) => { assert.strictEqual(res.statusCode, 400); wss.close(done); }); }); wss.on('connection', () => { done(new Error("Unexpected 'connection' event")); }); }); it('fails is the Sec-WebSocket-Extensions header is invalid', (done) => { const wss = new WebSocket.Server( { perMessageDeflate: true, port: 0 }, () => { const req = http.get({ port: wss.address().port, headers: { Connection: 'Upgrade', Upgrade: 'websocket', 'Sec-WebSocket-Key': 'dGhlIHNhbXBsZSBub25jZQ==', 'Sec-WebSocket-Version': 13, 'Sec-WebSocket-Extensions': 'permessage-deflate; server_max_window_bits=foo' } }); req.on('response', (res) => { assert.strictEqual(res.statusCode, 400); wss.close(done); }); } ); wss.on('connection', () => { done(new Error("Unexpected 'connection' event")); }); }); describe('`verifyClient`', () => { it('can reject client synchronously', (done) => { const wss = new WebSocket.Server( { verifyClient: () => false, port: 0 }, () => { const req = http.get({ port: wss.address().port, headers: { Connection: 'Upgrade', Upgrade: 'websocket', 'Sec-WebSocket-Key': 'dGhlIHNhbXBsZSBub25jZQ==', 'Sec-WebSocket-Version': 8 } }); req.on('response', (res) => { assert.strictEqual(res.statusCode, 401); wss.close(done); }); } ); wss.on('connection', () => { done(new Error("Unexpected 'connection' event")); }); }); it('can accept client synchronously', (done) => { const server = https.createServer({ cert: fs.readFileSync('test/fixtures/certificate.pem'), key: fs.readFileSync('test/fixtures/key.pem') }); const wss = new WebSocket.Server({ verifyClient: (info) => { assert.strictEqual(info.origin, 'https://example.com'); assert.strictEqual(info.req.headers.foo, 'bar'); assert.ok(info.secure, true); return true; }, server }); wss.on('connection', () => { wss.close(); server.close(done); }); server.listen(0, () => { const ws = new WebSocket(`wss://localhost:${server.address().port}`, { headers: { Origin: 'https://example.com', foo: 'bar' }, rejectUnauthorized: false }); }); }); it('can accept client asynchronously', (done) => { const wss = new WebSocket.Server( { verifyClient: (o, cb) => process.nextTick(cb, true), port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); } ); wss.on('connection', () => wss.close(done)); }); it('can reject client asynchronously', (done) => { const wss = new WebSocket.Server( { verifyClient: (info, cb) => process.nextTick(cb, false), port: 0 }, () => { const req = http.get({ port: wss.address().port, headers: { Connection: 'Upgrade', Upgrade: 'websocket', 'Sec-WebSocket-Key': 'dGhlIHNhbXBsZSBub25jZQ==', 'Sec-WebSocket-Version': 8 } }); req.on('response', (res) => { assert.strictEqual(res.statusCode, 401); wss.close(done); }); } ); wss.on('connection', () => { done(new Error("Unexpected 'connection' event")); }); }); it('can reject client asynchronously w/ status code', (done) => { const wss = new WebSocket.Server( { verifyClient: (info, cb) => process.nextTick(cb, false, 404), port: 0 }, () => { const req = http.get({ port: wss.address().port, headers: { Connection: 'Upgrade', Upgrade: 'websocket', 'Sec-WebSocket-Key': 'dGhlIHNhbXBsZSBub25jZQ==', 'Sec-WebSocket-Version': 8 } }); req.on('response', (res) => { assert.strictEqual(res.statusCode, 404); wss.close(done); }); } ); wss.on('connection', () => { done(new Error("Unexpected 'connection' event")); }); }); it('can reject client asynchronously w/ custom headers', (done) => { const wss = new WebSocket.Server( { verifyClient: (info, cb) => { process.nextTick(cb, false, 503, '', { 'Retry-After': 120 }); }, port: 0 }, () => { const req = http.get({ port: wss.address().port, headers: { Connection: 'Upgrade', Upgrade: 'websocket', 'Sec-WebSocket-Key': 'dGhlIHNhbXBsZSBub25jZQ==', 'Sec-WebSocket-Version': 8 } }); req.on('response', (res) => { assert.strictEqual(res.statusCode, 503); assert.strictEqual(res.headers['retry-after'], '120'); wss.close(done); }); } ); wss.on('connection', () => { done(new Error("Unexpected 'connection' event")); }); }); }); it("doesn't emit the 'connection' event if socket is closed prematurely", (done) => { const server = http.createServer(); server.listen(0, () => { const wss = new WebSocket.Server({ verifyClient: ({ req: { socket } }, cb) => { assert.strictEqual(socket.readable, true); assert.strictEqual(socket.writable, true); socket.on('end', () => { assert.strictEqual(socket.readable, false); assert.strictEqual(socket.writable, true); cb(true); }); }, server }); wss.on('connection', () => { done(new Error("Unexpected 'connection' event")); }); const socket = net.connect( { port: server.address().port, allowHalfOpen: true }, () => { socket.end( [ 'GET / HTTP/1.1', 'Host: localhost', 'Upgrade: websocket', 'Connection: Upgrade', 'Sec-WebSocket-Key: dGhlIHNhbXBsZSBub25jZQ==', 'Sec-WebSocket-Version: 13', '\r\n' ].join('\r\n') ); } ); socket.on('end', () => { wss.close(); server.close(done); }); }); }); it('handles data passed along with the upgrade request', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const req = http.request({ port: wss.address().port, headers: { Connection: 'Upgrade', Upgrade: 'websocket', 'Sec-WebSocket-Key': 'dGhlIHNhbXBsZSBub25jZQ==', 'Sec-WebSocket-Version': 13 } }); req.write(Buffer.from([0x81, 0x05, 0x48, 0x65, 0x6c, 0x6c, 0x6f])); req.end(); }); wss.on('connection', (ws) => { ws.on('message', (data) => { assert.strictEqual(data, 'Hello'); wss.close(done); }); }); }); describe('`handleProtocols`', () => { it('allows to select a subprotocol', (done) => { const handleProtocols = (protocols, request) => { assert.ok(request instanceof http.IncomingMessage); assert.strictEqual(request.url, '/'); return protocols.pop(); }; const wss = new WebSocket.Server({ handleProtocols, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`, [ 'foo', 'bar' ]); ws.on('open', () => { assert.strictEqual(ws.protocol, 'bar'); wss.close(done); }); }); }); }); it("emits the 'headers' event", (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); wss.on('headers', (headers, request) => { assert.deepStrictEqual(headers.slice(0, 3), [ 'HTTP/1.1 101 Switching Protocols', 'Upgrade: websocket', 'Connection: Upgrade' ]); assert.ok(request instanceof http.IncomingMessage); assert.strictEqual(request.url, '/'); wss.on('connection', () => wss.close(done)); }); }); }); }); describe('permessage-deflate', () => { it('is disabled by default', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); }); wss.on('connection', (ws, req) => { assert.strictEqual( req.headers['sec-websocket-extensions'], 'permessage-deflate; client_max_window_bits' ); assert.strictEqual(ws.extensions, ''); wss.close(done); }); }); it('uses configuration options', (done) => { const wss = new WebSocket.Server( { perMessageDeflate: { clientMaxWindowBits: 8 }, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('upgrade', (res) => { assert.strictEqual( res.headers['sec-websocket-extensions'], 'permessage-deflate; client_max_window_bits=8' ); wss.close(done); }); } ); }); }); }); ws-7.2.1/test/websocket.integration.js000066400000000000000000000022201357512353100200000ustar00rootroot00000000000000'use strict'; const assert = require('assert'); const WebSocket = require('..'); describe('WebSocket', () => { it('communicates successfully with echo service (ws)', (done) => { const ws = new WebSocket('ws://echo.websocket.org/', { origin: 'http://www.websocket.org', protocolVersion: 13 }); const str = Date.now().toString(); let dataReceived = false; ws.on('open', () => ws.send(str)); ws.on('close', () => { assert.ok(dataReceived); done(); }); ws.on('message', (data) => { dataReceived = true; assert.strictEqual(data, str); ws.close(); }); }); it('communicates successfully with echo service (wss)', (done) => { const ws = new WebSocket('wss://echo.websocket.org/', { origin: 'https://www.websocket.org', protocolVersion: 13 }); const str = Date.now().toString(); let dataReceived = false; ws.on('open', () => ws.send(str)); ws.on('close', () => { assert.ok(dataReceived); done(); }); ws.on('message', (data) => { dataReceived = true; assert.strictEqual(data, str); ws.close(); }); }); }); ws-7.2.1/test/websocket.test.js000066400000000000000000002140651357512353100164500ustar00rootroot00000000000000/* eslint no-unused-vars: ["error", { "varsIgnorePattern": "^ws$" }] */ 'use strict'; const assert = require('assert'); const crypto = require('crypto'); const https = require('https'); const http = require('http'); const tls = require('tls'); const fs = require('fs'); const { URL } = require('url'); const WebSocket = require('..'); const { GUID, NOOP } = require('../lib/constants'); class CustomAgent extends http.Agent { addRequest() {} } describe('WebSocket', () => { describe('#ctor', () => { it('throws an error when using an invalid url', () => { assert.throws( () => new WebSocket('ws+unix:'), /^Error: Invalid URL: ws\+unix:$/ ); }); it('accepts `url.URL` objects as url', function(done) { const agent = new CustomAgent(); agent.addRequest = (req, opts) => { assert.strictEqual(opts.host, '::1'); assert.strictEqual(req.path, '/'); done(); }; const ws = new WebSocket(new URL('ws://[::1]'), { agent }); }); describe('options', () => { it('accepts the `options` object as 3rd argument', () => { const agent = new CustomAgent(); let count = 0; let ws; agent.addRequest = () => count++; ws = new WebSocket('ws://localhost', undefined, { agent }); ws = new WebSocket('ws://localhost', null, { agent }); ws = new WebSocket('ws://localhost', [], { agent }); assert.strictEqual(count, 3); }); it('accepts the `maxPayload` option', (done) => { const maxPayload = 20480; const wss = new WebSocket.Server( { perMessageDeflate: true, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`, { perMessageDeflate: true, maxPayload }); ws.on('open', () => { assert.strictEqual(ws._receiver._maxPayload, maxPayload); assert.strictEqual( ws._receiver._extensions['permessage-deflate']._maxPayload, maxPayload ); wss.close(done); }); } ); }); it('throws an error when using an invalid `protocolVersion`', () => { const options = { agent: new CustomAgent(), protocolVersion: 1000 }; assert.throws( () => new WebSocket('ws://localhost', options), /^RangeError: Unsupported protocol version: 1000 \(supported versions: 8, 13\)$/ ); }); }); }); describe('Constants', () => { const readyStates = { CONNECTING: 0, OPEN: 1, CLOSING: 2, CLOSED: 3 }; Object.keys(readyStates).forEach((state) => { describe(`\`${state}\``, () => { it('is enumerable property of class', () => { const propertyDescripter = Object.getOwnPropertyDescriptor( WebSocket, state ); assert.strictEqual(propertyDescripter.value, readyStates[state]); assert.strictEqual(propertyDescripter.enumerable, true); }); it('is property of instance', () => { const ws = new WebSocket('ws://localhost', { agent: new CustomAgent() }); assert.strictEqual(ws[state], readyStates[state]); }); }); }); }); describe('Attributes', () => { describe('`binaryType`', () => { it("defaults to 'nodebuffer'", () => { const ws = new WebSocket('ws://localhost', { agent: new CustomAgent() }); assert.strictEqual(ws.binaryType, 'nodebuffer'); }); it("can be changed to 'arraybuffer' or 'fragments'", () => { const ws = new WebSocket('ws://localhost', { agent: new CustomAgent() }); ws.binaryType = 'arraybuffer'; assert.strictEqual(ws.binaryType, 'arraybuffer'); ws.binaryType = 'foo'; assert.strictEqual(ws.binaryType, 'arraybuffer'); ws.binaryType = 'fragments'; assert.strictEqual(ws.binaryType, 'fragments'); ws.binaryType = ''; assert.strictEqual(ws.binaryType, 'fragments'); ws.binaryType = 'nodebuffer'; assert.strictEqual(ws.binaryType, 'nodebuffer'); }); }); describe('`bufferedAmount`', () => { it('defaults to zero', () => { const ws = new WebSocket('ws://localhost', { agent: new CustomAgent() }); assert.strictEqual(ws.bufferedAmount, 0); }); it('defaults to zero upon "open"', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.onopen = () => { assert.strictEqual(ws.bufferedAmount, 0); wss.close(done); }; }); }); it('takes into account the data in the sender queue', (done) => { const wss = new WebSocket.Server( { perMessageDeflate: true, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`, { perMessageDeflate: { threshold: 0 } }); ws.on('open', () => { ws.send('foo'); ws.send('bar', (err) => { assert.ifError(err); assert.strictEqual(ws.bufferedAmount, 0); wss.close(done); }); assert.strictEqual(ws.bufferedAmount, 3); }); } ); }); it('takes into account the data in the socket queue', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); }); wss.on('connection', (ws) => { const data = Buffer.alloc(1024, 61); while (ws._socket.bufferSize === 0) { ws.send(data); } assert.ok(ws._socket.bufferSize > 0); assert.strictEqual(ws.bufferedAmount, ws._socket.bufferSize); ws.on('close', () => wss.close(done)); ws.close(); }); }); }); describe('`extensions`', () => { it('exposes the negotiated extensions names (1/2)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); assert.strictEqual(ws.extensions, ''); ws.on('open', () => { assert.strictEqual(ws.extensions, ''); ws.on('close', () => wss.close(done)); }); }); wss.on('connection', (ws) => { assert.strictEqual(ws.extensions, ''); ws.close(); }); }); it('exposes the negotiated extensions names (2/2)', (done) => { const wss = new WebSocket.Server( { perMessageDeflate: true, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); assert.strictEqual(ws.extensions, ''); ws.on('open', () => { assert.strictEqual(ws.extensions, 'permessage-deflate'); ws.on('close', () => wss.close(done)); }); } ); wss.on('connection', (ws) => { assert.strictEqual(ws.extensions, 'permessage-deflate'); ws.close(); }); }); }); describe('`protocol`', () => { it('exposes the subprotocol selected by the server', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const port = wss.address().port; const ws = new WebSocket(`ws://localhost:${port}`, 'foo'); assert.strictEqual(ws.extensions, ''); ws.on('open', () => { assert.strictEqual(ws.protocol, 'foo'); ws.on('close', () => wss.close(done)); }); }); wss.on('connection', (ws) => { assert.strictEqual(ws.protocol, 'foo'); ws.close(); }); }); }); describe('`readyState`', () => { it('defaults to `CONNECTING`', () => { const ws = new WebSocket('ws://localhost', { agent: new CustomAgent() }); assert.strictEqual(ws.readyState, WebSocket.CONNECTING); }); it('is set to `OPEN` once connection is established', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => { assert.strictEqual(ws.readyState, WebSocket.OPEN); ws.close(); }); ws.on('close', () => wss.close(done)); }); }); it('is set to `CLOSED` once connection is closed', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('close', () => { assert.strictEqual(ws.readyState, WebSocket.CLOSED); wss.close(done); }); ws.on('open', () => ws.close(1001)); }); }); it('is set to `CLOSED` once connection is terminated', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('close', () => { assert.strictEqual(ws.readyState, WebSocket.CLOSED); wss.close(done); }); ws.on('open', () => ws.terminate()); }); }); }); describe('`url`', () => { it('exposes the server url', () => { const url = 'ws://localhost'; const ws = new WebSocket(url, { agent: new CustomAgent() }); assert.strictEqual(ws.url, url); }); }); }); describe('Events', () => { it("emits an 'error' event if an error occurs", (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('error', (err) => { assert.ok(err instanceof RangeError); assert.strictEqual( err.message, 'Invalid WebSocket frame: invalid opcode 5' ); ws.on('close', (code, reason) => { assert.strictEqual(code, 1002); assert.strictEqual(reason, ''); wss.close(done); }); }); }); wss.on('connection', (ws) => { ws._socket.write(Buffer.from([0x85, 0x00])); }); }); it('does not re-emit `net.Socket` errors', (done) => { const codes = ['EPIPE', 'ECONNABORTED', 'ECANCELED', 'ECONNRESET']; const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => { ws._socket.on('error', (err) => { assert.ok(err instanceof Error); assert.ok(codes.includes(err.code), `Unexpected code: ${err.code}`); ws.on('close', (code, message) => { assert.strictEqual(message, ''); assert.strictEqual(code, 1006); wss.close(done); }); }); for (const client of wss.clients) client.terminate(); ws.send('foo'); ws.send('bar'); }); }); }); it("emits an 'upgrade' event", (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('upgrade', (res) => { assert.ok(res instanceof http.IncomingMessage); wss.close(done); }); }); }); it("emits a 'ping' event", (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('ping', () => wss.close(done)); }); wss.on('connection', (ws) => ws.ping()); }); it("emits a 'pong' event", (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('pong', () => wss.close(done)); }); wss.on('connection', (ws) => ws.pong()); }); }); describe('Connection establishing', () => { const server = http.createServer(); beforeEach((done) => server.listen(0, done)); afterEach((done) => server.close(done)); it('fails if the Sec-WebSocket-Accept header is invalid', (done) => { server.once('upgrade', (req, socket) => { socket.on('end', socket.end); socket.write( 'HTTP/1.1 101 Switching Protocols\r\n' + 'Upgrade: websocket\r\n' + 'Connection: Upgrade\r\n' + 'Sec-WebSocket-Accept: CxYS6+NgJSBG74mdgLvGscRvpns=\r\n' + '\r\n' ); }); const ws = new WebSocket(`ws://localhost:${server.address().port}`); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual(err.message, 'Invalid Sec-WebSocket-Accept header'); done(); }); }); it('close event is raised when server closes connection', (done) => { server.once('upgrade', (req, socket) => { const key = crypto .createHash('sha1') .update(req.headers['sec-websocket-key'] + GUID) .digest('base64'); socket.end( 'HTTP/1.1 101 Switching Protocols\r\n' + 'Upgrade: websocket\r\n' + 'Connection: Upgrade\r\n' + `Sec-WebSocket-Accept: ${key}\r\n` + '\r\n' ); }); const ws = new WebSocket(`ws://localhost:${server.address().port}`); ws.on('close', (code, reason) => { assert.strictEqual(code, 1006); assert.strictEqual(reason, ''); done(); }); }); it('error is emitted if server aborts connection', (done) => { server.once('upgrade', (req, socket) => { socket.end( `HTTP/1.1 401 ${http.STATUS_CODES[401]}\r\n` + 'Connection: close\r\n' + 'Content-type: text/html\r\n' + `Content-Length: ${http.STATUS_CODES[401].length}\r\n` + '\r\n' ); }); const ws = new WebSocket(`ws://localhost:${server.address().port}`); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual(err.message, 'Unexpected server response: 401'); done(); }); }); it('unexpected response can be read when sent by server', (done) => { server.once('upgrade', (req, socket) => { socket.end( `HTTP/1.1 401 ${http.STATUS_CODES[401]}\r\n` + 'Connection: close\r\n' + 'Content-type: text/html\r\n' + 'Content-Length: 3\r\n' + '\r\n' + 'foo' ); }); const ws = new WebSocket(`ws://localhost:${server.address().port}`); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', () => done(new Error("Unexpected 'error' event"))); ws.on('unexpected-response', (req, res) => { assert.strictEqual(res.statusCode, 401); let data = ''; res.on('data', (v) => { data += v; }); res.on('end', () => { assert.strictEqual(data, 'foo'); done(); }); }); }); it('request can be aborted when unexpected response is sent by server', (done) => { server.once('upgrade', (req, socket) => { socket.end( `HTTP/1.1 401 ${http.STATUS_CODES[401]}\r\n` + 'Connection: close\r\n' + 'Content-type: text/html\r\n' + 'Content-Length: 3\r\n' + '\r\n' + 'foo' ); }); const ws = new WebSocket(`ws://localhost:${server.address().port}`); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', () => done(new Error("Unexpected 'error' event"))); ws.on('unexpected-response', (req, res) => { assert.strictEqual(res.statusCode, 401); res.on('end', done); req.abort(); }); }); it('fails if the opening handshake timeout expires', (done) => { server.once('upgrade', (req, socket) => socket.on('end', socket.end)); const port = server.address().port; const ws = new WebSocket(`ws://localhost:${port}`, null, { handshakeTimeout: 100 }); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual(err.message, 'Opening handshake has timed out'); done(); }); }); it('fails if the Sec-WebSocket-Extensions response header is invalid', (done) => { server.once('upgrade', (req, socket) => { const key = crypto .createHash('sha1') .update(req.headers['sec-websocket-key'] + GUID) .digest('base64'); socket.end( 'HTTP/1.1 101 Switching Protocols\r\n' + 'Upgrade: websocket\r\n' + 'Connection: Upgrade\r\n' + `Sec-WebSocket-Accept: ${key}\r\n` + 'Sec-WebSocket-Extensions: foo;=\r\n' + '\r\n' ); }); const ws = new WebSocket(`ws://localhost:${server.address().port}`); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'Invalid Sec-WebSocket-Extensions header' ); ws.on('close', () => done()); }); }); it('fails if server sends a subprotocol when none was requested', (done) => { const wss = new WebSocket.Server({ server }); wss.on('headers', (headers) => { headers.push('Sec-WebSocket-Protocol: foo'); }); const ws = new WebSocket(`ws://localhost:${server.address().port}`); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'Server sent a subprotocol but none was requested' ); ws.on('close', () => wss.close(done)); }); }); it('fails if server sends an invalid subprotocol', (done) => { const wss = new WebSocket.Server({ handleProtocols: () => 'baz', server }); const ws = new WebSocket(`ws://localhost:${server.address().port}`, [ 'foo', 'bar' ]); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual(err.message, 'Server sent an invalid subprotocol'); ws.on('close', () => wss.close(done)); }); }); it('fails if server sends no subprotocol', (done) => { const wss = new WebSocket.Server({ handleProtocols() {}, server }); const ws = new WebSocket(`ws://localhost:${server.address().port}`, [ 'foo', 'bar' ]); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual(err.message, 'Server sent no subprotocol'); ws.on('close', () => wss.close(done)); }); }); it('does not follow redirects by default', (done) => { server.once('upgrade', (req, socket) => { socket.end( 'HTTP/1.1 301 Moved Permanently\r\n' + 'Location: ws://localhost:8080\r\n' + '\r\n' ); }); const ws = new WebSocket(`ws://localhost:${server.address().port}`); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual(err.message, 'Unexpected server response: 301'); assert.strictEqual(ws._redirects, 0); ws.on('close', () => done()); }); }); it('honors the `followRedirects` option', (done) => { const wss = new WebSocket.Server({ noServer: true, path: '/foo' }); server.once('upgrade', (req, socket) => { socket.end('HTTP/1.1 302 Found\r\nLocation: /foo\r\n\r\n'); server.once('upgrade', (req, socket, head) => { wss.handleUpgrade(req, socket, head, NOOP); }); }); const port = server.address().port; const ws = new WebSocket(`ws://localhost:${port}`, { followRedirects: true }); ws.on('open', () => { assert.strictEqual(ws.url, `ws://localhost:${port}/foo`); assert.strictEqual(ws._redirects, 1); ws.on('close', () => done()); ws.close(); }); }); it('honors the `maxRedirects` option', (done) => { const onUpgrade = (req, socket) => { socket.end('HTTP/1.1 302 Found\r\nLocation: /\r\n\r\n'); }; server.on('upgrade', onUpgrade); const ws = new WebSocket(`ws://localhost:${server.address().port}`, { followRedirects: true, maxRedirects: 1 }); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual(err.message, 'Maximum redirects exceeded'); assert.strictEqual(ws._redirects, 2); server.removeListener('upgrade', onUpgrade); ws.on('close', () => done()); }); }); }); describe('Connection with query string', () => { it('connects when pathname is not null', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const port = wss.address().port; const ws = new WebSocket(`ws://localhost:${port}/?token=qwerty`); ws.on('open', () => wss.close(done)); }); }); it('connects when pathname is null', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const port = wss.address().port; const ws = new WebSocket(`ws://localhost:${port}?token=qwerty`); ws.on('open', () => wss.close(done)); }); }); }); describe('#ping', () => { it('throws an error if `readyState` is `CONNECTING`', () => { const ws = new WebSocket('ws://localhost', { lookup() {} }); assert.throws( () => ws.ping(), /^Error: WebSocket is not open: readyState 0 \(CONNECTING\)$/ ); assert.throws( () => ws.ping(NOOP), /^Error: WebSocket is not open: readyState 0 \(CONNECTING\)$/ ); }); it('increases `bufferedAmount` if `readyState` is 2 or 3', (done) => { const ws = new WebSocket('ws://localhost', { lookup() {} }); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket was closed before the connection was established' ); assert.strictEqual(ws.readyState, WebSocket.CLOSING); assert.strictEqual(ws.bufferedAmount, 0); ws.ping('hi'); assert.strictEqual(ws.bufferedAmount, 2); ws.ping(); assert.strictEqual(ws.bufferedAmount, 2); ws.on('close', () => { assert.strictEqual(ws.readyState, WebSocket.CLOSED); ws.ping('hi'); assert.strictEqual(ws.bufferedAmount, 4); ws.ping(); assert.strictEqual(ws.bufferedAmount, 4); done(); }); }); ws.close(); }); it('calls the callback w/ an error if `readyState` is 2 or 3', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); }); wss.on('connection', (ws) => { ws.close(); assert.strictEqual(ws.bufferedAmount, 0); ws.ping('hi', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket is not open: readyState 2 (CLOSING)' ); assert.strictEqual(ws.bufferedAmount, 2); ws.on('close', () => { ws.ping((err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket is not open: readyState 3 (CLOSED)' ); assert.strictEqual(ws.bufferedAmount, 2); wss.close(done); }); }); }); }); }); it('can send a ping with no data', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => { ws.ping(() => ws.ping()); }); }); wss.on('connection', (ws) => { let pings = 0; ws.on('ping', (data) => { assert.ok(Buffer.isBuffer(data)); assert.strictEqual(data.length, 0); if (++pings === 2) wss.close(done); }); }); }); it('can send a ping with data', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => { ws.ping('hi', () => ws.ping('hi', true)); }); }); wss.on('connection', (ws) => { let pings = 0; ws.on('ping', (message) => { assert.strictEqual(message.toString(), 'hi'); if (++pings === 2) wss.close(done); }); }); }); it('can send numbers as ping payload', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.ping(0)); }); wss.on('connection', (ws) => { ws.on('ping', (message) => { assert.strictEqual(message.toString(), '0'); wss.close(done); }); }); }); }); describe('#pong', () => { it('throws an error if `readyState` is `CONNECTING`', () => { const ws = new WebSocket('ws://localhost', { lookup() {} }); assert.throws( () => ws.pong(), /^Error: WebSocket is not open: readyState 0 \(CONNECTING\)$/ ); assert.throws( () => ws.pong(NOOP), /^Error: WebSocket is not open: readyState 0 \(CONNECTING\)$/ ); }); it('increases `bufferedAmount` if `readyState` is 2 or 3', (done) => { const ws = new WebSocket('ws://localhost', { lookup() {} }); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket was closed before the connection was established' ); assert.strictEqual(ws.readyState, WebSocket.CLOSING); assert.strictEqual(ws.bufferedAmount, 0); ws.pong('hi'); assert.strictEqual(ws.bufferedAmount, 2); ws.pong(); assert.strictEqual(ws.bufferedAmount, 2); ws.on('close', () => { assert.strictEqual(ws.readyState, WebSocket.CLOSED); ws.pong('hi'); assert.strictEqual(ws.bufferedAmount, 4); ws.pong(); assert.strictEqual(ws.bufferedAmount, 4); done(); }); }); ws.close(); }); it('calls the callback w/ an error if `readyState` is 2 or 3', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); }); wss.on('connection', (ws) => { ws.close(); assert.strictEqual(ws.bufferedAmount, 0); ws.pong('hi', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket is not open: readyState 2 (CLOSING)' ); assert.strictEqual(ws.bufferedAmount, 2); ws.on('close', () => { ws.pong((err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket is not open: readyState 3 (CLOSED)' ); assert.strictEqual(ws.bufferedAmount, 2); wss.close(done); }); }); }); }); }); it('can send a pong with no data', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => { ws.pong(() => ws.pong()); }); }); wss.on('connection', (ws) => { let pongs = 0; ws.on('pong', (data) => { assert.ok(Buffer.isBuffer(data)); assert.strictEqual(data.length, 0); if (++pongs === 2) wss.close(done); }); }); }); it('can send a pong with data', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => { ws.pong('hi', () => ws.pong('hi', true)); }); }); wss.on('connection', (ws) => { let pongs = 0; ws.on('pong', (message) => { assert.strictEqual(message.toString(), 'hi'); if (++pongs === 2) wss.close(done); }); }); }); it('can send numbers as pong payload', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.pong(0)); }); wss.on('connection', (ws) => { ws.on('pong', (message) => { assert.strictEqual(message.toString(), '0'); wss.close(done); }); }); }); }); describe('#send', () => { it('throws an error if `readyState` is `CONNECTING`', () => { const ws = new WebSocket('ws://localhost', { lookup() {} }); assert.throws( () => ws.send('hi'), /^Error: WebSocket is not open: readyState 0 \(CONNECTING\)$/ ); assert.throws( () => ws.send('hi', NOOP), /^Error: WebSocket is not open: readyState 0 \(CONNECTING\)$/ ); }); it('increases `bufferedAmount` if `readyState` is 2 or 3', (done) => { const ws = new WebSocket('ws://localhost', { lookup() {} }); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket was closed before the connection was established' ); assert.strictEqual(ws.readyState, WebSocket.CLOSING); assert.strictEqual(ws.bufferedAmount, 0); ws.send('hi'); assert.strictEqual(ws.bufferedAmount, 2); ws.send(); assert.strictEqual(ws.bufferedAmount, 2); ws.on('close', () => { assert.strictEqual(ws.readyState, WebSocket.CLOSED); ws.send('hi'); assert.strictEqual(ws.bufferedAmount, 4); ws.send(); assert.strictEqual(ws.bufferedAmount, 4); done(); }); }); ws.close(); }); it('calls the callback w/ an error if `readyState` is 2 or 3', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); }); wss.on('connection', (ws) => { ws.close(); assert.strictEqual(ws.bufferedAmount, 0); ws.send('hi', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket is not open: readyState 2 (CLOSING)' ); assert.strictEqual(ws.bufferedAmount, 2); ws.on('close', () => { ws.send('hi', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket is not open: readyState 3 (CLOSED)' ); assert.strictEqual(ws.bufferedAmount, 4); wss.close(done); }); }); }); }); }); it('can send a big binary message', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const array = new Float32Array(5 * 1024 * 1024); for (let i = 0; i < array.length; i++) { array[i] = i / 5; } const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.send(array, { compress: false })); ws.on('message', (msg) => { assert.ok(msg.equals(Buffer.from(array.buffer))); wss.close(done); }); }); wss.on('connection', (ws) => { ws.on('message', (msg) => ws.send(msg, { compress: false })); }); }); it('can send text data', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.send('hi')); ws.on('message', (message) => { assert.strictEqual(message, 'hi'); wss.close(done); }); }); wss.on('connection', (ws) => { ws.on('message', (msg) => ws.send(msg)); }); }); it('does not override the `fin` option', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => { ws.send('fragment', { fin: false }); ws.send('fragment', { fin: true }); }); }); wss.on('connection', (ws) => { ws.on('message', (msg) => { assert.strictEqual(msg, 'fragmentfragment'); wss.close(done); }); }); }); it('sends numbers as strings', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.send(0)); }); wss.on('connection', (ws) => { ws.on('message', (msg) => { assert.strictEqual(msg, '0'); wss.close(done); }); }); }); it('can send binary data as an array', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const array = new Float32Array(6); for (let i = 0; i < array.length; ++i) { array[i] = i / 2; } const partial = array.subarray(2, 5); const buf = Buffer.from(partial.buffer).slice( partial.byteOffset, partial.byteOffset + partial.byteLength ); const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.send(partial, { binary: true })); ws.on('message', (message) => { assert.ok(message.equals(buf)); wss.close(done); }); }); wss.on('connection', (ws) => { ws.on('message', (msg) => ws.send(msg)); }); }); it('can send binary data as a buffer', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const buf = Buffer.from('foobar'); const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.send(buf, { binary: true })); ws.on('message', (message) => { assert.ok(message.equals(buf)); wss.close(done); }); }); wss.on('connection', (ws) => { ws.on('message', (msg) => ws.send(msg)); }); }); it('can send an `ArrayBuffer`', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const array = new Float32Array(5); for (let i = 0; i < array.length; ++i) { array[i] = i / 2; } const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.send(array.buffer)); ws.onmessage = (event) => { assert.ok(event.data.equals(Buffer.from(array.buffer))); wss.close(done); }; }); wss.on('connection', (ws) => { ws.on('message', (msg) => ws.send(msg)); }); }); it('can send a `Buffer`', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const buf = Buffer.from('foobar'); const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.send(buf)); ws.onmessage = (event) => { assert.ok(event.data.equals(buf)); wss.close(done); }; }); wss.on('connection', (ws) => { ws.on('message', (msg) => ws.send(msg)); }); }); it('calls the callback when data is written out', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => { ws.send('hi', (err) => { assert.ifError(err); wss.close(done); }); }); }); }); it('works when the `data` argument is falsy', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.send()); }); wss.on('connection', (ws) => { ws.on('message', (message) => { assert.ok(message.equals(Buffer.alloc(0))); wss.close(done); }); }); }); it('can send text data with `mask` option set to `false`', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.send('hi', { mask: false })); }); wss.on('connection', (ws) => { ws.on('message', (message) => { assert.strictEqual(message, 'hi'); wss.close(done); }); }); }); it('can send binary data with `mask` option set to `false`', (done) => { const array = new Float32Array(5); for (let i = 0; i < array.length; ++i) { array[i] = i / 2; } const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.send(array, { mask: false })); }); wss.on('connection', (ws) => { ws.on('message', (message) => { assert.ok(message.equals(Buffer.from(array.buffer))); wss.close(done); }); }); }); }); describe('#close', () => { it('closes the connection if called while connecting (1/2)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket was closed before the connection was established' ); ws.on('close', () => wss.close(done)); }); ws.close(1001); }); }); it('closes the connection if called while connecting (2/2)', (done) => { const wss = new WebSocket.Server( { verifyClient: (info, cb) => setTimeout(cb, 300, true), port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket was closed before the connection was established' ); ws.on('close', () => wss.close(done)); }); setTimeout(() => ws.close(1001), 150); } ); }); it('can be called from an error listener while connecting', (done) => { const ws = new WebSocket('ws://localhost:1337'); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual(err.code, 'ECONNREFUSED'); ws.close(); ws.on('close', () => done()); }); }).timeout(4000); it("can be called from a listener of the 'upgrade' event", (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket was closed before the connection was established' ); ws.on('close', () => wss.close(done)); }); ws.on('upgrade', () => ws.close()); }); }); it('throws an error if the first argument is invalid (1/2)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => { assert.throws( () => ws.close('error'), /^TypeError: First argument must be a valid error code number$/ ); wss.close(done); }); }); }); it('throws an error if the first argument is invalid (2/2)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => { assert.throws( () => ws.close(1004), /^TypeError: First argument must be a valid error code number$/ ); wss.close(done); }); }); }); it('sends the close status code only when necessary', (done) => { let sent; const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => { ws._socket.once('data', (data) => { sent = data; }); }); }); wss.on('connection', (ws) => { ws._socket.once('data', (received) => { assert.ok(received.slice(0, 2).equals(Buffer.from([0x88, 0x80]))); assert.ok(sent.equals(Buffer.from([0x88, 0x00]))); ws.on('close', (code, reason) => { assert.strictEqual(code, 1005); assert.strictEqual(reason, ''); wss.close(done); }); }); ws.close(); }); }); it('works when close reason is not specified', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.close(1000)); }); wss.on('connection', (ws) => { ws.on('close', (code, message) => { assert.strictEqual(message, ''); assert.strictEqual(code, 1000); wss.close(done); }); }); }); it('works when close reason is specified', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => ws.close(1000, 'some reason')); }); wss.on('connection', (ws) => { ws.on('close', (code, message) => { assert.strictEqual(message, 'some reason'); assert.strictEqual(code, 1000); wss.close(done); }); }); }); it('permits all buffered data to be delivered', (done) => { const wss = new WebSocket.Server( { perMessageDeflate: { threshold: 0 }, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const messages = []; ws.on('message', (message) => messages.push(message)); ws.on('close', (code) => { assert.strictEqual(code, 1005); assert.deepStrictEqual(messages, ['foo', 'bar', 'baz']); wss.close(done); }); } ); wss.on('connection', (ws) => { const callback = (err) => assert.ifError(err); ws.send('foo', callback); ws.send('bar', callback); ws.send('baz', callback); ws.close(); ws.close(); }); }); it('allows close code 1013', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('close', (code) => { assert.strictEqual(code, 1013); wss.close(done); }); }); wss.on('connection', (ws) => ws.close(1013)); }); it('does nothing if `readyState` is `CLOSED`', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('close', (code) => { assert.strictEqual(code, 1005); assert.strictEqual(ws.readyState, WebSocket.CLOSED); ws.close(); wss.close(done); }); }); wss.on('connection', (ws) => ws.close()); }); it('sets a timer for the closing handshake to complete', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('close', (code, reason) => { assert.strictEqual(code, 1000); assert.strictEqual(reason, 'some reason'); wss.close(done); }); ws.on('open', () => { let callbackCalled = false; assert.strictEqual(ws._closeTimer, null); ws.send('foo', () => { callbackCalled = true; }); ws.close(1000, 'some reason'); // // Check that the close timer is set even if the `Sender.close()` // callback is not called. // assert.strictEqual(callbackCalled, false); assert.strictEqual(ws._closeTimer._idleTimeout, 30000); }); }); }); }); describe('#terminate', () => { it('closes the connection if called while connecting (1/2)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket was closed before the connection was established' ); ws.on('close', () => wss.close(done)); }); ws.terminate(); }); }); it('closes the connection if called while connecting (2/2)', (done) => { const wss = new WebSocket.Server( { verifyClient: (info, cb) => setTimeout(cb, 300, true), port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket was closed before the connection was established' ); ws.on('close', () => wss.close(done)); }); setTimeout(() => ws.terminate(), 150); } ); }); it('can be called from an error listener while connecting', (done) => { const ws = new WebSocket('ws://localhost:1337'); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual(err.code, 'ECONNREFUSED'); ws.terminate(); ws.on('close', () => done()); }); }).timeout(4000); it("can be called from a listener of the 'upgrade' event", (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('open', () => done(new Error("Unexpected 'open' event"))); ws.on('error', (err) => { assert.ok(err instanceof Error); assert.strictEqual( err.message, 'WebSocket was closed before the connection was established' ); ws.on('close', () => wss.close(done)); }); ws.on('upgrade', () => ws.terminate()); }); }); it('does nothing if `readyState` is `CLOSED`', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.on('close', (code) => { assert.strictEqual(code, 1006); assert.strictEqual(ws.readyState, WebSocket.CLOSED); ws.terminate(); wss.close(done); }); }); wss.on('connection', (ws) => ws.terminate()); }); }); describe('WHATWG API emulation', () => { it('supports the `on{close,error,message,open}` attributes', () => { const ws = new WebSocket('ws://localhost', { agent: new CustomAgent() }); assert.strictEqual(ws.onmessage, undefined); assert.strictEqual(ws.onclose, undefined); assert.strictEqual(ws.onerror, undefined); assert.strictEqual(ws.onopen, undefined); ws.onmessage = NOOP; ws.onerror = NOOP; ws.onclose = NOOP; ws.onopen = NOOP; assert.strictEqual(ws.onmessage, NOOP); assert.strictEqual(ws.onclose, NOOP); assert.strictEqual(ws.onerror, NOOP); assert.strictEqual(ws.onopen, NOOP); }); it('works like the `EventEmitter` interface', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.onmessage = (messageEvent) => { assert.strictEqual(messageEvent.data, 'foo'); ws.onclose = (closeEvent) => { assert.strictEqual(closeEvent.wasClean, true); assert.strictEqual(closeEvent.code, 1005); assert.strictEqual(closeEvent.reason, ''); wss.close(done); }; ws.close(); }; ws.onopen = () => ws.send('foo'); }); wss.on('connection', (ws) => { ws.on('message', (msg) => ws.send(msg)); }); }); it("doesn't return listeners added with `on`", () => { const ws = new WebSocket('ws://localhost', { agent: new CustomAgent() }); ws.on('open', NOOP); assert.deepStrictEqual(ws.listeners('open'), [NOOP]); assert.strictEqual(ws.onopen, undefined); }); it("doesn't remove listeners added with `on`", () => { const ws = new WebSocket('ws://localhost', { agent: new CustomAgent() }); ws.on('close', NOOP); ws.onclose = NOOP; let listeners = ws.listeners('close'); assert.strictEqual(listeners.length, 2); assert.strictEqual(listeners[0], NOOP); assert.strictEqual(listeners[1]._listener, NOOP); ws.onclose = NOOP; listeners = ws.listeners('close'); assert.strictEqual(listeners.length, 2); assert.strictEqual(listeners[0], NOOP); assert.strictEqual(listeners[1]._listener, NOOP); }); it('adds listeners for custom events with `addEventListener`', () => { const ws = new WebSocket('ws://localhost', { agent: new CustomAgent() }); ws.addEventListener('foo', NOOP); assert.strictEqual(ws.listeners('foo')[0], NOOP); // // Fails silently when the `listener` is not a function. // ws.addEventListener('bar', {}); assert.strictEqual(ws.listeners('bar').length, 0); }); it('supports the `removeEventListener` method', () => { const ws = new WebSocket('ws://localhost', { agent: new CustomAgent() }); ws.addEventListener('message', NOOP); ws.addEventListener('open', NOOP); ws.addEventListener('foo', NOOP); assert.strictEqual(ws.listeners('message')[0]._listener, NOOP); assert.strictEqual(ws.listeners('open')[0]._listener, NOOP); assert.strictEqual(ws.listeners('foo')[0], NOOP); ws.removeEventListener('message', () => {}); assert.strictEqual(ws.listeners('message')[0]._listener, NOOP); ws.removeEventListener('message', NOOP); ws.removeEventListener('open', NOOP); ws.removeEventListener('foo', NOOP); assert.strictEqual(ws.listenerCount('message'), 0); assert.strictEqual(ws.listenerCount('open'), 0); assert.strictEqual(ws.listenerCount('foo'), 0); }); it('wraps text data in a `MessageEvent`', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.addEventListener('open', () => ws.send('hi')); ws.addEventListener('message', (messageEvent) => { assert.strictEqual(messageEvent.data, 'hi'); wss.close(done); }); }); wss.on('connection', (ws) => { ws.on('message', (msg) => ws.send(msg)); }); }); it('receives a `CloseEvent` when server closes (1000)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.addEventListener('close', (closeEvent) => { assert.ok(closeEvent.wasClean); assert.strictEqual(closeEvent.reason, ''); assert.strictEqual(closeEvent.code, 1000); wss.close(done); }); }); wss.on('connection', (ws) => ws.close(1000)); }); it('receives a `CloseEvent` when server closes (4000)', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.addEventListener('close', (closeEvent) => { assert.ok(closeEvent.wasClean); assert.strictEqual(closeEvent.reason, 'some daft reason'); assert.strictEqual(closeEvent.code, 4000); wss.close(done); }); }); wss.on('connection', (ws) => ws.close(4000, 'some daft reason')); }); it('sets `target` and `type` on events', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const err = new Error('forced'); const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.addEventListener('open', (openEvent) => { assert.strictEqual(openEvent.type, 'open'); assert.strictEqual(openEvent.target, ws); }); ws.addEventListener('message', (messageEvent) => { assert.strictEqual(messageEvent.type, 'message'); assert.strictEqual(messageEvent.target, ws); wss.close(); }); ws.addEventListener('close', (closeEvent) => { assert.strictEqual(closeEvent.type, 'close'); assert.strictEqual(closeEvent.target, ws); ws.emit('error', err); }); ws.addEventListener('error', (errorEvent) => { assert.strictEqual(errorEvent.message, 'forced'); assert.strictEqual(errorEvent.type, 'error'); assert.strictEqual(errorEvent.target, ws); assert.strictEqual(errorEvent.error, err); done(); }); }); wss.on('connection', (client) => client.send('hi')); }); it('passes binary data as a Node.js `Buffer` by default', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.onmessage = (evt) => { assert.ok(Buffer.isBuffer(evt.data)); wss.close(done); }; }); wss.on('connection', (ws) => ws.send(new Uint8Array(4096))); }); it('ignores `binaryType` for text messages', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); ws.binaryType = 'arraybuffer'; ws.onmessage = (evt) => { assert.strictEqual(evt.data, 'foo'); wss.close(done); }; }); wss.on('connection', (ws) => ws.send('foo')); }); it('allows to update `binaryType` on the fly', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); function testType(binaryType, next) { const buf = Buffer.from(binaryType); ws.binaryType = binaryType; ws.onmessage = (evt) => { if (binaryType === 'nodebuffer') { assert.ok(Buffer.isBuffer(evt.data)); assert.ok(evt.data.equals(buf)); } else if (binaryType === 'arraybuffer') { assert.ok(evt.data instanceof ArrayBuffer); assert.ok(Buffer.from(evt.data).equals(buf)); } else if (binaryType === 'fragments') { assert.deepStrictEqual(evt.data, [buf]); } next(); }; ws.send(buf); } ws.onopen = () => { testType('nodebuffer', () => { testType('arraybuffer', () => { testType('fragments', () => wss.close(done)); }); }); }; }); wss.on('connection', (ws) => { ws.on('message', (msg) => ws.send(msg)); }); }); }); describe('SSL', () => { it('connects to secure websocket server', (done) => { const server = https.createServer({ cert: fs.readFileSync('test/fixtures/certificate.pem'), key: fs.readFileSync('test/fixtures/key.pem') }); const wss = new WebSocket.Server({ server }); wss.on('connection', () => { wss.close(); server.close(done); }); server.listen(0, () => { const ws = new WebSocket(`wss://localhost:${server.address().port}`, { rejectUnauthorized: false }); }); }); it('connects to secure websocket server with client side certificate', (done) => { const server = https.createServer({ cert: fs.readFileSync('test/fixtures/certificate.pem'), ca: [fs.readFileSync('test/fixtures/ca1-cert.pem')], key: fs.readFileSync('test/fixtures/key.pem'), requestCert: true }); let success = false; const wss = new WebSocket.Server({ verifyClient: (info) => { success = !!info.req.client.authorized; return true; }, server }); wss.on('connection', () => { assert.ok(success); server.close(done); wss.close(); }); server.listen(0, () => { const ws = new WebSocket(`wss://localhost:${server.address().port}`, { cert: fs.readFileSync('test/fixtures/agent1-cert.pem'), key: fs.readFileSync('test/fixtures/agent1-key.pem'), rejectUnauthorized: false }); }); }); it('cannot connect to secure websocket server via ws://', (done) => { const server = https.createServer({ cert: fs.readFileSync('test/fixtures/certificate.pem'), key: fs.readFileSync('test/fixtures/key.pem') }); const wss = new WebSocket.Server({ server }); server.listen(0, () => { const ws = new WebSocket(`ws://localhost:${server.address().port}`, { rejectUnauthorized: false }); ws.on('error', () => { server.close(done); wss.close(); }); }); }); it('can send and receive text data', (done) => { const server = https.createServer({ cert: fs.readFileSync('test/fixtures/certificate.pem'), key: fs.readFileSync('test/fixtures/key.pem') }); const wss = new WebSocket.Server({ server }); wss.on('connection', (ws) => { ws.on('message', (message) => { assert.strictEqual(message, 'foobar'); server.close(done); wss.close(); }); }); server.listen(0, () => { const ws = new WebSocket(`wss://localhost:${server.address().port}`, { rejectUnauthorized: false }); ws.on('open', () => ws.send('foobar')); }); }); it('can send a big binary message', (done) => { const buf = crypto.randomBytes(5 * 1024 * 1024); const server = https.createServer({ cert: fs.readFileSync('test/fixtures/certificate.pem'), key: fs.readFileSync('test/fixtures/key.pem') }); const wss = new WebSocket.Server({ server }); wss.on('connection', (ws) => { ws.on('message', (message) => ws.send(message)); }); server.listen(0, () => { const ws = new WebSocket(`wss://localhost:${server.address().port}`, { rejectUnauthorized: false }); ws.on('open', () => ws.send(buf)); ws.on('message', (message) => { assert.ok(buf.equals(message)); server.close(done); wss.close(); }); }); }).timeout(4000); it('allows to disable sending the SNI extension', (done) => { const original = tls.connect; tls.connect = (options) => { assert.strictEqual(options.servername, ''); tls.connect = original; done(); }; const ws = new WebSocket('wss://127.0.0.1', { servername: '' }); }); }); describe('Request headers', () => { it('adds the authorization header if the url has userinfo', (done) => { const agent = new CustomAgent(); const auth = 'test:testpass'; agent.addRequest = (req) => { assert.strictEqual( req.getHeader('authorization'), `Basic ${Buffer.from(auth).toString('base64')}` ); done(); }; const ws = new WebSocket(`ws://${auth}@localhost`, { agent }); }); it('adds custom headers', (done) => { const agent = new CustomAgent(); agent.addRequest = (req) => { assert.strictEqual(req.getHeader('cookie'), 'foo=bar'); done(); }; const ws = new WebSocket('ws://localhost', { headers: { Cookie: 'foo=bar' }, agent }); }); it('excludes default ports from host header', () => { const options = { lookup() {} }; const variants = [ ['wss://localhost:8443', 'localhost:8443'], ['wss://localhost:443', 'localhost'], ['ws://localhost:88', 'localhost:88'], ['ws://localhost:80', 'localhost'] ]; for (const [url, host] of variants) { const ws = new WebSocket(url, options); assert.strictEqual(ws._req.getHeader('host'), host); } }); it("doesn't add the origin header by default", (done) => { const agent = new CustomAgent(); agent.addRequest = (req) => { assert.strictEqual(req.getHeader('origin'), undefined); done(); }; const ws = new WebSocket('ws://localhost', { agent }); }); it('honors the `origin` option (1/2)', (done) => { const agent = new CustomAgent(); agent.addRequest = (req) => { assert.strictEqual(req.getHeader('origin'), 'https://example.com:8000'); done(); }; const ws = new WebSocket('ws://localhost', { origin: 'https://example.com:8000', agent }); }); it('honors the `origin` option (2/2)', (done) => { const agent = new CustomAgent(); agent.addRequest = (req) => { assert.strictEqual( req.getHeader('sec-websocket-origin'), 'https://example.com:8000' ); done(); }; const ws = new WebSocket('ws://localhost', { origin: 'https://example.com:8000', protocolVersion: 8, agent }); }); }); describe('permessage-deflate', () => { it('is enabled by default', (done) => { const agent = new CustomAgent(); agent.addRequest = (req) => { assert.strictEqual( req.getHeader('sec-websocket-extensions'), 'permessage-deflate; client_max_window_bits' ); done(); }; const ws = new WebSocket('ws://localhost', { agent }); }); it('can be disabled', (done) => { const agent = new CustomAgent(); agent.addRequest = (req) => { assert.strictEqual( req.getHeader('sec-websocket-extensions'), undefined ); done(); }; const ws = new WebSocket('ws://localhost', { perMessageDeflate: false, agent }); }); it('can send extension parameters', (done) => { const agent = new CustomAgent(); const value = 'permessage-deflate; server_no_context_takeover;' + ' client_no_context_takeover; server_max_window_bits=10;' + ' client_max_window_bits'; agent.addRequest = (req) => { assert.strictEqual(req.getHeader('sec-websocket-extensions'), value); done(); }; const ws = new WebSocket('ws://localhost', { perMessageDeflate: { clientNoContextTakeover: true, serverNoContextTakeover: true, clientMaxWindowBits: true, serverMaxWindowBits: 10 }, agent }); }); it('can send and receive text data', (done) => { const wss = new WebSocket.Server( { perMessageDeflate: { threshold: 0 }, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`, { perMessageDeflate: { threshold: 0 } }); ws.on('open', () => ws.send('hi', { compress: true })); ws.on('message', (message) => { assert.strictEqual(message, 'hi'); wss.close(done); }); } ); wss.on('connection', (ws) => { ws.on('message', (message) => ws.send(message, { compress: true })); }); }); it('can send and receive a `TypedArray`', (done) => { const array = new Float32Array(5); for (let i = 0; i < array.length; i++) { array[i] = i / 2; } const wss = new WebSocket.Server( { perMessageDeflate: { threshold: 0 }, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`, { perMessageDeflate: { threshold: 0 } }); ws.on('open', () => ws.send(array, { compress: true })); ws.on('message', (message) => { assert.ok(message.equals(Buffer.from(array.buffer))); wss.close(done); }); } ); wss.on('connection', (ws) => { ws.on('message', (message) => ws.send(message, { compress: true })); }); }); it('can send and receive an `ArrayBuffer`', (done) => { const array = new Float32Array(5); for (let i = 0; i < array.length; i++) { array[i] = i / 2; } const wss = new WebSocket.Server( { perMessageDeflate: { threshold: 0 }, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`, { perMessageDeflate: { threshold: 0 } }); ws.on('open', () => ws.send(array.buffer, { compress: true })); ws.on('message', (message) => { assert.ok(message.equals(Buffer.from(array.buffer))); wss.close(done); }); } ); wss.on('connection', (ws) => { ws.on('message', (message) => ws.send(message, { compress: true })); }); }); it('consumes all received data when connection is closed abnormally', (done) => { const wss = new WebSocket.Server( { perMessageDeflate: { threshold: 0 }, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const messages = []; ws.on('message', (message) => messages.push(message)); ws.on('close', (code) => { assert.strictEqual(code, 1006); assert.deepStrictEqual(messages, ['foo', 'bar', 'baz', 'qux']); wss.close(done); }); } ); wss.on('connection', (ws) => { ws.send('foo'); ws.send('bar'); ws.send('baz'); ws.send('qux', () => ws._socket.end()); }); }); describe('#send', () => { it('ignores the `compress` option if the extension is disabled', (done) => { const wss = new WebSocket.Server({ port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`, { perMessageDeflate: false }); ws.on('open', () => ws.send('hi', { compress: true })); ws.on('message', (message) => { assert.strictEqual(message, 'hi'); wss.close(done); }); }); wss.on('connection', (ws) => { ws.on('message', (message) => ws.send(message, { compress: true })); }); }); }); describe('#terminate', () => { it('can be used while data is being compressed', (done) => { const wss = new WebSocket.Server( { perMessageDeflate: { threshold: 0 }, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`, { perMessageDeflate: { threshold: 0 } }); ws.on('open', () => { ws.send('hi', () => done(new Error('Unexpected callback invocation')) ); ws.terminate(); }); } ); wss.on('connection', (ws) => { ws.on('close', () => { wss.close(done); }); }); }); it('can be used while data is being decompressed', (done) => { const wss = new WebSocket.Server( { perMessageDeflate: true, port: 0 }, () => { const ws = new WebSocket(`ws://localhost:${wss.address().port}`); const messages = []; ws.on('message', (message) => { if (messages.push(message) > 1) return; process.nextTick(() => { assert.strictEqual(ws._receiver._state, 5); ws.terminate(); }); }); ws.on('close', (code, reason) => { assert.deepStrictEqual(messages, ['', '', '', '']); assert.strictEqual(code, 1006); assert.strictEqual(reason, ''); wss.close(done); }); } ); wss.on('connection', (ws) => { const buf = Buffer.from('c10100c10100c10100c10100', 'hex'); ws._socket.write(buf); }); }); }); }); });