@rangojs/router 0.0.0-experimental.0f44aca1

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