-
Notifications
You must be signed in to change notification settings - Fork 42
/
Copy pathsse.py
247 lines (203 loc) · 8.89 KB
/
sse.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
import logging
from datetime import datetime, timezone
from typing import (
Any,
AsyncIterable,
Awaitable,
Callable,
Coroutine,
Iterator,
Mapping,
Optional,
Union,
)
import anyio
from starlette.background import BackgroundTask
from starlette.concurrency import iterate_in_threadpool
from starlette.datastructures import MutableHeaders
from starlette.responses import Response
from starlette.types import Receive, Scope, Send
from sse_starlette.event import ServerSentEvent, ensure_bytes
logger = logging.getLogger(__name__)
class SendTimeoutError(TimeoutError):
pass
class AppStatus:
"""Helper to capture a shutdown signal from Uvicorn so we can gracefully terminate SSE streams."""
should_exit = False
should_exit_event: Union[anyio.Event, None] = None
original_handler = None
@staticmethod
def handle_exit(*args, **kwargs):
# Mark that the app should exit, and signal all waiters.
AppStatus.should_exit = True
if AppStatus.should_exit_event is not None:
AppStatus.should_exit_event.set()
if AppStatus.original_handler is not None:
AppStatus.original_handler(*args, **kwargs)
try:
from uvicorn.main import Server
AppStatus.original_handler = Server.handle_exit
Server.handle_exit = AppStatus.handle_exit # type: ignore
except ImportError:
logger.debug(
"Uvicorn not installed. Graceful shutdown on server termination disabled."
)
Content = Union[str, bytes, dict, ServerSentEvent, Any]
SyncContentStream = Iterator[Content]
AsyncContentStream = AsyncIterable[Content]
ContentStream = Union[AsyncContentStream, SyncContentStream]
class EventSourceResponse(Response):
"""
Streaming response that sends data conforming to the SSE (Server-Sent Events) specification.
"""
DEFAULT_PING_INTERVAL = 15
DEFAULT_SEPARATOR = "\r\n"
def __init__(
self,
content: ContentStream,
status_code: int = 200,
headers: Optional[Mapping[str, str]] = None,
media_type: str = "text/event-stream",
background: Optional[BackgroundTask] = None,
ping: Optional[int] = None,
sep: Optional[str] = None,
ping_message_factory: Optional[Callable[[], ServerSentEvent]] = None,
data_sender_callable: Optional[
Callable[[], Coroutine[None, None, None]]
] = None,
send_timeout: Optional[float] = None,
) -> None:
# Validate separator
if sep not in (None, "\r\n", "\r", "\n"):
raise ValueError(f"sep must be one of: \\r\\n, \\r, \\n, got: {sep}")
self.sep = sep or self.DEFAULT_SEPARATOR
# If content is sync, wrap it for async iteration
if isinstance(content, AsyncIterable):
self.body_iterator = content
else:
self.body_iterator = iterate_in_threadpool(content)
self.status_code = status_code
self.media_type = self.media_type if media_type is None else media_type
self.background = background
self.data_sender_callable = data_sender_callable
self.send_timeout = send_timeout
# Build SSE-specific headers.
_headers = MutableHeaders()
if headers is not None: # pragma: no cover
_headers.update(headers)
# "The no-store response directive indicates that any caches of any kind (private or shared)
# should not store this response."
# -- https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Cache-Control
# allow cache control header to be set by user to support fan out proxies
# https://www.fastly.com/blog/server-sent-events-fastly
_headers.setdefault("Cache-Control", "no-store")
# mandatory for servers-sent events headers
_headers["Connection"] = "keep-alive"
_headers["X-Accel-Buffering"] = "no"
self.init_headers(_headers)
self.ping_interval = self.DEFAULT_PING_INTERVAL if ping is None else ping
self.ping_message_factory = ping_message_factory
self.active = True
# https://github.com/sysid/sse-starlette/pull/55#issuecomment-1732374113
self._send_lock = anyio.Lock()
@property
def ping_interval(self) -> Union[int, float]:
return self._ping_interval
@ping_interval.setter
def ping_interval(self, value: Union[int, float]) -> None:
if not isinstance(value, (int, float)):
raise TypeError("ping interval must be int")
if value < 0:
raise ValueError("ping interval must be greater than 0")
self._ping_interval = value
def enable_compression(self, force: bool = False) -> None:
raise NotImplementedError("Compression is not supported for SSE streams.")
async def _stream_response(self, send: Send) -> None:
"""Send out SSE data to the client as it becomes available in the iterator."""
await send(
{
"type": "http.response.start",
"status": self.status_code,
"headers": self.raw_headers,
}
)
async for data in self.body_iterator:
chunk = ensure_bytes(data, self.sep)
logger.debug("chunk: %s", chunk)
with anyio.move_on_after(self.send_timeout) as cancel_scope:
await send(
{"type": "http.response.body", "body": chunk, "more_body": True}
)
if cancel_scope and cancel_scope.cancel_called:
if hasattr(self.body_iterator, "aclose"):
await self.body_iterator.aclose()
raise SendTimeoutError()
async with self._send_lock:
self.active = False
await send({"type": "http.response.body", "body": b"", "more_body": False})
@staticmethod
async def _listen_for_disconnect(receive: Receive) -> None:
"""Watch for a disconnect message from the client."""
while True:
message = await receive()
if message["type"] == "http.disconnect":
logger.debug("Got event: http.disconnect. Stop streaming.")
break
@staticmethod
async def _listen_for_exit_signal() -> None:
"""Watch for shutdown signals (e.g. SIGINT, SIGTERM) so we can break the event loop."""
# Check if should_exit was set before anybody started waiting
if AppStatus.should_exit:
return
if AppStatus.should_exit_event is None:
AppStatus.should_exit_event = anyio.Event()
# Check if should_exit got set while we set up the event
if AppStatus.should_exit:
return
await AppStatus.should_exit_event.wait()
async def _ping(self, send: Send) -> None:
"""Periodically send ping messages to keep the connection alive on proxies.
- frequenccy ca every 15 seconds.
- Alternatively one can send periodically a comment line (one starting with a ':' character)
"""
while self.active:
await anyio.sleep(self._ping_interval)
sse_ping = (
self.ping_message_factory()
if self.ping_message_factory
else ServerSentEvent(
comment=f"ping - {datetime.now(timezone.utc)}", sep=self.sep
)
)
ping_bytes = ensure_bytes(sse_ping, self.sep)
logger.debug("ping: %s", ping_bytes)
async with self._send_lock:
if self.active:
await send(
{
"type": "http.response.body",
"body": ping_bytes,
"more_body": True,
}
)
async def __call__(self, scope: Scope, receive: Receive, send: Send) -> None:
"""Entrypoint for Starlette's ASGI contract. We spin up tasks:
- _stream_response to push events
- _ping to keep the connection alive
- _listen_for_exit_signal to respond to server shutdown
- _listen_for_disconnect to respond to client disconnect
"""
async with anyio.create_task_group() as task_group:
# https://trio.readthedocs.io/en/latest/reference-core.html#custom-supervisors
async def cancel_on_finish(coro: Callable[[], Awaitable[None]]):
await coro()
task_group.cancel_scope.cancel()
task_group.start_soon(cancel_on_finish, lambda: self._stream_response(send))
task_group.start_soon(cancel_on_finish, lambda: self._ping(send))
task_group.start_soon(cancel_on_finish, self._listen_for_exit_signal)
if self.data_sender_callable:
task_group.start_soon(self.data_sender_callable)
# Wait for the client to disconnect last
await cancel_on_finish(lambda: self._listen_for_disconnect(receive))
if self.background is not None:
await self.background()