@rangojs/router 0.0.0-experimental.3 → 0.0.0-experimental.30

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