@rangojs/router 0.0.0-experimental.32 → 0.0.0-experimental.3232cd17

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 (376) hide show
  1. package/AGENTS.md +4 -0
  2. package/README.md +198 -44
  3. package/dist/bin/rango.js +287 -105
  4. package/dist/testing/vitest.js +82 -0
  5. package/dist/vite/index.js +3248 -1117
  6. package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  7. package/package.json +73 -21
  8. package/skills/api-client/SKILL.md +211 -0
  9. package/skills/breadcrumbs/SKILL.md +107 -1
  10. package/skills/bundle-analysis/SKILL.md +159 -0
  11. package/skills/cache-guide/SKILL.md +245 -21
  12. package/skills/caching/SKILL.md +302 -6
  13. package/skills/composability/SKILL.md +27 -2
  14. package/skills/css/SKILL.md +76 -0
  15. package/skills/document-cache/SKILL.md +78 -55
  16. package/skills/handler-use/SKILL.md +364 -0
  17. package/skills/hooks/SKILL.md +270 -30
  18. package/skills/host-router/SKILL.md +82 -22
  19. package/skills/i18n/SKILL.md +276 -0
  20. package/skills/intercept/SKILL.md +49 -5
  21. package/skills/layout/SKILL.md +35 -9
  22. package/skills/links/SKILL.md +249 -17
  23. package/skills/loader/SKILL.md +294 -30
  24. package/skills/middleware/SKILL.md +52 -13
  25. package/skills/migrate-nextjs/SKILL.md +584 -0
  26. package/skills/migrate-react-router/SKILL.md +769 -0
  27. package/skills/mime-routes/SKILL.md +27 -0
  28. package/skills/observability/SKILL.md +137 -0
  29. package/skills/parallel/SKILL.md +203 -7
  30. package/skills/prerender/SKILL.md +123 -100
  31. package/skills/rango/SKILL.md +250 -22
  32. package/skills/react-compiler/SKILL.md +168 -0
  33. package/skills/response-routes/SKILL.md +122 -47
  34. package/skills/route/SKILL.md +97 -5
  35. package/skills/router-setup/SKILL.md +90 -5
  36. package/skills/server-actions/SKILL.md +775 -0
  37. package/skills/streams-and-websockets/SKILL.md +283 -0
  38. package/skills/tailwind/SKILL.md +27 -3
  39. package/skills/testing/SKILL.md +129 -0
  40. package/skills/testing/bindings.md +89 -0
  41. package/skills/testing/cache-prerender.md +124 -0
  42. package/skills/testing/client-components.md +122 -0
  43. package/skills/testing/e2e-parity.md +125 -0
  44. package/skills/testing/flight.md +92 -0
  45. package/skills/testing/handles.md +129 -0
  46. package/skills/testing/loader.md +128 -0
  47. package/skills/testing/middleware.md +99 -0
  48. package/skills/testing/render-handler.md +121 -0
  49. package/skills/testing/response-routes.md +95 -0
  50. package/skills/testing/reverse-and-types.md +84 -0
  51. package/skills/testing/server-actions.md +107 -0
  52. package/skills/testing/server-tree.md +128 -0
  53. package/skills/testing/setup.md +120 -0
  54. package/skills/typesafety/SKILL.md +329 -27
  55. package/skills/use-cache/SKILL.md +36 -5
  56. package/skills/view-transitions/SKILL.md +294 -0
  57. package/src/__augment-tests__/augment.ts +81 -0
  58. package/src/__augment-tests__/augmented.check.ts +116 -0
  59. package/src/__internal.ts +67 -40
  60. package/src/browser/action-coordinator.ts +53 -36
  61. package/src/browser/action-fence.ts +47 -0
  62. package/src/browser/app-shell.ts +39 -0
  63. package/src/browser/app-version.ts +14 -0
  64. package/src/browser/cookie-name.ts +140 -0
  65. package/src/browser/event-controller.ts +86 -147
  66. package/src/browser/history-state.ts +21 -0
  67. package/src/browser/index.ts +3 -3
  68. package/src/browser/invalidate-client-cache.ts +52 -0
  69. package/src/browser/link-interceptor.ts +4 -0
  70. package/src/browser/navigation-bridge.ts +148 -19
  71. package/src/browser/navigation-client.ts +187 -67
  72. package/src/browser/navigation-store-handle.ts +38 -0
  73. package/src/browser/navigation-store.ts +76 -67
  74. package/src/browser/navigation-transaction.ts +18 -66
  75. package/src/browser/partial-update.ts +123 -94
  76. package/src/browser/prefetch/cache.ts +214 -36
  77. package/src/browser/prefetch/fetch.ts +260 -38
  78. package/src/browser/prefetch/policy.ts +6 -0
  79. package/src/browser/prefetch/queue.ts +126 -20
  80. package/src/browser/prefetch/resource-ready.ts +77 -0
  81. package/src/browser/rango-state.ts +158 -76
  82. package/src/browser/react/Link.tsx +93 -11
  83. package/src/browser/react/NavigationProvider.tsx +115 -34
  84. package/src/browser/react/ScrollRestoration.tsx +10 -6
  85. package/src/browser/react/context.ts +7 -2
  86. package/src/browser/react/filter-segment-order.ts +49 -7
  87. package/src/browser/react/index.ts +0 -48
  88. package/src/browser/react/location-state-shared.ts +166 -8
  89. package/src/browser/react/location-state.ts +39 -14
  90. package/src/browser/react/use-action.ts +6 -15
  91. package/src/browser/react/use-handle.ts +23 -69
  92. package/src/browser/react/use-link-status.ts +0 -4
  93. package/src/browser/react/use-navigation.ts +22 -5
  94. package/src/browser/react/use-params.ts +20 -10
  95. package/src/browser/react/use-reverse.ts +106 -0
  96. package/src/browser/react/use-router.ts +46 -11
  97. package/src/browser/react/use-search-params.ts +0 -5
  98. package/src/browser/react/use-segments.ts +11 -21
  99. package/src/browser/response-adapter.ts +52 -1
  100. package/src/browser/rsc-router.tsx +215 -76
  101. package/src/browser/scroll-restoration.ts +46 -39
  102. package/src/browser/segment-reconciler.ts +36 -9
  103. package/src/browser/segment-structure-assert.ts +2 -2
  104. package/src/browser/server-action-bridge.ts +176 -50
  105. package/src/browser/types.ts +95 -11
  106. package/src/browser/validate-redirect-origin.ts +43 -16
  107. package/src/build/collect-fallback-refs.ts +107 -0
  108. package/src/build/generate-manifest.ts +65 -40
  109. package/src/build/generate-route-types.ts +5 -0
  110. package/src/build/index.ts +8 -2
  111. package/src/build/prefix-tree-utils.ts +123 -0
  112. package/src/build/route-trie.ts +137 -32
  113. package/src/build/route-types/codegen.ts +4 -4
  114. package/src/build/route-types/include-resolution.ts +9 -2
  115. package/src/build/route-types/param-extraction.ts +6 -3
  116. package/src/build/route-types/per-module-writer.ts +7 -4
  117. package/src/build/route-types/router-processing.ts +278 -96
  118. package/src/build/route-types/scan-filter.ts +9 -2
  119. package/src/build/route-types/source-scan.ts +118 -0
  120. package/src/build/runtime-discovery.ts +9 -20
  121. package/src/cache/cache-error.ts +104 -0
  122. package/src/cache/cache-policy.ts +68 -28
  123. package/src/cache/cache-runtime.ts +149 -43
  124. package/src/cache/cache-scope.ts +148 -81
  125. package/src/cache/cache-tag.ts +98 -0
  126. package/src/cache/cf/cf-cache-store.ts +2550 -93
  127. package/src/cache/cf/index.ts +11 -17
  128. package/src/cache/document-cache.ts +78 -27
  129. package/src/cache/handle-snapshot.ts +63 -0
  130. package/src/cache/index.ts +23 -20
  131. package/src/cache/memory-segment-store.ts +136 -37
  132. package/src/cache/profile-registry.ts +6 -30
  133. package/src/cache/read-through-swr.ts +41 -11
  134. package/src/cache/segment-codec.ts +0 -16
  135. package/src/cache/tag-invalidation.ts +230 -0
  136. package/src/cache/taint.ts +55 -0
  137. package/src/cache/types.ts +33 -100
  138. package/src/cache/vercel/index.ts +11 -0
  139. package/src/cache/vercel/vercel-cache-store.ts +799 -0
  140. package/src/client.rsc.tsx +6 -21
  141. package/src/client.tsx +108 -290
  142. package/src/component-utils.ts +19 -0
  143. package/src/context-var.ts +84 -2
  144. package/src/debug.ts +2 -2
  145. package/src/decode-loader-results.ts +36 -0
  146. package/src/defer.ts +196 -0
  147. package/src/deps/ssr.ts +0 -1
  148. package/src/errors.ts +30 -4
  149. package/src/handle.ts +70 -22
  150. package/src/handles/MetaTags.tsx +0 -14
  151. package/src/handles/breadcrumbs.ts +16 -5
  152. package/src/handles/meta.ts +0 -39
  153. package/src/host/cookie-handler.ts +0 -36
  154. package/src/host/errors.ts +0 -24
  155. package/src/host/index.ts +8 -2
  156. package/src/host/pattern-matcher.ts +7 -50
  157. package/src/host/router.ts +107 -99
  158. package/src/host/testing.ts +40 -27
  159. package/src/host/types.ts +37 -4
  160. package/src/host/utils.ts +1 -1
  161. package/src/href-client.ts +137 -22
  162. package/src/index.rsc.ts +52 -26
  163. package/src/index.ts +100 -38
  164. package/src/internal-debug.ts +2 -4
  165. package/src/loader-store.ts +500 -0
  166. package/src/loader.rsc.ts +20 -13
  167. package/src/loader.ts +12 -11
  168. package/src/missing-id-error.ts +68 -0
  169. package/src/network-error-thrower.tsx +1 -6
  170. package/src/outlet-context.ts +1 -1
  171. package/src/outlet-provider.tsx +1 -5
  172. package/src/prerender/param-hash.ts +10 -11
  173. package/src/prerender/store.ts +37 -41
  174. package/src/prerender.ts +198 -82
  175. package/src/redirect-origin.ts +100 -0
  176. package/src/response-utils.ts +37 -0
  177. package/src/reverse.ts +65 -15
  178. package/src/root-error-boundary.tsx +1 -19
  179. package/src/route-content-wrapper.tsx +7 -72
  180. package/src/route-definition/dsl-helpers.ts +437 -274
  181. package/src/route-definition/helper-factories.ts +29 -139
  182. package/src/route-definition/helpers-types.ts +113 -37
  183. package/src/route-definition/index.ts +3 -0
  184. package/src/route-definition/redirect.ts +52 -10
  185. package/src/route-definition/resolve-handler-use.ts +161 -0
  186. package/src/route-definition/use-item-types.ts +32 -0
  187. package/src/route-map-builder.ts +7 -17
  188. package/src/route-types.ts +37 -41
  189. package/src/router/basename.ts +14 -0
  190. package/src/router/content-negotiation.ts +108 -9
  191. package/src/router/error-handling.ts +13 -17
  192. package/src/router/find-match.ts +45 -22
  193. package/src/router/handler-context.ts +83 -41
  194. package/src/router/intercept-resolution.ts +25 -23
  195. package/src/router/lazy-includes.ts +19 -53
  196. package/src/router/loader-resolution.ts +213 -30
  197. package/src/router/logging.ts +5 -8
  198. package/src/router/manifest.ts +49 -45
  199. package/src/router/match-api.ts +120 -204
  200. package/src/router/match-context.ts +0 -22
  201. package/src/router/match-handlers.ts +58 -58
  202. package/src/router/match-middleware/background-revalidation.ts +27 -6
  203. package/src/router/match-middleware/cache-lookup.ts +205 -249
  204. package/src/router/match-middleware/cache-store.ts +45 -32
  205. package/src/router/match-middleware/intercept-resolution.ts +8 -28
  206. package/src/router/match-middleware/segment-resolution.ts +52 -18
  207. package/src/router/match-pipelines.ts +1 -42
  208. package/src/router/match-result.ts +104 -40
  209. package/src/router/metrics.ts +5 -34
  210. package/src/router/middleware-types.ts +13 -142
  211. package/src/router/middleware.ts +173 -143
  212. package/src/router/navigation-snapshot.ts +131 -0
  213. package/src/router/params-util.ts +23 -0
  214. package/src/router/pattern-matching.ts +109 -63
  215. package/src/router/prerender-match.ts +190 -54
  216. package/src/router/preview-match.ts +32 -102
  217. package/src/router/request-classification.ts +276 -0
  218. package/src/router/revalidation.ts +63 -55
  219. package/src/router/route-snapshot.ts +244 -0
  220. package/src/router/router-context.ts +6 -28
  221. package/src/router/router-interfaces.ts +100 -35
  222. package/src/router/router-options.ts +91 -11
  223. package/src/router/router-registry.ts +2 -5
  224. package/src/router/segment-resolution/fresh.ts +242 -75
  225. package/src/router/segment-resolution/helpers.ts +63 -24
  226. package/src/router/segment-resolution/loader-cache.ts +41 -37
  227. package/src/router/segment-resolution/revalidation.ts +456 -372
  228. package/src/router/segment-resolution/static-store.ts +19 -5
  229. package/src/router/segment-resolution/streamed-handler-telemetry.ts +52 -0
  230. package/src/router/segment-resolution/view-transition-default.ts +36 -0
  231. package/src/router/segment-resolution.ts +4 -1
  232. package/src/router/segment-wrappers.ts +2 -3
  233. package/src/router/state-cookie-name.ts +33 -0
  234. package/src/router/substitute-pattern-params.ts +56 -0
  235. package/src/router/telemetry-otel.ts +0 -20
  236. package/src/router/telemetry.ts +96 -19
  237. package/src/router/timeout.ts +0 -20
  238. package/src/router/trie-matching.ts +91 -46
  239. package/src/router/types.ts +10 -63
  240. package/src/router/url-params.ts +44 -0
  241. package/src/router.ts +134 -43
  242. package/src/rsc/handler-context.ts +3 -2
  243. package/src/rsc/handler.ts +492 -383
  244. package/src/rsc/helpers.ts +162 -46
  245. package/src/rsc/index.ts +1 -1
  246. package/src/rsc/json-route-result.ts +38 -0
  247. package/src/rsc/loader-fetch.ts +23 -3
  248. package/src/rsc/manifest-init.ts +33 -42
  249. package/src/rsc/origin-guard.ts +39 -25
  250. package/src/rsc/progressive-enhancement.ts +30 -3
  251. package/src/rsc/redirect-guard.ts +99 -0
  252. package/src/rsc/response-error.ts +79 -12
  253. package/src/rsc/response-route-handler.ts +90 -63
  254. package/src/rsc/rsc-rendering.ts +56 -54
  255. package/src/rsc/runtime-warnings.ts +23 -10
  256. package/src/rsc/server-action.ts +74 -67
  257. package/src/rsc/ssr-setup.ts +18 -2
  258. package/src/rsc/types.ts +25 -6
  259. package/src/runtime-env.ts +18 -0
  260. package/src/search-params.ts +4 -20
  261. package/src/segment-content-promise.ts +67 -0
  262. package/src/segment-loader-promise.ts +134 -0
  263. package/src/segment-system.tsx +272 -129
  264. package/src/serialize.ts +243 -0
  265. package/src/server/context.ts +309 -61
  266. package/src/server/cookie-store.ts +80 -5
  267. package/src/server/handle-store.ts +26 -24
  268. package/src/server/loader-registry.ts +10 -28
  269. package/src/server/request-context.ts +338 -126
  270. package/src/ssr/index.tsx +23 -15
  271. package/src/static-handler.ts +27 -18
  272. package/src/testing/cache-status.ts +162 -0
  273. package/src/testing/collect-handle.ts +40 -0
  274. package/src/testing/dispatch.ts +618 -0
  275. package/src/testing/dom.entry.ts +22 -0
  276. package/src/testing/e2e/fixture.ts +188 -0
  277. package/src/testing/e2e/index.ts +128 -0
  278. package/src/testing/e2e/matchers.ts +35 -0
  279. package/src/testing/e2e/page-helpers.ts +272 -0
  280. package/src/testing/e2e/parity.ts +387 -0
  281. package/src/testing/e2e/server.ts +195 -0
  282. package/src/testing/flight-matchers.ts +97 -0
  283. package/src/testing/flight-normalize.ts +11 -0
  284. package/src/testing/flight-runtime.d.ts +57 -0
  285. package/src/testing/flight-tree.ts +682 -0
  286. package/src/testing/flight.entry.ts +52 -0
  287. package/src/testing/flight.ts +232 -0
  288. package/src/testing/generated-routes.ts +183 -0
  289. package/src/testing/index.ts +99 -0
  290. package/src/testing/internal/context.ts +348 -0
  291. package/src/testing/internal/flight-client-globals.ts +30 -0
  292. package/src/testing/internal/seed-vars.ts +54 -0
  293. package/src/testing/render-handler.ts +330 -0
  294. package/src/testing/render-route.tsx +566 -0
  295. package/src/testing/run-loader.ts +378 -0
  296. package/src/testing/run-middleware.ts +205 -0
  297. package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
  298. package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
  299. package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
  300. package/src/testing/vitest-stubs/version.ts +5 -0
  301. package/src/testing/vitest.ts +305 -0
  302. package/src/theme/ThemeProvider.tsx +0 -52
  303. package/src/theme/ThemeScript.tsx +0 -6
  304. package/src/theme/constants.ts +0 -12
  305. package/src/theme/index.ts +0 -7
  306. package/src/theme/theme-context.ts +1 -5
  307. package/src/theme/theme-script.ts +0 -14
  308. package/src/theme/use-theme.ts +0 -3
  309. package/src/types/boundaries.ts +0 -35
  310. package/src/types/cache-types.ts +17 -8
  311. package/src/types/error-types.ts +30 -90
  312. package/src/types/global-namespace.ts +54 -41
  313. package/src/types/handler-context.ts +233 -81
  314. package/src/types/index.ts +1 -10
  315. package/src/types/loader-types.ts +44 -15
  316. package/src/types/request-scope.ts +107 -0
  317. package/src/types/route-config.ts +6 -50
  318. package/src/types/route-entry.ts +19 -7
  319. package/src/types/segments.ts +37 -14
  320. package/src/urls/include-helper.ts +33 -70
  321. package/src/urls/index.ts +1 -11
  322. package/src/urls/path-helper-types.ts +58 -11
  323. package/src/urls/path-helper.ts +57 -111
  324. package/src/urls/pattern-types.ts +48 -19
  325. package/src/urls/response-types.ts +25 -22
  326. package/src/urls/type-extraction.ts +58 -139
  327. package/src/urls/urls-function.ts +1 -18
  328. package/src/use-loader.tsx +346 -89
  329. package/src/vite/debug.ts +185 -0
  330. package/src/vite/discovery/bundle-postprocess.ts +36 -38
  331. package/src/vite/discovery/discover-routers.ts +130 -85
  332. package/src/vite/discovery/discovery-errors.ts +194 -0
  333. package/src/vite/discovery/gate-state.ts +171 -0
  334. package/src/vite/discovery/prerender-collection.ts +192 -99
  335. package/src/vite/discovery/route-types-writer.ts +40 -84
  336. package/src/vite/discovery/self-gen-tracking.ts +27 -1
  337. package/src/vite/discovery/state.ts +51 -6
  338. package/src/vite/discovery/virtual-module-codegen.ts +14 -34
  339. package/src/vite/index.ts +8 -0
  340. package/src/vite/plugin-types.ts +187 -69
  341. package/src/vite/plugins/cjs-to-esm.ts +8 -18
  342. package/src/vite/plugins/client-ref-dedup.ts +16 -11
  343. package/src/vite/plugins/client-ref-hashing.ts +28 -15
  344. package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
  345. package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  346. package/src/vite/plugins/cloudflare-protocol-stub.ts +194 -0
  347. package/src/vite/plugins/expose-action-id.ts +49 -98
  348. package/src/vite/plugins/expose-id-utils.ts +11 -50
  349. package/src/vite/plugins/expose-ids/export-analysis.ts +76 -34
  350. package/src/vite/plugins/expose-ids/handler-transform.ts +10 -48
  351. package/src/vite/plugins/expose-ids/loader-transform.ts +3 -20
  352. package/src/vite/plugins/expose-ids/router-transform.ts +20 -16
  353. package/src/vite/plugins/expose-internal-ids.ts +554 -317
  354. package/src/vite/plugins/performance-tracks.ts +89 -0
  355. package/src/vite/plugins/refresh-cmd.ts +89 -27
  356. package/src/vite/plugins/use-cache-transform.ts +73 -83
  357. package/src/vite/plugins/vercel-output.ts +258 -0
  358. package/src/vite/plugins/version-injector.ts +21 -25
  359. package/src/vite/plugins/version-plugin.ts +41 -20
  360. package/src/vite/plugins/virtual-entries.ts +2 -17
  361. package/src/vite/rango.ts +257 -289
  362. package/src/vite/router-discovery.ts +930 -140
  363. package/src/vite/utils/ast-handler-extract.ts +15 -31
  364. package/src/vite/utils/banner.ts +4 -4
  365. package/src/vite/utils/bundle-analysis.ts +10 -15
  366. package/src/vite/utils/client-chunks.ts +184 -0
  367. package/src/vite/utils/forward-user-plugins.ts +171 -0
  368. package/src/vite/utils/manifest-utils.ts +4 -59
  369. package/src/vite/utils/package-resolution.ts +20 -52
  370. package/src/vite/utils/prerender-utils.ts +27 -29
  371. package/src/vite/utils/shared-utils.ts +92 -42
  372. package/src/browser/action-response-classifier.ts +0 -99
  373. package/src/browser/react/use-client-cache.ts +0 -58
  374. package/src/browser/shallow.ts +0 -40
  375. package/src/handles/index.ts +0 -7
  376. package/src/router/middleware-cookies.ts +0 -55
