@rangojs/router 0.0.0-experimental.7 → 0.0.0-experimental.71

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 (307) hide show
  1. package/AGENTS.md +9 -0
  2. package/README.md +942 -4
  3. package/dist/bin/rango.js +1689 -0
  4. package/dist/vite/index.js +4951 -930
  5. package/package.json +70 -60
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +294 -0
  8. package/skills/caching/SKILL.md +93 -23
  9. package/skills/composability/SKILL.md +172 -0
  10. package/skills/debug-manifest/SKILL.md +12 -8
  11. package/skills/document-cache/SKILL.md +18 -16
  12. package/skills/fonts/SKILL.md +167 -0
  13. package/skills/hooks/SKILL.md +334 -72
  14. package/skills/host-router/SKILL.md +218 -0
  15. package/skills/intercept/SKILL.md +131 -8
  16. package/skills/layout/SKILL.md +100 -3
  17. package/skills/links/SKILL.md +92 -31
  18. package/skills/loader/SKILL.md +404 -44
  19. package/skills/middleware/SKILL.md +173 -34
  20. package/skills/mime-routes/SKILL.md +128 -0
  21. package/skills/parallel/SKILL.md +204 -1
  22. package/skills/prerender/SKILL.md +685 -0
  23. package/skills/rango/SKILL.md +85 -16
  24. package/skills/response-routes/SKILL.md +411 -0
  25. package/skills/route/SKILL.md +257 -14
  26. package/skills/router-setup/SKILL.md +210 -32
  27. package/skills/tailwind/SKILL.md +129 -0
  28. package/skills/theme/SKILL.md +9 -8
  29. package/skills/typesafety/SKILL.md +328 -89
  30. package/skills/use-cache/SKILL.md +324 -0
  31. package/src/__internal.ts +102 -4
  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/app-version.ts +14 -0
  36. package/src/browser/event-controller.ts +92 -64
  37. package/src/browser/history-state.ts +80 -0
  38. package/src/browser/intercept-utils.ts +52 -0
  39. package/src/browser/link-interceptor.ts +24 -4
  40. package/src/browser/logging.ts +55 -0
  41. package/src/browser/merge-segment-loaders.ts +20 -12
  42. package/src/browser/navigation-bridge.ts +296 -558
  43. package/src/browser/navigation-client.ts +179 -69
  44. package/src/browser/navigation-store.ts +73 -55
  45. package/src/browser/navigation-transaction.ts +297 -0
  46. package/src/browser/network-error-handler.ts +61 -0
  47. package/src/browser/partial-update.ts +328 -313
  48. package/src/browser/prefetch/cache.ts +206 -0
  49. package/src/browser/prefetch/fetch.ts +150 -0
  50. package/src/browser/prefetch/observer.ts +65 -0
  51. package/src/browser/prefetch/policy.ts +48 -0
  52. package/src/browser/prefetch/queue.ts +160 -0
  53. package/src/browser/prefetch/resource-ready.ts +77 -0
  54. package/src/browser/rango-state.ts +112 -0
  55. package/src/browser/react/Link.tsx +230 -74
  56. package/src/browser/react/NavigationProvider.tsx +87 -11
  57. package/src/browser/react/context.ts +11 -0
  58. package/src/browser/react/filter-segment-order.ts +11 -0
  59. package/src/browser/react/index.ts +12 -12
  60. package/src/browser/react/location-state-shared.ts +95 -53
  61. package/src/browser/react/location-state.ts +60 -15
  62. package/src/browser/react/mount-context.ts +6 -1
  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 +29 -51
  66. package/src/browser/react/use-client-cache.ts +5 -3
  67. package/src/browser/react/use-handle.ts +30 -126
  68. package/src/browser/react/use-href.tsx +2 -2
  69. package/src/browser/react/use-link-status.ts +6 -5
  70. package/src/browser/react/use-navigation.ts +22 -63
  71. package/src/browser/react/use-params.ts +65 -0
  72. package/src/browser/react/use-pathname.ts +47 -0
  73. package/src/browser/react/use-router.ts +76 -0
  74. package/src/browser/react/use-search-params.ts +56 -0
  75. package/src/browser/react/use-segments.ts +80 -97
  76. package/src/browser/response-adapter.ts +73 -0
  77. package/src/browser/rsc-router.tsx +214 -58
  78. package/src/browser/scroll-restoration.ts +127 -52
  79. package/src/browser/segment-reconciler.ts +221 -0
  80. package/src/browser/segment-structure-assert.ts +16 -0
  81. package/src/browser/server-action-bridge.ts +510 -603
  82. package/src/browser/shallow.ts +6 -1
  83. package/src/browser/types.ts +141 -48
  84. package/src/browser/validate-redirect-origin.ts +29 -0
  85. package/src/build/generate-manifest.ts +235 -24
  86. package/src/build/generate-route-types.ts +39 -0
  87. package/src/build/index.ts +13 -0
  88. package/src/build/route-trie.ts +265 -0
  89. package/src/build/route-types/ast-helpers.ts +25 -0
  90. package/src/build/route-types/ast-route-extraction.ts +98 -0
  91. package/src/build/route-types/codegen.ts +102 -0
  92. package/src/build/route-types/include-resolution.ts +418 -0
  93. package/src/build/route-types/param-extraction.ts +48 -0
  94. package/src/build/route-types/per-module-writer.ts +128 -0
  95. package/src/build/route-types/router-processing.ts +618 -0
  96. package/src/build/route-types/scan-filter.ts +85 -0
  97. package/src/build/runtime-discovery.ts +231 -0
  98. package/src/cache/background-task.ts +34 -0
  99. package/src/cache/cache-key-utils.ts +44 -0
  100. package/src/cache/cache-policy.ts +125 -0
  101. package/src/cache/cache-runtime.ts +342 -0
  102. package/src/cache/cache-scope.ts +167 -309
  103. package/src/cache/cf/cf-cache-store.ts +571 -17
  104. package/src/cache/cf/index.ts +13 -3
  105. package/src/cache/document-cache.ts +116 -77
  106. package/src/cache/handle-capture.ts +81 -0
  107. package/src/cache/handle-snapshot.ts +41 -0
  108. package/src/cache/index.ts +1 -15
  109. package/src/cache/memory-segment-store.ts +191 -13
  110. package/src/cache/profile-registry.ts +73 -0
  111. package/src/cache/read-through-swr.ts +134 -0
  112. package/src/cache/segment-codec.ts +256 -0
  113. package/src/cache/taint.ts +153 -0
  114. package/src/cache/types.ts +72 -122
  115. package/src/client.rsc.tsx +3 -1
  116. package/src/client.tsx +105 -179
  117. package/src/component-utils.ts +4 -4
  118. package/src/components/DefaultDocument.tsx +5 -1
  119. package/src/context-var.ts +156 -0
  120. package/src/debug.ts +19 -9
  121. package/src/errors.ts +108 -2
  122. package/src/handle.ts +55 -29
  123. package/src/handles/MetaTags.tsx +73 -20
  124. package/src/handles/breadcrumbs.ts +66 -0
  125. package/src/handles/index.ts +1 -0
  126. package/src/handles/meta.ts +30 -13
  127. package/src/host/cookie-handler.ts +21 -15
  128. package/src/host/errors.ts +8 -8
  129. package/src/host/index.ts +4 -7
  130. package/src/host/pattern-matcher.ts +27 -27
  131. package/src/host/router.ts +61 -39
  132. package/src/host/testing.ts +8 -8
  133. package/src/host/types.ts +15 -7
  134. package/src/host/utils.ts +1 -1
  135. package/src/href-client.ts +119 -29
  136. package/src/index.rsc.ts +155 -19
  137. package/src/index.ts +223 -30
  138. package/src/internal-debug.ts +11 -0
  139. package/src/loader.rsc.ts +26 -157
  140. package/src/loader.ts +27 -10
  141. package/src/network-error-thrower.tsx +3 -1
  142. package/src/outlet-provider.tsx +45 -0
  143. package/src/prerender/param-hash.ts +37 -0
  144. package/src/prerender/store.ts +186 -0
  145. package/src/prerender.ts +524 -0
  146. package/src/reverse.ts +351 -0
  147. package/src/root-error-boundary.tsx +41 -29
  148. package/src/route-content-wrapper.tsx +7 -4
  149. package/src/route-definition/dsl-helpers.ts +982 -0
  150. package/src/route-definition/helper-factories.ts +200 -0
  151. package/src/route-definition/helpers-types.ts +434 -0
  152. package/src/route-definition/index.ts +55 -0
  153. package/src/route-definition/redirect.ts +101 -0
  154. package/src/route-definition/resolve-handler-use.ts +149 -0
  155. package/src/route-definition.ts +1 -1428
  156. package/src/route-map-builder.ts +217 -123
  157. package/src/route-name.ts +53 -0
  158. package/src/route-types.ts +70 -8
  159. package/src/router/content-negotiation.ts +215 -0
  160. package/src/router/debug-manifest.ts +72 -0
  161. package/src/router/error-handling.ts +9 -9
  162. package/src/router/find-match.ts +160 -0
  163. package/src/router/handler-context.ts +435 -86
  164. package/src/router/intercept-resolution.ts +402 -0
  165. package/src/router/lazy-includes.ts +237 -0
  166. package/src/router/loader-resolution.ts +356 -128
  167. package/src/router/logging.ts +251 -0
  168. package/src/router/manifest.ts +154 -35
  169. package/src/router/match-api.ts +555 -0
  170. package/src/router/match-context.ts +5 -3
  171. package/src/router/match-handlers.ts +440 -0
  172. package/src/router/match-middleware/background-revalidation.ts +108 -93
  173. package/src/router/match-middleware/cache-lookup.ts +459 -10
  174. package/src/router/match-middleware/cache-store.ts +98 -26
  175. package/src/router/match-middleware/intercept-resolution.ts +57 -17
  176. package/src/router/match-middleware/segment-resolution.ts +80 -6
  177. package/src/router/match-pipelines.ts +10 -45
  178. package/src/router/match-result.ts +135 -35
  179. package/src/router/metrics.ts +240 -15
  180. package/src/router/middleware-cookies.ts +55 -0
  181. package/src/router/middleware-types.ts +220 -0
  182. package/src/router/middleware.ts +324 -369
  183. package/src/router/navigation-snapshot.ts +182 -0
  184. package/src/router/pattern-matching.ts +211 -43
  185. package/src/router/prerender-match.ts +502 -0
  186. package/src/router/preview-match.ts +98 -0
  187. package/src/router/request-classification.ts +310 -0
  188. package/src/router/revalidation.ts +137 -38
  189. package/src/router/route-snapshot.ts +245 -0
  190. package/src/router/router-context.ts +41 -21
  191. package/src/router/router-interfaces.ts +484 -0
  192. package/src/router/router-options.ts +618 -0
  193. package/src/router/router-registry.ts +24 -0
  194. package/src/router/segment-resolution/fresh.ts +748 -0
  195. package/src/router/segment-resolution/helpers.ts +268 -0
  196. package/src/router/segment-resolution/loader-cache.ts +199 -0
  197. package/src/router/segment-resolution/revalidation.ts +1379 -0
  198. package/src/router/segment-resolution/static-store.ts +67 -0
  199. package/src/router/segment-resolution.ts +21 -0
  200. package/src/router/segment-wrappers.ts +291 -0
  201. package/src/router/telemetry-otel.ts +299 -0
  202. package/src/router/telemetry.ts +300 -0
  203. package/src/router/timeout.ts +148 -0
  204. package/src/router/trie-matching.ts +239 -0
  205. package/src/router/types.ts +78 -3
  206. package/src/router.ts +740 -4252
  207. package/src/rsc/handler-context.ts +45 -0
  208. package/src/rsc/handler.ts +907 -797
  209. package/src/rsc/helpers.ts +140 -6
  210. package/src/rsc/index.ts +0 -20
  211. package/src/rsc/loader-fetch.ts +229 -0
  212. package/src/rsc/manifest-init.ts +90 -0
  213. package/src/rsc/nonce.ts +14 -0
  214. package/src/rsc/origin-guard.ts +141 -0
  215. package/src/rsc/progressive-enhancement.ts +391 -0
  216. package/src/rsc/response-error.ts +37 -0
  217. package/src/rsc/response-route-handler.ts +347 -0
  218. package/src/rsc/rsc-rendering.ts +246 -0
  219. package/src/rsc/runtime-warnings.ts +42 -0
  220. package/src/rsc/server-action.ts +356 -0
  221. package/src/rsc/ssr-setup.ts +128 -0
  222. package/src/rsc/types.ts +46 -11
  223. package/src/search-params.ts +230 -0
  224. package/src/segment-system.tsx +165 -17
  225. package/src/server/context.ts +315 -58
  226. package/src/server/cookie-store.ts +190 -0
  227. package/src/server/fetchable-loader-store.ts +37 -0
  228. package/src/server/handle-store.ts +113 -15
  229. package/src/server/loader-registry.ts +24 -64
  230. package/src/server/request-context.ts +607 -81
  231. package/src/server.ts +35 -130
  232. package/src/ssr/index.tsx +103 -30
  233. package/src/static-handler.ts +126 -0
  234. package/src/theme/ThemeProvider.tsx +21 -15
  235. package/src/theme/ThemeScript.tsx +5 -5
  236. package/src/theme/constants.ts +5 -2
  237. package/src/theme/index.ts +4 -14
  238. package/src/theme/theme-context.ts +4 -30
  239. package/src/theme/theme-script.ts +21 -18
  240. package/src/types/boundaries.ts +158 -0
  241. package/src/types/cache-types.ts +198 -0
  242. package/src/types/error-types.ts +192 -0
  243. package/src/types/global-namespace.ts +100 -0
  244. package/src/types/handler-context.ts +791 -0
  245. package/src/types/index.ts +88 -0
  246. package/src/types/loader-types.ts +210 -0
  247. package/src/types/route-config.ts +170 -0
  248. package/src/types/route-entry.ts +109 -0
  249. package/src/types/segments.ts +151 -0
  250. package/src/types.ts +1 -1623
  251. package/src/urls/include-helper.ts +197 -0
  252. package/src/urls/index.ts +53 -0
  253. package/src/urls/path-helper-types.ts +346 -0
  254. package/src/urls/path-helper.ts +364 -0
  255. package/src/urls/pattern-types.ts +107 -0
  256. package/src/urls/response-types.ts +116 -0
  257. package/src/urls/type-extraction.ts +372 -0
  258. package/src/urls/urls-function.ts +98 -0
  259. package/src/urls.ts +1 -802
  260. package/src/use-loader.tsx +161 -81
  261. package/src/vite/discovery/bundle-postprocess.ts +181 -0
  262. package/src/vite/discovery/discover-routers.ts +348 -0
  263. package/src/vite/discovery/prerender-collection.ts +439 -0
  264. package/src/vite/discovery/route-types-writer.ts +258 -0
  265. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  266. package/src/vite/discovery/state.ts +117 -0
  267. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  268. package/src/vite/index.ts +15 -1129
  269. package/src/vite/plugin-types.ts +103 -0
  270. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  271. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  272. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  273. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -53
  274. package/src/vite/plugins/expose-id-utils.ts +299 -0
  275. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  276. package/src/vite/plugins/expose-ids/handler-transform.ts +209 -0
  277. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  278. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  279. package/src/vite/plugins/expose-ids/types.ts +45 -0
  280. package/src/vite/plugins/expose-internal-ids.ts +786 -0
  281. package/src/vite/plugins/performance-tracks.ts +88 -0
  282. package/src/vite/plugins/refresh-cmd.ts +127 -0
  283. package/src/vite/plugins/use-cache-transform.ts +323 -0
  284. package/src/vite/plugins/version-injector.ts +83 -0
  285. package/src/vite/plugins/version-plugin.ts +266 -0
  286. package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
  287. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  288. package/src/vite/rango.ts +462 -0
  289. package/src/vite/router-discovery.ts +918 -0
  290. package/src/vite/utils/ast-handler-extract.ts +517 -0
  291. package/src/vite/utils/banner.ts +36 -0
  292. package/src/vite/utils/bundle-analysis.ts +137 -0
  293. package/src/vite/utils/manifest-utils.ts +70 -0
  294. package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
  295. package/src/vite/utils/prerender-utils.ts +207 -0
  296. package/src/vite/utils/shared-utils.ts +170 -0
  297. package/CLAUDE.md +0 -43
  298. package/src/browser/lru-cache.ts +0 -69
  299. package/src/browser/request-controller.ts +0 -164
  300. package/src/cache/memory-store.ts +0 -253
  301. package/src/href-context.ts +0 -33
  302. package/src/href.ts +0 -255
  303. package/src/server/route-manifest-cache.ts +0 -173
  304. package/src/vite/expose-handle-id.ts +0 -209
  305. package/src/vite/expose-loader-id.ts +0 -426
  306. package/src/vite/expose-location-state-id.ts +0 -177
  307. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -0,0 +1,502 @@
