@rangojs/router 0.0.0-experimental.002d056c

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 +9 -0
  2. package/README.md +899 -0
  3. package/dist/bin/rango.js +1606 -0
  4. package/dist/vite/index.js +5153 -0
  5. package/package.json +177 -0
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +262 -0
  8. package/skills/caching/SKILL.md +253 -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 +638 -0
  43. package/src/browser/navigation-client.ts +261 -0
  44. package/src/browser/navigation-store.ts +806 -0
  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 +582 -0
  48. package/src/browser/prefetch/cache.ts +206 -0
  49. package/src/browser/prefetch/fetch.ts +145 -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 +128 -0
  53. package/src/browser/rango-state.ts +112 -0
  54. package/src/browser/react/Link.tsx +368 -0
  55. package/src/browser/react/NavigationProvider.tsx +413 -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 +464 -0
  79. package/src/browser/scroll-restoration.ts +397 -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 +547 -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 +479 -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 +982 -0
  105. package/src/cache/cf/index.ts +29 -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 +44 -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 +281 -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 +160 -0
  170. package/src/router/handler-context.ts +451 -0
  171. package/src/router/intercept-resolution.ts +397 -0
  172. package/src/router/lazy-includes.ts +236 -0
  173. package/src/router/loader-resolution.ts +420 -0
  174. package/src/router/logging.ts +251 -0
  175. package/src/router/manifest.ts +269 -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 +193 -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 +749 -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 +320 -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 +1242 -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 +291 -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 +1006 -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 +237 -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 +920 -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 +109 -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 +108 -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 +48 -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 +363 -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 +266 -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 +445 -0
  298. package/src/vite/router-discovery.ts +777 -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,777 @@
