Realtime

Realtime Concepts

Learn about channels and other features in Supabase Realtime


You can use Supabase Realtime to build real-time applications with collaborative/multiplayer functionality. It includes 3 core features:

  • Broadcast: sends rapid, ephemeral messages to other connected clients. You can use it to track mouse movements, for example.
  • Presence: sends user state between connected clients. You can use it to show an "online" status, which disappears when a user is disconnected.
  • Postgres Changes: receives database changes in real-time.

Channels

A Channel is the basic building block of Realtime. You can think of a Channel as a chatroom, similar to a Discord or Slack channel, where participants are able to see who's online and send and receive messages.

When you initialize your Supabase Realtime client, you define a topic that uniquely references a channel. Clients can bi-directionally send and receive messages over a Channel.


_10
import { createClient } from '@supabase/supabase-js'
_10
_10
const client = createClient('https://<project>.supabase.co', '<your-anon-key>')
_10
_10
const roomOne = client.channel('room-one') // set your topic here

Broadcast

Realtime Broadcast follows the publish-subscribe pattern where a client publishes messages to a channel based on a unique topic. For example, a user could send a message to a channel with topic room-one.


_10
roomOne.send({
_10
type: 'broadcast',
_10
event: 'test',
_10
payload: { message: 'hello, world' },
_10
})

Other clients can receive the message in real-time by subscribing to the Channel with topic room-one. These clients continue to receive messages as long as they are subscribed and connected to the same Channel topic.

An example use-case is sharing a user's cursor position with other clients in an online game.

Presence

Presence can be used to share an individual's state with others within a Channel.


_10
const presenceTrackStatus = await roomOne.track({
_10
user: 'user-1',
_10
online_at: new Date().toISOString(),
_10
})

Each client maintains their own state, and this is then combined into a "shared state" for that Channel topic. It's commonly used for sharing statuses (eg: "online" or "inactive"). The neat thing about Presence is that if a client is suddenly disconnected (for example, they go offline), their state is automatically removed from the shared state. If you've ever tried to build an “I'm online” feature which handles unexpected disconnects, you'll appreciate how useful this is.

When a new client subscribes to a channel, it will immediately receive the channel's latest state in a single message because the state is held by the Realtime server.

Choosing between broadcast and presence

We recommend using Broadcast by default, and then Presence when required. Presence utilizes an in-memory conflict-free replicated data type (CRDT) to track and synchronize shared state in an eventually consistent manner. It computes the difference between existing state and new state changes and sends the necessary updates to clients via Broadcast. This is computationally heavy, so you should use it sparingly. If you use Presence, it's best to throttle your changes so that you are sending updates less frequently.

Realtime extensions

Channels provide a generic networking solution. Supabase Realtime is designed to leverage this networking primitive with "extensions". We currently support one extension: Postgres changes.

Postgres changes

The Postgres Changes extension listens for database changes and sends them to clients. Clients are required to subscribe with a JWT dictating which changes they are allowed to receive based on the database's Row Level Security.


_11
const allChanges = client
_11
.channel('schema-db-changes')
_11
.on(
_11
'postgres_changes',
_11
{
_11
event: '*',
_11
schema: 'public',
_11
},
_11
(payload) => console.log(payload)
_11
)
_11
.subscribe()

Anyone with access to a valid JWT signed with the project's JWT secret is able to listen to your database's changes, unless tables have Row Level Security enabled and policies in place.

Clients can choose to receive INSERT, UPDATE, DELETE, or * (all) changes for all changes in a schema, a table in a schema, or a column's value in a table. Your clients should only listen to tables in the public schema and you must first enable the tables you want your clients to listen to.