Source code for litestar_vite.codegen._export

"""Unified asset export pipeline for deterministic code generation.

This module provides a single entry point for exporting all integration artifacts:
- openapi.json (OpenAPI schema with Inertia types registered)
- routes.json (route metadata)
- routes.ts (Ziggy-style typed routes)
- inertia-pages.json (Inertia page props metadata)

Both CLI and Plugin should call this function to guarantee byte-identical output.
"""

from dataclasses import dataclass, field
from functools import partial
from pathlib import Path
from typing import TYPE_CHECKING, Any

if TYPE_CHECKING:
    from collections.abc import Callable

    from litestar import Litestar

    from litestar_vite.config import TypeGenConfig, ViteConfig


[docs] @dataclass class ExportResult: """Result of the export operation.""" exported_files: list[str] = field(default_factory=list) # pyright: ignore[reportUnknownVariableType] """Files that were written (content changed).""" unchanged_files: list[str] = field(default_factory=list) # pyright: ignore[reportUnknownVariableType] """Files that were skipped (content unchanged).""" openapi_schema: "dict[str, Any] | None" = None """The OpenAPI schema dict (for downstream use)."""
def fmt_path(path: Path) -> str: """Format path for display, using relative path when possible. Returns: The result. """ try: return str(path.relative_to(Path.cwd())) except ValueError: return str(path)
[docs] def export_integration_assets( app: "Litestar", config: "ViteConfig", *, serializer: "Callable[[Any], bytes] | None" = None ) -> ExportResult: """Export all integration artifacts with deterministic output. This is the single source of truth for code generation. Both CLI commands and Plugin startup should call this function to ensure byte-identical output. The export order is critical: 1. Register Inertia page prop types in OpenAPI schema (mutates schema_dict) 2. Export openapi.json (now includes session prop types) 3. Export routes.json (uses schema for component refs) 4. Export routes.ts (if enabled) 5. Export inertia-pages.json (if enabled) Args: app: The Litestar application instance. config: The ViteConfig instance. serializer: Optional custom serializer for OpenAPI schema encoding. Returns: ExportResult with lists of exported and unchanged files. """ from litestar._openapi.plugin import OpenAPIPlugin from litestar.serialization import encode_json, get_serializer from litestar_vite.codegen._inertia import generate_inertia_pages_json from litestar_vite.config import InertiaConfig, InertiaTypeGenConfig, TypeGenConfig result = ExportResult() if not isinstance(config.types, TypeGenConfig): return result types_config = config.types # Check if OpenAPI is available openapi_plugin = next((p for p in app.plugins._plugins if isinstance(p, OpenAPIPlugin)), None) # pyright: ignore[reportPrivateUsage] has_openapi = openapi_plugin is not None and openapi_plugin._openapi_config is not None # pyright: ignore[reportPrivateUsage] if not has_openapi: return result # Get serializer for OpenAPI encoding if serializer is None: encoders: Any try: encoders = app.type_encoders # pyright: ignore[reportUnknownMemberType] except AttributeError: encoders = None serializer = partial(encode_json, serializer=get_serializer(encoders if isinstance(encoders, dict) else None)) # pyright: ignore[reportUnknownArgumentType] # Step 1: Get OpenAPI schema and register Inertia types schema_dict = app.openapi_schema.to_schema() # Register Inertia page prop types in OpenAPI schema BEFORE exporting # This ensures types like EmailSent, NoProps, CurrentTeam are included inertia_pages_data: dict[str, Any] | None = None if isinstance(config.inertia, InertiaConfig) and types_config.generate_page_props: inertia_type_gen = config.inertia.type_gen or InertiaTypeGenConfig() inertia_pages_data = generate_inertia_pages_json( app, openapi_schema=schema_dict, include_default_auth=inertia_type_gen.include_default_auth, include_default_flash=inertia_type_gen.include_default_flash, inertia_config=config.inertia, types_config=types_config, ) result.openapi_schema = schema_dict # Step 2: Export openapi.json export_openapi(schema_dict=schema_dict, types_config=types_config, serializer=serializer, result=result) # Step 3: Export routes.json (always pass openapi_schema for consistency) export_routes_json(app=app, types_config=types_config, openapi_schema=schema_dict, result=result) # Step 4: Export routes.ts (if enabled) if types_config.generate_routes: export_routes_ts(app=app, types_config=types_config, openapi_schema=schema_dict, result=result) # Step 5: Export inertia-pages.json (if enabled) if ( isinstance(config.inertia, InertiaConfig) and types_config.generate_page_props and types_config.page_props_path and inertia_pages_data is not None ): export_inertia_pages(pages_data=inertia_pages_data, types_config=types_config, result=result) return result
def export_openapi( *, schema_dict: "dict[str, Any]", types_config: "TypeGenConfig", serializer: "Callable[[Any], bytes]", result: ExportResult, ) -> None: """Export OpenAPI schema to file.""" from litestar_vite.codegen._utils import encode_deterministic_json, write_if_changed openapi_path = types_config.openapi_path if openapi_path is None: openapi_path = types_config.output / "openapi.json" schema_content = encode_deterministic_json(schema_dict, serializer=serializer) if write_if_changed(openapi_path, schema_content): result.exported_files.append(f"openapi: {fmt_path(openapi_path)}") else: result.unchanged_files.append("openapi.json") def export_routes_json( *, app: "Litestar", types_config: "TypeGenConfig", openapi_schema: "dict[str, Any]", result: ExportResult ) -> None: """Export routes metadata to JSON file.""" from litestar_vite.codegen._routes import generate_routes_json from litestar_vite.codegen._utils import encode_deterministic_json, write_if_changed try: from litestar import __version__ as _version litestar_version = str(_version) except ImportError: litestar_version = "unknown" routes_path = types_config.routes_path if routes_path is None: routes_path = types_config.output / "routes.json" # Always pass openapi_schema for consistent output between CLI and plugin routes_data = generate_routes_json(app, include_components=True, openapi_schema=openapi_schema) routes_data["litestar_version"] = litestar_version routes_content = encode_deterministic_json(routes_data) if write_if_changed(routes_path, routes_content): result.exported_files.append(fmt_path(routes_path)) else: result.unchanged_files.append("routes.json") def export_routes_ts( *, app: "Litestar", types_config: "TypeGenConfig", openapi_schema: "dict[str, Any]", result: ExportResult ) -> None: """Export typed routes TypeScript file.""" from litestar_vite.codegen._routes import generate_routes_ts from litestar_vite.codegen._utils import write_if_changed routes_ts_path = types_config.routes_ts_path if routes_ts_path is None: routes_ts_path = types_config.output / "routes.ts" # Always pass openapi_schema for consistent output routes_ts_content = generate_routes_ts(app, openapi_schema=openapi_schema, global_route=types_config.global_route) if write_if_changed(routes_ts_path, routes_ts_content): result.exported_files.append(fmt_path(routes_ts_path)) else: result.unchanged_files.append("routes.ts") def export_inertia_pages(*, pages_data: "dict[str, Any]", types_config: "TypeGenConfig", result: ExportResult) -> None: """Export Inertia pages metadata to JSON file.""" from litestar_vite.codegen._utils import encode_deterministic_json, write_if_changed page_props_path = types_config.page_props_path if page_props_path is None: return pages_content = encode_deterministic_json(pages_data) if write_if_changed(page_props_path, pages_content): result.exported_files.append(fmt_path(page_props_path)) else: result.unchanged_files.append("inertia-pages.json")