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