Skip to content

Commit de99140

Browse files
authoredMay 9, 2023
Merge pull request #8893 from marmelab/realtime-supabase
[Doc] Add Supabase realtime adapter
2 parents dfb0d75 + deef4f0 commit de99140

File tree

1 file changed

+93
-3
lines changed

1 file changed

+93
-3
lines changed
 

‎docs/RealtimeDataProvider.md

+93-3
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,16 @@ title: "Realtime DataProvider Requirements"
55

66
# Realtime DataProvider Requirements
77

8+
`ra-realtime` provides helper functions to add real-time capabilities to an existing data provider if you use the following real-time backends:
9+
10+
- [Supabase](#supabase)
11+
- [API Platform](#api-platform)
12+
- [Mercure](#mercure)
13+
14+
For other backends, you'll need to write your own implementation. Check the [Writing a custom adapter](#writing-a-custom-adapter) section below for more information.
15+
16+
## Realtime Methods & Signature
17+
818
To enable real-time features, the `dataProvider` must implement three new methods:
919

1020
- `subscribe(topic, callback)`
@@ -20,7 +30,87 @@ In addition, to support the lock features, the `dataProvider` must implement 4 m
2030
- `getLock(resource, { id, meta })`
2131
- `getLocks(resource, { meta })`
2232

23-
## API-Platform Adapter
33+
## Supabase
34+
35+
The `ra-realtime` package contains a function augmenting a regular (API-based) `dataProvider` with real-time methods based on the capabilities of [Supabase](https://supabase.com/docs/guides/realtime).
36+
37+
This adapter subscribes to [Postgres Changes](https://supabase.com/docs/guides/realtime/extensions/postgres-changes), and transforms the events into the format expected by `ra-realtime`.
38+
39+
```jsx
40+
import { createClient } from '@supabase/supabase-js';
41+
import { supabaseDataProvider } from 'ra-supabase';
42+
import { addRealTimeMethodsBasedOnSupabase, ListLive } from '@react-admin/ra-realtime';
43+
import { Admin, Resource, Datagrid, TextField, EmailField } from 'react-admin';
44+
45+
const supabaseClient = createClient(
46+
process.env.SUPABASE_URL,
47+
process.env.SUPABASE_ANON_KEY
48+
);
49+
50+
const dataProvider = supabaseDataProvider({
51+
instanceUrl: process.env.SUPABASE_URL,
52+
apiKey: process.env.SUPABASE_ANON_KEY,
53+
supabaseClient
54+
});
55+
56+
const realTimeDataProvider = addRealTimeMethodsBasedOnSupabase({
57+
dataProvider,
58+
supabaseClient,
59+
});
60+
61+
export const App = () => (
62+
<Admin dataProvider={realTimeDataProvider}>
63+
<Resource name="sales" list={SaleList} />
64+
</Admin>
65+
);
66+
67+
const SaleList = () => (
68+
<ListLive>
69+
<Datagrid rowClick="edit">
70+
<TextField source="id" />
71+
<TextField source="first_name" />
72+
<TextField source="last_name" />
73+
<EmailField source="email" />
74+
</Datagrid>
75+
</ListLive>
76+
);
77+
```
78+
79+
**Tip:** Realtime features are not enabled in Supabase by default, you need to enable them. This can be done either from the [Replication](https://app.supabase.com/project/_/database/replication) section of your Supabase Dashboard, or by running the following SQL query with the [SQL Editor](https://app.supabase.com/project/_/sql):
80+
81+
```sql
82+
begin;
83+
84+
-- remove the supabase_realtime publication
85+
drop
86+
publication if exists supabase_realtime;
87+
88+
-- re-create the supabase_realtime publication with no tables
89+
create publication supabase_realtime;
90+
91+
commit;
92+
93+
-- add a table to the publication
94+
alter
95+
publication supabase_realtime add table sales;
96+
alter
97+
publication supabase_realtime add table contacts;
98+
alter
99+
publication supabase_realtime add table contactNotes;
100+
```
101+
102+
Have a look at the Supabase [Replication Setup](https://supabase.com/docs/guides/realtime/extensions/postgres-changes#replication-setup) documentation section for more info.
103+
104+
`addRealTimeMethodsBasedOnSupabase` accepts the following parameters:
105+
106+
| Prop | Required | Type | Default | Description |
107+
| ----------------- | -------- | ---------------- | ------- | -------------------------------------------------------- |
108+
| `dataProvider` | Required | `DataProvider` | - | The base dataProvider to augment with realtime methods |
109+
| `supabaseClient` | Required | `SupabaseClient` | - | The Supabase JS Client |
110+
111+
**Tip**: You may choose to sign your own tokens to customize claims that can be checked in your RLS policies. In order to use these custom tokens with `addRealTimeMethodsBasedOnSupabase`, you must pass an `apikey` field in both Realtime's `headers` and `params` when creating the `supabaseClient`. Please follow the instructions from the [Supabase documentation](https://supabase.com/docs/guides/realtime/extensions/postgres-changes#custom-tokens) for more information about how to do so.
112+
113+
## API-Platform
24114

25115
The `ra-realtime` package contains a function augmenting a regular (API-based) `dataProvider` with real-time methods based on the capabilities of [API-Platform](https://api-platform.com/). Use it as follows:
26116

@@ -70,7 +160,7 @@ const GreetingsList = () => (
70160
);
71161
```
72162

73-
## Mercure Adapter
163+
## Mercure
74164

75165
The `ra-realtime` package contains a function augmenting a regular (API-based) `dataProvider` with real-time methods based on [a Mercure hub](https://mercure.rocks/). Use it as follows:
76166

@@ -95,7 +185,7 @@ const App = () => (
95185

96186
If you're using another transport for real-time messages (WebSockets, long polling, GraphQL subscriptions, etc.), you'll have to implement `subscribe`, `unsubscribe`, and `publish` yourself in your `dataProvider`. As an example, here is an implementation using a local variable, that `ra-realtime` uses in tests:
97187

98-
```ts
188+
```jsx
99189
let subscriptions = [];
100190

101191
const dataProvider = {

0 commit comments

Comments
 (0)