@devlusoft/devix 0.3.2 → 0.4.0-beta.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (53) hide show
  1. package/dist/cli/build.js +2 -4
  2. package/dist/cli/build.js.map +2 -2
  3. package/dist/cli/dev.js +12 -14
  4. package/dist/cli/dev.js.map +3 -3
  5. package/dist/cli/generate.js +1 -3
  6. package/dist/cli/generate.js.map +2 -2
  7. package/dist/cli/index.js +6 -8
  8. package/dist/cli/index.js.map +3 -3
  9. package/dist/cli/start.js +1 -1
  10. package/dist/cli/start.js.map +2 -2
  11. package/dist/runtime/context.d.ts +6 -1
  12. package/dist/runtime/context.js +1 -1
  13. package/dist/runtime/context.js.map +2 -2
  14. package/dist/runtime/create-handler.d.ts +2 -2
  15. package/dist/runtime/create-handler.js.map +1 -1
  16. package/dist/runtime/fetch.d.ts +10 -1
  17. package/dist/runtime/fetch.js.map +2 -2
  18. package/dist/runtime/head.d.ts +4 -0
  19. package/dist/runtime/head.js +1 -1
  20. package/dist/runtime/head.js.map +3 -3
  21. package/dist/runtime/index.d.ts +4 -3
  22. package/dist/runtime/index.js +1 -1
  23. package/dist/runtime/index.js.map +3 -3
  24. package/dist/runtime/router-provider.d.ts +5 -2
  25. package/dist/runtime/router-provider.js +1 -1
  26. package/dist/runtime/router-provider.js.map +3 -3
  27. package/dist/runtime/server-app.d.ts +15 -0
  28. package/dist/runtime/server-app.js +2 -0
  29. package/dist/runtime/server-app.js.map +7 -0
  30. package/dist/server/api.js +1 -1
  31. package/dist/server/api.js.map +3 -3
  32. package/dist/server/handler-store.js +1 -1
  33. package/dist/server/handler-store.js.map +3 -3
  34. package/dist/server/public-index.js +1 -1
  35. package/dist/server/public-index.js.map +3 -3
  36. package/dist/server/render.d.ts +29 -1
  37. package/dist/server/render.js +1 -1
  38. package/dist/server/render.js.map +4 -4
  39. package/dist/server/routes.js +1 -1
  40. package/dist/server/routes.js.map +2 -2
  41. package/dist/server/types.d.ts +11 -8
  42. package/dist/types.d.ts +9 -2
  43. package/dist/utils/banner.js +1 -1
  44. package/dist/utils/banner.js.map +3 -3
  45. package/dist/utils/response.d.ts +14 -1
  46. package/dist/utils/response.js +1 -1
  47. package/dist/utils/response.js.map +3 -3
  48. package/dist/vite/codegen/routes-dts.js +1 -3
  49. package/dist/vite/codegen/routes-dts.js.map +2 -2
  50. package/dist/vite/codegen/scan-api.js.map +1 -1
  51. package/dist/vite/index.js +1 -3
  52. package/dist/vite/index.js.map +2 -2
  53. package/package.json +2 -1
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/runtime/router-provider.tsx", "../../src/runtime/head.tsx", "../../src/runtime/context.tsx", "../../src/runtime/error-boundary.tsx"],
4
- "sourcesContent": ["import {ComponentType, ReactNode, useCallback, useContext, useEffect, useRef, useState} from \"react\";\nimport {RouterContext} from 'virtual:devix/context'\nimport {ErrorProps, LayoutProps, PageProps} from \"../server/types\";\nimport {Metadata, Viewport} from \"../types\";\nimport {getDefaultErrorPage, loadErrorPage, matchClientRoute} from \"virtual:devix/client-routes\";\nimport {buildHeadNodes} from \"./head\";\nimport {PageMetaContext, RouteDataContext} from \"./context\";\nimport {DevixErrorBoundary} from \"./error-boundary\";\n\ninterface RouteState {\n pathname: string\n params: Record<string, string>\n loaderData: unknown\n layoutsData: unknown[]\n Page: ComponentType<PageProps>\n layouts: ComponentType<LayoutProps>[]\n metadata: Metadata | null\n viewport?: Viewport\n pendingError?: ErrorProps\n ErrorPage?: ComponentType<ErrorProps>\n}\n\nexport function useRouter() {\n return useContext(RouterContext)\n}\n\nexport function useNavigate() {\n const ctx = useContext(RouterContext)\n if (!ctx) throw new Error(\"useNavigate must be used within a RouterProvider\")\n return ctx.navigate\n}\n\nexport function useParams<T extends Record<string, string>>() {\n const ctx = useContext(RouteDataContext)\n if (!ctx) throw new Error(\"useParams must be used within a route or layout\")\n return ctx.params as T\n}\n\ntype LoaderReturnType<T> = T extends (...args: any[]) => Promise<infer R>\n ? R\n : T extends (...args: any[]) => infer R\n ? R\n : T\n\nexport function useLoaderData<T>() {\n const ctx = useContext(RouteDataContext)\n if (!ctx) throw new Error(\"useLoaderData must be used within a route or layout\")\n return ctx.loaderData as LoaderReturnType<T>\n}\n\n\ninterface RouterProviderProps {\n initialData: unknown\n initialParams: Record<string, string>\n initialPage: ComponentType<PageProps>\n initialLayouts?: ComponentType<LayoutProps>[]\n initialLayoutsData?: unknown[]\n initialMeta?: Metadata | null\n initialViewport?: Viewport\n initialError?: ErrorProps\n initialErrorPage?: ComponentType<ErrorProps>\n clientEntry: string\n}\n\nexport function RouterProvider({\n initialData,\n initialParams,\n initialPage,\n initialLayouts = [],\n initialLayoutsData = [],\n initialMeta,\n initialViewport,\n initialError,\n initialErrorPage,\n clientEntry,\n }: RouterProviderProps) {\n\n const [state, setState] = useState<RouteState>({\n pathname: window.location.pathname,\n params: initialParams,\n loaderData: initialData,\n layoutsData: initialLayoutsData,\n Page: initialPage,\n layouts: initialLayouts,\n metadata: initialMeta ?? null,\n viewport: initialViewport,\n pendingError: initialError,\n ErrorPage: initialErrorPage,\n })\n\n const navigatingRef = useRef<AbortController | null>(null)\n const [isNavigating, setIsNavigating] = useState(false)\n\n const loadRoute = useCallback(async (to: string, controller: AbortController) => {\n const pathname = to.split('?')[0]\n const matched = matchClientRoute(pathname)\n if (!matched) {\n const ErrorPage = await loadErrorPage() ?? getDefaultErrorPage()\n setState(prev => ({\n ...prev,\n pathname: pathname,\n pendingError: {statusCode: 404, message: 'Not found'},\n ErrorPage: ErrorPage ?? undefined,\n }))\n return\n }\n\n const [pageMod, ...layoutMods] = await Promise.all([\n matched.load(),\n ...matched.loadLayouts.map(l => l()),\n ])\n\n if (controller.signal.aborted) return\n if (!pageMod.default) return\n\n const dataRes = await fetch(`/_data${to}`, {\n headers: {Accept: 'application/json'},\n signal: controller.signal,\n })\n\n if (controller.signal.aborted) return\n\n if (!dataRes.ok) {\n if (dataRes.status === 404) {\n window.location.href = to\n return\n }\n console.error(`/_data${to} returned ${dataRes.status}`)\n return\n }\n\n const data = await dataRes.json()\n\n window.scrollTo(0, 0)\n setState({\n pathname,\n params: data.params ?? {},\n loaderData: data.loaderData,\n layoutsData: (data.layouts ?? []).map((l: any) => l.loaderData),\n Page: pageMod.default,\n layouts: layoutMods.map(m => m.default),\n metadata: data.metadata ?? null,\n viewport: data.viewport,\n })\n }, [])\n\n const navigate = useCallback(async (to: string) => {\n navigatingRef.current?.abort()\n const controller = new AbortController()\n navigatingRef.current = controller\n\n setIsNavigating(true)\n try {\n window.history.pushState(null, \"\", to)\n await loadRoute(to, controller)\n } finally {\n if (!controller.signal.aborted) setIsNavigating(false)\n }\n }, [loadRoute])\n\n useEffect(() => {\n const handlePop = () => {\n navigatingRef.current?.abort()\n const controller = new AbortController()\n navigatingRef.current = controller\n\n const to = window.location.pathname + window.location.search\n loadRoute(to, controller).catch(err => {\n if (err.name !== 'AbortError') console.error('[router] popstate error:', err)\n })\n }\n window.addEventListener(\"popstate\", handlePop)\n return () => window.removeEventListener(\"popstate\", handlePop)\n }, [loadRoute])\n\n let content: ReactNode\n\n if (state.pendingError) {\n content = state.ErrorPage\n ? <state.ErrorPage {...state.pendingError} />\n : <h1>{state.pendingError.statusCode}</h1>\n } else {\n let tree: ReactNode = (\n <RouteDataContext value={{loaderData: state.loaderData, params: state.params}}>\n <state.Page data={state.loaderData} params={state.params} url={state.pathname}/>\n </RouteDataContext>\n )\n\n for (let i = state.layouts.length - 1; i >= 0; i--) {\n const Layout = state.layouts[i]\n const layoutData = state.layoutsData[i]\n tree = (\n <RouteDataContext value={{loaderData: layoutData, params: state.params}}>\n <Layout data={layoutData} params={state.params}>{tree}</Layout>\n </RouteDataContext>\n )\n }\n\n content = (\n <DevixErrorBoundary key={state.pathname} ErrorPage={state.ErrorPage}>\n {tree}\n </DevixErrorBoundary>\n )\n }\n\n return (\n <PageMetaContext value={{\n metadata: state.metadata,\n viewport: state.viewport,\n clientEntry,\n }}>\n {state.metadata && buildHeadNodes(state.metadata, state.viewport)}\n <RouterContext value={{...state, isNavigating, navigate}}>\n {content}\n </RouterContext>\n </PageMetaContext>\n )\n}", "import {Metadata, Viewport} from \"../types\";\nimport {ReactNode} from \"react\";\n\ntype MetaTag =\n | { tag: 'title'; children: string }\n | { tag: 'meta'; name?: string; property?: string; content: string }\n | { tag: 'link'; rel: string; href: string; hrefLang?: string }\n\nfunction collectTags(metadata: Metadata, viewport?: Viewport): MetaTag[] {\n const tags: MetaTag[] = []\n\n if (metadata.title)\n tags.push({tag: 'title', children: metadata.title})\n if (metadata.description)\n tags.push({tag: 'meta', name: 'description', content: metadata.description})\n if (metadata.keywords?.length)\n tags.push({tag: 'meta', name: 'keywords', content: metadata.keywords.join(', ')})\n\n const ogTitle = metadata.og?.title ?? metadata.title\n if (ogTitle) tags.push({tag: 'meta', property: 'og:title', content: ogTitle})\n const ogDesc = metadata.og?.description ?? metadata.description\n if (ogDesc) tags.push({tag: 'meta', property: 'og:description', content: ogDesc})\n if (metadata.og?.image) tags.push({tag: 'meta', property: 'og:image', content: metadata.og.image})\n if (metadata.og?.type) tags.push({tag: 'meta', property: 'og:type', content: metadata.og.type})\n if (metadata.og?.url) tags.push({tag: 'meta', property: 'og:url', content: metadata.og.url})\n\n const twTitle = metadata.twitter?.title ?? metadata.title\n if (twTitle) tags.push({tag: 'meta', name: 'twitter:title', content: twTitle})\n const twDesc = metadata.twitter?.description ?? metadata.description\n if (twDesc) tags.push({tag: 'meta', name: 'twitter:description', content: twDesc})\n if (metadata.twitter?.card) tags.push({\n tag: 'meta', name: 'twitter:card', content:\n metadata.twitter.card\n })\n if (metadata.twitter?.image) tags.push({\n tag: 'meta', name: 'twitter:image', content:\n metadata.twitter.image\n })\n if (metadata.twitter?.creator) tags.push({\n tag: 'meta', name: 'twitter:creator', content:\n metadata.twitter.creator\n })\n\n if (metadata.canonical) tags.push({tag: 'link', rel: 'canonical', href: metadata.canonical})\n if (metadata.robots) tags.push({tag: 'meta', name: 'robots', content: metadata.robots})\n if (metadata.alternates) {\n for (const [lang, href] of Object.entries(metadata.alternates))\n tags.push({tag: 'link', rel: 'alternate', href, hrefLang: lang})\n }\n\n if (viewport) {\n const parts: string[] = []\n if (viewport.width !== undefined) parts.push(`width=${viewport.width}`)\n if (viewport.initialScale !== undefined) parts.push(`initial-scale=${viewport.initialScale}`)\n if (viewport.maximumScale !== undefined) parts.push(`maximum-scale=${viewport.maximumScale}`)\n if (viewport.userScalable !== undefined) parts.push(`user-scalable=${viewport.userScalable ? 'yes' :\n 'no'}`)\n if (parts.length) tags.push({tag: 'meta', name: 'viewport', content: parts.join(', ')})\n if (viewport.themeColor) tags.push({\n tag: 'meta', name: 'theme-color', content: viewport.themeColor\n })\n }\n\n return tags\n}\n\nexport function buildHeadNodes(metadata: Metadata, viewport?: Viewport): ReactNode {\n const tags = collectTags(metadata, viewport)\n\n return <>\n {tags.map((t, i) => {\n if (t.tag === 'title') return <title key={i}>{t.children}</title>\n if (t.tag === 'link') return <link key={i} rel={t.rel} href={t.href} hrefLang={t.hrefLang}/>\n return <meta key={i} name={t.name} property={t.property} content={t.content}/>\n })}\n </>\n}", "import {createContext, Context, ComponentType} from \"react\";\nimport {Metadata, Viewport} from \"../types\";\nimport {LayoutProps, PageProps} from \"../server/types\";\n\nexport interface RouterContextValue {\n pathname: string\n params: Record<string, string>\n loaderData: unknown\n layoutsData: unknown[]\n Page: ComponentType<PageProps>\n layouts: ComponentType<LayoutProps>[]\n metadata: Metadata | null\n viewport?: Viewport\n navigate: (to: string) => void\n isNavigating: boolean\n}\n\nexport interface PageMetaContextValue {\n metadata: Metadata | null\n viewport?: Viewport\n clientEntry?: string\n}\n\nexport interface RouteDataContextValue {\n loaderData: unknown\n params: Record<string, string>\n}\n\nconst g = globalThis as any\n\ng.__devix_RouterContext__ ??= createContext<RouterContextValue | null>(null)\nexport const RouterContext: Context<RouterContextValue | null> = g.__devix_RouterContext__\n\ng.__devix_PageMetaContext__ ??= createContext<PageMetaContextValue | null>(null)\ng.__devix_RouteDataContext__ ??= createContext<RouteDataContextValue | null>(null)\n\nexport const PageMetaContext: Context<PageMetaContextValue | null> = g.__devix_PageMetaContext__\nexport const RouteDataContext: Context<RouteDataContextValue | null> = g.__devix_RouteDataContext__\n\n", "import {Component, ComponentType, ReactNode} from \"react\";\nimport {ErrorProps} from \"../server/types\";\n\ninterface Props {\n ErrorPage?: ComponentType<ErrorProps>\n children: ReactNode\n}\n\ninterface State {\n error: ErrorProps | null\n}\n\nexport class DevixErrorBoundary extends Component<Props, State> {\n state: State = { error: null }\n\n static getDerivedStateFromError(err: unknown): State {\n if (err instanceof DevixError) {\n return {\n error: {statusCode: err.statusCode, message: err.message}\n }\n }\n return {\n error: {statusCode: 500, message: err instanceof Error ? err.message : 'Unknown error'}\n }\n }\n\n render() {\n if (this.state.error && this.props.ErrorPage) {\n return <this.props.ErrorPage {...this.state.error} />\n }\n if (this.state.error) {\n return <h1>{this.state.error.statusCode}</h1>\n }\n return this.props.children\n }\n}\n\nexport class DevixError extends Error {\n statusCode: number\n constructor(statusCode: number, message: string) {\n super(message)\n this.statusCode = statusCode\n }\n}\n"],
5
- "mappings": "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",
6
- "names": ["useCallback", "useContext", "useEffect", "useRef", "useState", "RouterContext", "getDefaultErrorPage", "loadErrorPage", "matchClientRoute", "Fragment", "jsx", "collectTags", "metadata", "viewport", "tags", "ogTitle", "ogDesc", "twTitle", "twDesc", "lang", "href", "parts", "buildHeadNodes", "t", "i", "createContext", "g", "RouterContext", "PageMetaContext", "RouteDataContext", "Component", "jsx", "DevixErrorBoundary", "err", "DevixError", "statusCode", "message", "jsx", "jsxs", "useRouter", "useContext", "RouterContext", "useNavigate", "ctx", "useParams", "RouteDataContext", "useLoaderData", "RouterProvider", "initialData", "initialParams", "initialPage", "initialLayouts", "initialLayoutsData", "initialMeta", "initialViewport", "initialError", "initialErrorPage", "clientEntry", "state", "setState", "useState", "navigatingRef", "useRef", "isNavigating", "setIsNavigating", "loadRoute", "useCallback", "to", "controller", "pathname", "matched", "matchClientRoute", "ErrorPage", "loadErrorPage", "getDefaultErrorPage", "prev", "pageMod", "layoutMods", "l", "dataRes", "data", "m", "navigate", "useEffect", "handlePop", "err", "content", "tree", "i", "Layout", "layoutData", "DevixErrorBoundary", "PageMetaContext", "buildHeadNodes"]
4
+ "sourcesContent": ["import {ComponentType, ReactNode, useCallback, useContext, useEffect, useRef, useState} from \"react\";\nimport {RouterContext} from 'virtual:devix/context'\nimport {ErrorProps, LayoutProps, PageProps} from \"../server/types\";\nimport {Metadata, Viewport} from \"../types\";\nimport {getDefaultErrorPage, loadErrorPage, matchClientRoute} from \"virtual:devix/client-routes\";\nimport {HeadSlot} from \"./head\";\nimport {NavigateOptions, PageMetaContext, RouteDataContext} from \"./context\";\nimport {DevixErrorBoundary} from \"./error-boundary\";\nimport type {Redirect} from \"../utils/response\";\n\ninterface RouteState {\n pathname: string\n params: Record<string, string>\n loaderData: unknown\n layoutsData: unknown[]\n Page: ComponentType<PageProps>\n layouts: ComponentType<LayoutProps>[]\n metadata: Metadata | null\n viewport?: Viewport\n pendingError?: ErrorProps\n ErrorPage?: ComponentType<ErrorProps>\n}\n\nexport function useRouter() {\n return useContext(RouterContext)\n}\n\nconst noopNavigate = () => Promise.resolve()\nconst noopRevalidate = () => Promise.resolve()\n\nexport function useNavigate() {\n const ctx = useContext(RouterContext)\n return ctx?.navigate ?? noopNavigate\n}\n\nexport function useRevalidate() {\n const ctx = useContext(RouterContext)\n return ctx?.revalidate ?? noopRevalidate\n}\n\nexport function useParams<T extends Record<string, string>>() {\n const ctx = useContext(RouteDataContext)\n if (!ctx) throw new Error(\"useParams must be used within a route or layout\")\n return ctx.params as T\n}\n\ntype LoaderReturnType<T> = T extends (...args: any[]) => Promise<infer R>\n ? [Exclude<R, Redirect | void | undefined>] extends [never] ? undefined : Exclude<R, Redirect | void | undefined>\n : T extends (...args: any[]) => infer R\n ? [Exclude<R, Redirect | void | undefined>] extends [never] ? undefined : Exclude<R, Redirect | void | undefined>\n : T\n\nexport function useLoaderData<T>() {\n const ctx = useContext(RouteDataContext)\n if (!ctx) throw new Error(\"useLoaderData must be used within a route or layout\")\n return ctx.loaderData as LoaderReturnType<T>\n}\n\n\ninterface RouterProviderProps {\n initialData: unknown\n initialParams: Record<string, string>\n initialPage: ComponentType<PageProps>\n initialLayouts?: ComponentType<LayoutProps>[]\n initialLayoutsData?: unknown[]\n initialMeta?: Metadata | null\n initialViewport?: Viewport\n initialError?: ErrorProps\n initialErrorPage?: ComponentType<ErrorProps>\n clientEntry: string\n}\n\nexport function RouterProvider({\n initialData,\n initialParams,\n initialPage,\n initialLayouts = [],\n initialLayoutsData = [],\n initialMeta,\n initialViewport,\n initialError,\n initialErrorPage,\n clientEntry,\n }: RouterProviderProps) {\n\n const [state, setState] = useState<RouteState>({\n pathname: window.location.pathname,\n params: initialParams,\n loaderData: initialData,\n layoutsData: initialLayoutsData,\n Page: initialPage,\n layouts: initialLayouts,\n metadata: initialMeta ?? null,\n viewport: initialViewport,\n pendingError: initialError,\n ErrorPage: initialErrorPage,\n })\n\n const navigatingRef = useRef<AbortController | null>(null)\n const [isNavigating, setIsNavigating] = useState(false)\n\n const loadRoute = useCallback(async (to: string, controller: AbortController) => {\n const pathname = to.split('?')[0]\n const matched = matchClientRoute(pathname)\n if (!matched) {\n const ErrorPage = await loadErrorPage() ?? getDefaultErrorPage()\n setState(prev => ({\n ...prev,\n pathname: pathname,\n pendingError: {statusCode: 404, message: 'Not found'},\n ErrorPage: ErrorPage ?? undefined,\n }))\n return\n }\n\n const [pageMod, ...layoutMods] = await Promise.all([\n matched.load(),\n ...matched.loadLayouts.map(l => l()),\n ])\n\n if (controller.signal.aborted) return\n if (!pageMod.default) return\n\n const dataRes = await fetch(`/_data${to}`, {\n headers: {Accept: 'application/json'},\n signal: controller.signal,\n })\n\n if (controller.signal.aborted) return\n\n if (!dataRes.ok) {\n if (dataRes.status === 404) {\n window.location.href = to\n return\n }\n const ErrorPage = await loadErrorPage() ?? getDefaultErrorPage()\n setState(prev => ({\n ...prev,\n pathname,\n pendingError: {statusCode: dataRes.status, message: 'Server error'},\n ErrorPage: ErrorPage ?? undefined,\n }))\n return\n }\n\n const data = await dataRes.json()\n\n if (data.redirect) {\n if (data.redirectReplace) {\n window.history.replaceState(null, '', data.redirect)\n } else {\n window.history.pushState(null, '', data.redirect)\n }\n await loadRoute(data.redirect, controller)\n return\n }\n\n window.scrollTo(0, 0)\n setState({\n pathname,\n params: data.params ?? {},\n loaderData: data.loaderData,\n layoutsData: (data.layouts ?? []).map((l: any) => l.loaderData),\n Page: pageMod.default,\n layouts: layoutMods.map(m => m.default),\n metadata: data.metadata ?? null,\n viewport: data.viewport,\n })\n }, [])\n\n const navigate = useCallback(async (to: string, options?: NavigateOptions) => {\n navigatingRef.current?.abort()\n const controller = new AbortController()\n navigatingRef.current = controller\n\n setIsNavigating(true)\n const run = async () => {\n window.history[options?.replace ? 'replaceState' : 'pushState'](null, '', to)\n await loadRoute(to, controller)\n }\n try {\n if (options?.viewTransition && 'startViewTransition' in document) {\n await (document as any).startViewTransition(run).finished\n } else {\n await run()\n }\n } finally {\n if (!controller.signal.aborted) setIsNavigating(false)\n }\n }, [loadRoute])\n\n const revalidate = useCallback(async () => {\n const to = window.location.pathname + window.location.search\n const controller = new AbortController()\n const dataRes = await fetch(`/_data${to}`, {\n headers: {Accept: 'application/json'},\n signal: controller.signal,\n })\n if (!dataRes.ok) return\n const data = await dataRes.json()\n if (data.redirect) {\n await navigate(data.redirect, {replace: data.redirectReplace})\n return\n }\n setState(prev => ({\n ...prev,\n loaderData: data.loaderData,\n layoutsData: (data.layouts ?? []).map((l: any) => l.loaderData),\n params: data.params ?? prev.params,\n metadata: data.metadata ?? prev.metadata,\n viewport: data.viewport ?? prev.viewport,\n }))\n }, [navigate])\n\n useEffect(() => {\n const handlePop = () => {\n navigatingRef.current?.abort()\n const controller = new AbortController()\n navigatingRef.current = controller\n\n const to = window.location.pathname + window.location.search\n loadRoute(to, controller).catch(err => {\n if (err.name !== 'AbortError') console.error('[router] popstate error:', err)\n })\n }\n window.addEventListener(\"popstate\", handlePop)\n return () => window.removeEventListener(\"popstate\", handlePop)\n }, [loadRoute])\n\n let content: ReactNode\n\n if (state.pendingError) {\n content = state.ErrorPage\n ? <state.ErrorPage {...state.pendingError} />\n : <h1>{state.pendingError.statusCode}</h1>\n } else {\n let tree: ReactNode = (\n <RouteDataContext value={{loaderData: state.loaderData, params: state.params}}>\n <state.Page data={state.loaderData} params={state.params} url={state.pathname}/>\n </RouteDataContext>\n )\n\n for (let i = state.layouts.length - 1; i >= 0; i--) {\n const Layout = state.layouts[i]\n const layoutData = state.layoutsData[i]\n tree = (\n <RouteDataContext value={{loaderData: layoutData, params: state.params}}>\n <Layout data={layoutData} params={state.params}>{tree}</Layout>\n </RouteDataContext>\n )\n }\n\n content = (\n <DevixErrorBoundary key={state.pathname} ErrorPage={state.ErrorPage}>\n {tree}\n </DevixErrorBoundary>\n )\n }\n\n return (\n <PageMetaContext value={{\n metadata: state.metadata,\n viewport: state.viewport,\n clientEntry,\n }}>\n <HeadSlot metadata={state.metadata} viewport={state.viewport}/>\n <RouterContext value={{...state, isNavigating, navigate, revalidate}}>\n {content}\n </RouterContext>\n </PageMetaContext>\n )\n}", "import {Metadata, Viewport} from \"../types\";\nimport {ReactNode} from \"react\";\n\ntype MetaTag =\n | { tag: 'title'; children: string }\n | { tag: 'meta'; name?: string; property?: string; content: string }\n | { tag: 'link'; rel: string; href: string; hrefLang?: string }\n\nfunction collectTags(metadata: Metadata, viewport?: Viewport): MetaTag[] {\n const tags: MetaTag[] = []\n\n if (metadata.title)\n tags.push({tag: 'title', children: metadata.title})\n if (metadata.description)\n tags.push({tag: 'meta', name: 'description', content: metadata.description})\n if (metadata.keywords?.length)\n tags.push({tag: 'meta', name: 'keywords', content: metadata.keywords.join(', ')})\n\n const ogTitle = metadata.og?.title ?? metadata.title\n if (ogTitle) tags.push({tag: 'meta', property: 'og:title', content: ogTitle})\n const ogDesc = metadata.og?.description ?? metadata.description\n if (ogDesc) tags.push({tag: 'meta', property: 'og:description', content: ogDesc})\n if (metadata.og?.image) tags.push({tag: 'meta', property: 'og:image', content: metadata.og.image})\n if (metadata.og?.type) tags.push({tag: 'meta', property: 'og:type', content: metadata.og.type})\n if (metadata.og?.url) tags.push({tag: 'meta', property: 'og:url', content: metadata.og.url})\n\n const twTitle = metadata.twitter?.title ?? metadata.title\n if (twTitle) tags.push({tag: 'meta', name: 'twitter:title', content: twTitle})\n const twDesc = metadata.twitter?.description ?? metadata.description\n if (twDesc) tags.push({tag: 'meta', name: 'twitter:description', content: twDesc})\n if (metadata.twitter?.card) tags.push({\n tag: 'meta', name: 'twitter:card', content:\n metadata.twitter.card\n })\n if (metadata.twitter?.image) tags.push({\n tag: 'meta', name: 'twitter:image', content:\n metadata.twitter.image\n })\n if (metadata.twitter?.creator) tags.push({\n tag: 'meta', name: 'twitter:creator', content:\n metadata.twitter.creator\n })\n\n if (metadata.canonical) tags.push({tag: 'link', rel: 'canonical', href: metadata.canonical})\n if (metadata.robots) tags.push({tag: 'meta', name: 'robots', content: metadata.robots})\n if (metadata.alternates) {\n for (const [lang, href] of Object.entries(metadata.alternates))\n tags.push({tag: 'link', rel: 'alternate', href, hrefLang: lang})\n }\n\n if (viewport) {\n const parts: string[] = []\n if (viewport.width !== undefined) parts.push(`width=${viewport.width}`)\n if (viewport.initialScale !== undefined) parts.push(`initial-scale=${viewport.initialScale}`)\n if (viewport.maximumScale !== undefined) parts.push(`maximum-scale=${viewport.maximumScale}`)\n if (viewport.userScalable !== undefined) parts.push(`user-scalable=${viewport.userScalable ? 'yes' :\n 'no'}`)\n if (parts.length) tags.push({tag: 'meta', name: 'viewport', content: parts.join(', ')})\n if (viewport.themeColor) tags.push({\n tag: 'meta', name: 'theme-color', content: viewport.themeColor\n })\n }\n\n return tags\n}\n\nexport function HeadSlot({metadata, viewport}: {metadata: Metadata | null, viewport?: Viewport}) {\n if (typeof window === 'undefined' || !metadata) return null\n return <>{buildHeadNodes(metadata, viewport)}</>\n}\n\nexport function buildHeadNodes(metadata: Metadata, viewport?: Viewport): ReactNode {\n const tags = collectTags(metadata, viewport)\n\n return <>\n {tags.map((t, i) => {\n if (t.tag === 'title') return <title key={i}>{t.children}</title>\n if (t.tag === 'link') return <link key={i} rel={t.rel} href={t.href} hrefLang={t.hrefLang}/>\n return <meta key={i} name={t.name} property={t.property} content={t.content}/>\n })}\n </>\n}", "import {createContext, Context, ComponentType} from \"react\";\nimport {Metadata, Viewport} from \"../types\";\nimport {LayoutProps, PageProps} from \"../server/types\";\n\nexport interface NavigateOptions {\n replace?: boolean\n viewTransition?: boolean\n}\n\nexport interface RouterContextValue {\n pathname: string\n params: Record<string, string>\n loaderData: unknown\n layoutsData: unknown[]\n Page: ComponentType<PageProps>\n layouts: ComponentType<LayoutProps>[]\n metadata: Metadata | null\n viewport?: Viewport\n navigate: (to: string, options?: NavigateOptions) => Promise<void>\n revalidate: () => Promise<void>\n isNavigating: boolean\n}\n\nexport interface PageMetaContextValue {\n metadata: Metadata | null\n viewport?: Viewport\n clientEntry?: string\n}\n\nexport interface RouteDataContextValue {\n loaderData: unknown\n params: Record<string, string>\n}\n\nconst g = globalThis as any\n\ng.__devix_RouterContext__ ??= createContext<RouterContextValue | null>(null)\nexport const RouterContext: Context<RouterContextValue | null> = g.__devix_RouterContext__\n\ng.__devix_PageMetaContext__ ??= createContext<PageMetaContextValue | null>(null)\ng.__devix_RouteDataContext__ ??= createContext<RouteDataContextValue | null>(null)\n\nexport const PageMetaContext: Context<PageMetaContextValue | null> = g.__devix_PageMetaContext__\nexport const RouteDataContext: Context<RouteDataContextValue | null> = g.__devix_RouteDataContext__\n\n", "import {Component, ComponentType, ReactNode} from \"react\";\nimport {ErrorProps} from \"../server/types\";\n\ninterface Props {\n ErrorPage?: ComponentType<ErrorProps>\n children: ReactNode\n}\n\ninterface State {\n error: ErrorProps | null\n}\n\nexport class DevixErrorBoundary extends Component<Props, State> {\n state: State = { error: null }\n\n static getDerivedStateFromError(err: unknown): State {\n if (err instanceof DevixError) {\n return {\n error: {statusCode: err.statusCode, message: err.message}\n }\n }\n return {\n error: {statusCode: 500, message: err instanceof Error ? err.message : 'Unknown error'}\n }\n }\n\n render() {\n if (this.state.error && this.props.ErrorPage) {\n return <this.props.ErrorPage {...this.state.error} />\n }\n if (this.state.error) {\n return <h1>{this.state.error.statusCode}</h1>\n }\n return this.props.children\n }\n}\n\nexport class DevixError extends Error {\n statusCode: number\n constructor(statusCode: number, message: string) {\n super(message)\n this.statusCode = statusCode\n }\n}\n"],
5
+ "mappings": "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",
6
+ "names": ["useCallback", "useContext", "useEffect", "useRef", "useState", "RouterContext", "getDefaultErrorPage", "loadErrorPage", "matchClientRoute", "Fragment", "jsx", "collectTags", "metadata", "viewport", "tags", "ogTitle", "ogDesc", "twTitle", "twDesc", "lang", "href", "parts", "HeadSlot", "buildHeadNodes", "t", "i", "createContext", "g", "RouterContext", "PageMetaContext", "RouteDataContext", "Component", "jsx", "DevixErrorBoundary", "err", "DevixError", "statusCode", "message", "jsx", "jsxs", "useRouter", "useContext", "RouterContext", "noopNavigate", "noopRevalidate", "useNavigate", "useRevalidate", "useParams", "ctx", "RouteDataContext", "useLoaderData", "RouterProvider", "initialData", "initialParams", "initialPage", "initialLayouts", "initialLayoutsData", "initialMeta", "initialViewport", "initialError", "initialErrorPage", "clientEntry", "state", "setState", "useState", "navigatingRef", "useRef", "isNavigating", "setIsNavigating", "loadRoute", "useCallback", "to", "controller", "pathname", "matched", "matchClientRoute", "ErrorPage", "loadErrorPage", "getDefaultErrorPage", "prev", "pageMod", "layoutMods", "l", "dataRes", "data", "m", "navigate", "options", "run", "revalidate", "useEffect", "handlePop", "err", "content", "tree", "i", "Layout", "layoutData", "DevixErrorBoundary", "PageMetaContext", "HeadSlot"]
7
7
  }
