Source code for litestar_vite.inertia.request
from functools import cached_property
from typing import TYPE_CHECKING, Optional
from urllib.parse import unquote
from litestar import Request
from litestar.connection.base import (
AuthT,
StateT,
UserT,
empty_receive,
empty_send,
)
from litestar_vite.inertia._utils import InertiaHeaders
__all__ = ("InertiaDetails", "InertiaRequest")
if TYPE_CHECKING:
from litestar.types import Receive, Scope, Send
[docs]
class InertiaDetails:
"""InertiaDetails holds all the values sent by Inertia client in headers and provide convenient properties."""
[docs]
def __init__(self, request: "Request[UserT, AuthT, StateT]") -> None:
"""Initialize :class:`InertiaDetails`"""
self.request = request
def _get_header_value(self, name: "InertiaHeaders") -> "Optional[str]":
"""Parse request header
Check for uri encoded header and unquotes it in readable format.
Args:
name: The header name.
Returns:
The header value.
"""
if value := self.request.headers.get(name.value.lower()):
is_uri_encoded = self.request.headers.get(f"{name.value.lower()}-uri-autoencoded") == "true"
return unquote(value) if is_uri_encoded else value
return None
def _get_route_component(self) -> "Optional[str]":
"""Get the route component.
Checks for the `component` key within the route handler configuration.
Args:
request: The request object.
Returns:
The route component.
"""
rh = self.request.scope.get("route_handler") # pyright: ignore[reportUnknownMemberType]
if rh:
return rh.opt.get("component")
return None
[docs]
def __bool__(self) -> bool:
"""Check if request is sent by an Inertia client.
Returns:
True if the request is sent by an Inertia client, False otherwise.
"""
return self._get_header_value(InertiaHeaders.ENABLED) == "true"
@cached_property
def route_component(self) -> "Optional[str]":
"""Get the route component.
Returns:
The route component.
"""
return self._get_route_component()
@cached_property
def partial_component(self) -> "Optional[str]":
"""Get the partial component.
Returns:
The partial component.
"""
return self._get_header_value(InertiaHeaders.PARTIAL_COMPONENT)
@cached_property
def partial_data(self) -> "Optional[str]":
"""Get the partial data.
Returns:
The partial data.
"""
return self._get_header_value(InertiaHeaders.PARTIAL_DATA)
@cached_property
def referer(self) -> "Optional[str]":
"""Get the referer.
Returns:
The referer.
"""
return self._get_header_value(InertiaHeaders.REFERER)
@cached_property
def is_partial_render(self) -> bool:
"""Check if the request is a partial render.
Returns:
True if the request is a partial render, False otherwise.
"""
return bool(self.partial_component == self.route_component and self.partial_data)
@cached_property
def partial_keys(self) -> list[str]:
"""Get the partial keys.
Returns:
The partial keys.
"""
return self.partial_data.split(",") if self.partial_data is not None else []
[docs]
class InertiaRequest(Request[UserT, AuthT, StateT]):
"""Inertia Request class to work with Inertia client."""
__slots__ = ("inertia",)
[docs]
def __init__(self, scope: "Scope", receive: "Receive" = empty_receive, send: "Send" = empty_send) -> None:
"""Initialize :class:`InertiaRequest`"""
super().__init__(scope=scope, receive=receive, send=send)
self.inertia = InertiaDetails(self)
@property
def is_inertia(self) -> bool:
"""True if the request contained inertia headers."""
return bool(self.inertia)
@property
def inertia_enabled(self) -> bool:
"""True if the route handler contains an inertia enabled configuration."""
return bool(self.inertia.route_component is not None)
@property
def is_partial_render(self) -> bool:
"""True if the route handler contains an inertia enabled configuration."""
return self.inertia.is_partial_render
@property
def partial_keys(self) -> "set[str]":
"""True if the route handler contains an inertia enabled configuration."""
return set(self.inertia.partial_keys)