@rangojs/router 0.0.0-experimental.debug-cache-2383ca26 → 0.0.0-experimental.df410dd5

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