Skip to content
4th April 2024: This is a preview, whilst production-ready, it means some APIs might change

Realtime

The SDK includes built-in support for realtime updates using Cloudflare Durable Objects and WebSockets. With just a few lines of setup, you can enable bidirectional communication between clients and the server — without polling.


Setup

You’ll need to connect three parts:

1. Client Setup

On the client side, initialize the realtime connection with a key. This key determines which group of clients should share updates. More on this below.

src/client.tsx
import { initRealtimeClient } from "@redwoodjs/sdk/realtime/client";
initRealtimeClient({
key: window.location.pathname, // Used to group related clients
});

2. Export the Durable Object

src/worker.ts
export { RealtimeDurableObject } from "@redwoodjs/sdk/realtime/durableObject";

3. Wire Up the Worker Route

src/worker.ts
import { realtimeRoute } from "@redwoodjs/sdk/realtime/worker";
export default defineApp([
realtimeRoute((env) => env.REALTIME_DURABLE_OBJECT),
// ... your routes
]);

4. Add the Durable Object to wrangler.jsonc

"durable_objects": {
"bindings": [
// ...
{
"name": "REALTIME_DURABLE_OBJECT",
"class_name": "RealtimeDurableObject",
},
],
},

Example: Collaborative Notes

To see realtime updates in action, check out our collaborative notes example.

Multiple users can edit the same note at the same time, and see each other’s changes as they happen.


Understanding Realtime in the SDK

React Server Components provide a way of describing the UI you want to render based the latest app state. When an event happens (whether user or system-initiated), the app state is updated, the server re-renders the UI, and the client receives the result.

In RedwoodSDK, we build on top of this model for real time updates.

  1. An event happens — user action or external trigger
  2. App state is updated — using your existing RSC action handlers
  3. Re-render is triggered — either automatically (as a result of an action) or explicitly via renderRealtimeClients()
  4. Each client re-renders — by calling your server code and receiving the latest state

This means you don’t need to wire up subscriptions or manually track diffs — just write your app as usual and let the server deliver updated UI to each connected client.


Scoping Updates

Each realtime connection is scoped by a key. This key determines which clients are considered part of the same group, so they can receive the same updates.

All clients with the same key are connected to the same Durable Object instance. Updates affecting one client are pushed to others in the same group.

For example:

initRealtimeClient({ key: "/chat/room-42" });

Only clients in room-42 will receive updates related to that room.


Client ➝ Server ➝ Client Updates

For updates that begin from user interaction, consider this example:

const Note = async ({ appContext }: RouteOptions) => {
return <Editor content={appContext.content} />;
};

Here, the appContext has been populated with the latest content for the relevant note.

This is just a normal React Server Component. What’s new is that when one client triggers an action, all other clients with the same key will re-run this server logic too.


Server ➝ Client Updates

You can update clients even when the event didn’t originate from a user action — for example, background events, notifications, or admin triggers.

Use renderRealtimeClients() to trigger a re-render for all clients connected to a given key:

import { renderRealtimeClients } from "@redwoodjs/sdk/realtime/worker";
await renderRealtimeClients({
durableObjectNamespace: env.REALTIME_DURABLE_OBJECT,
key: "/note/some-id",
});

Why WebSockets and Durable Objects?

To support realtime updates on Cloudflare, WebSockets and Durable Objects are a natural fit. WebSockets give us a persistent, bidirectional connection - not just for pushing updates to the client, but also for sending actions back to the server. Since that connection is already open, we can reuse it for both directions, avoiding the overhead of establishing new HTTP requests.

Durable Objects are well-suited for managing these connections: they can persist across requests, maintain in-memory state, and coordinate updates between connected clients.

API Reference

initRealtimeClient({ key?: string }): Promise<void>

Initialises the realtime WebSocket client.

  • key: (optional) Identifies which group of clients this user belongs to.
realtimeRoute((env) => DurableObjectNamespace): RouteDefinition

Connects the WebSocket route in your worker to the appropriate Durable Object.

renderRealtimeClients({
durableObjectNamespace,
key?: string,
}): Promise<void>

Triggers a re-render for all clients with a given key.

  • durableObjectNamespace: your binding to the Durable Object
  • key: the scope of clients to re-render