socket.io, uws, websocket, and ws are all npm packages that enable real-time, bidirectional communication between clients and servers using the WebSocket protocol or compatible fallbacks. socket.io provides a high-level, feature-rich abstraction with automatic reconnection, rooms, and fallback to HTTP long-polling. ws is a lightweight, standards-compliant WebSocket implementation focused on performance and simplicity. The websocket package offers a lower-level, spec-aligned API with support for both client and server roles in Node.js. uws was an ultra-fast WebSocket implementation but has been officially deprecated and should not be used in new projects.
When building real-time web applications — like live chat, collaborative editors, or financial tickers — choosing the right WebSocket library is critical. The four packages under review (socket.io, uws, websocket, and ws) each take a different approach to enabling bidirectional communication. Let’s break down their technical trade-offs with real code examples.
uws Is No Longer ViableBefore diving into comparisons, note that uws is officially deprecated. According to its npm page and GitHub repository, the project has been archived, and the author states: “This project is no longer maintained. Please use the official uWebSockets.js bindings or switch to ws.” Despite past claims of extreme speed, uws should not be used in any new project due to lack of security updates, Node.js version compatibility, and active maintenance.
// ❌ Do NOT use uws in new code
// const uWS = require('uws'); // Deprecated — avoid
With that out of the way, let’s compare the three viable options.
socket.io: Adds a Layer on Top of WebSocketssocket.io doesn’t just use WebSockets — it wraps them in its own protocol that includes acknowledgments, namespaces, and event names. It also falls back to HTTP long-polling if WebSockets aren’t available.
// socket.io server
const io = require('socket.io')(3000);
io.on('connection', (socket) => {
socket.join('room1');
socket.emit('welcome', { msg: 'Hello!' });
socket.on('chat', (data) => {
io.to('room1').emit('message', data);
});
});
// socket.io client (browser)
const socket = io('http://localhost:3000');
socket.on('welcome', (data) => console.log(data.msg));
socket.emit('chat', { text: 'Hi everyone!' });
This abstraction simplifies common patterns but adds overhead: every message includes metadata like event names and packet IDs, increasing bandwidth usage.
ws: Pure WebSocket, Minimal Overheadws implements the WebSocket standard directly with no extra framing. It’s ideal when you control both client and server and want maximum efficiency.
// ws server
const WebSocket = require('ws');
const wss = new WebSocket.Server({ port: 8080 });
wss.on('connection', (ws) => {
ws.send(JSON.stringify({ type: 'welcome' }));
ws.on('message', (data) => {
const msg = JSON.parse(data);
// Broadcast to all clients
wss.clients.forEach(client => {
if (client.readyState === WebSocket.OPEN) {
client.send(data);
}
});
});
});
// ws client (Node.js or browser with wrapper)
const ws = new WebSocket('ws://localhost:8080');
ws.onopen = () => ws.send(JSON.stringify({ text: 'Hello' }));
ws.onmessage = (event) => console.log(JSON.parse(event.data));
Note: In browsers, you’d use the native WebSocket object; ws is primarily a server-side library (though it can be used in Node.js clients).
websocket: Low-Level Control with Full Spec ComplianceThe websocket package gives you fine-grained access to the WebSocket handshake and frame parsing. It’s more verbose but useful when you need to inspect or modify protocol details.
// websocket server
const WebSocketServer = require('websocket').server;
const http = require('http');
const server = http.createServer();
server.listen(8081);
const wsServer = new WebSocketServer({ httpServer: server });
wsServer.on('request', (request) => {
const connection = request.accept(null, request.origin);
connection.sendUTF(JSON.stringify({ type: 'welcome' }));
connection.on('message', (message) => {
if (message.type === 'utf8') {
const data = JSON.parse(message.utf8Data);
// Echo to sender
connection.sendUTF(message.utf8Data);
}
});
});
// websocket client (Node.js)
const WebSocketClient = require('websocket').client;
const client = new WebSocketClient();
client.on('connect', (connection) => {
connection.sendUTF(JSON.stringify({ text: 'Hi' }));
connection.on('message', (msg) => {
if (msg.type === 'utf8') console.log(JSON.parse(msg.utf8Data));
});
});
client.connect('ws://localhost:8081/');
This level of control is rarely needed in typical apps but can be essential for protocol testing or integration with non-standard clients.
socket.io automatically handles reconnection, heartbeat (ping/pong), and disconnection detection. You get this out of the box.// socket.io auto-reconnect is default behavior
const socket = io({
reconnection: true,
reconnectionAttempts: Infinity
});
ws requires manual ping/pong handling if you want to detect dead connections:// ws: manual heartbeat
wss.on('connection', (ws) => {
const interval = setInterval(() => {
if (ws.readyState === WebSocket.OPEN) {
ws.ping();
}
}, 30000);
ws.on('close', () => clearInterval(interval));
});
websocket also leaves heartbeat logic to you, though it exposes raw ping/pong events.| Feature | socket.io | ws | websocket |
|---|---|---|---|
| Automatic reconnection | ✅ | ❌ | ❌ |
| Rooms/broadcasting | ✅ | ❌* | ❌ |
| Fallback to HTTP | ✅ | ❌ | ❌ |
| Native browser support | ✅ (via client lib) | ✅ (native WebSocket) | ❌ (Node-only client) |
| Message acknowledgment | ✅ | ❌ | ❌ |
| Compression (permessage-deflate) | ✅ | ✅ | ✅ |
* Broadcasting in ws requires manual iteration over wss.clients.
socket.io if you must support very old browsers (e.g., IE9) that lack WebSocket support — it gracefully degrades to long-polling.ws or websocket only if you can assume modern browser support (WebSocket has been widely supported since ~2012).socket.io emits clear events like 'connect_error', 'disconnect', and 'reconnect_failed'.ws uses standard 'error' and 'close' events but gives less context by default.websocket provides detailed error codes per RFC 6455 (e.g., 1002 for protocol error).socket.io when:ws when:websocket when:uws — it’s deprecated.For most real-world applications, ws is the sweet spot: fast, well-maintained, and close to the metal without unnecessary complexity. Reach for socket.io only when you truly need its higher-level features and are willing to accept the protocol overhead. Avoid uws entirely, and reserve websocket for niche, low-level use cases.
Remember: real-time doesn’t have to mean complicated. Sometimes, the simplest WebSocket implementation is the most powerful.
Choose ws if you want a fast, lightweight, and well-maintained WebSocket implementation that follows the standard closely while offering a clean, modern API. It’s the de facto choice for most real-time Node.js applications that don’t need Socket.IO’s extra features. It supports native compression, backpressure handling, and integrates smoothly with Express and other frameworks.
Choose socket.io if you need built-in features like automatic reconnection, room management, broadcasting, and fallback to HTTP long-polling for older browsers. It’s ideal for applications requiring rapid development of real-time features (e.g., chat apps, live dashboards) where developer productivity outweighs the need for minimal overhead. However, be aware that its custom protocol adds latency and bandwidth compared to raw WebSockets.
Choose the websocket package if you need strict adherence to the WebSocket RFC 6455 specification and require both client and server implementations in a single library. It’s useful in environments where you must control low-level handshake details or integrate with legacy systems expecting precise protocol compliance. However, it lacks higher-level conveniences like automatic reconnection or message buffering, requiring more boilerplate code.
Do not choose uws for new projects — it is officially deprecated and unmaintained. The author archived the repository and recommends using ws instead. While it once offered extreme performance, lack of updates, security patches, and compatibility with modern Node.js versions makes it unsuitable for production use today.
ws is a simple to use, blazing fast, and thoroughly tested WebSocket client and server implementation.
Passes the quite extensive Autobahn test suite: server, client.
Note: This module does not work in the browser. The client in the docs is a
reference to a backend with the role of a client in the WebSocket communication.
Browser clients must use the native
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.
protocolVersion: 8)protocolVersion: 13)npm install ws
bufferutil is an optional module that can be installed alongside the ws module:
npm install --save-optional bufferutil
This is a binary addon that improves the performance of certain operations such as masking and unmasking the data payload of the WebSocket frames. Prebuilt binaries are available for the most popular platforms, so you don't necessarily need to have a C++ compiler installed on your machine.
To force ws to not use bufferutil, use the
WS_NO_BUFFER_UTIL environment variable. This
can be useful to enhance security in systems where a user can put a package in
the package search path of an application of another user, due to how the
Node.js resolver algorithm works.
If you are running on an old version of Node.js (prior to v18.14.0), ws also supports the utf-8-validate module:
npm install --save-optional utf-8-validate
This contains a binary polyfill for buffer.isUtf8().
To force ws not to use utf-8-validate, use the
WS_NO_UTF_8_VALIDATE environment variable.
See /doc/ws.md for Node.js-like documentation of ws classes and
utility functions.
ws supports the permessage-deflate extension 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 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.
See the docs for more options.
import WebSocket, { WebSocketServer } from 'ws';
const wss = new WebSocketServer({
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 if context takeover is disabled.
}
});
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.
import WebSocket from 'ws';
const ws = new WebSocket('ws://www.host.com/path', {
perMessageDeflate: false
});
import WebSocket from 'ws';
const ws = new WebSocket('ws://www.host.com/path');
ws.on('error', console.error);
ws.on('open', function open() {
ws.send('something');
});
ws.on('message', function message(data) {
console.log('received: %s', data);
});
import WebSocket from 'ws';
const ws = new WebSocket('ws://www.host.com/path');
ws.on('error', console.error);
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);
});
import { WebSocketServer } from 'ws';
const wss = new WebSocketServer({ port: 8080 });
wss.on('connection', function connection(ws) {
ws.on('error', console.error);
ws.on('message', function message(data) {
console.log('received: %s', data);
});
ws.send('something');
});
import { createServer } from 'https';
import { readFileSync } from 'fs';
import { WebSocketServer } from 'ws';
const server = createServer({
cert: readFileSync('/path/to/cert.pem'),
key: readFileSync('/path/to/key.pem')
});
const wss = new WebSocketServer({ server });
wss.on('connection', function connection(ws) {
ws.on('error', console.error);
ws.on('message', function message(data) {
console.log('received: %s', data);
});
ws.send('something');
});
server.listen(8080);
import { createServer } from 'http';
import { WebSocketServer } from 'ws';
const server = createServer();
const wss1 = new WebSocketServer({ noServer: true });
const wss2 = new WebSocketServer({ noServer: true });
wss1.on('connection', function connection(ws) {
ws.on('error', console.error);
// ...
});
wss2.on('connection', function connection(ws) {
ws.on('error', console.error);
// ...
});
server.on('upgrade', function upgrade(request, socket, head) {
const { pathname } = new URL(request.url, 'wss://base.url');
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);
import { createServer } from 'http';
import { WebSocketServer } from 'ws';
function onSocketError(err) {
console.error(err);
}
const server = createServer();
const wss = new WebSocketServer({ noServer: true });
wss.on('connection', function connection(ws, request, client) {
ws.on('error', console.error);
ws.on('message', function message(data) {
console.log(`Received message ${data} from user ${client}`);
});
});
server.on('upgrade', function upgrade(request, socket, head) {
socket.on('error', onSocketError);
// This function is not defined on purpose. Implement it with your own logic.
authenticate(request, function next(err, client) {
if (err || !client) {
socket.write('HTTP/1.1 401 Unauthorized\r\n\r\n');
socket.destroy();
return;
}
socket.removeListener('error', onSocketError);
wss.handleUpgrade(request, socket, head, function done(ws) {
wss.emit('connection', ws, request, client);
});
});
});
server.listen(8080);
Also see the provided example using express-session.
A client WebSocket broadcasting to all connected WebSocket clients, including itself.
import WebSocket, { WebSocketServer } from 'ws';
const wss = new WebSocketServer({ port: 8080 });
wss.on('connection', function connection(ws) {
ws.on('error', console.error);
ws.on('message', function message(data, isBinary) {
wss.clients.forEach(function each(client) {
if (client.readyState === WebSocket.OPEN) {
client.send(data, { binary: isBinary });
}
});
});
});
A client WebSocket broadcasting to every other connected WebSocket clients, excluding itself.
import WebSocket, { WebSocketServer } from 'ws';
const wss = new WebSocketServer({ port: 8080 });
wss.on('connection', function connection(ws) {
ws.on('error', console.error);
ws.on('message', function message(data, isBinary) {
wss.clients.forEach(function each(client) {
if (client !== ws && client.readyState === WebSocket.OPEN) {
client.send(data, { binary: isBinary });
}
});
});
});
import WebSocket from 'ws';
const ws = new WebSocket('wss://websocket-echo.com/');
ws.on('error', console.error);
ws.on('open', function open() {
console.log('connected');
ws.send(Date.now());
});
ws.on('close', function close() {
console.log('disconnected');
});
ws.on('message', function message(data) {
console.log(`Round-trip time: ${Date.now() - data} ms`);
setTimeout(function timeout() {
ws.send(Date.now());
}, 500);
});
import WebSocket, { createWebSocketStream } from 'ws';
const ws = new WebSocket('wss://websocket-echo.com/');
const duplex = createWebSocketStream(ws, { encoding: 'utf8' });
duplex.on('error', console.error);
duplex.pipe(process.stdout);
process.stdin.pipe(duplex);
For a full example with a browser client communicating with a ws server, see the examples folder.
Otherwise, see the test cases.
The remote IP address can be obtained from the raw socket.
import { WebSocketServer } from 'ws';
const wss = new WebSocketServer({ port: 8080 });
wss.on('connection', function connection(ws, req) {
const ip = req.socket.remoteAddress;
ws.on('error', console.error);
});
When the server runs behind a proxy like NGINX, the de-facto standard is to use
the X-Forwarded-For header.
wss.on('connection', function connection(ws, req) {
const ip = req.headers['x-forwarded-for'].split(',')[0].trim();
ws.on('error', console.error);
});
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.
import { WebSocketServer } from 'ws';
function heartbeat() {
this.isAlive = true;
}
const wss = new WebSocketServer({ port: 8080 });
wss.on('connection', function connection(ws) {
ws.isAlive = true;
ws.on('error', console.error);
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();
});
}, 30000);
wss.on('close', function close() {
clearInterval(interval);
});
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:
import WebSocket from '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://websocket-echo.com/');
client.on('error', console.error);
client.on('open', heartbeat);
client.on('ping', heartbeat);
client.on('close', function clear() {
clearTimeout(this.pingTimeout);
});
Use a custom http.Agent implementation like https-proxy-agent or
socks-proxy-agent.
We're using the GitHub releases for changelog entries.