@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.
Files changed (305) hide show
  1. package/AGENTS.md +5 -0
  2. package/README.md +899 -0
  3. package/dist/bin/rango.js +1601 -0
  4. package/dist/vite/index.js +5214 -0
  5. package/package.json +176 -0
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +262 -0
  8. package/skills/caching/SKILL.md +220 -0
  9. package/skills/composability/SKILL.md +172 -0
  10. package/skills/debug-manifest/SKILL.md +112 -0
  11. package/skills/document-cache/SKILL.md +182 -0
  12. package/skills/fonts/SKILL.md +167 -0
  13. package/skills/hooks/SKILL.md +704 -0
  14. package/skills/host-router/SKILL.md +218 -0
  15. package/skills/intercept/SKILL.md +313 -0
  16. package/skills/layout/SKILL.md +310 -0
  17. package/skills/links/SKILL.md +239 -0
  18. package/skills/loader/SKILL.md +596 -0
  19. package/skills/middleware/SKILL.md +339 -0
  20. package/skills/mime-routes/SKILL.md +128 -0
  21. package/skills/parallel/SKILL.md +305 -0
  22. package/skills/prerender/SKILL.md +643 -0
  23. package/skills/rango/SKILL.md +118 -0
  24. package/skills/response-routes/SKILL.md +411 -0
  25. package/skills/route/SKILL.md +385 -0
  26. package/skills/router-setup/SKILL.md +439 -0
  27. package/skills/tailwind/SKILL.md +129 -0
  28. package/skills/theme/SKILL.md +79 -0
  29. package/skills/typesafety/SKILL.md +623 -0
  30. package/skills/use-cache/SKILL.md +324 -0
  31. package/src/__internal.ts +273 -0
  32. package/src/bin/rango.ts +321 -0
  33. package/src/browser/action-coordinator.ts +97 -0
  34. package/src/browser/action-response-classifier.ts +99 -0
  35. package/src/browser/event-controller.ts +899 -0
  36. package/src/browser/history-state.ts +80 -0
  37. package/src/browser/index.ts +18 -0
  38. package/src/browser/intercept-utils.ts +52 -0
  39. package/src/browser/link-interceptor.ts +141 -0
  40. package/src/browser/logging.ts +55 -0
  41. package/src/browser/merge-segment-loaders.ts +134 -0
  42. package/src/browser/navigation-bridge.ts +645 -0
  43. package/src/browser/navigation-client.ts +215 -0
  44. package/src/browser/navigation-store.ts +806 -0
  45. package/src/browser/navigation-transaction.ts +295 -0
  46. package/src/browser/network-error-handler.ts +61 -0
  47. package/src/browser/partial-update.ts +550 -0
  48. package/src/browser/prefetch/cache.ts +146 -0
  49. package/src/browser/prefetch/fetch.ts +135 -0
  50. package/src/browser/prefetch/observer.ts +65 -0
  51. package/src/browser/prefetch/policy.ts +42 -0
  52. package/src/browser/prefetch/queue.ts +88 -0
  53. package/src/browser/rango-state.ts +112 -0
  54. package/src/browser/react/Link.tsx +360 -0
  55. package/src/browser/react/NavigationProvider.tsx +386 -0
  56. package/src/browser/react/ScrollRestoration.tsx +94 -0
  57. package/src/browser/react/context.ts +59 -0
  58. package/src/browser/react/filter-segment-order.ts +11 -0
  59. package/src/browser/react/index.ts +52 -0
  60. package/src/browser/react/location-state-shared.ts +162 -0
  61. package/src/browser/react/location-state.ts +107 -0
  62. package/src/browser/react/mount-context.ts +37 -0
  63. package/src/browser/react/nonce-context.ts +23 -0
  64. package/src/browser/react/shallow-equal.ts +27 -0
  65. package/src/browser/react/use-action.ts +218 -0
  66. package/src/browser/react/use-client-cache.ts +58 -0
  67. package/src/browser/react/use-handle.ts +162 -0
  68. package/src/browser/react/use-href.tsx +40 -0
  69. package/src/browser/react/use-link-status.ts +135 -0
  70. package/src/browser/react/use-mount.ts +31 -0
  71. package/src/browser/react/use-navigation.ts +99 -0
  72. package/src/browser/react/use-params.ts +65 -0
  73. package/src/browser/react/use-pathname.ts +47 -0
  74. package/src/browser/react/use-router.ts +63 -0
  75. package/src/browser/react/use-search-params.ts +56 -0
  76. package/src/browser/react/use-segments.ts +171 -0
  77. package/src/browser/response-adapter.ts +73 -0
  78. package/src/browser/rsc-router.tsx +431 -0
  79. package/src/browser/scroll-restoration.ts +400 -0
  80. package/src/browser/segment-reconciler.ts +216 -0
  81. package/src/browser/segment-structure-assert.ts +83 -0
  82. package/src/browser/server-action-bridge.ts +667 -0
  83. package/src/browser/shallow.ts +40 -0
  84. package/src/browser/types.ts +538 -0
  85. package/src/browser/validate-redirect-origin.ts +29 -0
  86. package/src/build/generate-manifest.ts +438 -0
  87. package/src/build/generate-route-types.ts +36 -0
  88. package/src/build/index.ts +35 -0
  89. package/src/build/route-trie.ts +265 -0
  90. package/src/build/route-types/ast-helpers.ts +25 -0
  91. package/src/build/route-types/ast-route-extraction.ts +98 -0
  92. package/src/build/route-types/codegen.ts +102 -0
  93. package/src/build/route-types/include-resolution.ts +411 -0
  94. package/src/build/route-types/param-extraction.ts +48 -0
  95. package/src/build/route-types/per-module-writer.ts +128 -0
  96. package/src/build/route-types/router-processing.ts +469 -0
  97. package/src/build/route-types/scan-filter.ts +78 -0
  98. package/src/build/runtime-discovery.ts +231 -0
  99. package/src/cache/background-task.ts +34 -0
  100. package/src/cache/cache-key-utils.ts +44 -0
  101. package/src/cache/cache-policy.ts +125 -0
  102. package/src/cache/cache-runtime.ts +338 -0
  103. package/src/cache/cache-scope.ts +382 -0
  104. package/src/cache/cf/cf-cache-store.ts +540 -0
  105. package/src/cache/cf/index.ts +25 -0
  106. package/src/cache/document-cache.ts +369 -0
  107. package/src/cache/handle-capture.ts +81 -0
  108. package/src/cache/handle-snapshot.ts +41 -0
  109. package/src/cache/index.ts +43 -0
  110. package/src/cache/memory-segment-store.ts +328 -0
  111. package/src/cache/profile-registry.ts +73 -0
  112. package/src/cache/read-through-swr.ts +134 -0
  113. package/src/cache/segment-codec.ts +256 -0
  114. package/src/cache/taint.ts +98 -0
  115. package/src/cache/types.ts +342 -0
  116. package/src/client.rsc.tsx +85 -0
  117. package/src/client.tsx +601 -0
  118. package/src/component-utils.ts +76 -0
  119. package/src/components/DefaultDocument.tsx +27 -0
  120. package/src/context-var.ts +86 -0
  121. package/src/debug.ts +243 -0
  122. package/src/default-error-boundary.tsx +88 -0
  123. package/src/deps/browser.ts +8 -0
  124. package/src/deps/html-stream-client.ts +2 -0
  125. package/src/deps/html-stream-server.ts +2 -0
  126. package/src/deps/rsc.ts +10 -0
  127. package/src/deps/ssr.ts +2 -0
  128. package/src/errors.ts +365 -0
  129. package/src/handle.ts +135 -0
  130. package/src/handles/MetaTags.tsx +246 -0
  131. package/src/handles/breadcrumbs.ts +66 -0
  132. package/src/handles/index.ts +7 -0
  133. package/src/handles/meta.ts +264 -0
  134. package/src/host/cookie-handler.ts +165 -0
  135. package/src/host/errors.ts +97 -0
  136. package/src/host/index.ts +53 -0
  137. package/src/host/pattern-matcher.ts +214 -0
  138. package/src/host/router.ts +352 -0
  139. package/src/host/testing.ts +79 -0
  140. package/src/host/types.ts +146 -0
  141. package/src/host/utils.ts +25 -0
  142. package/src/href-client.ts +222 -0
  143. package/src/index.rsc.ts +233 -0
  144. package/src/index.ts +277 -0
  145. package/src/internal-debug.ts +11 -0
  146. package/src/loader.rsc.ts +89 -0
  147. package/src/loader.ts +64 -0
  148. package/src/network-error-thrower.tsx +23 -0
  149. package/src/outlet-context.ts +15 -0
  150. package/src/outlet-provider.tsx +45 -0
  151. package/src/prerender/param-hash.ts +37 -0
  152. package/src/prerender/store.ts +185 -0
  153. package/src/prerender.ts +463 -0
  154. package/src/reverse.ts +330 -0
  155. package/src/root-error-boundary.tsx +289 -0
  156. package/src/route-content-wrapper.tsx +196 -0
  157. package/src/route-definition/dsl-helpers.ts +934 -0
  158. package/src/route-definition/helper-factories.ts +200 -0
  159. package/src/route-definition/helpers-types.ts +430 -0
  160. package/src/route-definition/index.ts +52 -0
  161. package/src/route-definition/redirect.ts +93 -0
  162. package/src/route-definition.ts +1 -0
  163. package/src/route-map-builder.ts +275 -0
  164. package/src/route-name.ts +53 -0
  165. package/src/route-types.ts +259 -0
  166. package/src/router/content-negotiation.ts +116 -0
  167. package/src/router/debug-manifest.ts +72 -0
  168. package/src/router/error-handling.ts +287 -0
  169. package/src/router/find-match.ts +158 -0
  170. package/src/router/handler-context.ts +451 -0
  171. package/src/router/intercept-resolution.ts +395 -0
  172. package/src/router/lazy-includes.ts +234 -0
  173. package/src/router/loader-resolution.ts +420 -0
  174. package/src/router/logging.ts +248 -0
  175. package/src/router/manifest.ts +267 -0
  176. package/src/router/match-api.ts +620 -0
  177. package/src/router/match-context.ts +266 -0
  178. package/src/router/match-handlers.ts +440 -0
  179. package/src/router/match-middleware/background-revalidation.ts +223 -0
  180. package/src/router/match-middleware/cache-lookup.ts +634 -0
  181. package/src/router/match-middleware/cache-store.ts +295 -0
  182. package/src/router/match-middleware/index.ts +81 -0
  183. package/src/router/match-middleware/intercept-resolution.ts +306 -0
  184. package/src/router/match-middleware/segment-resolution.ts +192 -0
  185. package/src/router/match-pipelines.ts +179 -0
  186. package/src/router/match-result.ts +219 -0
  187. package/src/router/metrics.ts +282 -0
  188. package/src/router/middleware-cookies.ts +55 -0
  189. package/src/router/middleware-types.ts +222 -0
  190. package/src/router/middleware.ts +748 -0
  191. package/src/router/pattern-matching.ts +563 -0
  192. package/src/router/prerender-match.ts +402 -0
  193. package/src/router/preview-match.ts +170 -0
  194. package/src/router/revalidation.ts +289 -0
  195. package/src/router/router-context.ts +316 -0
  196. package/src/router/router-interfaces.ts +452 -0
  197. package/src/router/router-options.ts +592 -0
  198. package/src/router/router-registry.ts +24 -0
  199. package/src/router/segment-resolution/fresh.ts +570 -0
  200. package/src/router/segment-resolution/helpers.ts +263 -0
  201. package/src/router/segment-resolution/loader-cache.ts +198 -0
  202. package/src/router/segment-resolution/revalidation.ts +1239 -0
  203. package/src/router/segment-resolution/static-store.ts +67 -0
  204. package/src/router/segment-resolution.ts +21 -0
  205. package/src/router/segment-wrappers.ts +289 -0
  206. package/src/router/telemetry-otel.ts +299 -0
  207. package/src/router/telemetry.ts +300 -0
  208. package/src/router/timeout.ts +148 -0
  209. package/src/router/trie-matching.ts +239 -0
  210. package/src/router/types.ts +170 -0
  211. package/src/router.ts +1002 -0
  212. package/src/rsc/handler-context.ts +45 -0
  213. package/src/rsc/handler.ts +1089 -0
  214. package/src/rsc/helpers.ts +198 -0
  215. package/src/rsc/index.ts +36 -0
  216. package/src/rsc/loader-fetch.ts +209 -0
  217. package/src/rsc/manifest-init.ts +86 -0
  218. package/src/rsc/nonce.ts +32 -0
  219. package/src/rsc/origin-guard.ts +141 -0
  220. package/src/rsc/progressive-enhancement.ts +379 -0
  221. package/src/rsc/response-error.ts +37 -0
  222. package/src/rsc/response-route-handler.ts +347 -0
  223. package/src/rsc/rsc-rendering.ts +235 -0
  224. package/src/rsc/runtime-warnings.ts +42 -0
  225. package/src/rsc/server-action.ts +348 -0
  226. package/src/rsc/ssr-setup.ts +128 -0
  227. package/src/rsc/types.ts +263 -0
  228. package/src/search-params.ts +230 -0
  229. package/src/segment-system.tsx +454 -0
  230. package/src/server/context.ts +591 -0
  231. package/src/server/cookie-store.ts +190 -0
  232. package/src/server/fetchable-loader-store.ts +37 -0
  233. package/src/server/handle-store.ts +308 -0
  234. package/src/server/loader-registry.ts +133 -0
  235. package/src/server/request-context.ts +914 -0
  236. package/src/server/root-layout.tsx +10 -0
  237. package/src/server/tsconfig.json +14 -0
  238. package/src/server.ts +51 -0
  239. package/src/ssr/index.tsx +365 -0
  240. package/src/static-handler.ts +114 -0
  241. package/src/theme/ThemeProvider.tsx +297 -0
  242. package/src/theme/ThemeScript.tsx +61 -0
  243. package/src/theme/constants.ts +62 -0
  244. package/src/theme/index.ts +48 -0
  245. package/src/theme/theme-context.ts +44 -0
  246. package/src/theme/theme-script.ts +155 -0
  247. package/src/theme/types.ts +182 -0
  248. package/src/theme/use-theme.ts +44 -0
  249. package/src/types/boundaries.ts +158 -0
  250. package/src/types/cache-types.ts +198 -0
  251. package/src/types/error-types.ts +192 -0
  252. package/src/types/global-namespace.ts +100 -0
  253. package/src/types/handler-context.ts +687 -0
  254. package/src/types/index.ts +88 -0
  255. package/src/types/loader-types.ts +183 -0
  256. package/src/types/route-config.ts +170 -0
  257. package/src/types/route-entry.ts +102 -0
  258. package/src/types/segments.ts +148 -0
  259. package/src/types.ts +1 -0
  260. package/src/urls/include-helper.ts +197 -0
  261. package/src/urls/index.ts +53 -0
  262. package/src/urls/path-helper-types.ts +339 -0
  263. package/src/urls/path-helper.ts +329 -0
  264. package/src/urls/pattern-types.ts +95 -0
  265. package/src/urls/response-types.ts +106 -0
  266. package/src/urls/type-extraction.ts +372 -0
  267. package/src/urls/urls-function.ts +98 -0
  268. package/src/urls.ts +1 -0
  269. package/src/use-loader.tsx +354 -0
  270. package/src/vite/discovery/bundle-postprocess.ts +184 -0
  271. package/src/vite/discovery/discover-routers.ts +344 -0
  272. package/src/vite/discovery/prerender-collection.ts +385 -0
  273. package/src/vite/discovery/route-types-writer.ts +258 -0
  274. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  275. package/src/vite/discovery/state.ts +110 -0
  276. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  277. package/src/vite/index.ts +16 -0
  278. package/src/vite/plugin-types.ts +131 -0
  279. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  280. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  281. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  282. package/src/vite/plugins/expose-action-id.ts +365 -0
  283. package/src/vite/plugins/expose-id-utils.ts +287 -0
  284. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  285. package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
  286. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  287. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  288. package/src/vite/plugins/expose-ids/types.ts +45 -0
  289. package/src/vite/plugins/expose-internal-ids.ts +569 -0
  290. package/src/vite/plugins/refresh-cmd.ts +65 -0
  291. package/src/vite/plugins/use-cache-transform.ts +323 -0
  292. package/src/vite/plugins/version-injector.ts +83 -0
  293. package/src/vite/plugins/version-plugin.ts +254 -0
  294. package/src/vite/plugins/version.d.ts +12 -0
  295. package/src/vite/plugins/virtual-entries.ts +123 -0
  296. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  297. package/src/vite/rango.ts +510 -0
  298. package/src/vite/router-discovery.ts +785 -0
  299. package/src/vite/utils/ast-handler-extract.ts +517 -0
  300. package/src/vite/utils/banner.ts +36 -0
  301. package/src/vite/utils/bundle-analysis.ts +137 -0
  302. package/src/vite/utils/manifest-utils.ts +70 -0
  303. package/src/vite/utils/package-resolution.ts +121 -0
  304. package/src/vite/utils/prerender-utils.ts +189 -0
  305. 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
+ }
@@ -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
+ }