@rangojs/router 0.0.0-experimental.0f44aca1
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.
- package/AGENTS.md +5 -0
- package/README.md +899 -0
- package/dist/bin/rango.js +1601 -0
- package/dist/vite/index.js +5214 -0
- package/package.json +176 -0
- package/skills/breadcrumbs/SKILL.md +250 -0
- package/skills/cache-guide/SKILL.md +262 -0
- package/skills/caching/SKILL.md +220 -0
- package/skills/composability/SKILL.md +172 -0
- package/skills/debug-manifest/SKILL.md +112 -0
- package/skills/document-cache/SKILL.md +182 -0
- package/skills/fonts/SKILL.md +167 -0
- package/skills/hooks/SKILL.md +704 -0
- package/skills/host-router/SKILL.md +218 -0
- package/skills/intercept/SKILL.md +313 -0
- package/skills/layout/SKILL.md +310 -0
- package/skills/links/SKILL.md +239 -0
- package/skills/loader/SKILL.md +596 -0
- package/skills/middleware/SKILL.md +339 -0
- package/skills/mime-routes/SKILL.md +128 -0
- package/skills/parallel/SKILL.md +305 -0
- package/skills/prerender/SKILL.md +643 -0
- package/skills/rango/SKILL.md +118 -0
- package/skills/response-routes/SKILL.md +411 -0
- package/skills/route/SKILL.md +385 -0
- package/skills/router-setup/SKILL.md +439 -0
- package/skills/tailwind/SKILL.md +129 -0
- package/skills/theme/SKILL.md +79 -0
- package/skills/typesafety/SKILL.md +623 -0
- package/skills/use-cache/SKILL.md +324 -0
- package/src/__internal.ts +273 -0
- package/src/bin/rango.ts +321 -0
- package/src/browser/action-coordinator.ts +97 -0
- package/src/browser/action-response-classifier.ts +99 -0
- package/src/browser/event-controller.ts +899 -0
- package/src/browser/history-state.ts +80 -0
- package/src/browser/index.ts +18 -0
- package/src/browser/intercept-utils.ts +52 -0
- package/src/browser/link-interceptor.ts +141 -0
- package/src/browser/logging.ts +55 -0
- package/src/browser/merge-segment-loaders.ts +134 -0
- package/src/browser/navigation-bridge.ts +645 -0
- package/src/browser/navigation-client.ts +215 -0
- package/src/browser/navigation-store.ts +806 -0
- package/src/browser/navigation-transaction.ts +295 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +550 -0
- package/src/browser/prefetch/cache.ts +146 -0
- package/src/browser/prefetch/fetch.ts +135 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +42 -0
- package/src/browser/prefetch/queue.ts +88 -0
- package/src/browser/rango-state.ts +112 -0
- package/src/browser/react/Link.tsx +360 -0
- package/src/browser/react/NavigationProvider.tsx +386 -0
- package/src/browser/react/ScrollRestoration.tsx +94 -0
- package/src/browser/react/context.ts +59 -0
- package/src/browser/react/filter-segment-order.ts +11 -0
- package/src/browser/react/index.ts +52 -0
- package/src/browser/react/location-state-shared.ts +162 -0
- package/src/browser/react/location-state.ts +107 -0
- package/src/browser/react/mount-context.ts +37 -0
- package/src/browser/react/nonce-context.ts +23 -0
- package/src/browser/react/shallow-equal.ts +27 -0
- package/src/browser/react/use-action.ts +218 -0
- package/src/browser/react/use-client-cache.ts +58 -0
- package/src/browser/react/use-handle.ts +162 -0
- package/src/browser/react/use-href.tsx +40 -0
- package/src/browser/react/use-link-status.ts +135 -0
- package/src/browser/react/use-mount.ts +31 -0
- package/src/browser/react/use-navigation.ts +99 -0
- package/src/browser/react/use-params.ts +65 -0
- package/src/browser/react/use-pathname.ts +47 -0
- package/src/browser/react/use-router.ts +63 -0
- package/src/browser/react/use-search-params.ts +56 -0
- package/src/browser/react/use-segments.ts +171 -0
- package/src/browser/response-adapter.ts +73 -0
- package/src/browser/rsc-router.tsx +431 -0
- package/src/browser/scroll-restoration.ts +400 -0
- package/src/browser/segment-reconciler.ts +216 -0
- package/src/browser/segment-structure-assert.ts +83 -0
- package/src/browser/server-action-bridge.ts +667 -0
- package/src/browser/shallow.ts +40 -0
- package/src/browser/types.ts +538 -0
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +438 -0
- package/src/build/generate-route-types.ts +36 -0
- package/src/build/index.ts +35 -0
- package/src/build/route-trie.ts +265 -0
- package/src/build/route-types/ast-helpers.ts +25 -0
- package/src/build/route-types/ast-route-extraction.ts +98 -0
- package/src/build/route-types/codegen.ts +102 -0
- package/src/build/route-types/include-resolution.ts +411 -0
- package/src/build/route-types/param-extraction.ts +48 -0
- package/src/build/route-types/per-module-writer.ts +128 -0
- package/src/build/route-types/router-processing.ts +469 -0
- package/src/build/route-types/scan-filter.ts +78 -0
- package/src/build/runtime-discovery.ts +231 -0
- package/src/cache/background-task.ts +34 -0
- package/src/cache/cache-key-utils.ts +44 -0
- package/src/cache/cache-policy.ts +125 -0
- package/src/cache/cache-runtime.ts +338 -0
- package/src/cache/cache-scope.ts +382 -0
- package/src/cache/cf/cf-cache-store.ts +540 -0
- package/src/cache/cf/index.ts +25 -0
- package/src/cache/document-cache.ts +369 -0
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/handle-snapshot.ts +41 -0
- package/src/cache/index.ts +43 -0
- package/src/cache/memory-segment-store.ts +328 -0
- package/src/cache/profile-registry.ts +73 -0
- package/src/cache/read-through-swr.ts +134 -0
- package/src/cache/segment-codec.ts +256 -0
- package/src/cache/taint.ts +98 -0
- package/src/cache/types.ts +342 -0
- package/src/client.rsc.tsx +85 -0
- package/src/client.tsx +601 -0
- package/src/component-utils.ts +76 -0
- package/src/components/DefaultDocument.tsx +27 -0
- package/src/context-var.ts +86 -0
- package/src/debug.ts +243 -0
- package/src/default-error-boundary.tsx +88 -0
- package/src/deps/browser.ts +8 -0
- package/src/deps/html-stream-client.ts +2 -0
- package/src/deps/html-stream-server.ts +2 -0
- package/src/deps/rsc.ts +10 -0
- package/src/deps/ssr.ts +2 -0
- package/src/errors.ts +365 -0
- package/src/handle.ts +135 -0
- package/src/handles/MetaTags.tsx +246 -0
- package/src/handles/breadcrumbs.ts +66 -0
- package/src/handles/index.ts +7 -0
- package/src/handles/meta.ts +264 -0
- package/src/host/cookie-handler.ts +165 -0
- package/src/host/errors.ts +97 -0
- package/src/host/index.ts +53 -0
- package/src/host/pattern-matcher.ts +214 -0
- package/src/host/router.ts +352 -0
- package/src/host/testing.ts +79 -0
- package/src/host/types.ts +146 -0
- package/src/host/utils.ts +25 -0
- package/src/href-client.ts +222 -0
- package/src/index.rsc.ts +233 -0
- package/src/index.ts +277 -0
- package/src/internal-debug.ts +11 -0
- package/src/loader.rsc.ts +89 -0
- package/src/loader.ts +64 -0
- package/src/network-error-thrower.tsx +23 -0
- package/src/outlet-context.ts +15 -0
- package/src/outlet-provider.tsx +45 -0
- package/src/prerender/param-hash.ts +37 -0
- package/src/prerender/store.ts +185 -0
- package/src/prerender.ts +463 -0
- package/src/reverse.ts +330 -0
- package/src/root-error-boundary.tsx +289 -0
- package/src/route-content-wrapper.tsx +196 -0
- package/src/route-definition/dsl-helpers.ts +934 -0
- package/src/route-definition/helper-factories.ts +200 -0
- package/src/route-definition/helpers-types.ts +430 -0
- package/src/route-definition/index.ts +52 -0
- package/src/route-definition/redirect.ts +93 -0
- package/src/route-definition.ts +1 -0
- package/src/route-map-builder.ts +275 -0
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +259 -0
- package/src/router/content-negotiation.ts +116 -0
- package/src/router/debug-manifest.ts +72 -0
- package/src/router/error-handling.ts +287 -0
- package/src/router/find-match.ts +158 -0
- package/src/router/handler-context.ts +451 -0
- package/src/router/intercept-resolution.ts +395 -0
- package/src/router/lazy-includes.ts +234 -0
- package/src/router/loader-resolution.ts +420 -0
- package/src/router/logging.ts +248 -0
- package/src/router/manifest.ts +267 -0
- package/src/router/match-api.ts +620 -0
- package/src/router/match-context.ts +266 -0
- package/src/router/match-handlers.ts +440 -0
- package/src/router/match-middleware/background-revalidation.ts +223 -0
- package/src/router/match-middleware/cache-lookup.ts +634 -0
- package/src/router/match-middleware/cache-store.ts +295 -0
- package/src/router/match-middleware/index.ts +81 -0
- package/src/router/match-middleware/intercept-resolution.ts +306 -0
- package/src/router/match-middleware/segment-resolution.ts +192 -0
- package/src/router/match-pipelines.ts +179 -0
- package/src/router/match-result.ts +219 -0
- package/src/router/metrics.ts +282 -0
- package/src/router/middleware-cookies.ts +55 -0
- package/src/router/middleware-types.ts +222 -0
- package/src/router/middleware.ts +748 -0
- package/src/router/pattern-matching.ts +563 -0
- package/src/router/prerender-match.ts +402 -0
- package/src/router/preview-match.ts +170 -0
- package/src/router/revalidation.ts +289 -0
- package/src/router/router-context.ts +316 -0
- package/src/router/router-interfaces.ts +452 -0
- package/src/router/router-options.ts +592 -0
- package/src/router/router-registry.ts +24 -0
- package/src/router/segment-resolution/fresh.ts +570 -0
- package/src/router/segment-resolution/helpers.ts +263 -0
- package/src/router/segment-resolution/loader-cache.ts +198 -0
- package/src/router/segment-resolution/revalidation.ts +1239 -0
- package/src/router/segment-resolution/static-store.ts +67 -0
- package/src/router/segment-resolution.ts +21 -0
- package/src/router/segment-wrappers.ts +289 -0
- package/src/router/telemetry-otel.ts +299 -0
- package/src/router/telemetry.ts +300 -0
- package/src/router/timeout.ts +148 -0
- package/src/router/trie-matching.ts +239 -0
- package/src/router/types.ts +170 -0
- package/src/router.ts +1002 -0
- package/src/rsc/handler-context.ts +45 -0
- package/src/rsc/handler.ts +1089 -0
- package/src/rsc/helpers.ts +198 -0
- package/src/rsc/index.ts +36 -0
- package/src/rsc/loader-fetch.ts +209 -0
- package/src/rsc/manifest-init.ts +86 -0
- package/src/rsc/nonce.ts +32 -0
- package/src/rsc/origin-guard.ts +141 -0
- package/src/rsc/progressive-enhancement.ts +379 -0
- package/src/rsc/response-error.ts +37 -0
- package/src/rsc/response-route-handler.ts +347 -0
- package/src/rsc/rsc-rendering.ts +235 -0
- package/src/rsc/runtime-warnings.ts +42 -0
- package/src/rsc/server-action.ts +348 -0
- package/src/rsc/ssr-setup.ts +128 -0
- package/src/rsc/types.ts +263 -0
- package/src/search-params.ts +230 -0
- package/src/segment-system.tsx +454 -0
- package/src/server/context.ts +591 -0
- package/src/server/cookie-store.ts +190 -0
- package/src/server/fetchable-loader-store.ts +37 -0
- package/src/server/handle-store.ts +308 -0
- package/src/server/loader-registry.ts +133 -0
- package/src/server/request-context.ts +914 -0
- package/src/server/root-layout.tsx +10 -0
- package/src/server/tsconfig.json +14 -0
- package/src/server.ts +51 -0
- package/src/ssr/index.tsx +365 -0
- package/src/static-handler.ts +114 -0
- package/src/theme/ThemeProvider.tsx +297 -0
- package/src/theme/ThemeScript.tsx +61 -0
- package/src/theme/constants.ts +62 -0
- package/src/theme/index.ts +48 -0
- package/src/theme/theme-context.ts +44 -0
- package/src/theme/theme-script.ts +155 -0
- package/src/theme/types.ts +182 -0
- package/src/theme/use-theme.ts +44 -0
- package/src/types/boundaries.ts +158 -0
- package/src/types/cache-types.ts +198 -0
- package/src/types/error-types.ts +192 -0
- package/src/types/global-namespace.ts +100 -0
- package/src/types/handler-context.ts +687 -0
- package/src/types/index.ts +88 -0
- package/src/types/loader-types.ts +183 -0
- package/src/types/route-config.ts +170 -0
- package/src/types/route-entry.ts +102 -0
- package/src/types/segments.ts +148 -0
- package/src/types.ts +1 -0
- package/src/urls/include-helper.ts +197 -0
- package/src/urls/index.ts +53 -0
- package/src/urls/path-helper-types.ts +339 -0
- package/src/urls/path-helper.ts +329 -0
- package/src/urls/pattern-types.ts +95 -0
- package/src/urls/response-types.ts +106 -0
- package/src/urls/type-extraction.ts +372 -0
- package/src/urls/urls-function.ts +98 -0
- package/src/urls.ts +1 -0
- package/src/use-loader.tsx +354 -0
- package/src/vite/discovery/bundle-postprocess.ts +184 -0
- package/src/vite/discovery/discover-routers.ts +344 -0
- package/src/vite/discovery/prerender-collection.ts +385 -0
- package/src/vite/discovery/route-types-writer.ts +258 -0
- package/src/vite/discovery/self-gen-tracking.ts +47 -0
- package/src/vite/discovery/state.ts +110 -0
- package/src/vite/discovery/virtual-module-codegen.ts +203 -0
- package/src/vite/index.ts +16 -0
- package/src/vite/plugin-types.ts +131 -0
- package/src/vite/plugins/cjs-to-esm.ts +93 -0
- package/src/vite/plugins/client-ref-dedup.ts +115 -0
- package/src/vite/plugins/client-ref-hashing.ts +105 -0
- package/src/vite/plugins/expose-action-id.ts +365 -0
- package/src/vite/plugins/expose-id-utils.ts +287 -0
- package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
- package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
- package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
- package/src/vite/plugins/expose-ids/types.ts +45 -0
- package/src/vite/plugins/expose-internal-ids.ts +569 -0
- package/src/vite/plugins/refresh-cmd.ts +65 -0
- package/src/vite/plugins/use-cache-transform.ts +323 -0
- package/src/vite/plugins/version-injector.ts +83 -0
- package/src/vite/plugins/version-plugin.ts +254 -0
- package/src/vite/plugins/version.d.ts +12 -0
- package/src/vite/plugins/virtual-entries.ts +123 -0
- package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
- package/src/vite/rango.ts +510 -0
- package/src/vite/router-discovery.ts +785 -0
- package/src/vite/utils/ast-handler-extract.ts +517 -0
- package/src/vite/utils/banner.ts +36 -0
- package/src/vite/utils/bundle-analysis.ts +137 -0
- package/src/vite/utils/manifest-utils.ts +70 -0
- package/src/vite/utils/package-resolution.ts +121 -0
- package/src/vite/utils/prerender-utils.ts +189 -0
- package/src/vite/utils/shared-utils.ts +169 -0
|
@@ -0,0 +1,185 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Prerender Store
|
|
3
|
+
*
|
|
4
|
+
* Reads pre-rendered segment data from the worker bundle at build time.
|
|
5
|
+
* The manifest module is lazily loaded via globalThis.__loadPrerenderManifestModule,
|
|
6
|
+
* a function injected into the RSC entry that returns the manifest module
|
|
7
|
+
* containing a key-to-specifier map and a `loadPrerenderAsset` function
|
|
8
|
+
* that anchors import() resolution relative to the manifest file.
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import type {
|
|
12
|
+
SerializedSegmentData,
|
|
13
|
+
SegmentHandleData,
|
|
14
|
+
} from "../cache/types.js";
|
|
15
|
+
|
|
16
|
+
export interface PrerenderEntry {
|
|
17
|
+
segments: SerializedSegmentData[];
|
|
18
|
+
handles: Record<string, SegmentHandleData>;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
export interface PrerenderStore {
|
|
22
|
+
get(
|
|
23
|
+
routeName: string,
|
|
24
|
+
paramHash: string,
|
|
25
|
+
meta?: { pathname: string; isPassthroughRoute?: boolean },
|
|
26
|
+
): PrerenderEntry | null | Promise<PrerenderEntry | null>;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
export interface StaticEntry {
|
|
30
|
+
encoded: string;
|
|
31
|
+
handles: Record<string, unknown[]>;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
export interface StaticStore {
|
|
35
|
+
get(handlerId: string): Promise<StaticEntry | null>;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
interface PrerenderManifestModule {
|
|
39
|
+
default: Record<string, string>;
|
|
40
|
+
loadPrerenderAsset: (
|
|
41
|
+
specifier: string,
|
|
42
|
+
) => Promise<{ default: PrerenderEntry }>;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
declare global {
|
|
46
|
+
// Injected by closeBundle post-processing: lazy loader for the prerender
|
|
47
|
+
// manifest module. The module exports a key→specifier map and a
|
|
48
|
+
// loadPrerenderAsset function that anchors import() relative to the manifest.
|
|
49
|
+
// eslint-disable-next-line no-var
|
|
50
|
+
var __loadPrerenderManifestModule:
|
|
51
|
+
| (() => Promise<PrerenderManifestModule>)
|
|
52
|
+
| undefined;
|
|
53
|
+
// Injected by closeBundle post-processing: map of handlerId -> () => import("./assets/__st-*.js")
|
|
54
|
+
// Asset default export is either a string (no handles) or { encoded, handles } object.
|
|
55
|
+
// eslint-disable-next-line no-var
|
|
56
|
+
var __STATIC_MANIFEST:
|
|
57
|
+
| Record<string, () => Promise<{ default: string | StaticEntry }>>
|
|
58
|
+
| undefined;
|
|
59
|
+
// Injected by virtual module in dev mode for on-demand prerender
|
|
60
|
+
// eslint-disable-next-line no-var
|
|
61
|
+
var __PRERENDER_DEV_URL: string | undefined;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Create a dev-mode prerender store that fetches on-demand from the
|
|
66
|
+
* Vite dev server's /__rsc_prerender endpoint (runs in Node.js where
|
|
67
|
+
* node:fs works, unlike workerd).
|
|
68
|
+
*/
|
|
69
|
+
export function createDevPrerenderStore(devUrl: string): PrerenderStore {
|
|
70
|
+
return {
|
|
71
|
+
async get(routeName, paramHash, meta) {
|
|
72
|
+
if (!meta?.pathname) return null;
|
|
73
|
+
const isIntercept = paramHash.endsWith("/i");
|
|
74
|
+
let url = `${devUrl}/__rsc_prerender?pathname=${encodeURIComponent(meta.pathname)}&routeName=${encodeURIComponent(routeName)}`;
|
|
75
|
+
if (isIntercept) url += "&intercept=1";
|
|
76
|
+
if (meta.isPassthroughRoute) url += "&passthrough=1";
|
|
77
|
+
try {
|
|
78
|
+
const res = await fetch(url);
|
|
79
|
+
if (!res.ok) return null;
|
|
80
|
+
return res.json();
|
|
81
|
+
} catch {
|
|
82
|
+
return null;
|
|
83
|
+
}
|
|
84
|
+
},
|
|
85
|
+
};
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
/**
|
|
89
|
+
* Create a prerender store.
|
|
90
|
+
* Dev mode: on-demand fetch from Vite dev server (node:fs works there).
|
|
91
|
+
* Production: backed by globalThis.__loadPrerenderManifestModule which lazily
|
|
92
|
+
* loads the manifest module on first access.
|
|
93
|
+
* Returns null if no prerender data is available.
|
|
94
|
+
*/
|
|
95
|
+
export function createPrerenderStore(): PrerenderStore | null {
|
|
96
|
+
if (globalThis.__PRERENDER_DEV_URL) {
|
|
97
|
+
return createDevPrerenderStore(globalThis.__PRERENDER_DEV_URL);
|
|
98
|
+
}
|
|
99
|
+
if (!globalThis.__loadPrerenderManifestModule) return null;
|
|
100
|
+
|
|
101
|
+
const cache = new Map<string, Promise<PrerenderEntry | null>>();
|
|
102
|
+
let manifestModulePromise: Promise<PrerenderManifestModule | null> | null =
|
|
103
|
+
null;
|
|
104
|
+
|
|
105
|
+
function loadManifestModule(): Promise<PrerenderManifestModule | null> {
|
|
106
|
+
if (!manifestModulePromise) {
|
|
107
|
+
manifestModulePromise = globalThis.__loadPrerenderManifestModule!().catch(
|
|
108
|
+
() => null,
|
|
109
|
+
);
|
|
110
|
+
}
|
|
111
|
+
return manifestModulePromise;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
return {
|
|
115
|
+
get(routeName: string, paramHash: string): Promise<PrerenderEntry | null> {
|
|
116
|
+
const key = `${routeName}/${paramHash}`;
|
|
117
|
+
const cached = cache.get(key);
|
|
118
|
+
if (cached) return cached;
|
|
119
|
+
|
|
120
|
+
const promise = loadManifestModule().then((mod) => {
|
|
121
|
+
if (!mod) return null;
|
|
122
|
+
const specifier = mod.default[key];
|
|
123
|
+
if (!specifier) return null;
|
|
124
|
+
return mod
|
|
125
|
+
.loadPrerenderAsset(specifier)
|
|
126
|
+
.then((asset) => asset.default)
|
|
127
|
+
.catch(() => null);
|
|
128
|
+
});
|
|
129
|
+
cache.set(key, promise);
|
|
130
|
+
return promise;
|
|
131
|
+
},
|
|
132
|
+
};
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
/**
|
|
136
|
+
* Load the prerender manifest index for test introspection.
|
|
137
|
+
* Returns the key→specifier map or null if unavailable.
|
|
138
|
+
*/
|
|
139
|
+
export async function loadPrerenderManifestIndex(): Promise<Record<
|
|
140
|
+
string,
|
|
141
|
+
string
|
|
142
|
+
> | null> {
|
|
143
|
+
if (!globalThis.__loadPrerenderManifestModule) return null;
|
|
144
|
+
try {
|
|
145
|
+
const mod = await globalThis.__loadPrerenderManifestModule();
|
|
146
|
+
return mod.default;
|
|
147
|
+
} catch {
|
|
148
|
+
return null;
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
/**
|
|
153
|
+
* Create a static segment store.
|
|
154
|
+
* Production only: backed by globalThis.__STATIC_MANIFEST injected at build time.
|
|
155
|
+
* Returns null if no static data is available (dev mode or no Static handlers).
|
|
156
|
+
*/
|
|
157
|
+
export function createStaticStore(): StaticStore | null {
|
|
158
|
+
const manifest = globalThis.__STATIC_MANIFEST;
|
|
159
|
+
if (!manifest || Object.keys(manifest).length === 0) return null;
|
|
160
|
+
|
|
161
|
+
const cache = new Map<string, Promise<StaticEntry | null>>();
|
|
162
|
+
|
|
163
|
+
return {
|
|
164
|
+
get(handlerId: string): Promise<StaticEntry | null> {
|
|
165
|
+
const cached = cache.get(handlerId);
|
|
166
|
+
if (cached) return cached;
|
|
167
|
+
|
|
168
|
+
const importFn = manifest[handlerId];
|
|
169
|
+
if (!importFn) return Promise.resolve(null);
|
|
170
|
+
|
|
171
|
+
const promise = importFn()
|
|
172
|
+
.then((mod) => {
|
|
173
|
+
const val = mod.default;
|
|
174
|
+
// Normalize: string-only (no handles) or { encoded, handles }
|
|
175
|
+
if (typeof val === "string") {
|
|
176
|
+
return { encoded: val, handles: {} } as StaticEntry;
|
|
177
|
+
}
|
|
178
|
+
return val as StaticEntry;
|
|
179
|
+
})
|
|
180
|
+
.catch(() => null);
|
|
181
|
+
cache.set(handlerId, promise);
|
|
182
|
+
return promise;
|
|
183
|
+
},
|
|
184
|
+
};
|
|
185
|
+
}
|
package/src/prerender.ts
ADDED
|
@@ -0,0 +1,463 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pre-render handler definition for build-time rendering of route segments.
|
|
3
|
+
*
|
|
4
|
+
* Prerender wraps a handler so that in production (phase 2)
|
|
5
|
+
* it can be pre-rendered at build time and served as a static Flight payload.
|
|
6
|
+
* In dev mode (phase 1), it behaves as a normal handler — the handler runs
|
|
7
|
+
* on every request just like a regular path() handler.
|
|
8
|
+
*
|
|
9
|
+
* The $$id is auto-generated by the Vite exposeInternalIds plugin
|
|
10
|
+
* based on file path and export name. No manual naming required.
|
|
11
|
+
*
|
|
12
|
+
* @example
|
|
13
|
+
* ```ts
|
|
14
|
+
* // Static page — no params
|
|
15
|
+
* export const DocsPage = Prerender(async (ctx) => {
|
|
16
|
+
* return <div>Documentation</div>;
|
|
17
|
+
* });
|
|
18
|
+
*
|
|
19
|
+
* // Dynamic page — params first, handler second
|
|
20
|
+
* export const DocsArticle = Prerender(
|
|
21
|
+
* async () => [{ slug: "getting-started" }, { slug: "api-reference" }],
|
|
22
|
+
* async (ctx) => {
|
|
23
|
+
* return <div>{ctx.params.slug}</div>;
|
|
24
|
+
* }
|
|
25
|
+
* );
|
|
26
|
+
* ```
|
|
27
|
+
*/
|
|
28
|
+
import type { ReactNode } from "react";
|
|
29
|
+
import type {
|
|
30
|
+
Handler,
|
|
31
|
+
HandlerContext,
|
|
32
|
+
DefaultEnv,
|
|
33
|
+
ExtractParams,
|
|
34
|
+
} from "./types.js";
|
|
35
|
+
import type { Handle } from "./handle.js";
|
|
36
|
+
import type { ContextVar } from "./context-var.js";
|
|
37
|
+
import type { ReverseFunction } from "./reverse.js";
|
|
38
|
+
import type { DefaultReverseRouteMap } from "./types/global-namespace.js";
|
|
39
|
+
import { isCachedFunction } from "./cache/taint.js";
|
|
40
|
+
|
|
41
|
+
// -- Named route resolution types -------------------------------------------
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* Reverse function for build contexts (BuildContext, StaticBuildContext, GetParamsContext).
|
|
45
|
+
* Global names get full autocomplete and param validation from the generated route map.
|
|
46
|
+
* Local `.name` calls are accepted but not validated (the include() scope is unknown
|
|
47
|
+
* at the type level).
|
|
48
|
+
*/
|
|
49
|
+
type BuildReverseFunction = [DefaultReverseRouteMap] extends [
|
|
50
|
+
Record<string, string>,
|
|
51
|
+
]
|
|
52
|
+
? // No generated route map — permissive fallback
|
|
53
|
+
(
|
|
54
|
+
name: string,
|
|
55
|
+
params?: Record<string, string>,
|
|
56
|
+
search?: Record<string, unknown>,
|
|
57
|
+
) => string
|
|
58
|
+
: // Generated route map available — typed globals + permissive locals
|
|
59
|
+
ReverseFunction<DefaultReverseRouteMap> & {
|
|
60
|
+
(
|
|
61
|
+
name: `.${string}`,
|
|
62
|
+
params?: Record<string, string>,
|
|
63
|
+
search?: Record<string, unknown>,
|
|
64
|
+
): string;
|
|
65
|
+
};
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* Default route map for Prerender named route resolution.
|
|
69
|
+
* Uses GeneratedRouteMap (from gen file) to avoid circular dependencies.
|
|
70
|
+
*/
|
|
71
|
+
type DefaultPrerenderRouteMap = keyof RSCRouter.GeneratedRouteMap extends never
|
|
72
|
+
? {}
|
|
73
|
+
: RSCRouter.GeneratedRouteMap;
|
|
74
|
+
|
|
75
|
+
/** Extract params from a route map entry (string pattern or { path } object). */
|
|
76
|
+
type ExtractParamsFromEntry<TEntry> = TEntry extends string
|
|
77
|
+
? ExtractParams<TEntry>
|
|
78
|
+
: TEntry extends { readonly path: infer P extends string }
|
|
79
|
+
? ExtractParams<P>
|
|
80
|
+
: Record<string, string>;
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Resolve params from Prerender's type parameter.
|
|
84
|
+
* Accepts named routes (global or .local) and explicit param objects.
|
|
85
|
+
*
|
|
86
|
+
* Resolution order:
|
|
87
|
+
* 1. ".local" string → look up in TRouteMap
|
|
88
|
+
* 2. Global route name string → look up in DefaultPrerenderRouteMap
|
|
89
|
+
* 3. Record<string, any> object → use as-is (explicit params)
|
|
90
|
+
* 4. Fallback → {}
|
|
91
|
+
*/
|
|
92
|
+
type ResolvePrerenderParams<
|
|
93
|
+
T,
|
|
94
|
+
TRouteMap extends {} = DefaultPrerenderRouteMap,
|
|
95
|
+
> = T extends `.${infer Local}`
|
|
96
|
+
? Local extends keyof TRouteMap
|
|
97
|
+
? ExtractParamsFromEntry<TRouteMap[Local]>
|
|
98
|
+
: Record<string, string>
|
|
99
|
+
: T extends keyof DefaultPrerenderRouteMap
|
|
100
|
+
? ExtractParamsFromEntry<DefaultPrerenderRouteMap[T]>
|
|
101
|
+
: T extends Record<string, any>
|
|
102
|
+
? T
|
|
103
|
+
: {};
|
|
104
|
+
|
|
105
|
+
// -- Types ------------------------------------------------------------------
|
|
106
|
+
|
|
107
|
+
export interface PrerenderOptions {
|
|
108
|
+
/**
|
|
109
|
+
* Keep handler in server bundle for live fallback (default: false).
|
|
110
|
+
* false: handler replaced with stub, source-only APIs excluded from bundle.
|
|
111
|
+
* true: handler stays in bundle, unknown params render live at request time.
|
|
112
|
+
*/
|
|
113
|
+
passthrough?: boolean;
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* Maximum number of param sets to render in parallel (default: 1).
|
|
117
|
+
* Only applies to dynamic Prerender handlers with getParams().
|
|
118
|
+
* Set to higher values to speed up builds with many routes.
|
|
119
|
+
*
|
|
120
|
+
* @example
|
|
121
|
+
* ```typescript
|
|
122
|
+
* export const BlogPost = Prerender(
|
|
123
|
+
* async () => allPosts.map(p => ({ slug: p.slug })),
|
|
124
|
+
* async (ctx) => <PostPage slug={ctx.params.slug} />,
|
|
125
|
+
* { concurrency: 4 },
|
|
126
|
+
* );
|
|
127
|
+
* ```
|
|
128
|
+
*/
|
|
129
|
+
concurrency?: number;
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
/**
|
|
133
|
+
* Context passed to Prerender() handlers at build time.
|
|
134
|
+
* Has a synthetic URL from getParams, params, and pathname.
|
|
135
|
+
* No request, env, headers, cookies.
|
|
136
|
+
*/
|
|
137
|
+
export interface BuildContext<TParams> {
|
|
138
|
+
/** Params extracted from the route pattern (populated from getParams). */
|
|
139
|
+
params: TParams;
|
|
140
|
+
|
|
141
|
+
/** True during build-time pre-rendering, false during passthrough live render. */
|
|
142
|
+
build: true;
|
|
143
|
+
|
|
144
|
+
/** Read a variable set by getParams or a parent handler. */
|
|
145
|
+
get: {
|
|
146
|
+
<T>(contextVar: ContextVar<T>): T | undefined;
|
|
147
|
+
(key: string): any;
|
|
148
|
+
};
|
|
149
|
+
|
|
150
|
+
/** Set a variable readable by child layouts and parallels. */
|
|
151
|
+
set: {
|
|
152
|
+
<T>(contextVar: ContextVar<T>, value: T): void;
|
|
153
|
+
(key: string, value: any): void;
|
|
154
|
+
};
|
|
155
|
+
|
|
156
|
+
/** Push handle data (frozen into pre-rendered output at build time). */
|
|
157
|
+
use: <T>(handle: Handle<T>) => (data: T) => void;
|
|
158
|
+
|
|
159
|
+
/** Synthetic URL built from pattern + params (no real request). */
|
|
160
|
+
url: URL;
|
|
161
|
+
|
|
162
|
+
/** Pathname portion of the synthetic URL. */
|
|
163
|
+
pathname: string;
|
|
164
|
+
|
|
165
|
+
/** URLSearchParams from the synthetic URL (always empty for prerender). */
|
|
166
|
+
searchParams: URLSearchParams;
|
|
167
|
+
|
|
168
|
+
/** Typed search params — always {} for prerender (no real query string). */
|
|
169
|
+
search: {};
|
|
170
|
+
|
|
171
|
+
/** URL generation by route name. */
|
|
172
|
+
reverse: BuildReverseFunction;
|
|
173
|
+
|
|
174
|
+
/**
|
|
175
|
+
* Signal that this param set should not produce a local prerender artifact.
|
|
176
|
+
* At runtime the handler runs live instead. Only valid on routes declared
|
|
177
|
+
* with `{ passthrough: true }`.
|
|
178
|
+
*/
|
|
179
|
+
passthrough: () => PrerenderPassthroughResult;
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
/**
|
|
183
|
+
* Context passed to Static() handlers at build time.
|
|
184
|
+
* No URL, no params, no pathname — just renders content.
|
|
185
|
+
*/
|
|
186
|
+
export interface StaticBuildContext {
|
|
187
|
+
/** Always true for Static handlers at build time. */
|
|
188
|
+
build: true;
|
|
189
|
+
|
|
190
|
+
/** Read a variable (available for type consistency with BuildContext). */
|
|
191
|
+
get: {
|
|
192
|
+
<T>(contextVar: ContextVar<T>): T | undefined;
|
|
193
|
+
(key: string): any;
|
|
194
|
+
};
|
|
195
|
+
|
|
196
|
+
/** Set a variable (available for type consistency with BuildContext). */
|
|
197
|
+
set: {
|
|
198
|
+
<T>(contextVar: ContextVar<T>, value: T): void;
|
|
199
|
+
(key: string, value: any): void;
|
|
200
|
+
};
|
|
201
|
+
|
|
202
|
+
/** Push handle data (frozen into pre-rendered output at build time). */
|
|
203
|
+
use: <T>(handle: Handle<T>) => (data: T) => void;
|
|
204
|
+
|
|
205
|
+
/** URL generation by route name. */
|
|
206
|
+
reverse: BuildReverseFunction;
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
/**
|
|
210
|
+
* Context passed to getParams() at build time.
|
|
211
|
+
* Allows sharing data with handler invocations via set().
|
|
212
|
+
*/
|
|
213
|
+
export interface GetParamsContext {
|
|
214
|
+
/** Always true during build-time getParams execution. */
|
|
215
|
+
build: true;
|
|
216
|
+
|
|
217
|
+
/** Set a variable that will be available to each handler invocation via ctx.get(). */
|
|
218
|
+
set: {
|
|
219
|
+
<T>(contextVar: ContextVar<T>, value: T): void;
|
|
220
|
+
(key: string, value: any): void;
|
|
221
|
+
};
|
|
222
|
+
|
|
223
|
+
/** URL generation by route name. */
|
|
224
|
+
reverse: BuildReverseFunction;
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
/**
|
|
228
|
+
* Context type for passthrough Prerender handlers.
|
|
229
|
+
*
|
|
230
|
+
* When `passthrough: true`, the handler runs both at build time and at request
|
|
231
|
+
* time. The context is a full `HandlerContext` with `build: boolean`:
|
|
232
|
+
* - `ctx.build === true`: build-time, env/request/res throw at runtime
|
|
233
|
+
* - `ctx.build === false`: live request, full context available
|
|
234
|
+
*
|
|
235
|
+
* For `passthrough: false` (default), handlers receive `BuildContext` only.
|
|
236
|
+
*/
|
|
237
|
+
export type PrerenderPassthroughContext<
|
|
238
|
+
TParams = {},
|
|
239
|
+
TEnv = DefaultEnv,
|
|
240
|
+
> = HandlerContext<TParams, TEnv> & {
|
|
241
|
+
passthrough: () => PrerenderPassthroughResult;
|
|
242
|
+
};
|
|
243
|
+
|
|
244
|
+
export interface PrerenderHandlerDefinition<
|
|
245
|
+
TParams extends Record<string, any> = any,
|
|
246
|
+
> {
|
|
247
|
+
readonly __brand: "prerenderHandler";
|
|
248
|
+
/** Auto-generated unique ID (injected by Vite plugin). */
|
|
249
|
+
$$id: string;
|
|
250
|
+
/** In dev mode, the actual handler function that path() can call. */
|
|
251
|
+
handler: Handler<TParams>;
|
|
252
|
+
/** Returns the list of param objects to pre-render (dynamic routes). */
|
|
253
|
+
getParams?: (ctx: GetParamsContext) => Promise<TParams[]> | TParams[];
|
|
254
|
+
/** Pre-render options. */
|
|
255
|
+
options?: PrerenderOptions;
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
// -- Overloads --------------------------------------------------------------
|
|
259
|
+
//
|
|
260
|
+
// T accepts: named route string (global or .local) OR explicit param object.
|
|
261
|
+
// Named routes resolve params from GeneratedRouteMap, e.g.:
|
|
262
|
+
// Prerender<"locale.detail"> → params = { locale: string; slug: string }
|
|
263
|
+
// Explicit params work as before:
|
|
264
|
+
// Prerender<{ slug: string }> → params = { slug: string }
|
|
265
|
+
|
|
266
|
+
// Overload 1: Static handler, no passthrough (build-time only)
|
|
267
|
+
export function Prerender<
|
|
268
|
+
T extends
|
|
269
|
+
| keyof DefaultPrerenderRouteMap
|
|
270
|
+
| `.${keyof TRouteMap & string}`
|
|
271
|
+
| Record<string, any> = {},
|
|
272
|
+
TRouteMap extends {} = DefaultPrerenderRouteMap,
|
|
273
|
+
>(
|
|
274
|
+
handler: (
|
|
275
|
+
ctx: BuildContext<ResolvePrerenderParams<T, TRouteMap>>,
|
|
276
|
+
) => ReactNode | Promise<ReactNode>,
|
|
277
|
+
options?: PrerenderOptions & { passthrough?: false },
|
|
278
|
+
__injectedId?: string,
|
|
279
|
+
): PrerenderHandlerDefinition<ResolvePrerenderParams<T, TRouteMap>>;
|
|
280
|
+
|
|
281
|
+
// Overload 2: Static handler, passthrough (build + live — full HandlerContext)
|
|
282
|
+
export function Prerender<
|
|
283
|
+
T extends
|
|
284
|
+
| keyof DefaultPrerenderRouteMap
|
|
285
|
+
| `.${keyof TRouteMap & string}`
|
|
286
|
+
| Record<string, any> = {},
|
|
287
|
+
TRouteMap extends {} = DefaultPrerenderRouteMap,
|
|
288
|
+
TEnv = DefaultEnv,
|
|
289
|
+
>(
|
|
290
|
+
handler: (
|
|
291
|
+
ctx: PrerenderPassthroughContext<
|
|
292
|
+
ResolvePrerenderParams<T, TRouteMap>,
|
|
293
|
+
TEnv
|
|
294
|
+
>,
|
|
295
|
+
) =>
|
|
296
|
+
| ReactNode
|
|
297
|
+
| PrerenderPassthroughResult
|
|
298
|
+
| Promise<ReactNode | PrerenderPassthroughResult>,
|
|
299
|
+
options: PrerenderOptions & { passthrough: true },
|
|
300
|
+
__injectedId?: string,
|
|
301
|
+
): PrerenderHandlerDefinition<ResolvePrerenderParams<T, TRouteMap>>;
|
|
302
|
+
|
|
303
|
+
// Overload 3: Dynamic handler, no passthrough (build-time only)
|
|
304
|
+
export function Prerender<
|
|
305
|
+
T extends
|
|
306
|
+
| keyof DefaultPrerenderRouteMap
|
|
307
|
+
| `.${keyof TRouteMap & string}`
|
|
308
|
+
| Record<string, any>,
|
|
309
|
+
TRouteMap extends {} = DefaultPrerenderRouteMap,
|
|
310
|
+
>(
|
|
311
|
+
getParams: (
|
|
312
|
+
ctx: GetParamsContext,
|
|
313
|
+
) =>
|
|
314
|
+
| Promise<ResolvePrerenderParams<T, TRouteMap>[]>
|
|
315
|
+
| ResolvePrerenderParams<T, TRouteMap>[],
|
|
316
|
+
handler: (
|
|
317
|
+
ctx: BuildContext<ResolvePrerenderParams<T, TRouteMap>>,
|
|
318
|
+
) => ReactNode | Promise<ReactNode>,
|
|
319
|
+
options?: PrerenderOptions & { passthrough?: false },
|
|
320
|
+
__injectedId?: string,
|
|
321
|
+
): PrerenderHandlerDefinition<ResolvePrerenderParams<T, TRouteMap>>;
|
|
322
|
+
|
|
323
|
+
// Overload 4: Dynamic handler, passthrough (build + live — full HandlerContext)
|
|
324
|
+
export function Prerender<
|
|
325
|
+
T extends
|
|
326
|
+
| keyof DefaultPrerenderRouteMap
|
|
327
|
+
| `.${keyof TRouteMap & string}`
|
|
328
|
+
| Record<string, any>,
|
|
329
|
+
TRouteMap extends {} = DefaultPrerenderRouteMap,
|
|
330
|
+
TEnv = DefaultEnv,
|
|
331
|
+
>(
|
|
332
|
+
getParams: (
|
|
333
|
+
ctx: GetParamsContext,
|
|
334
|
+
) =>
|
|
335
|
+
| Promise<ResolvePrerenderParams<T, TRouteMap>[]>
|
|
336
|
+
| ResolvePrerenderParams<T, TRouteMap>[],
|
|
337
|
+
handler: (
|
|
338
|
+
ctx: PrerenderPassthroughContext<
|
|
339
|
+
ResolvePrerenderParams<T, TRouteMap>,
|
|
340
|
+
TEnv
|
|
341
|
+
>,
|
|
342
|
+
) =>
|
|
343
|
+
| ReactNode
|
|
344
|
+
| PrerenderPassthroughResult
|
|
345
|
+
| Promise<ReactNode | PrerenderPassthroughResult>,
|
|
346
|
+
options: PrerenderOptions & { passthrough: true },
|
|
347
|
+
__injectedId?: string,
|
|
348
|
+
): PrerenderHandlerDefinition<ResolvePrerenderParams<T, TRouteMap>>;
|
|
349
|
+
|
|
350
|
+
// -- Implementation ---------------------------------------------------------
|
|
351
|
+
|
|
352
|
+
export function Prerender<TParams extends Record<string, any>>(
|
|
353
|
+
handlerOrGetParams: Function,
|
|
354
|
+
handlerOrOptions?: Function | PrerenderOptions,
|
|
355
|
+
optionsOrId?: PrerenderOptions | string,
|
|
356
|
+
maybeId?: string,
|
|
357
|
+
): PrerenderHandlerDefinition<TParams> {
|
|
358
|
+
// Resolve overloads:
|
|
359
|
+
// 1 fn arg: Prerender(handler, options?, __injectedId?)
|
|
360
|
+
// 2 fn args: Prerender(getParams, handler, options?, __injectedId?)
|
|
361
|
+
let handler: Handler<TParams>;
|
|
362
|
+
let getParams: (() => Promise<TParams[]> | TParams[]) | undefined;
|
|
363
|
+
let options: PrerenderOptions | undefined;
|
|
364
|
+
let id: string;
|
|
365
|
+
|
|
366
|
+
if (typeof handlerOrOptions === "function") {
|
|
367
|
+
// Two function args: getParams + handler
|
|
368
|
+
getParams = handlerOrGetParams as () => Promise<TParams[]> | TParams[];
|
|
369
|
+
handler = handlerOrOptions as Handler<TParams>;
|
|
370
|
+
if (typeof optionsOrId === "string") {
|
|
371
|
+
id = optionsOrId;
|
|
372
|
+
} else {
|
|
373
|
+
options = optionsOrId as PrerenderOptions | undefined;
|
|
374
|
+
id = maybeId ?? "";
|
|
375
|
+
}
|
|
376
|
+
} else {
|
|
377
|
+
// Single function arg: handler only
|
|
378
|
+
handler = handlerOrGetParams as Handler<TParams>;
|
|
379
|
+
if (typeof handlerOrOptions === "object" && handlerOrOptions !== null) {
|
|
380
|
+
options = handlerOrOptions as PrerenderOptions;
|
|
381
|
+
}
|
|
382
|
+
if (typeof optionsOrId === "string") {
|
|
383
|
+
id = optionsOrId;
|
|
384
|
+
} else {
|
|
385
|
+
id = maybeId ?? "";
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
|
|
389
|
+
if (isCachedFunction(handler)) {
|
|
390
|
+
throw new Error(
|
|
391
|
+
'A "use cache" function cannot be used as a Prerender() handler. ' +
|
|
392
|
+
"Prerender handlers are rendered at build time. Remove the " +
|
|
393
|
+
'"use cache" directive — Prerender already provides caching.',
|
|
394
|
+
);
|
|
395
|
+
}
|
|
396
|
+
if (getParams && isCachedFunction(getParams)) {
|
|
397
|
+
throw new Error(
|
|
398
|
+
'A "use cache" function cannot be used as Prerender() getParams. ' +
|
|
399
|
+
"getParams runs at build time to enumerate params. Remove the " +
|
|
400
|
+
'"use cache" directive.',
|
|
401
|
+
);
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
if (!id) {
|
|
405
|
+
throw new Error(
|
|
406
|
+
"[rsc-router] Prerender: missing $$id. " +
|
|
407
|
+
"Ensure the exposeInternalIds Vite plugin is configured.",
|
|
408
|
+
);
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
return {
|
|
412
|
+
__brand: "prerenderHandler" as const,
|
|
413
|
+
$$id: id,
|
|
414
|
+
handler,
|
|
415
|
+
...(getParams ? { getParams } : {}),
|
|
416
|
+
...(options ? { options } : {}),
|
|
417
|
+
};
|
|
418
|
+
}
|
|
419
|
+
|
|
420
|
+
// -- Passthrough sentinel ---------------------------------------------------
|
|
421
|
+
|
|
422
|
+
/**
|
|
423
|
+
* Sentinel returned by `ctx.passthrough()` to signal that a specific param set
|
|
424
|
+
* should not produce a local prerender artifact. The build skips writing the
|
|
425
|
+
* entry; at runtime the handler runs live (requires `{ passthrough: true }`).
|
|
426
|
+
*/
|
|
427
|
+
export const PRERENDER_PASSTHROUGH: Readonly<{
|
|
428
|
+
__brand: "prerenderPassthrough";
|
|
429
|
+
}> = Object.freeze({
|
|
430
|
+
__brand: "prerenderPassthrough" as const,
|
|
431
|
+
});
|
|
432
|
+
|
|
433
|
+
export type PrerenderPassthroughResult = typeof PRERENDER_PASSTHROUGH;
|
|
434
|
+
|
|
435
|
+
/**
|
|
436
|
+
* Type guard to check if a value is the passthrough sentinel.
|
|
437
|
+
*/
|
|
438
|
+
export function isPrerenderPassthrough(
|
|
439
|
+
value: unknown,
|
|
440
|
+
): value is PrerenderPassthroughResult {
|
|
441
|
+
return (
|
|
442
|
+
typeof value === "object" &&
|
|
443
|
+
value !== null &&
|
|
444
|
+
"__brand" in value &&
|
|
445
|
+
(value as { __brand: unknown }).__brand === "prerenderPassthrough"
|
|
446
|
+
);
|
|
447
|
+
}
|
|
448
|
+
|
|
449
|
+
// -- Type guard -------------------------------------------------------------
|
|
450
|
+
|
|
451
|
+
/**
|
|
452
|
+
* Type guard to check if a value is a PrerenderHandlerDefinition.
|
|
453
|
+
*/
|
|
454
|
+
export function isPrerenderHandler(
|
|
455
|
+
value: unknown,
|
|
456
|
+
): value is PrerenderHandlerDefinition {
|
|
457
|
+
return (
|
|
458
|
+
typeof value === "object" &&
|
|
459
|
+
value !== null &&
|
|
460
|
+
"__brand" in value &&
|
|
461
|
+
(value as { __brand: unknown }).__brand === "prerenderHandler"
|
|
462
|
+
);
|
|
463
|
+
}
|