rembrembdocs

You can use WebSockets for all or some of the communication with your server, see wsLink for how to set it up on the client.

tip

The document here outlines the specific details of using WebSockets. For general usage of subscriptions, see our subscriptions guide.

Creating a WebSocket-server

bash

yarn add ws

server/wsServer.ts

ts

import { applyWSSHandler } from '@trpc/server/adapters/ws';

import { WebSocketServer } from 'ws';

import { appRouter } from './routers/app';

import { createContext } from './trpc';

const wss = new WebSocketServer({

port: 3001,

});

const handler = applyWSSHandler({

wss,

router: appRouter,

createContext,

// Enable heartbeat messages to keep connection open (disabled by default)

keepAlive: {

`enabled: true,`

`// server ping message interval in milliseconds`

`pingMs: 30000,`

`// connection is terminated if pong message is not received in this many milliseconds`

`pongWaitMs: 5000,`

},

});

wss.on('connection', (ws) => {

console.log(`++ Connection (${wss.clients.size})`);

ws.once('close', () => {

``console.log(`-- Connection (${wss.clients.size})`);``

});

});

console.log('WebSocket Server listening on ws://localhost:3001');

process.on('SIGTERM', () => {

console.log('SIGTERM');

handler.broadcastReconnectNotification();

wss.close();

});

Setting TRPCClient to use WebSockets

tip

You can use Links to route queries and/or mutations to HTTP transport and subscriptions over WebSockets.

client.ts

tsx

import { createTRPCClient, createWSClient, wsLink } from '@trpc/client';

import type { AppRouter } from './server';

// create persistent WebSocket connection

const wsClient = createWSClient({

url: `ws://localhost:3001`,

});

// configure TRPCClient to use WebSockets transport

const client = createTRPCClient<AppRouter>({

links: [

`wsLink({`

  `client: wsClient,`

`}),`

],

});

Authentication / connection params

tip

If you're doing a web application, you can ignore this section as the cookies are sent as part of the request.

In order to authenticate with WebSockets, you can define connectionParams to createWSClient. This will be sent as the first message when the client establishes a WebSocket connection.

server/context.ts

ts

import type { CreateWSSContextFnOptions } from '@trpc/server/adapters/ws';

export const createContext = async (opts: CreateWSSContextFnOptions) => {

const token = opts.info.connectionParams?.token;

     `const token: string | undefined`

// [... authenticate]

return {};

};

export type Context = Awaited<ReturnType<typeof createContext>>;

client/trpc.ts

ts

import { createTRPCClient, createWSClient, wsLink } from '@trpc/client';

import type { AppRouter } from './server';

import superjson from 'superjson';

const wsClient = createWSClient({

url: `ws://localhost:3000`,

connectionParams: async () => {

`return {`

  `token: 'supersecret',`

`};`

},

});

export const trpc = createTRPCClient<AppRouter>({

links: [wsLink({ client: wsClient, transformer: superjson })],

});

Automatic tracking of id using tracked() (recommended)

If you yield an event using our tracked()-helper and include an id, the client will automatically reconnect when it gets disconnected and send the last known ID when reconnecting as part of the lastEventId-input.

You can send an initial lastEventId when initializing the subscription and it will be automatically updated as the browser receives data.

info

If you're fetching data based on the lastEventId, and capturing all events is critical, you may want to use ReadableStream's or a similar pattern as an intermediary as is done in our full-stack SSE example to prevent newly emitted events being ignored while yield'ing the original batch based on lastEventId.

ts

import EventEmitter, { on } from 'events';

import { initTRPC, tracked } from '@trpc/server';

import { z } from 'zod';

type Post = { id: string; title: string };

const t = initTRPC.create();

const publicProcedure = t.procedure;

const router = t.router;

const ee = new EventEmitter();

export const subRouter = router({

onPostAdd: publicProcedure

`.input(`

  `z`

    `.object({`

      `// lastEventId is the last event id that the client has received`

      `// On the first call, it will be whatever was passed in the initial setup`

      `// If the client reconnects, it will be the last event id that the client received`

      `lastEventId: z.string().nullish(),`

    `})`

    `.optional(),`

`)`

`.subscription(async function* (opts) {`

  `if (opts.input?.lastEventId) {`

    `// [...] get the posts since the last event id and yield them`

  `}`

  `// listen for new events`

  `for await (const [data] of on(ee, 'add', {`

    `// Passing the AbortSignal from the request automatically cancels the event emitter when the subscription is aborted`

    `signal: opts.signal,`

  `})) {`

    `const post = data as Post;`

    `// tracking the post id ensures the client can reconnect at any time and get the latest events since this id`

    `yield tracked(post.id, post);`

  `}`

`}),`

});

WebSockets RPC Specification

You can read more details by drilling into the TypeScript definitions:

query / mutation

Request

ts

interface RequestMessage {

id: number | string;

jsonrpc?: '2.0';

method: 'query' | 'mutation';

params: {

`path: string;`

`input?: unknown; // <-- pass input of procedure, serialized by transformer`

};

}

Response

... below, or an error.

ts

interface ResponseMessage {

id: number | string;

jsonrpc?: '2.0';

result: {

`type: 'data'; // always 'data' for mutation / queries`

`data: TOutput; // output from procedure`

};

}

subscription / subscription.stop

Start a subscription

ts

interface SubscriptionRequest {

id: number | string;

jsonrpc?: '2.0';

method: 'subscription';

params: {

`path: string;`

`input?: unknown; // <-- pass input of procedure, serialized by transformer`

};

}

To cancel a subscription, call subscription.stop

ts

interface SubscriptionStopRequest {

id: number | string; // <-- id of your created subscription

jsonrpc?: '2.0';

method: 'subscription.stop';

}

Subscription response shape

... below, or an error.

ts

interface SubscriptionResponse {

id: number | string;

jsonrpc?: '2.0';

result:

`| {`

    `type: 'data';`

    `data: TData; // subscription emitted data`

  `}`

`| {`

    `type: 'started'; // subscription started`

  `}`

`| {`

    `type: 'stopped'; // subscription stopped`

  `};`

}

Connection params

If the connection is initialized with ?connectionParams=1, the first message has to be connection params.

ts

interface ConnectionParamsMessage {

data: Record<string, string> | null;

method: 'connectionParams';

}

Errors

See https://www.jsonrpc.org/specification#error_object or Error Formatting.

Notifications from Server to Client

{ id: null, type: 'reconnect' }

Tells clients to reconnect before shutting down the server. Invoked by wssHandler.broadcastReconnectNotification().