1
+ /**
2
+ * Router Discovery Plugin
3
+ *
4
+ * Vite plugin that discovers router instances at dev/build time via the RSC
5
+ * environment. Delegates to extracted modules for discovery, route types
6
+ * generation, virtual module codegen, and bundle post-processing.
7
+ */
8
+
9
+ import type { Plugin } from "vite";
10
+ import { createServer as createViteServer } from "vite";
11
+ import { resolve } from "node:path";
12
+ import { readFileSync } from "node:fs";
13
+ import {
14
+ formatNestedRouterConflictError,
15
+ findNestedRouterConflict,
16
+ findRouterFiles,
17
+ } from "../build/generate-route-types.js";
18
+ import { createVersionPlugin } from "./plugins/version-plugin.js";
19
+ import { createVirtualStubPlugin } from "./plugins/virtual-stub-plugin.js";
20
+ import {
21
+ exposeInternalIds,
22
+ exposeRouterId,
23
+ } from "./plugins/expose-internal-ids.js";
24
+ import { hashClientRefs } from "./plugins/client-ref-hashing.js";
25
+ import { extractHandlerExportsFromChunk } from "./utils/bundle-analysis.js";
26
+ import {
27
+ createDiscoveryState,
28
+ VIRTUAL_ROUTES_MANIFEST_ID,
29
+ type DiscoveryState,
30
+ type PluginOptions,
31
+ } from "./discovery/state.js";
32
+ import { consumeSelfGenWrite } from "./discovery/self-gen-tracking.js";
33
+ import { discoverRouters } from "./discovery/discover-routers.js";
34
+ import {
35
+ writeCombinedRouteTypesWithTracking,
36
+ writeRouteTypesFiles,
37
+ supplementGenFilesWithRuntimeRoutes,
38
+ } from "./discovery/route-types-writer.js";
39
+ import {
40
+ generateRoutesManifestModule,
41
+ generatePerRouterModule,
42
+ } from "./discovery/virtual-module-codegen.js";
43
+ import { postprocessBundle } from "./discovery/bundle-postprocess.js";
44
+ import { resetStagedBuildAssets } from "./utils/prerender-utils.js";
45
+
46
+ export { VIRTUAL_ROUTES_MANIFEST_ID };
47
+
48
+ // ============================================================================
49
+ // Temp Server Factory
50
+ // ============================================================================
51
+
52
+ /**
53
+ * Create a minimal Vite server for router discovery.
54
+ *
55
+ * Both dev-mode prerender and build-mode discovery need a temp RSC server
56
+ * to import user router files via module runner. This factory centralizes
57
+ * the shared config and the mode-specific differences:
58
+ * - Dev: path-based IDs (no forceBuild), separate cacheDir
59
+ * - Build: hashed IDs (forceBuild), hashClientRefs for production bundles
60
+ *
61
+ * Returns the ViteDevServer instance. Callers access .environments.rsc as needed.
62
+ */
63
+ async function createTempRscServer(
64
+ state: DiscoveryState,
65
+ options: { forceBuild?: boolean; cacheDir?: string } = {},
66
+ ) {
67
+ const { default: rsc } = await import("@vitejs/plugin-rsc");
68
+ return createViteServer({
69
+ root: state.projectRoot,
70
+ configFile: false,
71
+ server: { middlewareMode: true },
72
+ appType: "custom",
73
+ logLevel: "silent",
74
+ resolve: { alias: state.userResolveAlias },
75
+ esbuild: { jsx: "automatic", jsxImportSource: "react" },
76
+ ...(options.cacheDir && { cacheDir: options.cacheDir }),
77
+ plugins: [
78
+ rsc({
79
+ entries: {
80
+ client: "virtual:entry-client",
81
+ ssr: "virtual:entry-ssr",
82
+ rsc: state.resolvedEntryPath!,
83
+ },
84
+ }),
85
+ // hashClientRefs only in build mode — production bundles need hashed refs
86
+ ...(options.forceBuild ? [hashClientRefs(state.projectRoot)] : []),
87
+ createVersionPlugin(),
88
+ createVirtualStubPlugin(),
89
+ // Dev prerender must use dev-mode IDs (path-based) to match the workerd
90
+ // runtime. forceBuild produces hashed IDs for production bundle consistency.
91
+ exposeInternalIds(options.forceBuild ? { forceBuild: true } : undefined),
92
+ exposeRouterId(),
93
+ ],
94
+ });
95
+ }
96
+
97
+ /**
98
+ * Plugin that discovers router instances at dev/build time via the RSC environment.
99
+ *
100
+ * Uses `server.environments.rsc.runner.import()` to load the user's router file
101
+ * with full TS/TSX compilation. This triggers `createRouter()` which populates
102
+ * the `RouterRegistry`. The plugin then generates manifests for each router.
103
+ *
104
+ * In dev mode, this runs in `configureServer` (post-middleware setup).
105
+ * In build mode, this will run in `buildStart` (future).
106
+ *
107
+ * @internal
108
+ */
109
+ export function createRouterDiscoveryPlugin(
110
+ entryPath: string | undefined,
111
+ opts?: PluginOptions,
112
+ ): Plugin {
113
+ const s = createDiscoveryState(entryPath, opts);
114
+
115
+ return {
116
+ name: "@rangojs/router:discovery",
117
+
118
+ config() {
119
+ const config: any = {
120
+ define: {
121
+ __RANGO_DEBUG__: JSON.stringify(!!process.env.INTERNAL_RANGO_DEBUG),
122
+ },
123
+ };
124
+ if (opts?.enableBuildPrerender) {
125
+ config.environments = {
126
+ rsc: {
127
+ build: {
128
+ rollupOptions: {
129
+ output: {
130
+ manualChunks(id: string) {
131
+ if (s.resolvedPrerenderModules?.has(id)) {
132
+ return "__prerender-handlers";
133
+ }
134
+ if (s.resolvedStaticModules?.has(id)) {
135
+ return "__static-handlers";
136
+ }
137
+ },
138
+ },
139
+ },
140
+ },
141
+ },
142
+ };
143
+ }
144
+ return config;
145
+ },
146
+
147
+ configResolved(config) {
148
+ s.projectRoot = config.root;
149
+ s.isBuildMode = config.command === "build";
150
+ // Capture user's resolve aliases for the temp server
151
+ s.userResolveAlias = config.resolve.alias;
152
+ // Node preset: pick up auto-discovered router path from the config() hook.
153
+ // The auto-discover plugin runs in config() using Vite's resolved root,
154
+ // populating the mutable ref before configResolved fires.
155
+ if (!s.resolvedEntryPath && opts?.routerPathRef?.path) {
156
+ s.resolvedEntryPath = opts.routerPathRef.path;
157
+ }
158
+ // Cloudflare preset: read entry from resolved environment config.
159
+ // The @cloudflare/vite-plugin reads wrangler config (toml/json/jsonc)
160
+ // and sets optimizeDeps.entries on the RSC environment.
161
+ if (!s.resolvedEntryPath) {
162
+ const rscEnvConfig = (config.environments as any)?.["rsc"];
163
+ const entries = rscEnvConfig?.optimizeDeps?.entries;
164
+ if (typeof entries === "string") {
165
+ s.resolvedEntryPath = entries;
166
+ } else if (Array.isArray(entries) && entries.length > 0) {
167
+ s.resolvedEntryPath = entries[0];
168
+ }
169
+ }
170
+ // Generate combined named-routes.gen.ts from static source parsing.
171
+ // Runs before the dev server starts so the gen file exists immediately for IDE.
172
+ // In build mode, the runtime discovery in buildStart produces the definitive
173
+ // named-routes.gen.ts (including dynamically generated routes).
174
+ // preserveIfLarger prevents overwriting a previously generated complete
175
+ // file with a partial one.
176
+ if (opts?.staticRouteTypesGeneration !== false) {
177
+ s.cachedRouterFiles = findRouterFiles(s.projectRoot, s.scanFilter);
178
+ writeCombinedRouteTypesWithTracking(s, { preserveIfLarger: true });
179
+ }
180
+ // Resolve prerenderHandlerModules and staticHandlerModules from the consolidated IDs plugin's API.
181
+ if (opts?.enableBuildPrerender) {
182
+ const idsPlugin = config.plugins.find(
183
+ (p: any) => p.name === "@rangojs/router:expose-internal-ids",
184
+ );
185
+ s.resolvedPrerenderModules = (
186
+ idsPlugin?.api as any
187
+ )?.prerenderHandlerModules;
188
+ s.resolvedStaticModules = (idsPlugin?.api as any)?.staticHandlerModules;
189
+ }
190
+ },
191
+
192
+ // Dev mode: discover routers and populate manifest in memory.
193
+ // Skipped in build mode (buildStart handles it).
194
+ configureServer(server) {
195
+ if (s.isBuildMode) return;
196
+ // Skip if this is a temp server created by buildStart
197
+ if ((globalThis as any).__rscRouterDiscoveryActive) return;
198
+ s.devServer = server;
199
+
200
+ // Discovery promise that the handler can await if requests arrive
201
+ // before discovery completes
202
+ let resolveDiscovery: () => void;
203
+ const discoveryPromise = new Promise<void>((resolve) => {
204
+ resolveDiscovery = resolve;
205
+ });
206
+
207
+ // Compute dev server origin from resolved URLs (preferred) or config port (fallback).
208
+ // Called after discovery (or in the load hook) when the server may be listening.
209
+ const getDevServerOrigin = () =>
210
+ server.resolvedUrls?.local?.[0]?.replace(/\/$/, "") ||
211
+ `http://localhost:${server.config.server.port || 5173}`;
212
+
213
+ // Shared temp server for Cloudflare dev (no module runner in workerd).
214
+ // Used by both discover() (route type generation) and the prerender
215
+ // middleware (on-demand prerender evaluation). Created lazily, closed on
216
+ // server shutdown.
217
+ let prerenderTempServer: any = null;
218
+ let prerenderNodeRegistry: Map<string, any> | null = null;
219
+
220
+ // Clean up the temporary server when the dev server shuts down
221
+ server.httpServer?.on("close", () => {
222
+ if (prerenderTempServer) {
223
+ prerenderTempServer.close().catch(() => {});
224
+ prerenderTempServer = null;
225
+ }
226
+ });
227
+
228
+ async function getOrCreateTempServer(): Promise<any | null> {
229
+ if (prerenderNodeRegistry) {
230
+ return (prerenderTempServer.environments as any)?.rsc ?? null;
231
+ }
232
+ try {
233
+ prerenderTempServer = await createTempRscServer(s, {
234
+ cacheDir: "node_modules/.vite_prerender",
235
+ });
236
+
237
+ const tempRscEnv = (prerenderTempServer.environments as any)?.rsc;
238
+ if (tempRscEnv?.runner) {
239
+ await tempRscEnv.runner.import(s.resolvedEntryPath!);
240
+ const serverMod = await tempRscEnv.runner.import(
241
+ "@rangojs/router/server",
242
+ );
243
+ prerenderNodeRegistry = serverMod.RouterRegistry;
244
+ return tempRscEnv;
245
+ }
246
+ } catch (err: any) {
247
+ console.warn(
248
+ `[rsc-router] Failed to create temp runner: ${err.message}`,
249
+ );
250
+ }
251
+ return null;
252
+ }
253
+
254
+ const discover = async () => {
255
+ const rscEnv = (server.environments as any)?.rsc;
256
+ if (!rscEnv?.runner) {
257
+ // Cloudflare dev: no module runner available (workerd-based RSC env).
258
+ // Set devServerOrigin so the virtual module can inject __PRERENDER_DEV_URL
259
+ // for on-demand prerender via the /__rsc_prerender endpoint.
260
+ s.devServerOrigin = getDevServerOrigin();
261
+
262
+ // Create a temp Node.js server to run runtime discovery and generate
263
+ // named route types (static parser can't resolve factory calls).
264
+ try {
265
+ const tempRscEnv = await getOrCreateTempServer();
266
+ if (tempRscEnv) {
267
+ await discoverRouters(s, tempRscEnv);
268
+ writeRouteTypesFiles(s);
269
+ }
270
+ } catch (err: any) {
271
+ console.warn(
272
+ `[rsc-router] Cloudflare dev discovery failed: ${err.message}\n${err.stack}`,
273
+ );
274
+ }
275
+
276
+ resolveDiscovery!();
277
+ return;
278
+ }
279
+
280
+ try {
281
+ // Set the readiness gate BEFORE discovery so early requests
282
+ // block until manifest is populated
283
+ const serverMod = await rscEnv.runner.import(
284
+ "@rangojs/router/server",
285
+ );
286
+ if (serverMod?.setManifestReadyPromise) {
287
+ serverMod.setManifestReadyPromise(discoveryPromise);
288
+ }
289
+
290
+ await discoverRouters(s, rscEnv);
291
+
292
+ // Store server origin for dev prerender endpoint (virtual module injection)
293
+ s.devServerOrigin = getDevServerOrigin();
294
+
295
+ // Update named-routes.gen.ts from runtime discovery.
296
+ // The runtime manifest is the source of truth: it evaluates dynamic
297
+ // routes (e.g. Array.from loops) that the static parser cannot see.
298
+ // writeRouteTypesFiles() only writes when content changes, so this
299
+ // won't cause unnecessary HMR triggers.
300
+ writeRouteTypesFiles(s);
301
+
302
+ // Populate the route map and per-router data in the RSC env
303
+ await propagateDiscoveryState(rscEnv);
304
+ } catch (err: any) {
305
+ console.warn(
306
+ `[rsc-router] Router discovery failed: ${err.message}\n${err.stack}`,
307
+ );
308
+ } finally {
309
+ resolveDiscovery!();
310
+ }
311
+ };
312
+
313
+ // Schedule after all plugins have finished configureServer.
314
+ // Store the promise so the virtual module's load hook can await it.
315
+ s.discoveryDone = new Promise<void>((resolve) => {
316
+ setTimeout(() => discover().then(resolve, resolve), 0);
317
+ });
318
+
319
+ // Dev-mode on-demand prerender endpoint.
320
+ // When workerd hits a prerender route, it fetches this endpoint instead of
321
+ // trying to run node:fs-dependent handlers in the Cloudflare environment.
322
+ //
323
+ // Node.js preset: uses the main server's RSC environment directly (router
324
+ // instances are already discovered and have matchForPrerender).
325
+ // Cloudflare preset: lazily creates a Node.js temp server because the main
326
+ // RSC environment uses workerd where node:fs can't access the host filesystem.
327
+
328
+ // Registry from the main server's RSC environment (populated by discoverRouters)
329
+ let mainRegistry: Map<string, any> | null = null;
330
+
331
+ // Push discovery state (manifest, trie, precomputed entries) to the
332
+ // server module so runtime request handling uses the current routes.
333
+ // Shared by initial discovery and HMR-triggered re-discovery.
334
+ const propagateDiscoveryState = async (rscEnv: any) => {
335
+ const serverMod = await rscEnv.runner.import("@rangojs/router/server");
336
+ if (!serverMod) return;
337
+ // Clear stale per-router and global route data before repopulating.
338
+ // Without this, removed routers/routes survive in the per-router maps
339
+ // and shrunk precomputed entries or tries are never purged.
340
+ if (serverMod.clearAllRouterData) {
341
+ serverMod.clearAllRouterData();
342
+ }
343
+ mainRegistry = serverMod.RouterRegistry ?? null;
344
+ if (s.mergedRouteManifest && serverMod.setCachedManifest) {
345
+ serverMod.setCachedManifest(s.mergedRouteManifest);
346
+ }
347
+ if (
348
+ s.mergedPrecomputedEntries &&
349
+ s.mergedPrecomputedEntries.length > 0 &&
350
+ serverMod.setPrecomputedEntries
351
+ ) {
352
+ serverMod.setPrecomputedEntries(s.mergedPrecomputedEntries);
353
+ }
354
+ if (s.mergedRouteTrie && serverMod.setRouteTrie) {
355
+ serverMod.setRouteTrie(s.mergedRouteTrie);
356
+ }
357
+ if (serverMod.setRouterManifest) {
358
+ for (const [routerId, manifest] of s.perRouterManifestDataMap) {
359
+ serverMod.setRouterManifest(routerId, manifest);
360
+ }
361
+ }
362
+ if (serverMod.setRouterTrie) {
363
+ for (const [routerId, trie] of s.perRouterTrieMap) {
364
+ serverMod.setRouterTrie(routerId, trie);
365
+ }
366
+ }
367
+ if (serverMod.setRouterPrecomputedEntries) {
368
+ for (const [routerId, entries] of s.perRouterPrecomputedMap) {
369
+ serverMod.setRouterPrecomputedEntries(routerId, entries);
370
+ }
371
+ }
372
+ };
373
+
374
+ server.middlewares.use("/__rsc_prerender", async (req: any, res: any) => {
375
+ if (s.discoveryDone) await s.discoveryDone;
376
+
377
+ const url = new URL(req.url || "/", "http://localhost");
378
+ const pathname = url.searchParams.get("pathname");
379
+ if (!pathname) {
380
+ res.statusCode = 400;
381
+ res.end("Missing pathname");
382
+ return;
383
+ }
384
+
385
+ // Prefer the main server's registry (Node.js preset: module runner available).
386
+ // Fall back to a temp server for Cloudflare where the main RSC env uses workerd.
387
+ let registry = mainRegistry;
388
+
389
+ if (!registry) {
390
+ // No main registry: the RSC env has no module runner (Cloudflare dev).
391
+ // Lazily create a Node.js temp server for prerender evaluation.
392
+ if (!prerenderNodeRegistry) {
393
+ await getOrCreateTempServer();
394
+ }
395
+ registry = prerenderNodeRegistry;
396
+ }
397
+
398
+ if (!registry || registry.size === 0) {
399
+ res.statusCode = 503;
400
+ res.end("Prerender runner not available");
401
+ return;
402
+ }
403
+
404
+ const wantIntercept = url.searchParams.get("intercept") === "1";
405
+ const wantRouteName = url.searchParams.get("routeName");
406
+ const wantPassthrough = url.searchParams.get("passthrough") === "1";
407
+
408
+ for (const [, routerInstance] of registry) {
409
+ if (!routerInstance.matchForPrerender) continue;
410
+ try {
411
+ const result = await routerInstance.matchForPrerender(
412
+ pathname,
413
+ {},
414
+ undefined,
415
+ wantPassthrough,
416
+ );
417
+ if (!result) continue;
418
+ if (result.passthrough) continue;
419
+ // When routeName is specified, only accept a match for that route.
420
+ // This prevents returning the wrong entry when multiple routers
421
+ // have prerenderable routes sharing the same pathname.
422
+ if (wantRouteName && result.routeName !== wantRouteName) continue;
423
+ res.setHeader("content-type", "application/json");
424
+ let payload: Record<string, unknown>;
425
+ if (wantIntercept && result.interceptSegments?.length) {
426
+ payload = {
427
+ segments: [...result.segments, ...result.interceptSegments],
428
+ handles: {
429
+ ...result.handles,
430
+ ...(result.interceptHandles || {}),
431
+ },
432
+ };
433
+ } else {
434
+ payload = { segments: result.segments, handles: result.handles };
435
+ }
436
+ res.end(JSON.stringify(payload));
437
+ return;
438
+ } catch (err: any) {
439
+ console.warn(
440
+ `[rsc-router] Dev prerender failed for ${pathname}: ${err.message}`,
441
+ );
442
+ }
443
+ }
444
+
445
+ res.statusCode = 404;
446
+ res.end("No prerender match");
447
+ });
448
+
449
+ // Watch url module and router files for changes and regenerate named-routes.gen.ts.
450
+ // Process files containing urls( or createRouter( to update the combined route map.
451
+ if (opts?.staticRouteTypesGeneration !== false) {
452
+ const isGeneratedRouteFile = (filePath: string): boolean =>
453
+ filePath.endsWith(".gen.ts") &&
454
+ (filePath.includes("named-routes.gen.ts") ||
455
+ filePath.includes("urls.gen.ts"));
456
+
457
+ const regenerateGeneratedRouteFiles = () => {
458
+ if (s.perRouterManifests.length > 0) {
459
+ writeRouteTypesFiles(s);
460
+ } else {
461
+ writeCombinedRouteTypesWithTracking(s);
462
+ }
463
+ };
464
+
465
+ const maybeHandleGeneratedRouteFileMutation = (
466
+ filePath: string,
467
+ ): boolean => {
468
+ if (!isGeneratedRouteFile(filePath)) return false;
469
+ if (consumeSelfGenWrite(s, filePath)) return true;
470
+ // In Cloudflare dev (no module runner), perRouterManifests is never
471
+ // refreshed after HMR so regenerateGeneratedRouteFiles() would use
472
+ // stale data and revert user edits. Source files own route state;
473
+ // gen files are derived output. Skip regeneration and let the next
474
+ // source-file change rebuild them from the static parser.
475
+ const hasRunner = !!(server.environments as any)?.rsc?.runner;
476
+ if (!hasRunner) return true;
477
+ regenerateGeneratedRouteFiles();
478
+ return true;
479
+ };
480
+
481
+ // Debounce timer for batching rapid route-file changes (e.g. afterEach
482
+ // restoring two files in quick succession). The cheap checks (extension,
483
+ // scanFilter, content sniff) run synchronously to gate non-route files;
484
+ // only the expensive regeneration is debounced.
485
+ let routeChangeTimer: ReturnType<typeof setTimeout> | undefined;
486
+
487
+ // Re-run runtime discovery so factory-generated routes that the
488
+ // static parser cannot see are refreshed after source changes.
489
+ let runtimeRediscoveryInProgress = false;
490
+ const refreshRuntimeDiscovery = async () => {
491
+ const rscEnv = (server.environments as any)?.rsc;
492
+ if (!rscEnv?.runner || runtimeRediscoveryInProgress) return;
493
+ runtimeRediscoveryInProgress = true;
494
+ try {
495
+ await discoverRouters(s, rscEnv);
496
+ writeRouteTypesFiles(s);
497
+ await propagateDiscoveryState(rscEnv);
498
+ } catch (err: any) {
499
+ console.warn(
500
+ `[rsc-router] Runtime re-discovery failed: ${err.message}`,
501
+ );
502
+ } finally {
503
+ runtimeRediscoveryInProgress = false;
504
+ }
505
+ };
506
+
507
+ const scheduleRouteRegeneration = () => {
508
+ clearTimeout(routeChangeTimer);
509
+ routeChangeTimer = setTimeout(() => {
510
+ routeChangeTimer = undefined;
511
+ try {
512
+ writeCombinedRouteTypesWithTracking(s);
513
+ if (s.perRouterManifests.length > 0) {
514
+ supplementGenFilesWithRuntimeRoutes(s);
515
+ }
516
+ } catch (err: any) {
517
+ console.error(
518
+ `[rsc-router] Route regeneration error: ${err.message}`,
519
+ );
520
+ }
521
+ // Async: re-run runtime discovery to refresh factory-generated
522
+ // routes that the static parser cannot resolve.
523
+ if (s.perRouterManifests.length > 0) {
524
+ refreshRuntimeDiscovery().catch((err: any) => {
525
+ console.warn(
526
+ `[rsc-router] Runtime re-discovery error: ${err.message}`,
527
+ );
528
+ });
529
+ }
530
+ }, 100);
531
+ };
532
+
533
+ const handleRouteFileChange = (filePath: string) => {
534
+ if (maybeHandleGeneratedRouteFileMutation(filePath)) return;
535
+ if (
536
+ !filePath.endsWith(".ts") &&
537
+ !filePath.endsWith(".tsx") &&
538
+ !filePath.endsWith(".js") &&
539
+ !filePath.endsWith(".jsx")
540
+ )
541
+ return;
542
+ // Apply scan filter as early-exit before reading file
543
+ if (s.scanFilter && !s.scanFilter(filePath)) return;
544
+ try {
545
+ const source = readFileSync(filePath, "utf-8");
546
+ const trimmed = source.trimStart();
547
+ if (
548
+ trimmed.startsWith('"use client"') ||
549
+ trimmed.startsWith("'use client'")
550
+ )
551
+ return;
552
+ const hasUrls = source.includes("urls(");
553
+ const hasCreateRouter = /\bcreateRouter\s*[<(]/.test(source);
554
+ if (!hasUrls && !hasCreateRouter) return;
555
+ // Invalidate cache when a router file changes (new router added/removed)
556
+ if (hasCreateRouter) {
557
+ const nestedRouterConflict = findNestedRouterConflict([
558
+ ...(s.cachedRouterFiles ?? []),
559
+ resolve(filePath),
560
+ ]);
561
+ if (nestedRouterConflict) {
562
+ server.config.logger.error(
563
+ formatNestedRouterConflictError(nestedRouterConflict),
564
+ );
565
+ return;
566
+ }
567
+ s.cachedRouterFiles = undefined;
568
+ }
569
+ scheduleRouteRegeneration();
570
+ } catch {
571
+ // Ignore read errors for deleted/moved files
572
+ }
573
+ };
574
+
575
+ // Handle both "add" and "change" events: editors with atomic saves
576
+ // (unlink + rename) emit "add" instead of "change", and chokidar's
577
+ // polling mode on CI Linux can also emit "add" for overwrites.
578
+ server.watcher.on("add", handleRouteFileChange);
579
+ server.watcher.on("change", handleRouteFileChange);
580
+
581
+ // Regenerate gen files when they are deleted (e.g. manual cleanup).
582
+ // Same no-runner guard as change/add: stale perRouterManifests would
583
+ // reintroduce reverted content.
584
+ server.watcher.on("unlink", (filePath) => {
585
+ if (!isGeneratedRouteFile(filePath)) return;
586
+ const hasRunner = !!(server.environments as any)?.rsc?.runner;
587
+ if (!hasRunner) return;
588
+ regenerateGeneratedRouteFiles();
589
+ });
590
+ }
591
+ },
592
+
593
+ // Build mode: create a temporary Vite dev server to access the RSC
594
+ // environment's module runner, then discover routers and generate manifests.
595
+ // The manifest data is stored for the virtual module's load hook.
596
+ async buildStart() {
597
+ if (!s.isBuildMode) return;
598
+ // Only run once across environment builds
599
+ if (s.mergedRouteManifest !== null) return;
600
+ resetStagedBuildAssets(s.projectRoot);
601
+ s.prerenderManifestEntries = null;
602
+ s.staticManifestEntries = null;
603
+
604
+ let tempServer: any = null;
605
+ // Signal to user-space code (e.g. reverse.ts) that build-time discovery
606
+ // is active. Uses globalThis because the temp server's module runner
607
+ // creates a separate module context — there is no shared import path
608
+ // between the vite plugin and user code loaded via runner.import().
609
+ (globalThis as any).__rscRouterDiscoveryActive = true;
610
+ try {
611
+ tempServer = await createTempRscServer(s, { forceBuild: true });
612
+
613
+ const rscEnv = (tempServer.environments as any)?.rsc;
614
+ if (!rscEnv?.runner) {
615
+ console.warn(
616
+ "[rsc-router] RSC environment runner not available during build, skipping manifest generation",
617
+ );
618
+ return;
619
+ }
620
+
621
+ // Point resolvedStaticModules at the temp server's expose-internal-ids
622
+ // plugin so that discoverRouters() can access the static handler module
623
+ // map after the temp server's transforms populate it.
624
+ const tempIdsPlugin = (tempServer as any).config?.plugins?.find(
625
+ (p: any) => p.name === "@rangojs/router:expose-internal-ids",
626
+ );
627
+ if (tempIdsPlugin?.api?.staticHandlerModules) {
628
+ s.resolvedStaticModules = tempIdsPlugin.api.staticHandlerModules;
629
+ }
630
+
631
+ await discoverRouters(s, rscEnv);
632
+ // Update named-routes.gen.ts from runtime discovery.
633
+ // The runtime manifest includes dynamically generated routes
634
+ // that the static parser cannot extract from source code.
635
+ writeRouteTypesFiles(s);
636
+ } catch (err: any) {
637
+ // Extract the user source file from the stack trace (skip internal frames)
638
+ const sourceFile = err.stack
639
+ ?.split("\n")
640
+ .find(
641
+ (line: string) =>
642
+ line.includes(s.projectRoot) && !line.includes("node_modules"),
643
+ )
644
+ ?.match(/\(([^)]+)\)/)?.[1];
645
+ // Extract the route name from "Unknown route: <name>" errors
646
+ const routeName = err.message?.match(/Unknown route: (.+)/)?.[1];
647
+ const details = [
648
+ routeName ? ` Route name: ${routeName}` : null,
649
+ sourceFile ? ` File: ${sourceFile}` : null,
650
+ err.stack ? ` Stack:\n${err.stack}` : null,
651
+ ]
652
+ .filter(Boolean)
653
+ .join("\n");
654
+ throw new Error(
655
+ `[rsc-router] Build-time router discovery failed:\n${details}`,
656
+ );
657
+ } finally {
658
+ delete (globalThis as any).__rscRouterDiscoveryActive;
659
+ if (tempServer) {
660
+ await tempServer.close();
661
+ }
662
+ }
663
+ },
664
+
665
+ // Virtual module: provides the pre-generated route manifest as a JS module
666
+ // that calls setCachedManifest() at import time.
667
+ resolveId(id) {
668
+ if (id === VIRTUAL_ROUTES_MANIFEST_ID) {
669
+ return "\0" + VIRTUAL_ROUTES_MANIFEST_ID;
670
+ }
671
+ // Per-router virtual modules: virtual:rsc-router/routes-manifest/<routerId>
672
+ if (id.startsWith(VIRTUAL_ROUTES_MANIFEST_ID + "/")) {
673
+ return "\0" + id;
674
+ }
675
+ // virtual:rsc-router/prerender-paths removed: prerender data is served through the worker
676
+ return null;
677
+ },
678
+
679
+ async load(id) {
680
+ if (id === "\0" + VIRTUAL_ROUTES_MANIFEST_ID) {
681
+ // In dev mode, wait for discovery to complete before emitting module content.
682
+ // This is critical for Cloudflare dev where the worker runs in a separate
683
+ // Miniflare process and can only receive manifest data via the virtual module.
684
+ if (s.discoveryDone) {
685
+ await s.discoveryDone;
686
+ }
687
+ return generateRoutesManifestModule(s);
688
+ }
689
+ // Per-router virtual modules: pure data exports (no side effects).
690
+ // ensureRouterManifest() imports the module and stores the data.
691
+ const perRouterPrefix = "\0" + VIRTUAL_ROUTES_MANIFEST_ID + "/";
692
+ if (id.startsWith(perRouterPrefix)) {
693
+ if (s.discoveryDone) {
694
+ await s.discoveryDone;
695
+ }
696
+ const routerId = id.slice(perRouterPrefix.length);
697
+ return generatePerRouterModule(s, routerId);
698
+ }
699
+ // virtual:rsc-router/prerender-paths load handler removed
700
+ return null;
701
+ },
702
+
703
+ // Record handler chunk metadata and RSC entry filename during RSC build.
704
+ // Used by closeBundle for handler code eviction and prerender data injection.
705
+ generateBundle(_options: any, bundle: any) {
706
+ if (this.environment?.name !== "rsc") return;
707
+
708
+ // Record RSC entry chunk filename for closeBundle injection
709
+ for (const [fileName, chunk] of Object.entries(bundle) as [
710
+ string,
711
+ any,
712
+ ][]) {
713
+ if (chunk.type === "chunk" && chunk.isEntry) {
714
+ s.rscEntryFileName = fileName;
715
+ break;
716
+ }
717
+ }
718
+
719
+ if (!s.resolvedPrerenderModules?.size && !s.resolvedStaticModules?.size)
720
+ return;
721
+
722
+ for (const [fileName, chunk] of Object.entries(bundle) as [
723
+ string,
724
+ any,
725
+ ][]) {
726
+ if (chunk.type !== "chunk") continue;
727
+
728
+ // Prerender handlers chunk
729
+ if (
730
+ fileName.includes("__prerender-handlers") &&
731
+ s.resolvedPrerenderModules?.size
732
+ ) {
733
+ const handlers = extractHandlerExportsFromChunk(
734
+ chunk.code,
735
+ s.resolvedPrerenderModules,
736
+ "Prerender",
737
+ true,
738
+ );
739
+ if (handlers.length > 0) {
740
+ s.handlerChunkInfo = { fileName, exports: handlers };
741
+ }
742
+ }
743
+
744
+ // Static handlers chunk
745
+ if (
746
+ fileName.includes("__static-handlers") &&
747
+ s.resolvedStaticModules?.size
748
+ ) {
749
+ const handlers = extractHandlerExportsFromChunk(
750
+ chunk.code,
751
+ s.resolvedStaticModules,
752
+ "Static",
753
+ false,
754
+ );
755
+ if (handlers.length > 0) {
756
+ s.staticHandlerChunkInfo = { fileName, exports: handlers };
757
+ }
758
+ }
759
+ }
760
+ },
761
+
762
+ // Build-time pre-rendering: evict handler code and inject collected prerender data.
763
+ // Collection now happens in-process during discoverRouters() via RSC runner.
764
+ // closeBundle only needs to evict handlers and inject the in-memory data.
765
+ closeBundle: {
766
+ order: "post" as const,
767
+ sequential: true,
768
+ async handler(this: any) {
769
+ if (!s.isBuildMode) return;
770
+ // Only run for the RSC environment — other environments (client, ssr) have
771
+ // no prerender/static data to process and would just do redundant file I/O.
772
+ if (this.environment && this.environment.name !== "rsc") return;
773
+ postprocessBundle(s);
774
+ },
775
+ },
776
+ };
777
+ }