Important: This documentation covers Yarn 1 (Classic).
For Yarn 2+ docs and migration guide, see yarnpkg.com.

Package detail

websocket

theturtle323.4mApache-2.01.0.35TypeScript support: definitely-typed

Websocket Client & Server Library implementing the WebSocket protocol as specified in RFC 6455.

websocket, websockets, socket, networking, comet, push, RFC-6455, realtime, server, client

readme

WebSocket Client & Server Implementation for Node

npm version

NPM Downloads

Codeship Status for theturtle32/WebSocket-Node

Overview

This is a (mostly) pure JavaScript implementation of the WebSocket protocol versions 8 and 13 for Node. There are some example client and server applications that implement various interoperability testing protocols in the "test/scripts" folder.

Documentation

You can read the full API documentation in the docs folder.

Changelog

Current Version: 1.0.35 - Release 2024-05-12

View the full changelog

Browser Support

All current browsers are fully* supported.

  • Firefox 7-9 (Old) (Protocol Version 8)
  • Firefox 10+ (Protocol Version 13)
  • Chrome 14,15 (Old) (Protocol Version 8)
  • Chrome 16+ (Protocol Version 13)
  • Internet Explorer 10+ (Protocol Version 13)
  • Safari 6+ (Protocol Version 13)

(Not all W3C WebSocket features are supported by browsers. More info in the Full API documentation)

Benchmarks

There are some basic benchmarking sections in the Autobahn test suite. I've put up a benchmark page that shows the results from the Autobahn tests run against AutobahnServer 0.4.10, WebSocket-Node 1.0.2, WebSocket-Node 1.0.4, and ws 0.3.4.

