Testland
Browse all skills & agents

websocket-tests

Test WebSocket protocol behavior - opening handshake (HTTP Upgrade with Sec-WebSocket-Key + Sec-WebSocket-Version: 13), control frames (ping 0x9 / pong 0xA / close 0x8), close-frame status codes (1000 normal, 1001 going-away, 1006 abnormal, 1011 server error), subprotocol negotiation, backpressure, and reconnect with jitter. Use ws (Node), websockets (Python), or Playwright frame inspection per language.

websocket-tests

Per RFC 6455, tests must cover the handshake, control frames, close codes, and subprotocol negotiation - not just message round-trips.

When to use

  • Testing a real-time service: chat, notifications, collaborative editing, live dashboards.
  • Validating reconnection logic after server-side restart or network partition.
  • Pre-deployment gate: the close-code matrix is correct + ping/pong keepalive is wired.

Step 1 - Pick the client lib

StackLibrary
Nodews (npm install ws)
Browser e2ePlaywright page.on('websocket', …) for frame inspection
Pythonwebsockets
Gogithub.com/gorilla/websocket
Javaio.javalin/javalin-testtools or org.glassfish.tyrus/tyrus-client

Step 2 - Handshake assertions

Per RFC 6455, the opening handshake requires:

  • HTTP GET, version 1.1+
  • Upgrade: websocket
  • Connection: Upgrade
  • Sec-WebSocket-Key: base64 16-byte nonce
  • Sec-WebSocket-Version: 13

The server validates by hashing key + "258EAFA5-E914-47DA-95CA-C5AB0DC85B11" (SHA-1, base64) into Sec-WebSocket-Accept and replying with 101 Switching Protocols.

Test that an upgrade is performed (101) and that the accept header matches:

import { WebSocket } from 'ws';
import crypto from 'crypto';

test('handshake completes with correct accept header', async () => {
  const ws = new WebSocket('ws://localhost:8080/');
  await new Promise((res, rej) => {
    ws.once('upgrade', (msg) => {
      expect(msg.statusCode).toBe(101);
      const expectedKey = crypto
        .createHash('sha1')
        .update(ws._req?.getHeader('Sec-WebSocket-Key') + '258EAFA5-E914-47DA-95CA-C5AB0DC85B11')
        .digest('base64');
      expect(msg.headers['sec-websocket-accept']).toBe(expectedKey);
      res(null);
    });
    ws.once('error', rej);
  });
  ws.close();
});

Step 3 - Subprotocol negotiation

Per RFC 6455, the Sec-WebSocket-Protocol header negotiates a subprotocol. Server selects one or none.

test('server picks v2 subprotocol when offered', async () => {
  const ws = new WebSocket('ws://localhost:8080/', ['chat-v1', 'chat-v2']);
  await new Promise((r) => ws.once('open', r));
  expect(ws.protocol).toBe('chat-v2');
  ws.close();
});

test('server rejects unknown subprotocol', async () => {
  const ws = new WebSocket('ws://localhost:8080/', ['unknown-v99']);
  await new Promise((res) => ws.once('close', (code) => { expect(code).toBe(1002); res(null); }));
});

Step 4 - Ping/pong keepalive

Per RFC 6455, control frames include ping (0x9), pong (0xA), close (0x8); payloads ≤ 125 bytes.

test('client receives pong within 5s of ping', async () => {
  const ws = new WebSocket('ws://localhost:8080/');
  await new Promise((r) => ws.once('open', r));

  const pongReceived = new Promise((resolve) => {
    ws.on('pong', () => resolve(true));
  });
  ws.ping('keepalive');

  const got = await Promise.race([
    pongReceived,
    new Promise((r) => setTimeout(() => r(false), 5000)),
  ]);
  expect(got).toBe(true);
  ws.close();
});

Step 5 - Close-frame status code matrix

Per RFC 6455, standard close codes:

CodeMeaning
1000Normal closure
1001Endpoint going away
1002Protocol error
1006Abnormal closure (no close frame received)
1011Server error

Test the matrix per scenario:

test('server sends 1011 on internal error', async () => {
  const ws = new WebSocket('ws://localhost:8080/');
  await new Promise((r) => ws.once('open', r));

  ws.send(JSON.stringify({ trigger: 'crash' }));
  const code = await new Promise((r) => ws.once('close', (c) => r(c)));
  expect(code).toBe(1011);
});

test('graceful shutdown sends 1001', async () => {
  // server initiates shutdown; client observes 1001
  const ws = new WebSocket('ws://localhost:8080/');
  await new Promise((r) => ws.once('open', r));

  await fetch('http://localhost:8080/admin/shutdown', { method: 'POST' });
  const code = await new Promise((r) => ws.once('close', (c) => r(c)));
  expect(code).toBe(1001);
});

Step 6 - Backpressure / large message tests

Per RFC 6455, control frame payloads are ≤ 125 bytes; data frames have no upper bound but implementations apply limits. Test server's maxPayload config:

test('server rejects message > maxPayload', async () => {
  const ws = new WebSocket('ws://localhost:8080/');
  await new Promise((r) => ws.once('open', r));

  const big = 'a'.repeat(2 * 1024 * 1024); // 2 MB
  ws.send(big);

  const code = await new Promise((r) => ws.once('close', (c) => r(c)));
  expect(code).toBe(1009); // Message too big
});

Step 7 - Reconnect with jitter

Reconnect logic should exponential backoff + jitter (not RFC 6455 itself, but field-tested practice):

test('client reconnects within 30s after server bounce', async () => {
  const ws = createReconnectingClient('ws://localhost:8080/');
  await waitForState(ws, 'open');

  await bounceServer();

  const reconnected = await waitForState(ws, 'open', { timeout: 30_000 });
  expect(reconnected).toBe(true);
});

Cross-ref qa-resilience-drills/error-budget-tests for SLO-driven reconnect budget.

Step 8 - Playwright frame inspection (browser e2e)

test('app sends auth frame on open', async ({ page }) => {
  page.on('websocket', (ws) => {
    ws.on('framesent', (event) => {
      const data = JSON.parse(event.payload as string);
      if (data.type === 'auth') {
        expect(data.token).toBeTruthy();
      }
    });
  });
  await page.goto('https://localhost:3000/dashboard');
});

Anti-patterns

Anti-patternWhy it failsFix
Skip handshake test (assume browser handles it)Server-side Sec-WebSocket-Accept bug shipsStep 2
Test only happy-path message exchangeReconnect/close-code bugs slip throughStep 5 + Step 7
Use HTTP polling fallback as "good enough"Different code path; doesn't validate WSTest the actual WS path
Hard-code reconnect interval, no jitterThundering herd on server bounceExponential backoff + jitter
Skip subprotocol test in versioned APIsOld clients silently get v2 server response shapeStep 3

Limitations

  • RFC 6455 is the base spec. WebSocket-over-HTTP/2 (RFC 8441) changes the bootstrap; verify per stack if HTTP/2 is in play.
  • Browser autobahn-suite-style edge cases (fragmentation, extension negotiation) are out of scope here; use Autobahn TestSuite for full conformance.

References