@@ -0,0 +1,15 @@
1
+ import { ComponentType } from 'react';
2
+ import { LayoutProps, PageProps } from '../server/types';
3
+ import { Metadata, Viewport } from '../types';
4
+ export interface ServerAppProps {
5
+ pathname: string;
6
+ params: Record<string, string>;
7
+ loaderData: unknown;
8
+ layoutsData: unknown[];
9
+ Page: ComponentType<PageProps>;
10
+ layouts: ComponentType<LayoutProps>[];
11
+ metadata: Metadata | null;
12
+ viewport?: Viewport;
13
+ clientEntry: string;
14
+ }
15
+ export declare function ServerApp({ pathname, params, loaderData, layoutsData, Page, layouts, metadata, viewport, clientEntry, }: ServerAppProps): import("react/jsx-runtime").JSX.Element;
@@ -0,0 +1,2 @@
1
+ import{createContext as h}from"react";var l=globalThis;l.__devix_RouterContext__??=h(null);var y=l.__devix_RouterContext__;l.__devix_PageMetaContext__??=h(null);l.__devix_RouteDataContext__??=h(null);var P=l.__devix_PageMetaContext__,d=l.__devix_RouteDataContext__;import{Fragment as _,jsx as u}from"react/jsx-runtime";function M(t,e){let o=[];t.title&&o.push({tag:"title",children:t.title}),t.description&&o.push({tag:"meta",name:"description",content:t.description}),t.keywords?.length&&o.push({tag:"meta",name:"keywords",content:t.keywords.join(", ")});let r=t.og?.title??t.title;r&&o.push({tag:"meta",property:"og:title",content:r});let i=t.og?.description??t.description;i&&o.push({tag:"meta",property:"og:description",content:i}),t.og?.image&&o.push({tag:"meta",property:"og:image",content:t.og.image}),t.og?.type&&o.push({tag:"meta",property:"og:type",content:t.og.type}),t.og?.url&&o.push({tag:"meta",property:"og:url",content:t.og.url});let s=t.twitter?.title??t.title;s&&o.push({tag:"meta",name:"twitter:title",content:s});let p=t.twitter?.description??t.description;if(p&&o.push({tag:"meta",name:"twitter:description",content:p}),t.twitter?.card&&o.push({tag:"meta",name:"twitter:card",content:t.twitter.card}),t.twitter?.image&&o.push({tag:"meta",name:"twitter:image",content:t.twitter.image}),t.twitter?.creator&&o.push({tag:"meta",name:"twitter:creator",content:t.twitter.creator}),t.canonical&&o.push({tag:"link",rel:"canonical",href:t.canonical}),t.robots&&o.push({tag:"meta",name:"robots",content:t.robots}),t.alternates)for(let[n,f]of Object.entries(t.alternates))o.push({tag:"link",rel:"alternate",href:f,hrefLang:n});if(e){let n=[];e.width!==void 0&&n.push(`width=${e.width}`),e.initialScale!==void 0&&n.push(`initial-scale=${e.initialScale}`),e.maximumScale!==void 0&&n.push(`maximum-scale=${e.maximumScale}`),e.userScalable!==void 0&&n.push(`user-scalable=${e.userScalable?"yes":"no"}`),n.length&&o.push({tag:"meta",name:"viewport",content:n.join(", ")}),e.themeColor&&o.push({tag:"meta",name:"theme-color",content:e.themeColor})}return o}function w({metadata:t,viewport:e}){return typeof window>"u"||!t?null:u(_,{children:k(t,e)})}function k(t,e){let o=M(t,e);return u(_,{children:o.map((r,i)=>r.tag==="title"?u("title",{children:r.children},i):r.tag==="link"?u("link",{rel:r.rel,href:r.href,hrefLang:r.hrefLang},i):u("meta",{name:r.name,property:r.property,content:r.content},i))})}import{Component as V}from"react";import{jsx as R}from"react/jsx-runtime";var c=class extends V{state={error:null};static getDerivedStateFromError(e){return e instanceof x?{error:{statusCode:e.statusCode,message:e.message}}:{error:{statusCode:500,message:e instanceof Error?e.message:"Unknown error"}}}render(){return this.state.error&&this.props.ErrorPage?R(this.props.ErrorPage,{...this.state.error}):this.state.error?R("h1",{children:this.state.error.statusCode}):this.props.children}},x=class extends Error{statusCode;constructor(e,o){super(o),this.statusCode=e}};import{jsx as a,jsxs as T}from"react/jsx-runtime";var D=(t,e)=>Promise.resolve(),S=()=>Promise.resolve();function I({pathname:t,params:e,loaderData:o,layoutsData:r,Page:i,layouts:s,metadata:p,viewport:n,clientEntry:f}){let m=a(d,{value:{loaderData:o,params:e},children:a(i,{data:o,params:e,url:t})});for(let g=s.length-1;g>=0;g--){let v=s[g],C=r[g];m=a(d,{value:{loaderData:C,params:e},children:a(v,{data:C,params:e,children:m})})}return T(P,{value:{metadata:p,viewport:n,clientEntry:f},children:[a(w,{metadata:p,viewport:n}),a(y,{value:{pathname:t,params:e,loaderData:o,layoutsData:r,Page:i,layouts:s,metadata:p,viewport:n,isNavigating:!1,navigate:D,revalidate:S},children:a(c,{children:m},t)})]})}export{I as ServerApp};
2
+ //# sourceMappingURL=server-app.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/runtime/context.tsx", "../../src/runtime/head.tsx", "../../src/runtime/error-boundary.tsx", "../../src/runtime/server-app.tsx"],
4
+ "sourcesContent": ["import {createContext, Context, ComponentType} from \"react\";\nimport {Metadata, Viewport} from \"../types\";\nimport {LayoutProps, PageProps} from \"../server/types\";\n\nexport interface NavigateOptions {\n replace?: boolean\n viewTransition?: boolean\n}\n\nexport interface RouterContextValue {\n pathname: string\n params: Record<string, string>\n loaderData: unknown\n layoutsData: unknown[]\n Page: ComponentType<PageProps>\n layouts: ComponentType<LayoutProps>[]\n metadata: Metadata | null\n viewport?: Viewport\n navigate: (to: string, options?: NavigateOptions) => Promise<void>\n revalidate: () => Promise<void>\n isNavigating: boolean\n}\n\nexport interface PageMetaContextValue {\n metadata: Metadata | null\n viewport?: Viewport\n clientEntry?: string\n}\n\nexport interface RouteDataContextValue {\n loaderData: unknown\n params: Record<string, string>\n}\n\nconst g = globalThis as any\n\ng.__devix_RouterContext__ ??= createContext<RouterContextValue | null>(null)\nexport const RouterContext: Context<RouterContextValue | null> = g.__devix_RouterContext__\n\ng.__devix_PageMetaContext__ ??= createContext<PageMetaContextValue | null>(null)\ng.__devix_RouteDataContext__ ??= createContext<RouteDataContextValue | null>(null)\n\nexport const PageMetaContext: Context<PageMetaContextValue | null> = g.__devix_PageMetaContext__\nexport const RouteDataContext: Context<RouteDataContextValue | null> = g.__devix_RouteDataContext__\n\n", "import {Metadata, Viewport} from \"../types\";\nimport {ReactNode} from \"react\";\n\ntype MetaTag =\n | { tag: 'title'; children: string }\n | { tag: 'meta'; name?: string; property?: string; content: string }\n | { tag: 'link'; rel: string; href: string; hrefLang?: string }\n\nfunction collectTags(metadata: Metadata, viewport?: Viewport): MetaTag[] {\n const tags: MetaTag[] = []\n\n if (metadata.title)\n tags.push({tag: 'title', children: metadata.title})\n if (metadata.description)\n tags.push({tag: 'meta', name: 'description', content: metadata.description})\n if (metadata.keywords?.length)\n tags.push({tag: 'meta', name: 'keywords', content: metadata.keywords.join(', ')})\n\n const ogTitle = metadata.og?.title ?? metadata.title\n if (ogTitle) tags.push({tag: 'meta', property: 'og:title', content: ogTitle})\n const ogDesc = metadata.og?.description ?? metadata.description\n if (ogDesc) tags.push({tag: 'meta', property: 'og:description', content: ogDesc})\n if (metadata.og?.image) tags.push({tag: 'meta', property: 'og:image', content: metadata.og.image})\n if (metadata.og?.type) tags.push({tag: 'meta', property: 'og:type', content: metadata.og.type})\n if (metadata.og?.url) tags.push({tag: 'meta', property: 'og:url', content: metadata.og.url})\n\n const twTitle = metadata.twitter?.title ?? metadata.title\n if (twTitle) tags.push({tag: 'meta', name: 'twitter:title', content: twTitle})\n const twDesc = metadata.twitter?.description ?? metadata.description\n if (twDesc) tags.push({tag: 'meta', name: 'twitter:description', content: twDesc})\n if (metadata.twitter?.card) tags.push({\n tag: 'meta', name: 'twitter:card', content:\n metadata.twitter.card\n })\n if (metadata.twitter?.image) tags.push({\n tag: 'meta', name: 'twitter:image', content:\n metadata.twitter.image\n })\n if (metadata.twitter?.creator) tags.push({\n tag: 'meta', name: 'twitter:creator', content:\n metadata.twitter.creator\n })\n\n if (metadata.canonical) tags.push({tag: 'link', rel: 'canonical', href: metadata.canonical})\n if (metadata.robots) tags.push({tag: 'meta', name: 'robots', content: metadata.robots})\n if (metadata.alternates) {\n for (const [lang, href] of Object.entries(metadata.alternates))\n tags.push({tag: 'link', rel: 'alternate', href, hrefLang: lang})\n }\n\n if (viewport) {\n const parts: string[] = []\n if (viewport.width !== undefined) parts.push(`width=${viewport.width}`)\n if (viewport.initialScale !== undefined) parts.push(`initial-scale=${viewport.initialScale}`)\n if (viewport.maximumScale !== undefined) parts.push(`maximum-scale=${viewport.maximumScale}`)\n if (viewport.userScalable !== undefined) parts.push(`user-scalable=${viewport.userScalable ? 'yes' :\n 'no'}`)\n if (parts.length) tags.push({tag: 'meta', name: 'viewport', content: parts.join(', ')})\n if (viewport.themeColor) tags.push({\n tag: 'meta', name: 'theme-color', content: viewport.themeColor\n })\n }\n\n return tags\n}\n\nexport function HeadSlot({metadata, viewport}: {metadata: Metadata | null, viewport?: Viewport}) {\n if (typeof window === 'undefined' || !metadata) return null\n return <>{buildHeadNodes(metadata, viewport)}</>\n}\n\nexport function buildHeadNodes(metadata: Metadata, viewport?: Viewport): ReactNode {\n const tags = collectTags(metadata, viewport)\n\n return <>\n {tags.map((t, i) => {\n if (t.tag === 'title') return <title key={i}>{t.children}</title>\n if (t.tag === 'link') return <link key={i} rel={t.rel} href={t.href} hrefLang={t.hrefLang}/>\n return <meta key={i} name={t.name} property={t.property} content={t.content}/>\n })}\n </>\n}", "import {Component, ComponentType, ReactNode} from \"react\";\nimport {ErrorProps} from \"../server/types\";\n\ninterface Props {\n ErrorPage?: ComponentType<ErrorProps>\n children: ReactNode\n}\n\ninterface State {\n error: ErrorProps | null\n}\n\nexport class DevixErrorBoundary extends Component<Props, State> {\n state: State = { error: null }\n\n static getDerivedStateFromError(err: unknown): State {\n if (err instanceof DevixError) {\n return {\n error: {statusCode: err.statusCode, message: err.message}\n }\n }\n return {\n error: {statusCode: 500, message: err instanceof Error ? err.message : 'Unknown error'}\n }\n }\n\n render() {\n if (this.state.error && this.props.ErrorPage) {\n return <this.props.ErrorPage {...this.state.error} />\n }\n if (this.state.error) {\n return <h1>{this.state.error.statusCode}</h1>\n }\n return this.props.children\n }\n}\n\nexport class DevixError extends Error {\n statusCode: number\n constructor(statusCode: number, message: string) {\n super(message)\n this.statusCode = statusCode\n }\n}\n", "import {ComponentType, ReactNode} from 'react'\nimport {RouterContext, PageMetaContext, RouteDataContext, NavigateOptions} from './context'\nimport {HeadSlot} from './head'\nimport {DevixErrorBoundary} from './error-boundary'\nimport {LayoutProps, PageProps} from '../server/types'\nimport {Metadata, Viewport} from '../types'\n\nconst noopNavigate = (_to: string, _opts?: NavigateOptions) => Promise.resolve()\nconst noopRevalidate = () => Promise.resolve()\n\nexport interface ServerAppProps {\n pathname: string\n params: Record<string, string>\n loaderData: unknown\n layoutsData: unknown[]\n Page: ComponentType<PageProps>\n layouts: ComponentType<LayoutProps>[]\n metadata: Metadata | null\n viewport?: Viewport\n clientEntry: string\n}\n\nexport function ServerApp({\n pathname, params, loaderData, layoutsData,\n Page, layouts, metadata, viewport, clientEntry,\n}: ServerAppProps) {\n let tree: ReactNode = (\n <RouteDataContext value={{loaderData, params}}>\n <Page data={loaderData as any} params={params} url={pathname}/>\n </RouteDataContext>\n )\n\n for (let i = layouts.length - 1; i >= 0; i--) {\n const Layout = layouts[i]\n const layoutData = layoutsData[i]\n tree = (\n <RouteDataContext value={{loaderData: layoutData, params}}>\n <Layout data={layoutData as any} params={params}>{tree}</Layout>\n </RouteDataContext>\n )\n }\n\n return (\n <PageMetaContext value={{metadata, viewport, clientEntry}}>\n <HeadSlot metadata={metadata} viewport={viewport}/>\n <RouterContext value={{\n pathname,\n params,\n loaderData,\n layoutsData,\n Page,\n layouts,\n metadata,\n viewport,\n isNavigating: false,\n navigate: noopNavigate,\n revalidate: noopRevalidate,\n }}>\n <DevixErrorBoundary key={pathname}>\n {tree}\n </DevixErrorBoundary>\n </RouterContext>\n </PageMetaContext>\n )\n}\n"],
5
+ "mappings": "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",
6
+ "names": ["createContext", "g", "RouterContext", "PageMetaContext", "RouteDataContext", "Fragment", "jsx", "collectTags", "metadata", "viewport", "tags", "ogTitle", "ogDesc", "twTitle", "twDesc", "lang", "href", "parts", "HeadSlot", "buildHeadNodes", "t", "Component", "jsx", "DevixErrorBoundary", "err", "DevixError", "statusCode", "message", "jsx", "jsxs", "noopNavigate", "_to", "_opts", "noopRevalidate", "ServerApp", "pathname", "params", "loaderData", "layoutsData", "Page", "layouts", "metadata", "viewport", "clientEntry", "tree", "RouteDataContext", "i", "Layout", "layoutData", "PageMetaContext", "HeadSlot", "RouterContext", "DevixErrorBoundary"]
7
+ }
@@ -1,2 +1,2 @@
1
- function x(e){return e.replace(/\.(tsx|ts|jsx|js)$/,"").replace(/\(.*?\)\//g,"").replace(/^index$|\/index$/,"").replace(/\[([^\]]+)]/g,":$1")||"/"}function k(e,t){let n=e.slice(t.length+1).replace(/\\/g,"/"),r=x(n);return r==="/"?"/api":`/api/${r}`.replace("/api//","/api/")}function h(e){return e.slice(0,e.lastIndexOf("/"))}var l=null;function y(e,t,n){if(l)return l;let r=[],a=[];for(let o of t)a.push({dir:h(o),key:o});for(let o of e){let s=k(o,n),i=[...s.matchAll(/:([^/]+)/g)].map(d=>d[1]),u=s.replace(/:[^/]+/g,"([^/]+)").replace(/\//g,"\\/");r.push({path:s,key:o,params:i,regex:new RegExp(`^${u}$`)})}return r.sort((o,s)=>{let i=(o.path.match(/:/g)||[]).length,u=(s.path.match(/:/g)||[]).length;return i!==u?i-u:s.path.length-o.path.length}),l={routes:r,middlewares:a},l}function w(e,t){let n=h(e);return t.filter(r=>n.startsWith(r.dir)).sort((r,a)=>r.dir.split("/").length-a.dir.split("/").length)}function A(e,t){for(let n of t){let r=e.match(n.regex);if(r){let a={};return n.params.forEach((o,s)=>{a[o]=decodeURIComponent(r[s+1])}),{route:n,params:a}}}return null}var c=class{params;_state=new Map;constructor(t={}){this.params=t}set(t,n){this._state.set(t,n)}get(t){return this._state.get(t)}};import{Component as q}from"react";import{jsx as I}from"react/jsx-runtime";var f=class extends Error{statusCode;constructor(t,n){super(n),this.statusCode=t}};var T="__devix_handler__";import{AsyncLocalStorage as D}from"node:async_hooks";var M=new D;function H(e,t){return M.run(e,t)}function v(e){return typeof e=="object"&&e!==null&&T in e}async function _(e){let t=e.headers.get("Content-Type")??"";return t.includes("application/json")?e.json():t.includes("multipart/form-data")||t.includes("application/x-www-form-urlencoded")?e.formData():e.text()}function N(e){return e instanceof Response?e:e==null?new Response(null,{status:204}):new Response(JSON.stringify(e),{headers:{"Content-Type":"application/json"}})}async function Z(e,t,n){try{let{pathname:r}=new URL(e,"http://localhost"),{routes:a,middlewares:o}=y(Object.keys(n.routes),Object.keys(n.middlewares),n.apiDir),s=A(r,a);if(!s)return new Response("Not Found",{status:404});let{route:i,params:u}=s,d=new c(u),C=await H({request:t,ctx:d},async()=>{let E=w(i.key,o);for(let m of E){let R=await n.middlewares[m.key]();if(R.middleware){let g=await R.middleware(d,t);if(g instanceof Response)return g}}let P=await n.routes[i.key](),S=t.method.toUpperCase(),p=P[S];if(!p)return new Response("Method Not Allowed",{status:405});if(v(p)){if(p.fn.length===0)return p.fn();let m=await _(t);return p.fn(m)}return p(d,t)});return N(C)}catch(r){return console.error("[devix] api error:",r),r instanceof f?new Response(r.message,{status:r.statusCode}):new Response("Internal Server Error",{status:500})}}export{Z as handleApiRequest};
1
+ function h(e){return e.replace(/\.(tsx|ts|jsx|js)$/,"").replace(/\(.*?\)\//g,"").replace(/^index$|\/index$/,"").replace(/\[([^\]]+)]/g,":$1")||"/"}function M(e,t){let n=e.slice(t.length+1).replace(/\\/g,"/"),r=h(n);return r==="/"?"/api":`/api/${r}`.replace("/api//","/api/")}function w(e){return e.slice(0,e.lastIndexOf("/"))}var p=null;function A(e,t,n){if(p)return p;let r=[],a=[];for(let o of t)a.push({dir:w(o),key:o});for(let o of e){let s=M(o,n),i=[...s.matchAll(/:([^/]+)/g)].map(l=>l[1]),u=s.replace(/:[^/]+/g,"([^/]+)").replace(/\//g,"\\/");r.push({path:s,key:o,params:i,regex:new RegExp(`^${u}$`)})}return r.sort((o,s)=>{let i=(o.path.match(/:/g)||[]).length,u=(s.path.match(/:/g)||[]).length;return i!==u?i-u:s.path.length-o.path.length}),p={routes:r,middlewares:a},p}function T(e,t){let n=w(e);return t.filter(r=>n.startsWith(r.dir)).sort((r,a)=>r.dir.split("/").length-a.dir.split("/").length)}function H(e,t){for(let n of t){let r=e.match(n.regex);if(r){let a={};return n.params.forEach((o,s)=>{a[o]=decodeURIComponent(r[s+1])}),{route:n,params:a}}}return null}var c=class{params;_state=new Map;constructor(t={}){this.params=t}set(t,n){this._state.set(t,n)}get(t){return this._state.get(t)}};import{Component as O}from"react";import{jsx as J}from"react/jsx-runtime";var f=class extends Error{statusCode;constructor(t,n){super(n),this.statusCode=t}};var C="__devix_handler__";import{AsyncLocalStorage as v}from"node:async_hooks";var R=Symbol.for("@devlusoft/devix.handlerStore"),g=globalThis;g[R]||(g[R]=new v);var _=g[R];function S(e,t){return _.run(e,t)}function N(e){return typeof e=="object"&&e!==null&&C in e}async function b(e){let t=e.headers.get("Content-Type")??"";return t.includes("application/json")?e.json():t.includes("multipart/form-data")||t.includes("application/x-www-form-urlencoded")?e.formData():e.text()}function B(e){return e instanceof Response?e:e==null?new Response(null,{status:204}):new Response(JSON.stringify(e),{headers:{"Content-Type":"application/json"}})}async function te(e,t,n){try{let{pathname:r}=new URL(e,"http://localhost"),{routes:a,middlewares:o}=A(Object.keys(n.routes),Object.keys(n.middlewares),n.apiDir),s=H(r,a);if(!s)return new Response("Not Found",{status:404});let{route:i,params:u}=s,l=new c(u),E=await S({request:t,ctx:l},async()=>{let P=T(i.key,o);for(let m of P){let x=await n.middlewares[m.key]();if(x.middleware){let y=await x.middleware(l,t);if(y instanceof Response)return y}}let k=await n.routes[i.key](),D=t.method.toUpperCase(),d=k[D];if(!d)return new Response("Method Not Allowed",{status:405});if(N(d)){if(d.fn.length===0)return d.fn();let m=await b(t);return d.fn(m)}return d(l,t)});return B(E)}catch(r){return console.error("[devix] api error:",r),r instanceof f?new Response(r.message,{status:r.statusCode}):new Response("Internal Server Error",{status:500})}}export{te as handleApiRequest};
2
2
  //# sourceMappingURL=api.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/utils/patterns.ts", "../../src/server/api-router.ts", "../../src/runtime/api-context.ts", "../../src/runtime/error-boundary.tsx", "../../src/runtime/create-handler.ts", "../../src/server/handler-store.ts", "../../src/server/api.ts"],
4
- "sourcesContent": ["export function routePattern(rel: string): string {\n return rel\n .replace(/\\.(tsx|ts|jsx|js)$/, '')\n .replace(/\\(.*?\\)\\//g, '')\n .replace(/^index$|\\/index$/, '')\n .replace(/\\[([^\\]]+)]/g, ':$1')\n || '/'\n}", "import {routePattern} from \"../utils/patterns\";\n\nexport interface ApiRoute {\n path: string\n key: string\n params: string[]\n regex: RegExp\n}\n\nexport interface ApiMiddleware {\n dir: string\n key: string\n}\n\nexport interface ApiResult {\n routes: ApiRoute[]\n middlewares: ApiMiddleware[]\n}\n\nexport function keyToRoutePattern(key: string, apiDir: string): string {\n const rel = key.slice(apiDir.length + 1).replace(/\\\\/g, '/')\n const pattern = routePattern(rel)\n return pattern === '/' ? '/api' : `/api/${pattern}`.replace('/api//', '/api/')\n}\n\nfunction keyToDir(key: string): string {\n return key.slice(0, key.lastIndexOf('/'))\n}\n\nlet cache: ApiResult | null = null\n\nexport function invalidateApiCache() {\n cache = null\n}\n\nexport function buildRoutes(routeKeys: string[], middlewareKeys: string[], apiDir: string): ApiResult {\n if (cache) return cache\n\n const routes: ApiRoute[] = []\n const middlewares: ApiMiddleware[] = []\n\n for (const key of middlewareKeys) {\n middlewares.push({dir: keyToDir(key), key})\n }\n\n for (const key of routeKeys) {\n const pattern = keyToRoutePattern(key, apiDir)\n const params = [...pattern.matchAll(/:([^/]+)/g)].map(m => m[1])\n const regexStr = pattern\n .replace(/:[^/]+/g, '([^/]+)')\n .replace(/\\//g, '\\\\/')\n routes.push({path: pattern, key, params, regex: new RegExp(`^${regexStr}$`)})\n }\n routes.sort((a, b) => {\n const aScore = (a.path.match(/:/g) || []).length\n const bScore = (b.path.match(/:/g) || []).length\n if (aScore !== bScore) return aScore - bScore\n return b.path.length - a.path.length\n })\n\n cache = {routes, middlewares}\n return cache\n}\n\nexport function collectMiddlewareChain(routeKey: string, middlewares: ApiMiddleware[]): ApiMiddleware[] {\n const routeDir = keyToDir(routeKey)\n\n return middlewares\n .filter(mw => routeDir.startsWith(mw.dir))\n .sort((a, b) => a.dir.split('/').length - b.dir.split('/').length)\n}\n\nexport function matchRoute(\n pathname: string,\n routes: ApiRoute[]\n): {route: ApiRoute; params: Record<string, string>} | null {\n for (const route of routes) {\n const match = pathname.match(route.regex)\n if (match) {\n const params: Record<string, string> = {}\n route.params.forEach((name, i) => {\n params[name] = decodeURIComponent(match[i + 1])\n })\n return {route, params}\n }\n }\n return null\n}\n", "import type {DevixHandler} from './create-handler'\n\nexport class RouteContext {\n readonly params: Record<string, string>\n private _state = new Map<string, unknown>()\n\n constructor(params: Record<string, string> = {}) {\n this.params = params\n }\n\n set<T>(key: string, value: T): void {\n this._state.set(key, value)\n }\n\n get<T>(key: string): T | undefined {\n return this._state.get(key) as T\n }\n}\n\nexport type RouteResult = Response | Record<string, unknown> | unknown[] | null | void\n\nexport type RouteHandler = (ctx: RouteContext, req: Request) => Promise<RouteResult> | RouteResult\n\nexport interface MiddlewareModule {\n middleware: (ctx: RouteContext, req: Request) => Promise<Response | null> | Response | null\n}\n\ntype AnyHandler = RouteHandler | DevixHandler<any, any>\n\nexport interface RouteModule {\n GET?: AnyHandler\n POST?: AnyHandler\n PUT?: AnyHandler\n PATCH?: AnyHandler\n DELETE?: AnyHandler\n HEAD?: AnyHandler\n OPTIONS?: AnyHandler\n}\n", "import {Component, ComponentType, ReactNode} from \"react\";\nimport {ErrorProps} from \"../server/types\";\n\ninterface Props {\n ErrorPage?: ComponentType<ErrorProps>\n children: ReactNode\n}\n\ninterface State {\n error: ErrorProps | null\n}\n\nexport class DevixErrorBoundary extends Component<Props, State> {\n state: State = { error: null }\n\n static getDerivedStateFromError(err: unknown): State {\n if (err instanceof DevixError) {\n return {\n error: {statusCode: err.statusCode, message: err.message}\n }\n }\n return {\n error: {statusCode: 500, message: err instanceof Error ? err.message : 'Unknown error'}\n }\n }\n\n render() {\n if (this.state.error && this.props.ErrorPage) {\n return <this.props.ErrorPage {...this.state.error} />\n }\n if (this.state.error) {\n return <h1>{this.state.error.statusCode}</h1>\n }\n return this.props.children\n }\n}\n\nexport class DevixError extends Error {\n statusCode: number\n constructor(statusCode: number, message: string) {\n super(message)\n this.statusCode = statusCode\n }\n}\n", "export const HANDLER_BRAND = '__devix_handler__' as const\n\nexport interface DevixHandler<TBody = undefined, TReturn = unknown> {\n readonly [HANDLER_BRAND]: true\n readonly fn: (...args: any[]) => any\n readonly __body?: TBody\n readonly __return?: TReturn\n}\n\ntype ExtractBody<TFn> = TFn extends (body: infer B) => any ? B : undefined\n\nexport function createHandler<TFn extends (...args: any[]) => any>(\n fn: TFn,\n): DevixHandler<ExtractBody<TFn>, Awaited<ReturnType<TFn>>> {\n return {[HANDLER_BRAND]: true, fn}\n}\n", "import {AsyncLocalStorage} from 'node:async_hooks'\nimport type {RouteContext} from '../runtime/api-context'\n\ninterface HandlerStore {\n request: Request\n ctx: RouteContext\n}\n\nconst storage = new AsyncLocalStorage<HandlerStore>()\n\nexport function withHandlerStore<T>(store: HandlerStore, fn: () => T): T {\n return storage.run(store, fn)\n}\n\nfunction getStore(hookName: string): HandlerStore {\n const store = storage.getStore()\n if (!store) throw new Error(`[devix] ${hookName}() called outside of a request handler`)\n return store\n}\n\nexport function useRequest(): Request {\n return getStore('useRequest').request\n}\n\nexport function useCtx(): RouteContext {\n return getStore('useCtx').ctx\n}\n\nexport function useParams(): Record<string, string> {\n return getStore('useParams').ctx.params\n}\n", "import {buildRoutes, matchRoute, collectMiddlewareChain} from './api-router'\nimport {RouteContext} from '../runtime/api-context'\nimport type {RouteModule, MiddlewareModule, RouteResult} from '../runtime/api-context'\nimport type {ApiGlob} from './types'\nimport {DevixError} from '../runtime/error-boundary'\nimport {HANDLER_BRAND, type DevixHandler} from '../runtime/create-handler'\nimport {withHandlerStore} from './handler-store'\n\nfunction isDevixHandler(h: unknown): h is DevixHandler<any, any> {\n return typeof h === 'object' && h !== null && HANDLER_BRAND in h\n}\n\nasync function parseBody(request: Request): Promise<unknown> {\n const ct = request.headers.get('Content-Type') ?? ''\n if (ct.includes('application/json')) return request.json()\n if (ct.includes('multipart/form-data') || ct.includes('application/x-www-form-urlencoded')) {\n return request.formData()\n }\n return request.text()\n}\n\nfunction resultToResponse(result: RouteResult): Response {\n if (result instanceof Response) return result\n if (result == null) return new Response(null, {status: 204})\n return new Response(JSON.stringify(result), {\n headers: {'Content-Type': 'application/json'},\n })\n}\n\nexport async function handleApiRequest(\n url: string,\n request: Request,\n glob: ApiGlob,\n): Promise<Response> {\n try {\n const {pathname} = new URL(url, 'http://localhost')\n const {routes, middlewares} = buildRoutes(\n Object.keys(glob.routes),\n Object.keys(glob.middlewares),\n glob.apiDir,\n )\n const matched = matchRoute(pathname, routes)\n\n if (!matched) return new Response('Not Found', {status: 404})\n\n const {route, params} = matched\n const ctx = new RouteContext(params)\n\n const result = await withHandlerStore({request, ctx}, async () => {\n const middlewareChain = collectMiddlewareChain(route.key, middlewares)\n for (const mw of middlewareChain) {\n const mod = await glob.middlewares[mw.key]() as MiddlewareModule\n if (mod.middleware) {\n const mwResult = await mod.middleware(ctx, request)\n if (mwResult instanceof Response) return mwResult\n }\n }\n\n const mod = await glob.routes[route.key]() as RouteModule\n const method = request.method.toUpperCase() as keyof RouteModule\n const handler = mod[method]\n\n if (!handler) return new Response('Method Not Allowed', {status: 405})\n\n if (isDevixHandler(handler)) {\n if (handler.fn.length === 0) {\n return handler.fn() as Promise<RouteResult>\n }\n const body = await parseBody(request)\n return handler.fn(body) as Promise<RouteResult>\n }\n\n return handler(ctx, request)\n })\n\n return resultToResponse(result)\n } catch (err) {\n console.error('[devix] api error:', err)\n if (err instanceof DevixError) {\n return new Response(err.message, {status: err.statusCode})\n }\n return new Response('Internal Server Error', {status: 500})\n }\n}\n"],
5
- "mappings": "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",
6
- "names": ["routePattern", "rel", "keyToRoutePattern", "key", "apiDir", "rel", "pattern", "routePattern", "keyToDir", "cache", "buildRoutes", "routeKeys", "middlewareKeys", "apiDir", "cache", "routes", "middlewares", "key", "keyToDir", "pattern", "keyToRoutePattern", "params", "m", "regexStr", "a", "b", "aScore", "bScore", "collectMiddlewareChain", "routeKey", "routeDir", "mw", "matchRoute", "pathname", "route", "match", "name", "i", "RouteContext", "params", "key", "value", "Component", "jsx", "DevixError", "statusCode", "message", "HANDLER_BRAND", "AsyncLocalStorage", "storage", "withHandlerStore", "store", "fn", "isDevixHandler", "h", "HANDLER_BRAND", "parseBody", "request", "ct", "resultToResponse", "result", "handleApiRequest", "url", "glob", "pathname", "routes", "middlewares", "buildRoutes", "matched", "matchRoute", "route", "params", "ctx", "RouteContext", "withHandlerStore", "middlewareChain", "collectMiddlewareChain", "mw", "mod", "mwResult", "method", "handler", "body", "err", "DevixError"]
4
+ "sourcesContent": ["export function routePattern(rel: string): string {\n return rel\n .replace(/\\.(tsx|ts|jsx|js)$/, '')\n .replace(/\\(.*?\\)\\//g, '')\n .replace(/^index$|\\/index$/, '')\n .replace(/\\[([^\\]]+)]/g, ':$1')\n || '/'\n}", "import {routePattern} from \"../utils/patterns\";\n\nexport interface ApiRoute {\n path: string\n key: string\n params: string[]\n regex: RegExp\n}\n\nexport interface ApiMiddleware {\n dir: string\n key: string\n}\n\nexport interface ApiResult {\n routes: ApiRoute[]\n middlewares: ApiMiddleware[]\n}\n\nexport function keyToRoutePattern(key: string, apiDir: string): string {\n const rel = key.slice(apiDir.length + 1).replace(/\\\\/g, '/')\n const pattern = routePattern(rel)\n return pattern === '/' ? '/api' : `/api/${pattern}`.replace('/api//', '/api/')\n}\n\nfunction keyToDir(key: string): string {\n return key.slice(0, key.lastIndexOf('/'))\n}\n\nlet cache: ApiResult | null = null\n\nexport function invalidateApiCache() {\n cache = null\n}\n\nexport function buildRoutes(routeKeys: string[], middlewareKeys: string[], apiDir: string): ApiResult {\n if (cache) return cache\n\n const routes: ApiRoute[] = []\n const middlewares: ApiMiddleware[] = []\n\n for (const key of middlewareKeys) {\n middlewares.push({dir: keyToDir(key), key})\n }\n\n for (const key of routeKeys) {\n const pattern = keyToRoutePattern(key, apiDir)\n const params = [...pattern.matchAll(/:([^/]+)/g)].map(m => m[1])\n const regexStr = pattern\n .replace(/:[^/]+/g, '([^/]+)')\n .replace(/\\//g, '\\\\/')\n routes.push({path: pattern, key, params, regex: new RegExp(`^${regexStr}$`)})\n }\n routes.sort((a, b) => {\n const aScore = (a.path.match(/:/g) || []).length\n const bScore = (b.path.match(/:/g) || []).length\n if (aScore !== bScore) return aScore - bScore\n return b.path.length - a.path.length\n })\n\n cache = {routes, middlewares}\n return cache\n}\n\nexport function collectMiddlewareChain(routeKey: string, middlewares: ApiMiddleware[]): ApiMiddleware[] {\n const routeDir = keyToDir(routeKey)\n\n return middlewares\n .filter(mw => routeDir.startsWith(mw.dir))\n .sort((a, b) => a.dir.split('/').length - b.dir.split('/').length)\n}\n\nexport function matchRoute(\n pathname: string,\n routes: ApiRoute[]\n): {route: ApiRoute; params: Record<string, string>} | null {\n for (const route of routes) {\n const match = pathname.match(route.regex)\n if (match) {\n const params: Record<string, string> = {}\n route.params.forEach((name, i) => {\n params[name] = decodeURIComponent(match[i + 1])\n })\n return {route, params}\n }\n }\n return null\n}\n", "import type {DevixHandler} from './create-handler'\n\nexport class RouteContext {\n readonly params: Record<string, string>\n private _state = new Map<string, unknown>()\n\n constructor(params: Record<string, string> = {}) {\n this.params = params\n }\n\n set<T>(key: string, value: T): void {\n this._state.set(key, value)\n }\n\n get<T>(key: string): T | undefined {\n return this._state.get(key) as T\n }\n}\n\nexport type RouteResult = Response | Record<string, unknown> | unknown[] | null | void\n\nexport type RouteHandler = (ctx: RouteContext, req: Request) => Promise<RouteResult> | RouteResult\n\nexport interface MiddlewareModule {\n middleware: (ctx: RouteContext, req: Request) => Promise<Response | null> | Response | null\n}\n\ntype AnyHandler = RouteHandler | DevixHandler<any, any>\n\nexport interface RouteModule {\n GET?: AnyHandler\n POST?: AnyHandler\n PUT?: AnyHandler\n PATCH?: AnyHandler\n DELETE?: AnyHandler\n HEAD?: AnyHandler\n OPTIONS?: AnyHandler\n}\n", "import {Component, ComponentType, ReactNode} from \"react\";\nimport {ErrorProps} from \"../server/types\";\n\ninterface Props {\n ErrorPage?: ComponentType<ErrorProps>\n children: ReactNode\n}\n\ninterface State {\n error: ErrorProps | null\n}\n\nexport class DevixErrorBoundary extends Component<Props, State> {\n state: State = { error: null }\n\n static getDerivedStateFromError(err: unknown): State {\n if (err instanceof DevixError) {\n return {\n error: {statusCode: err.statusCode, message: err.message}\n }\n }\n return {\n error: {statusCode: 500, message: err instanceof Error ? err.message : 'Unknown error'}\n }\n }\n\n render() {\n if (this.state.error && this.props.ErrorPage) {\n return <this.props.ErrorPage {...this.state.error} />\n }\n if (this.state.error) {\n return <h1>{this.state.error.statusCode}</h1>\n }\n return this.props.children\n }\n}\n\nexport class DevixError extends Error {\n statusCode: number\n constructor(statusCode: number, message: string) {\n super(message)\n this.statusCode = statusCode\n }\n}\n", "export const HANDLER_BRAND = '__devix_handler__' as const\n\nexport interface DevixHandler<TBody = undefined, TReturn = unknown> {\n readonly [HANDLER_BRAND]: true\n readonly fn: (...args: any[]) => any\n readonly __body: TBody\n readonly __return: TReturn\n}\n\ntype ExtractBody<TFn> = TFn extends (body: infer B) => any ? B : undefined\n\nexport function createHandler<TFn extends (...args: any[]) => any>(\n fn: TFn,\n): DevixHandler<ExtractBody<TFn>, Awaited<ReturnType<TFn>>> {\n return {[HANDLER_BRAND]: true, fn} as unknown as DevixHandler<ExtractBody<TFn>, Awaited<ReturnType<TFn>>>\n}\n", "import {AsyncLocalStorage} from 'node:async_hooks'\nimport type {RouteContext} from '../runtime/api-context'\n\ninterface HandlerStore {\n request: Request\n ctx: RouteContext\n}\n\nconst ALS_KEY = Symbol.for('@devlusoft/devix.handlerStore')\nconst g = globalThis as Record<symbol, unknown>\nif (!g[ALS_KEY]) g[ALS_KEY] = new AsyncLocalStorage<HandlerStore>()\nconst storage = g[ALS_KEY] as AsyncLocalStorage<HandlerStore>\n\nexport function withHandlerStore<T>(store: HandlerStore, fn: () => T): T {\n return storage.run(store, fn)\n}\n\nfunction getStore(hookName: string): HandlerStore {\n const store = storage.getStore()\n if (!store) throw new Error(`[devix] ${hookName}() called outside of a request handler`)\n return store\n}\n\nexport function useRequest(): Request {\n return getStore('useRequest').request\n}\n\nexport function useCtx(): RouteContext {\n return getStore('useCtx').ctx\n}\n\nexport function useParams(): Record<string, string> {\n return getStore('useParams').ctx.params\n}\n", "import {buildRoutes, matchRoute, collectMiddlewareChain} from './api-router'\nimport {RouteContext} from '../runtime/api-context'\nimport type {RouteModule, MiddlewareModule, RouteResult} from '../runtime/api-context'\nimport type {ApiGlob} from './types'\nimport {DevixError} from '../runtime/error-boundary'\nimport {HANDLER_BRAND, type DevixHandler} from '../runtime/create-handler'\nimport {withHandlerStore} from './handler-store'\n\nfunction isDevixHandler(h: unknown): h is DevixHandler<any, any> {\n return typeof h === 'object' && h !== null && HANDLER_BRAND in h\n}\n\nasync function parseBody(request: Request): Promise<unknown> {\n const ct = request.headers.get('Content-Type') ?? ''\n if (ct.includes('application/json')) return request.json()\n if (ct.includes('multipart/form-data') || ct.includes('application/x-www-form-urlencoded')) {\n return request.formData()\n }\n return request.text()\n}\n\nfunction resultToResponse(result: RouteResult): Response {\n if (result instanceof Response) return result\n if (result == null) return new Response(null, {status: 204})\n return new Response(JSON.stringify(result), {\n headers: {'Content-Type': 'application/json'},\n })\n}\n\nexport async function handleApiRequest(\n url: string,\n request: Request,\n glob: ApiGlob,\n): Promise<Response> {\n try {\n const {pathname} = new URL(url, 'http://localhost')\n const {routes, middlewares} = buildRoutes(\n Object.keys(glob.routes),\n Object.keys(glob.middlewares),\n glob.apiDir,\n )\n const matched = matchRoute(pathname, routes)\n\n if (!matched) return new Response('Not Found', {status: 404})\n\n const {route, params} = matched\n const ctx = new RouteContext(params)\n\n const result = await withHandlerStore({request, ctx}, async () => {\n const middlewareChain = collectMiddlewareChain(route.key, middlewares)\n for (const mw of middlewareChain) {\n const mod = await glob.middlewares[mw.key]() as MiddlewareModule\n if (mod.middleware) {\n const mwResult = await mod.middleware(ctx, request)\n if (mwResult instanceof Response) return mwResult\n }\n }\n\n const mod = await glob.routes[route.key]() as RouteModule\n const method = request.method.toUpperCase() as keyof RouteModule\n const handler = mod[method]\n\n if (!handler) return new Response('Method Not Allowed', {status: 405})\n\n if (isDevixHandler(handler)) {\n if (handler.fn.length === 0) {\n return handler.fn() as Promise<RouteResult>\n }\n const body = await parseBody(request)\n return handler.fn(body) as Promise<RouteResult>\n }\n\n return handler(ctx, request)\n })\n\n return resultToResponse(result)\n } catch (err) {\n console.error('[devix] api error:', err)\n if (err instanceof DevixError) {\n return new Response(err.message, {status: err.statusCode})\n }\n return new Response('Internal Server Error', {status: 500})\n }\n}\n"],
5
+ "mappings": "AAAO,SAASA,EAAaC,EAAqB,CAC9C,OAAOA,EACE,QAAQ,qBAAsB,EAAE,EAChC,QAAQ,aAAc,EAAE,EACxB,QAAQ,mBAAoB,EAAE,EAC9B,QAAQ,eAAgB,KAAK,GAC/B,GACX,CCYO,SAASC,EAAkBC,EAAaC,EAAwB,CACnE,IAAMC,EAAMF,EAAI,MAAMC,EAAO,OAAS,CAAC,EAAE,QAAQ,MAAO,GAAG,EACrDE,EAAUC,EAAaF,CAAG,EAChC,OAAOC,IAAY,IAAM,OAAS,QAAQA,CAAO,GAAG,QAAQ,SAAU,OAAO,CACjF,CAEA,SAASE,EAASL,EAAqB,CACnC,OAAOA,EAAI,MAAM,EAAGA,EAAI,YAAY,GAAG,CAAC,CAC5C,CAEA,IAAIM,EAA0B,KAMvB,SAASC,EAAYC,EAAqBC,EAA0BC,EAA2B,CAClG,GAAIC,EAAO,OAAOA,EAElB,IAAMC,EAAqB,CAAC,EACtBC,EAA+B,CAAC,EAEtC,QAAWC,KAAOL,EACdI,EAAY,KAAK,CAAC,IAAKE,EAASD,CAAG,EAAG,IAAAA,CAAG,CAAC,EAG9C,QAAWA,KAAON,EAAW,CACzB,IAAMQ,EAAUC,EAAkBH,EAAKJ,CAAM,EACvCQ,EAAS,CAAC,GAAGF,EAAQ,SAAS,WAAW,CAAC,EAAE,IAAIG,GAAKA,EAAE,CAAC,CAAC,EACzDC,EAAWJ,EACZ,QAAQ,UAAW,SAAS,EAC5B,QAAQ,MAAO,KAAK,EACzBJ,EAAO,KAAK,CAAC,KAAMI,EAAS,IAAAF,EAAK,OAAAI,EAAQ,MAAO,IAAI,OAAO,IAAIE,CAAQ,GAAG,CAAC,CAAC,CAChF,CACA,OAAAR,EAAO,KAAK,CAACS,EAAGC,IAAM,CAClB,IAAMC,GAAUF,EAAE,KAAK,MAAM,IAAI,GAAK,CAAC,GAAG,OACpCG,GAAUF,EAAE,KAAK,MAAM,IAAI,GAAK,CAAC,GAAG,OAC1C,OAAIC,IAAWC,EAAeD,EAASC,EAChCF,EAAE,KAAK,OAASD,EAAE,KAAK,MAClC,CAAC,EAEDV,EAAQ,CAAC,OAAAC,EAAQ,YAAAC,CAAW,EACrBF,CACX,CAEO,SAASc,EAAuBC,EAAkBb,EAA+C,CACpG,IAAMc,EAAWZ,EAASW,CAAQ,EAElC,OAAOb,EACF,OAAOe,GAAMD,EAAS,WAAWC,EAAG,GAAG,CAAC,EACxC,KAAK,CAACP,EAAGC,IAAMD,EAAE,IAAI,MAAM,GAAG,EAAE,OAASC,EAAE,IAAI,MAAM,GAAG,EAAE,MAAM,CACzE,CAEO,SAASO,EACZC,EACAlB,EACwD,CACxD,QAAWmB,KAASnB,EAAQ,CACxB,IAAMoB,EAAQF,EAAS,MAAMC,EAAM,KAAK,EACxC,GAAIC,EAAO,CACP,IAAMd,EAAiC,CAAC,EACxC,OAAAa,EAAM,OAAO,QAAQ,CAACE,EAAMC,IAAM,CAC9BhB,EAAOe,CAAI,EAAI,mBAAmBD,EAAME,EAAI,CAAC,CAAC,CAClD,CAAC,EACM,CAAC,MAAAH,EAAO,OAAAb,CAAM,CACzB,CACJ,CACA,OAAO,IACX,CCrFO,IAAMiB,EAAN,KAAmB,CACb,OACD,OAAS,IAAI,IAErB,YAAYC,EAAiC,CAAC,EAAG,CAC7C,KAAK,OAASA,CAClB,CAEA,IAAOC,EAAaC,EAAgB,CAChC,KAAK,OAAO,IAAID,EAAKC,CAAK,CAC9B,CAEA,IAAOD,EAA4B,CAC/B,OAAO,KAAK,OAAO,IAAIA,CAAG,CAC9B,CACJ,ECjBA,OAAQ,aAAAE,MAA0C,QA4B/B,cAAAC,MAAA,oBASZ,IAAMC,EAAN,cAAyB,KAAM,CAClC,WACA,YAAYC,EAAoBC,EAAiB,CAC7C,MAAMA,CAAO,EACb,KAAK,WAAaD,CACtB,CACJ,EC3CO,IAAME,EAAgB,oBCA7B,OAAQ,qBAAAC,MAAwB,mBAQhC,IAAMC,EAAU,OAAO,IAAI,+BAA+B,EACpD,EAAI,WACL,EAAEA,CAAO,IAAG,EAAEA,CAAO,EAAI,IAAID,GAClC,IAAME,EAAU,EAAED,CAAO,EAElB,SAASE,EAAoBC,EAAqBC,EAAgB,CACrE,OAAOH,EAAQ,IAAIE,EAAOC,CAAE,CAChC,CCPA,SAASC,EAAeC,EAAyC,CAC7D,OAAO,OAAOA,GAAM,UAAYA,IAAM,MAAQC,KAAiBD,CACnE,CAEA,eAAeE,EAAUC,EAAoC,CACzD,IAAMC,EAAKD,EAAQ,QAAQ,IAAI,cAAc,GAAK,GAClD,OAAIC,EAAG,SAAS,kBAAkB,EAAUD,EAAQ,KAAK,EACrDC,EAAG,SAAS,qBAAqB,GAAKA,EAAG,SAAS,mCAAmC,EAC9ED,EAAQ,SAAS,EAErBA,EAAQ,KAAK,CACxB,CAEA,SAASE,EAAiBC,EAA+B,CACrD,OAAIA,aAAkB,SAAiBA,EACnCA,GAAU,KAAa,IAAI,SAAS,KAAM,CAAC,OAAQ,GAAG,CAAC,EACpD,IAAI,SAAS,KAAK,UAAUA,CAAM,EAAG,CACxC,QAAS,CAAC,eAAgB,kBAAkB,CAChD,CAAC,CACL,CAEA,eAAsBC,GAClBC,EACAL,EACAM,EACiB,CACjB,GAAI,CACA,GAAM,CAAC,SAAAC,CAAQ,EAAI,IAAI,IAAIF,EAAK,kBAAkB,EAC5C,CAAC,OAAAG,EAAQ,YAAAC,CAAW,EAAIC,EAC1B,OAAO,KAAKJ,EAAK,MAAM,EACvB,OAAO,KAAKA,EAAK,WAAW,EAC5BA,EAAK,MACT,EACMK,EAAUC,EAAWL,EAAUC,CAAM,EAE3C,GAAI,CAACG,EAAS,OAAO,IAAI,SAAS,YAAa,CAAC,OAAQ,GAAG,CAAC,EAE5D,GAAM,CAAC,MAAAE,EAAO,OAAAC,CAAM,EAAIH,EAClBI,EAAM,IAAIC,EAAaF,CAAM,EAE7BX,EAAS,MAAMc,EAAiB,CAAC,QAAAjB,EAAS,IAAAe,CAAG,EAAG,SAAY,CAC9D,IAAMG,EAAkBC,EAAuBN,EAAM,IAAKJ,CAAW,EACrE,QAAWW,KAAMF,EAAiB,CAC9B,IAAMG,EAAM,MAAMf,EAAK,YAAYc,EAAG,GAAG,EAAE,EAC3C,GAAIC,EAAI,WAAY,CAChB,IAAMC,EAAW,MAAMD,EAAI,WAAWN,EAAKf,CAAO,EAClD,GAAIsB,aAAoB,SAAU,OAAOA,CAC7C,CACJ,CAEA,IAAMD,EAAM,MAAMf,EAAK,OAAOO,EAAM,GAAG,EAAE,EACnCU,EAASvB,EAAQ,OAAO,YAAY,EACpCwB,EAAUH,EAAIE,CAAM,EAE1B,GAAI,CAACC,EAAS,OAAO,IAAI,SAAS,qBAAsB,CAAC,OAAQ,GAAG,CAAC,EAErE,GAAI5B,EAAe4B,CAAO,EAAG,CACzB,GAAIA,EAAQ,GAAG,SAAW,EACtB,OAAOA,EAAQ,GAAG,EAEtB,IAAMC,EAAO,MAAM1B,EAAUC,CAAO,EACpC,OAAOwB,EAAQ,GAAGC,CAAI,CAC1B,CAEA,OAAOD,EAAQT,EAAKf,CAAO,CAC/B,CAAC,EAED,OAAOE,EAAiBC,CAAM,CAClC,OAASuB,EAAK,CAEV,OADA,QAAQ,MAAM,qBAAsBA,CAAG,EACnCA,aAAeC,EACR,IAAI,SAASD,EAAI,QAAS,CAAC,OAAQA,EAAI,UAAU,CAAC,EAEtD,IAAI,SAAS,wBAAyB,CAAC,OAAQ,GAAG,CAAC,CAC9D,CACJ",
6
+ "names": ["routePattern", "rel", "keyToRoutePattern", "key", "apiDir", "rel", "pattern", "routePattern", "keyToDir", "cache", "buildRoutes", "routeKeys", "middlewareKeys", "apiDir", "cache", "routes", "middlewares", "key", "keyToDir", "pattern", "keyToRoutePattern", "params", "m", "regexStr", "a", "b", "aScore", "bScore", "collectMiddlewareChain", "routeKey", "routeDir", "mw", "matchRoute", "pathname", "route", "match", "name", "i", "RouteContext", "params", "key", "value", "Component", "jsx", "DevixError", "statusCode", "message", "HANDLER_BRAND", "AsyncLocalStorage", "ALS_KEY", "storage", "withHandlerStore", "store", "fn", "isDevixHandler", "h", "HANDLER_BRAND", "parseBody", "request", "ct", "resultToResponse", "result", "handleApiRequest", "url", "glob", "pathname", "routes", "middlewares", "buildRoutes", "matched", "matchRoute", "route", "params", "ctx", "RouteContext", "withHandlerStore", "middlewareChain", "collectMiddlewareChain", "mw", "mod", "mwResult", "method", "handler", "body", "err", "DevixError"]
7
7
  }
@@ -1,2 +1,2 @@
1
- import{AsyncLocalStorage as n}from"node:async_hooks";var o=new n;function s(t,e){return o.run(t,e)}function r(t){let e=o.getStore();if(!e)throw new Error(`[devix] ${t}() called outside of a request handler`);return e}function a(){return r("useRequest").request}function i(){return r("useCtx").ctx}function c(){return r("useParams").ctx.params}export{i as useCtx,c as useParams,a as useRequest,s as withHandlerStore};
1
+ import{AsyncLocalStorage as u}from"node:async_hooks";var r=Symbol.for("@devlusoft/devix.handlerStore"),o=globalThis;o[r]||(o[r]=new u);var s=o[r];function c(t,e){return s.run(t,e)}function n(t){let e=s.getStore();if(!e)throw new Error(`[devix] ${t}() called outside of a request handler`);return e}function i(){return n("useRequest").request}function l(){return n("useCtx").ctx}function d(){return n("useParams").ctx.params}export{l as useCtx,d as useParams,i as useRequest,c as withHandlerStore};
2
2
  //# sourceMappingURL=handler-store.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/server/handler-store.ts"],
4
- "sourcesContent": ["import {AsyncLocalStorage} from 'node:async_hooks'\nimport type {RouteContext} from '../runtime/api-context'\n\ninterface HandlerStore {\n request: Request\n ctx: RouteContext\n}\n\nconst storage = new AsyncLocalStorage<HandlerStore>()\n\nexport function withHandlerStore<T>(store: HandlerStore, fn: () => T): T {\n return storage.run(store, fn)\n}\n\nfunction getStore(hookName: string): HandlerStore {\n const store = storage.getStore()\n if (!store) throw new Error(`[devix] ${hookName}() called outside of a request handler`)\n return store\n}\n\nexport function useRequest(): Request {\n return getStore('useRequest').request\n}\n\nexport function useCtx(): RouteContext {\n return getStore('useCtx').ctx\n}\n\nexport function useParams(): Record<string, string> {\n return getStore('useParams').ctx.params\n}\n"],
5
- "mappings": "AAAA,OAAQ,qBAAAA,MAAwB,mBAQhC,IAAMC,EAAU,IAAID,EAEb,SAASE,EAAoBC,EAAqBC,EAAgB,CACrE,OAAOH,EAAQ,IAAIE,EAAOC,CAAE,CAChC,CAEA,SAASC,EAASC,EAAgC,CAC9C,IAAMH,EAAQF,EAAQ,SAAS,EAC/B,GAAI,CAACE,EAAO,MAAM,IAAI,MAAM,WAAWG,CAAQ,wCAAwC,EACvF,OAAOH,CACX,CAEO,SAASI,GAAsB,CAClC,OAAOF,EAAS,YAAY,EAAE,OAClC,CAEO,SAASG,GAAuB,CACnC,OAAOH,EAAS,QAAQ,EAAE,GAC9B,CAEO,SAASI,GAAoC,CAChD,OAAOJ,EAAS,WAAW,EAAE,IAAI,MACrC",
6
- "names": ["AsyncLocalStorage", "storage", "withHandlerStore", "store", "fn", "getStore", "hookName", "useRequest", "useCtx", "useParams"]
4
+ "sourcesContent": ["import {AsyncLocalStorage} from 'node:async_hooks'\nimport type {RouteContext} from '../runtime/api-context'\n\ninterface HandlerStore {\n request: Request\n ctx: RouteContext\n}\n\nconst ALS_KEY = Symbol.for('@devlusoft/devix.handlerStore')\nconst g = globalThis as Record<symbol, unknown>\nif (!g[ALS_KEY]) g[ALS_KEY] = new AsyncLocalStorage<HandlerStore>()\nconst storage = g[ALS_KEY] as AsyncLocalStorage<HandlerStore>\n\nexport function withHandlerStore<T>(store: HandlerStore, fn: () => T): T {\n return storage.run(store, fn)\n}\n\nfunction getStore(hookName: string): HandlerStore {\n const store = storage.getStore()\n if (!store) throw new Error(`[devix] ${hookName}() called outside of a request handler`)\n return store\n}\n\nexport function useRequest(): Request {\n return getStore('useRequest').request\n}\n\nexport function useCtx(): RouteContext {\n return getStore('useCtx').ctx\n}\n\nexport function useParams(): Record<string, string> {\n return getStore('useParams').ctx.params\n}\n"],
5
+ "mappings": "AAAA,OAAQ,qBAAAA,MAAwB,mBAQhC,IAAMC,EAAU,OAAO,IAAI,+BAA+B,EACpDC,EAAI,WACLA,EAAED,CAAO,IAAGC,EAAED,CAAO,EAAI,IAAID,GAClC,IAAMG,EAAUD,EAAED,CAAO,EAElB,SAASG,EAAoBC,EAAqBC,EAAgB,CACrE,OAAOH,EAAQ,IAAIE,EAAOC,CAAE,CAChC,CAEA,SAASC,EAASC,EAAgC,CAC9C,IAAMH,EAAQF,EAAQ,SAAS,EAC/B,GAAI,CAACE,EAAO,MAAM,IAAI,MAAM,WAAWG,CAAQ,wCAAwC,EACvF,OAAOH,CACX,CAEO,SAASI,GAAsB,CAClC,OAAOF,EAAS,YAAY,EAAE,OAClC,CAEO,SAASG,GAAuB,CACnC,OAAOH,EAAS,QAAQ,EAAE,GAC9B,CAEO,SAASI,GAAoC,CAChD,OAAOJ,EAAS,WAAW,EAAE,IAAI,MACrC",
6
+ "names": ["AsyncLocalStorage", "ALS_KEY", "g", "storage", "withHandlerStore", "store", "fn", "getStore", "hookName", "useRequest", "useCtx", "useParams"]
7
7
  }
@@ -1,2 +1,2 @@
1
- import{AsyncLocalStorage as o}from"node:async_hooks";var n=new o;function e(r){let t=n.getStore();if(!t)throw new Error(`[devix] ${r}() called outside of a request handler`);return t}function u(){return e("useRequest").request}function s(){return e("useCtx").ctx}function a(){return e("useParams").ctx.params}export{s as useCtx,a as useParams,u as useRequest};
1
+ import{AsyncLocalStorage as s}from"node:async_hooks";var e=Symbol.for("@devlusoft/devix.handlerStore"),t=globalThis;t[e]||(t[e]=new s);var u=t[e];function r(n){let o=u.getStore();if(!o)throw new Error(`[devix] ${n}() called outside of a request handler`);return o}function a(){return r("useRequest").request}function c(){return r("useCtx").ctx}function i(){return r("useParams").ctx.params}export{c as useCtx,i as useParams,a as useRequest};
2
2
  //# sourceMappingURL=public-index.js.map
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/server/handler-store.ts"],
4
- "sourcesContent": ["import {AsyncLocalStorage} from 'node:async_hooks'\nimport type {RouteContext} from '../runtime/api-context'\n\ninterface HandlerStore {\n request: Request\n ctx: RouteContext\n}\n\nconst storage = new AsyncLocalStorage<HandlerStore>()\n\nexport function withHandlerStore<T>(store: HandlerStore, fn: () => T): T {\n return storage.run(store, fn)\n}\n\nfunction getStore(hookName: string): HandlerStore {\n const store = storage.getStore()\n if (!store) throw new Error(`[devix] ${hookName}() called outside of a request handler`)\n return store\n}\n\nexport function useRequest(): Request {\n return getStore('useRequest').request\n}\n\nexport function useCtx(): RouteContext {\n return getStore('useCtx').ctx\n}\n\nexport function useParams(): Record<string, string> {\n return getStore('useParams').ctx.params\n}\n"],
5
- "mappings": "AAAA,OAAQ,qBAAAA,MAAwB,mBAQhC,IAAMC,EAAU,IAAID,EAMpB,SAASE,EAASC,EAAgC,CAC9C,IAAMC,EAAQC,EAAQ,SAAS,EAC/B,GAAI,CAACD,EAAO,MAAM,IAAI,MAAM,WAAWD,CAAQ,wCAAwC,EACvF,OAAOC,CACX,CAEO,SAASE,GAAsB,CAClC,OAAOJ,EAAS,YAAY,EAAE,OAClC,CAEO,SAASK,GAAuB,CACnC,OAAOL,EAAS,QAAQ,EAAE,GAC9B,CAEO,SAASM,GAAoC,CAChD,OAAON,EAAS,WAAW,EAAE,IAAI,MACrC",
6
- "names": ["AsyncLocalStorage", "storage", "getStore", "hookName", "store", "storage", "useRequest", "useCtx", "useParams"]
4
+ "sourcesContent": ["import {AsyncLocalStorage} from 'node:async_hooks'\nimport type {RouteContext} from '../runtime/api-context'\n\ninterface HandlerStore {\n request: Request\n ctx: RouteContext\n}\n\nconst ALS_KEY = Symbol.for('@devlusoft/devix.handlerStore')\nconst g = globalThis as Record<symbol, unknown>\nif (!g[ALS_KEY]) g[ALS_KEY] = new AsyncLocalStorage<HandlerStore>()\nconst storage = g[ALS_KEY] as AsyncLocalStorage<HandlerStore>\n\nexport function withHandlerStore<T>(store: HandlerStore, fn: () => T): T {\n return storage.run(store, fn)\n}\n\nfunction getStore(hookName: string): HandlerStore {\n const store = storage.getStore()\n if (!store) throw new Error(`[devix] ${hookName}() called outside of a request handler`)\n return store\n}\n\nexport function useRequest(): Request {\n return getStore('useRequest').request\n}\n\nexport function useCtx(): RouteContext {\n return getStore('useCtx').ctx\n}\n\nexport function useParams(): Record<string, string> {\n return getStore('useParams').ctx.params\n}\n"],
5
+ "mappings": "AAAA,OAAQ,qBAAAA,MAAwB,mBAQhC,IAAMC,EAAU,OAAO,IAAI,+BAA+B,EACpDC,EAAI,WACLA,EAAED,CAAO,IAAGC,EAAED,CAAO,EAAI,IAAID,GAClC,IAAMG,EAAUD,EAAED,CAAO,EAMzB,SAASG,EAASC,EAAgC,CAC9C,IAAMC,EAAQC,EAAQ,SAAS,EAC/B,GAAI,CAACD,EAAO,MAAM,IAAI,MAAM,WAAWD,CAAQ,wCAAwC,EACvF,OAAOC,CACX,CAEO,SAASE,GAAsB,CAClC,OAAOJ,EAAS,YAAY,EAAE,OAClC,CAEO,SAASK,GAAuB,CACnC,OAAOL,EAAS,QAAQ,EAAE,GAC9B,CAEO,SAASM,GAAoC,CAChD,OAAON,EAAS,WAAW,EAAE,IAAI,MACrC",
6
+ "names": ["AsyncLocalStorage", "ALS_KEY", "g", "storage", "getStore", "hookName", "store", "storage", "useRequest", "useCtx", "useParams"]
7
7
  }
@@ -3,11 +3,35 @@ import type { Manifest } from "vite";
3
3
  export declare function runLoader(url: string, request: Request, glob: PageGlob, options?: {
4
4
  loaderTimeout?: number;
5
5
  }): Promise<{
6
+ error: true;
6
7
  loaderData: null;
7
8
  params: {};
8
9
  layouts: never[];
9
10
  metadata: null;
10
11
  viewport: undefined;
12
+ redirect?: undefined;
13
+ redirectStatus?: undefined;
14
+ redirectReplace?: undefined;
15
+ } | {
16
+ loaderData: null;
17
+ params: {};
18
+ layouts: never[];
19
+ metadata: null;
20
+ viewport: undefined;
21
+ error?: undefined;
22
+ redirect?: undefined;
23
+ redirectStatus?: undefined;
24
+ redirectReplace?: undefined;
25
+ } | {
26
+ redirect: string | undefined;
27
+ redirectStatus: number | undefined;
28
+ redirectReplace: boolean | undefined;
29
+ error?: undefined;
30
+ loaderData?: undefined;
31
+ params?: undefined;
32
+ layouts?: undefined;
33
+ metadata?: undefined;
34
+ viewport?: undefined;
11
35
  } | {
12
36
  loaderData: unknown;
13
37
  params: Record<string, string>;
@@ -16,13 +40,17 @@ export declare function runLoader(url: string, request: Request, glob: PageGlob,
16
40
  }[];
17
41
  metadata: import("../types").Metadata;
18
42
  viewport: import("../types").Viewport | undefined;
43
+ error?: undefined;
44
+ redirect?: undefined;
45
+ redirectStatus?: undefined;
46
+ redirectReplace?: undefined;
19
47
  }>;
20
48
  export declare function render(url: string, request: Request, glob: PageGlob, options?: {
21
49
  manifest?: Manifest;
22
50
  loaderTimeout?: number;
23
51
  }): Promise<{
24
52
  html: string;
25
- statusCode: number;
53
+ statusCode: number | undefined;
26
54
  headers: {
27
55
  Location: string | undefined;
28
56
  };
@@ -1,2 +1,2 @@
1
- import{createElement as T}from"react";import{renderToString as X,renderToStaticMarkup as F}from"react-dom/server";import{Fragment as H,jsx as C}from"react/jsx-runtime";function Y(t,r){let e=[];t.title&&e.push({tag:"title",children:t.title}),t.description&&e.push({tag:"meta",name:"description",content:t.description}),t.keywords?.length&&e.push({tag:"meta",name:"keywords",content:t.keywords.join(", ")});let a=t.og?.title??t.title;a&&e.push({tag:"meta",property:"og:title",content:a});let n=t.og?.description??t.description;n&&e.push({tag:"meta",property:"og:description",content:n}),t.og?.image&&e.push({tag:"meta",property:"og:image",content:t.og.image}),t.og?.type&&e.push({tag:"meta",property:"og:type",content:t.og.type}),t.og?.url&&e.push({tag:"meta",property:"og:url",content:t.og.url});let o=t.twitter?.title??t.title;o&&e.push({tag:"meta",name:"twitter:title",content:o});let s=t.twitter?.description??t.description;if(s&&e.push({tag:"meta",name:"twitter:description",content:s}),t.twitter?.card&&e.push({tag:"meta",name:"twitter:card",content:t.twitter.card}),t.twitter?.image&&e.push({tag:"meta",name:"twitter:image",content:t.twitter.image}),t.twitter?.creator&&e.push({tag:"meta",name:"twitter:creator",content:t.twitter.creator}),t.canonical&&e.push({tag:"link",rel:"canonical",href:t.canonical}),t.robots&&e.push({tag:"meta",name:"robots",content:t.robots}),t.alternates)for(let[i,l]of Object.entries(t.alternates))e.push({tag:"link",rel:"alternate",href:l,hrefLang:i});if(r){let i=[];r.width!==void 0&&i.push(`width=${r.width}`),r.initialScale!==void 0&&i.push(`initial-scale=${r.initialScale}`),r.maximumScale!==void 0&&i.push(`maximum-scale=${r.maximumScale}`),r.userScalable!==void 0&&i.push(`user-scalable=${r.userScalable?"yes":"no"}`),i.length&&e.push({tag:"meta",name:"viewport",content:i.join(", ")}),r.themeColor&&e.push({tag:"meta",name:"theme-color",content:r.themeColor})}return e}function E(t,r){let e=Y(t,r);return C(H,{children:e.map((a,n)=>a.tag==="title"?C("title",{children:a.children},n):a.tag==="link"?C("link",{rel:a.rel,href:a.href,hrefLang:a.hrefLang},n):C("meta",{name:a.name,property:a.property,content:a.content},n))})}function O(t){return t.replace(/\.(tsx|ts|jsx|js)$/,"").replace(/\(.*?\)\//g,"").replace(/^index$|\/index$/,"").replace(/\[([^\]]+)]/g,":$1")||"/"}function J(t,r){let e=t.slice(r.length+1).replace(/\\/g,"/"),a=O(e);return a==="/"?"/":`/${a}`}function j(t){return t.slice(0,t.lastIndexOf("/"))}var D=null;function $(t,r,e){if(D)return D;let a=[],n=[];for(let o of r)n.push({dir:j(o),key:o});for(let o of t){let s=J(o,e),i=[...s.matchAll(/:([^/]+)/g)].map(d=>d[1]),l=s.replace(/:[^/]+/g,"([^/]+)").replace(/\//g,"\\/");a.push({path:s,key:o,params:i,regex:new RegExp(`^${l}$`)})}return a.sort((o,s)=>{let i=(o.path.match(/:/g)||[]).length,l=(s.path.match(/:/g)||[]).length;return i!==l?i-l:s.path.length-o.path.length}),D={pages:a,layouts:n},D}function N(t,r){let e=j(t);return r.filter(a=>e.startsWith(a.dir)).sort((a,n)=>a.dir.split("/").length-n.dir.split("/").length)}function I(t,r){for(let e of r){let a=t.match(e.regex);if(a){let n={};return e.params.forEach((o,s)=>{n[o]=decodeURIComponent(a[s+1])}),{page:e,params:n}}}return null}async function b(t,r){let e=t.generateMetadata?await t.generateMetadata(r):t.metadata??{},a=t.generateViewport?await t.generateViewport(r):t.viewport;return{metadata:e,viewport:a}}function U(...t){let r={};for(let e of t){if(!e)continue;let{og:a,twitter:n,...o}=e;Object.assign(r,o),a&&(r.og={...r.og,...a}),n&&(r.twitter={...r.twitter,...n})}return r}import{createContext as S}from"react";var y=globalThis;y.__devix_RouterContext__??=S(null);var it=y.__devix_RouterContext__;y.__devix_PageMetaContext__??=S(null);y.__devix_RouteDataContext__??=S(null);var st=y.__devix_PageMetaContext__,A=y.__devix_RouteDataContext__;function M(t){return JSON.stringify(t).replace(/<\/script>/gi,"<\\/script>")}function G(t){return t.replace(/"/g,"&quot;")}function V(t,r){let e;return Promise.race([t.finally(()=>clearTimeout(e)),new Promise((a,n)=>{e=setTimeout(()=>n(new Error(`timed out after ${r}ms`)),r)})])}var W="/@id/virtual:devix/entry-client";async function q(t,r,e,a){let{pages:n,layouts:o}=$(Object.keys(e.pages),Object.keys(e.layouts),e.pagesDir),s=I(t,n);if(!s)return null;let{page:i,params:l}=s,d=N(i.key,o),g={params:l,request:r},c=await e.pages[i.key]();if(c.guard){let p=await c.guard(g);if(p)return{redirect:p}}let m=c.loader?await V(c.loader(g),a):null,f=await Promise.all(d.map(p=>e.layouts[p.key]())),h=await V(Promise.all(f.map(p=>p.loader?p.loader(g):null)),a),w=await b(c,{...g,loaderData:m}),R=await Promise.all(f.map((p,k)=>b(p,{...g,loaderData:h[k]}))),_=U(...R.map(p=>p.metadata),w.metadata),v=w.viewport??R.findLast(p=>p.viewport)?.viewport,x=f[0],L=x?.generateLang?await x.generateLang({...g,loaderData:m}):x?.lang??"en";return{pageMod:c,layoutMods:f,params:l,loaderData:m,layoutsData:h,metadata:_,viewport:v,lang:L}}async function Pt(t,r,e,a){let{pathname:n}=new URL(t,"http://localhost"),o;try{let c=a?.loaderTimeout??1e4;o=await q(n,r,e,c)}catch(c){return console.error("[devix] render error:",c),{loaderData:null,params:{},layouts:[],metadata:null,viewport:void 0}}if(!o||"redirect"in o)return{loaderData:null,params:{},layouts:[],metadata:null,viewport:void 0};let{loaderData:s,params:i,layoutsData:l,metadata:d,viewport:g}=o;return{loaderData:s,params:i,layouts:l.map(c=>({loaderData:c})),metadata:d,viewport:g}}async function Mt(t,r,e,a){let n=a?.manifest?`/${Object.values(a.manifest).find(u=>u.isEntry)?.file}`:W,s=(a?.manifest?Object.values(a.manifest).find(u=>u.isEntry)?.css??[]:[]).map(u=>`<link rel="stylesheet" href="/${u}">`).join(""),{pathname:i}=new URL(t,"http://localhost"),l;try{let u=a?.loaderTimeout??1e4;l=await q(i,r,e,u)}catch(u){return console.error("[devix] render error:",u),{html:`<html lang="en"><head><meta charset="utf-8">${s}</head><body><script>window.__DEVIX__=null;window.__LOADER_DATA__=null;window.__LAYOUTS_DATA__=[];</script><script type="module" src="${n}"></script><div id="devix-root"></div></body></html>`,statusCode:500,headers:{}}}if(!l){let u=`<script>window.__DEVIX__=${M({metadata:null,viewport:void 0,clientEntry:n})};window.__LOADER_DATA__=null;window.__LAYOUTS_DATA__=[];</script>`,P=`<script type="module" src="${n}"></script>`;return{html:`<html lang="en"><head><meta charset="utf-8">${s}${u}</head><body><div id="devix-root"></div>${P}</body></html>`,statusCode:404,headers:{}}}if("redirect"in l)return{html:"",statusCode:302,headers:{Location:l.redirect}};let{pageMod:d,layoutMods:g,params:c,loaderData:m,layoutsData:f,metadata:h,viewport:w,lang:R}=l,_=T(A,{value:{loaderData:m,params:c}},T(d.default,{data:m,params:c,url:i}));for(let u=g.length-1;u>=0;u--){let P=f[u];_=T(A,{value:{loaderData:P,params:c}},T(g[u].default,{data:P,params:c},_))}let v=X(_),x=h?F(E(h,w)):"",L=`<script>window.__DEVIX__=${M({metadata:h,viewport:w,clientEntry:n})};window.__LOADER_DATA__=${M(m)};window.__LAYOUTS_DATA__=${M(f)};</script>`,p=`<script type="module" src="${n}"></script>`,k=d.headers??{};return{html:`<html lang="${G(R)}"><head><meta charset="utf-8">${x}${s}${L}</head><body><div id="devix-root">${v}</div>${p}</body></html>`,statusCode:200,headers:k}}async function Rt(t){let{pages:r}=$(Object.keys(t.pages),Object.keys(t.layouts),t.pagesDir),e=[];for(let a of r)if(a.params.length===0)e.push(a.path);else{let n=await t.pages[a.key]();if(!n.generateStaticParams)continue;let o=await n.generateStaticParams();for(let s of o){let i=a.path;for(let[l,d]of Object.entries(s))i=i.replace(`:${l}`,encodeURIComponent(d));e.push(i)}}return e}export{Rt as getStaticRoutes,Mt as render,Pt as runLoader};
1
+ import{createElement as lt}from"react";import{renderToString as ct,renderToStaticMarkup as ut}from"react-dom/server";import{Fragment as G,jsx as P}from"react/jsx-runtime";function et(t,e){let r=[];t.title&&r.push({tag:"title",children:t.title}),t.description&&r.push({tag:"meta",name:"description",content:t.description}),t.keywords?.length&&r.push({tag:"meta",name:"keywords",content:t.keywords.join(", ")});let o=t.og?.title??t.title;o&&r.push({tag:"meta",property:"og:title",content:o});let n=t.og?.description??t.description;n&&r.push({tag:"meta",property:"og:description",content:n}),t.og?.image&&r.push({tag:"meta",property:"og:image",content:t.og.image}),t.og?.type&&r.push({tag:"meta",property:"og:type",content:t.og.type}),t.og?.url&&r.push({tag:"meta",property:"og:url",content:t.og.url});let a=t.twitter?.title??t.title;a&&r.push({tag:"meta",name:"twitter:title",content:a});let s=t.twitter?.description??t.description;if(s&&r.push({tag:"meta",name:"twitter:description",content:s}),t.twitter?.card&&r.push({tag:"meta",name:"twitter:card",content:t.twitter.card}),t.twitter?.image&&r.push({tag:"meta",name:"twitter:image",content:t.twitter.image}),t.twitter?.creator&&r.push({tag:"meta",name:"twitter:creator",content:t.twitter.creator}),t.canonical&&r.push({tag:"link",rel:"canonical",href:t.canonical}),t.robots&&r.push({tag:"meta",name:"robots",content:t.robots}),t.alternates)for(let[i,c]of Object.entries(t.alternates))r.push({tag:"link",rel:"alternate",href:c,hrefLang:i});if(e){let i=[];e.width!==void 0&&i.push(`width=${e.width}`),e.initialScale!==void 0&&i.push(`initial-scale=${e.initialScale}`),e.maximumScale!==void 0&&i.push(`maximum-scale=${e.maximumScale}`),e.userScalable!==void 0&&i.push(`user-scalable=${e.userScalable?"yes":"no"}`),i.length&&r.push({tag:"meta",name:"viewport",content:i.join(", ")}),e.themeColor&&r.push({tag:"meta",name:"theme-color",content:e.themeColor})}return r}function J({metadata:t,viewport:e}){return typeof window>"u"||!t?null:P(G,{children:A(t,e)})}function A(t,e){let r=et(t,e);return P(G,{children:r.map((o,n)=>o.tag==="title"?P("title",{children:o.children},n):o.tag==="link"?P("link",{rel:o.rel,href:o.href,hrefLang:o.hrefLang},n):P("meta",{name:o.name,property:o.property,content:o.content},n))})}import{createContext as V}from"react";var _=globalThis;_.__devix_RouterContext__??=V(null);var H=_.__devix_RouterContext__;_.__devix_PageMetaContext__??=V(null);_.__devix_RouteDataContext__??=V(null);var Y=_.__devix_PageMetaContext__,N=_.__devix_RouteDataContext__;import{Component as rt}from"react";import{jsx as B}from"react/jsx-runtime";var S=class extends rt{state={error:null};static getDerivedStateFromError(e){return e instanceof O?{error:{statusCode:e.statusCode,message:e.message}}:{error:{statusCode:500,message:e instanceof Error?e.message:"Unknown error"}}}render(){return this.state.error&&this.props.ErrorPage?B(this.props.ErrorPage,{...this.state.error}):this.state.error?B("h1",{children:this.state.error.statusCode}):this.props.children}},O=class extends Error{statusCode;constructor(e,r){super(r),this.statusCode=e}};import{jsx as w,jsxs as at}from"react/jsx-runtime";var ot=(t,e)=>Promise.resolve(),nt=()=>Promise.resolve();function F({pathname:t,params:e,loaderData:r,layoutsData:o,Page:n,layouts:a,metadata:s,viewport:i,clientEntry:c}){let g=w(N,{value:{loaderData:r,params:e},children:w(n,{data:r,params:e,url:t})});for(let u=a.length-1;u>=0;u--){let p=a[u],m=o[u];g=w(N,{value:{loaderData:m,params:e},children:w(p,{data:m,params:e,children:g})})}return at(Y,{value:{metadata:s,viewport:i,clientEntry:c},children:[w(J,{metadata:s,viewport:i}),w(H,{value:{pathname:t,params:e,loaderData:r,layoutsData:o,Page:n,layouts:a,metadata:s,viewport:i,isNavigating:!1,navigate:ot,revalidate:nt},children:w(S,{children:g},t)})]})}function X(t){return t.replace(/\.(tsx|ts|jsx|js)$/,"").replace(/\(.*?\)\//g,"").replace(/^index$|\/index$/,"").replace(/\[([^\]]+)]/g,":$1")||"/"}function it(t,e){let r=t.slice(e.length+1).replace(/\\/g,"/"),o=X(r);return o==="/"?"/":`/${o}`}function q(t){return t.slice(0,t.lastIndexOf("/"))}var k=null;function j(t,e,r){if(k)return k;let o=[],n=[];for(let a of e)n.push({dir:q(a),key:a});for(let a of t){let s=it(a,r),i=[...s.matchAll(/:([^/]+)/g)].map(g=>g[1]),c=s.replace(/:[^/]+/g,"([^/]+)").replace(/\//g,"\\/");o.push({path:s,key:a,params:i,regex:new RegExp(`^${c}$`)})}return o.sort((a,s)=>{let i=(a.path.match(/:/g)||[]).length,c=(s.path.match(/:/g)||[]).length;return i!==c?i-c:s.path.length-a.path.length}),k={pages:o,layouts:n},k}function W(t,e){let r=q(t);return e.filter(o=>r.startsWith(o.dir)).sort((o,n)=>o.dir.split("/").length-n.dir.split("/").length)}function z(t,e){for(let r of e){let o=t.match(r.regex);if(o){let n={};return r.params.forEach((a,s)=>{n[a]=decodeURIComponent(o[s+1])}),{page:r,params:n}}}return null}async function I(t,e){let r=t.generateMetadata?await t.generateMetadata(e):t.metadata??{},o=t.generateViewport?await t.generateViewport(e):t.viewport;return{metadata:r,viewport:o}}function K(...t){let e={};for(let r of t){if(!r)continue;let{og:o,twitter:n,...a}=r;Object.assign(e,a),o&&(e.og={...e.og,...o}),n&&(e.twitter={...e.twitter,...n})}return e}function v(t){return JSON.stringify(t).replace(/<\/script>/gi,"<\\/script>")}function Q(t){return t.replace(/"/g,"&quot;")}function U(t,e){let r;return Promise.race([t.finally(()=>clearTimeout(r)),new Promise((o,n)=>{r=setTimeout(()=>n(new Error(`timed out after ${e}ms`)),e)})])}var st=Symbol("devix.redirect");function L(t){return typeof t=="object"&&t!==null&&st in t}var pt="/@id/virtual:devix/entry-client";function Z(t){return typeof t=="string"?{url:t,status:302,replace:!1}:L(t)?{url:t.url,status:t.status,replace:t.replace}:null}async function tt(t,e,r,o){let{pages:n,layouts:a}=j(Object.keys(r.pages),Object.keys(r.layouts),r.pagesDir),s=z(t,n);if(!s)return null;let{page:i,params:c}=s,g=W(i.key,a),[u,...p]=await Promise.all([r.pages[i.key](),...g.map(l=>r.layouts[l.key]())]),m;for(let l of p)if(l.guard){let f=await l.guard({params:c,request:e,guardData:m}),b=Z(f);if(b!==null)return{redirect:b.url,redirectStatus:b.status,redirectReplace:b.replace};f!=null&&(m=f)}if(u.guard){let l=await u.guard({params:c,request:e,guardData:m}),f=Z(l);if(f!==null)return{redirect:f.url,redirectStatus:f.status,redirectReplace:f.replace};l!=null&&(m=l)}let h={params:c,request:e,guardData:m},y=u.loader?await U(u.loader(h),o):null;if(L(y))return{redirect:y.url,redirectStatus:y.status,redirectReplace:y.replace};let x=y,C=await U(Promise.all(p.map(l=>l.loader?l.loader(h):null)),o);for(let l of C)if(L(l))return{redirect:l.url,redirectStatus:l.status,redirectReplace:l.replace};let R=C,T=await I(u,{...h,loaderData:x}),D=await Promise.all(p.map((l,f)=>I(l,{...h,loaderData:R[f]}))),E=K(...D.map(l=>l.metadata),T.metadata),$=T.viewport??D.findLast(l=>l.viewport)?.viewport,M=p[0],d=M?.generateLang?await M.generateLang({...h,loaderData:R[0]}):M?.lang??"en";return{pageMod:u,layoutMods:p,params:c,loaderData:x,layoutsData:R,metadata:E,viewport:$,lang:d}}async function Bt(t,e,r,o){let{pathname:n}=new URL(t,"http://localhost"),a;try{let p=o?.loaderTimeout??1e4;a=await tt(n,e,r,p)}catch(p){return console.error("[devix] render error:",p),{error:!0,loaderData:null,params:{},layouts:[],metadata:null,viewport:void 0}}if(!a)return{loaderData:null,params:{},layouts:[],metadata:null,viewport:void 0};if("redirect"in a)return{redirect:a.redirect,redirectStatus:a.redirectStatus,redirectReplace:a.redirectReplace};let{loaderData:s,params:i,layoutsData:c,metadata:g,viewport:u}=a;return{loaderData:s,params:i,layouts:c.map(p=>({loaderData:p})),metadata:g,viewport:u}}async function Ft(t,e,r,o){let n=o?.manifest?`/${Object.values(o.manifest).find(d=>d.isEntry)?.file}`:pt,s=(o?.manifest?Object.values(o.manifest).find(d=>d.isEntry)?.css??[]:[]).map(d=>`<link rel="stylesheet" href="/${d}">`).join(""),{pathname:i}=new URL(t,"http://localhost"),c;try{let d=o?.loaderTimeout??1e4;c=await tt(i,e,r,d)}catch(d){return console.error("[devix] render error:",d),{html:`<html lang="en"><head><meta charset="utf-8">${s}</head><body><script>window.__DEVIX__=null;window.__LOADER_DATA__=null;window.__LAYOUTS_DATA__=[];</script><script type="module" src="${n}"></script><div id="devix-root"></div></body></html>`,statusCode:500,headers:{}}}if(!c){let d=`<script>window.__DEVIX__=${v({metadata:null,viewport:void 0,clientEntry:n})};window.__LOADER_DATA__=null;window.__LAYOUTS_DATA__=[];</script>`,l=`<script type="module" src="${n}"></script>`;return{html:`<html lang="en"><head><meta charset="utf-8">${s}${d}</head><body><div id="devix-root"></div>${l}</body></html>`,statusCode:404,headers:{}}}if("redirect"in c)return{html:"",statusCode:c.redirectStatus,headers:{Location:c.redirect}};let{pageMod:g,layoutMods:u,params:p,loaderData:m,layoutsData:h,metadata:y,viewport:x,lang:C}=c,R=ct(lt(F,{pathname:i,params:p,loaderData:m,layoutsData:h,Page:g.default,layouts:u.map(d=>d.default),metadata:y??null,viewport:x,clientEntry:n})),T=y?ut(A(y,x)):"",D=`<script>window.__DEVIX__=${v({metadata:y,viewport:x,clientEntry:n})};window.__LOADER_DATA__=${v(m??null)};window.__LAYOUTS_DATA__=${v(h)};</script>`,E=`<script type="module" src="${n}"></script>`,$=g.headers??{};return{html:`<html lang="${Q(C)}"><head><meta charset="utf-8">${T}${s}${D}</head><body><div id="devix-root">${R}</div>${E}</body></html>`,statusCode:200,headers:$}}async function Xt(t){let{pages:e}=j(Object.keys(t.pages),Object.keys(t.layouts),t.pagesDir),r=[];for(let o of e)if(o.params.length===0)r.push(o.path);else{let n=await t.pages[o.key]();if(!n.generateStaticParams)continue;let a=await n.generateStaticParams();for(let s of a){let i=o.path;for(let[c,g]of Object.entries(s))i=i.replace(`:${c}`,encodeURIComponent(g));r.push(i)}}return r}export{Xt as getStaticRoutes,Ft as render,Bt as runLoader};
2
2
  //# sourceMappingURL=render.js.map