1
+ import { type ReactNode } from "react";
2
+ import { createHandleStore } from "../server/handle-store.js";
3
+ import { getRequestContext } from "../server/request-context.js";
4
+ import {
5
+ runWithRequestContext,
6
+ type RequestContext,
7
+ } from "../server/request-context.js";
8
+ import { contextGet, contextSet } from "../context-var.js";
9
+ import {
10
+ createPrerenderContext,
11
+ createStaticContext,
12
+ createReverseFunction,
13
+ } from "./handler-context.js";
14
+ import { isPrerenderPassthrough } from "../prerender.js";
15
+ import { isRouteRootScoped } from "../route-map-builder.js";
16
+ import { setupBuildUse } from "./loader-resolution.js";
17
+ import { loadManifest } from "./manifest.js";
18
+ import { traverseBack } from "./pattern-matching.js";
19
+ import type { RouterContext } from "./router-context.js";
20
+ import { runWithRouterContext } from "./router-context.js";
21
+ import type { EntryData, InterceptEntry } from "../server/context";
22
+ import type {
23
+ HandlerContext,
24
+ InternalHandlerContext,
25
+ ResolvedSegment,
26
+ } from "../types";
27
+ import type {
28
+ SerializedSegmentData,
29
+ SegmentHandleData,
30
+ } from "../cache/types.js";
31
+ import type { RouteMatchResult } from "./pattern-matching.js";
32
+
33
+ export interface PrerenderMatchDeps<TEnv = any> {
34
+ findMatch: (pathname: string) => RouteMatchResult<TEnv> | null;
35
+ buildRouterContext: () => RouterContext<TEnv>;
36
+ mergedRouteMap: Record<string, string>;
37
+ resolveAllSegments: (
38
+ entries: EntryData[],
39
+ routeKey: string,
40
+ params: Record<string, string>,
41
+ context: HandlerContext<any, TEnv>,
42
+ loaderPromises: Map<string, Promise<any>>,
43
+ options?: { skipLoaders?: boolean },
44
+ ) => Promise<ResolvedSegment[]>;
45
+ }
46
+
47
+ /**
48
+ * Build-time pre-render match. Resolves segments with a BuildContext
49
+ * (no request/env/headers/cookies), skipping middleware and loaders.
50
+ */
51
+ export async function matchForPrerender<TEnv = any>(
52
+ pathname: string,
53
+ params: Record<string, string>,
54
+ deps: PrerenderMatchDeps<TEnv>,
55
+ buildVars?: Record<string, any>,
56
+ isPassthroughRoute?: boolean,
57
+ buildEnv?: TEnv,
58
+ /** Dev-only: check getParams() for passthrough routes to skip unknown params. */
59
+ devMode?: boolean,
60
+ ): Promise<{
61
+ segments: SerializedSegmentData[];
62
+ handles: Record<string, SegmentHandleData>;
63
+ routeName: string;
64
+ params: Record<string, string>;
65
+ interceptSegments?: SerializedSegmentData[];
66
+ interceptHandles?: Record<string, SegmentHandleData>;
67
+ passthrough?: true;
68
+ } | null> {
69
+ // 1. Find the matching route entry
70
+ const matched = deps.findMatch(pathname);
71
+ if (!matched) return null;
72
+
73
+ // Use params from trie match if available, fall back to provided params
74
+ const matchedParams = matched.params ?? params;
75
+ const matchedPassthroughRoute = isPassthroughRoute ?? matched.pt === true;
76
+
77
+ // Build RouterContext for loadManifest/traverseBack
78
+ const routerCtx = deps.buildRouterContext();
79
+
80
+ return runWithRouterContext(routerCtx, async () => {
81
+ // 2. Load the manifest entry tree
82
+ const manifestEntry = await loadManifest(
83
+ matched.entry,
84
+ matched.routeKey,
85
+ pathname,
86
+ undefined,
87
+ false,
88
+ );
89
+
90
+ // 3. Build ancestor chain [root, ..., route]
91
+ const entries: EntryData[] = [];
92
+ for (const entry of traverseBack(manifestEntry)) {
93
+ entries.push(entry);
94
+ }
95
+
96
+ // 3b. Dev-mode passthrough shortcut: if the route is a Passthrough route
97
+ // and has getParams(), check if the matched params are in the known list.
98
+ // In production, only known params are pre-rendered; unknown params fall
99
+ // through to the live handler. Mirror that behavior in dev mode to avoid
100
+ // rendering unknown params with build: true.
101
+ // Vars collected from getParams() probe — merged into render context below.
102
+ let devProbeBuildVars: Record<string, any> | undefined;
103
+
104
+ if (devMode && matchedPassthroughRoute) {
105
+ const routeEntry = entries.find(
106
+ (
107
+ e,
108
+ ): e is EntryData & {
109
+ type: "route";
110
+ prerenderDef: { getParams: (ctx: any) => Promise<any[]> | any[] };
111
+ } =>
112
+ e.type === "route" &&
113
+ !!(e as any).isPassthrough &&
114
+ !!(e as any).prerenderDef?.getParams,
115
+ );
116
+ if (routeEntry) {
117
+ try {
118
+ const probeBuildVars: Record<string, any> = {};
119
+ const knownParamsList = await routeEntry.prerenderDef.getParams({
120
+ build: true as const,
121
+ dev: true,
122
+ set: ((keyOrVar: any, value: any) => {
123
+ contextSet(probeBuildVars, keyOrVar, value);
124
+ }) as any,
125
+ reverse: createReverseFunction(deps.mergedRouteMap),
126
+ get env() {
127
+ if (buildEnv !== undefined) return buildEnv;
128
+ throw new Error(
129
+ "[rsc-router] ctx.env is not available during dev-mode getParams(). " +
130
+ "Configure buildEnv in your rango() plugin options to enable build-time env access.",
131
+ );
132
+ },
133
+ });
134
+ // Compare only the keys returned by getParams — ignore mount params
135
+ // from include() prefixes that aren't part of the handler's params.
136
+ const isKnown = knownParamsList.some((known: Record<string, any>) => {
137
+ const knownKeys = Object.keys(known);
138
+ return knownKeys.every(
139
+ (k) => String(known[k]) === String(matchedParams[k]),
140
+ );
141
+ });
142
+ if (!isKnown) {
143
+ return {
144
+ segments: [],
145
+ handles: {},
146
+ routeName: matched.routeKey,
147
+ params: matchedParams,
148
+ passthrough: true as const,
149
+ };
150
+ }
151
+ // Preserve vars set by getParams() for the render context
152
+ if (
153
+ Object.keys(probeBuildVars).length > 0 ||
154
+ Object.getOwnPropertySymbols(probeBuildVars).length > 0
155
+ ) {
156
+ devProbeBuildVars = probeBuildVars;
157
+ }
158
+ } catch (err: any) {
159
+ // Mirror production semantics (prerender-collection.ts):
160
+ // Skip errors are intentional — treat as passthrough.
161
+ // All other errors propagate so dev surfaces them.
162
+ if (err?.name === "Skip") {
163
+ return {
164
+ segments: [],
165
+ handles: {},
166
+ routeName: matched.routeKey,
167
+ params: matchedParams,
168
+ passthrough: true as const,
169
+ };
170
+ }
171
+ throw err;
172
+ }
173
+ }
174
+ }
175
+
176
+ // 4. Create handle store for collecting handle data
177
+ const handleStore = createHandleStore();
178
+
179
+ // 5. Create a minimal request context with the handle store
180
+ // Shallow-copy getParams vars so each param set is independent.
181
+ // In dev mode, merge vars from the getParams() probe if the caller
182
+ // didn't provide buildVars (production passes them from expandPrerenderRoutes).
183
+ const effectiveBuildVars = buildVars ?? devProbeBuildVars;
184
+ const variables: Record<string, any> = effectiveBuildVars
185
+ ? { ...effectiveBuildVars }
186
+ : {};
187
+ const stubRes = new Response(null, { status: 200 });
188
+ const minimalRequestContext: RequestContext<TEnv> = {
189
+ env: buildEnv ?? ({} as TEnv),
190
+ request: new Request("http://prerender" + pathname),
191
+ url: new URL("http://prerender" + pathname),
192
+ originalUrl: new URL("http://prerender" + pathname),
193
+ pathname,
194
+ searchParams: new URLSearchParams(),
195
+ _variables: variables,
196
+ get: ((keyOrVar: any) => contextGet(variables, keyOrVar)) as any,
197
+ set: ((keyOrVar: any, value: any) => {
198
+ contextSet(variables, keyOrVar, value);
199
+ }) as any,
200
+ params: matchedParams,
201
+ res: stubRes,
202
+ cookie: () => undefined,
203
+ cookies: () => ({}),
204
+ setCookie: () => {},
205
+ deleteCookie: () => {},
206
+ header: () => {},
207
+ setStatus: () => {},
208
+ _setStatus: () => {},
209
+ use: (() => {
210
+ throw new Error("use() not available during pre-rendering");
211
+ }) as any,
212
+ method: "GET",
213
+ _handleStore: handleStore,
214
+ waitUntil: () => {},
215
+ onResponse: () => {},
216
+ _onResponseCallbacks: [],
217
+ setLocationState() {},
218
+ _locationState: undefined,
219
+ _renderBarrier: Promise.resolve(),
220
+ _resolveRenderBarrier: () => {},
221
+ _reportedErrors: new WeakSet<object>(),
222
+ reverse: createReverseFunction(
223
+ deps.mergedRouteMap,
224
+ matched.routeKey,
225
+ matchedParams,
226
+ matched.routeKey ? isRouteRootScoped(matched.routeKey) : undefined,
227
+ ),
228
+ };
229
+
230
+ return runWithRequestContext(minimalRequestContext, async () => {
231
+ // 6. Create prerender context with synthetic URL.
232
+ // Prerender handlers get params, pathname, url, searchParams, search,
233
+ // reverse, use(handle), and optionally env (when buildEnv is configured).
234
+ const buildCtx = createPrerenderContext<TEnv>(
235
+ matchedParams,
236
+ pathname,
237
+ deps.mergedRouteMap,
238
+ matched.routeKey,
239
+ variables,
240
+ matchedPassthroughRoute,
241
+ buildEnv,
242
+ devMode,
243
+ );
244
+
245
+ // 7. Wire use() for handles only (loaders throw)
246
+ setupBuildUse(buildCtx);
247
+
248
+ // 8. Resolve all segments with skipLoaders
249
+ const loaderPromises = new Map<string, Promise<any>>();
250
+ const allSegments = await deps.resolveAllSegments(
251
+ entries,
252
+ matched.routeKey,
253
+ matchedParams,
254
+ buildCtx,
255
+ loaderPromises,
256
+ { skipLoaders: true },
257
+ );
258
+
259
+ // 9. Detect passthrough sentinel: handler returned ctx.passthrough()
260
+ for (const seg of allSegments) {
261
+ if (isPrerenderPassthrough(seg.component)) {
262
+ return {
263
+ segments: [],
264
+ handles: {},
265
+ routeName: matched.routeKey,
266
+ params: matchedParams,
267
+ passthrough: true as const,
268
+ };
269
+ }
270
+ }
271
+
272
+ // 10. Filter out any loader segments (belt-and-suspenders)
273
+ const nonLoaderSegments = allSegments.filter((s) => s.type !== "loader");
274
+
275
+ // 11. Wait for handles to settle
276
+ handleStore.seal();
277
+ await handleStore.settled;
278
+
279
+ // 12. Serialize segments using the cache serializer
280
+ const { serializeSegments } = await import("../cache/segment-codec.js");
281
+ const serializedSegments = await serializeSegments(nonLoaderSegments);
282
+
283
+ // 13. Collect handle data per segment (skip segments with no handle data)
284
+ const handles: Record<string, SegmentHandleData> = {};
285
+ for (const seg of nonLoaderSegments) {
286
+ const segHandles = handleStore.getDataForSegment(seg.id);
287
+ if (Object.keys(segHandles).length > 0) {
288
+ handles[seg.id] = segHandles;
289
+ }
290
+ }
291
+
292
+ // Use the trie-level route key (e.g., "docs", "docs.article")
293
+ const routeName = matched.routeKey;
294
+
295
+ // 14. Resolve intercept segments for this route (if any ancestor defines
296
+ // an intercept targeting this route). At build time we skip when()
297
+ // evaluation -- we pre-render all intercepts unconditionally and let
298
+ // runtime matching decide which to serve.
299
+ let interceptSegments: SerializedSegmentData[] | undefined;
300
+ let interceptHandles: Record<string, SegmentHandleData> | undefined;
301
+
302
+ const foundIntercepts: {
303
+ intercept: InterceptEntry;
304
+ entry: EntryData;
305
+ }[] = [];
306
+ let current: EntryData | null = manifestEntry;
307
+ while (current) {
308
+ if (current.intercept && current.intercept.length > 0) {
309
+ for (const ic of current.intercept) {
310
+ if (ic.routeName === matched.routeKey) {
311
+ foundIntercepts.push({ intercept: ic, entry: current });
312
+ }
313
+ }
314
+ }
315
+ if (current.layout && current.layout.length > 0) {
316
+ for (const siblingLayout of current.layout) {
317
+ if (siblingLayout.intercept && siblingLayout.intercept.length > 0) {
318
+ for (const ic of siblingLayout.intercept) {
319
+ if (ic.routeName === matched.routeKey) {
320
+ foundIntercepts.push({
321
+ intercept: ic,
322
+ entry: siblingLayout,
323
+ });
324
+ }
325
+ }
326
+ }
327
+ }
328
+ }
329
+ current = current.parent;
330
+ }
331
+
332
+ if (foundIntercepts.length > 0) {
333
+ const interceptResolvedSegments: typeof nonLoaderSegments = [];
334
+
335
+ for (const { intercept, entry: parentEntry } of foundIntercepts) {
336
+ // Resolve handler
337
+ const handlerRaw =
338
+ typeof intercept.handler === "function"
339
+ ? intercept.handler(buildCtx)
340
+ : intercept.handler;
341
+ const handlerResolved =
342
+ handlerRaw instanceof Promise ? await handlerRaw : handlerRaw;
343
+ if (handlerResolved instanceof Response) {
344
+ // Handler returned a redirect/response -- skip this intercept
345
+ continue;
346
+ }
347
+ const component: ReactNode = handlerResolved;
348
+
349
+ // Resolve layout (if any)
350
+ let layoutElement: ReactNode | undefined;
351
+ if (intercept.layout) {
352
+ if (typeof intercept.layout === "function") {
353
+ const layoutResult = await intercept.layout(buildCtx);
354
+ if (layoutResult instanceof Response) continue;
355
+ layoutElement = layoutResult;
356
+ } else {
357
+ layoutElement = intercept.layout;
358
+ }
359
+ }
360
+
361
+ interceptResolvedSegments.push({
362
+ id: `${parentEntry.shortCode}.${intercept.slotName}`,
363
+ namespace: `intercept:${intercept.routeName}`,
364
+ type: "parallel" as const,
365
+ index: 0,
366
+ component,
367
+ loading: intercept.loading === false ? null : intercept.loading,
368
+ layout: layoutElement,
369
+ params: matchedParams,
370
+ slot: intercept.slotName,
371
+ belongsToRoute: true,
372
+ parallelName: `intercept:${intercept.routeName}.${intercept.slotName}`,
373
+ });
374
+ }
375
+
376
+ if (interceptResolvedSegments.length > 0) {
377
+ // Wait for handles again (intercept handlers may have called use())
378
+ await handleStore.settled;
379
+ interceptSegments = await serializeSegments(
380
+ interceptResolvedSegments,
381
+ );
382
+ interceptHandles = {};
383
+ for (const seg of interceptResolvedSegments) {
384
+ const segHandles = handleStore.getDataForSegment(seg.id);
385
+ if (Object.keys(segHandles).length > 0) {
386
+ interceptHandles[seg.id] = segHandles;
387
+ }
388
+ }
389
+ }
390
+ }
391
+
392
+ return {
393
+ segments: serializedSegments,
394
+ handles,
395
+ routeName,
396
+ params: matchedParams,
397
+ interceptSegments,
398
+ interceptHandles,
399
+ };
400
+ });
401
+ });
402
+ }
403
+
404
+ /**
405
+ * Render a single Static handler at build time.
406
+ * Creates a minimal BuildContext, calls the handler, and RSC-serializes
407
+ * the component. Returns the encoded Flight string (or null on failure).
408
+ * Used by the Vite plugin to collect static segment data at build time.
409
+ */
410
+ export async function renderStaticSegment<TEnv = any>(
411
+ handler: Function,
412
+ handlerId: string,
413
+ mergedRouteMap: Record<string, string>,
414
+ routeName?: string,
415
+ buildEnv?: TEnv,
416
+ devMode?: boolean,
417
+ ): Promise<{ encoded: string; handles: Record<string, unknown[]> } | null> {
418
+ const syntheticUrl = new URL("http://prerender/");
419
+ const syntheticRequest = new Request(syntheticUrl);
420
+
421
+ // Create a HandleStore to capture handle data pushed during rendering
422
+ const handleStore = createHandleStore();
423
+
424
+ // Minimal request context so setupBuildUse can find the HandleStore
425
+ const stubRes = new Response(null, { status: 200 });
426
+ const minimalRequestContext: RequestContext<TEnv> = {
427
+ env: buildEnv ?? ({} as TEnv),
428
+ request: syntheticRequest,
429
+ url: syntheticUrl,
430
+ originalUrl: syntheticUrl,
431
+ pathname: "/",
432
+ searchParams: syntheticUrl.searchParams,
433
+ _variables: {},
434
+ get: () => undefined as any,
435
+ set: () => {},
436
+ params: {},
437
+ res: stubRes,
438
+ cookie: () => undefined,
439
+ cookies: () => ({}),
440
+ setCookie: () => {},
441
+ deleteCookie: () => {},
442
+ header: () => {},
443
+ setStatus: () => {},
444
+ _setStatus: () => {},
445
+ use: (() => {
446
+ throw new Error("use() not available during static pre-rendering");
447
+ }) as any,
448
+ method: "GET",
449
+ _handleStore: handleStore,
450
+ waitUntil: () => {},
451
+ onResponse: () => {},
452
+ _onResponseCallbacks: [],
453
+ setLocationState() {},
454
+ _locationState: undefined,
455
+ _renderBarrier: Promise.resolve(),
456
+ _resolveRenderBarrier: () => {},
457
+ _reportedErrors: new WeakSet<object>(),
458
+ reverse: createReverseFunction(
459
+ mergedRouteMap,
460
+ routeName,
461
+ {},
462
+ routeName ? isRouteRootScoped(routeName) : undefined,
463
+ ),
464
+ };
465
+
466
+ return runWithRequestContext(minimalRequestContext, async () => {
467
+ // Static handlers get only reverse, use(handle), and optionally env.
468
+ const buildCtx = createStaticContext<TEnv>(
469
+ mergedRouteMap,
470
+ routeName,
471
+ buildEnv,
472
+ devMode,
473
+ );
474
+
475
+ // Set segment ID so handle pushes are keyed correctly
476
+ (buildCtx as InternalHandlerContext<any, TEnv>)._currentSegmentId =
477
+ handlerId;
478
+
479
+ setupBuildUse(buildCtx);
480
+
481
+ const raw = await handler(buildCtx);
482
+ const component = raw?.type ? raw : raw;
483
+
484
+ const segment: ResolvedSegment = {
485
+ id: handlerId,
486
+ namespace: handlerId,
487
+ type: "layout",
488
+ index: 0,
489
+ component,
490
+ params: {},
491
+ belongsToRoute: false,
492
+ };
493
+
494
+ const { serializeSegments } = await import("../cache/segment-codec.js");
495
+ const [serialized] = await serializeSegments([segment]);
496
+
497
+ // Collect handle data pushed during rendering
498
+ const handles = handleStore.getDataForSegment(handlerId);
499
+
500
+ return { encoded: serialized.encoded, handles };
501
+ });
502
+ }
@@ -0,0 +1,98 @@
1
+ import { negotiateRoute } from "./content-negotiation.js";
2
+ import { runWithRouterLogContext, withRouterLogScope } from "./logging.js";
3
+ import type { EntryData } from "../server/context";
4
+ import type { RouteMatchResult } from "./pattern-matching.js";
5
+ import type { MiddlewareFn } from "./middleware.js";
6
+ import { resolveRoute } from "./route-snapshot.js";
7
+
8
+ export interface PreviewMatchDeps<TEnv = any> {
9
+ findMatch: (pathname: string) => RouteMatchResult<TEnv> | null;
10
+ }
11
+
12
+ /**
13
+ * Preview match - returns route middleware without segment resolution.
14
+ * Also returns responseType and handler for response routes (non-RSC short-circuit).
15
+ */
16
+ export async function previewMatch<TEnv = any>(
17
+ request: Request,
18
+ _context: TEnv,
19
+ deps: PreviewMatchDeps<TEnv>,
20
+ ): Promise<{
21
+ routeMiddleware?: Array<{
22
+ handler: MiddlewareFn;
23
+ params: Record<string, string>;
24
+ }>;
25
+ responseType?: string;
26
+ handler?: Function;
27
+ params?: Record<string, string>;
28
+ negotiated?: boolean;
29
+ manifestEntry?: EntryData;
30
+ routeKey?: string;
31
+ } | null> {
32
+ return runWithRouterLogContext(
33
+ { request, transaction: "previewMatch" },
34
+ async () =>
35
+ withRouterLogScope("previewMatch", async () => {
36
+ const url = new URL(request.url);
37
+ const pathname = url.pathname;
38
+
39
+ // Route resolution via snapshot (lite mode: skip entries/cacheScope
40
+ // since previewMatch only needs matched, manifestEntry, routeMiddleware,
41
+ // and responseType)
42
+ const result = await resolveRoute<TEnv>(pathname, {
43
+ findMatch: deps.findMatch,
44
+ lite: true,
45
+ });
46
+
47
+ if (!result) {
48
+ return null;
49
+ }
50
+
51
+ // Skip redirect check - will be handled in full match
52
+ if (result.type === "redirect") {
53
+ return { routeMiddleware: undefined };
54
+ }
55
+
56
+ const snapshot = result.snapshot;
57
+ const { matched, manifestEntry, routeMiddleware, responseType } =
58
+ snapshot;
59
+
60
+ const negotiation = await negotiateRoute(request, pathname, snapshot);
61
+ if (negotiation) {
62
+ return {
63
+ routeMiddleware:
64
+ negotiation.routeMiddleware.length > 0
65
+ ? negotiation.routeMiddleware
66
+ : undefined,
67
+ responseType: negotiation.responseType,
68
+ handler: negotiation.handler,
69
+ params: matched.params,
70
+ negotiated: true,
71
+ manifestEntry: negotiation.manifestEntry,
72
+ routeKey: matched.routeKey,
73
+ };
74
+ }
75
+
76
+ // No negotiation or RSC won — return default route info
77
+ const hasVariants =
78
+ matched.negotiateVariants && matched.negotiateVariants.length > 0;
79
+ return {
80
+ routeMiddleware:
81
+ routeMiddleware.length > 0 ? routeMiddleware : undefined,
82
+ params: matched.params,
83
+ routeKey: matched.routeKey,
84
+ ...(responseType
85
+ ? {
86
+ responseType,
87
+ handler:
88
+ manifestEntry.type === "route"
89
+ ? manifestEntry.handler
90
+ : undefined,
91
+ manifestEntry,
92
+ }
93
+ : {}),
94
+ ...(hasVariants ? { negotiated: true } : {}),
95
+ };
96
+ }),
97
+ );
98
+ }