from __future__ import annotations
import asyncio
import json
from asyncio import Queue as AsyncQueue
from asyncio.futures import Future
from typing import Any
from urllib.parse import urljoin
from tornado.httpserver import HTTPServer
from tornado.httputil import HTTPServerRequest
from tornado.log import enable_pretty_logging
from tornado.platform.asyncio import AsyncIOMainLoop
from tornado.web import Application, RequestHandler, StaticFileHandler
from tornado.websocket import WebSocketHandler
from tornado.wsgi import WSGIContainer
from typing_extensions import TypeAlias
from reactpy.backend._common import (
ASSETS_PATH,
CLIENT_BUILD_DIR,
MODULES_PATH,
STREAM_PATH,
CommonOptions,
read_client_index_html,
)
from reactpy.backend.types import Connection, Location
from reactpy.config import REACTPY_WEB_MODULES_DIR
from reactpy.core.hooks import ConnectionContext
from reactpy.core.hooks import use_connection as _use_connection
from reactpy.core.layout import Layout
from reactpy.core.serve import serve_layout
from reactpy.core.types import ComponentConstructor
# BackendType.Options
Options = CommonOptions
# BackendType.configure
# BackendType.create_development_app
def create_development_app() -> Application:
return Application(debug=True)
# BackendType.serve_development_app
async def serve_development_app(
app: Application,
host: str,
port: int,
started: asyncio.Event | None = None,
) -> None:
enable_pretty_logging()
AsyncIOMainLoop.current().install()
server = HTTPServer(app)
server.listen(port, host)
if started:
# at this point the server is accepting connection
started.set()
try:
# block forever - tornado has already set up its own background tasks
await asyncio.get_running_loop().create_future()
finally:
# stop accepting new connections
server.stop()
# wait for existing connections to complete
await server.close_all_connections()
[docs]def use_request() -> HTTPServerRequest:
"""Get the current ``HTTPServerRequest``"""
return use_connection().carrier
def use_connection() -> Connection[HTTPServerRequest]:
conn = _use_connection()
if not isinstance(conn.carrier, HTTPServerRequest): # nocov
msg = f"Connection has unexpected carrier {conn.carrier}. Are you running with a Flask server?"
raise TypeError(msg)
return conn
_RouteHandlerSpecs: TypeAlias = "list[tuple[str, type[RequestHandler], Any]]"
def _setup_common_routes(options: Options) -> _RouteHandlerSpecs:
return [
(
rf"{MODULES_PATH}/(.*)",
StaticFileHandler,
{"path": str(REACTPY_WEB_MODULES_DIR.current)},
),
(
rf"{ASSETS_PATH}/(.*)",
StaticFileHandler,
{"path": str(CLIENT_BUILD_DIR / "assets")},
),
] + (
[
(
r"/(.*)",
IndexHandler,
{"index_html": read_client_index_html(options)},
),
]
if options.serve_index_route
else []
)
def _add_handler(
app: Application, options: Options, handlers: _RouteHandlerSpecs
) -> None:
prefixed_handlers: list[Any] = [
(urljoin(options.url_prefix, route_pattern), *tuple(handler_info))
for route_pattern, *handler_info in handlers
]
app.add_handlers(r".*", prefixed_handlers)
def _setup_single_view_dispatcher_route(
constructor: ComponentConstructor, options: Options
) -> _RouteHandlerSpecs:
return [
(
rf"{STREAM_PATH}/(.*)",
ModelStreamHandler,
{"component_constructor": constructor, "url_prefix": options.url_prefix},
),
(
str(STREAM_PATH),
ModelStreamHandler,
{"component_constructor": constructor, "url_prefix": options.url_prefix},
),
]
[docs]class IndexHandler(RequestHandler): # type: ignore
_index_html: str
def initialize(self, index_html: str) -> None:
self._index_html = index_html
async def get(self, _: str) -> None:
self.finish(self._index_html)
[docs]class ModelStreamHandler(WebSocketHandler): # type: ignore
"""A web-socket handler that serves up a new model stream to each new client"""
_dispatch_future: Future[None]
_message_queue: AsyncQueue[str]
def initialize(
self, component_constructor: ComponentConstructor, url_prefix: str
) -> None:
self._component_constructor = component_constructor
self._url_prefix = url_prefix
[docs] async def open(self, path: str = "", *args: Any, **kwargs: Any) -> None:
message_queue: AsyncQueue[str] = AsyncQueue()
async def send(value: Any) -> None:
await self.write_message(json.dumps(value))
async def recv() -> Any:
return json.loads(await message_queue.get())
self._message_queue = message_queue
self._dispatch_future = asyncio.ensure_future(
serve_layout(
Layout(
ConnectionContext(
self._component_constructor(),
value=Connection(
scope=_FAKE_WSGI_CONTAINER.environ(self.request),
location=Location(
pathname=f"/{path[len(self._url_prefix) :]}",
search=(
f"?{self.request.query}"
if self.request.query
else ""
),
),
carrier=self.request,
),
)
),
send,
recv,
)
)
[docs] async def on_message(self, message: str | bytes) -> None:
await self._message_queue.put(
message if isinstance(message, str) else message.decode()
)
[docs] def on_close(self) -> None:
if not self._dispatch_future.done():
self._dispatch_future.cancel()
# The interface for WSGIContainer.environ changed in Tornado version 6.3 from
# a staticmethod to an instance method. Since we're not that concerned with
# the details of the WSGI app itself, we can just use a fake one.
# see: https://github.com/tornadoweb/tornado/pull/3231#issuecomment-1518957578
_FAKE_WSGI_CONTAINER = WSGIContainer(lambda *a, **kw: iter([]))