sse client
This commit is contained in:
parent
1d92b331e2
commit
cadd4889b6
79
pocketbase/services/utils/sse.py
Normal file
79
pocketbase/services/utils/sse.py
Normal file
@ -0,0 +1,79 @@
|
||||
from __future__ import annotations
|
||||
|
||||
from dataclasses import dataclass
|
||||
|
||||
import httpx
|
||||
|
||||
|
||||
@dataclass
|
||||
class Event:
|
||||
"""Representation of an event"""
|
||||
|
||||
id: str = ""
|
||||
event: str = "message"
|
||||
data: str = ""
|
||||
retry: int | None = None
|
||||
|
||||
|
||||
class SSEClient:
|
||||
"""Implementation of a server side event client"""
|
||||
|
||||
FIELD_SEPARATOR = ":"
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
url: str,
|
||||
method: str = "GET",
|
||||
headers: dict | None = None,
|
||||
payload: dict | None = None,
|
||||
encoding="utf-8",
|
||||
) -> None:
|
||||
self.url = url
|
||||
self.method = method
|
||||
self.headers = headers
|
||||
self.payload = payload
|
||||
self.encoding = encoding
|
||||
|
||||
def _read(self):
|
||||
"""Read the incoming event source stream and yield event chunks"""
|
||||
data = b""
|
||||
with httpx.stream(
|
||||
self.method, self.url, headers=self.headers, data=self.payload, timeout=None
|
||||
) as r:
|
||||
for chunk in r.iter_bytes():
|
||||
for line in chunk.splitlines(True):
|
||||
data += line
|
||||
if data.endswith((b"\r\r", b"\n\n", b"\r\n\r\n")):
|
||||
yield data
|
||||
data = b""
|
||||
if data:
|
||||
yield data
|
||||
|
||||
def events(self):
|
||||
for chunk in self._read():
|
||||
event = Event()
|
||||
for line in chunk.splitlines():
|
||||
line = line.decode(self.encoding)
|
||||
if not line.strip() or line.startswith(self.FIELD_SEPARATOR):
|
||||
continue
|
||||
data = line.split(self.FIELD_SEPARATOR, 1)
|
||||
field = data[0]
|
||||
if field not in event.__dict__:
|
||||
continue
|
||||
if len(data) > 1:
|
||||
if data[1].startswith(" "):
|
||||
value = data[1][1:]
|
||||
else:
|
||||
value = data[1]
|
||||
else:
|
||||
value = ""
|
||||
if field == "data":
|
||||
event.data += value + "\n"
|
||||
else:
|
||||
setattr(event, field, value)
|
||||
if not event.data:
|
||||
continue
|
||||
if event.data.endswith("\n"):
|
||||
event.data = event.data[0:-1]
|
||||
event.event = event.event or "message"
|
||||
yield event
|
||||
Loading…
x
Reference in New Issue
Block a user