Skip to content

Commit 0a96f70

Browse files
authored
feat: remove the need of calling listen method (#274)
* fix: remove listen * deprecate old listen method * fix reconnection * remove listen instruction from README * fix reconnection and fix tests * revert auth test * omit test folder from coverage * remove install supabase cli step since npx handles it * ignore vscode config files * update docstrings * format
1 parent 2ecec40 commit 0a96f70

File tree

9 files changed

+176
-129
lines changed

9 files changed

+176
-129
lines changed

.github/workflows/ci.yml

+1-7
Original file line numberDiff line numberDiff line change
@@ -27,14 +27,8 @@ jobs:
2727
- name: Set up Poetry
2828
run: pipx install poetry==1.8.5 --python python${{ matrix.python-version }}
2929

30-
31-
- name: Install Supabase CLI
32-
uses: supabase/setup-cli@v1
33-
with:
34-
version: latest
35-
3630
- name: Start Supabase local development setup
37-
run: supabase start --workdir infra -x studio,inbucket,edge-runtime,logflare,vector,supavisor,imgproxy,storage-api
31+
run: make run_infra
3832

3933
- name: Run Tests
4034
run: make run_tests

.gitignore

+3
Original file line numberDiff line numberDiff line change
@@ -141,3 +141,6 @@ dmypy.json
141141

142142
.idea/*
143143
# End of https://www.toptal.com/developers/gitignore/api/python
144+
145+
146+
.vscode/*

Makefile

+1-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ run_tests: tests
2121
local_tests: run_infra sleep tests
2222

2323
tests_only:
24-
poetry run pytest --cov=./ --cov-report=xml --cov-report=html -vv
24+
poetry run pytest --cov=realtime --cov-report=xml --cov-report=html -vv
2525

2626
sleep:
2727
sleep 2

README.md

-3
Original file line numberDiff line numberDiff line change
@@ -59,9 +59,6 @@ def _on_subscribe(status: RealtimeSubscribeStates, err: Optional[Exception]):
5959
print('Realtime channel was unexpectedly closed.')
6060

6161
await channel.subscribe(_on_subscribe)
62-
63-
# Listen for all incoming events, often the last thing you want to do.
64-
await client.listen()
6562
```
6663

6764
### Notes:

realtime/_async/channel.py

+49-29
Original file line numberDiff line numberDiff line change
@@ -33,9 +33,9 @@
3333

3434
class AsyncRealtimeChannel:
3535
"""
36-
`Channel` is an abstraction for a topic listener for an existing socket connection.
37-
Each Channel has its own topic and a list of event-callbacks that responds to messages.
38-
Should only be instantiated through `connection.RealtimeClient().channel(topic)`.
36+
Channel is an abstraction for a topic subscription on an existing socket connection.
37+
Each Channel has its own topic and a list of event-callbacks that respond to messages.
38+
Should only be instantiated through `AsyncRealtimeClient.channel(topic)`.
3939
"""
4040

4141
def __init__(
@@ -52,13 +52,14 @@ def __init__(
5252
:param params: Optional parameters for connection.
5353
"""
5454
self.socket = socket
55-
self.params = params or RealtimeChannelOptions(
56-
config={
55+
self.params = params or {}
56+
if self.params.get("config") is None:
57+
self.params["config"] = {
5758
"broadcast": {"ack": False, "self": False},
5859
"presence": {"key": ""},
5960
"private": False,
6061
}
61-
)
62+
6263
self.topic = topic
6364
self._joined_once = False
6465
self.bindings: Dict[str, List[Binding]] = {}
@@ -97,7 +98,7 @@ def on_close(*args):
9798
logger.info(f"channel {self.topic} closed")
9899
self.rejoin_timer.reset()
99100
self.state = ChannelStates.CLOSED
100-
self.socket.remove_channel(self)
101+
self.socket._remove_channel(self)
101102

102103
def on_error(payload, *args):
103104
if self.is_leaving or self.is_closed:
@@ -148,12 +149,16 @@ async def subscribe(
148149
] = None,
149150
) -> AsyncRealtimeChannel:
150151
"""
151-
Subscribe to the channel.
152+
Subscribe to the channel. Can only be called once per channel instance.
152153
153-
:return: The Channel instance for method chaining.
154+
:param callback: Optional callback function that receives subscription state updates
155+
and any errors that occur during subscription
156+
:return: The Channel instance for method chaining
157+
:raises: Exception if called multiple times on the same channel instance
154158
"""
155159
if not self.socket.is_connected:
156160
await self.socket.connect()
161+
157162
if self._joined_once:
158163
raise Exception(
159164
"Tried to subscribe multiple times. 'subscribe' can only be called a single time per channel instance"
@@ -249,6 +254,10 @@ def on_join_push_timeout(*args):
249254
return self
250255

251256
async def unsubscribe(self):
257+
"""
258+
Unsubscribe from the channel and leave the topic.
259+
Sets channel state to LEAVING and cleans up timers and pushes.
260+
"""
252261
self.state = ChannelStates.LEAVING
253262

254263
self.rejoin_timer.reset()
@@ -269,6 +278,15 @@ def _close(*args):
269278
async def push(
270279
self, event: str, payload: Dict[str, Any], timeout: Optional[int] = None
271280
) -> AsyncPush:
281+
"""
282+
Push a message to the channel.
283+
284+
:param event: The event name to push
285+
:param payload: The payload to send
286+
:param timeout: Optional timeout in milliseconds
287+
:return: AsyncPush instance representing the push operation
288+
:raises: Exception if called before subscribing to the channel
289+
"""
272290
if not self._joined_once:
273291
raise Exception(
274292
f"tried to push '{event}' to '{self.topic}' before joining. Use channel.subscribe() before pushing events"
@@ -350,9 +368,9 @@ def on_broadcast(
350368
"""
351369
Set up a listener for a specific broadcast event.
352370
353-
:param event: The name of the broadcast event to listen for.
354-
:param callback: The callback function to execute when the event is received.
355-
:return: The Channel instance for method chaining.
371+
:param event: The name of the broadcast event to listen for
372+
:param callback: Function called with the payload when a matching broadcast is received
373+
:return: The Channel instance for method chaining
356374
"""
357375
return self._on(
358376
"broadcast",
@@ -369,13 +387,14 @@ def on_postgres_changes(
369387
filter: Optional[str] = None,
370388
) -> AsyncRealtimeChannel:
371389
"""
372-
Set up a listener for a specific Postgres changes event.
373-
374-
:param event: The name of the Postgres changes event to listen for.
375-
:param table: The table name for which changes should be monitored.
376-
:param callback: The callback function to execute when the event is received.
377-
:param schema: The database schema where the table exists. Default is 'public'.
378-
:return: The Channel instance for method chaining.
390+
Set up a listener for Postgres database changes.
391+
392+
:param event: The type of database event to listen for (INSERT, UPDATE, DELETE, or *)
393+
:param callback: Function called with the payload when a matching change is detected
394+
:param table: The table name to monitor. Defaults to "*" for all tables
395+
:param schema: The database schema to monitor. Defaults to "public"
396+
:param filter: Optional filter string to apply
397+
:return: The Channel instance for method chaining
379398
"""
380399

381400
binding_filter = {"event": event, "schema": schema, "table": table}
@@ -402,22 +421,24 @@ def on_system(
402421
# Presence methods
403422
async def track(self, user_status: Dict[str, Any]) -> None:
404423
"""
405-
Track a user's presence.
424+
Track presence status for the current user.
406425
407-
:param user_status: User's presence status.
408-
:return: None
426+
:param user_status: Dictionary containing the user's presence information
409427
"""
410428
await self.send_presence("track", user_status)
411429

412430
async def untrack(self) -> None:
413431
"""
414-
Untrack a user's presence.
415-
416-
:return: None
432+
Stop tracking presence for the current user.
417433
"""
418434
await self.send_presence("untrack", {})
419435

420436
def presence_state(self) -> RealtimePresenceState:
437+
"""
438+
Get the current state of presence on this channel.
439+
440+
:return: Dictionary mapping presence keys to lists of presence payloads
441+
"""
421442
return self.presence.state
422443

423444
def on_presence_sync(self, callback: Callable[[], None]) -> AsyncRealtimeChannel:
@@ -457,11 +478,10 @@ def on_presence_leave(
457478
# Broadcast methods
458479
async def send_broadcast(self, event: str, data: Any) -> None:
459480
"""
460-
Sends a broadcast message to the current channel.
481+
Send a broadcast message through this channel.
461482
462-
:param event: The name of the broadcast event.
463-
:param data: The data to be sent with the message.
464-
:return: An asyncio.Future object representing the send operation.
483+
:param event: The name of the broadcast event
484+
:param data: The payload to broadcast
465485
"""
466486
await self.push(
467487
ChannelEvents.broadcast,

0 commit comments

Comments
 (0)