@@ -10,14 +10,21 @@ import type { Plugin } from "vite";
10
10
  import { createServer as createViteServer } from "vite";
11
11
  import { resolve } from "node:path";
12
12
  import { readFileSync } from "node:fs";
13
+ import { createRequire, register } from "node:module";
14
+ import { pathToFileURL } from "node:url";
13
15
  import {
14
16
  formatNestedRouterConflictError,
15
17
  findNestedRouterConflict,
16
18
  findRouterFiles,
17
19
  createScanFilter,
18
20
  } from "../build/generate-route-types.js";
21
+ import { firstCodeMatchIndex } from "../build/route-types/source-scan.js";
19
22
  import { createVersionPlugin } from "./plugins/version-plugin.js";
20
23
  import { createVirtualStubPlugin } from "./plugins/virtual-stub-plugin.js";
24
+ import {
25
+ BUILD_ENV_GLOBAL_KEY,
26
+ createCloudflareProtocolStubPlugin,
27
+ } from "./plugins/cloudflare-protocol-stub.js";
21
28
  import {
22
29
  exposeInternalIds,
23
30
  exposeRouterId,
@@ -30,7 +37,10 @@ import {
30
37
  type DiscoveryState,
31
38
  type PluginOptions,
32
39
  } from "./discovery/state.js";
33
- import { consumeSelfGenWrite } from "./discovery/self-gen-tracking.js";
40
+ import {
41
+ consumeSelfGenWrite,
42
+ peekSelfGenWrite,
43
+ } from "./discovery/self-gen-tracking.js";
34
44
  import { discoverRouters } from "./discovery/discover-routers.js";
35
45
  import {
36
46
  writeCombinedRouteTypesWithTracking,
@@ -42,10 +52,65 @@ import {
42
52
  generatePerRouterModule,
43
53
  } from "./discovery/virtual-module-codegen.js";
44
54
  import { postprocessBundle } from "./discovery/bundle-postprocess.js";
55
+ import { createDiscoveryGate } from "./discovery/gate-state.js";
45
56
  import { resetStagedBuildAssets } from "./utils/prerender-utils.js";
57
+ import { resolveRscEntryFromConfig } from "./utils/shared-utils.js";
58
+ import {
59
+ pickForwardedRunnerConfig,
60
+ selectForwardableResolvePlugins,
61
+ } from "./utils/forward-user-plugins.js";
62
+ import { createRangoDebugger, timed, timedSync, NS } from "./debug.js";
63
+
64
+ const debugDiscovery = createRangoDebugger(NS.discovery);
65
+ const debugRoutes = createRangoDebugger(NS.routes);
66
+ const debugBuild = createRangoDebugger(NS.build);
67
+ const debugDev = createRangoDebugger(NS.dev);
46
68
 
47
69
  export { VIRTUAL_ROUTES_MANIFEST_ID };
48
70
 
71
+ // ============================================================================
72
+ // Node ESM Loader Hook Registration
73
+ // ============================================================================
74
+
75
+ /**
76
+ * Registers a Node ESM loader hook that resolves `cloudflare:*` specifiers
77
+ * to a data: URL stub. Defense-in-depth alongside the Vite transform in
78
+ * `cloudflare-protocol-stub.ts`:
79
+ *
80
+ * - The Vite transform catches `cloudflare:*` imports in modules that flow
81
+ * through Vite's plugin pipeline. That's the vast majority of cases.
82
+ * - The Node loader catches imports in modules that Vite/Rollup externalize
83
+ * (e.g. the `partyserver` package, which has a top-level
84
+ * `import { DurableObject, env } from "cloudflare:workers"` and ships
85
+ * shapes plugin-rsc marks as external). Externalized modules are loaded
86
+ * via Node's native ESM loader, which rejects URL schemes.
87
+ *
88
+ * Registration is process-global and one-shot. The hook only intercepts
89
+ * `cloudflare:*` specifiers; everything else passes through via
90
+ * `nextResolve()`. It runs in a separate worker thread (Node ESM loader
91
+ * architecture), so it can't read the `globalThis[BUILD_ENV_GLOBAL_KEY]`
92
+ * bridge that the Vite transform uses — the stubs served here always
93
+ * return `env = {}`. That's fine because externalized libraries don't
94
+ * typically access `env` at module top level; user source (where real
95
+ * `env` matters at build time) flows through the Vite transform.
96
+ */
97
+ let loaderHookRegistered = false;
98
+ function ensureCloudflareProtocolLoaderRegistered(): void {
99
+ if (loaderHookRegistered) return;
100
+ loaderHookRegistered = true;
101
+ try {
102
+ register(
103
+ new URL("./plugins/cloudflare-protocol-loader-hook.mjs", import.meta.url),
104
+ );
105
+ } catch (err: any) {
106
+ // register() requires Node 18.19+ / 20.6+. Older Node still has the
107
+ // Vite transform as primary defense.
108
+ console.warn(
109
+ `[rango] Could not register Node ESM loader hook for cloudflare:* imports (${err?.message ?? err}). Falling back to Vite transform only.`,
110
+ );
111
+ }
112
+ }
113
+
49
114
  // ============================================================================
50
115
  // Temp Server Factory
51
116
  // ============================================================================
@@ -65,15 +130,32 @@ async function createTempRscServer(
65
130
  state: DiscoveryState,
66
131
  options: { forceBuild?: boolean; cacheDir?: string } = {},
67
132
  ) {
133
+ // Install the Node ESM loader hook before any module evaluation so
134
+ // `cloudflare:*` specifiers in externalized/loader-delegated modules
135
+ // (e.g. packages plugin-rsc marks as external) resolve to stubs
136
+ // instead of crashing Node's native loader.
137
+ ensureCloudflareProtocolLoaderRegistered();
68
138
  const { default: rsc } = await import("@vitejs/plugin-rsc");
139
+ // Mirror the user's resolution config + plugins so discovery (and the
140
+ // prerender/static rendering that shares this runner) resolves modules the
141
+ // same way the real environment does. Falls back to the legacy alias-only
142
+ // behavior if configResolved hasn't populated the parity slice yet.
143
+ const runnerConfig = state.userRunnerConfig;
144
+ const resolveConfig = runnerConfig?.resolve ?? {
145
+ alias: state.userResolveAlias,
146
+ };
147
+ const oxcConfig = runnerConfig?.oxc ?? {
148
+ jsx: { runtime: "automatic", importSource: "react" },
149
+ };
69
150
  return createViteServer({
70
151
  root: state.projectRoot,
71
152
  configFile: false,
72
153
  server: { middlewareMode: true },
73
154
  appType: "custom",
74
155
  logLevel: "silent",
75
- resolve: { alias: state.userResolveAlias },
76
- esbuild: { jsx: "automatic", jsxImportSource: "react" },
156
+ resolve: resolveConfig,
157
+ ...(runnerConfig?.define ? { define: runnerConfig.define } : {}),
158
+ oxc: oxcConfig as any,
77
159
  ...(options.cacheDir && { cacheDir: options.cacheDir }),
78
160
  plugins: [
79
161
  rsc({
@@ -87,14 +169,124 @@ async function createTempRscServer(
87
169
  ...(options.forceBuild ? [hashClientRefs(state.projectRoot)] : []),
88
170
  createVersionPlugin(),
89
171
  createVirtualStubPlugin(),
172
+ createCloudflareProtocolStubPlugin(),
90
173
  // Dev prerender must use dev-mode IDs (path-based) to match the workerd
91
174
  // runtime. forceBuild produces hashed IDs for production bundle consistency.
92
175
  exposeInternalIds(options.forceBuild ? { forceBuild: true } : undefined),
93
176
  exposeRouterId(),
177
+ // Forwarded user resolution plugins (e.g. vite-tsconfig-paths). Stripped
178
+ // to resolveId/load and placed last so framework resolution runs first;
179
+ // Vite re-sorts by `enforce`, so `enforce: "pre"` resolvers still lead.
180
+ ...state.userResolvePlugins,
94
181
  ],
95
182
  });
96
183
  }
97
184
 
185
+ // ============================================================================
186
+ // Build-Time Env Resolution
187
+ // ============================================================================
188
+
189
+ import type {
190
+ BuildEnvOption,
191
+ BuildEnvFactoryContext,
192
+ BuildEnvResult,
193
+ } from "./plugin-types.js";
194
+
195
+ /**
196
+ * Resolve the buildEnv option into a concrete { env, dispose? } result.
197
+ * Handles all four input shapes: false, "auto", factory, plain object.
198
+ */
199
+ async function resolveBuildEnv(
200
+ option: BuildEnvOption | undefined,
201
+ factoryCtx: BuildEnvFactoryContext,
202
+ ): Promise<BuildEnvResult | null> {
203
+ if (!option) return null;
204
+
205
+ if (option === "auto") {
206
+ if (factoryCtx.preset !== "cloudflare") {
207
+ throw new Error(
208
+ '[rango] buildEnv: "auto" is only supported with preset: "cloudflare". ' +
209
+ "Use a factory function or plain object for other presets.",
210
+ );
211
+ }
212
+ try {
213
+ // Resolve wrangler from the user's project root (not the router package)
214
+ const userRequire = createRequire(
215
+ resolve(factoryCtx.root, "package.json"),
216
+ );
217
+ const wranglerPath = userRequire.resolve("wrangler");
218
+ const { getPlatformProxy } = (await import(
219
+ pathToFileURL(wranglerPath).href
220
+ )) as {
221
+ getPlatformProxy: (opts?: any) => Promise<any>;
222
+ };
223
+ const proxy = await getPlatformProxy();
224
+ return {
225
+ env: proxy.env as Record<string, unknown>,
226
+ dispose: proxy.dispose,
227
+ };
228
+ } catch (err: any) {
229
+ throw new Error(
230
+ '[rango] buildEnv: "auto" requires wrangler to be installed.\n' +
231
+ `Install it with: pnpm add -D wrangler\n${err.message}`,
232
+ );
233
+ }
234
+ }
235
+
236
+ if (typeof option === "function") {
237
+ return await option(factoryCtx);
238
+ }
239
+
240
+ // Plain object
241
+ return { env: option };
242
+ }
243
+
244
+ /**
245
+ * Acquire build-time env bindings and store on discovery state.
246
+ * Returns true if env was acquired, false if buildEnv is disabled.
247
+ */
248
+ async function acquireBuildEnv(
249
+ s: DiscoveryState,
250
+ command: "serve" | "build",
251
+ mode: string,
252
+ ): Promise<boolean> {
253
+ const option = s.opts?.buildEnv;
254
+ if (!option) return false;
255
+
256
+ const result = await resolveBuildEnv(option, {
257
+ root: s.projectRoot,
258
+ mode,
259
+ command,
260
+ preset: s.opts?.preset ?? "node",
261
+ });
262
+ if (!result) return false;
263
+
264
+ s.resolvedBuildEnv = result.env;
265
+ s.buildEnvDispose = result.dispose ?? null;
266
+ // Bridge the resolved env into `cloudflare:workers`'s stubbed `env`
267
+ // export so user code that does `import { env } from "cloudflare:workers"`
268
+ // sees the real bindings proxy during discovery + prerender instead of
269
+ // an empty object. The stub reads this global at module-evaluation time.
270
+ (globalThis as Record<string, unknown>)[BUILD_ENV_GLOBAL_KEY] = result.env;
271
+ return true;
272
+ }
273
+
274
+ /**
275
+ * Release build-time env resources and clear state.
276
+ */
277
+ async function releaseBuildEnv(s: DiscoveryState): Promise<void> {
278
+ if (s.buildEnvDispose) {
279
+ try {
280
+ await s.buildEnvDispose();
281
+ } catch (err: any) {
282
+ console.warn(`[rango] buildEnv dispose failed: ${err.message}`);
283
+ }
284
+ s.buildEnvDispose = null;
285
+ }
286
+ s.resolvedBuildEnv = undefined;
287
+ delete (globalThis as Record<string, unknown>)[BUILD_ENV_GLOBAL_KEY];
288
+ }
289
+
98
290
  /**
99
291
  * Plugin that discovers router instances at dev/build time via the RSC environment.
100
292
  *
@@ -112,6 +304,8 @@ export function createRouterDiscoveryPlugin(
112
304
  opts?: PluginOptions,
113
305
  ): Plugin {
114
306
  const s = createDiscoveryState(entryPath, opts);
307
+ let viteCommand: "serve" | "build" = "build";
308
+ let viteMode = "production";
115
309
 
116
310
  return {
117
311
  name: "@rangojs/router:discovery",
@@ -122,58 +316,50 @@ export function createRouterDiscoveryPlugin(
122
316
  __RANGO_DEBUG__: JSON.stringify(!!process.env.INTERNAL_RANGO_DEBUG),
123
317
  },
124
318
  };
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
- }
319
+ // Prerender/static handler modules are bundled naturally with the
320
+ // rest of the RSC entry. A previous design forced them into dedicated
321
+ // __prerender-handlers / __static-handlers chunks via manualChunks,
322
+ // but Rollup hoisted all shared dependencies into those chunks,
323
+ // inflating them to ~1 MB with active runtime code. Handler code is
324
+ // evicted in closeBundle regardless of which chunk it lands in.
145
325
  return config;
146
326
  },
147
327
 
148
328
  configResolved(config) {
149
329
  s.projectRoot = config.root;
330
+ // Compile the optional discovery scan filter (glob include/exclude) now
331
+ // that the project root is known. findRouterFiles() below — and the
332
+ // build/HMR rediscovery paths — honor s.scanFilter.
333
+ s.scanFilter = opts?.discovery
334
+ ? createScanFilter(s.projectRoot, opts.discovery)
335
+ : undefined;
150
336
  s.isBuildMode = config.command === "build";
337
+ viteCommand = config.command as "serve" | "build";
338
+ viteMode = config.mode;
151
339
  // Capture user's resolve aliases for the temp server
152
340
  s.userResolveAlias = config.resolve.alias;
341
+ // Capture the data-only resolution config (resolve.*, define, oxc) and
342
+ // the user's resolution plugins (resolveId/load) so the discovery temp
343
+ // server resolves modules the same way the real environment does.
344
+ // Without this, both flavors of user resolution are absent during
345
+ // discovery/prerender/static rendering even though they apply at request
346
+ // time: third-party resolvers (e.g. vite-tsconfig-paths, forwarded as
347
+ // plugins) and Vite 8's native resolve.tsconfigPaths (forwarded in the
348
+ // data slice). See utils/forward-user-plugins.ts.
349
+ s.userRunnerConfig = pickForwardedRunnerConfig(config);
350
+ s.userResolvePlugins = selectForwardableResolvePlugins(
351
+ config.plugins as any,
352
+ );
153
353
  // Node preset: pick up auto-discovered router path from the config() hook.
154
354
  // The auto-discover plugin runs in config() using Vite's resolved root,
155
355
  // populating the mutable ref before configResolved fires.
156
356
  if (!s.resolvedEntryPath && opts?.routerPathRef?.path) {
157
357
  s.resolvedEntryPath = opts.routerPathRef.path;
158
358
  }
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.
359
+ // Cloudflare preset: entry comes from the resolved RSC env config.
162
360
  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
- });
361
+ const entry = resolveRscEntryFromConfig(config);
362
+ if (entry) s.resolvedEntryPath = entry;
177
363
  }
178
364
  // Generate combined named-routes.gen.ts from static source parsing.
179
365
  // Runs before the dev server starts so the gen file exists immediately for IDE.
@@ -212,6 +398,17 @@ export function createRouterDiscoveryPlugin(
212
398
  resolveDiscovery = resolve;
213
399
  });
214
400
 
401
+ // Manifest-readiness gate + rediscovery scheduler.
402
+ // The virtual:rsc-router/routes-manifest module's `load()` hook
403
+ // awaits `s.discoveryDone`; the gate is reset on each discovery
404
+ // cycle so workerd's HMR reloads block until the new gen file is
405
+ // written. State machine + transitions are extracted into
406
+ // ./discovery/gate-state.ts and unit-tested there — see the
407
+ // module's JSDoc for the four-flag contract.
408
+ const gate = createDiscoveryGate(s, debugDiscovery);
409
+ const beginDiscoveryGate = gate.beginGate;
410
+ const resolveDiscoveryGate = gate.resolveGate;
411
+
215
412
  // Compute dev server origin from resolved URLs (preferred) or config port (fallback).
216
413
  // Called after discovery (or in the load hook) when the server may be listening.
217
414
  const getDevServerOrigin = () =>
@@ -225,18 +422,113 @@ export function createRouterDiscoveryPlugin(
225
422
  let prerenderTempServer: any = null;
226
423
  let prerenderNodeRegistry: Map<string, any> | null = null;
227
424
 
228
- // Clean up the temporary server when the dev server shuts down
425
+ // Clean up the temporary server and build env when the dev server shuts down
229
426
  server.httpServer?.on("close", () => {
230
427
  if (prerenderTempServer) {
231
428
  prerenderTempServer.close().catch(() => {});
232
429
  prerenderTempServer = null;
233
430
  }
431
+ releaseBuildEnv(s).catch(() => {});
234
432
  });
235
433
 
434
+ // Mirror the build-path contract (the buildStart hook below, which sets
435
+ // __rscRouterDiscoveryActive before running user modules):
436
+ // set __rscRouterDiscoveryActive before running user modules so any
437
+ // module-level router.reverse() calls return a placeholder instead
438
+ // of throwing. The temp Vite server's module runner has its own
439
+ // module context; the flag must be on globalThis to cross that
440
+ // boundary. Cleared in finally so the dev request handlers run with
441
+ // strict reverse() semantics afterwards.
442
+ async function importEntryAndRegistry(tempRscEnv: any): Promise<void> {
443
+ const flagAlreadySet = !!(globalThis as any).__rscRouterDiscoveryActive;
444
+ if (!flagAlreadySet) {
445
+ (globalThis as any).__rscRouterDiscoveryActive = true;
446
+ }
447
+ try {
448
+ debugDiscovery?.(
449
+ "importEntryAndRegistry: importing entry (flag=%s)",
450
+ (globalThis as any).__rscRouterDiscoveryActive ?? false,
451
+ );
452
+ await tempRscEnv.runner.import(s.resolvedEntryPath!);
453
+ debugDiscovery?.(
454
+ "importEntryAndRegistry: entry import OK, fetching RouterRegistry",
455
+ );
456
+ const serverMod = await tempRscEnv.runner.import(
457
+ "@rangojs/router/server",
458
+ );
459
+ prerenderNodeRegistry = serverMod.RouterRegistry;
460
+ debugDiscovery?.(
461
+ "importEntryAndRegistry: registry size=%d",
462
+ prerenderNodeRegistry?.size ?? 0,
463
+ );
464
+ } finally {
465
+ if (!flagAlreadySet) {
466
+ delete (globalThis as any).__rscRouterDiscoveryActive;
467
+ debugDiscovery?.(
468
+ "importEntryAndRegistry: cleared __rscRouterDiscoveryActive",
469
+ );
470
+ }
471
+ }
472
+ }
473
+
236
474
  async function getOrCreateTempServer(): Promise<any | null> {
237
- if (prerenderNodeRegistry) {
238
- return (prerenderTempServer.environments as any)?.rsc ?? null;
475
+ // Reuse path: if a temp server is already alive, prefer reusing
476
+ // it over orphaning the existing instance and spinning up a new
477
+ // one. This handles two cases:
478
+ //
479
+ // 1. Steady-state cache hit (cold-start completed, registry
480
+ // cached) — return the env immediately.
481
+ // 2. Recovery from a failed refresh: refreshTempRscEnv() may
482
+ // have invalidated and nulled the registry, then thrown
483
+ // during importEntryAndRegistry. Without reuse, the next
484
+ // call would `createTempRscServer` and overwrite the
485
+ // handle, leaking the previous server. Try to re-import on
486
+ // the existing runner first; only if THAT fails do we
487
+ // close the orphan and create new.
488
+ if (prerenderTempServer) {
489
+ const existingEnv = (prerenderTempServer.environments as any)?.rsc;
490
+ if (existingEnv?.runner) {
491
+ if (prerenderNodeRegistry) {
492
+ debugDiscovery?.(
493
+ "getOrCreateTempServer: cached temp runner reused",
494
+ );
495
+ return existingEnv;
496
+ }
497
+ // Server alive but registry missing — likely after a prior
498
+ // refresh's invalidate + import threw. Try to re-import.
499
+ debugDiscovery?.(
500
+ "getOrCreateTempServer: server alive but registry missing — re-importing",
501
+ );
502
+ try {
503
+ await importEntryAndRegistry(existingEnv);
504
+ return existingEnv;
505
+ } catch (err: any) {
506
+ debugDiscovery?.(
507
+ "getOrCreateTempServer: reuse import failed (%s) — closing orphan and creating fresh",
508
+ err?.message ?? String(err),
509
+ );
510
+ await prerenderTempServer.close().catch(() => {});
511
+ prerenderTempServer = null;
512
+ prerenderNodeRegistry = null;
513
+ // Fall through to create-new path below.
514
+ }
515
+ } else {
516
+ // Server reference exists but its rsc env is unhealthy
517
+ // (no runner). Close and recreate.
518
+ debugDiscovery?.(
519
+ "getOrCreateTempServer: existing server has no rsc.runner — closing and recreating",
520
+ );
521
+ await prerenderTempServer.close().catch(() => {});
522
+ prerenderTempServer = null;
523
+ prerenderNodeRegistry = null;
524
+ }
239
525
  }
526
+
527
+ // Create path: no existing temp server (or just nullified above).
528
+ debugDiscovery?.(
529
+ "getOrCreateTempServer: creating new temp server, entry=%s",
530
+ s.resolvedEntryPath ?? "(unset)",
531
+ );
240
532
  try {
241
533
  prerenderTempServer = await createTempRscServer(s, {
242
534
  cacheDir: "node_modules/.vite_prerender",
@@ -244,58 +536,189 @@ export function createRouterDiscoveryPlugin(
244
536
 
245
537
  const tempRscEnv = (prerenderTempServer.environments as any)?.rsc;
246
538
  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;
539
+ await importEntryAndRegistry(tempRscEnv);
252
540
  return tempRscEnv;
253
541
  }
542
+ debugDiscovery?.(
543
+ "getOrCreateTempServer: tempRscEnv.runner unavailable",
544
+ );
254
545
  } catch (err: any) {
255
- console.warn(
256
- `[rsc-router] Failed to create temp runner: ${err.message}`,
546
+ debugDiscovery?.(
547
+ "getOrCreateTempServer: FAILED message=%s",
548
+ err.message,
257
549
  );
550
+ console.warn(`[rango] Failed to create temp runner: ${err.message}`);
258
551
  }
259
552
  return null;
260
553
  }
261
554
 
555
+ // Clear the package-level singleton registries that survive a Vite
556
+ // moduleGraph.invalidateAll(). createRouter() / createHostRouter()
557
+ // call .set(id, ...) on these Maps; for "router removed" or
558
+ // "router id changed" edits, the OLD entry would persist after
559
+ // re-import without an explicit .clear(), leaving ghost routes
560
+ // in discoverRouters' output.
561
+ //
562
+ // We import the same module the runner imports, so the .clear()
563
+ // here mutates the same Map the freshly re-imported entry will
564
+ // populate.
565
+ async function clearTempRegistries(tempRscEnv: any): Promise<void> {
566
+ try {
567
+ const serverMod = await tempRscEnv.runner.import(
568
+ "@rangojs/router/server",
569
+ );
570
+ if (typeof serverMod?.RouterRegistry?.clear === "function") {
571
+ serverMod.RouterRegistry.clear();
572
+ }
573
+ if (typeof serverMod?.HostRouterRegistry?.clear === "function") {
574
+ serverMod.HostRouterRegistry.clear();
575
+ }
576
+ debugDiscovery?.(
577
+ "clearTempRegistries: cleared RouterRegistry + HostRouterRegistry",
578
+ );
579
+ } catch (err: any) {
580
+ // Non-fatal: if the import fails here, importEntryAndRegistry
581
+ // below will fail loudly with the same root cause and the
582
+ // caller will surface it.
583
+ debugDiscovery?.(
584
+ "clearTempRegistries: import @rangojs/router/server failed (%s)",
585
+ err?.message ?? String(err),
586
+ );
587
+ }
588
+ }
589
+
590
+ // HMR refresh: keep the temp Vite server alive across HMR cycles and
591
+ // invalidate its module graph instead of close+recreate. Closing the
592
+ // temp server during workerd's first post-cold-start module-fetch
593
+ // window disrupted the main dev server's transport — the user-visible
594
+ // symptom was a `transport was disconnected, cannot call "fetchModule"`
595
+ // error on the first urls.tsx edit (workerd's cache was cold, so its
596
+ // eval was still in flight when our close() ran). Module-graph
597
+ // invalidation is the architecturally cleaner refresh: same Vite
598
+ // instance, same transport, fresh source.
599
+ //
600
+ // Falls back to close+recreate when neither the env-level nor
601
+ // server-level moduleGraph exposes invalidateAll() (defensive — Vite
602
+ // versions / preset configurations may differ in which graph carries
603
+ // the module-runner cache).
604
+ async function refreshTempRscEnv(): Promise<any | null> {
605
+ let tempRscEnv = await getOrCreateTempServer();
606
+ if (!tempRscEnv) return null;
607
+
608
+ // Module-runner cache is on the per-environment graph in Vite 6+;
609
+ // older / non-environments setups carry it on the server graph.
610
+ // Try env first, server second.
611
+ const envGraph = (tempRscEnv as any).moduleGraph;
612
+ const serverGraph = (prerenderTempServer as any)?.moduleGraph;
613
+ const target = envGraph?.invalidateAll
614
+ ? envGraph
615
+ : serverGraph?.invalidateAll
616
+ ? serverGraph
617
+ : null;
618
+
619
+ if (!target) {
620
+ // No invalidate method available — fall back to close+recreate.
621
+ // This preserves the previous behavior in case a Vite version
622
+ // doesn't expose invalidateAll on either graph.
623
+ debugDiscovery?.(
624
+ "refreshTempRscEnv: invalidateAll unavailable on env+server graphs, falling back to close+recreate",
625
+ );
626
+ if (prerenderTempServer) {
627
+ await prerenderTempServer.close().catch(() => {});
628
+ prerenderTempServer = null;
629
+ prerenderNodeRegistry = null;
630
+ }
631
+ return await getOrCreateTempServer();
632
+ }
633
+
634
+ debugDiscovery?.(
635
+ "refreshTempRscEnv: invalidating module graph (%s)",
636
+ envGraph?.invalidateAll ? "env" : "server",
637
+ );
638
+ target.invalidateAll();
639
+ // Drop the cached registry so importEntryAndRegistry re-reads it
640
+ // through the now-invalidated module runner.
641
+ prerenderNodeRegistry = null;
642
+ // Clear singleton Maps that Vite's moduleGraph invalidation can't
643
+ // reach (RouterRegistry / HostRouterRegistry). Without this, an
644
+ // edit that REMOVES a createRouter() call or CHANGES a router id
645
+ // would leave the old entry in the registry, and discoverRouters
646
+ // would still emit its routes alongside whatever the new source
647
+ // declares.
648
+ await clearTempRegistries(tempRscEnv);
649
+ await importEntryAndRegistry(tempRscEnv);
650
+ return tempRscEnv;
651
+ }
652
+
262
653
  const discover = async () => {
654
+ const discoverStart = performance.now();
263
655
  const rscEnv = (server.environments as any)?.rsc;
264
656
  if (!rscEnv?.runner) {
265
657
  // Cloudflare dev: no module runner available (workerd-based RSC env).
266
658
  // Set devServerOrigin so the virtual module can inject __PRERENDER_DEV_URL
267
659
  // for on-demand prerender via the /__rsc_prerender endpoint.
660
+ debugDiscovery?.(
661
+ "dev: cloudflare path start, __rscRouterDiscoveryActive=%s",
662
+ (globalThis as any).__rscRouterDiscoveryActive ?? false,
663
+ );
268
664
  s.devServerOrigin = getDevServerOrigin();
269
665
 
270
666
  // Create a temp Node.js server to run runtime discovery and generate
271
667
  // named route types (static parser can't resolve factory calls).
272
668
  try {
273
- const tempRscEnv = await getOrCreateTempServer();
669
+ // Acquire build-time env bindings for dev prerender
670
+ await timed(debugDiscovery, "acquireBuildEnv", () =>
671
+ acquireBuildEnv(s, viteCommand, viteMode),
672
+ );
673
+
674
+ const tempRscEnv = await timed(
675
+ debugDiscovery,
676
+ "getOrCreateTempServer",
677
+ () => getOrCreateTempServer(),
678
+ );
274
679
  if (tempRscEnv) {
275
- await discoverRouters(s, tempRscEnv);
276
- writeRouteTypesFiles(s);
680
+ await timed(debugDiscovery, "discoverRouters (cloudflare)", () =>
681
+ discoverRouters(s, tempRscEnv),
682
+ );
683
+ timedSync(debugDiscovery, "writeRouteTypesFiles", () =>
684
+ writeRouteTypesFiles(s),
685
+ );
277
686
  }
278
687
  } catch (err: any) {
279
688
  console.warn(
280
- `[rsc-router] Cloudflare dev discovery failed: ${err.message}\n${err.stack}`,
689
+ `[rango] Cloudflare dev discovery failed: ${err.message}\n${err.stack}`,
281
690
  );
282
691
  }
283
692
 
693
+ debugDiscovery?.(
694
+ "dev discovery done (%sms)",
695
+ (performance.now() - discoverStart).toFixed(1),
696
+ );
284
697
  resolveDiscovery!();
285
698
  return;
286
699
  }
287
700
 
288
701
  try {
702
+ // Acquire build-time env bindings for dev prerender (Node.js path)
703
+ debugDiscovery?.("dev: node path start");
704
+ await timed(debugDiscovery, "acquireBuildEnv", () =>
705
+ acquireBuildEnv(s, viteCommand, viteMode),
706
+ );
707
+
289
708
  // Set the readiness gate BEFORE discovery so early requests
290
709
  // block until manifest is populated
291
- const serverMod = await rscEnv.runner.import(
292
- "@rangojs/router/server",
710
+ const serverMod = await timed(
711
+ debugDiscovery,
712
+ "import @rangojs/router/server",
713
+ () => rscEnv.runner.import("@rangojs/router/server"),
293
714
  );
294
715
  if (serverMod?.setManifestReadyPromise) {
295
716
  serverMod.setManifestReadyPromise(discoveryPromise);
296
717
  }
297
718
 
298
- await discoverRouters(s, rscEnv);
719
+ await timed(debugDiscovery, "discoverRouters", () =>
720
+ discoverRouters(s, rscEnv),
721
+ );
299
722
 
300
723
  // Store server origin for dev prerender endpoint (virtual module injection)
301
724
  s.devServerOrigin = getDevServerOrigin();
@@ -305,24 +728,36 @@ export function createRouterDiscoveryPlugin(
305
728
  // routes (e.g. Array.from loops) that the static parser cannot see.
306
729
  // writeRouteTypesFiles() only writes when content changes, so this
307
730
  // won't cause unnecessary HMR triggers.
308
- writeRouteTypesFiles(s);
731
+ timedSync(debugDiscovery, "writeRouteTypesFiles", () =>
732
+ writeRouteTypesFiles(s),
733
+ );
309
734
 
310
735
  // Populate the route map and per-router data in the RSC env
311
- await propagateDiscoveryState(rscEnv);
736
+ await timed(debugDiscovery, "propagateDiscoveryState", () =>
737
+ propagateDiscoveryState(rscEnv),
738
+ );
312
739
  } catch (err: any) {
313
740
  console.warn(
314
- `[rsc-router] Router discovery failed: ${err.message}\n${err.stack}`,
741
+ `[rango] Router discovery failed: ${err.message}\n${err.stack}`,
315
742
  );
316
743
  } finally {
744
+ debugDiscovery?.(
745
+ "dev discovery done (%sms)",
746
+ (performance.now() - discoverStart).toFixed(1),
747
+ );
317
748
  resolveDiscovery!();
318
749
  }
319
750
  };
320
751
 
321
752
  // 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
- });
753
+ // The gate (s.discoveryDone) is reset via beginDiscoveryGate() and
754
+ // resolved when discover() finishes, so the virtual manifest module's
755
+ // load() awaits the populated state.
756
+ beginDiscoveryGate();
757
+ setTimeout(
758
+ () => discover().then(resolveDiscoveryGate, resolveDiscoveryGate),
759
+ 0,
760
+ );
326
761
 
327
762
  // Dev-mode on-demand prerender endpoint.
328
763
  // When workerd hits a prerender route, it fetches this endpoint instead of
@@ -362,24 +797,30 @@ export function createRouterDiscoveryPlugin(
362
797
  if (s.mergedRouteTrie && serverMod.setRouteTrie) {
363
798
  serverMod.setRouteTrie(s.mergedRouteTrie);
364
799
  }
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
- }
800
+ const perRouterSetters: Array<[Map<string, any>, string]> = [
801
+ [s.perRouterManifestDataMap, "setRouterManifest"],
802
+ [s.perRouterTrieMap, "setRouterTrie"],
803
+ [s.perRouterPrecomputedMap, "setRouterPrecomputedEntries"],
804
+ ];
805
+ for (const [map, fn] of perRouterSetters) {
806
+ const setter = serverMod[fn];
807
+ if (typeof setter !== "function") continue;
808
+ for (const [routerId, value] of map) setter(routerId, value);
379
809
  }
380
810
  };
381
811
 
382
812
  server.middlewares.use("/__rsc_prerender", async (req: any, res: any) => {
813
+ const reqStart = debugDev ? performance.now() : 0;
814
+ const logResult = (status: number, note: string) => {
815
+ debugDev?.(
816
+ "/__rsc_prerender %s -> %d %s (%sms)",
817
+ req.url,
818
+ status,
819
+ note,
820
+ (performance.now() - reqStart).toFixed(1),
821
+ );
822
+ };
823
+
383
824
  if (s.discoveryDone) await s.discoveryDone;
384
825
 
385
826
  const url = new URL(req.url || "/", "http://localhost");
@@ -387,12 +828,36 @@ export function createRouterDiscoveryPlugin(
387
828
  if (!pathname) {
388
829
  res.statusCode = 400;
389
830
  res.end("Missing pathname");
831
+ logResult(400, "missing pathname");
390
832
  return;
391
833
  }
392
834
 
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;
835
+ // Import the user's entry module to force re-evaluation of any
836
+ // HMR-invalidated modules in the chain (entry router urls handlers).
837
+ // This ensures createRouter() re-runs with updated handler code before
838
+ // we read RouterRegistry. Without this, edits to prerender handler files
839
+ // produce stale content because the old router instance remains registered.
840
+ const rscEnv = (server.environments as any)?.rsc;
841
+ let registry: Map<string, any> | null = null;
842
+ if (rscEnv?.runner && s.resolvedEntryPath) {
843
+ try {
844
+ await rscEnv.runner.import(s.resolvedEntryPath);
845
+ const serverMod = await rscEnv.runner.import(
846
+ "@rangojs/router/server",
847
+ );
848
+ registry = serverMod.RouterRegistry ?? null;
849
+ } catch (err: any) {
850
+ console.warn(
851
+ `[rango] Dev prerender module refresh failed: ${err.message}`,
852
+ );
853
+ res.statusCode = 500;
854
+ res.end(`Prerender handler error: ${err.message}`);
855
+ logResult(500, "module refresh failed");
856
+ return;
857
+ }
858
+ } else {
859
+ registry = mainRegistry;
860
+ }
396
861
 
397
862
  if (!registry) {
398
863
  // No main registry: the RSC env has no module runner (Cloudflare dev).
@@ -406,6 +871,7 @@ export function createRouterDiscoveryPlugin(
406
871
  if (!registry || registry.size === 0) {
407
872
  res.statusCode = 503;
408
873
  res.end("Prerender runner not available");
874
+ logResult(503, "no registry");
409
875
  return;
410
876
  }
411
877
 
@@ -421,6 +887,8 @@ export function createRouterDiscoveryPlugin(
421
887
  {},
422
888
  undefined,
423
889
  wantPassthrough,
890
+ s.resolvedBuildEnv,
891
+ true, // devMode: check getParams for passthrough routes
424
892
  );
425
893
  if (!result) continue;
426
894
  if (result.passthrough) continue;
@@ -433,25 +901,26 @@ export function createRouterDiscoveryPlugin(
433
901
  if (wantIntercept && result.interceptSegments?.length) {
434
902
  payload = {
435
903
  segments: [...result.segments, ...result.interceptSegments],
436
- handles: {
437
- ...result.handles,
438
- ...(result.interceptHandles || {}),
439
- },
904
+ // Pre-encoded MERGED handle string from the producer (handles are
905
+ // Flight-encoded so Promise/ReactNode values survive the wire).
906
+ handles: result.interceptHandles ?? "",
440
907
  };
441
908
  } else {
442
909
  payload = { segments: result.segments, handles: result.handles };
443
910
  }
444
911
  res.end(JSON.stringify(payload));
912
+ logResult(200, `match ${result.routeName}`);
445
913
  return;
446
914
  } catch (err: any) {
447
915
  console.warn(
448
- `[rsc-router] Dev prerender failed for ${pathname}: ${err.message}`,
916
+ `[rango] Dev prerender failed for ${pathname}: ${err.message}`,
449
917
  );
450
918
  }
451
919
  }
452
920
 
453
921
  res.statusCode = 404;
454
922
  res.end("No prerender match");
923
+ logResult(404, "no match");
455
924
  });
456
925
 
457
926
  // Watch url module and router files for changes and regenerate named-routes.gen.ts.
@@ -494,21 +963,135 @@ export function createRouterDiscoveryPlugin(
494
963
 
495
964
  // Re-run runtime discovery so factory-generated routes that the
496
965
  // static parser cannot see are refreshed after source changes.
497
- let runtimeRediscoveryInProgress = false;
966
+ // The state-machine concerns (queued/pending/gatePending) are
967
+ // owned by the gate created above (./discovery/gate-state.ts).
968
+ // Here we provide just the env-specific work.
498
969
  const refreshRuntimeDiscovery = async () => {
499
970
  const rscEnv = (server.environments as any)?.rsc;
500
- if (!rscEnv?.runner || runtimeRediscoveryInProgress) return;
501
- runtimeRediscoveryInProgress = true;
971
+ const hasMainRunner = !!rscEnv?.runner;
972
+ // Cloudflare HMR has no main RSC runner (workerd is a separate
973
+ // runtime). When we have a populated runtime manifest from cold
974
+ // start, we can re-discover via the temp Node runner — the same
975
+ // mechanism getOrCreateTempServer() uses at startup. Without a
976
+ // populated manifest there's nothing useful to do, so bail
977
+ // before involving the gate machine at all.
978
+ if (!hasMainRunner && s.perRouterManifests.length === 0) return;
979
+ await gate.runRefreshCycle(async () => {
980
+ const hmrStart = performance.now();
981
+ try {
982
+ if (hasMainRunner) {
983
+ await timed(debugDiscovery, "hmr discoverRouters", () =>
984
+ discoverRouters(s, rscEnv),
985
+ );
986
+ timedSync(debugDiscovery, "hmr writeRouteTypesFiles", () =>
987
+ writeRouteTypesFiles(s),
988
+ );
989
+ await timed(debugDiscovery, "hmr propagateDiscoveryState", () =>
990
+ propagateDiscoveryState(rscEnv),
991
+ );
992
+ } else {
993
+ // Cloudflare HMR: invalidate the temp server's RSC module
994
+ // graph (or close+recreate as a fallback) so the runner
995
+ // re-reads the freshly edited source. Keeping the same
996
+ // Vite instance alive avoids disrupting workerd's transport
997
+ // during the first post-cold-start module-fetch window.
998
+ const tempRscEnv = await timed(
999
+ debugDiscovery,
1000
+ "hmr refreshTempRscEnv (cloudflare)",
1001
+ () => refreshTempRscEnv(),
1002
+ );
1003
+ if (!tempRscEnv) {
1004
+ throw new Error(
1005
+ "temp runner unavailable for cloudflare HMR rediscovery",
1006
+ );
1007
+ }
1008
+ await timed(
1009
+ debugDiscovery,
1010
+ "hmr discoverRouters (cloudflare)",
1011
+ () => discoverRouters(s, tempRscEnv),
1012
+ );
1013
+ timedSync(debugDiscovery, "hmr writeRouteTypesFiles", () =>
1014
+ writeRouteTypesFiles(s),
1015
+ );
1016
+ }
1017
+ if (s.lastDiscoveryError) {
1018
+ debugDiscovery?.(
1019
+ "hmr: cleared lastDiscoveryError (%s) after successful rediscovery",
1020
+ s.lastDiscoveryError.message,
1021
+ );
1022
+ s.lastDiscoveryError = null;
1023
+ }
1024
+ // Cloudflare dev: on a successful cycle drop the workerd runner's
1025
+ // cached worker-entry chain so the next request re-evaluates
1026
+ // createRouter() with the new routes. Fired here in the work path
1027
+ // (not the caller's .then()) so a queued follow-up cycle that
1028
+ // succeeds after an earlier failed cycle still reloads:
1029
+ // runRefreshCycle recurses queued work without awaiting it, so the
1030
+ // original call already resolved on the failed cycle. A failed
1031
+ // cycle throws above and never reaches here, so a broken edit
1032
+ // never reloads the worker onto bad source.
1033
+ if (rscEnv && !rscEnv.runner) forceCloudflareWorkerReload(rscEnv);
1034
+ } catch (err: any) {
1035
+ s.lastDiscoveryError = {
1036
+ message: err?.message ?? String(err),
1037
+ at: Date.now(),
1038
+ };
1039
+ console.warn(
1040
+ `[rango] Runtime re-discovery failed: ${err.message}`,
1041
+ );
1042
+ debugDiscovery?.(
1043
+ "hmr: lastDiscoveryError set (%s) — manifest preserved at last-good; recovery mode active (any in-scan source change will trigger rediscovery)",
1044
+ err?.message,
1045
+ );
1046
+ } finally {
1047
+ debugDiscovery?.(
1048
+ "hmr re-discovery done (%sms)",
1049
+ (performance.now() - hmrStart).toFixed(1),
1050
+ );
1051
+ }
1052
+ });
1053
+ };
1054
+
1055
+ // Cloudflare dev only. workerd serves every request through the
1056
+ // runner-worker singleton, which re-resolves the worker entry per
1057
+ // request via runner.import("virtual:cloudflare/worker-entry"). The
1058
+ // route table lives in the user's createRouter() instance, captured
1059
+ // when that entry chain (entry -> router -> urls) was last evaluated
1060
+ // and then cached in the runner's evaluatedModules. The route-file
1061
+ // watcher refreshes discovery + types on the Node side, but the worker
1062
+ // keeps serving the cached (stale) router: route-definition modules
1063
+ // have no import.meta.hot boundary, so Vite never sends the worker an
1064
+ // HMR update for them and the entry chain is never evicted.
1065
+ //
1066
+ // Fix: after discovery completes, (1) invalidate the worker env's
1067
+ // Node-side module graph, then (2) send a full-reload to the worker.
1068
+ // Step (2) alone is insufficient: the full-reload handler clears the
1069
+ // runner's evaluatedModules and re-imports entrypoints, but each
1070
+ // re-import fetches the module back through this Node-side graph, which
1071
+ // still holds the pre-edit transform of urls.tsx — so createRouter()
1072
+ // rebuilds the stale route table and the new route 404s/hits the
1073
+ // catch-all. Invalidating the graph forces a fresh transform on
1074
+ // re-fetch (the same mechanism refreshTempRscEnv uses for discovery),
1075
+ // so the re-import re-runs createRouter() with the new routes. This is
1076
+ // the programmatic equivalent of the dev-server "r + enter" restart,
1077
+ // scoped to the worker environment instead of tearing down the server.
1078
+ const forceCloudflareWorkerReload = (rscEnv: any) => {
1079
+ if (!rscEnv?.hot) return;
502
1080
  try {
503
- await discoverRouters(s, rscEnv);
504
- writeRouteTypesFiles(s);
505
- await propagateDiscoveryState(rscEnv);
1081
+ const graph = rscEnv.moduleGraph;
1082
+ if (graph?.invalidateAll) {
1083
+ graph.invalidateAll();
1084
+ debugDiscovery?.("hmr: invalidated workerd rsc module graph");
1085
+ }
1086
+ rscEnv.hot.send({ type: "full-reload" });
1087
+ debugDiscovery?.(
1088
+ "hmr: forced workerd rsc env reload (full-reload)",
1089
+ );
506
1090
  } catch (err: any) {
507
- console.warn(
508
- `[rsc-router] Runtime re-discovery failed: ${err.message}`,
1091
+ debugDiscovery?.(
1092
+ "hmr: workerd reload failed: %s",
1093
+ err?.message ?? err,
509
1094
  );
510
- } finally {
511
- runtimeRediscoveryInProgress = false;
512
1095
  }
513
1096
  };
514
1097
 
@@ -516,23 +1099,50 @@ export function createRouterDiscoveryPlugin(
516
1099
  clearTimeout(routeChangeTimer);
517
1100
  routeChangeTimer = setTimeout(() => {
518
1101
  routeChangeTimer = undefined;
1102
+ const regenStart = debugDiscovery ? performance.now() : 0;
1103
+ const rscEnv = (server.environments as any)?.rsc;
1104
+ const skipStaticWrite =
1105
+ !rscEnv?.runner && s.perRouterManifests.length > 0;
519
1106
  try {
520
- writeCombinedRouteTypesWithTracking(s);
521
- if (s.perRouterManifests.length > 0) {
522
- supplementGenFilesWithRuntimeRoutes(s);
1107
+ // In cloudflare dev with a populated runtime manifest, the
1108
+ // static parser produces a strictly smaller (and actively
1109
+ // wrong) gen file — supplementGenFilesWithRuntimeRoutes can
1110
+ // only restore factory-only prefixes, and apps with mixed
1111
+ // static+factory routes under shared prefixes (cf-stress)
1112
+ // collapse to the 19-route static view. Skip the static
1113
+ // write entirely; runtime rediscovery below will overwrite
1114
+ // the gen file with the authoritative manifest.
1115
+ if (skipStaticWrite) {
1116
+ debugDiscovery?.(
1117
+ "watcher: skipping static write (cloudflare HMR — runtime rediscovery owns gen file)",
1118
+ );
1119
+ } else {
1120
+ writeCombinedRouteTypesWithTracking(s);
1121
+ if (s.perRouterManifests.length > 0) {
1122
+ supplementGenFilesWithRuntimeRoutes(s);
1123
+ }
523
1124
  }
524
1125
  } catch (err: any) {
525
- console.error(
526
- `[rsc-router] Route regeneration error: ${err.message}`,
527
- );
1126
+ console.error(`[rango] Route regeneration error: ${err.message}`);
528
1127
  }
1128
+ debugDiscovery?.(
1129
+ "watcher: regenerated gen files (%sms)",
1130
+ (performance.now() - regenStart).toFixed(1),
1131
+ );
529
1132
  // Async: re-run runtime discovery to refresh factory-generated
530
- // routes that the static parser cannot resolve.
1133
+ // routes that the static parser cannot resolve. Resolves the
1134
+ // discovery gate when complete.
531
1135
  if (s.perRouterManifests.length > 0) {
1136
+ // The cloudflare workerd reload fires inside refreshRuntimeDiscovery
1137
+ // on the successful cycle (see forceCloudflareWorkerReload call
1138
+ // there) so queued follow-up cycles also trigger it.
532
1139
  refreshRuntimeDiscovery().catch((err: any) => {
533
1140
  console.warn(
534
- `[rsc-router] Runtime re-discovery error: ${err.message}`,
1141
+ `[rango] Runtime re-discovery error: ${err.message}`,
535
1142
  );
1143
+ // Even on error, unblock the gate so workerd's reload doesn't
1144
+ // hang indefinitely against the previous manifest.
1145
+ resolveDiscoveryGate();
536
1146
  });
537
1147
  }
538
1148
  }, 100);
@@ -545,21 +1155,74 @@ export function createRouterDiscoveryPlugin(
545
1155
  !filePath.endsWith(".tsx") &&
546
1156
  !filePath.endsWith(".js") &&
547
1157
  !filePath.endsWith(".jsx")
548
- )
1158
+ ) {
1159
+ if (s.lastDiscoveryError) {
1160
+ debugDiscovery?.(
1161
+ "watcher: skip non-source %s [LASTERR %s]",
1162
+ filePath,
1163
+ s.lastDiscoveryError.message,
1164
+ );
1165
+ }
549
1166
  return;
1167
+ }
550
1168
  // Apply scan filter as early-exit before reading file
551
- if (s.scanFilter && !s.scanFilter(filePath)) return;
1169
+ if (s.scanFilter && !s.scanFilter(filePath)) {
1170
+ if (s.lastDiscoveryError) {
1171
+ debugDiscovery?.(
1172
+ "watcher: skip scan-filter %s [LASTERR %s]",
1173
+ filePath,
1174
+ s.lastDiscoveryError.message,
1175
+ );
1176
+ }
1177
+ return;
1178
+ }
1179
+ // Recovery mode: when the previous HMR re-discovery failed, the
1180
+ // import graph is incomplete and the manifest is stuck at the
1181
+ // last-good state. The fix may land in a non-route file (e.g. a
1182
+ // helper imported by the router, a missing module being created,
1183
+ // or a "use client" component) that the narrow content sniff
1184
+ // would otherwise filter out. While in recovery, treat any
1185
+ // in-scan source change as a candidate for rediscovery; the
1186
+ // tighter filter resumes once discovery succeeds again.
1187
+ const inRecoveryMode = !!s.lastDiscoveryError;
552
1188
  try {
553
1189
  const source = readFileSync(filePath, "utf-8");
554
1190
  const trimmed = source.trimStart();
555
- if (
1191
+ const isUseClient =
556
1192
  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;
1193
+ trimmed.startsWith("'use client'");
1194
+ if (!inRecoveryMode && isUseClient) return;
1195
+ // Cheap raw pre-check first; only when a candidate token is present
1196
+ // do we confirm it occurs in real code (not a comment/string) via a
1197
+ // single allocation-free code-region scan. Most saved files contain
1198
+ // neither token and skip the scan entirely. This avoids a comment or
1199
+ // string mention spuriously marking a file relevant and triggering an
1200
+ // unnecessary re-discovery on save.
1201
+ let hasUrls = source.includes("urls(");
1202
+ let hasCreateRouter = /\bcreateRouter\s*[<(]/.test(source);
1203
+ if (hasUrls) hasUrls = firstCodeMatchIndex(source, /urls\(/g) >= 0;
1204
+ if (hasCreateRouter) {
1205
+ hasCreateRouter =
1206
+ firstCodeMatchIndex(source, /\bcreateRouter\s*[<(]/g) >= 0;
1207
+ }
1208
+ if (!inRecoveryMode && !hasUrls && !hasCreateRouter) return;
1209
+ if (inRecoveryMode) {
1210
+ debugDiscovery?.(
1211
+ "watcher: recovery rediscovery for %s (urls=%s, router=%s, useClient=%s) [LASTERR %s]",
1212
+ filePath,
1213
+ hasUrls,
1214
+ hasCreateRouter,
1215
+ isUseClient,
1216
+ s.lastDiscoveryError!.message,
1217
+ );
1218
+ } else {
1219
+ debugDiscovery?.(
1220
+ "watcher: %s matches (urls=%s, router=%s)",
1221
+ filePath,
1222
+ hasUrls,
1223
+ hasCreateRouter,
1224
+ );
1225
+ }
563
1226
  // Invalidate cache when a router file changes (new router added/removed)
564
1227
  if (hasCreateRouter) {
565
1228
  const nestedRouterConflict = findNestedRouterConflict([
@@ -574,8 +1237,27 @@ export function createRouterDiscoveryPlugin(
574
1237
  }
575
1238
  s.cachedRouterFiles = undefined;
576
1239
  }
1240
+ // Note the event in the gate machine IMMEDIATELY (before the
1241
+ // 100ms debounce and any downstream HMR fanout). This sets
1242
+ // both `pendingEvents` (so refresh's finally holds the gate
1243
+ // through the tail window even if no rediscovery is queued)
1244
+ // and resets `discoveryDone` to a fresh pending promise (so
1245
+ // workerd reloads triggered by the same source change can't
1246
+ // observe a stale resolved gate from cold-start). Resolved
1247
+ // by the trailing refreshRuntimeDiscovery() cycle.
1248
+ if (s.perRouterManifests.length > 0) {
1249
+ gate.noteRouteEvent();
1250
+ }
577
1251
  scheduleRouteRegeneration();
578
- } catch {
1252
+ } catch (readErr: any) {
1253
+ if (s.lastDiscoveryError) {
1254
+ debugDiscovery?.(
1255
+ "watcher: read error %s: %s [LASTERR %s]",
1256
+ filePath,
1257
+ readErr?.message,
1258
+ s.lastDiscoveryError.message,
1259
+ );
1260
+ }
579
1261
  // Ignore read errors for deleted/moved files
580
1262
  }
581
1263
  };
@@ -604,11 +1286,24 @@ export function createRouterDiscoveryPlugin(
604
1286
  async buildStart() {
605
1287
  if (!s.isBuildMode) return;
606
1288
  // Only run once across environment builds
607
- if (s.mergedRouteManifest !== null) return;
1289
+ if (s.mergedRouteManifest !== null) {
1290
+ debugDiscovery?.(
1291
+ "build: skip (already discovered, env=%s)",
1292
+ this.environment?.name ?? "?",
1293
+ );
1294
+ return;
1295
+ }
1296
+ const buildStartTime = performance.now();
1297
+ debugDiscovery?.("build: start (env=%s)", this.environment?.name ?? "?");
608
1298
  resetStagedBuildAssets(s.projectRoot);
609
1299
  s.prerenderManifestEntries = null;
610
1300
  s.staticManifestEntries = null;
611
1301
 
1302
+ // Acquire build-time env bindings if configured
1303
+ await timed(debugDiscovery, "build acquireBuildEnv", () =>
1304
+ acquireBuildEnv(s, viteCommand, viteMode),
1305
+ );
1306
+
612
1307
  let tempServer: any = null;
613
1308
  // Signal to user-space code (e.g. reverse.ts) that build-time discovery
614
1309
  // is active. Uses globalThis because the temp server's module runner
@@ -616,12 +1311,16 @@ export function createRouterDiscoveryPlugin(
616
1311
  // between the vite plugin and user code loaded via runner.import().
617
1312
  (globalThis as any).__rscRouterDiscoveryActive = true;
618
1313
  try {
619
- tempServer = await createTempRscServer(s, { forceBuild: true });
1314
+ tempServer = await timed(
1315
+ debugDiscovery,
1316
+ "build createTempRscServer",
1317
+ () => createTempRscServer(s, { forceBuild: true }),
1318
+ );
620
1319
 
621
1320
  const rscEnv = (tempServer.environments as any)?.rsc;
622
1321
  if (!rscEnv?.runner) {
623
1322
  console.warn(
624
- "[rsc-router] RSC environment runner not available during build, skipping manifest generation",
1323
+ "[rango] RSC environment runner not available during build, skipping manifest generation",
625
1324
  );
626
1325
  return;
627
1326
  }
@@ -636,11 +1335,15 @@ export function createRouterDiscoveryPlugin(
636
1335
  s.resolvedStaticModules = tempIdsPlugin.api.staticHandlerModules;
637
1336
  }
638
1337
 
639
- await discoverRouters(s, rscEnv);
1338
+ await timed(debugDiscovery, "build discoverRouters", () =>
1339
+ discoverRouters(s, rscEnv),
1340
+ );
640
1341
  // Update named-routes.gen.ts from runtime discovery.
641
1342
  // The runtime manifest includes dynamically generated routes
642
1343
  // that the static parser cannot extract from source code.
643
- writeRouteTypesFiles(s);
1344
+ timedSync(debugDiscovery, "build writeRouteTypesFiles", () =>
1345
+ writeRouteTypesFiles(s),
1346
+ );
644
1347
  } catch (err: any) {
645
1348
  // Extract the user source file from the stack trace (skip internal frames)
646
1349
  const sourceFile = err.stack
@@ -660,13 +1363,50 @@ export function createRouterDiscoveryPlugin(
660
1363
  .filter(Boolean)
661
1364
  .join("\n");
662
1365
  throw new Error(
663
- `[rsc-router] Build-time router discovery failed:\n${details}`,
1366
+ `[rango] Build-time router discovery failed:\n${details}`,
1367
+ { cause: err },
664
1368
  );
665
1369
  } finally {
666
1370
  delete (globalThis as any).__rscRouterDiscoveryActive;
667
1371
  if (tempServer) {
668
- await tempServer.close();
1372
+ await timed(debugDiscovery, "build tempServer.close", () =>
1373
+ tempServer.close(),
1374
+ );
669
1375
  }
1376
+ await releaseBuildEnv(s);
1377
+ debugDiscovery?.(
1378
+ "build discovery done (%sms)",
1379
+ (performance.now() - buildStartTime).toFixed(1),
1380
+ );
1381
+ }
1382
+ },
1383
+
1384
+ // Suppress vite's HMR cascade for our own gen-file writes.
1385
+ //
1386
+ // After every cf HMR cycle, refreshTempRscEnv → writeRouteTypesFiles
1387
+ // writes the configured gen files (default `router.named-routes.gen.ts`,
1388
+ // but the source filenames and gen suffix are user-configurable). The
1389
+ // chokidar watcher then fires twice independently: our
1390
+ // `handleRouteFileChange` (already short-circuited by
1391
+ // `consumeSelfGenWrite` inside `maybeHandleGeneratedRouteFileMutation`),
1392
+ // AND vite's own HMR pipeline (which invalidates the gen file's
1393
+ // importers and triggers a second workerd full reload — visible to the
1394
+ // user as a duplicate "[Rango] HMR: version changed" on the client).
1395
+ //
1396
+ // `peekSelfGenWrite` is the authoritative filter: its map only contains
1397
+ // paths that `markSelfGenWrite` has registered, so it natively works
1398
+ // for any configured gen-file name. It is non-consuming so the chokidar
1399
+ // handler that fires later can still consume the same entry. Returning
1400
+ // [] tells vite "no modules invalidated by this change" — safe because
1401
+ // `s.perRouterManifests` is already up-to-date (the write that just
1402
+ // happened is the consequence of our just-completed rediscovery).
1403
+ handleHotUpdate(ctx) {
1404
+ if (peekSelfGenWrite(s, ctx.file)) {
1405
+ debugDiscovery?.(
1406
+ "handleHotUpdate: suppressing self-write HMR cascade for %s",
1407
+ ctx.file,
1408
+ );
1409
+ return [];
670
1410
  }
671
1411
  },
672
1412
 
@@ -690,19 +1430,38 @@ export function createRouterDiscoveryPlugin(
690
1430
  // This is critical for Cloudflare dev where the worker runs in a separate
691
1431
  // Miniflare process and can only receive manifest data via the virtual module.
692
1432
  if (s.discoveryDone) {
693
- await s.discoveryDone;
1433
+ await timed(
1434
+ debugRoutes,
1435
+ "await discoveryDone (manifest)",
1436
+ () => s.discoveryDone,
1437
+ );
694
1438
  }
695
- return generateRoutesManifestModule(s);
1439
+ const code = await timed(
1440
+ debugRoutes,
1441
+ "generateRoutesManifestModule",
1442
+ () => generateRoutesManifestModule(s),
1443
+ );
1444
+ debugRoutes?.("manifest module emitted (%d bytes)", code?.length ?? 0);
1445
+ return code;
696
1446
  }
697
1447
  // Per-router virtual modules: pure data exports (no side effects).
698
1448
  // ensureRouterManifest() imports the module and stores the data.
699
1449
  const perRouterPrefix = "\0" + VIRTUAL_ROUTES_MANIFEST_ID + "/";
700
1450
  if (id.startsWith(perRouterPrefix)) {
701
1451
  if (s.discoveryDone) {
702
- await s.discoveryDone;
1452
+ await timed(
1453
+ debugRoutes,
1454
+ "await discoveryDone (per-router)",
1455
+ () => s.discoveryDone,
1456
+ );
703
1457
  }
704
1458
  const routerId = id.slice(perRouterPrefix.length);
705
- return generatePerRouterModule(s, routerId);
1459
+ const code = await timed(
1460
+ debugRoutes,
1461
+ `generatePerRouterModule ${routerId}`,
1462
+ () => generatePerRouterModule(s, routerId),
1463
+ );
1464
+ return code;
706
1465
  }
707
1466
  // virtual:rsc-router/prerender-paths load handler removed
708
1467
  return null;
@@ -712,6 +1471,7 @@ export function createRouterDiscoveryPlugin(
712
1471
  // Used by closeBundle for handler code eviction and prerender data injection.
713
1472
  generateBundle(_options: any, bundle: any) {
714
1473
  if (this.environment?.name !== "rsc") return;
1474
+ const genStart = debugBuild ? performance.now() : 0;
715
1475
 
716
1476
  // Record RSC entry chunk filename for closeBundle injection
717
1477
  for (const [fileName, chunk] of Object.entries(bundle) as [
@@ -724,8 +1484,19 @@ export function createRouterDiscoveryPlugin(
724
1484
  }
725
1485
  }
726
1486
 
727
- if (!s.resolvedPrerenderModules?.size && !s.resolvedStaticModules?.size)
1487
+ if (!s.resolvedPrerenderModules?.size && !s.resolvedStaticModules?.size) {
1488
+ debugBuild?.(
1489
+ "generateBundle (rsc): no handlers to scan (%sms)",
1490
+ (performance.now() - genStart).toFixed(1),
1491
+ );
728
1492
  return;
1493
+ }
1494
+
1495
+ // Clear maps at the start of each RSC generateBundle pass.
1496
+ // Vite 6 multi-environment builds run RSC twice (analysis + production);
1497
+ // clearing prevents stale/duplicate records from the analysis pass.
1498
+ s.handlerChunkInfoMap.clear();
1499
+ s.staticHandlerChunkInfoMap.clear();
729
1500
 
730
1501
  for (const [fileName, chunk] of Object.entries(bundle) as [
731
1502
  string,
@@ -733,27 +1504,28 @@ export function createRouterDiscoveryPlugin(
733
1504
  ][]) {
734
1505
  if (chunk.type !== "chunk") continue;
735
1506
 
736
- // Prerender handlers chunk
737
- if (
738
- fileName.includes("__prerender-handlers") &&
739
- s.resolvedPrerenderModules?.size
740
- ) {
1507
+ // Scan all chunks for handler exports (handlers may land in any chunk)
1508
+ if (s.resolvedPrerenderModules?.size) {
741
1509
  const handlers = extractHandlerExportsFromChunk(
742
1510
  chunk.code,
743
1511
  s.resolvedPrerenderModules,
744
1512
  "Prerender",
745
- true,
1513
+ false,
746
1514
  );
747
1515
  if (handlers.length > 0) {
748
- s.handlerChunkInfo = { fileName, exports: handlers };
1516
+ const existing = s.handlerChunkInfoMap.get(fileName);
1517
+ if (existing) {
1518
+ existing.exports.push(...handlers);
1519
+ } else {
1520
+ s.handlerChunkInfoMap.set(fileName, {
1521
+ fileName,
1522
+ exports: handlers,
1523
+ });
1524
+ }
749
1525
  }
750
1526
  }
751
1527
 
752
- // Static handlers chunk
753
- if (
754
- fileName.includes("__static-handlers") &&
755
- s.resolvedStaticModules?.size
756
- ) {
1528
+ if (s.resolvedStaticModules?.size) {
757
1529
  const handlers = extractHandlerExportsFromChunk(
758
1530
  chunk.code,
759
1531
  s.resolvedStaticModules,
@@ -761,10 +1533,26 @@ export function createRouterDiscoveryPlugin(
761
1533
  false,
762
1534
  );
763
1535
  if (handlers.length > 0) {
764
- s.staticHandlerChunkInfo = { fileName, exports: handlers };
1536
+ const existing = s.staticHandlerChunkInfoMap.get(fileName);
1537
+ if (existing) {
1538
+ existing.exports.push(...handlers);
1539
+ } else {
1540
+ s.staticHandlerChunkInfoMap.set(fileName, {
1541
+ fileName,
1542
+ exports: handlers,
1543
+ });
1544
+ }
765
1545
  }
766
1546
  }
767
1547
  }
1548
+
1549
+ debugBuild?.(
1550
+ "generateBundle (rsc): scanned %d chunks, %d prerender chunk(s), %d static chunk(s) (%sms)",
1551
+ Object.keys(bundle).length,
1552
+ s.handlerChunkInfoMap.size,
1553
+ s.staticHandlerChunkInfoMap.size,
1554
+ (performance.now() - genStart).toFixed(1),
1555
+ );
768
1556
  },
769
1557
 
770
1558
  // Build-time pre-rendering: evict handler code and inject collected prerender data.
@@ -778,7 +1566,9 @@ export function createRouterDiscoveryPlugin(
778
1566
  // Only run for the RSC environment — other environments (client, ssr) have
779
1567
  // no prerender/static data to process and would just do redundant file I/O.
780
1568
  if (this.environment && this.environment.name !== "rsc") return;
781
- postprocessBundle(s);
1569
+ timedSync(debugBuild, "closeBundle postprocessBundle", () =>
1570
+ postprocessBundle(s),
1571
+ );
782
1572
  },
783
1573
  },
784
1574
  };