(These benchmarks are quite a bit outdated at this point, so take them with a grain of salt. Anyone up for running new benchmarks? I'll link to your report.)

Autobahn Tests

The very complete Autobahn Test Suite is used by most WebSocket implementations to test spec compliance and interoperability.

Installation

In your project root:

$ npm install websocket

Then in your code:

var WebSocketServer = require('websocket').server;
var WebSocketClient = require('websocket').client;
var WebSocketFrame  = require('websocket').frame;
var WebSocketRouter = require('websocket').router;
var W3CWebSocket = require('websocket').w3cwebsocket;

Current Features:

  • Licensed under the Apache License, Version 2.0
  • Protocol version "8" and "13" (Draft-08 through the final RFC) framing and handshake
  • Can handle/aggregate received fragmented messages
  • Can fragment outgoing messages
  • Router to mount multiple applications to various path and protocol combinations
  • TLS supported for outbound connections via WebSocketClient
  • TLS supported for server connections (use https.createServer instead of http.createServer)
    • Thanks to pors for confirming this!
  • Cookie setting and parsing
  • Tunable settings
    • Max Receivable Frame Size
    • Max Aggregate ReceivedMessage Size
    • Whether to fragment outgoing messages
    • Fragmentation chunk size for outgoing messages
    • Whether to automatically send ping frames for the purposes of keepalive
    • Keep-alive ping interval
    • Whether or not to automatically assemble received fragments (allows application to handle individual fragments directly)
    • How long to wait after sending a close frame for acknowledgment before closing the socket.
  • W3C WebSocket API for applications running on both Node and browsers (via the W3CWebSocket class).

Known Issues/Missing Features:

  • No API for user-provided protocol extensions.

Usage Examples

Server Example

Here's a short example showing a server that echos back anything sent to it, whether utf-8 or binary.

#!/usr/bin/env node
var WebSocketServer = require('websocket').server;
var http = require('http');

var server = http.createServer(function(request, response) {
    console.log((new Date()) + ' Received request for ' + request.url);
    response.writeHead(404);
    response.end();
});
server.listen(8080, function() {
    console.log((new Date()) + ' Server is listening on port 8080');
});

wsServer = new WebSocketServer({
    httpServer: server,
    // You should not use autoAcceptConnections for production
    // applications, as it defeats all standard cross-origin protection
    // facilities built into the protocol and the browser.  You should
    // *always* verify the connection's origin and decide whether or not
    // to accept it.
    autoAcceptConnections: false
});

function originIsAllowed(origin) {
  // put logic here to detect whether the specified origin is allowed.
  return true;
}

wsServer.on('request', function(request) {
    if (!originIsAllowed(request.origin)) {
      // Make sure we only accept requests from an allowed origin
      request.reject();
      console.log((new Date()) + ' Connection from origin ' + request.origin + ' rejected.');
      return;
    }

    var connection = request.accept('echo-protocol', request.origin);
    console.log((new Date()) + ' Connection accepted.');
    connection.on('message', function(message) {
        if (message.type === 'utf8') {
            console.log('Received Message: ' + message.utf8Data);
            connection.sendUTF(message.utf8Data);
        }
        else if (message.type === 'binary') {
            console.log('Received Binary Message of ' + message.binaryData.length + ' bytes');
            connection.sendBytes(message.binaryData);
        }
    });
    connection.on('close', function(reasonCode, description) {
        console.log((new Date()) + ' Peer ' + connection.remoteAddress + ' disconnected.');
    });
});

Client Example

This is a simple example client that will print out any utf-8 messages it receives on the console, and periodically sends a random number.

This code demonstrates a client in Node.js, not in the browser

#!/usr/bin/env node
var WebSocketClient = require('websocket').client;

var client = new WebSocketClient();

client.on('connectFailed', function(error) {
    console.log('Connect Error: ' + error.toString());
});

client.on('connect', function(connection) {
    console.log('WebSocket Client Connected');
    connection.on('error', function(error) {
        console.log("Connection Error: " + error.toString());
    });
    connection.on('close', function() {
        console.log('echo-protocol Connection Closed');
    });
    connection.on('message', function(message) {
        if (message.type === 'utf8') {
            console.log("Received: '" + message.utf8Data + "'");
        }
    });

    function sendNumber() {
        if (connection.connected) {
            var number = Math.round(Math.random() * 0xFFFFFF);
            connection.sendUTF(number.toString());
            setTimeout(sendNumber, 1000);
        }
    }
    sendNumber();
});

client.connect('ws://localhost:8080/', 'echo-protocol');

Client Example using the W3C WebSocket API

Same example as above but using the W3C WebSocket API.

var W3CWebSocket = require('websocket').w3cwebsocket;

var client = new W3CWebSocket('ws://localhost:8080/', 'echo-protocol');

client.onerror = function() {
    console.log('Connection Error');
};

client.onopen = function() {
    console.log('WebSocket Client Connected');

    function sendNumber() {
        if (client.readyState === client.OPEN) {
            var number = Math.round(Math.random() * 0xFFFFFF);
            client.send(number.toString());
            setTimeout(sendNumber, 1000);
        }
    }
    sendNumber();
};

client.onclose = function() {
    console.log('echo-protocol Client Closed');
};

client.onmessage = function(e) {
    if (typeof e.data === 'string') {
        console.log("Received: '" + e.data + "'");
    }
};

Request Router Example

For an example of using the request router, see libwebsockets-test-server.js in the test folder.

Resources

A presentation on the state of the WebSockets protocol that I gave on July 23, 2011 at the LA Hacker News meetup. WebSockets: The Real-Time Web, Delivered

changelog

Changelog

Version 1.0.35

Released 2024-05-12

Version 1.0.34

Released 2021-04-14

  • Updated browser shim to use the native globalThis property when available. See this MDN page for context. Resolves #415

Version 1.0.33

Released 2020-12-08

  • Added new configuration options to WebSocketServer allowing implementors to bypass parsing WebSocket extensions and HTTP Cookies if they are not needed. (Thanks, @aetheon)
  • Added new upgradeError event to WebSocketServer to allow for visibility into and logging of any parsing errors that might occur during the HTTP Upgrade phase. (Thanks, @aetheon)

Version 1.0.32

Released 2020-08-28

  • Refactor to use N-API modules from ws project. (Thanks, @andreek)
  • Removed some documentation notations about very old browsers and very old Websocket protocol drafts that are no longer relevant today in 2020.
  • Removed outdated notations and instructions about building native extensions, since those functions are now delegated to dependencies.
  • Add automated unit test executionn via Github Actions (Thanks, @nebojsa94)
  • Accept new connection close code 1015 ("TLS Handshake"). (More information at the WebSocket Close Code Number Registry)

Version 1.0.31

Released 2019-12-06

Version 1.0.30

Released 2019-09-12

  • Moved gulp back to devDependencies

Version 1.0.29

Released 2019-07-03

  • Updated some dependencies and updated the .gitignore and .npmignore files

Version 1.0.28

Released 2018-09-19

  • Updated to latest version of nan

Version 1.0.27

Released 2018-09-19

  • Allowing additional request headers to be specified in the tlsOptions config parameter for WebSocketClient. See pull request #323
  • Resolving deprecation warnings relating to usage of new Buffer

Version 1.0.26

Released 2018-04-27

  • No longer using the deprecated noAssert parameter for functions reading and writing binary numeric data. (Thanks, @BridgeAR)

Version 1.0.25

Released 2017-10-18

Version 1.0.24

Released 2016-12-28

  • Fixed a bug when using native keepalive on Node >= 6.0. (Thanks, @prossin)
  • Upgrading outdated dependencies

Version 1.0.23

Released 2016-05-18

  • Official support for Node 6.x
  • Updating dependencies. Specifically, updating nan to ^2.3.3

Version 1.0.22

Released 2015-09-28

  • Updating to work with nan 2.x

Version 1.0.21

Released 2015-07-22

  • Incremented and re-published to work around an aborted npm publish of v1.0.20.

Version 1.0.20

Released 2015-07-22

  • Added EventTarget to the W3CWebSocket interface (Thanks, @ibc!)
  • Corrected an inaccurate error message. (Thanks, @lekoaf!)

Version 1.0.19

Released 2015-05-28

Version 1.0.18

Released 2015-03-19

Version 1.0.17

Released 2015-01-17

  • Resolves issue #179 - Allow toBuffer to work with empty data

Version 1.0.16

Released 2015-01-16

Version 1.0.15

Released 2015-01-13

  • Resolves issue #177 - WebSocketClient ignores options unless it has a tlsOptions property

Version 1.0.14

Released 2014-12-03

  • Resolves issue #173 - To allow the W3CWebSocket interface to accept an optional non-standard configuration object as its third parameter, which will be ignored when running in a browser context.

Version 1.0.13

Released 2014-11-29

  • Fixes issue #171 - Code to prevent calling req.accept/req.reject multiple times breaks sanity checks in req.accept

Version 1.0.12

Released 2014-11-28

  • Fixes issue #170 - Non-native XOR implementation broken after making JSHint happy

Version 1.0.11

Released 2014-11-25

  • Fixes some undefined behavior surrounding closing WebSocket connections and more reliably handles edge cases.
  • Adds an implementation of the W3C WebSocket API for browsers to facilitate sharing code between client and server via browserify. (Thanks, @ibc!)
  • WebSocketConnection.prototype.close now accepts optional reasonCode and description parameters.
  • Calling accept or reject more than once on a WebSocketRequest will now throw an error. Issue #149
  • Handling connections dropped by client before accepted by server Issue #167
  • Integrating Gulp and JSHint (Thanks, @ibc!)
  • Starting to add individual unit tests (using substack's tape and faucet)

Version 1.0.10

Released 2014-10-22

  • Fixed Issue #146 that was causing WebSocketClient to throw errors when instantiated if passed tlsOptions.

Version 1.0.9

Released 2014-10-20

  • Fixing an insidious corner-case bug that prevented WebSocketConnection from firing the close event in certain cases when there was an error on the underlying Socket, leading to connections sticking around forever, stuck erroneously in the connected state. These "ghost" connections would cause an error event when trying to write to them.
  • Removed deprecated websocketVersion property. Use webSocketVersion instead (case difference).
  • Allowing user to specify all properties for tlsOptions in WebSocketClient, not just a few whitelisted properties. This keeps us from having to constantly add new config properties for new versions of Node. (Thanks, jesusprubio)
  • Removing support for Node 0.4.x and 0.6.x.
  • Adding fuzzingclient.json spec file for the Autobahn Test Suite.
  • Now more fairly emitting message events from the WebSocketConnection. Previously, all buffered frames for a connection would be processed and all message events emitted before moving on to processing the next connection with available data. Now We process one frame per connection (most of the time) in a more fair round-robin fashion.
  • Now correctly calling the EventEmitter superclass constructor during class instance initialization.
  • WebSocketClient.prototype.connect now accepts the empty string ('') to mean "no subprotocol requested." Previously either null or an empty array ([]) was required.
  • Fixing a TypeError bug in WebSocketRouter (Thanks, a0000778)
  • Fixing a potential race condition when attaching event listeners to the underlying Socket. (Thanks RichardBsolut)
  • WebSocketClient now accepts an optional options hash to be passed to (http|https).request. (Thanks mildred and aus) This enables the following new abilities, amongst others:
    • Use WebSocket-Node from behind HTTP/HTTPS proxy servers using koichik/node-tunnel or similar.
    • Specify the local port and local address to bind the outgoing request socket to.
  • Adding option to ignore X-Forwarded-For headers when accepting connections from untrusted clients.
  • Adding ability to mount a WebSocketServer instance to an arbitrary number of Node http/https servers.
  • Adding browser shim so Browserify won't blow up when trying to package up code that uses WebSocket-Node. The shim is a no-op, it does not implement a wrapper providing the WebSocket-Node API in the browser.
  • Incorporating upstream enhancements for the native C++ UTF-8 validation and xor masking functions. (Thanks einaros and kkoopa)

Version 1.0.8

Released 2012-12-26

  • Fixed remaining naming inconsistency of "websocketVersion" as opposed to "webSocketVersion" throughout the code, and added deprecation warnings for use of the old casing throughout.
  • Fixed an issue with our case-insensitive handling of WebSocket subprotocols. Clients that requested a mixed-case subprotocol would end up failing the connection when the server accepted the connection, returning a lower-case version of the subprotocol name. Now we return the subprotocol name in the exact casing that was requested by the client, while still maintaining the case-insensitive verification logic for convenience and practicality.
  • Making sure that any socket-level activity timeout that may have been set on a TCP socket is removed when initializing a connection.
  • Added support for native TCP Keep-Alive instead of using the WebSocket ping/pong packets to serve that function.
  • Fixed cookie parsing to be compliant with RFC 2109

Version 1.0.7

Released 2012-08-12

  • Native modules are now optional! If they fail to compile, WebSocket-Node will still work but will not verify that received UTF-8 data is valid, and xor masking/unmasking of payload data for security purposes will not be as efficient as it is performed in JavaScript instead of native code.
  • Reduced Node.JS version requirement back to v0.6.10

Version 1.0.6

Released 2012-05-22

  • Now requires Node v0.6.13 since that's the first version that I can manage to successfully build the native UTF-8 validator with node-gyp through npm.

Version 1.0.5

Released 2012-05-21

  • Fixes the issues that users were having building the native UTF-8 validator on Windows platforms. Special Thanks to:
  • Fixed accidental global variable usage (Thanks, hakobera!)
  • Added callbacks to the send* methods that provide notification of messages being sent on the wire and any socket errors that may occur when sending a message. (Thanks, zerodivisi0n!)
  • Added option to disable logging in the echo-server in the test folder (Thanks, oberstet!)

Version 1.0.4

Released 2011-12-18

  • Now validates that incoming UTF-8 messages do, in fact, contain valid UTF-8 data. The connection is dropped with prejudice if invalid data is received. This strict behavior conforms to the WebSocket RFC and is verified by the Autobahn Test Suite. This is accomplished in a performant way by using a native C++ Node module created by einaros.
  • Updated handling of connection closure to pass more of the Autobahn Test Suite.

Version 1.0.3

Released 2011-12-18

  • Substantial speed increase (~150% on my machine, depending on the circumstances) due to an optimization in FastBufferList.js that drastically reduces the number of memory alloctions and buffer copying. (kazuyukitanimura)

Version 1.0.2

Released 2011-11-28

  • Fixing whiteboard example to work under Node 0.6.x (theturtle32)
  • Now correctly emitting a close event with a 1006 error code if there is a TCP error while writing to the socket during the handshake. (theturtle32)
  • Catching errors when writing to the TCP socket during the handshake. (justoneplanet)
  • No longer outputting console.warn messages when there is an error writing to the TCP socket (justoneplanet)
  • Fixing some formatting errors, commas, semicolons, etc. (kaisellgren)

Version 1.0.1

Released 2011-11-21

  • Now works with Node 0.6.2 as well as 0.4.12
  • Support TLS in WebSocketClient
  • Added support for setting and reading cookies
  • Added WebSocketServer.prototype.broadcast(data) convenience method
  • Added resourceURL property to WebSocketRequest objects. It is a Node URL object with the resource and any query string params already parsed.
  • The WebSocket request router no longer includes the entire query string when trying to match the path name of the request.
  • WebSocketRouterRequest objects now include all the properties and events of WebSocketRequest objects.
  • Removed more console.log statements. Please rely on the various events emitted to be notified of error conditions. I decided that it is not a library's place to spew information to the console.
  • Renamed the websocketVersion property to webSocketVersion throughout the code to fix inconsistent capitalization. websocketVersion has been kept for compatibility but is deprecated and may be removed in the future.
  • Now outputting the sanitized version of custom header names rather than the raw value. This prevents invalid HTTP from being put onto the wire if given an illegal header name.

I decided it's time to start maintaining a changelog now, starting with version 1.0.1.