Tauri now joins FastAPI/Django/axum as a first-class Mizan backend. The
React frontend calls Mizan-registered functions through Tauri's IPC
with the same {result, invalidate, merge} envelope the HTTP path uses;
the schema flows Pydantic → decoru → Rust → KDL → TS in one
mizan-generate invocation.
New packages:
* backends/mizan-tauri — Tauri plugin exposing a single `mizan_invoke`
command that routes through mizan-core's FUNCTIONS / CONTEXTS
registries. No per-function tauri::command; the linkme slice IS the
dispatch table.
* frontends/mizan-tauri-transport — TS package exporting
tauriTransport() that wraps invoke('plugin:mizan|mizan_invoke', ...)
and re-shapes errors into MizanError. Pairs with mizan-tauri.
@mizan/base — pluggable transport:
* Adds MizanTransport interface + transport config field.
* Existing fetch-based body factored into httpTransport() (default).
* mizanCall/mizanFetch delegate to config.transport; merge/invalidate
side-effects stay in the kernel (transport-agnostic).
* Consumers swap via configure({ transport: tauriTransport() }).
mizan-codegen — Rust source + Pydantic pre-step:
* [source.rust] runs a Cargo bin (cargo run --bin <name>) and parses
KDL from stdout. The bin uses mizan_core::build_ir() after
force-linking the consumer's #[derive(Mizan)] / #[mizan::client]
registrations.
* [source.rust.pydantic] is an optional pre-step that pipes an
embedded Python bridge (scripts/run_decoru.py) to python and writes
decoru-emitted Rust types into the consumer crate. The bridge
auto-discovers BaseModel subclasses AND Enum subclasses
(last-variant-is-default convention so decoru's impl Default keeps
compiling against enum-typed fields without explicit Pydantic
defaults).
* Pure-Rust usage stays intact — omit pydantic block and write Rust
types by hand.
mizan-macros:
* #[mizan::client] now supports Result<T, MizanError> returns. The
dispatch wrapper `?`-unwraps the user fn so server-side errors
surface as the protocol's standard {code, message, details?}
envelope; T-returning functions stay unchanged.
* #[derive(Mizan)] strips the r# raw-identifier prefix and honors
field-level #[serde(rename = "...")] when emitting IR field names.
Matches serde's wire shape — fixes IR-vs-JSON drift for Rust-keyword
fields (e.g. `r#type` → `type`).
react.tsx template:
* Conditionally emits context-related imports / useContextSubscription
helper based on has_global || !named_contexts.is_empty(). Consumers
without contexts (mutation/RPC-only apps like claude-manage) no
longer get dead imports that trip noUnusedLocals.
Verified end-to-end: cargo build clean across mizan-tauri,
mizan-codegen, AFI rust_app; AFI three-way KDL parity tests pass;
claude-manage migration drives the full stack (Pydantic schema →
generated TS api → Tauri-IPC transport → mizan-core dispatch).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
190 lines
6.1 KiB
Django/Jinja
190 lines
6.1 KiB
Django/Jinja
'use client'
|
|
|
|
// AUTO-GENERATED by mizan — do not edit
|
|
|
|
{% set has_contexts = has_global || !named_contexts.is_empty() -%}
|
|
import {
|
|
{% if has_contexts -%}
|
|
createContext,
|
|
{% endif -%}
|
|
useCallback,
|
|
{% if has_contexts -%}
|
|
useContext,
|
|
useEffect,
|
|
{% endif -%}
|
|
useRef,
|
|
useState,
|
|
{% if has_contexts -%}
|
|
useSyncExternalStore,
|
|
{% endif -%}
|
|
type ReactNode,
|
|
} from 'react'
|
|
import {
|
|
configure,
|
|
mizanCall,
|
|
mizanFetch,
|
|
{% if has_contexts -%}
|
|
registerContext,
|
|
type ContextState,
|
|
{% endif -%}
|
|
} from '@mizan/base'
|
|
|
|
{% if !stage1_imports.is_empty() -%}
|
|
import { {{ stage1_imports|join(", ") }} } from './index'
|
|
|
|
{% endif -%}
|
|
{% if has_contexts -%}
|
|
// Internal — runs inside a Provider, registers with the kernel exactly once.
|
|
function useContextSubscription<T>(
|
|
name: string,
|
|
params: Record<string, any>,
|
|
fetchFn: () => Promise<T>,
|
|
initialData?: T,
|
|
): ContextState<T> {
|
|
const ref = useRef<ReturnType<typeof registerContext> | null>(null)
|
|
if (!ref.current) {
|
|
ref.current = registerContext(name, params, fetchFn, initialData)
|
|
}
|
|
const handle = ref.current
|
|
|
|
useEffect(() => {
|
|
if (handle.getState().status === 'idle') handle.refetch()
|
|
return () => handle.unregister()
|
|
}, [handle])
|
|
|
|
return useSyncExternalStore(handle.subscribe, handle.getState, handle.getState)
|
|
}
|
|
{% endif %}
|
|
|
|
// Internal — wraps an imperative call() with isPending / error state.
|
|
interface MutationHook<TArgs, TResult> {
|
|
mutate: (args: TArgs) => Promise<TResult>
|
|
isPending: boolean
|
|
error: Error | null
|
|
}
|
|
|
|
function useMutation<TArgs, TResult>(
|
|
callFn: (args: TArgs) => Promise<TResult>,
|
|
): MutationHook<TArgs, TResult> {
|
|
const [isPending, setIsPending] = useState(false)
|
|
const [error, setError] = useState<Error | null>(null)
|
|
|
|
const mutate = useCallback(async (args: TArgs) => {
|
|
setIsPending(true)
|
|
setError(null)
|
|
try {
|
|
return await callFn(args)
|
|
} catch (e) {
|
|
setError(e as Error)
|
|
throw e
|
|
} finally {
|
|
setIsPending(false)
|
|
}
|
|
}, [callFn])
|
|
|
|
return { mutate, isPending, error }
|
|
}
|
|
{% if has_global %}
|
|
// ── Global Context ──
|
|
|
|
const GlobalCtx = createContext<ContextState<GlobalContextData> | null>(null)
|
|
|
|
export function GlobalContextProvider({ children }: { children: ReactNode }) {
|
|
const ssrData = typeof window !== 'undefined' ? (window as any).__MIZAN_SSR_DATA__ : undefined
|
|
const state = useContextSubscription('global', {}, () => fetchGlobalContext({} as any), ssrData)
|
|
return <GlobalCtx.Provider value={state}>{children}</GlobalCtx.Provider>
|
|
}
|
|
|
|
export function useGlobalContext(): ContextState<GlobalContextData> {
|
|
const ctx = useContext(GlobalCtx)
|
|
if (!ctx) throw new Error('useGlobalContext requires <MizanContext> or <GlobalContextProvider>')
|
|
return ctx
|
|
}
|
|
{% for fn in global_fns %}
|
|
export function use{{ fn.pascal }}(): {{ fn.output_type }} | null {
|
|
return useGlobalContext().data?.{{ fn.name }} ?? null
|
|
}
|
|
{% endfor -%}
|
|
{% endif -%}
|
|
{% for ctx in named_contexts %}
|
|
// ── {{ ctx.pascal }} Context ──
|
|
|
|
const {{ ctx.pascal }}Ctx = createContext<ContextState<{{ ctx.pascal }}ContextData> | null>(null)
|
|
|
|
{% if ctx.has_params -%}
|
|
export function {{ ctx.pascal }}Context({ children, ...params }: {{ ctx.pascal }}ContextParams & { children: ReactNode }) {
|
|
const state = useContextSubscription('{{ ctx.name }}', params, () => fetch{{ ctx.pascal }}Context(params))
|
|
return <{{ ctx.pascal }}Ctx.Provider value={state}>{children}</{{ ctx.pascal }}Ctx.Provider>
|
|
}
|
|
{% else -%}
|
|
export function {{ ctx.pascal }}Context({ children }: { children: ReactNode }) {
|
|
const state = useContextSubscription('{{ ctx.name }}', {}, () => fetch{{ ctx.pascal }}Context({} as any))
|
|
return <{{ ctx.pascal }}Ctx.Provider value={state}>{children}</{{ ctx.pascal }}Ctx.Provider>
|
|
}
|
|
{% endif %}
|
|
export function use{{ ctx.pascal }}Context(): ContextState<{{ ctx.pascal }}ContextData> {
|
|
const ctx = useContext({{ ctx.pascal }}Ctx)
|
|
if (!ctx) throw new Error('use{{ ctx.pascal }}Context requires <{{ ctx.pascal }}Context>')
|
|
return ctx
|
|
}
|
|
{% for fn in ctx.fns %}
|
|
export function use{{ fn.pascal }}(): {{ fn.output_type }} | null {
|
|
return use{{ ctx.pascal }}Context().data?.{{ fn.name }} ?? null
|
|
}
|
|
{% endfor -%}
|
|
{% endfor -%}
|
|
{% for call in calls %}
|
|
{% if call.has_input -%}
|
|
export function use{{ call.pascal }}() {
|
|
return useMutation<Parameters<typeof call{{ call.pascal }}>[0], Awaited<ReturnType<typeof call{{ call.pascal }}>>>(call{{ call.pascal }})
|
|
}
|
|
{% else -%}
|
|
export function use{{ call.pascal }}() {
|
|
return useMutation<void, Awaited<ReturnType<typeof call{{ call.pascal }}>>>(() => call{{ call.pascal }}() as any)
|
|
}
|
|
{% endif -%}
|
|
{% endfor %}
|
|
// ── MizanContext root provider ──
|
|
|
|
export interface MizanContextProps {
|
|
/** Base URL for protocol endpoints. Defaults to "/api/mizan". */
|
|
baseUrl?: string
|
|
/** Set to `false` for backends without a `/session/` endpoint (e.g. FastAPI). */
|
|
session?: boolean
|
|
children: ReactNode
|
|
}
|
|
|
|
/**
|
|
* Root provider — calls configure() once and mounts the global context (if defined).
|
|
* Must wrap any component using Mizan-generated hooks.
|
|
*/
|
|
export function MizanContext({ baseUrl, session, children }: MizanContextProps) {
|
|
const configured = useRef(false)
|
|
if (!configured.current) {
|
|
const opts: Parameters<typeof configure>[0] = {}
|
|
if (baseUrl !== undefined) opts.baseUrl = baseUrl
|
|
if (session !== undefined) opts.session = session
|
|
if (Object.keys(opts).length > 0) configure(opts)
|
|
configured.current = true
|
|
}
|
|
{%- if has_global %}
|
|
return <GlobalContextProvider>{children}</GlobalContextProvider>
|
|
{%- else %}
|
|
return <>{children}</>
|
|
{%- endif %}
|
|
}
|
|
|
|
// ── Imperative escape hatch ──
|
|
|
|
/**
|
|
* Returns the imperative kernel API. For test harnesses or rare cases where
|
|
* a typed generated hook does not fit. Most app code should use the typed hooks.
|
|
*/
|
|
export function useMizan() {
|
|
return { call: mizanCall, fetch: mizanFetch }
|
|
}
|
|
|
|
export type { ContextState } from '@mizan/base'
|
|
export { configure, initSession, MizanError } from '@mizan/base'
|
|
|