Files
mizan/backends/mizan-fastapi
Ryth Azhur 43bcf3f26f Mizan codegen substrate: Rust kernel + Rust codegen binary, JS generator deleted
The Mizan codegen substrate moves off JavaScript template-literal emission
onto a compiled Rust binary that consumes the same OpenAPI + x-mizan-* IR
the JS substrate consumed. Three structural wins fall out of one move:

1. Moat closes. The codegen logic (how `affects` becomes auto-invalidation,
   how named contexts collapse onto bundled fetches, how the registry-to-
   Provider mapping is shaped) ships compiled instead of as source bytes
   in every consumer's node_modules.

2. Pattern F (lines.push append-walls) becomes structurally unauthorable.
   The emit substrate is askama templates in templates/<target>/*.j2 —
   actual target-language files with {{ ... }} substitution markers,
   syntax-highlighted natively, type-checked against the render context
   structs at compile time. The Rust emit modules build typed render
   contexts and call .render(); no string-builder surface exists.

3. OpenAPI `default`-bearing fields now emit as non-optional in TS / Python
   / Rust — the server always populates them, so consumer code reads them
   without nullable checks. Surfaced by Blazr's typecheck on regeneration.

Layout:
  frontends/mizan-rust/        — Rust port of @mizan/base; #[cfg(feature="pyo3")]
                                 exposes PyMizanClient for the Python target.
  protocol/mizan-codegen/      — codegen binary source + askama templates.
  protocol/mizan-generate/     — npm-package shim. bin/launcher.mjs dispatches
                                 to the platform-appropriate prebuilt binary.
                                 Old generator/ JS tree deleted.
  tests/rust/                  — wire-parity drivers. drive_kernel exercises
                                 raw client.call() / fetch_context(); drive_emitted
                                 exercises the typed crate the codegen emits.
  tests/afi/afi_codegen_app.py — codegen entrypoint module (imports + registers).
  backends/mizan-fastapi/.../schema.py — adds outputNullable so the Rust
                                 codegen can wrap T | None responses in Option<T>.

Verification:
  - 20 mizan-codegen tests green (IR deserialization, byte-equivalent
    parity vs JS baseline for stage1/rust/python/react/vue/svelte,
    structural test for channels).
  - tests/rust/run_wire_parity.py — 12/12 probes green via the Rust binary
    driving the FastAPI fixture end-to-end.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-17 18:26:32 -04:00
..

mizan-fastapi

FastAPI backend adapter for the Mizan protocol. One decorator on a server function. Typed React client generated. Invalidation automatic.

Scope

mizan-fastapi targets the AFI-common subset — RPC dispatch, context bundling, JSON-body invalidation, and auth gating. Forms, Channels, Shapes, and SSR are out of scope for the FastAPI adapter — FastAPI projects use native equivalents (Pydantic, native WebSockets, ORM-of-choice, FastAPI's own SSR ecosystem).

Install

uv add mizan-fastapi

Setup

# main.py
from fastapi import FastAPI
from fastapi.exceptions import RequestValidationError

from mizan_fastapi import (
    MizanError,
    mizan_exception_handler,
    mizan_validation_handler,
    router as mizan_router,
)


app = FastAPI()
app.include_router(mizan_router, prefix="/api/mizan")
app.add_exception_handler(MizanError, mizan_exception_handler)
app.add_exception_handler(RequestValidationError, mizan_validation_handler)

The exception handlers render every error path through the Mizan envelope ({"error": {"code", "message", "details"}}) so the kernel's MizanError parses status + code on the frontend regardless of which failure happened.

Define server functions

from mizan_core.client.function import client
from mizan_core.registry import register
from pydantic import BaseModel


class EchoOutput(BaseModel):
    message: str


@client
def echo(request, text: str) -> EchoOutput:
    return EchoOutput(message=text)


register(echo, "echo")

mizan-fastapi has no auto-discovery (FastAPI doesn't have an app registry to walk). Register every @client-decorated function explicitly. A typical project keeps registrations in main.py (alongside the FastAPI app) or in a dedicated clients.py imported during startup.

@client parameters

@client                              # plain RPC function
@client(context="global")            # singleton context — fetched once, SSR-hydrated
@client(context="user")              # named context — fetched per provider mount
@client(affects="user")              # mutation — invalidates the user context
@client(affects=user_profile)        # mutation — invalidates a specific function
@client(auth=True)                   # requires authentication
@client(auth="staff")                # requires is_staff
@client(auth="superuser")            # requires is_superuser
@client(auth=lambda req: ...)        # custom predicate
@client(rev=2)                       # cache revision (busts on bump)

websocket=True, Forms, and Channels parameters are accepted by the decorator (they're a mizan-core primitive) but ignored by mizan-fastapi — those features only have effect when paired with mizan-django.

Auth integration

The executor expects request.state.user to be populated by your FastAPI middleware or dependency tree before dispatch:

from fastapi import Request


@app.middleware("http")
async def attach_user(request: Request, call_next):
    request.state.user = await resolve_user_from_token(request)
    return await call_next(request)

Where resolve_user_from_token returns either a user object with is_authenticated, is_staff, is_superuser attributes, or None for an anonymous request. The executor branches on those for auth=True, auth="staff", auth="superuser" requirements.

Generate the frontend

The codegen is mizan-generate (in protocol/mizan-generate/). Point a config at your FastAPI app and run the CLI:

// frontend/fastapi.config.mjs
import path from "path"
import { fileURLToPath } from "url"

const __dirname = path.dirname(fileURLToPath(import.meta.url))
const root = path.resolve(__dirname, "..")

export default {
    source: {
        fastapi: {
            module: "main",                           // module to import for @client side effects
            cwd: path.join(root, "backend"),          // python cwd for module resolution
            command: ["uv", "run", "python"],         // optional — defaults to ["python"]
        },
    },
    output: "src/api",
}
npx mizan-generate --config fastapi.config.mjs

The codegen drives python -m mizan_fastapi.cli <module> under the hood, then emits Stage 1 (typed callXxx/fetchXxx over the runtime kernel) + Stage 2 (<MizanContext> provider, per-context providers, use{Hook}() hooks) into src/api/.

// app.tsx
import { MizanContext } from "./api"

export default function App({ children }) {
    return <MizanContext baseUrl="/api/mizan">{children}</MizanContext>
}
// any component
import { useEcho, useCurrentUser } from "./api"

const echo = useEcho()
echo.mutate({ text: "hi" }).then(r => console.log(r.message))

const user = useCurrentUser()  // global context — auto-fetched, auto-refreshed on mutation

Running tests

uv sync --extra dev
uv run pytest

Schema export CLI

For codegen consumption (or any tooling that wants the Mizan schema):

python -m mizan_fastapi.cli <module>

Imports the named module (which must register every @client function as import-time side effects), then prints the OpenAPI schema as JSON to stdout. Mirrors mizan-django's manage.py export_mizan_schema so the codegen consumes either backend the same subprocess way.

Architecture

mizan-fastapi is one of two reference backend adapters (the other is backends/mizan-django). Both implement the same Mizan protocol on top of the shared cores/mizan-python core (@client, registry, MWT, HMAC cache keys). The AFI conformance suite at tests/afi/ gates that the two adapters emit equivalent schemas for the same registered functions. See docs/AFI_ARCHITECTURE.md.

A live e2e harness exercises this adapter end-to-end at examples/fastapi-react-site/ (real Chromium → React with generated hooks → FastAPI server, 14/14 Playwright tests).