from __future__ import annotations
import asyncio
import os
from collections.abc import Awaitable, Sequence
from dataclasses import dataclass
from pathlib import Path, PurePosixPath
from typing import TYPE_CHECKING, Any, cast
from reactpy import __file__ as _reactpy_file_path
from reactpy import html
from reactpy.config import REACTPY_WEB_MODULES_DIR
from reactpy.core.types import VdomDict
from reactpy.utils import vdom_to_html
if TYPE_CHECKING:
import uvicorn
from asgiref.typing import ASGIApplication
PATH_PREFIX = PurePosixPath("/_reactpy")
MODULES_PATH = PATH_PREFIX / "modules"
ASSETS_PATH = PATH_PREFIX / "assets"
STREAM_PATH = PATH_PREFIX / "stream"
CLIENT_BUILD_DIR = Path(_reactpy_file_path).parent / "static"
[docs]async def serve_with_uvicorn(
app: ASGIApplication | Any,
host: str,
port: int,
started: asyncio.Event | None,
) -> None:
"""Run a development server for an ASGI application"""
import uvicorn
server = uvicorn.Server(
uvicorn.Config(
app,
host=host,
port=port,
loop="asyncio",
)
)
server.config.setup_event_loop()
coros: list[Awaitable[Any]] = [server.serve()]
# If a started event is provided, then use it signal based on `server.started`
if started:
coros.append(_check_if_started(server, started))
try:
await asyncio.gather(*coros)
finally:
# Since we aren't using the uvicorn's `run()` API, we can't guarantee uvicorn's
# order of operations. So we need to make sure `shutdown()` always has an initialized
# list of `self.servers` to use.
if not hasattr(server, "servers"): # nocov
server.servers = []
await asyncio.wait_for(server.shutdown(), timeout=3)
async def _check_if_started(server: uvicorn.Server, started: asyncio.Event) -> None:
while not server.started:
await asyncio.sleep(0.2)
started.set()
[docs]def safe_client_build_dir_path(path: str) -> Path:
"""Prevent path traversal out of :data:`CLIENT_BUILD_DIR`"""
return traversal_safe_path(
CLIENT_BUILD_DIR, *("index.html" if path in {"", "/"} else path).split("/")
)
[docs]def safe_web_modules_dir_path(path: str) -> Path:
"""Prevent path traversal out of :data:`reactpy.config.REACTPY_WEB_MODULES_DIR`"""
return traversal_safe_path(REACTPY_WEB_MODULES_DIR.current, *path.split("/"))
[docs]def traversal_safe_path(root: str | Path, *unsafe: str | Path) -> Path:
"""Raise a ``ValueError`` if the ``unsafe`` path resolves outside the root dir."""
root = os.path.abspath(root)
# Resolve relative paths but not symlinks - symlinks should be ok since their
# presence and where they point is under the control of the developer.
path = os.path.abspath(os.path.join(root, *unsafe))
if os.path.commonprefix([root, path]) != root:
# If the common prefix is not root directory we resolved outside the root dir
msg = "Unsafe path"
raise ValueError(msg)
return Path(path)
def read_client_index_html(options: CommonOptions) -> str:
return (
(CLIENT_BUILD_DIR / "index.html")
.read_text()
.format(__head__=vdom_head_elements_to_html(options.head))
)
def vdom_head_elements_to_html(head: Sequence[VdomDict] | VdomDict | str) -> str:
if isinstance(head, str):
return head
elif isinstance(head, dict):
if head.get("tagName") == "head":
head = cast(VdomDict, {**head, "tagName": ""})
return vdom_to_html(head)
else:
return vdom_to_html(html._(*head))
[docs]@dataclass
class CommonOptions:
"""Options for ReactPy's built-in backed server implementations"""
head: Sequence[VdomDict] | VdomDict | str = (html.title("ReactPy"),)
"""Add elements to the ``<head>`` of the application.
For example, this can be used to customize the title of the page, link extra
scripts, or load stylesheets.
"""
url_prefix: str = ""
"""The URL prefix where ReactPy resources will be served from"""
serve_index_route: bool = True
"""Automatically generate and serve the index route (``/``)"""
def __post_init__(self) -> None:
if self.url_prefix and not self.url_prefix.startswith("/"):
msg = "Expected 'url_prefix' to start with '/'"
raise ValueError(msg)