@rangojs/router 0.0.0-experimental.13 → 0.0.0-experimental.131

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 (1214) hide show
  1. package/AGENTS.md +9 -0
  2. package/README.md +1040 -5
  3. package/dist/__internal.d.ts +83 -0
  4. package/dist/__internal.d.ts.map +1 -0
  5. package/dist/__internal.js +19 -0
  6. package/dist/__internal.js.map +1 -0
  7. package/dist/__mocks__/version.d.ts +7 -0
  8. package/dist/__mocks__/version.d.ts.map +1 -0
  9. package/dist/__mocks__/version.js +7 -0
  10. package/dist/__mocks__/version.js.map +1 -0
  11. package/dist/__tests__/client-href.test.d.ts +2 -0
  12. package/dist/__tests__/client-href.test.d.ts.map +1 -0
  13. package/dist/__tests__/client-href.test.js +74 -0
  14. package/dist/__tests__/client-href.test.js.map +1 -0
  15. package/dist/__tests__/component-utils.test.d.ts +2 -0
  16. package/dist/__tests__/component-utils.test.d.ts.map +1 -0
  17. package/dist/__tests__/component-utils.test.js +51 -0
  18. package/dist/__tests__/component-utils.test.js.map +1 -0
  19. package/dist/__tests__/event-controller.test.d.ts +2 -0
  20. package/dist/__tests__/event-controller.test.d.ts.map +1 -0
  21. package/dist/__tests__/event-controller.test.js +538 -0
  22. package/dist/__tests__/event-controller.test.js.map +1 -0
  23. package/dist/__tests__/helpers/route-tree.d.ts +118 -0
  24. package/dist/__tests__/helpers/route-tree.d.ts.map +1 -0
  25. package/dist/__tests__/helpers/route-tree.js +374 -0
  26. package/dist/__tests__/helpers/route-tree.js.map +1 -0
  27. package/dist/__tests__/match-result.test.d.ts +2 -0
  28. package/dist/__tests__/match-result.test.d.ts.map +1 -0
  29. package/dist/__tests__/match-result.test.js +154 -0
  30. package/dist/__tests__/match-result.test.js.map +1 -0
  31. package/dist/__tests__/navigation-store.test.d.ts +2 -0
  32. package/dist/__tests__/navigation-store.test.d.ts.map +1 -0
  33. package/dist/__tests__/navigation-store.test.js +440 -0
  34. package/dist/__tests__/navigation-store.test.js.map +1 -0
  35. package/dist/__tests__/partial-update.test.d.ts +2 -0
  36. package/dist/__tests__/partial-update.test.d.ts.map +1 -0
  37. package/dist/__tests__/partial-update.test.js +1009 -0
  38. package/dist/__tests__/partial-update.test.js.map +1 -0
  39. package/dist/__tests__/reverse-types.test.d.ts +8 -0
  40. package/dist/__tests__/reverse-types.test.d.ts.map +1 -0
  41. package/dist/__tests__/reverse-types.test.js +656 -0
  42. package/dist/__tests__/reverse-types.test.js.map +1 -0
  43. package/dist/__tests__/route-definition.test.d.ts +2 -0
  44. package/dist/__tests__/route-definition.test.d.ts.map +1 -0
  45. package/dist/__tests__/route-definition.test.js +55 -0
  46. package/dist/__tests__/route-definition.test.js.map +1 -0
  47. package/dist/__tests__/router-helpers.test.d.ts +2 -0
  48. package/dist/__tests__/router-helpers.test.d.ts.map +1 -0
  49. package/dist/__tests__/router-helpers.test.js +377 -0
  50. package/dist/__tests__/router-helpers.test.js.map +1 -0
  51. package/dist/__tests__/router-integration-2.test.d.ts +2 -0
  52. package/dist/__tests__/router-integration-2.test.d.ts.map +1 -0
  53. package/dist/__tests__/router-integration-2.test.js +426 -0
  54. package/dist/__tests__/router-integration-2.test.js.map +1 -0
  55. package/dist/__tests__/router-integration.test.d.ts +2 -0
  56. package/dist/__tests__/router-integration.test.d.ts.map +1 -0
  57. package/dist/__tests__/router-integration.test.js +1051 -0
  58. package/dist/__tests__/router-integration.test.js.map +1 -0
  59. package/dist/__tests__/search-params.test.d.ts +5 -0
  60. package/dist/__tests__/search-params.test.d.ts.map +1 -0
  61. package/dist/__tests__/search-params.test.js +306 -0
  62. package/dist/__tests__/search-params.test.js.map +1 -0
  63. package/dist/__tests__/segment-system.test.d.ts +2 -0
  64. package/dist/__tests__/segment-system.test.d.ts.map +1 -0
  65. package/dist/__tests__/segment-system.test.js +627 -0
  66. package/dist/__tests__/segment-system.test.js.map +1 -0
  67. package/dist/__tests__/static-handler-types.test.d.ts +8 -0
  68. package/dist/__tests__/static-handler-types.test.d.ts.map +1 -0
  69. package/dist/__tests__/static-handler-types.test.js +63 -0
  70. package/dist/__tests__/static-handler-types.test.js.map +1 -0
  71. package/dist/__tests__/urls.test.d.ts +2 -0
  72. package/dist/__tests__/urls.test.d.ts.map +1 -0
  73. package/dist/__tests__/urls.test.js +421 -0
  74. package/dist/__tests__/urls.test.js.map +1 -0
  75. package/dist/__tests__/use-mount.test.d.ts +2 -0
  76. package/dist/__tests__/use-mount.test.d.ts.map +1 -0
  77. package/dist/__tests__/use-mount.test.js +35 -0
  78. package/dist/__tests__/use-mount.test.js.map +1 -0
  79. package/dist/bin/rango.d.ts +2 -0
  80. package/dist/bin/rango.d.ts.map +1 -0
  81. package/dist/bin/rango.js +1712 -212
  82. package/dist/bin/rango.js.map +1 -0
  83. package/dist/browser/event-controller.d.ts +191 -0
  84. package/dist/browser/event-controller.d.ts.map +1 -0
  85. package/dist/browser/event-controller.js +559 -0
  86. package/dist/browser/event-controller.js.map +1 -0
  87. package/dist/browser/index.d.ts +2 -0
  88. package/dist/browser/index.d.ts.map +1 -0
  89. package/dist/browser/index.js +14 -0
  90. package/dist/browser/index.js.map +1 -0
  91. package/dist/browser/link-interceptor.d.ts +38 -0
  92. package/dist/browser/link-interceptor.d.ts.map +1 -0
  93. package/dist/browser/link-interceptor.js +99 -0
  94. package/dist/browser/link-interceptor.js.map +1 -0
  95. package/dist/browser/logging.d.ts +10 -0
  96. package/dist/browser/logging.d.ts.map +1 -0
  97. package/dist/browser/logging.js +29 -0
  98. package/dist/browser/logging.js.map +1 -0
  99. package/dist/browser/lru-cache.d.ts +17 -0
  100. package/dist/browser/lru-cache.d.ts.map +1 -0
  101. package/dist/browser/lru-cache.js +50 -0
  102. package/dist/browser/lru-cache.js.map +1 -0
  103. package/dist/browser/merge-segment-loaders.d.ts +39 -0
  104. package/dist/browser/merge-segment-loaders.d.ts.map +1 -0
  105. package/dist/browser/merge-segment-loaders.js +102 -0
  106. package/dist/browser/merge-segment-loaders.js.map +1 -0
  107. package/dist/browser/navigation-bridge.d.ts +102 -0
  108. package/dist/browser/navigation-bridge.d.ts.map +1 -0
  109. package/dist/browser/navigation-bridge.js +708 -0
  110. package/dist/browser/navigation-bridge.js.map +1 -0
  111. package/dist/browser/navigation-client.d.ts +25 -0
  112. package/dist/browser/navigation-client.d.ts.map +1 -0
  113. package/dist/browser/navigation-client.js +157 -0
  114. package/dist/browser/navigation-client.js.map +1 -0
  115. package/dist/browser/navigation-store.d.ts +101 -0
  116. package/dist/browser/navigation-store.d.ts.map +1 -0
  117. package/dist/browser/navigation-store.js +625 -0
  118. package/dist/browser/navigation-store.js.map +1 -0
  119. package/dist/browser/partial-update.d.ts +75 -0
  120. package/dist/browser/partial-update.d.ts.map +1 -0
  121. package/dist/browser/partial-update.js +426 -0
  122. package/dist/browser/partial-update.js.map +1 -0
  123. package/dist/browser/react/Link.d.ts +86 -0
  124. package/dist/browser/react/Link.d.ts.map +1 -0
  125. package/dist/browser/react/Link.js +128 -0
  126. package/dist/browser/react/Link.js.map +1 -0
  127. package/dist/browser/react/NavigationProvider.d.ts +63 -0
  128. package/dist/browser/react/NavigationProvider.d.ts.map +1 -0
  129. package/dist/browser/react/NavigationProvider.js +216 -0
  130. package/dist/browser/react/NavigationProvider.js.map +1 -0
  131. package/dist/browser/react/ScrollRestoration.d.ts +75 -0
  132. package/dist/browser/react/ScrollRestoration.d.ts.map +1 -0
  133. package/dist/browser/react/ScrollRestoration.js +57 -0
  134. package/dist/browser/react/ScrollRestoration.js.map +1 -0
  135. package/dist/browser/react/context.d.ts +46 -0
  136. package/dist/browser/react/context.d.ts.map +1 -0
  137. package/dist/browser/react/context.js +10 -0
  138. package/dist/browser/react/context.js.map +1 -0
  139. package/dist/browser/react/index.d.ts +11 -0
  140. package/dist/browser/react/index.d.ts.map +1 -0
  141. package/dist/browser/react/index.js +22 -0
  142. package/dist/browser/react/index.js.map +1 -0
  143. package/dist/browser/react/location-state-shared.d.ts +63 -0
  144. package/dist/browser/react/location-state-shared.d.ts.map +1 -0
  145. package/dist/browser/react/location-state-shared.js +81 -0
  146. package/dist/browser/react/location-state-shared.js.map +1 -0
  147. package/dist/browser/react/location-state.d.ts +23 -0
  148. package/dist/browser/react/location-state.d.ts.map +1 -0
  149. package/dist/browser/react/location-state.js +29 -0
  150. package/dist/browser/react/location-state.js.map +1 -0
  151. package/dist/browser/react/mount-context.d.ts +24 -0
  152. package/dist/browser/react/mount-context.d.ts.map +1 -0
  153. package/dist/browser/react/mount-context.js +24 -0
  154. package/dist/browser/react/mount-context.js.map +1 -0
  155. package/dist/browser/react/use-action.d.ts +64 -0
  156. package/dist/browser/react/use-action.d.ts.map +1 -0
  157. package/dist/browser/react/use-action.js +134 -0
  158. package/dist/browser/react/use-action.js.map +1 -0
  159. package/dist/browser/react/use-client-cache.d.ts +41 -0
  160. package/dist/browser/react/use-client-cache.d.ts.map +1 -0
  161. package/{src/browser/react/use-client-cache.ts → dist/browser/react/use-client-cache.js} +9 -26
  162. package/dist/browser/react/use-client-cache.js.map +1 -0
  163. package/dist/browser/react/use-handle.d.ts +31 -0
  164. package/dist/browser/react/use-handle.d.ts.map +1 -0
  165. package/dist/browser/react/use-handle.js +144 -0
  166. package/dist/browser/react/use-handle.js.map +1 -0
  167. package/dist/browser/react/use-href.d.ts +33 -0
  168. package/dist/browser/react/use-href.d.ts.map +1 -0
  169. package/dist/browser/react/use-href.js +39 -0
  170. package/dist/browser/react/use-href.js.map +1 -0
  171. package/dist/browser/react/use-link-status.d.ts +37 -0
  172. package/dist/browser/react/use-link-status.d.ts.map +1 -0
  173. package/dist/browser/react/use-link-status.js +99 -0
  174. package/dist/browser/react/use-link-status.js.map +1 -0
  175. package/dist/browser/react/use-mount.d.ts +25 -0
  176. package/dist/browser/react/use-mount.d.ts.map +1 -0
  177. package/dist/browser/react/use-mount.js +30 -0
  178. package/dist/browser/react/use-mount.js.map +1 -0
  179. package/dist/browser/react/use-navigation.d.ts +27 -0
  180. package/dist/browser/react/use-navigation.d.ts.map +1 -0
  181. package/dist/browser/react/use-navigation.js +87 -0
  182. package/dist/browser/react/use-navigation.js.map +1 -0
  183. package/dist/browser/react/use-segments.d.ts +38 -0
  184. package/dist/browser/react/use-segments.d.ts.map +1 -0
  185. package/dist/browser/react/use-segments.js +130 -0
  186. package/dist/browser/react/use-segments.js.map +1 -0
  187. package/dist/browser/request-controller.d.ts +26 -0
  188. package/dist/browser/request-controller.d.ts.map +1 -0
  189. package/dist/browser/request-controller.js +147 -0
  190. package/dist/browser/request-controller.js.map +1 -0
  191. package/dist/browser/rsc-router.d.ts +129 -0
  192. package/dist/browser/rsc-router.d.ts.map +1 -0
  193. package/dist/browser/rsc-router.js +195 -0
  194. package/dist/browser/rsc-router.js.map +1 -0
  195. package/dist/browser/scroll-restoration.d.ts +93 -0
  196. package/dist/browser/scroll-restoration.d.ts.map +1 -0
  197. package/dist/browser/scroll-restoration.js +321 -0
  198. package/dist/browser/scroll-restoration.js.map +1 -0
  199. package/dist/browser/segment-structure-assert.d.ts +17 -0
  200. package/dist/browser/segment-structure-assert.d.ts.map +1 -0
  201. package/dist/browser/segment-structure-assert.js +59 -0
  202. package/dist/browser/segment-structure-assert.js.map +1 -0
  203. package/dist/browser/server-action-bridge.d.ts +26 -0
  204. package/dist/browser/server-action-bridge.d.ts.map +1 -0
  205. package/dist/browser/server-action-bridge.js +668 -0
  206. package/dist/browser/server-action-bridge.js.map +1 -0
  207. package/dist/browser/shallow.d.ts +12 -0
  208. package/dist/browser/shallow.d.ts.map +1 -0
  209. package/dist/browser/shallow.js +34 -0
  210. package/dist/browser/shallow.js.map +1 -0
  211. package/dist/browser/types.d.ts +369 -0
  212. package/dist/browser/types.d.ts.map +1 -0
  213. package/dist/browser/types.js +2 -0
  214. package/dist/browser/types.js.map +1 -0
  215. package/dist/build/__tests__/generate-cli.test.d.ts +2 -0
  216. package/dist/build/__tests__/generate-cli.test.d.ts.map +1 -0
  217. package/dist/build/__tests__/generate-cli.test.js +237 -0
  218. package/dist/build/__tests__/generate-cli.test.js.map +1 -0
  219. package/dist/build/__tests__/generate-manifest.test.d.ts +2 -0
  220. package/dist/build/__tests__/generate-manifest.test.d.ts.map +1 -0
  221. package/dist/build/__tests__/generate-manifest.test.js +119 -0
  222. package/dist/build/__tests__/generate-manifest.test.js.map +1 -0
  223. package/dist/build/__tests__/generate-route-types.test.d.ts +2 -0
  224. package/dist/build/__tests__/generate-route-types.test.d.ts.map +1 -0
  225. package/dist/build/__tests__/generate-route-types.test.js +620 -0
  226. package/dist/build/__tests__/generate-route-types.test.js.map +1 -0
  227. package/dist/build/__tests__/per-router-manifest.test.d.ts +2 -0
  228. package/dist/build/__tests__/per-router-manifest.test.d.ts.map +1 -0
  229. package/dist/build/__tests__/per-router-manifest.test.js +308 -0
  230. package/dist/build/__tests__/per-router-manifest.test.js.map +1 -0
  231. package/dist/build/generate-manifest.d.ts +81 -0
  232. package/dist/build/generate-manifest.d.ts.map +1 -0
  233. package/dist/build/generate-manifest.js +276 -0
  234. package/dist/build/generate-manifest.js.map +1 -0
  235. package/dist/build/generate-route-types.d.ts +115 -0
  236. package/dist/build/generate-route-types.d.ts.map +1 -0
  237. package/dist/build/generate-route-types.js +740 -0
  238. package/dist/build/generate-route-types.js.map +1 -0
  239. package/dist/build/index.d.ts +21 -0
  240. package/dist/build/index.d.ts.map +1 -0
  241. package/dist/build/index.js +21 -0
  242. package/dist/build/index.js.map +1 -0
  243. package/dist/build/route-trie.d.ts +71 -0
  244. package/dist/build/route-trie.d.ts.map +1 -0
  245. package/dist/build/route-trie.js +175 -0
  246. package/dist/build/route-trie.js.map +1 -0
  247. package/dist/cache/__tests__/cache-scope.test.d.ts +2 -0
  248. package/dist/cache/__tests__/cache-scope.test.d.ts.map +1 -0
  249. package/dist/cache/__tests__/cache-scope.test.js +208 -0
  250. package/dist/cache/__tests__/cache-scope.test.js.map +1 -0
  251. package/dist/cache/__tests__/document-cache.test.d.ts +2 -0
  252. package/dist/cache/__tests__/document-cache.test.d.ts.map +1 -0
  253. package/dist/cache/__tests__/document-cache.test.js +345 -0
  254. package/dist/cache/__tests__/document-cache.test.js.map +1 -0
  255. package/dist/cache/__tests__/memory-segment-store.test.d.ts +2 -0
  256. package/dist/cache/__tests__/memory-segment-store.test.d.ts.map +1 -0
  257. package/dist/cache/__tests__/memory-segment-store.test.js +425 -0
  258. package/dist/cache/__tests__/memory-segment-store.test.js.map +1 -0
  259. package/dist/cache/__tests__/memory-store.test.d.ts +2 -0
  260. package/dist/cache/__tests__/memory-store.test.d.ts.map +1 -0
  261. package/dist/cache/__tests__/memory-store.test.js +367 -0
  262. package/dist/cache/__tests__/memory-store.test.js.map +1 -0
  263. package/dist/cache/cache-scope.d.ts +102 -0
  264. package/dist/cache/cache-scope.d.ts.map +1 -0
  265. package/dist/cache/cache-scope.js +440 -0
  266. package/dist/cache/cache-scope.js.map +1 -0
  267. package/dist/cache/cf/__tests__/cf-cache-store.test.d.ts +2 -0
  268. package/dist/cache/cf/__tests__/cf-cache-store.test.d.ts.map +1 -0
  269. package/dist/cache/cf/__tests__/cf-cache-store.test.js +330 -0
  270. package/dist/cache/cf/__tests__/cf-cache-store.test.js.map +1 -0
  271. package/dist/cache/cf/cf-cache-store.d.ts +165 -0
  272. package/dist/cache/cf/cf-cache-store.d.ts.map +1 -0
  273. package/dist/cache/cf/cf-cache-store.js +242 -0
  274. package/dist/cache/cf/cf-cache-store.js.map +1 -0
  275. package/dist/cache/cf/index.d.ts +14 -0
  276. package/dist/cache/cf/index.d.ts.map +1 -0
  277. package/dist/cache/cf/index.js +17 -0
  278. package/dist/cache/cf/index.js.map +1 -0
  279. package/dist/cache/document-cache.d.ts +64 -0
  280. package/dist/cache/document-cache.d.ts.map +1 -0
  281. package/dist/cache/document-cache.js +228 -0
  282. package/dist/cache/document-cache.js.map +1 -0
  283. package/dist/cache/index.d.ts +19 -0
  284. package/dist/cache/index.d.ts.map +1 -0
  285. package/dist/cache/index.js +21 -0
  286. package/dist/cache/index.js.map +1 -0
  287. package/dist/cache/memory-segment-store.d.ts +110 -0
  288. package/dist/cache/memory-segment-store.d.ts.map +1 -0
  289. package/dist/cache/memory-segment-store.js +117 -0
  290. package/dist/cache/memory-segment-store.js.map +1 -0
  291. package/dist/cache/memory-store.d.ts +41 -0
  292. package/dist/cache/memory-store.d.ts.map +1 -0
  293. package/dist/cache/memory-store.js +191 -0
  294. package/dist/cache/memory-store.js.map +1 -0
  295. package/dist/cache/types.d.ts +317 -0
  296. package/dist/cache/types.d.ts.map +1 -0
  297. package/dist/cache/types.js +12 -0
  298. package/dist/cache/types.js.map +1 -0
  299. package/dist/client.d.ts +248 -0
  300. package/dist/client.d.ts.map +1 -0
  301. package/dist/client.js +367 -0
  302. package/dist/client.js.map +1 -0
  303. package/dist/client.rsc.d.ts +26 -0
  304. package/dist/client.rsc.d.ts.map +1 -0
  305. package/dist/client.rsc.js +46 -0
  306. package/dist/client.rsc.js.map +1 -0
  307. package/dist/component-utils.d.ts +36 -0
  308. package/dist/component-utils.d.ts.map +1 -0
  309. package/dist/component-utils.js +61 -0
  310. package/dist/component-utils.js.map +1 -0
  311. package/dist/components/DefaultDocument.d.ts +13 -0
  312. package/dist/components/DefaultDocument.d.ts.map +1 -0
  313. package/dist/components/DefaultDocument.js +15 -0
  314. package/dist/components/DefaultDocument.js.map +1 -0
  315. package/dist/debug.d.ts +58 -0
  316. package/dist/debug.d.ts.map +1 -0
  317. package/dist/debug.js +157 -0
  318. package/dist/debug.js.map +1 -0
  319. package/dist/default-error-boundary.d.ts +11 -0
  320. package/dist/default-error-boundary.d.ts.map +1 -0
  321. package/dist/default-error-boundary.js +45 -0
  322. package/dist/default-error-boundary.js.map +1 -0
  323. package/dist/deps/browser.d.ts +2 -0
  324. package/dist/deps/browser.d.ts.map +1 -0
  325. package/dist/deps/browser.js +3 -0
  326. package/dist/deps/browser.js.map +1 -0
  327. package/dist/deps/html-stream-client.d.ts +2 -0
  328. package/dist/deps/html-stream-client.d.ts.map +1 -0
  329. package/dist/deps/html-stream-client.js +3 -0
  330. package/dist/deps/html-stream-client.js.map +1 -0
  331. package/dist/deps/html-stream-server.d.ts +2 -0
  332. package/dist/deps/html-stream-server.d.ts.map +1 -0
  333. package/dist/deps/html-stream-server.js +3 -0
  334. package/dist/deps/html-stream-server.js.map +1 -0
  335. package/dist/deps/rsc.d.ts +2 -0
  336. package/dist/deps/rsc.d.ts.map +1 -0
  337. package/dist/deps/rsc.js +4 -0
  338. package/dist/deps/rsc.js.map +1 -0
  339. package/dist/deps/ssr.d.ts +2 -0
  340. package/dist/deps/ssr.d.ts.map +1 -0
  341. package/dist/deps/ssr.js +3 -0
  342. package/dist/deps/ssr.js.map +1 -0
  343. package/dist/errors.d.ts +174 -0
  344. package/dist/errors.d.ts.map +1 -0
  345. package/dist/errors.js +241 -0
  346. package/dist/errors.js.map +1 -0
  347. package/dist/handle.d.ts +78 -0
  348. package/dist/handle.d.ts.map +1 -0
  349. package/dist/handle.js +82 -0
  350. package/dist/handle.js.map +1 -0
  351. package/dist/handles/MetaTags.d.ts +14 -0
  352. package/dist/handles/MetaTags.d.ts.map +1 -0
  353. package/dist/handles/MetaTags.js +136 -0
  354. package/dist/handles/MetaTags.js.map +1 -0
  355. package/dist/handles/index.d.ts +6 -0
  356. package/dist/handles/index.d.ts.map +1 -0
  357. package/{src/handles/index.ts → dist/handles/index.js} +1 -1
  358. package/dist/handles/index.js.map +1 -0
  359. package/dist/handles/meta.d.ts +39 -0
  360. package/dist/handles/meta.d.ts.map +1 -0
  361. package/dist/handles/meta.js +202 -0
  362. package/dist/handles/meta.js.map +1 -0
  363. package/dist/host/__tests__/errors.test.d.ts +2 -0
  364. package/dist/host/__tests__/errors.test.d.ts.map +1 -0
  365. package/dist/host/__tests__/errors.test.js +76 -0
  366. package/dist/host/__tests__/errors.test.js.map +1 -0
  367. package/dist/host/__tests__/pattern-comprehensive.test.d.ts +2 -0
  368. package/dist/host/__tests__/pattern-comprehensive.test.d.ts.map +1 -0
  369. package/dist/host/__tests__/pattern-comprehensive.test.js +732 -0
  370. package/dist/host/__tests__/pattern-comprehensive.test.js.map +1 -0
  371. package/dist/host/__tests__/pattern-matcher.test.d.ts +2 -0
  372. package/dist/host/__tests__/pattern-matcher.test.d.ts.map +1 -0
  373. package/dist/host/__tests__/pattern-matcher.test.js +251 -0
  374. package/dist/host/__tests__/pattern-matcher.test.js.map +1 -0
  375. package/dist/host/__tests__/router.test.d.ts +2 -0
  376. package/dist/host/__tests__/router.test.d.ts.map +1 -0
  377. package/dist/host/__tests__/router.test.js +241 -0
  378. package/dist/host/__tests__/router.test.js.map +1 -0
  379. package/dist/host/__tests__/testing.test.d.ts +2 -0
  380. package/dist/host/__tests__/testing.test.d.ts.map +1 -0
  381. package/dist/host/__tests__/testing.test.js +64 -0
  382. package/dist/host/__tests__/testing.test.js.map +1 -0
  383. package/dist/host/__tests__/utils.test.d.ts +2 -0
  384. package/dist/host/__tests__/utils.test.d.ts.map +1 -0
  385. package/dist/host/__tests__/utils.test.js +29 -0
  386. package/dist/host/__tests__/utils.test.js.map +1 -0
  387. package/dist/host/cookie-handler.d.ts +34 -0
  388. package/dist/host/cookie-handler.d.ts.map +1 -0
  389. package/dist/host/cookie-handler.js +124 -0
  390. package/dist/host/cookie-handler.js.map +1 -0
  391. package/dist/host/errors.d.ts +56 -0
  392. package/dist/host/errors.d.ts.map +1 -0
  393. package/dist/host/errors.js +79 -0
  394. package/dist/host/errors.js.map +1 -0
  395. package/dist/host/index.d.ts +29 -0
  396. package/dist/host/index.d.ts.map +1 -0
  397. package/dist/host/index.js +32 -0
  398. package/dist/host/index.js.map +1 -0
  399. package/dist/host/pattern-matcher.d.ts +36 -0
  400. package/dist/host/pattern-matcher.d.ts.map +1 -0
  401. package/dist/host/pattern-matcher.js +172 -0
  402. package/dist/host/pattern-matcher.js.map +1 -0
  403. package/dist/host/router.d.ts +26 -0
  404. package/dist/host/router.d.ts.map +1 -0
  405. package/dist/host/router.js +218 -0
  406. package/dist/host/router.js.map +1 -0
  407. package/dist/host/testing.d.ts +36 -0
  408. package/dist/host/testing.d.ts.map +1 -0
  409. package/dist/host/testing.js +55 -0
  410. package/dist/host/testing.js.map +1 -0
  411. package/dist/host/types.d.ts +115 -0
  412. package/dist/host/types.d.ts.map +1 -0
  413. package/dist/host/types.js +7 -0
  414. package/dist/host/types.js.map +1 -0
  415. package/dist/host/utils.d.ts +21 -0
  416. package/dist/host/utils.d.ts.map +1 -0
  417. package/dist/host/utils.js +23 -0
  418. package/dist/host/utils.js.map +1 -0
  419. package/dist/href-client.d.ts +131 -0
  420. package/dist/href-client.d.ts.map +1 -0
  421. package/dist/href-client.js +64 -0
  422. package/dist/href-client.js.map +1 -0
  423. package/{src/href-context.ts → dist/href-context.d.ts} +7 -11
  424. package/dist/href-context.d.ts.map +1 -0
  425. package/dist/href-context.js +21 -0
  426. package/dist/href-context.js.map +1 -0
  427. package/dist/index.d.ts +73 -0
  428. package/dist/index.d.ts.map +1 -0
  429. package/dist/index.js +91 -0
  430. package/dist/index.js.map +1 -0
  431. package/dist/index.rsc.d.ts +32 -0
  432. package/dist/index.rsc.d.ts.map +1 -0
  433. package/dist/index.rsc.js +40 -0
  434. package/dist/index.rsc.js.map +1 -0
  435. package/dist/internal-debug.d.ts +2 -0
  436. package/dist/internal-debug.d.ts.map +1 -0
  437. package/dist/internal-debug.js +5 -0
  438. package/dist/internal-debug.js.map +1 -0
  439. package/dist/loader.d.ts +14 -0
  440. package/dist/loader.d.ts.map +1 -0
  441. package/dist/loader.js +20 -0
  442. package/dist/loader.js.map +1 -0
  443. package/dist/loader.rsc.d.ts +19 -0
  444. package/dist/loader.rsc.d.ts.map +1 -0
  445. package/dist/loader.rsc.js +99 -0
  446. package/dist/loader.rsc.js.map +1 -0
  447. package/dist/network-error-thrower.d.ts +17 -0
  448. package/dist/network-error-thrower.d.ts.map +1 -0
  449. package/dist/network-error-thrower.js +14 -0
  450. package/dist/network-error-thrower.js.map +1 -0
  451. package/dist/outlet-context.d.ts +13 -0
  452. package/dist/outlet-context.d.ts.map +1 -0
  453. package/dist/outlet-context.js +3 -0
  454. package/dist/outlet-context.js.map +1 -0
  455. package/dist/prerender/__tests__/param-hash.test.d.ts +2 -0
  456. package/dist/prerender/__tests__/param-hash.test.d.ts.map +1 -0
  457. package/dist/prerender/__tests__/param-hash.test.js +148 -0
  458. package/dist/prerender/__tests__/param-hash.test.js.map +1 -0
  459. package/dist/prerender/param-hash.d.ts +16 -0
  460. package/dist/prerender/param-hash.d.ts.map +1 -0
  461. package/dist/prerender/param-hash.js +36 -0
  462. package/dist/prerender/param-hash.js.map +1 -0
  463. package/dist/prerender/store.d.ts +38 -0
  464. package/dist/prerender/store.d.ts.map +1 -0
  465. package/dist/prerender/store.js +61 -0
  466. package/dist/prerender/store.js.map +1 -0
  467. package/dist/prerender.d.ts +66 -0
  468. package/dist/prerender.d.ts.map +1 -0
  469. package/dist/prerender.js +57 -0
  470. package/dist/prerender.js.map +1 -0
  471. package/dist/reverse.d.ts +196 -0
  472. package/dist/reverse.d.ts.map +1 -0
  473. package/dist/reverse.js +78 -0
  474. package/dist/reverse.js.map +1 -0
  475. package/dist/root-error-boundary.d.ts +33 -0
  476. package/dist/root-error-boundary.d.ts.map +1 -0
  477. package/dist/root-error-boundary.js +165 -0
  478. package/dist/root-error-boundary.js.map +1 -0
  479. package/dist/route-content-wrapper.d.ts +46 -0
  480. package/dist/route-content-wrapper.d.ts.map +1 -0
  481. package/dist/route-content-wrapper.js +77 -0
  482. package/dist/route-content-wrapper.js.map +1 -0
  483. package/dist/route-definition.d.ts +421 -0
  484. package/dist/route-definition.d.ts.map +1 -0
  485. package/dist/route-definition.js +868 -0
  486. package/dist/route-definition.js.map +1 -0
  487. package/dist/route-map-builder.d.ts +155 -0
  488. package/dist/route-map-builder.d.ts.map +1 -0
  489. package/dist/route-map-builder.js +237 -0
  490. package/dist/route-map-builder.js.map +1 -0
  491. package/dist/route-types.d.ts +165 -0
  492. package/dist/route-types.d.ts.map +1 -0
  493. package/dist/route-types.js +7 -0
  494. package/dist/route-types.js.map +1 -0
  495. package/dist/router/__tests__/handler-context.test.d.ts +2 -0
  496. package/dist/router/__tests__/handler-context.test.d.ts.map +1 -0
  497. package/dist/router/__tests__/handler-context.test.js +65 -0
  498. package/dist/router/__tests__/handler-context.test.js.map +1 -0
  499. package/dist/router/__tests__/loader-cycle-detection.test.d.ts +2 -0
  500. package/dist/router/__tests__/loader-cycle-detection.test.d.ts.map +1 -0
  501. package/dist/router/__tests__/loader-cycle-detection.test.js +221 -0
  502. package/dist/router/__tests__/loader-cycle-detection.test.js.map +1 -0
  503. package/dist/router/__tests__/match-context.test.d.ts +2 -0
  504. package/dist/router/__tests__/match-context.test.d.ts.map +1 -0
  505. package/dist/router/__tests__/match-context.test.js +92 -0
  506. package/dist/router/__tests__/match-context.test.js.map +1 -0
  507. package/dist/router/__tests__/match-pipelines.test.d.ts +2 -0
  508. package/dist/router/__tests__/match-pipelines.test.d.ts.map +1 -0
  509. package/dist/router/__tests__/match-pipelines.test.js +417 -0
  510. package/dist/router/__tests__/match-pipelines.test.js.map +1 -0
  511. package/dist/router/__tests__/match-result.test.d.ts +2 -0
  512. package/dist/router/__tests__/match-result.test.d.ts.map +1 -0
  513. package/dist/router/__tests__/match-result.test.js +457 -0
  514. package/dist/router/__tests__/match-result.test.js.map +1 -0
  515. package/dist/router/__tests__/on-error.test.d.ts +2 -0
  516. package/dist/router/__tests__/on-error.test.d.ts.map +1 -0
  517. package/dist/router/__tests__/on-error.test.js +678 -0
  518. package/dist/router/__tests__/on-error.test.js.map +1 -0
  519. package/dist/router/__tests__/pattern-matching.test.d.ts +2 -0
  520. package/dist/router/__tests__/pattern-matching.test.d.ts.map +1 -0
  521. package/dist/router/__tests__/pattern-matching.test.js +629 -0
  522. package/dist/router/__tests__/pattern-matching.test.js.map +1 -0
  523. package/dist/router/__tests__/segment-resolution-parallel-loading.test.d.ts +2 -0
  524. package/dist/router/__tests__/segment-resolution-parallel-loading.test.d.ts.map +1 -0
  525. package/dist/router/__tests__/segment-resolution-parallel-loading.test.js +155 -0
  526. package/dist/router/__tests__/segment-resolution-parallel-loading.test.js.map +1 -0
  527. package/dist/router/error-handling.d.ts +77 -0
  528. package/dist/router/error-handling.d.ts.map +1 -0
  529. package/dist/router/error-handling.js +202 -0
  530. package/dist/router/error-handling.js.map +1 -0
  531. package/dist/router/handler-context.d.ts +20 -0
  532. package/dist/router/handler-context.d.ts.map +1 -0
  533. package/dist/router/handler-context.js +198 -0
  534. package/dist/router/handler-context.js.map +1 -0
  535. package/dist/router/intercept-resolution.d.ts +66 -0
  536. package/dist/router/intercept-resolution.d.ts.map +1 -0
  537. package/dist/router/intercept-resolution.js +246 -0
  538. package/dist/router/intercept-resolution.js.map +1 -0
  539. package/dist/router/loader-resolution.d.ts +64 -0
  540. package/dist/router/loader-resolution.d.ts.map +1 -0
  541. package/dist/router/loader-resolution.js +284 -0
  542. package/dist/router/loader-resolution.js.map +1 -0
  543. package/dist/router/logging.d.ts +15 -0
  544. package/dist/router/logging.d.ts.map +1 -0
  545. package/dist/router/logging.js +99 -0
  546. package/dist/router/logging.js.map +1 -0
  547. package/dist/router/manifest.d.ts +22 -0
  548. package/dist/router/manifest.d.ts.map +1 -0
  549. package/dist/router/manifest.js +181 -0
  550. package/dist/router/manifest.js.map +1 -0
  551. package/dist/router/match-api.d.ts +35 -0
  552. package/dist/router/match-api.d.ts.map +1 -0
  553. package/dist/router/match-api.js +406 -0
  554. package/dist/router/match-api.js.map +1 -0
  555. package/dist/router/match-context.d.ts +206 -0
  556. package/dist/router/match-context.d.ts.map +1 -0
  557. package/dist/router/match-context.js +17 -0
  558. package/dist/router/match-context.js.map +1 -0
  559. package/dist/router/match-middleware/background-revalidation.d.ts +127 -0
  560. package/dist/router/match-middleware/background-revalidation.d.ts.map +1 -0
  561. package/dist/router/match-middleware/background-revalidation.js +75 -0
  562. package/dist/router/match-middleware/background-revalidation.js.map +1 -0
  563. package/dist/router/match-middleware/cache-lookup.d.ts +112 -0
  564. package/dist/router/match-middleware/cache-lookup.d.ts.map +1 -0
  565. package/dist/router/match-middleware/cache-lookup.js +257 -0
  566. package/dist/router/match-middleware/cache-lookup.js.map +1 -0
  567. package/dist/router/match-middleware/cache-store.d.ts +113 -0
  568. package/dist/router/match-middleware/cache-store.d.ts.map +1 -0
  569. package/dist/router/match-middleware/cache-store.js +108 -0
  570. package/dist/router/match-middleware/cache-store.js.map +1 -0
  571. package/dist/router/match-middleware/index.d.ts +81 -0
  572. package/dist/router/match-middleware/index.d.ts.map +1 -0
  573. package/dist/router/match-middleware/index.js +80 -0
  574. package/dist/router/match-middleware/index.js.map +1 -0
  575. package/dist/router/match-middleware/intercept-resolution.d.ts +117 -0
  576. package/dist/router/match-middleware/intercept-resolution.d.ts.map +1 -0
  577. package/dist/router/match-middleware/intercept-resolution.js +134 -0
  578. package/dist/router/match-middleware/intercept-resolution.js.map +1 -0
  579. package/dist/router/match-middleware/segment-resolution.d.ts +99 -0
  580. package/dist/router/match-middleware/segment-resolution.d.ts.map +1 -0
  581. package/dist/router/match-middleware/segment-resolution.js +53 -0
  582. package/dist/router/match-middleware/segment-resolution.js.map +1 -0
  583. package/dist/router/match-pipelines.d.ts +147 -0
  584. package/dist/router/match-pipelines.d.ts.map +1 -0
  585. package/dist/router/match-pipelines.js +82 -0
  586. package/dist/router/match-pipelines.js.map +1 -0
  587. package/dist/router/match-result.d.ts +126 -0
  588. package/dist/router/match-result.d.ts.map +1 -0
  589. package/dist/router/match-result.js +93 -0
  590. package/dist/router/match-result.js.map +1 -0
  591. package/dist/router/metrics.d.ts +20 -0
  592. package/dist/router/metrics.d.ts.map +1 -0
  593. package/dist/router/metrics.js +47 -0
  594. package/dist/router/metrics.js.map +1 -0
  595. package/dist/router/middleware.d.ts +249 -0
  596. package/dist/router/middleware.d.ts.map +1 -0
  597. package/dist/router/middleware.js +434 -0
  598. package/dist/router/middleware.js.map +1 -0
  599. package/dist/router/middleware.test.d.ts +2 -0
  600. package/dist/router/middleware.test.d.ts.map +1 -0
  601. package/dist/router/middleware.test.js +816 -0
  602. package/dist/router/middleware.test.js.map +1 -0
  603. package/dist/router/pattern-matching.d.ts +149 -0
  604. package/dist/router/pattern-matching.d.ts.map +1 -0
  605. package/dist/router/pattern-matching.js +349 -0
  606. package/dist/router/pattern-matching.js.map +1 -0
  607. package/dist/router/revalidation.d.ts +44 -0
  608. package/dist/router/revalidation.d.ts.map +1 -0
  609. package/dist/router/revalidation.js +147 -0
  610. package/dist/router/revalidation.js.map +1 -0
  611. package/dist/router/router-context.d.ts +135 -0
  612. package/dist/router/router-context.d.ts.map +1 -0
  613. package/dist/router/router-context.js +36 -0
  614. package/dist/router/router-context.js.map +1 -0
  615. package/dist/router/segment-resolution.d.ts +127 -0
  616. package/dist/router/segment-resolution.d.ts.map +1 -0
  617. package/dist/router/segment-resolution.js +919 -0
  618. package/dist/router/segment-resolution.js.map +1 -0
  619. package/dist/router/trie-matching.d.ts +40 -0
  620. package/dist/router/trie-matching.d.ts.map +1 -0
  621. package/dist/router/trie-matching.js +127 -0
  622. package/dist/router/trie-matching.js.map +1 -0
  623. package/dist/router/types.d.ts +136 -0
  624. package/dist/router/types.d.ts.map +1 -0
  625. package/dist/router/types.js +7 -0
  626. package/dist/router/types.js.map +1 -0
  627. package/dist/router.d.ts +753 -0
  628. package/dist/router.d.ts.map +1 -0
  629. package/dist/router.gen.d.ts +6 -0
  630. package/dist/router.gen.d.ts.map +1 -0
  631. package/dist/router.gen.js +6 -0
  632. package/dist/router.gen.js.map +1 -0
  633. package/dist/router.js +1304 -0
  634. package/dist/router.js.map +1 -0
  635. package/dist/rsc/__tests__/helpers.test.d.ts +2 -0
  636. package/dist/rsc/__tests__/helpers.test.d.ts.map +1 -0
  637. package/dist/rsc/__tests__/helpers.test.js +140 -0
  638. package/dist/rsc/__tests__/helpers.test.js.map +1 -0
  639. package/dist/rsc/handler.d.ts +45 -0
  640. package/dist/rsc/handler.d.ts.map +1 -0
  641. package/dist/rsc/handler.js +1172 -0
  642. package/dist/rsc/handler.js.map +1 -0
  643. package/dist/rsc/helpers.d.ts +16 -0
  644. package/dist/rsc/helpers.d.ts.map +1 -0
  645. package/dist/rsc/helpers.js +55 -0
  646. package/dist/rsc/helpers.js.map +1 -0
  647. package/dist/rsc/index.d.ts +22 -0
  648. package/dist/rsc/index.d.ts.map +1 -0
  649. package/dist/rsc/index.js +23 -0
  650. package/dist/rsc/index.js.map +1 -0
  651. package/dist/rsc/nonce.d.ts +9 -0
  652. package/dist/rsc/nonce.d.ts.map +1 -0
  653. package/dist/rsc/nonce.js +18 -0
  654. package/dist/rsc/nonce.js.map +1 -0
  655. package/dist/rsc/types.d.ts +206 -0
  656. package/dist/rsc/types.d.ts.map +1 -0
  657. package/dist/rsc/types.js +8 -0
  658. package/dist/rsc/types.js.map +1 -0
  659. package/dist/search-params.d.ts +103 -0
  660. package/dist/search-params.d.ts.map +1 -0
  661. package/dist/search-params.js +74 -0
  662. package/dist/search-params.js.map +1 -0
  663. package/dist/segment-system.d.ts +75 -0
  664. package/dist/segment-system.d.ts.map +1 -0
  665. package/dist/segment-system.js +336 -0
  666. package/dist/segment-system.js.map +1 -0
  667. package/dist/server/context.d.ts +245 -0
  668. package/dist/server/context.d.ts.map +1 -0
  669. package/dist/server/context.js +197 -0
  670. package/dist/server/context.js.map +1 -0
  671. package/dist/server/fetchable-loader-store.d.ts +18 -0
  672. package/dist/server/fetchable-loader-store.d.ts.map +1 -0
  673. package/dist/server/fetchable-loader-store.js +18 -0
  674. package/dist/server/fetchable-loader-store.js.map +1 -0
  675. package/dist/server/handle-store.d.ts +85 -0
  676. package/dist/server/handle-store.d.ts.map +1 -0
  677. package/dist/server/handle-store.js +142 -0
  678. package/dist/server/handle-store.js.map +1 -0
  679. package/dist/server/loader-registry.d.ts +55 -0
  680. package/dist/server/loader-registry.d.ts.map +1 -0
  681. package/dist/server/loader-registry.js +132 -0
  682. package/dist/server/loader-registry.js.map +1 -0
  683. package/dist/server/request-context.d.ts +226 -0
  684. package/dist/server/request-context.d.ts.map +1 -0
  685. package/dist/server/request-context.js +290 -0
  686. package/dist/server/request-context.js.map +1 -0
  687. package/dist/server/root-layout.d.ts +4 -0
  688. package/dist/server/root-layout.d.ts.map +1 -0
  689. package/dist/server/root-layout.js +5 -0
  690. package/dist/server/root-layout.js.map +1 -0
  691. package/dist/server.d.ts +15 -0
  692. package/dist/server.d.ts.map +1 -0
  693. package/dist/server.js +20 -0
  694. package/dist/server.js.map +1 -0
  695. package/dist/ssr/__tests__/ssr-handler.test.d.ts +2 -0
  696. package/dist/ssr/__tests__/ssr-handler.test.d.ts.map +1 -0
  697. package/dist/ssr/__tests__/ssr-handler.test.js +132 -0
  698. package/dist/ssr/__tests__/ssr-handler.test.js.map +1 -0
  699. package/dist/ssr/index.d.ts +98 -0
  700. package/dist/ssr/index.d.ts.map +1 -0
  701. package/dist/ssr/index.js +158 -0
  702. package/dist/ssr/index.js.map +1 -0
  703. package/dist/static-handler.d.ts +50 -0
  704. package/dist/static-handler.d.ts.map +1 -0
  705. package/dist/static-handler.gen.d.ts +5 -0
  706. package/dist/static-handler.gen.d.ts.map +1 -0
  707. package/dist/static-handler.gen.js +5 -0
  708. package/dist/static-handler.gen.js.map +1 -0
  709. package/dist/static-handler.js +29 -0
  710. package/dist/static-handler.js.map +1 -0
  711. package/dist/testing/vitest.js +82 -0
  712. package/dist/theme/ThemeProvider.d.ts +20 -0
  713. package/dist/theme/ThemeProvider.d.ts.map +1 -0
  714. package/dist/theme/ThemeProvider.js +240 -0
  715. package/dist/theme/ThemeProvider.js.map +1 -0
  716. package/dist/theme/ThemeScript.d.ts +48 -0
  717. package/dist/theme/ThemeScript.d.ts.map +1 -0
  718. package/dist/theme/ThemeScript.js +13 -0
  719. package/dist/theme/ThemeScript.js.map +1 -0
  720. package/dist/theme/__tests__/theme.test.d.ts +2 -0
  721. package/dist/theme/__tests__/theme.test.d.ts.map +1 -0
  722. package/dist/theme/__tests__/theme.test.js +103 -0
  723. package/dist/theme/__tests__/theme.test.js.map +1 -0
  724. package/dist/theme/constants.d.ts +29 -0
  725. package/dist/theme/constants.d.ts.map +1 -0
  726. package/dist/theme/constants.js +48 -0
  727. package/dist/theme/constants.js.map +1 -0
  728. package/dist/theme/index.d.ts +31 -0
  729. package/dist/theme/index.d.ts.map +1 -0
  730. package/dist/theme/index.js +36 -0
  731. package/dist/theme/index.js.map +1 -0
  732. package/dist/theme/theme-context.d.ts +40 -0
  733. package/dist/theme/theme-context.d.ts.map +1 -0
  734. package/dist/theme/theme-context.js +60 -0
  735. package/dist/theme/theme-context.js.map +1 -0
  736. package/dist/theme/theme-script.d.ts +27 -0
  737. package/dist/theme/theme-script.d.ts.map +1 -0
  738. package/dist/theme/theme-script.js +147 -0
  739. package/dist/theme/theme-script.js.map +1 -0
  740. package/dist/theme/types.d.ts +163 -0
  741. package/dist/theme/types.d.ts.map +1 -0
  742. package/dist/theme/types.js +11 -0
  743. package/dist/theme/types.js.map +1 -0
  744. package/dist/theme/use-theme.d.ts +12 -0
  745. package/dist/theme/use-theme.d.ts.map +1 -0
  746. package/dist/theme/use-theme.js +40 -0
  747. package/dist/theme/use-theme.js.map +1 -0
  748. package/dist/types.d.ts +1479 -0
  749. package/dist/types.d.ts.map +1 -0
  750. package/dist/types.js +10 -0
  751. package/dist/types.js.map +1 -0
  752. package/dist/urls.d.ts +441 -0
  753. package/dist/urls.d.ts.map +1 -0
  754. package/dist/urls.gen.d.ts +8 -0
  755. package/dist/urls.gen.d.ts.map +1 -0
  756. package/dist/urls.gen.js +8 -0
  757. package/dist/urls.gen.js.map +1 -0
  758. package/dist/urls.js +443 -0
  759. package/dist/urls.js.map +1 -0
  760. package/dist/use-loader.d.ts +127 -0
  761. package/dist/use-loader.d.ts.map +1 -0
  762. package/dist/use-loader.js +237 -0
  763. package/dist/use-loader.js.map +1 -0
  764. package/dist/vite/__tests__/ast-handler-extract.test.d.ts +2 -0
  765. package/dist/vite/__tests__/ast-handler-extract.test.d.ts.map +1 -0
  766. package/dist/vite/__tests__/ast-handler-extract.test.js +294 -0
  767. package/dist/vite/__tests__/ast-handler-extract.test.js.map +1 -0
  768. package/dist/vite/__tests__/expose-id-utils.test.d.ts +2 -0
  769. package/dist/vite/__tests__/expose-id-utils.test.d.ts.map +1 -0
  770. package/dist/vite/__tests__/expose-id-utils.test.js +224 -0
  771. package/dist/vite/__tests__/expose-id-utils.test.js.map +1 -0
  772. package/dist/vite/__tests__/expose-internal-ids.test.d.ts +2 -0
  773. package/dist/vite/__tests__/expose-internal-ids.test.d.ts.map +1 -0
  774. package/dist/vite/__tests__/expose-internal-ids.test.js +647 -0
  775. package/dist/vite/__tests__/expose-internal-ids.test.js.map +1 -0
  776. package/dist/vite/__tests__/expose-router-id.test.d.ts +2 -0
  777. package/dist/vite/__tests__/expose-router-id.test.d.ts.map +1 -0
  778. package/dist/vite/__tests__/expose-router-id.test.js +39 -0
  779. package/dist/vite/__tests__/expose-router-id.test.js.map +1 -0
  780. package/dist/vite/ast-handler-extract.d.ts +49 -0
  781. package/dist/vite/ast-handler-extract.d.ts.map +1 -0
  782. package/dist/vite/ast-handler-extract.js +249 -0
  783. package/dist/vite/ast-handler-extract.js.map +1 -0
  784. package/dist/vite/expose-action-id.d.ts +19 -0
  785. package/dist/vite/expose-action-id.d.ts.map +1 -0
  786. package/dist/vite/expose-action-id.js +250 -0
  787. package/dist/vite/expose-action-id.js.map +1 -0
  788. package/dist/vite/expose-id-utils.d.ts +69 -0
  789. package/dist/vite/expose-id-utils.d.ts.map +1 -0
  790. package/dist/vite/expose-id-utils.js +289 -0
  791. package/dist/vite/expose-id-utils.js.map +1 -0
  792. package/dist/vite/expose-internal-ids.d.ts +22 -0
  793. package/dist/vite/expose-internal-ids.d.ts.map +1 -0
  794. package/dist/vite/expose-internal-ids.js +886 -0
  795. package/dist/vite/expose-internal-ids.js.map +1 -0
  796. package/dist/vite/index.d.ts +149 -0
  797. package/dist/vite/index.d.ts.map +1 -0
  798. package/dist/vite/index.js +6263 -2733
  799. package/dist/vite/index.js.bak +5448 -0
  800. package/dist/vite/index.js.map +1 -0
  801. package/dist/vite/package-resolution.d.ts +43 -0
  802. package/dist/vite/package-resolution.d.ts.map +1 -0
  803. package/{src/vite/package-resolution.ts → dist/vite/package-resolution.js} +53 -66
  804. package/dist/vite/package-resolution.js.map +1 -0
  805. package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  806. package/dist/vite/virtual-entries.d.ts +25 -0
  807. package/dist/vite/virtual-entries.d.ts.map +1 -0
  808. package/{src/vite/virtual-entries.ts → dist/vite/virtual-entries.js} +12 -16
  809. package/dist/vite/virtual-entries.js.map +1 -0
  810. package/package.json +123 -68
  811. package/skills/api-client/SKILL.md +211 -0
  812. package/skills/breadcrumbs/SKILL.md +312 -0
  813. package/skills/bundle-analysis/SKILL.md +159 -0
  814. package/skills/cache-guide/SKILL.md +486 -0
  815. package/skills/caching/SKILL.md +349 -24
  816. package/skills/composability/SKILL.md +197 -0
  817. package/skills/css/SKILL.md +76 -0
  818. package/skills/debug-manifest/SKILL.md +12 -8
  819. package/skills/document-cache/SKILL.md +87 -62
  820. package/skills/fonts/SKILL.md +6 -4
  821. package/skills/handler-use/SKILL.md +364 -0
  822. package/skills/hooks/SKILL.md +557 -79
  823. package/skills/host-router/SKILL.md +278 -0
  824. package/skills/i18n/SKILL.md +276 -0
  825. package/skills/intercept/SKILL.md +175 -8
  826. package/skills/layout/SKILL.md +128 -5
  827. package/skills/links/SKILL.md +304 -25
  828. package/skills/loader/SKILL.md +604 -54
  829. package/skills/middleware/SKILL.md +213 -37
  830. package/skills/migrate-nextjs/SKILL.md +584 -0
  831. package/skills/migrate-react-router/SKILL.md +769 -0
  832. package/skills/mime-routes/SKILL.md +41 -10
  833. package/skills/observability/SKILL.md +172 -0
  834. package/skills/parallel/SKILL.md +276 -3
  835. package/skills/prerender/SKILL.md +432 -52
  836. package/skills/rango/SKILL.md +313 -21
  837. package/skills/react-compiler/SKILL.md +168 -0
  838. package/skills/response-routes/SKILL.md +248 -120
  839. package/skills/route/SKILL.md +287 -21
  840. package/skills/router-setup/SKILL.md +231 -33
  841. package/skills/server-actions/SKILL.md +775 -0
  842. package/skills/streams-and-websockets/SKILL.md +283 -0
  843. package/skills/tailwind/SKILL.md +27 -3
  844. package/skills/testing/SKILL.md +129 -0
  845. package/skills/testing/bindings.md +89 -0
  846. package/skills/testing/cache-prerender.md +124 -0
  847. package/skills/testing/client-components.md +122 -0
  848. package/skills/testing/e2e-parity.md +125 -0
  849. package/skills/testing/flight.md +92 -0
  850. package/skills/testing/handles.md +129 -0
  851. package/skills/testing/loader.md +128 -0
  852. package/skills/testing/middleware.md +99 -0
  853. package/skills/testing/render-handler.md +121 -0
  854. package/skills/testing/response-routes.md +95 -0
  855. package/skills/testing/reverse-and-types.md +84 -0
  856. package/skills/testing/server-actions.md +107 -0
  857. package/skills/testing/server-tree.md +128 -0
  858. package/skills/testing/setup.md +120 -0
  859. package/skills/theme/SKILL.md +9 -8
  860. package/skills/typesafety/SKILL.md +547 -107
  861. package/skills/use-cache/SKILL.md +355 -0
  862. package/skills/view-transitions/SKILL.md +294 -0
  863. package/src/__augment-tests__/augment.ts +81 -0
  864. package/src/__augment-tests__/augmented.check.ts +116 -0
  865. package/src/__internal.ts +77 -44
  866. package/src/bin/rango.ts +312 -15
  867. package/src/browser/action-coordinator.ts +114 -0
  868. package/src/browser/action-fence.ts +47 -0
  869. package/src/browser/app-shell.ts +39 -0
  870. package/src/browser/app-version.ts +14 -0
  871. package/src/browser/cookie-name.ts +140 -0
  872. package/src/browser/event-controller.ts +162 -200
  873. package/src/browser/history-state.ts +101 -0
  874. package/src/browser/index.ts +3 -3
  875. package/src/browser/intercept-utils.ts +52 -0
  876. package/src/browser/invalidate-client-cache.ts +52 -0
  877. package/src/browser/link-interceptor.ts +24 -4
  878. package/src/browser/logging.ts +11 -0
  879. package/src/browser/merge-segment-loaders.ts +20 -12
  880. package/src/browser/navigation-bridge.ts +323 -563
  881. package/src/browser/navigation-client.ts +219 -75
  882. package/src/browser/navigation-store-handle.ts +38 -0
  883. package/src/browser/navigation-store.ts +104 -112
  884. package/src/browser/navigation-transaction.ts +247 -0
  885. package/src/browser/network-error-handler.ts +61 -0
  886. package/src/browser/partial-update.ts +328 -348
  887. package/src/browser/prefetch/cache.ts +324 -0
  888. package/src/browser/prefetch/fetch.ts +357 -0
  889. package/src/browser/prefetch/observer.ts +65 -0
  890. package/src/browser/prefetch/policy.ts +48 -0
  891. package/src/browser/prefetch/queue.ts +194 -0
  892. package/src/browser/prefetch/resource-ready.ts +77 -0
  893. package/src/browser/rango-state.ts +194 -0
  894. package/src/browser/react/Link.tsx +253 -71
  895. package/src/browser/react/NavigationProvider.tsx +155 -34
  896. package/src/browser/react/ScrollRestoration.tsx +10 -6
  897. package/src/browser/react/context.ts +11 -0
  898. package/src/browser/react/filter-segment-order.ts +53 -0
  899. package/src/browser/react/index.ts +0 -48
  900. package/src/browser/react/location-state-shared.ts +260 -60
  901. package/src/browser/react/location-state.ts +90 -20
  902. package/src/browser/react/mount-context.ts +6 -1
  903. package/src/browser/react/nonce-context.ts +23 -0
  904. package/src/browser/react/shallow-equal.ts +27 -0
  905. package/src/browser/react/use-action.ts +35 -66
  906. package/src/browser/react/use-handle.ts +39 -126
  907. package/src/browser/react/use-link-status.ts +6 -9
  908. package/src/browser/react/use-navigation.ts +44 -68
  909. package/src/browser/react/use-params.ts +75 -0
  910. package/src/browser/react/use-pathname.ts +47 -0
  911. package/src/browser/react/use-reverse.ts +106 -0
  912. package/src/browser/react/use-router.ts +98 -0
  913. package/src/browser/react/use-search-params.ts +51 -0
  914. package/src/browser/react/use-segments.ts +72 -99
  915. package/src/browser/response-adapter.ts +124 -0
  916. package/src/browser/rsc-router.tsx +290 -72
  917. package/src/browser/scroll-restoration.ts +132 -49
  918. package/src/browser/segment-reconciler.ts +243 -0
  919. package/src/browser/segment-structure-assert.ts +17 -1
  920. package/src/browser/server-action-bridge.ts +621 -613
  921. package/src/browser/types.ts +175 -50
  922. package/src/browser/validate-redirect-origin.ts +56 -0
  923. package/src/build/collect-fallback-refs.ts +107 -0
  924. package/src/build/generate-manifest.ts +123 -56
  925. package/src/build/generate-route-types.ts +41 -1038
  926. package/src/build/index.ts +9 -6
  927. package/src/build/prefix-tree-utils.ts +123 -0
  928. package/src/build/route-trie.ts +165 -34
  929. package/src/build/route-types/ast-helpers.ts +25 -0
  930. package/src/build/route-types/ast-route-extraction.ts +98 -0
  931. package/src/build/route-types/codegen.ts +113 -0
  932. package/src/build/route-types/include-resolution.ts +418 -0
  933. package/src/build/route-types/param-extraction.ts +51 -0
  934. package/src/build/route-types/per-module-writer.ts +131 -0
  935. package/src/build/route-types/router-processing.ts +651 -0
  936. package/src/build/route-types/scan-filter.ts +85 -0
  937. package/src/build/route-types/source-scan.ts +118 -0
  938. package/src/build/runtime-discovery.ts +220 -0
  939. package/src/cache/background-task.ts +34 -0
  940. package/src/cache/cache-error.ts +104 -0
  941. package/src/cache/cache-key-utils.ts +44 -0
  942. package/src/cache/cache-policy.ts +165 -0
  943. package/src/cache/cache-runtime.ts +444 -0
  944. package/src/cache/cache-scope.ts +231 -325
  945. package/src/cache/cache-tag.ts +98 -0
  946. package/src/cache/cf/cf-cache-store.ts +2644 -75
  947. package/src/cache/cf/index.ts +17 -17
  948. package/src/cache/document-cache.ts +172 -92
  949. package/src/cache/handle-capture.ts +81 -0
  950. package/src/cache/handle-snapshot.ts +104 -0
  951. package/src/cache/index.ts +11 -35
  952. package/src/cache/memory-segment-store.ts +307 -30
  953. package/src/cache/profile-registry.ts +49 -0
  954. package/src/cache/read-through-swr.ts +164 -0
  955. package/src/cache/segment-codec.ts +240 -0
  956. package/src/cache/tag-invalidation.ts +230 -0
  957. package/src/cache/taint.ts +153 -0
  958. package/src/cache/types.ts +94 -211
  959. package/src/client.rsc.tsx +8 -21
  960. package/src/client.tsx +123 -347
  961. package/src/cloudflare/index.ts +11 -0
  962. package/src/cloudflare/tracing.ts +109 -0
  963. package/src/component-utils.ts +23 -4
  964. package/src/components/DefaultDocument.tsx +5 -1
  965. package/src/context-var.ts +168 -0
  966. package/src/debug.ts +19 -9
  967. package/src/decode-loader-results.ts +36 -0
  968. package/src/defer.ts +196 -0
  969. package/src/deps/ssr.ts +0 -1
  970. package/src/errors.ts +106 -10
  971. package/src/handle.ts +76 -23
  972. package/src/handles/MetaTags.tsx +73 -34
  973. package/src/handles/breadcrumbs.ts +77 -0
  974. package/src/handles/meta.ts +30 -52
  975. package/src/host/cookie-handler.ts +21 -51
  976. package/src/host/errors.ts +8 -32
  977. package/src/host/index.ts +12 -9
  978. package/src/host/pattern-matcher.ts +34 -77
  979. package/src/host/router.ts +151 -121
  980. package/src/host/testing.ts +45 -32
  981. package/src/host/types.ts +52 -11
  982. package/src/host/utils.ts +2 -2
  983. package/src/href-client.ts +192 -57
  984. package/src/index.rsc.ts +174 -34
  985. package/src/index.ts +241 -73
  986. package/src/internal-debug.ts +8 -4
  987. package/src/loader-store.ts +500 -0
  988. package/src/loader.rsc.ts +31 -99
  989. package/src/loader.ts +30 -12
  990. package/src/missing-id-error.ts +68 -0
  991. package/src/network-error-thrower.tsx +4 -7
  992. package/src/outlet-context.ts +1 -1
  993. package/src/outlet-provider.tsx +41 -0
  994. package/src/prerender/param-hash.ts +14 -13
  995. package/src/prerender/store.ts +121 -21
  996. package/src/prerender.ts +445 -24
  997. package/src/redirect-origin.ts +100 -0
  998. package/src/response-utils.ts +37 -0
  999. package/src/reverse.ts +198 -128
  1000. package/src/root-error-boundary.tsx +42 -48
  1001. package/src/route-content-wrapper.tsx +10 -72
  1002. package/src/route-definition/dsl-helpers.ts +1116 -0
  1003. package/src/route-definition/helper-factories.ts +90 -0
  1004. package/src/route-definition/helpers-types.ts +506 -0
  1005. package/src/route-definition/index.ts +55 -0
  1006. package/src/route-definition/redirect.ts +135 -0
  1007. package/src/route-definition/resolve-handler-use.ts +161 -0
  1008. package/src/route-definition/use-item-types.ts +32 -0
  1009. package/src/route-definition.ts +1 -1450
  1010. package/src/route-map-builder.ts +82 -144
  1011. package/src/route-name.ts +53 -0
  1012. package/src/route-types.ts +72 -41
  1013. package/src/router/basename.ts +14 -0
  1014. package/src/router/content-negotiation.ts +215 -0
  1015. package/src/router/debug-manifest.ts +72 -0
  1016. package/src/router/error-handling.ts +22 -26
  1017. package/src/router/find-match.ts +181 -0
  1018. package/src/router/handler-context.ts +372 -125
  1019. package/src/router/instrument.ts +346 -0
  1020. package/src/router/intercept-resolution.ts +34 -27
  1021. package/src/router/lazy-includes.ts +200 -0
  1022. package/src/router/loader-resolution.ts +381 -147
  1023. package/src/router/logging.ts +106 -6
  1024. package/src/router/manifest.ts +88 -51
  1025. package/src/router/match-api.ts +162 -245
  1026. package/src/router/match-context.ts +4 -24
  1027. package/src/router/match-handlers.ts +440 -0
  1028. package/src/router/match-middleware/background-revalidation.ts +90 -89
  1029. package/src/router/match-middleware/cache-lookup.ts +297 -150
  1030. package/src/router/match-middleware/cache-store.ts +75 -33
  1031. package/src/router/match-middleware/intercept-resolution.ts +44 -43
  1032. package/src/router/match-middleware/segment-resolution.ts +64 -22
  1033. package/src/router/match-pipelines.ts +11 -87
  1034. package/src/router/match-result.ts +121 -50
  1035. package/src/router/metrics.ts +219 -28
  1036. package/src/router/middleware-types.ts +93 -0
  1037. package/src/router/middleware.ts +421 -413
  1038. package/src/router/navigation-snapshot.ts +131 -0
  1039. package/src/router/params-util.ts +23 -0
  1040. package/src/router/pattern-matching.ts +263 -79
  1041. package/src/router/prerender-match.ts +541 -0
  1042. package/src/router/preview-match.ts +100 -0
  1043. package/src/router/request-classification.ts +276 -0
  1044. package/src/router/revalidation.ts +143 -44
  1045. package/src/router/route-snapshot.ts +244 -0
  1046. package/src/router/router-context.ts +41 -47
  1047. package/src/router/router-interfaces.ts +525 -0
  1048. package/src/router/router-options.ts +726 -0
  1049. package/src/router/router-registry.ts +21 -0
  1050. package/src/router/segment-resolution/fresh.ts +747 -0
  1051. package/src/router/segment-resolution/helpers.ts +314 -0
  1052. package/src/router/segment-resolution/loader-cache.ts +207 -0
  1053. package/src/router/segment-resolution/revalidation.ts +1322 -0
  1054. package/src/router/segment-resolution/static-store.ts +81 -0
  1055. package/src/router/segment-resolution/streamed-handler-telemetry.ts +52 -0
  1056. package/src/router/segment-resolution/view-transition-default.ts +36 -0
  1057. package/src/router/segment-resolution.ts +24 -1354
  1058. package/src/router/segment-wrappers.ts +289 -0
  1059. package/src/router/state-cookie-name.ts +33 -0
  1060. package/src/router/substitute-pattern-params.ts +56 -0
  1061. package/src/router/telemetry-otel.ts +261 -0
  1062. package/src/router/telemetry.ts +377 -0
  1063. package/src/router/timeout.ts +128 -0
  1064. package/src/router/tracing.ts +206 -0
  1065. package/src/router/trie-matching.ts +172 -60
  1066. package/src/router/types.ts +23 -70
  1067. package/src/router/url-params.ts +44 -0
  1068. package/src/router.ts +748 -2376
  1069. package/src/rsc/handler-context.ts +46 -0
  1070. package/src/rsc/handler.ts +861 -1141
  1071. package/src/rsc/helpers.ts +269 -19
  1072. package/src/rsc/index.ts +1 -21
  1073. package/src/rsc/json-route-result.ts +38 -0
  1074. package/src/rsc/loader-fetch.ts +235 -0
  1075. package/src/rsc/manifest-init.ts +77 -0
  1076. package/src/rsc/nonce.ts +14 -0
  1077. package/src/rsc/origin-guard.ts +155 -0
  1078. package/src/rsc/progressive-enhancement.ts +413 -0
  1079. package/src/rsc/redirect-guard.ts +99 -0
  1080. package/src/rsc/response-error.ts +104 -0
  1081. package/src/rsc/response-route-handler.ts +374 -0
  1082. package/src/rsc/rsc-rendering.ts +261 -0
  1083. package/src/rsc/runtime-warnings.ts +55 -0
  1084. package/src/rsc/server-action.ts +376 -0
  1085. package/src/rsc/ssr-setup.ts +144 -0
  1086. package/src/rsc/types.ts +58 -12
  1087. package/src/runtime-env.ts +18 -0
  1088. package/src/search-params.ts +70 -74
  1089. package/src/segment-content-promise.ts +67 -0
  1090. package/src/segment-loader-promise.ts +134 -0
  1091. package/src/segment-system.tsx +292 -134
  1092. package/src/serialize.ts +243 -0
  1093. package/src/server/context.ts +439 -85
  1094. package/src/server/cookie-store.ts +265 -0
  1095. package/src/server/fetchable-loader-store.ts +11 -6
  1096. package/src/server/handle-store.ts +112 -31
  1097. package/src/server/loader-registry.ts +23 -82
  1098. package/src/server/request-context.ts +724 -143
  1099. package/src/server.ts +26 -164
  1100. package/src/ssr/index.tsx +113 -36
  1101. package/src/static-handler.ts +45 -18
  1102. package/src/testing/cache-status.ts +162 -0
  1103. package/src/testing/collect-handle.ts +40 -0
  1104. package/src/testing/dispatch.ts +618 -0
  1105. package/src/testing/dom.entry.ts +22 -0
  1106. package/src/testing/e2e/fixture.ts +188 -0
  1107. package/src/testing/e2e/index.ts +128 -0
  1108. package/src/testing/e2e/matchers.ts +35 -0
  1109. package/src/testing/e2e/page-helpers.ts +272 -0
  1110. package/src/testing/e2e/parity.ts +387 -0
  1111. package/src/testing/e2e/server.ts +195 -0
  1112. package/src/testing/flight-matchers.ts +97 -0
  1113. package/src/testing/flight-normalize.ts +11 -0
  1114. package/src/testing/flight-runtime.d.ts +57 -0
  1115. package/src/testing/flight-tree.ts +682 -0
  1116. package/src/testing/flight.entry.ts +52 -0
  1117. package/src/testing/flight.ts +232 -0
  1118. package/src/testing/generated-routes.ts +183 -0
  1119. package/src/testing/index.ts +99 -0
  1120. package/src/testing/internal/context.ts +348 -0
  1121. package/src/testing/internal/flight-client-globals.ts +30 -0
  1122. package/src/testing/internal/seed-vars.ts +54 -0
  1123. package/src/testing/render-handler.ts +330 -0
  1124. package/src/testing/render-route.tsx +566 -0
  1125. package/src/testing/run-loader.ts +378 -0
  1126. package/src/testing/run-middleware.ts +205 -0
  1127. package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
  1128. package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
  1129. package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
  1130. package/src/testing/vitest-stubs/version.ts +5 -0
  1131. package/src/testing/vitest.ts +305 -0
  1132. package/src/theme/ThemeProvider.tsx +21 -67
  1133. package/src/theme/ThemeScript.tsx +5 -11
  1134. package/src/theme/constants.ts +5 -14
  1135. package/src/theme/index.ts +3 -20
  1136. package/src/theme/theme-context.ts +5 -35
  1137. package/src/theme/theme-script.ts +21 -32
  1138. package/src/theme/use-theme.ts +0 -3
  1139. package/src/types/boundaries.ts +123 -0
  1140. package/src/types/cache-types.ts +207 -0
  1141. package/src/types/error-types.ts +132 -0
  1142. package/src/types/global-namespace.ts +113 -0
  1143. package/src/types/handler-context.ts +839 -0
  1144. package/src/types/index.ts +79 -0
  1145. package/src/types/loader-types.ts +212 -0
  1146. package/src/types/request-scope.ts +107 -0
  1147. package/src/types/route-config.ts +126 -0
  1148. package/src/types/route-entry.ts +114 -0
  1149. package/src/types/segments.ts +171 -0
  1150. package/src/types.ts +1 -1795
  1151. package/src/urls/include-helper.ts +160 -0
  1152. package/src/urls/index.ts +43 -0
  1153. package/src/urls/path-helper-types.ts +386 -0
  1154. package/src/urls/path-helper.ts +275 -0
  1155. package/src/urls/pattern-types.ts +124 -0
  1156. package/src/urls/response-types.ts +109 -0
  1157. package/src/urls/type-extraction.ts +291 -0
  1158. package/src/urls/urls-function.ts +81 -0
  1159. package/src/urls.ts +1 -1323
  1160. package/src/use-loader.tsx +406 -141
  1161. package/src/vite/debug.ts +185 -0
  1162. package/src/vite/discovery/bundle-postprocess.ts +182 -0
  1163. package/src/vite/discovery/discover-routers.ts +389 -0
  1164. package/src/vite/discovery/discovery-errors.ts +194 -0
  1165. package/src/vite/discovery/gate-state.ts +171 -0
  1166. package/src/vite/discovery/prerender-collection.ts +467 -0
  1167. package/src/vite/discovery/route-types-writer.ts +214 -0
  1168. package/src/vite/discovery/self-gen-tracking.ts +73 -0
  1169. package/src/vite/discovery/state.ts +161 -0
  1170. package/src/vite/discovery/virtual-module-codegen.ts +183 -0
  1171. package/src/vite/index.ts +17 -2259
  1172. package/src/vite/plugin-types.ts +221 -0
  1173. package/src/vite/plugins/cjs-to-esm.ts +83 -0
  1174. package/src/vite/plugins/client-ref-dedup.ts +120 -0
  1175. package/src/vite/plugins/client-ref-hashing.ts +118 -0
  1176. package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
  1177. package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  1178. package/src/vite/plugins/cloudflare-protocol-stub.ts +194 -0
  1179. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +88 -110
  1180. package/src/vite/{expose-id-utils.ts → plugins/expose-id-utils.ts} +13 -87
  1181. package/src/vite/plugins/expose-ids/export-analysis.ts +338 -0
  1182. package/src/vite/plugins/expose-ids/handler-transform.ts +141 -0
  1183. package/src/vite/plugins/expose-ids/loader-transform.ts +57 -0
  1184. package/src/vite/plugins/expose-ids/router-transform.ts +124 -0
  1185. package/src/vite/plugins/expose-ids/types.ts +45 -0
  1186. package/src/vite/plugins/expose-internal-ids.ts +806 -0
  1187. package/src/vite/plugins/performance-tracks.ts +89 -0
  1188. package/src/vite/plugins/refresh-cmd.ts +127 -0
  1189. package/src/vite/plugins/use-cache-transform.ts +313 -0
  1190. package/src/vite/plugins/version-injector.ts +79 -0
  1191. package/src/vite/plugins/version-plugin.ts +275 -0
  1192. package/src/vite/plugins/virtual-entries.ts +108 -0
  1193. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  1194. package/src/vite/rango.ts +444 -0
  1195. package/src/vite/router-discovery.ts +1581 -0
  1196. package/src/vite/{ast-handler-extract.ts → utils/ast-handler-extract.ts} +193 -37
  1197. package/src/vite/utils/banner.ts +36 -0
  1198. package/src/vite/utils/bundle-analysis.ts +132 -0
  1199. package/src/vite/utils/client-chunks.ts +184 -0
  1200. package/src/vite/utils/forward-user-plugins.ts +171 -0
  1201. package/src/vite/utils/manifest-utils.ts +15 -0
  1202. package/src/vite/utils/package-resolution.ts +89 -0
  1203. package/src/vite/utils/prerender-utils.ts +223 -0
  1204. package/src/vite/utils/shared-utils.ts +219 -0
  1205. package/CLAUDE.md +0 -43
  1206. package/src/browser/lru-cache.ts +0 -69
  1207. package/src/browser/request-controller.ts +0 -164
  1208. package/src/browser/shallow.ts +0 -35
  1209. package/src/cache/memory-store.ts +0 -253
  1210. package/src/router.gen.ts +0 -6
  1211. package/src/static-handler.gen.ts +0 -5
  1212. package/src/urls.gen.ts +0 -8
  1213. package/src/vite/expose-internal-ids.ts +0 -1167
  1214. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -0,0 +1,1581 @@
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, register } from "node:module";
14
+ import { pathToFileURL } from "node:url";
15
+ import {
16
+ formatNestedRouterConflictError,
17
+ findNestedRouterConflict,
18
+ findRouterFiles,
19
+ createScanFilter,
20
+ } from "../build/generate-route-types.js";
21
+ import { firstCodeMatchIndex } from "../build/route-types/source-scan.js";
22
+ import { createVersionPlugin } from "./plugins/version-plugin.js";
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";
28
+ import {
29
+ exposeInternalIds,
30
+ exposeRouterId,
31
+ } from "./plugins/expose-internal-ids.js";
32
+ import { hashClientRefs } from "./plugins/client-ref-hashing.js";
33
+ import { extractHandlerExportsFromChunk } from "./utils/bundle-analysis.js";
34
+ import {
35
+ createDiscoveryState,
36
+ VIRTUAL_ROUTES_MANIFEST_ID,
37
+ type DiscoveryState,
38
+ type PluginOptions,
39
+ } from "./discovery/state.js";
40
+ import {
41
+ consumeSelfGenWrite,
42
+ peekSelfGenWrite,
43
+ } from "./discovery/self-gen-tracking.js";
44
+ import { discoverRouters } from "./discovery/discover-routers.js";
45
+ import {
46
+ writeCombinedRouteTypesWithTracking,
47
+ writeRouteTypesFiles,
48
+ supplementGenFilesWithRuntimeRoutes,
49
+ } from "./discovery/route-types-writer.js";
50
+ import {
51
+ generateRoutesManifestModule,
52
+ generatePerRouterModule,
53
+ } from "./discovery/virtual-module-codegen.js";
54
+ import { postprocessBundle } from "./discovery/bundle-postprocess.js";
55
+ import { createDiscoveryGate } from "./discovery/gate-state.js";
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);
68
+
69
+ export { VIRTUAL_ROUTES_MANIFEST_ID };
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
+
114
+ // ============================================================================
115
+ // Temp Server Factory
116
+ // ============================================================================
117
+
118
+ /**
119
+ * Create a minimal Vite server for router discovery.
120
+ *
121
+ * Both dev-mode prerender and build-mode discovery need a temp RSC server
122
+ * to import user router files via module runner. This factory centralizes
123
+ * the shared config and the mode-specific differences:
124
+ * - Dev: path-based IDs (no forceBuild), separate cacheDir
125
+ * - Build: hashed IDs (forceBuild), hashClientRefs for production bundles
126
+ *
127
+ * Returns the ViteDevServer instance. Callers access .environments.rsc as needed.
128
+ */
129
+ async function createTempRscServer(
130
+ state: DiscoveryState,
131
+ options: { forceBuild?: boolean; cacheDir?: string } = {},
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();
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
+ };
150
+ return createViteServer({
151
+ root: state.projectRoot,
152
+ configFile: false,
153
+ server: { middlewareMode: true },
154
+ appType: "custom",
155
+ logLevel: "silent",
156
+ resolve: resolveConfig,
157
+ ...(runnerConfig?.define ? { define: runnerConfig.define } : {}),
158
+ oxc: oxcConfig as any,
159
+ ...(options.cacheDir && { cacheDir: options.cacheDir }),
160
+ plugins: [
161
+ rsc({
162
+ entries: {
163
+ client: "virtual:entry-client",
164
+ ssr: "virtual:entry-ssr",
165
+ rsc: state.resolvedEntryPath!,
166
+ },
167
+ }),
168
+ // hashClientRefs only in build mode — production bundles need hashed refs
169
+ ...(options.forceBuild ? [hashClientRefs(state.projectRoot)] : []),
170
+ createVersionPlugin(),
171
+ createVirtualStubPlugin(),
172
+ createCloudflareProtocolStubPlugin(),
173
+ // Dev prerender must use dev-mode IDs (path-based) to match the workerd
174
+ // runtime. forceBuild produces hashed IDs for production bundle consistency.
175
+ exposeInternalIds(options.forceBuild ? { forceBuild: true } : undefined),
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,
181
+ ],
182
+ });
183
+ }
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
+
290
+ /**
291
+ * Plugin that discovers router instances at dev/build time via the RSC environment.
292
+ *
293
+ * Uses `server.environments.rsc.runner.import()` to load the user's router file
294
+ * with full TS/TSX compilation. This triggers `createRouter()` which populates
295
+ * the `RouterRegistry`. The plugin then generates manifests for each router.
296
+ *
297
+ * In dev mode, this runs in `configureServer` (post-middleware setup).
298
+ * In build mode, this will run in `buildStart` (future).
299
+ *
300
+ * @internal
301
+ */
302
+ export function createRouterDiscoveryPlugin(
303
+ entryPath: string | undefined,
304
+ opts?: PluginOptions,
305
+ ): Plugin {
306
+ const s = createDiscoveryState(entryPath, opts);
307
+ let viteCommand: "serve" | "build" = "build";
308
+ let viteMode = "production";
309
+
310
+ return {
311
+ name: "@rangojs/router:discovery",
312
+
313
+ config() {
314
+ const config: any = {
315
+ define: {
316
+ __RANGO_DEBUG__: JSON.stringify(!!process.env.INTERNAL_RANGO_DEBUG),
317
+ },
318
+ };
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.
325
+ return config;
326
+ },
327
+
328
+ configResolved(config) {
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;
336
+ s.isBuildMode = config.command === "build";
337
+ viteCommand = config.command as "serve" | "build";
338
+ viteMode = config.mode;
339
+ // Capture user's resolve aliases for the temp server
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
+ );
353
+ // Node preset: pick up auto-discovered router path from the config() hook.
354
+ // The auto-discover plugin runs in config() using Vite's resolved root,
355
+ // populating the mutable ref before configResolved fires.
356
+ if (!s.resolvedEntryPath && opts?.routerPathRef?.path) {
357
+ s.resolvedEntryPath = opts.routerPathRef.path;
358
+ }
359
+ // Cloudflare preset: entry comes from the resolved RSC env config.
360
+ if (!s.resolvedEntryPath) {
361
+ const entry = resolveRscEntryFromConfig(config);
362
+ if (entry) s.resolvedEntryPath = entry;
363
+ }
364
+ // Generate combined named-routes.gen.ts from static source parsing.
365
+ // Runs before the dev server starts so the gen file exists immediately for IDE.
366
+ // In build mode, the runtime discovery in buildStart produces the definitive
367
+ // named-routes.gen.ts (including dynamically generated routes).
368
+ // preserveIfLarger prevents overwriting a previously generated complete
369
+ // file with a partial one.
370
+ if (opts?.staticRouteTypesGeneration !== false) {
371
+ s.cachedRouterFiles = findRouterFiles(s.projectRoot, s.scanFilter);
372
+ writeCombinedRouteTypesWithTracking(s, { preserveIfLarger: true });
373
+ }
374
+ // Resolve prerenderHandlerModules and staticHandlerModules from the consolidated IDs plugin's API.
375
+ if (opts?.enableBuildPrerender) {
376
+ const idsPlugin = config.plugins.find(
377
+ (p: any) => p.name === "@rangojs/router:expose-internal-ids",
378
+ );
379
+ s.resolvedPrerenderModules = (
380
+ idsPlugin?.api as any
381
+ )?.prerenderHandlerModules;
382
+ s.resolvedStaticModules = (idsPlugin?.api as any)?.staticHandlerModules;
383
+ }
384
+ },
385
+
386
+ // Dev mode: discover routers and populate manifest in memory.
387
+ // Skipped in build mode (buildStart handles it).
388
+ configureServer(server) {
389
+ if (s.isBuildMode) return;
390
+ // Skip if this is a temp server created by buildStart
391
+ if ((globalThis as any).__rscRouterDiscoveryActive) return;
392
+ s.devServer = server;
393
+
394
+ // Discovery promise that the handler can await if requests arrive
395
+ // before discovery completes
396
+ let resolveDiscovery: () => void;
397
+ const discoveryPromise = new Promise<void>((resolve) => {
398
+ resolveDiscovery = resolve;
399
+ });
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
+
412
+ // Compute dev server origin from resolved URLs (preferred) or config port (fallback).
413
+ // Called after discovery (or in the load hook) when the server may be listening.
414
+ const getDevServerOrigin = () =>
415
+ server.resolvedUrls?.local?.[0]?.replace(/\/$/, "") ||
416
+ `http://localhost:${server.config.server.port || 5173}`;
417
+
418
+ // Shared temp server for Cloudflare dev (no module runner in workerd).
419
+ // Used by both discover() (route type generation) and the prerender
420
+ // middleware (on-demand prerender evaluation). Created lazily, closed on
421
+ // server shutdown.
422
+ let prerenderTempServer: any = null;
423
+ let prerenderNodeRegistry: Map<string, any> | null = null;
424
+
425
+ // Clean up the temporary server and build env when the dev server shuts down
426
+ server.httpServer?.on("close", () => {
427
+ if (prerenderTempServer) {
428
+ prerenderTempServer.close().catch(() => {});
429
+ prerenderTempServer = null;
430
+ }
431
+ releaseBuildEnv(s).catch(() => {});
432
+ });
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
+
474
+ async function getOrCreateTempServer(): Promise<any | 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
+ }
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
+ );
532
+ try {
533
+ prerenderTempServer = await createTempRscServer(s, {
534
+ cacheDir: "node_modules/.vite_prerender",
535
+ });
536
+
537
+ const tempRscEnv = (prerenderTempServer.environments as any)?.rsc;
538
+ if (tempRscEnv?.runner) {
539
+ await importEntryAndRegistry(tempRscEnv);
540
+ return tempRscEnv;
541
+ }
542
+ debugDiscovery?.(
543
+ "getOrCreateTempServer: tempRscEnv.runner unavailable",
544
+ );
545
+ } catch (err: any) {
546
+ debugDiscovery?.(
547
+ "getOrCreateTempServer: FAILED message=%s",
548
+ err.message,
549
+ );
550
+ console.warn(`[rango] Failed to create temp runner: ${err.message}`);
551
+ }
552
+ return null;
553
+ }
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
+
653
+ const discover = async () => {
654
+ const discoverStart = performance.now();
655
+ const rscEnv = (server.environments as any)?.rsc;
656
+ if (!rscEnv?.runner) {
657
+ // Cloudflare dev: no module runner available (workerd-based RSC env).
658
+ // Set devServerOrigin so the virtual module can inject __PRERENDER_DEV_URL
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
+ );
664
+ s.devServerOrigin = getDevServerOrigin();
665
+
666
+ // Create a temp Node.js server to run runtime discovery and generate
667
+ // named route types (static parser can't resolve factory calls).
668
+ try {
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
+ );
679
+ if (tempRscEnv) {
680
+ await timed(debugDiscovery, "discoverRouters (cloudflare)", () =>
681
+ discoverRouters(s, tempRscEnv),
682
+ );
683
+ timedSync(debugDiscovery, "writeRouteTypesFiles", () =>
684
+ writeRouteTypesFiles(s),
685
+ );
686
+ }
687
+ } catch (err: any) {
688
+ console.warn(
689
+ `[rango] Cloudflare dev discovery failed: ${err.message}\n${err.stack}`,
690
+ );
691
+ }
692
+
693
+ debugDiscovery?.(
694
+ "dev discovery done (%sms)",
695
+ (performance.now() - discoverStart).toFixed(1),
696
+ );
697
+ resolveDiscovery!();
698
+ return;
699
+ }
700
+
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
+
708
+ // Set the readiness gate BEFORE discovery so early requests
709
+ // block until manifest is populated
710
+ const serverMod = await timed(
711
+ debugDiscovery,
712
+ "import @rangojs/router/server",
713
+ () => rscEnv.runner.import("@rangojs/router/server"),
714
+ );
715
+ if (serverMod?.setManifestReadyPromise) {
716
+ serverMod.setManifestReadyPromise(discoveryPromise);
717
+ }
718
+
719
+ await timed(debugDiscovery, "discoverRouters", () =>
720
+ discoverRouters(s, rscEnv),
721
+ );
722
+
723
+ // Store server origin for dev prerender endpoint (virtual module injection)
724
+ s.devServerOrigin = getDevServerOrigin();
725
+
726
+ // Update named-routes.gen.ts from runtime discovery.
727
+ // The runtime manifest is the source of truth: it evaluates dynamic
728
+ // routes (e.g. Array.from loops) that the static parser cannot see.
729
+ // writeRouteTypesFiles() only writes when content changes, so this
730
+ // won't cause unnecessary HMR triggers.
731
+ timedSync(debugDiscovery, "writeRouteTypesFiles", () =>
732
+ writeRouteTypesFiles(s),
733
+ );
734
+
735
+ // Populate the route map and per-router data in the RSC env
736
+ await timed(debugDiscovery, "propagateDiscoveryState", () =>
737
+ propagateDiscoveryState(rscEnv),
738
+ );
739
+ } catch (err: any) {
740
+ console.warn(
741
+ `[rango] Router discovery failed: ${err.message}\n${err.stack}`,
742
+ );
743
+ } finally {
744
+ debugDiscovery?.(
745
+ "dev discovery done (%sms)",
746
+ (performance.now() - discoverStart).toFixed(1),
747
+ );
748
+ resolveDiscovery!();
749
+ }
750
+ };
751
+
752
+ // Schedule after all plugins have finished configureServer.
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
+ );
761
+
762
+ // Dev-mode on-demand prerender endpoint.
763
+ // When workerd hits a prerender route, it fetches this endpoint instead of
764
+ // trying to run node:fs-dependent handlers in the Cloudflare environment.
765
+ //
766
+ // Node.js preset: uses the main server's RSC environment directly (router
767
+ // instances are already discovered and have matchForPrerender).
768
+ // Cloudflare preset: lazily creates a Node.js temp server because the main
769
+ // RSC environment uses workerd where node:fs can't access the host filesystem.
770
+
771
+ // Registry from the main server's RSC environment (populated by discoverRouters)
772
+ let mainRegistry: Map<string, any> | null = null;
773
+
774
+ // Push discovery state (manifest, trie, precomputed entries) to the
775
+ // server module so runtime request handling uses the current routes.
776
+ // Shared by initial discovery and HMR-triggered re-discovery.
777
+ const propagateDiscoveryState = async (rscEnv: any) => {
778
+ const serverMod = await rscEnv.runner.import("@rangojs/router/server");
779
+ if (!serverMod) return;
780
+ // Clear stale per-router and global route data before repopulating.
781
+ // Without this, removed routers/routes survive in the per-router maps
782
+ // and shrunk precomputed entries or tries are never purged.
783
+ if (serverMod.clearAllRouterData) {
784
+ serverMod.clearAllRouterData();
785
+ }
786
+ mainRegistry = serverMod.RouterRegistry ?? null;
787
+ if (s.mergedRouteManifest && serverMod.setCachedManifest) {
788
+ serverMod.setCachedManifest(s.mergedRouteManifest);
789
+ }
790
+ if (
791
+ s.mergedPrecomputedEntries &&
792
+ s.mergedPrecomputedEntries.length > 0 &&
793
+ serverMod.setPrecomputedEntries
794
+ ) {
795
+ serverMod.setPrecomputedEntries(s.mergedPrecomputedEntries);
796
+ }
797
+ if (s.mergedRouteTrie && serverMod.setRouteTrie) {
798
+ serverMod.setRouteTrie(s.mergedRouteTrie);
799
+ }
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);
809
+ }
810
+ };
811
+
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
+
824
+ if (s.discoveryDone) await s.discoveryDone;
825
+
826
+ const url = new URL(req.url || "/", "http://localhost");
827
+ const pathname = url.searchParams.get("pathname");
828
+ if (!pathname) {
829
+ res.statusCode = 400;
830
+ res.end("Missing pathname");
831
+ logResult(400, "missing pathname");
832
+ return;
833
+ }
834
+
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
+ }
861
+
862
+ if (!registry) {
863
+ // No main registry: the RSC env has no module runner (Cloudflare dev).
864
+ // Lazily create a Node.js temp server for prerender evaluation.
865
+ if (!prerenderNodeRegistry) {
866
+ await getOrCreateTempServer();
867
+ }
868
+ registry = prerenderNodeRegistry;
869
+ }
870
+
871
+ if (!registry || registry.size === 0) {
872
+ res.statusCode = 503;
873
+ res.end("Prerender runner not available");
874
+ logResult(503, "no registry");
875
+ return;
876
+ }
877
+
878
+ const wantIntercept = url.searchParams.get("intercept") === "1";
879
+ const wantRouteName = url.searchParams.get("routeName");
880
+ const wantPassthrough = url.searchParams.get("passthrough") === "1";
881
+
882
+ for (const [, routerInstance] of registry) {
883
+ if (!routerInstance.matchForPrerender) continue;
884
+ try {
885
+ const result = await routerInstance.matchForPrerender(
886
+ pathname,
887
+ {},
888
+ undefined,
889
+ wantPassthrough,
890
+ s.resolvedBuildEnv,
891
+ true, // devMode: check getParams for passthrough routes
892
+ );
893
+ if (!result) continue;
894
+ if (result.passthrough) continue;
895
+ // When routeName is specified, only accept a match for that route.
896
+ // This prevents returning the wrong entry when multiple routers
897
+ // have prerenderable routes sharing the same pathname.
898
+ if (wantRouteName && result.routeName !== wantRouteName) continue;
899
+ res.setHeader("content-type", "application/json");
900
+ let payload: Record<string, unknown>;
901
+ if (wantIntercept && result.interceptSegments?.length) {
902
+ payload = {
903
+ segments: [...result.segments, ...result.interceptSegments],
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 ?? "",
907
+ };
908
+ } else {
909
+ payload = { segments: result.segments, handles: result.handles };
910
+ }
911
+ res.end(JSON.stringify(payload));
912
+ logResult(200, `match ${result.routeName}`);
913
+ return;
914
+ } catch (err: any) {
915
+ // matchForPrerender now re-throws render failures instead of baking
916
+ // an error page (issue #587). In dev there is no frozen artifact, so
917
+ // we fall through (404 -> live handler). A `throw new Skip()` is the
918
+ // expected "skip this URL" signal, not a failure, so it stays quiet.
919
+ if (err?.name !== "Skip") {
920
+ console.warn(
921
+ `[rango] Dev prerender error for ${pathname} (serving live instead): ${err.message}`,
922
+ );
923
+ }
924
+ }
925
+ }
926
+
927
+ res.statusCode = 404;
928
+ res.end("No prerender match");
929
+ logResult(404, "no match");
930
+ });
931
+
932
+ // Watch url module and router files for changes and regenerate named-routes.gen.ts.
933
+ // Process files containing urls( or createRouter( to update the combined route map.
934
+ if (opts?.staticRouteTypesGeneration !== false) {
935
+ const isGeneratedRouteFile = (filePath: string): boolean =>
936
+ filePath.endsWith(".gen.ts") &&
937
+ (filePath.includes("named-routes.gen.ts") ||
938
+ filePath.includes("urls.gen.ts"));
939
+
940
+ const regenerateGeneratedRouteFiles = () => {
941
+ if (s.perRouterManifests.length > 0) {
942
+ writeRouteTypesFiles(s);
943
+ } else {
944
+ writeCombinedRouteTypesWithTracking(s);
945
+ }
946
+ };
947
+
948
+ const maybeHandleGeneratedRouteFileMutation = (
949
+ filePath: string,
950
+ ): boolean => {
951
+ if (!isGeneratedRouteFile(filePath)) return false;
952
+ if (consumeSelfGenWrite(s, filePath)) return true;
953
+ // In Cloudflare dev (no module runner), perRouterManifests is never
954
+ // refreshed after HMR so regenerateGeneratedRouteFiles() would use
955
+ // stale data and revert user edits. Source files own route state;
956
+ // gen files are derived output. Skip regeneration and let the next
957
+ // source-file change rebuild them from the static parser.
958
+ const hasRunner = !!(server.environments as any)?.rsc?.runner;
959
+ if (!hasRunner) return true;
960
+ regenerateGeneratedRouteFiles();
961
+ return true;
962
+ };
963
+
964
+ // Debounce timer for batching rapid route-file changes (e.g. afterEach
965
+ // restoring two files in quick succession). The cheap checks (extension,
966
+ // scanFilter, content sniff) run synchronously to gate non-route files;
967
+ // only the expensive regeneration is debounced.
968
+ let routeChangeTimer: ReturnType<typeof setTimeout> | undefined;
969
+
970
+ // Re-run runtime discovery so factory-generated routes that the
971
+ // static parser cannot see are refreshed after source changes.
972
+ // The state-machine concerns (queued/pending/gatePending) are
973
+ // owned by the gate created above (./discovery/gate-state.ts).
974
+ // Here we provide just the env-specific work.
975
+ const refreshRuntimeDiscovery = async () => {
976
+ const rscEnv = (server.environments as any)?.rsc;
977
+ const hasMainRunner = !!rscEnv?.runner;
978
+ // Cloudflare HMR has no main RSC runner (workerd is a separate
979
+ // runtime). When we have a populated runtime manifest from cold
980
+ // start, we can re-discover via the temp Node runner — the same
981
+ // mechanism getOrCreateTempServer() uses at startup. Without a
982
+ // populated manifest there's nothing useful to do, so bail
983
+ // before involving the gate machine at all.
984
+ if (!hasMainRunner && s.perRouterManifests.length === 0) return;
985
+ await gate.runRefreshCycle(async () => {
986
+ const hmrStart = performance.now();
987
+ try {
988
+ if (hasMainRunner) {
989
+ await timed(debugDiscovery, "hmr discoverRouters", () =>
990
+ discoverRouters(s, rscEnv),
991
+ );
992
+ timedSync(debugDiscovery, "hmr writeRouteTypesFiles", () =>
993
+ writeRouteTypesFiles(s),
994
+ );
995
+ await timed(debugDiscovery, "hmr propagateDiscoveryState", () =>
996
+ propagateDiscoveryState(rscEnv),
997
+ );
998
+ } else {
999
+ // Cloudflare HMR: invalidate the temp server's RSC module
1000
+ // graph (or close+recreate as a fallback) so the runner
1001
+ // re-reads the freshly edited source. Keeping the same
1002
+ // Vite instance alive avoids disrupting workerd's transport
1003
+ // during the first post-cold-start module-fetch window.
1004
+ const tempRscEnv = await timed(
1005
+ debugDiscovery,
1006
+ "hmr refreshTempRscEnv (cloudflare)",
1007
+ () => refreshTempRscEnv(),
1008
+ );
1009
+ if (!tempRscEnv) {
1010
+ throw new Error(
1011
+ "temp runner unavailable for cloudflare HMR rediscovery",
1012
+ );
1013
+ }
1014
+ await timed(
1015
+ debugDiscovery,
1016
+ "hmr discoverRouters (cloudflare)",
1017
+ () => discoverRouters(s, tempRscEnv),
1018
+ );
1019
+ timedSync(debugDiscovery, "hmr writeRouteTypesFiles", () =>
1020
+ writeRouteTypesFiles(s),
1021
+ );
1022
+ }
1023
+ if (s.lastDiscoveryError) {
1024
+ debugDiscovery?.(
1025
+ "hmr: cleared lastDiscoveryError (%s) after successful rediscovery",
1026
+ s.lastDiscoveryError.message,
1027
+ );
1028
+ s.lastDiscoveryError = null;
1029
+ }
1030
+ // Cloudflare dev: on a successful cycle drop the workerd runner's
1031
+ // cached worker-entry chain so the next request re-evaluates
1032
+ // createRouter() with the new routes. Fired here in the work path
1033
+ // (not the caller's .then()) so a queued follow-up cycle that
1034
+ // succeeds after an earlier failed cycle still reloads:
1035
+ // runRefreshCycle recurses queued work without awaiting it, so the
1036
+ // original call already resolved on the failed cycle. A failed
1037
+ // cycle throws above and never reaches here, so a broken edit
1038
+ // never reloads the worker onto bad source.
1039
+ if (rscEnv && !rscEnv.runner) forceCloudflareWorkerReload(rscEnv);
1040
+ } catch (err: any) {
1041
+ s.lastDiscoveryError = {
1042
+ message: err?.message ?? String(err),
1043
+ at: Date.now(),
1044
+ };
1045
+ console.warn(
1046
+ `[rango] Runtime re-discovery failed: ${err.message}`,
1047
+ );
1048
+ debugDiscovery?.(
1049
+ "hmr: lastDiscoveryError set (%s) — manifest preserved at last-good; recovery mode active (any in-scan source change will trigger rediscovery)",
1050
+ err?.message,
1051
+ );
1052
+ } finally {
1053
+ debugDiscovery?.(
1054
+ "hmr re-discovery done (%sms)",
1055
+ (performance.now() - hmrStart).toFixed(1),
1056
+ );
1057
+ }
1058
+ });
1059
+ };
1060
+
1061
+ // Cloudflare dev only. workerd serves every request through the
1062
+ // runner-worker singleton, which re-resolves the worker entry per
1063
+ // request via runner.import("virtual:cloudflare/worker-entry"). The
1064
+ // route table lives in the user's createRouter() instance, captured
1065
+ // when that entry chain (entry -> router -> urls) was last evaluated
1066
+ // and then cached in the runner's evaluatedModules. The route-file
1067
+ // watcher refreshes discovery + types on the Node side, but the worker
1068
+ // keeps serving the cached (stale) router: route-definition modules
1069
+ // have no import.meta.hot boundary, so Vite never sends the worker an
1070
+ // HMR update for them and the entry chain is never evicted.
1071
+ //
1072
+ // Fix: after discovery completes, (1) invalidate the worker env's
1073
+ // Node-side module graph, then (2) send a full-reload to the worker.
1074
+ // Step (2) alone is insufficient: the full-reload handler clears the
1075
+ // runner's evaluatedModules and re-imports entrypoints, but each
1076
+ // re-import fetches the module back through this Node-side graph, which
1077
+ // still holds the pre-edit transform of urls.tsx — so createRouter()
1078
+ // rebuilds the stale route table and the new route 404s/hits the
1079
+ // catch-all. Invalidating the graph forces a fresh transform on
1080
+ // re-fetch (the same mechanism refreshTempRscEnv uses for discovery),
1081
+ // so the re-import re-runs createRouter() with the new routes. This is
1082
+ // the programmatic equivalent of the dev-server "r + enter" restart,
1083
+ // scoped to the worker environment instead of tearing down the server.
1084
+ const forceCloudflareWorkerReload = (rscEnv: any) => {
1085
+ if (!rscEnv?.hot) return;
1086
+ try {
1087
+ const graph = rscEnv.moduleGraph;
1088
+ if (graph?.invalidateAll) {
1089
+ graph.invalidateAll();
1090
+ debugDiscovery?.("hmr: invalidated workerd rsc module graph");
1091
+ }
1092
+ rscEnv.hot.send({ type: "full-reload" });
1093
+ debugDiscovery?.(
1094
+ "hmr: forced workerd rsc env reload (full-reload)",
1095
+ );
1096
+ } catch (err: any) {
1097
+ debugDiscovery?.(
1098
+ "hmr: workerd reload failed: %s",
1099
+ err?.message ?? err,
1100
+ );
1101
+ }
1102
+ };
1103
+
1104
+ const scheduleRouteRegeneration = () => {
1105
+ clearTimeout(routeChangeTimer);
1106
+ routeChangeTimer = setTimeout(() => {
1107
+ routeChangeTimer = undefined;
1108
+ const regenStart = debugDiscovery ? performance.now() : 0;
1109
+ const rscEnv = (server.environments as any)?.rsc;
1110
+ const skipStaticWrite =
1111
+ !rscEnv?.runner && s.perRouterManifests.length > 0;
1112
+ try {
1113
+ // In cloudflare dev with a populated runtime manifest, the
1114
+ // static parser produces a strictly smaller (and actively
1115
+ // wrong) gen file — supplementGenFilesWithRuntimeRoutes can
1116
+ // only restore factory-only prefixes, and apps with mixed
1117
+ // static+factory routes under shared prefixes (cf-stress)
1118
+ // collapse to the 19-route static view. Skip the static
1119
+ // write entirely; runtime rediscovery below will overwrite
1120
+ // the gen file with the authoritative manifest.
1121
+ if (skipStaticWrite) {
1122
+ debugDiscovery?.(
1123
+ "watcher: skipping static write (cloudflare HMR — runtime rediscovery owns gen file)",
1124
+ );
1125
+ } else {
1126
+ writeCombinedRouteTypesWithTracking(s);
1127
+ if (s.perRouterManifests.length > 0) {
1128
+ supplementGenFilesWithRuntimeRoutes(s);
1129
+ }
1130
+ }
1131
+ } catch (err: any) {
1132
+ console.error(`[rango] Route regeneration error: ${err.message}`);
1133
+ }
1134
+ debugDiscovery?.(
1135
+ "watcher: regenerated gen files (%sms)",
1136
+ (performance.now() - regenStart).toFixed(1),
1137
+ );
1138
+ // Async: re-run runtime discovery to refresh factory-generated
1139
+ // routes that the static parser cannot resolve. Resolves the
1140
+ // discovery gate when complete.
1141
+ if (s.perRouterManifests.length > 0) {
1142
+ // The cloudflare workerd reload fires inside refreshRuntimeDiscovery
1143
+ // on the successful cycle (see forceCloudflareWorkerReload call
1144
+ // there) so queued follow-up cycles also trigger it.
1145
+ refreshRuntimeDiscovery().catch((err: any) => {
1146
+ console.warn(
1147
+ `[rango] Runtime re-discovery error: ${err.message}`,
1148
+ );
1149
+ // Even on error, unblock the gate so workerd's reload doesn't
1150
+ // hang indefinitely against the previous manifest.
1151
+ resolveDiscoveryGate();
1152
+ });
1153
+ }
1154
+ }, 100);
1155
+ };
1156
+
1157
+ const handleRouteFileChange = (filePath: string) => {
1158
+ if (maybeHandleGeneratedRouteFileMutation(filePath)) return;
1159
+ if (
1160
+ !filePath.endsWith(".ts") &&
1161
+ !filePath.endsWith(".tsx") &&
1162
+ !filePath.endsWith(".js") &&
1163
+ !filePath.endsWith(".jsx")
1164
+ ) {
1165
+ if (s.lastDiscoveryError) {
1166
+ debugDiscovery?.(
1167
+ "watcher: skip non-source %s [LASTERR %s]",
1168
+ filePath,
1169
+ s.lastDiscoveryError.message,
1170
+ );
1171
+ }
1172
+ return;
1173
+ }
1174
+ // Apply scan filter as early-exit before reading file
1175
+ if (s.scanFilter && !s.scanFilter(filePath)) {
1176
+ if (s.lastDiscoveryError) {
1177
+ debugDiscovery?.(
1178
+ "watcher: skip scan-filter %s [LASTERR %s]",
1179
+ filePath,
1180
+ s.lastDiscoveryError.message,
1181
+ );
1182
+ }
1183
+ return;
1184
+ }
1185
+ // Recovery mode: when the previous HMR re-discovery failed, the
1186
+ // import graph is incomplete and the manifest is stuck at the
1187
+ // last-good state. The fix may land in a non-route file (e.g. a
1188
+ // helper imported by the router, a missing module being created,
1189
+ // or a "use client" component) that the narrow content sniff
1190
+ // would otherwise filter out. While in recovery, treat any
1191
+ // in-scan source change as a candidate for rediscovery; the
1192
+ // tighter filter resumes once discovery succeeds again.
1193
+ const inRecoveryMode = !!s.lastDiscoveryError;
1194
+ try {
1195
+ const source = readFileSync(filePath, "utf-8");
1196
+ const trimmed = source.trimStart();
1197
+ const isUseClient =
1198
+ trimmed.startsWith('"use client"') ||
1199
+ trimmed.startsWith("'use client'");
1200
+ if (!inRecoveryMode && isUseClient) return;
1201
+ // Cheap raw pre-check first; only when a candidate token is present
1202
+ // do we confirm it occurs in real code (not a comment/string) via a
1203
+ // single allocation-free code-region scan. Most saved files contain
1204
+ // neither token and skip the scan entirely. This avoids a comment or
1205
+ // string mention spuriously marking a file relevant and triggering an
1206
+ // unnecessary re-discovery on save.
1207
+ let hasUrls = source.includes("urls(");
1208
+ let hasCreateRouter = /\bcreateRouter\s*[<(]/.test(source);
1209
+ if (hasUrls) hasUrls = firstCodeMatchIndex(source, /urls\(/g) >= 0;
1210
+ if (hasCreateRouter) {
1211
+ hasCreateRouter =
1212
+ firstCodeMatchIndex(source, /\bcreateRouter\s*[<(]/g) >= 0;
1213
+ }
1214
+ if (!inRecoveryMode && !hasUrls && !hasCreateRouter) return;
1215
+ if (inRecoveryMode) {
1216
+ debugDiscovery?.(
1217
+ "watcher: recovery rediscovery for %s (urls=%s, router=%s, useClient=%s) [LASTERR %s]",
1218
+ filePath,
1219
+ hasUrls,
1220
+ hasCreateRouter,
1221
+ isUseClient,
1222
+ s.lastDiscoveryError!.message,
1223
+ );
1224
+ } else {
1225
+ debugDiscovery?.(
1226
+ "watcher: %s matches (urls=%s, router=%s)",
1227
+ filePath,
1228
+ hasUrls,
1229
+ hasCreateRouter,
1230
+ );
1231
+ }
1232
+ // Invalidate cache when a router file changes (new router added/removed)
1233
+ if (hasCreateRouter) {
1234
+ const nestedRouterConflict = findNestedRouterConflict([
1235
+ ...(s.cachedRouterFiles ?? []),
1236
+ resolve(filePath),
1237
+ ]);
1238
+ if (nestedRouterConflict) {
1239
+ server.config.logger.error(
1240
+ formatNestedRouterConflictError(nestedRouterConflict),
1241
+ );
1242
+ return;
1243
+ }
1244
+ s.cachedRouterFiles = undefined;
1245
+ }
1246
+ // Note the event in the gate machine IMMEDIATELY (before the
1247
+ // 100ms debounce and any downstream HMR fanout). This sets
1248
+ // both `pendingEvents` (so refresh's finally holds the gate
1249
+ // through the tail window even if no rediscovery is queued)
1250
+ // and resets `discoveryDone` to a fresh pending promise (so
1251
+ // workerd reloads triggered by the same source change can't
1252
+ // observe a stale resolved gate from cold-start). Resolved
1253
+ // by the trailing refreshRuntimeDiscovery() cycle.
1254
+ if (s.perRouterManifests.length > 0) {
1255
+ gate.noteRouteEvent();
1256
+ }
1257
+ scheduleRouteRegeneration();
1258
+ } catch (readErr: any) {
1259
+ if (s.lastDiscoveryError) {
1260
+ debugDiscovery?.(
1261
+ "watcher: read error %s: %s [LASTERR %s]",
1262
+ filePath,
1263
+ readErr?.message,
1264
+ s.lastDiscoveryError.message,
1265
+ );
1266
+ }
1267
+ // Ignore read errors for deleted/moved files
1268
+ }
1269
+ };
1270
+
1271
+ // Handle both "add" and "change" events: editors with atomic saves
1272
+ // (unlink + rename) emit "add" instead of "change", and chokidar's
1273
+ // polling mode on CI Linux can also emit "add" for overwrites.
1274
+ server.watcher.on("add", handleRouteFileChange);
1275
+ server.watcher.on("change", handleRouteFileChange);
1276
+
1277
+ // Regenerate gen files when they are deleted (e.g. manual cleanup).
1278
+ // Same no-runner guard as change/add: stale perRouterManifests would
1279
+ // reintroduce reverted content.
1280
+ server.watcher.on("unlink", (filePath) => {
1281
+ if (!isGeneratedRouteFile(filePath)) return;
1282
+ const hasRunner = !!(server.environments as any)?.rsc?.runner;
1283
+ if (!hasRunner) return;
1284
+ regenerateGeneratedRouteFiles();
1285
+ });
1286
+ }
1287
+ },
1288
+
1289
+ // Build mode: create a temporary Vite dev server to access the RSC
1290
+ // environment's module runner, then discover routers and generate manifests.
1291
+ // The manifest data is stored for the virtual module's load hook.
1292
+ async buildStart() {
1293
+ if (!s.isBuildMode) return;
1294
+ // Only run once across environment builds
1295
+ if (s.mergedRouteManifest !== null) {
1296
+ debugDiscovery?.(
1297
+ "build: skip (already discovered, env=%s)",
1298
+ this.environment?.name ?? "?",
1299
+ );
1300
+ return;
1301
+ }
1302
+ const buildStartTime = performance.now();
1303
+ debugDiscovery?.("build: start (env=%s)", this.environment?.name ?? "?");
1304
+ resetStagedBuildAssets(s.projectRoot);
1305
+ s.prerenderManifestEntries = null;
1306
+ s.staticManifestEntries = null;
1307
+
1308
+ // Acquire build-time env bindings if configured
1309
+ await timed(debugDiscovery, "build acquireBuildEnv", () =>
1310
+ acquireBuildEnv(s, viteCommand, viteMode),
1311
+ );
1312
+
1313
+ let tempServer: any = null;
1314
+ // Signal to user-space code (e.g. reverse.ts) that build-time discovery
1315
+ // is active. Uses globalThis because the temp server's module runner
1316
+ // creates a separate module context — there is no shared import path
1317
+ // between the vite plugin and user code loaded via runner.import().
1318
+ (globalThis as any).__rscRouterDiscoveryActive = true;
1319
+ try {
1320
+ tempServer = await timed(
1321
+ debugDiscovery,
1322
+ "build createTempRscServer",
1323
+ () => createTempRscServer(s, { forceBuild: true }),
1324
+ );
1325
+
1326
+ const rscEnv = (tempServer.environments as any)?.rsc;
1327
+ if (!rscEnv?.runner) {
1328
+ console.warn(
1329
+ "[rango] RSC environment runner not available during build, skipping manifest generation",
1330
+ );
1331
+ return;
1332
+ }
1333
+
1334
+ // Point resolvedStaticModules at the temp server's expose-internal-ids
1335
+ // plugin so that discoverRouters() can access the static handler module
1336
+ // map after the temp server's transforms populate it.
1337
+ const tempIdsPlugin = (tempServer as any).config?.plugins?.find(
1338
+ (p: any) => p.name === "@rangojs/router:expose-internal-ids",
1339
+ );
1340
+ if (tempIdsPlugin?.api?.staticHandlerModules) {
1341
+ s.resolvedStaticModules = tempIdsPlugin.api.staticHandlerModules;
1342
+ }
1343
+
1344
+ await timed(debugDiscovery, "build discoverRouters", () =>
1345
+ discoverRouters(s, rscEnv),
1346
+ );
1347
+ // Update named-routes.gen.ts from runtime discovery.
1348
+ // The runtime manifest includes dynamically generated routes
1349
+ // that the static parser cannot extract from source code.
1350
+ timedSync(debugDiscovery, "build writeRouteTypesFiles", () =>
1351
+ writeRouteTypesFiles(s),
1352
+ );
1353
+ } catch (err: any) {
1354
+ // Extract the user source file from the stack trace (skip internal frames)
1355
+ const sourceFile = err.stack
1356
+ ?.split("\n")
1357
+ .find(
1358
+ (line: string) =>
1359
+ line.includes(s.projectRoot) && !line.includes("node_modules"),
1360
+ )
1361
+ ?.match(/\(([^)]+)\)/)?.[1];
1362
+ // Extract the route name from "Unknown route: <name>" errors
1363
+ const routeName = err.message?.match(/Unknown route: (.+)/)?.[1];
1364
+ const details = [
1365
+ routeName ? ` Route name: ${routeName}` : null,
1366
+ sourceFile ? ` File: ${sourceFile}` : null,
1367
+ err.stack ? ` Stack:\n${err.stack}` : null,
1368
+ ]
1369
+ .filter(Boolean)
1370
+ .join("\n");
1371
+ throw new Error(
1372
+ `[rango] Build-time router discovery failed:\n${details}`,
1373
+ { cause: err },
1374
+ );
1375
+ } finally {
1376
+ delete (globalThis as any).__rscRouterDiscoveryActive;
1377
+ if (tempServer) {
1378
+ await timed(debugDiscovery, "build tempServer.close", () =>
1379
+ tempServer.close(),
1380
+ );
1381
+ }
1382
+ await releaseBuildEnv(s);
1383
+ debugDiscovery?.(
1384
+ "build discovery done (%sms)",
1385
+ (performance.now() - buildStartTime).toFixed(1),
1386
+ );
1387
+ }
1388
+ },
1389
+
1390
+ // Suppress vite's HMR cascade for our own gen-file writes.
1391
+ //
1392
+ // After every cf HMR cycle, refreshTempRscEnv → writeRouteTypesFiles
1393
+ // writes the configured gen files (default `router.named-routes.gen.ts`,
1394
+ // but the source filenames and gen suffix are user-configurable). The
1395
+ // chokidar watcher then fires twice independently: our
1396
+ // `handleRouteFileChange` (already short-circuited by
1397
+ // `consumeSelfGenWrite` inside `maybeHandleGeneratedRouteFileMutation`),
1398
+ // AND vite's own HMR pipeline (which invalidates the gen file's
1399
+ // importers and triggers a second workerd full reload — visible to the
1400
+ // user as a duplicate "[Rango] HMR: version changed" on the client).
1401
+ //
1402
+ // `peekSelfGenWrite` is the authoritative filter: its map only contains
1403
+ // paths that `markSelfGenWrite` has registered, so it natively works
1404
+ // for any configured gen-file name. It is non-consuming so the chokidar
1405
+ // handler that fires later can still consume the same entry. Returning
1406
+ // [] tells vite "no modules invalidated by this change" — safe because
1407
+ // `s.perRouterManifests` is already up-to-date (the write that just
1408
+ // happened is the consequence of our just-completed rediscovery).
1409
+ handleHotUpdate(ctx) {
1410
+ if (peekSelfGenWrite(s, ctx.file)) {
1411
+ debugDiscovery?.(
1412
+ "handleHotUpdate: suppressing self-write HMR cascade for %s",
1413
+ ctx.file,
1414
+ );
1415
+ return [];
1416
+ }
1417
+ },
1418
+
1419
+ // Virtual module: provides the pre-generated route manifest as a JS module
1420
+ // that calls setCachedManifest() at import time.
1421
+ resolveId(id) {
1422
+ if (id === VIRTUAL_ROUTES_MANIFEST_ID) {
1423
+ return "\0" + VIRTUAL_ROUTES_MANIFEST_ID;
1424
+ }
1425
+ // Per-router virtual modules: virtual:rsc-router/routes-manifest/<routerId>
1426
+ if (id.startsWith(VIRTUAL_ROUTES_MANIFEST_ID + "/")) {
1427
+ return "\0" + id;
1428
+ }
1429
+ // virtual:rsc-router/prerender-paths removed: prerender data is served through the worker
1430
+ return null;
1431
+ },
1432
+
1433
+ async load(id) {
1434
+ if (id === "\0" + VIRTUAL_ROUTES_MANIFEST_ID) {
1435
+ // In dev mode, wait for discovery to complete before emitting module content.
1436
+ // This is critical for Cloudflare dev where the worker runs in a separate
1437
+ // Miniflare process and can only receive manifest data via the virtual module.
1438
+ if (s.discoveryDone) {
1439
+ await timed(
1440
+ debugRoutes,
1441
+ "await discoveryDone (manifest)",
1442
+ () => s.discoveryDone,
1443
+ );
1444
+ }
1445
+ const code = await timed(
1446
+ debugRoutes,
1447
+ "generateRoutesManifestModule",
1448
+ () => generateRoutesManifestModule(s),
1449
+ );
1450
+ debugRoutes?.("manifest module emitted (%d bytes)", code?.length ?? 0);
1451
+ return code;
1452
+ }
1453
+ // Per-router virtual modules: pure data exports (no side effects).
1454
+ // ensureRouterManifest() imports the module and stores the data.
1455
+ const perRouterPrefix = "\0" + VIRTUAL_ROUTES_MANIFEST_ID + "/";
1456
+ if (id.startsWith(perRouterPrefix)) {
1457
+ if (s.discoveryDone) {
1458
+ await timed(
1459
+ debugRoutes,
1460
+ "await discoveryDone (per-router)",
1461
+ () => s.discoveryDone,
1462
+ );
1463
+ }
1464
+ const routerId = id.slice(perRouterPrefix.length);
1465
+ const code = await timed(
1466
+ debugRoutes,
1467
+ `generatePerRouterModule ${routerId}`,
1468
+ () => generatePerRouterModule(s, routerId),
1469
+ );
1470
+ return code;
1471
+ }
1472
+ // virtual:rsc-router/prerender-paths load handler removed
1473
+ return null;
1474
+ },
1475
+
1476
+ // Record handler chunk metadata and RSC entry filename during RSC build.
1477
+ // Used by closeBundle for handler code eviction and prerender data injection.
1478
+ generateBundle(_options: any, bundle: any) {
1479
+ if (this.environment?.name !== "rsc") return;
1480
+ const genStart = debugBuild ? performance.now() : 0;
1481
+
1482
+ // Record RSC entry chunk filename for closeBundle injection
1483
+ for (const [fileName, chunk] of Object.entries(bundle) as [
1484
+ string,
1485
+ any,
1486
+ ][]) {
1487
+ if (chunk.type === "chunk" && chunk.isEntry) {
1488
+ s.rscEntryFileName = fileName;
1489
+ break;
1490
+ }
1491
+ }
1492
+
1493
+ if (!s.resolvedPrerenderModules?.size && !s.resolvedStaticModules?.size) {
1494
+ debugBuild?.(
1495
+ "generateBundle (rsc): no handlers to scan (%sms)",
1496
+ (performance.now() - genStart).toFixed(1),
1497
+ );
1498
+ return;
1499
+ }
1500
+
1501
+ // Clear maps at the start of each RSC generateBundle pass.
1502
+ // Vite 6 multi-environment builds run RSC twice (analysis + production);
1503
+ // clearing prevents stale/duplicate records from the analysis pass.
1504
+ s.handlerChunkInfoMap.clear();
1505
+ s.staticHandlerChunkInfoMap.clear();
1506
+
1507
+ for (const [fileName, chunk] of Object.entries(bundle) as [
1508
+ string,
1509
+ any,
1510
+ ][]) {
1511
+ if (chunk.type !== "chunk") continue;
1512
+
1513
+ // Scan all chunks for handler exports (handlers may land in any chunk)
1514
+ if (s.resolvedPrerenderModules?.size) {
1515
+ const handlers = extractHandlerExportsFromChunk(
1516
+ chunk.code,
1517
+ s.resolvedPrerenderModules,
1518
+ "Prerender",
1519
+ false,
1520
+ );
1521
+ if (handlers.length > 0) {
1522
+ const existing = s.handlerChunkInfoMap.get(fileName);
1523
+ if (existing) {
1524
+ existing.exports.push(...handlers);
1525
+ } else {
1526
+ s.handlerChunkInfoMap.set(fileName, {
1527
+ fileName,
1528
+ exports: handlers,
1529
+ });
1530
+ }
1531
+ }
1532
+ }
1533
+
1534
+ if (s.resolvedStaticModules?.size) {
1535
+ const handlers = extractHandlerExportsFromChunk(
1536
+ chunk.code,
1537
+ s.resolvedStaticModules,
1538
+ "Static",
1539
+ false,
1540
+ );
1541
+ if (handlers.length > 0) {
1542
+ const existing = s.staticHandlerChunkInfoMap.get(fileName);
1543
+ if (existing) {
1544
+ existing.exports.push(...handlers);
1545
+ } else {
1546
+ s.staticHandlerChunkInfoMap.set(fileName, {
1547
+ fileName,
1548
+ exports: handlers,
1549
+ });
1550
+ }
1551
+ }
1552
+ }
1553
+ }
1554
+
1555
+ debugBuild?.(
1556
+ "generateBundle (rsc): scanned %d chunks, %d prerender chunk(s), %d static chunk(s) (%sms)",
1557
+ Object.keys(bundle).length,
1558
+ s.handlerChunkInfoMap.size,
1559
+ s.staticHandlerChunkInfoMap.size,
1560
+ (performance.now() - genStart).toFixed(1),
1561
+ );
1562
+ },
1563
+
1564
+ // Build-time pre-rendering: evict handler code and inject collected prerender data.
1565
+ // Collection now happens in-process during discoverRouters() via RSC runner.
1566
+ // closeBundle only needs to evict handlers and inject the in-memory data.
1567
+ closeBundle: {
1568
+ order: "post" as const,
1569
+ sequential: true,
1570
+ async handler(this: any) {
1571
+ if (!s.isBuildMode) return;
1572
+ // Only run for the RSC environment — other environments (client, ssr) have
1573
+ // no prerender/static data to process and would just do redundant file I/O.
1574
+ if (this.environment && this.environment.name !== "rsc") return;
1575
+ timedSync(debugBuild, "closeBundle postprocessBundle", () =>
1576
+ postprocessBundle(s),
1577
+ );
1578
+ },
1579
+ },
1580
+ };
1581
+ }