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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (1214) hide show
  1. package/AGENTS.md +9 -0
  2. package/README.md +1040 -5
  3. package/dist/__internal.d.ts +83 -0
  4. package/dist/__internal.d.ts.map +1 -0
  5. package/dist/__internal.js +19 -0
  6. package/dist/__internal.js.map +1 -0
  7. package/dist/__mocks__/version.d.ts +7 -0
  8. package/dist/__mocks__/version.d.ts.map +1 -0
  9. package/dist/__mocks__/version.js +7 -0
  10. package/dist/__mocks__/version.js.map +1 -0
  11. package/dist/__tests__/client-href.test.d.ts +2 -0
  12. package/dist/__tests__/client-href.test.d.ts.map +1 -0
  13. package/dist/__tests__/client-href.test.js +74 -0
  14. package/dist/__tests__/client-href.test.js.map +1 -0
  15. package/dist/__tests__/component-utils.test.d.ts +2 -0
  16. package/dist/__tests__/component-utils.test.d.ts.map +1 -0
  17. package/dist/__tests__/component-utils.test.js +51 -0
  18. package/dist/__tests__/component-utils.test.js.map +1 -0
  19. package/dist/__tests__/event-controller.test.d.ts +2 -0
  20. package/dist/__tests__/event-controller.test.d.ts.map +1 -0
  21. package/dist/__tests__/event-controller.test.js +538 -0
  22. package/dist/__tests__/event-controller.test.js.map +1 -0
  23. package/dist/__tests__/helpers/route-tree.d.ts +118 -0
  24. package/dist/__tests__/helpers/route-tree.d.ts.map +1 -0
  25. package/dist/__tests__/helpers/route-tree.js +374 -0
  26. package/dist/__tests__/helpers/route-tree.js.map +1 -0
  27. package/dist/__tests__/match-result.test.d.ts +2 -0
  28. package/dist/__tests__/match-result.test.d.ts.map +1 -0
  29. package/dist/__tests__/match-result.test.js +154 -0
  30. package/dist/__tests__/match-result.test.js.map +1 -0
  31. package/dist/__tests__/navigation-store.test.d.ts +2 -0
  32. package/dist/__tests__/navigation-store.test.d.ts.map +1 -0
  33. package/dist/__tests__/navigation-store.test.js +440 -0
  34. package/dist/__tests__/navigation-store.test.js.map +1 -0
  35. package/dist/__tests__/partial-update.test.d.ts +2 -0
  36. package/dist/__tests__/partial-update.test.d.ts.map +1 -0
  37. package/dist/__tests__/partial-update.test.js +1009 -0
  38. package/dist/__tests__/partial-update.test.js.map +1 -0
  39. package/dist/__tests__/reverse-types.test.d.ts +8 -0
  40. package/dist/__tests__/reverse-types.test.d.ts.map +1 -0
  41. package/dist/__tests__/reverse-types.test.js +656 -0
  42. package/dist/__tests__/reverse-types.test.js.map +1 -0
  43. package/dist/__tests__/route-definition.test.d.ts +2 -0
  44. package/dist/__tests__/route-definition.test.d.ts.map +1 -0
  45. package/dist/__tests__/route-definition.test.js +55 -0
  46. package/dist/__tests__/route-definition.test.js.map +1 -0
  47. package/dist/__tests__/router-helpers.test.d.ts +2 -0
  48. package/dist/__tests__/router-helpers.test.d.ts.map +1 -0
  49. package/dist/__tests__/router-helpers.test.js +377 -0
  50. package/dist/__tests__/router-helpers.test.js.map +1 -0
  51. package/dist/__tests__/router-integration-2.test.d.ts +2 -0
  52. package/dist/__tests__/router-integration-2.test.d.ts.map +1 -0
  53. package/dist/__tests__/router-integration-2.test.js +426 -0
  54. package/dist/__tests__/router-integration-2.test.js.map +1 -0
  55. package/dist/__tests__/router-integration.test.d.ts +2 -0
  56. package/dist/__tests__/router-integration.test.d.ts.map +1 -0
  57. package/dist/__tests__/router-integration.test.js +1051 -0
  58. package/dist/__tests__/router-integration.test.js.map +1 -0
  59. package/dist/__tests__/search-params.test.d.ts +5 -0
  60. package/dist/__tests__/search-params.test.d.ts.map +1 -0
  61. package/dist/__tests__/search-params.test.js +306 -0
  62. package/dist/__tests__/search-params.test.js.map +1 -0
  63. package/dist/__tests__/segment-system.test.d.ts +2 -0
  64. package/dist/__tests__/segment-system.test.d.ts.map +1 -0
  65. package/dist/__tests__/segment-system.test.js +627 -0
  66. package/dist/__tests__/segment-system.test.js.map +1 -0
  67. package/dist/__tests__/static-handler-types.test.d.ts +8 -0
  68. package/dist/__tests__/static-handler-types.test.d.ts.map +1 -0
  69. package/dist/__tests__/static-handler-types.test.js +63 -0
  70. package/dist/__tests__/static-handler-types.test.js.map +1 -0
  71. package/dist/__tests__/urls.test.d.ts +2 -0
  72. package/dist/__tests__/urls.test.d.ts.map +1 -0
  73. package/dist/__tests__/urls.test.js +421 -0
  74. package/dist/__tests__/urls.test.js.map +1 -0
  75. package/dist/__tests__/use-mount.test.d.ts +2 -0
  76. package/dist/__tests__/use-mount.test.d.ts.map +1 -0
  77. package/dist/__tests__/use-mount.test.js +35 -0
  78. package/dist/__tests__/use-mount.test.js.map +1 -0
  79. package/dist/bin/rango.d.ts +2 -0
  80. package/dist/bin/rango.d.ts.map +1 -0
  81. package/dist/bin/rango.js +1712 -212
  82. package/dist/bin/rango.js.map +1 -0
  83. package/dist/browser/event-controller.d.ts +191 -0
  84. package/dist/browser/event-controller.d.ts.map +1 -0
  85. package/dist/browser/event-controller.js +559 -0
  86. package/dist/browser/event-controller.js.map +1 -0
  87. package/dist/browser/index.d.ts +2 -0
  88. package/dist/browser/index.d.ts.map +1 -0
  89. package/dist/browser/index.js +14 -0
  90. package/dist/browser/index.js.map +1 -0
  91. package/dist/browser/link-interceptor.d.ts +38 -0
  92. package/dist/browser/link-interceptor.d.ts.map +1 -0
  93. package/dist/browser/link-interceptor.js +99 -0
  94. package/dist/browser/link-interceptor.js.map +1 -0
  95. package/dist/browser/logging.d.ts +10 -0
  96. package/dist/browser/logging.d.ts.map +1 -0
  97. package/dist/browser/logging.js +29 -0
  98. package/dist/browser/logging.js.map +1 -0
  99. package/dist/browser/lru-cache.d.ts +17 -0
  100. package/dist/browser/lru-cache.d.ts.map +1 -0
  101. package/dist/browser/lru-cache.js +50 -0
  102. package/dist/browser/lru-cache.js.map +1 -0
  103. package/dist/browser/merge-segment-loaders.d.ts +39 -0
  104. package/dist/browser/merge-segment-loaders.d.ts.map +1 -0
  105. package/dist/browser/merge-segment-loaders.js +102 -0
  106. package/dist/browser/merge-segment-loaders.js.map +1 -0
  107. package/dist/browser/navigation-bridge.d.ts +102 -0
  108. package/dist/browser/navigation-bridge.d.ts.map +1 -0
  109. package/dist/browser/navigation-bridge.js +708 -0
  110. package/dist/browser/navigation-bridge.js.map +1 -0
  111. package/dist/browser/navigation-client.d.ts +25 -0
  112. package/dist/browser/navigation-client.d.ts.map +1 -0
  113. package/dist/browser/navigation-client.js +157 -0
  114. package/dist/browser/navigation-client.js.map +1 -0
  115. package/dist/browser/navigation-store.d.ts +101 -0
  116. package/dist/browser/navigation-store.d.ts.map +1 -0
  117. package/dist/browser/navigation-store.js +625 -0
  118. package/dist/browser/navigation-store.js.map +1 -0
  119. package/dist/browser/partial-update.d.ts +75 -0
  120. package/dist/browser/partial-update.d.ts.map +1 -0
  121. package/dist/browser/partial-update.js +426 -0
  122. package/dist/browser/partial-update.js.map +1 -0
  123. package/dist/browser/react/Link.d.ts +86 -0
  124. package/dist/browser/react/Link.d.ts.map +1 -0
  125. package/dist/browser/react/Link.js +128 -0
  126. package/dist/browser/react/Link.js.map +1 -0
  127. package/dist/browser/react/NavigationProvider.d.ts +63 -0
  128. package/dist/browser/react/NavigationProvider.d.ts.map +1 -0
  129. package/dist/browser/react/NavigationProvider.js +216 -0
  130. package/dist/browser/react/NavigationProvider.js.map +1 -0
  131. package/dist/browser/react/ScrollRestoration.d.ts +75 -0
  132. package/dist/browser/react/ScrollRestoration.d.ts.map +1 -0
  133. package/dist/browser/react/ScrollRestoration.js +57 -0
  134. package/dist/browser/react/ScrollRestoration.js.map +1 -0
  135. package/dist/browser/react/context.d.ts +46 -0
  136. package/dist/browser/react/context.d.ts.map +1 -0
  137. package/dist/browser/react/context.js +10 -0
  138. package/dist/browser/react/context.js.map +1 -0
  139. package/dist/browser/react/index.d.ts +11 -0
  140. package/dist/browser/react/index.d.ts.map +1 -0
  141. package/dist/browser/react/index.js +22 -0
  142. package/dist/browser/react/index.js.map +1 -0
  143. package/dist/browser/react/location-state-shared.d.ts +63 -0
  144. package/dist/browser/react/location-state-shared.d.ts.map +1 -0
  145. package/dist/browser/react/location-state-shared.js +81 -0
  146. package/dist/browser/react/location-state-shared.js.map +1 -0
  147. package/dist/browser/react/location-state.d.ts +23 -0
  148. package/dist/browser/react/location-state.d.ts.map +1 -0
  149. package/dist/browser/react/location-state.js +29 -0
  150. package/dist/browser/react/location-state.js.map +1 -0
  151. package/dist/browser/react/mount-context.d.ts +24 -0
  152. package/dist/browser/react/mount-context.d.ts.map +1 -0
  153. package/dist/browser/react/mount-context.js +24 -0
  154. package/dist/browser/react/mount-context.js.map +1 -0
  155. package/dist/browser/react/use-action.d.ts +64 -0
  156. package/dist/browser/react/use-action.d.ts.map +1 -0
  157. package/dist/browser/react/use-action.js +134 -0
  158. package/dist/browser/react/use-action.js.map +1 -0
  159. package/dist/browser/react/use-client-cache.d.ts +41 -0
  160. package/dist/browser/react/use-client-cache.d.ts.map +1 -0
  161. package/{src/browser/react/use-client-cache.ts → dist/browser/react/use-client-cache.js} +9 -26
  162. package/dist/browser/react/use-client-cache.js.map +1 -0
  163. package/dist/browser/react/use-handle.d.ts +31 -0
  164. package/dist/browser/react/use-handle.d.ts.map +1 -0
  165. package/dist/browser/react/use-handle.js +144 -0
  166. package/dist/browser/react/use-handle.js.map +1 -0
  167. package/dist/browser/react/use-href.d.ts +33 -0
  168. package/dist/browser/react/use-href.d.ts.map +1 -0
  169. package/dist/browser/react/use-href.js +39 -0
  170. package/dist/browser/react/use-href.js.map +1 -0
  171. package/dist/browser/react/use-link-status.d.ts +37 -0
  172. package/dist/browser/react/use-link-status.d.ts.map +1 -0
  173. package/dist/browser/react/use-link-status.js +99 -0
  174. package/dist/browser/react/use-link-status.js.map +1 -0
  175. package/dist/browser/react/use-mount.d.ts +25 -0
  176. package/dist/browser/react/use-mount.d.ts.map +1 -0
  177. package/dist/browser/react/use-mount.js +30 -0
  178. package/dist/browser/react/use-mount.js.map +1 -0
  179. package/dist/browser/react/use-navigation.d.ts +27 -0
  180. package/dist/browser/react/use-navigation.d.ts.map +1 -0
  181. package/dist/browser/react/use-navigation.js +87 -0
  182. package/dist/browser/react/use-navigation.js.map +1 -0
  183. package/dist/browser/react/use-segments.d.ts +38 -0
  184. package/dist/browser/react/use-segments.d.ts.map +1 -0
  185. package/dist/browser/react/use-segments.js +130 -0
  186. package/dist/browser/react/use-segments.js.map +1 -0
  187. package/dist/browser/request-controller.d.ts +26 -0
  188. package/dist/browser/request-controller.d.ts.map +1 -0
  189. package/dist/browser/request-controller.js +147 -0
  190. package/dist/browser/request-controller.js.map +1 -0
  191. package/dist/browser/rsc-router.d.ts +129 -0
  192. package/dist/browser/rsc-router.d.ts.map +1 -0
  193. package/dist/browser/rsc-router.js +195 -0
  194. package/dist/browser/rsc-router.js.map +1 -0
  195. package/dist/browser/scroll-restoration.d.ts +93 -0
  196. package/dist/browser/scroll-restoration.d.ts.map +1 -0
  197. package/dist/browser/scroll-restoration.js +321 -0
  198. package/dist/browser/scroll-restoration.js.map +1 -0
  199. package/dist/browser/segment-structure-assert.d.ts +17 -0
  200. package/dist/browser/segment-structure-assert.d.ts.map +1 -0
  201. package/dist/browser/segment-structure-assert.js +59 -0
  202. package/dist/browser/segment-structure-assert.js.map +1 -0
  203. package/dist/browser/server-action-bridge.d.ts +26 -0
  204. package/dist/browser/server-action-bridge.d.ts.map +1 -0
  205. package/dist/browser/server-action-bridge.js +668 -0
  206. package/dist/browser/server-action-bridge.js.map +1 -0
  207. package/dist/browser/shallow.d.ts +12 -0
  208. package/dist/browser/shallow.d.ts.map +1 -0
  209. package/dist/browser/shallow.js +34 -0
  210. package/dist/browser/shallow.js.map +1 -0
  211. package/dist/browser/types.d.ts +369 -0
  212. package/dist/browser/types.d.ts.map +1 -0
  213. package/dist/browser/types.js +2 -0
  214. package/dist/browser/types.js.map +1 -0
  215. package/dist/build/__tests__/generate-cli.test.d.ts +2 -0
  216. package/dist/build/__tests__/generate-cli.test.d.ts.map +1 -0
  217. package/dist/build/__tests__/generate-cli.test.js +237 -0
  218. package/dist/build/__tests__/generate-cli.test.js.map +1 -0
  219. package/dist/build/__tests__/generate-manifest.test.d.ts +2 -0
  220. package/dist/build/__tests__/generate-manifest.test.d.ts.map +1 -0
  221. package/dist/build/__tests__/generate-manifest.test.js +119 -0
  222. package/dist/build/__tests__/generate-manifest.test.js.map +1 -0
  223. package/dist/build/__tests__/generate-route-types.test.d.ts +2 -0
  224. package/dist/build/__tests__/generate-route-types.test.d.ts.map +1 -0
  225. package/dist/build/__tests__/generate-route-types.test.js +620 -0
  226. package/dist/build/__tests__/generate-route-types.test.js.map +1 -0
  227. package/dist/build/__tests__/per-router-manifest.test.d.ts +2 -0
  228. package/dist/build/__tests__/per-router-manifest.test.d.ts.map +1 -0
  229. package/dist/build/__tests__/per-router-manifest.test.js +308 -0
  230. package/dist/build/__tests__/per-router-manifest.test.js.map +1 -0
  231. package/dist/build/generate-manifest.d.ts +81 -0
  232. package/dist/build/generate-manifest.d.ts.map +1 -0
  233. package/dist/build/generate-manifest.js +276 -0
  234. package/dist/build/generate-manifest.js.map +1 -0
  235. package/dist/build/generate-route-types.d.ts +115 -0
  236. package/dist/build/generate-route-types.d.ts.map +1 -0
  237. package/dist/build/generate-route-types.js +740 -0
  238. package/dist/build/generate-route-types.js.map +1 -0
  239. package/dist/build/index.d.ts +21 -0
  240. package/dist/build/index.d.ts.map +1 -0
  241. package/dist/build/index.js +21 -0
  242. package/dist/build/index.js.map +1 -0
  243. package/dist/build/route-trie.d.ts +71 -0
  244. package/dist/build/route-trie.d.ts.map +1 -0
  245. package/dist/build/route-trie.js +175 -0
  246. package/dist/build/route-trie.js.map +1 -0
  247. package/dist/cache/__tests__/cache-scope.test.d.ts +2 -0
  248. package/dist/cache/__tests__/cache-scope.test.d.ts.map +1 -0
  249. package/dist/cache/__tests__/cache-scope.test.js +208 -0
  250. package/dist/cache/__tests__/cache-scope.test.js.map +1 -0
  251. package/dist/cache/__tests__/document-cache.test.d.ts +2 -0
  252. package/dist/cache/__tests__/document-cache.test.d.ts.map +1 -0
  253. package/dist/cache/__tests__/document-cache.test.js +345 -0
  254. package/dist/cache/__tests__/document-cache.test.js.map +1 -0
  255. package/dist/cache/__tests__/memory-segment-store.test.d.ts +2 -0
  256. package/dist/cache/__tests__/memory-segment-store.test.d.ts.map +1 -0
  257. package/dist/cache/__tests__/memory-segment-store.test.js +425 -0
  258. package/dist/cache/__tests__/memory-segment-store.test.js.map +1 -0
  259. package/dist/cache/__tests__/memory-store.test.d.ts +2 -0
  260. package/dist/cache/__tests__/memory-store.test.d.ts.map +1 -0
  261. package/dist/cache/__tests__/memory-store.test.js +367 -0
  262. package/dist/cache/__tests__/memory-store.test.js.map +1 -0
  263. package/dist/cache/cache-scope.d.ts +102 -0
  264. package/dist/cache/cache-scope.d.ts.map +1 -0
  265. package/dist/cache/cache-scope.js +440 -0
  266. package/dist/cache/cache-scope.js.map +1 -0
  267. package/dist/cache/cf/__tests__/cf-cache-store.test.d.ts +2 -0
  268. package/dist/cache/cf/__tests__/cf-cache-store.test.d.ts.map +1 -0
  269. package/dist/cache/cf/__tests__/cf-cache-store.test.js +330 -0
  270. package/dist/cache/cf/__tests__/cf-cache-store.test.js.map +1 -0
  271. package/dist/cache/cf/cf-cache-store.d.ts +165 -0
  272. package/dist/cache/cf/cf-cache-store.d.ts.map +1 -0
  273. package/dist/cache/cf/cf-cache-store.js +242 -0
  274. package/dist/cache/cf/cf-cache-store.js.map +1 -0
  275. package/dist/cache/cf/index.d.ts +14 -0
  276. package/dist/cache/cf/index.d.ts.map +1 -0
  277. package/dist/cache/cf/index.js +17 -0
  278. package/dist/cache/cf/index.js.map +1 -0
  279. package/dist/cache/document-cache.d.ts +64 -0
  280. package/dist/cache/document-cache.d.ts.map +1 -0
  281. package/dist/cache/document-cache.js +228 -0
  282. package/dist/cache/document-cache.js.map +1 -0
  283. package/dist/cache/index.d.ts +19 -0
  284. package/dist/cache/index.d.ts.map +1 -0
  285. package/dist/cache/index.js +21 -0
  286. package/dist/cache/index.js.map +1 -0
  287. package/dist/cache/memory-segment-store.d.ts +110 -0
  288. package/dist/cache/memory-segment-store.d.ts.map +1 -0
  289. package/dist/cache/memory-segment-store.js +117 -0
  290. package/dist/cache/memory-segment-store.js.map +1 -0
  291. package/dist/cache/memory-store.d.ts +41 -0
  292. package/dist/cache/memory-store.d.ts.map +1 -0
  293. package/dist/cache/memory-store.js +191 -0
  294. package/dist/cache/memory-store.js.map +1 -0
  295. package/dist/cache/types.d.ts +317 -0
  296. package/dist/cache/types.d.ts.map +1 -0
  297. package/dist/cache/types.js +12 -0
  298. package/dist/cache/types.js.map +1 -0
  299. package/dist/client.d.ts +248 -0
  300. package/dist/client.d.ts.map +1 -0
  301. package/dist/client.js +367 -0
  302. package/dist/client.js.map +1 -0
  303. package/dist/client.rsc.d.ts +26 -0
  304. package/dist/client.rsc.d.ts.map +1 -0
  305. package/dist/client.rsc.js +46 -0
  306. package/dist/client.rsc.js.map +1 -0
  307. package/dist/component-utils.d.ts +36 -0
  308. package/dist/component-utils.d.ts.map +1 -0
  309. package/dist/component-utils.js +61 -0
  310. package/dist/component-utils.js.map +1 -0
  311. package/dist/components/DefaultDocument.d.ts +13 -0
  312. package/dist/components/DefaultDocument.d.ts.map +1 -0
  313. package/dist/components/DefaultDocument.js +15 -0
  314. package/dist/components/DefaultDocument.js.map +1 -0
  315. package/dist/debug.d.ts +58 -0
  316. package/dist/debug.d.ts.map +1 -0
  317. package/dist/debug.js +157 -0
  318. package/dist/debug.js.map +1 -0
  319. package/dist/default-error-boundary.d.ts +11 -0
  320. package/dist/default-error-boundary.d.ts.map +1 -0
  321. package/dist/default-error-boundary.js +45 -0
  322. package/dist/default-error-boundary.js.map +1 -0
  323. package/dist/deps/browser.d.ts +2 -0
  324. package/dist/deps/browser.d.ts.map +1 -0
  325. package/dist/deps/browser.js +3 -0
  326. package/dist/deps/browser.js.map +1 -0
  327. package/dist/deps/html-stream-client.d.ts +2 -0
  328. package/dist/deps/html-stream-client.d.ts.map +1 -0
  329. package/dist/deps/html-stream-client.js +3 -0
  330. package/dist/deps/html-stream-client.js.map +1 -0
  331. package/dist/deps/html-stream-server.d.ts +2 -0
  332. package/dist/deps/html-stream-server.d.ts.map +1 -0
  333. package/dist/deps/html-stream-server.js +3 -0
  334. package/dist/deps/html-stream-server.js.map +1 -0
  335. package/dist/deps/rsc.d.ts +2 -0
  336. package/dist/deps/rsc.d.ts.map +1 -0
  337. package/dist/deps/rsc.js +4 -0
  338. package/dist/deps/rsc.js.map +1 -0
  339. package/dist/deps/ssr.d.ts +2 -0
  340. package/dist/deps/ssr.d.ts.map +1 -0
  341. package/dist/deps/ssr.js +3 -0
  342. package/dist/deps/ssr.js.map +1 -0
  343. package/dist/errors.d.ts +174 -0
  344. package/dist/errors.d.ts.map +1 -0
  345. package/dist/errors.js +241 -0
  346. package/dist/errors.js.map +1 -0
  347. package/dist/handle.d.ts +78 -0
  348. package/dist/handle.d.ts.map +1 -0
  349. package/dist/handle.js +82 -0
  350. package/dist/handle.js.map +1 -0
  351. package/dist/handles/MetaTags.d.ts +14 -0
  352. package/dist/handles/MetaTags.d.ts.map +1 -0
  353. package/dist/handles/MetaTags.js +136 -0
  354. package/dist/handles/MetaTags.js.map +1 -0
  355. package/dist/handles/index.d.ts +6 -0
  356. package/dist/handles/index.d.ts.map +1 -0
  357. package/{src/handles/index.ts → dist/handles/index.js} +1 -1
  358. package/dist/handles/index.js.map +1 -0
  359. package/dist/handles/meta.d.ts +39 -0
  360. package/dist/handles/meta.d.ts.map +1 -0
  361. package/dist/handles/meta.js +202 -0
  362. package/dist/handles/meta.js.map +1 -0
  363. package/dist/host/__tests__/errors.test.d.ts +2 -0
  364. package/dist/host/__tests__/errors.test.d.ts.map +1 -0
  365. package/dist/host/__tests__/errors.test.js +76 -0
  366. package/dist/host/__tests__/errors.test.js.map +1 -0
  367. package/dist/host/__tests__/pattern-comprehensive.test.d.ts +2 -0
  368. package/dist/host/__tests__/pattern-comprehensive.test.d.ts.map +1 -0
  369. package/dist/host/__tests__/pattern-comprehensive.test.js +732 -0
  370. package/dist/host/__tests__/pattern-comprehensive.test.js.map +1 -0
  371. package/dist/host/__tests__/pattern-matcher.test.d.ts +2 -0
  372. package/dist/host/__tests__/pattern-matcher.test.d.ts.map +1 -0
  373. package/dist/host/__tests__/pattern-matcher.test.js +251 -0
  374. package/dist/host/__tests__/pattern-matcher.test.js.map +1 -0
  375. package/dist/host/__tests__/router.test.d.ts +2 -0
  376. package/dist/host/__tests__/router.test.d.ts.map +1 -0
  377. package/dist/host/__tests__/router.test.js +241 -0
  378. package/dist/host/__tests__/router.test.js.map +1 -0
  379. package/dist/host/__tests__/testing.test.d.ts +2 -0
  380. package/dist/host/__tests__/testing.test.d.ts.map +1 -0
  381. package/dist/host/__tests__/testing.test.js +64 -0
  382. package/dist/host/__tests__/testing.test.js.map +1 -0
  383. package/dist/host/__tests__/utils.test.d.ts +2 -0
  384. package/dist/host/__tests__/utils.test.d.ts.map +1 -0
  385. package/dist/host/__tests__/utils.test.js +29 -0
  386. package/dist/host/__tests__/utils.test.js.map +1 -0
  387. package/dist/host/cookie-handler.d.ts +34 -0
  388. package/dist/host/cookie-handler.d.ts.map +1 -0
  389. package/dist/host/cookie-handler.js +124 -0
  390. package/dist/host/cookie-handler.js.map +1 -0
  391. package/dist/host/errors.d.ts +56 -0
  392. package/dist/host/errors.d.ts.map +1 -0
  393. package/dist/host/errors.js +79 -0
  394. package/dist/host/errors.js.map +1 -0
  395. package/dist/host/index.d.ts +29 -0
  396. package/dist/host/index.d.ts.map +1 -0
  397. package/dist/host/index.js +32 -0
  398. package/dist/host/index.js.map +1 -0
  399. package/dist/host/pattern-matcher.d.ts +36 -0
  400. package/dist/host/pattern-matcher.d.ts.map +1 -0
  401. package/dist/host/pattern-matcher.js +172 -0
  402. package/dist/host/pattern-matcher.js.map +1 -0
  403. package/dist/host/router.d.ts +26 -0
  404. package/dist/host/router.d.ts.map +1 -0
  405. package/dist/host/router.js +218 -0
  406. package/dist/host/router.js.map +1 -0
  407. package/dist/host/testing.d.ts +36 -0
  408. package/dist/host/testing.d.ts.map +1 -0
  409. package/dist/host/testing.js +55 -0
  410. package/dist/host/testing.js.map +1 -0
  411. package/dist/host/types.d.ts +115 -0
  412. package/dist/host/types.d.ts.map +1 -0
  413. package/dist/host/types.js +7 -0
  414. package/dist/host/types.js.map +1 -0
  415. package/dist/host/utils.d.ts +21 -0
  416. package/dist/host/utils.d.ts.map +1 -0
  417. package/dist/host/utils.js +23 -0
  418. package/dist/host/utils.js.map +1 -0
  419. package/dist/href-client.d.ts +131 -0
  420. package/dist/href-client.d.ts.map +1 -0
  421. package/dist/href-client.js +64 -0
  422. package/dist/href-client.js.map +1 -0
  423. package/{src/href-context.ts → dist/href-context.d.ts} +7 -11
  424. package/dist/href-context.d.ts.map +1 -0
  425. package/dist/href-context.js +21 -0
  426. package/dist/href-context.js.map +1 -0
  427. package/dist/index.d.ts +73 -0
  428. package/dist/index.d.ts.map +1 -0
  429. package/dist/index.js +91 -0
  430. package/dist/index.js.map +1 -0
  431. package/dist/index.rsc.d.ts +32 -0
  432. package/dist/index.rsc.d.ts.map +1 -0
  433. package/dist/index.rsc.js +40 -0
  434. package/dist/index.rsc.js.map +1 -0
  435. package/dist/internal-debug.d.ts +2 -0
  436. package/dist/internal-debug.d.ts.map +1 -0
  437. package/dist/internal-debug.js +5 -0
  438. package/dist/internal-debug.js.map +1 -0
  439. package/dist/loader.d.ts +14 -0
  440. package/dist/loader.d.ts.map +1 -0
  441. package/dist/loader.js +20 -0
  442. package/dist/loader.js.map +1 -0
  443. package/dist/loader.rsc.d.ts +19 -0
  444. package/dist/loader.rsc.d.ts.map +1 -0
  445. package/dist/loader.rsc.js +99 -0
  446. package/dist/loader.rsc.js.map +1 -0
  447. package/dist/network-error-thrower.d.ts +17 -0
  448. package/dist/network-error-thrower.d.ts.map +1 -0
  449. package/dist/network-error-thrower.js +14 -0
  450. package/dist/network-error-thrower.js.map +1 -0
  451. package/dist/outlet-context.d.ts +13 -0
  452. package/dist/outlet-context.d.ts.map +1 -0
  453. package/dist/outlet-context.js +3 -0
  454. package/dist/outlet-context.js.map +1 -0
  455. package/dist/prerender/__tests__/param-hash.test.d.ts +2 -0
  456. package/dist/prerender/__tests__/param-hash.test.d.ts.map +1 -0
  457. package/dist/prerender/__tests__/param-hash.test.js +148 -0
  458. package/dist/prerender/__tests__/param-hash.test.js.map +1 -0
  459. package/dist/prerender/param-hash.d.ts +16 -0
  460. package/dist/prerender/param-hash.d.ts.map +1 -0
  461. package/dist/prerender/param-hash.js +36 -0
  462. package/dist/prerender/param-hash.js.map +1 -0
  463. package/dist/prerender/store.d.ts +38 -0
  464. package/dist/prerender/store.d.ts.map +1 -0
  465. package/dist/prerender/store.js +61 -0
  466. package/dist/prerender/store.js.map +1 -0
  467. package/dist/prerender.d.ts +66 -0
  468. package/dist/prerender.d.ts.map +1 -0
  469. package/dist/prerender.js +57 -0
  470. package/dist/prerender.js.map +1 -0
  471. package/dist/reverse.d.ts +196 -0
  472. package/dist/reverse.d.ts.map +1 -0
  473. package/dist/reverse.js +78 -0
  474. package/dist/reverse.js.map +1 -0
  475. package/dist/root-error-boundary.d.ts +33 -0
  476. package/dist/root-error-boundary.d.ts.map +1 -0
  477. package/dist/root-error-boundary.js +165 -0
  478. package/dist/root-error-boundary.js.map +1 -0
  479. package/dist/route-content-wrapper.d.ts +46 -0
  480. package/dist/route-content-wrapper.d.ts.map +1 -0
  481. package/dist/route-content-wrapper.js +77 -0
  482. package/dist/route-content-wrapper.js.map +1 -0
  483. package/dist/route-definition.d.ts +421 -0
  484. package/dist/route-definition.d.ts.map +1 -0
  485. package/dist/route-definition.js +868 -0
  486. package/dist/route-definition.js.map +1 -0
  487. package/dist/route-map-builder.d.ts +155 -0
  488. package/dist/route-map-builder.d.ts.map +1 -0
  489. package/dist/route-map-builder.js +237 -0
  490. package/dist/route-map-builder.js.map +1 -0
  491. package/dist/route-types.d.ts +165 -0
  492. package/dist/route-types.d.ts.map +1 -0
  493. package/dist/route-types.js +7 -0
  494. package/dist/route-types.js.map +1 -0
  495. package/dist/router/__tests__/handler-context.test.d.ts +2 -0
  496. package/dist/router/__tests__/handler-context.test.d.ts.map +1 -0
  497. package/dist/router/__tests__/handler-context.test.js +65 -0
  498. package/dist/router/__tests__/handler-context.test.js.map +1 -0
  499. package/dist/router/__tests__/loader-cycle-detection.test.d.ts +2 -0
  500. package/dist/router/__tests__/loader-cycle-detection.test.d.ts.map +1 -0
  501. package/dist/router/__tests__/loader-cycle-detection.test.js +221 -0
  502. package/dist/router/__tests__/loader-cycle-detection.test.js.map +1 -0
  503. package/dist/router/__tests__/match-context.test.d.ts +2 -0
  504. package/dist/router/__tests__/match-context.test.d.ts.map +1 -0
  505. package/dist/router/__tests__/match-context.test.js +92 -0
  506. package/dist/router/__tests__/match-context.test.js.map +1 -0
  507. package/dist/router/__tests__/match-pipelines.test.d.ts +2 -0
  508. package/dist/router/__tests__/match-pipelines.test.d.ts.map +1 -0
  509. package/dist/router/__tests__/match-pipelines.test.js +417 -0
  510. package/dist/router/__tests__/match-pipelines.test.js.map +1 -0
  511. package/dist/router/__tests__/match-result.test.d.ts +2 -0
  512. package/dist/router/__tests__/match-result.test.d.ts.map +1 -0
  513. package/dist/router/__tests__/match-result.test.js +457 -0
  514. package/dist/router/__tests__/match-result.test.js.map +1 -0
  515. package/dist/router/__tests__/on-error.test.d.ts +2 -0
  516. package/dist/router/__tests__/on-error.test.d.ts.map +1 -0
  517. package/dist/router/__tests__/on-error.test.js +678 -0
  518. package/dist/router/__tests__/on-error.test.js.map +1 -0
  519. package/dist/router/__tests__/pattern-matching.test.d.ts +2 -0
  520. package/dist/router/__tests__/pattern-matching.test.d.ts.map +1 -0
  521. package/dist/router/__tests__/pattern-matching.test.js +629 -0
  522. package/dist/router/__tests__/pattern-matching.test.js.map +1 -0
  523. package/dist/router/__tests__/segment-resolution-parallel-loading.test.d.ts +2 -0
  524. package/dist/router/__tests__/segment-resolution-parallel-loading.test.d.ts.map +1 -0
  525. package/dist/router/__tests__/segment-resolution-parallel-loading.test.js +155 -0
  526. package/dist/router/__tests__/segment-resolution-parallel-loading.test.js.map +1 -0
  527. package/dist/router/error-handling.d.ts +77 -0
  528. package/dist/router/error-handling.d.ts.map +1 -0
  529. package/dist/router/error-handling.js +202 -0
  530. package/dist/router/error-handling.js.map +1 -0
  531. package/dist/router/handler-context.d.ts +20 -0
  532. package/dist/router/handler-context.d.ts.map +1 -0
  533. package/dist/router/handler-context.js +198 -0
  534. package/dist/router/handler-context.js.map +1 -0
  535. package/dist/router/intercept-resolution.d.ts +66 -0
  536. package/dist/router/intercept-resolution.d.ts.map +1 -0
  537. package/dist/router/intercept-resolution.js +246 -0
  538. package/dist/router/intercept-resolution.js.map +1 -0
  539. package/dist/router/loader-resolution.d.ts +64 -0
  540. package/dist/router/loader-resolution.d.ts.map +1 -0
  541. package/dist/router/loader-resolution.js +284 -0
  542. package/dist/router/loader-resolution.js.map +1 -0
  543. package/dist/router/logging.d.ts +15 -0
  544. package/dist/router/logging.d.ts.map +1 -0
  545. package/dist/router/logging.js +99 -0
  546. package/dist/router/logging.js.map +1 -0
  547. package/dist/router/manifest.d.ts +22 -0
  548. package/dist/router/manifest.d.ts.map +1 -0
  549. package/dist/router/manifest.js +181 -0
  550. package/dist/router/manifest.js.map +1 -0
  551. package/dist/router/match-api.d.ts +35 -0
  552. package/dist/router/match-api.d.ts.map +1 -0
  553. package/dist/router/match-api.js +406 -0
  554. package/dist/router/match-api.js.map +1 -0
  555. package/dist/router/match-context.d.ts +206 -0
  556. package/dist/router/match-context.d.ts.map +1 -0
  557. package/dist/router/match-context.js +17 -0
  558. package/dist/router/match-context.js.map +1 -0
  559. package/dist/router/match-middleware/background-revalidation.d.ts +127 -0
  560. package/dist/router/match-middleware/background-revalidation.d.ts.map +1 -0
  561. package/dist/router/match-middleware/background-revalidation.js +75 -0
  562. package/dist/router/match-middleware/background-revalidation.js.map +1 -0
  563. package/dist/router/match-middleware/cache-lookup.d.ts +112 -0
  564. package/dist/router/match-middleware/cache-lookup.d.ts.map +1 -0
  565. package/dist/router/match-middleware/cache-lookup.js +257 -0
  566. package/dist/router/match-middleware/cache-lookup.js.map +1 -0
  567. package/dist/router/match-middleware/cache-store.d.ts +113 -0
  568. package/dist/router/match-middleware/cache-store.d.ts.map +1 -0
  569. package/dist/router/match-middleware/cache-store.js +108 -0
  570. package/dist/router/match-middleware/cache-store.js.map +1 -0
  571. package/dist/router/match-middleware/index.d.ts +81 -0
  572. package/dist/router/match-middleware/index.d.ts.map +1 -0
  573. package/dist/router/match-middleware/index.js +80 -0
  574. package/dist/router/match-middleware/index.js.map +1 -0
  575. package/dist/router/match-middleware/intercept-resolution.d.ts +117 -0
  576. package/dist/router/match-middleware/intercept-resolution.d.ts.map +1 -0
  577. package/dist/router/match-middleware/intercept-resolution.js +134 -0
  578. package/dist/router/match-middleware/intercept-resolution.js.map +1 -0
  579. package/dist/router/match-middleware/segment-resolution.d.ts +99 -0
  580. package/dist/router/match-middleware/segment-resolution.d.ts.map +1 -0
  581. package/dist/router/match-middleware/segment-resolution.js +53 -0
  582. package/dist/router/match-middleware/segment-resolution.js.map +1 -0
  583. package/dist/router/match-pipelines.d.ts +147 -0
  584. package/dist/router/match-pipelines.d.ts.map +1 -0
  585. package/dist/router/match-pipelines.js +82 -0
  586. package/dist/router/match-pipelines.js.map +1 -0
  587. package/dist/router/match-result.d.ts +126 -0
  588. package/dist/router/match-result.d.ts.map +1 -0
  589. package/dist/router/match-result.js +93 -0
  590. package/dist/router/match-result.js.map +1 -0
  591. package/dist/router/metrics.d.ts +20 -0
  592. package/dist/router/metrics.d.ts.map +1 -0
  593. package/dist/router/metrics.js +47 -0
  594. package/dist/router/metrics.js.map +1 -0
  595. package/dist/router/middleware.d.ts +249 -0
  596. package/dist/router/middleware.d.ts.map +1 -0
  597. package/dist/router/middleware.js +434 -0
  598. package/dist/router/middleware.js.map +1 -0
  599. package/dist/router/middleware.test.d.ts +2 -0
  600. package/dist/router/middleware.test.d.ts.map +1 -0
  601. package/dist/router/middleware.test.js +816 -0
  602. package/dist/router/middleware.test.js.map +1 -0
  603. package/dist/router/pattern-matching.d.ts +149 -0
  604. package/dist/router/pattern-matching.d.ts.map +1 -0
  605. package/dist/router/pattern-matching.js +349 -0
  606. package/dist/router/pattern-matching.js.map +1 -0
  607. package/dist/router/revalidation.d.ts +44 -0
  608. package/dist/router/revalidation.d.ts.map +1 -0
  609. package/dist/router/revalidation.js +147 -0
  610. package/dist/router/revalidation.js.map +1 -0
  611. package/dist/router/router-context.d.ts +135 -0
  612. package/dist/router/router-context.d.ts.map +1 -0
  613. package/dist/router/router-context.js +36 -0
  614. package/dist/router/router-context.js.map +1 -0
  615. package/dist/router/segment-resolution.d.ts +127 -0
  616. package/dist/router/segment-resolution.d.ts.map +1 -0
  617. package/dist/router/segment-resolution.js +919 -0
  618. package/dist/router/segment-resolution.js.map +1 -0
  619. package/dist/router/trie-matching.d.ts +40 -0
  620. package/dist/router/trie-matching.d.ts.map +1 -0
  621. package/dist/router/trie-matching.js +127 -0
  622. package/dist/router/trie-matching.js.map +1 -0
  623. package/dist/router/types.d.ts +136 -0
  624. package/dist/router/types.d.ts.map +1 -0
  625. package/dist/router/types.js +7 -0
  626. package/dist/router/types.js.map +1 -0
  627. package/dist/router.d.ts +753 -0
  628. package/dist/router.d.ts.map +1 -0
  629. package/dist/router.gen.d.ts +6 -0
  630. package/dist/router.gen.d.ts.map +1 -0
  631. package/dist/router.gen.js +6 -0
  632. package/dist/router.gen.js.map +1 -0
  633. package/dist/router.js +1304 -0
  634. package/dist/router.js.map +1 -0
  635. package/dist/rsc/__tests__/helpers.test.d.ts +2 -0
  636. package/dist/rsc/__tests__/helpers.test.d.ts.map +1 -0
  637. package/dist/rsc/__tests__/helpers.test.js +140 -0
  638. package/dist/rsc/__tests__/helpers.test.js.map +1 -0
  639. package/dist/rsc/handler.d.ts +45 -0
  640. package/dist/rsc/handler.d.ts.map +1 -0
  641. package/dist/rsc/handler.js +1172 -0
  642. package/dist/rsc/handler.js.map +1 -0
  643. package/dist/rsc/helpers.d.ts +16 -0
  644. package/dist/rsc/helpers.d.ts.map +1 -0
  645. package/dist/rsc/helpers.js +55 -0
  646. package/dist/rsc/helpers.js.map +1 -0
  647. package/dist/rsc/index.d.ts +22 -0
  648. package/dist/rsc/index.d.ts.map +1 -0
  649. package/dist/rsc/index.js +23 -0
  650. package/dist/rsc/index.js.map +1 -0
  651. package/dist/rsc/nonce.d.ts +9 -0
  652. package/dist/rsc/nonce.d.ts.map +1 -0
  653. package/dist/rsc/nonce.js +18 -0
  654. package/dist/rsc/nonce.js.map +1 -0
  655. package/dist/rsc/types.d.ts +206 -0
  656. package/dist/rsc/types.d.ts.map +1 -0
  657. package/dist/rsc/types.js +8 -0
  658. package/dist/rsc/types.js.map +1 -0
  659. package/dist/search-params.d.ts +103 -0
  660. package/dist/search-params.d.ts.map +1 -0
  661. package/dist/search-params.js +74 -0
  662. package/dist/search-params.js.map +1 -0
  663. package/dist/segment-system.d.ts +75 -0
  664. package/dist/segment-system.d.ts.map +1 -0
  665. package/dist/segment-system.js +336 -0
  666. package/dist/segment-system.js.map +1 -0
  667. package/dist/server/context.d.ts +245 -0
  668. package/dist/server/context.d.ts.map +1 -0
  669. package/dist/server/context.js +197 -0
  670. package/dist/server/context.js.map +1 -0
  671. package/dist/server/fetchable-loader-store.d.ts +18 -0
  672. package/dist/server/fetchable-loader-store.d.ts.map +1 -0
  673. package/dist/server/fetchable-loader-store.js +18 -0
  674. package/dist/server/fetchable-loader-store.js.map +1 -0
  675. package/dist/server/handle-store.d.ts +85 -0
  676. package/dist/server/handle-store.d.ts.map +1 -0
  677. package/dist/server/handle-store.js +142 -0
  678. package/dist/server/handle-store.js.map +1 -0
  679. package/dist/server/loader-registry.d.ts +55 -0
  680. package/dist/server/loader-registry.d.ts.map +1 -0
  681. package/dist/server/loader-registry.js +132 -0
  682. package/dist/server/loader-registry.js.map +1 -0
  683. package/dist/server/request-context.d.ts +226 -0
  684. package/dist/server/request-context.d.ts.map +1 -0
  685. package/dist/server/request-context.js +290 -0
  686. package/dist/server/request-context.js.map +1 -0
  687. package/dist/server/root-layout.d.ts +4 -0
  688. package/dist/server/root-layout.d.ts.map +1 -0
  689. package/dist/server/root-layout.js +5 -0
  690. package/dist/server/root-layout.js.map +1 -0
  691. package/dist/server.d.ts +15 -0
  692. package/dist/server.d.ts.map +1 -0
  693. package/dist/server.js +20 -0
  694. package/dist/server.js.map +1 -0
  695. package/dist/ssr/__tests__/ssr-handler.test.d.ts +2 -0
  696. package/dist/ssr/__tests__/ssr-handler.test.d.ts.map +1 -0
  697. package/dist/ssr/__tests__/ssr-handler.test.js +132 -0
  698. package/dist/ssr/__tests__/ssr-handler.test.js.map +1 -0
  699. package/dist/ssr/index.d.ts +98 -0
  700. package/dist/ssr/index.d.ts.map +1 -0
  701. package/dist/ssr/index.js +158 -0
  702. package/dist/ssr/index.js.map +1 -0
  703. package/dist/static-handler.d.ts +50 -0
  704. package/dist/static-handler.d.ts.map +1 -0
  705. package/dist/static-handler.gen.d.ts +5 -0
  706. package/dist/static-handler.gen.d.ts.map +1 -0
  707. package/dist/static-handler.gen.js +5 -0
  708. package/dist/static-handler.gen.js.map +1 -0
  709. package/dist/static-handler.js +29 -0
  710. package/dist/static-handler.js.map +1 -0
  711. package/dist/testing/vitest.js +82 -0
  712. package/dist/theme/ThemeProvider.d.ts +20 -0
  713. package/dist/theme/ThemeProvider.d.ts.map +1 -0
  714. package/dist/theme/ThemeProvider.js +240 -0
  715. package/dist/theme/ThemeProvider.js.map +1 -0
  716. package/dist/theme/ThemeScript.d.ts +48 -0
  717. package/dist/theme/ThemeScript.d.ts.map +1 -0
  718. package/dist/theme/ThemeScript.js +13 -0
  719. package/dist/theme/ThemeScript.js.map +1 -0
  720. package/dist/theme/__tests__/theme.test.d.ts +2 -0
  721. package/dist/theme/__tests__/theme.test.d.ts.map +1 -0
  722. package/dist/theme/__tests__/theme.test.js +103 -0
  723. package/dist/theme/__tests__/theme.test.js.map +1 -0
  724. package/dist/theme/constants.d.ts +29 -0
  725. package/dist/theme/constants.d.ts.map +1 -0
  726. package/dist/theme/constants.js +48 -0
  727. package/dist/theme/constants.js.map +1 -0
  728. package/dist/theme/index.d.ts +31 -0
  729. package/dist/theme/index.d.ts.map +1 -0
  730. package/dist/theme/index.js +36 -0
  731. package/dist/theme/index.js.map +1 -0
  732. package/dist/theme/theme-context.d.ts +40 -0
  733. package/dist/theme/theme-context.d.ts.map +1 -0
  734. package/dist/theme/theme-context.js +60 -0
  735. package/dist/theme/theme-context.js.map +1 -0
  736. package/dist/theme/theme-script.d.ts +27 -0
  737. package/dist/theme/theme-script.d.ts.map +1 -0
  738. package/dist/theme/theme-script.js +147 -0
  739. package/dist/theme/theme-script.js.map +1 -0
  740. package/dist/theme/types.d.ts +163 -0
  741. package/dist/theme/types.d.ts.map +1 -0
  742. package/dist/theme/types.js +11 -0
  743. package/dist/theme/types.js.map +1 -0
  744. package/dist/theme/use-theme.d.ts +12 -0
  745. package/dist/theme/use-theme.d.ts.map +1 -0
  746. package/dist/theme/use-theme.js +40 -0
  747. package/dist/theme/use-theme.js.map +1 -0
  748. package/dist/types.d.ts +1479 -0
  749. package/dist/types.d.ts.map +1 -0
  750. package/dist/types.js +10 -0
  751. package/dist/types.js.map +1 -0
  752. package/dist/urls.d.ts +441 -0
  753. package/dist/urls.d.ts.map +1 -0
  754. package/dist/urls.gen.d.ts +8 -0
  755. package/dist/urls.gen.d.ts.map +1 -0
  756. package/dist/urls.gen.js +8 -0
  757. package/dist/urls.gen.js.map +1 -0
  758. package/dist/urls.js +443 -0
  759. package/dist/urls.js.map +1 -0
  760. package/dist/use-loader.d.ts +127 -0
  761. package/dist/use-loader.d.ts.map +1 -0
  762. package/dist/use-loader.js +237 -0
  763. package/dist/use-loader.js.map +1 -0
  764. package/dist/vite/__tests__/ast-handler-extract.test.d.ts +2 -0
  765. package/dist/vite/__tests__/ast-handler-extract.test.d.ts.map +1 -0
  766. package/dist/vite/__tests__/ast-handler-extract.test.js +294 -0
  767. package/dist/vite/__tests__/ast-handler-extract.test.js.map +1 -0
  768. package/dist/vite/__tests__/expose-id-utils.test.d.ts +2 -0
  769. package/dist/vite/__tests__/expose-id-utils.test.d.ts.map +1 -0
  770. package/dist/vite/__tests__/expose-id-utils.test.js +224 -0
  771. package/dist/vite/__tests__/expose-id-utils.test.js.map +1 -0
  772. package/dist/vite/__tests__/expose-internal-ids.test.d.ts +2 -0
  773. package/dist/vite/__tests__/expose-internal-ids.test.d.ts.map +1 -0
  774. package/dist/vite/__tests__/expose-internal-ids.test.js +647 -0
  775. package/dist/vite/__tests__/expose-internal-ids.test.js.map +1 -0
  776. package/dist/vite/__tests__/expose-router-id.test.d.ts +2 -0
  777. package/dist/vite/__tests__/expose-router-id.test.d.ts.map +1 -0
  778. package/dist/vite/__tests__/expose-router-id.test.js +39 -0
  779. package/dist/vite/__tests__/expose-router-id.test.js.map +1 -0
  780. package/dist/vite/ast-handler-extract.d.ts +49 -0
  781. package/dist/vite/ast-handler-extract.d.ts.map +1 -0
  782. package/dist/vite/ast-handler-extract.js +249 -0
  783. package/dist/vite/ast-handler-extract.js.map +1 -0
  784. package/dist/vite/expose-action-id.d.ts +19 -0
  785. package/dist/vite/expose-action-id.d.ts.map +1 -0
  786. package/dist/vite/expose-action-id.js +250 -0
  787. package/dist/vite/expose-action-id.js.map +1 -0
  788. package/dist/vite/expose-id-utils.d.ts +69 -0
  789. package/dist/vite/expose-id-utils.d.ts.map +1 -0
  790. package/dist/vite/expose-id-utils.js +289 -0
  791. package/dist/vite/expose-id-utils.js.map +1 -0
  792. package/dist/vite/expose-internal-ids.d.ts +22 -0
  793. package/dist/vite/expose-internal-ids.d.ts.map +1 -0
  794. package/dist/vite/expose-internal-ids.js +886 -0
  795. package/dist/vite/expose-internal-ids.js.map +1 -0
  796. package/dist/vite/index.d.ts +149 -0
  797. package/dist/vite/index.d.ts.map +1 -0
  798. package/dist/vite/index.js +6263 -2733
  799. package/dist/vite/index.js.bak +5448 -0
  800. package/dist/vite/index.js.map +1 -0
  801. package/dist/vite/package-resolution.d.ts +43 -0
  802. package/dist/vite/package-resolution.d.ts.map +1 -0
  803. package/{src/vite/package-resolution.ts → dist/vite/package-resolution.js} +53 -66
  804. package/dist/vite/package-resolution.js.map +1 -0
  805. package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  806. package/dist/vite/virtual-entries.d.ts +25 -0
  807. package/dist/vite/virtual-entries.d.ts.map +1 -0
  808. package/{src/vite/virtual-entries.ts → dist/vite/virtual-entries.js} +12 -16
  809. package/dist/vite/virtual-entries.js.map +1 -0
  810. package/package.json +123 -68
  811. package/skills/api-client/SKILL.md +211 -0
  812. package/skills/breadcrumbs/SKILL.md +312 -0
  813. package/skills/bundle-analysis/SKILL.md +159 -0
  814. package/skills/cache-guide/SKILL.md +486 -0
  815. package/skills/caching/SKILL.md +349 -24
  816. package/skills/composability/SKILL.md +197 -0
  817. package/skills/css/SKILL.md +76 -0
  818. package/skills/debug-manifest/SKILL.md +12 -8
  819. package/skills/document-cache/SKILL.md +87 -62
  820. package/skills/fonts/SKILL.md +6 -4
  821. package/skills/handler-use/SKILL.md +364 -0
  822. package/skills/hooks/SKILL.md +557 -79
  823. package/skills/host-router/SKILL.md +278 -0
  824. package/skills/i18n/SKILL.md +276 -0
  825. package/skills/intercept/SKILL.md +175 -8
  826. package/skills/layout/SKILL.md +128 -5
  827. package/skills/links/SKILL.md +304 -25
  828. package/skills/loader/SKILL.md +604 -54
  829. package/skills/middleware/SKILL.md +213 -37
  830. package/skills/migrate-nextjs/SKILL.md +584 -0
  831. package/skills/migrate-react-router/SKILL.md +769 -0
  832. package/skills/mime-routes/SKILL.md +41 -10
  833. package/skills/observability/SKILL.md +172 -0
  834. package/skills/parallel/SKILL.md +276 -3
  835. package/skills/prerender/SKILL.md +432 -52
  836. package/skills/rango/SKILL.md +313 -21
  837. package/skills/react-compiler/SKILL.md +168 -0
  838. package/skills/response-routes/SKILL.md +248 -120
  839. package/skills/route/SKILL.md +287 -21
  840. package/skills/router-setup/SKILL.md +231 -33
  841. package/skills/server-actions/SKILL.md +775 -0
  842. package/skills/streams-and-websockets/SKILL.md +283 -0
  843. package/skills/tailwind/SKILL.md +27 -3
  844. package/skills/testing/SKILL.md +129 -0
  845. package/skills/testing/bindings.md +89 -0
  846. package/skills/testing/cache-prerender.md +124 -0
  847. package/skills/testing/client-components.md +122 -0
  848. package/skills/testing/e2e-parity.md +125 -0
  849. package/skills/testing/flight.md +92 -0
  850. package/skills/testing/handles.md +129 -0
  851. package/skills/testing/loader.md +128 -0
  852. package/skills/testing/middleware.md +99 -0
  853. package/skills/testing/render-handler.md +121 -0
  854. package/skills/testing/response-routes.md +95 -0
  855. package/skills/testing/reverse-and-types.md +84 -0
  856. package/skills/testing/server-actions.md +107 -0
  857. package/skills/testing/server-tree.md +128 -0
  858. package/skills/testing/setup.md +120 -0
  859. package/skills/theme/SKILL.md +9 -8
  860. package/skills/typesafety/SKILL.md +547 -107
  861. package/skills/use-cache/SKILL.md +355 -0
  862. package/skills/view-transitions/SKILL.md +294 -0
  863. package/src/__augment-tests__/augment.ts +81 -0
  864. package/src/__augment-tests__/augmented.check.ts +116 -0
  865. package/src/__internal.ts +77 -44
  866. package/src/bin/rango.ts +312 -15
  867. package/src/browser/action-coordinator.ts +114 -0
  868. package/src/browser/action-fence.ts +47 -0
  869. package/src/browser/app-shell.ts +39 -0
  870. package/src/browser/app-version.ts +14 -0
  871. package/src/browser/cookie-name.ts +140 -0
  872. package/src/browser/event-controller.ts +162 -200
  873. package/src/browser/history-state.ts +101 -0
  874. package/src/browser/index.ts +3 -3
  875. package/src/browser/intercept-utils.ts +52 -0
  876. package/src/browser/invalidate-client-cache.ts +52 -0
  877. package/src/browser/link-interceptor.ts +24 -4
  878. package/src/browser/logging.ts +11 -0
  879. package/src/browser/merge-segment-loaders.ts +20 -12
  880. package/src/browser/navigation-bridge.ts +323 -563
  881. package/src/browser/navigation-client.ts +219 -75
  882. package/src/browser/navigation-store-handle.ts +38 -0
  883. package/src/browser/navigation-store.ts +104 -112
  884. package/src/browser/navigation-transaction.ts +247 -0
  885. package/src/browser/network-error-handler.ts +61 -0
  886. package/src/browser/partial-update.ts +328 -348
  887. package/src/browser/prefetch/cache.ts +324 -0
  888. package/src/browser/prefetch/fetch.ts +357 -0
  889. package/src/browser/prefetch/observer.ts +65 -0
  890. package/src/browser/prefetch/policy.ts +48 -0
  891. package/src/browser/prefetch/queue.ts +194 -0
  892. package/src/browser/prefetch/resource-ready.ts +77 -0
  893. package/src/browser/rango-state.ts +194 -0
  894. package/src/browser/react/Link.tsx +253 -71
  895. package/src/browser/react/NavigationProvider.tsx +155 -34
  896. package/src/browser/react/ScrollRestoration.tsx +10 -6
  897. package/src/browser/react/context.ts +11 -0
  898. package/src/browser/react/filter-segment-order.ts +53 -0
  899. package/src/browser/react/index.ts +0 -48
  900. package/src/browser/react/location-state-shared.ts +260 -60
  901. package/src/browser/react/location-state.ts +90 -20
  902. package/src/browser/react/mount-context.ts +6 -1
  903. package/src/browser/react/nonce-context.ts +23 -0
  904. package/src/browser/react/shallow-equal.ts +27 -0
  905. package/src/browser/react/use-action.ts +35 -66
  906. package/src/browser/react/use-handle.ts +39 -126
  907. package/src/browser/react/use-link-status.ts +6 -9
  908. package/src/browser/react/use-navigation.ts +44 -68
  909. package/src/browser/react/use-params.ts +75 -0
  910. package/src/browser/react/use-pathname.ts +47 -0
  911. package/src/browser/react/use-reverse.ts +106 -0
  912. package/src/browser/react/use-router.ts +98 -0
  913. package/src/browser/react/use-search-params.ts +51 -0
  914. package/src/browser/react/use-segments.ts +72 -99
  915. package/src/browser/response-adapter.ts +124 -0
  916. package/src/browser/rsc-router.tsx +290 -72
  917. package/src/browser/scroll-restoration.ts +132 -49
  918. package/src/browser/segment-reconciler.ts +243 -0
  919. package/src/browser/segment-structure-assert.ts +17 -1
  920. package/src/browser/server-action-bridge.ts +621 -613
  921. package/src/browser/types.ts +175 -50
  922. package/src/browser/validate-redirect-origin.ts +56 -0
  923. package/src/build/collect-fallback-refs.ts +107 -0
  924. package/src/build/generate-manifest.ts +123 -56
  925. package/src/build/generate-route-types.ts +41 -1038
  926. package/src/build/index.ts +9 -6
  927. package/src/build/prefix-tree-utils.ts +123 -0
  928. package/src/build/route-trie.ts +165 -34
  929. package/src/build/route-types/ast-helpers.ts +25 -0
  930. package/src/build/route-types/ast-route-extraction.ts +98 -0
  931. package/src/build/route-types/codegen.ts +113 -0
  932. package/src/build/route-types/include-resolution.ts +418 -0
  933. package/src/build/route-types/param-extraction.ts +51 -0
  934. package/src/build/route-types/per-module-writer.ts +131 -0
  935. package/src/build/route-types/router-processing.ts +651 -0
  936. package/src/build/route-types/scan-filter.ts +85 -0
  937. package/src/build/route-types/source-scan.ts +118 -0
  938. package/src/build/runtime-discovery.ts +220 -0
  939. package/src/cache/background-task.ts +34 -0
  940. package/src/cache/cache-error.ts +104 -0
  941. package/src/cache/cache-key-utils.ts +44 -0
  942. package/src/cache/cache-policy.ts +165 -0
  943. package/src/cache/cache-runtime.ts +444 -0
  944. package/src/cache/cache-scope.ts +231 -325
  945. package/src/cache/cache-tag.ts +98 -0
  946. package/src/cache/cf/cf-cache-store.ts +2644 -75
  947. package/src/cache/cf/index.ts +17 -17
  948. package/src/cache/document-cache.ts +172 -92
  949. package/src/cache/handle-capture.ts +81 -0
  950. package/src/cache/handle-snapshot.ts +104 -0
  951. package/src/cache/index.ts +11 -35
  952. package/src/cache/memory-segment-store.ts +307 -30
  953. package/src/cache/profile-registry.ts +49 -0
  954. package/src/cache/read-through-swr.ts +164 -0
  955. package/src/cache/segment-codec.ts +240 -0
  956. package/src/cache/tag-invalidation.ts +230 -0
  957. package/src/cache/taint.ts +153 -0
  958. package/src/cache/types.ts +94 -211
  959. package/src/client.rsc.tsx +8 -21
  960. package/src/client.tsx +123 -347
  961. package/src/cloudflare/index.ts +11 -0
  962. package/src/cloudflare/tracing.ts +109 -0
  963. package/src/component-utils.ts +23 -4
  964. package/src/components/DefaultDocument.tsx +5 -1
  965. package/src/context-var.ts +168 -0
  966. package/src/debug.ts +19 -9
  967. package/src/decode-loader-results.ts +36 -0
  968. package/src/defer.ts +196 -0
  969. package/src/deps/ssr.ts +0 -1
  970. package/src/errors.ts +106 -10
  971. package/src/handle.ts +76 -23
  972. package/src/handles/MetaTags.tsx +73 -34
  973. package/src/handles/breadcrumbs.ts +77 -0
  974. package/src/handles/meta.ts +30 -52
  975. package/src/host/cookie-handler.ts +21 -51
  976. package/src/host/errors.ts +8 -32
  977. package/src/host/index.ts +12 -9
  978. package/src/host/pattern-matcher.ts +34 -77
  979. package/src/host/router.ts +151 -121
  980. package/src/host/testing.ts +45 -32
  981. package/src/host/types.ts +52 -11
  982. package/src/host/utils.ts +2 -2
  983. package/src/href-client.ts +192 -57
  984. package/src/index.rsc.ts +174 -34
  985. package/src/index.ts +241 -73
  986. package/src/internal-debug.ts +8 -4
  987. package/src/loader-store.ts +500 -0
  988. package/src/loader.rsc.ts +31 -99
  989. package/src/loader.ts +30 -12
  990. package/src/missing-id-error.ts +68 -0
  991. package/src/network-error-thrower.tsx +4 -7
  992. package/src/outlet-context.ts +1 -1
  993. package/src/outlet-provider.tsx +41 -0
  994. package/src/prerender/param-hash.ts +14 -13
  995. package/src/prerender/store.ts +121 -21
  996. package/src/prerender.ts +445 -24
  997. package/src/redirect-origin.ts +100 -0
  998. package/src/response-utils.ts +37 -0
  999. package/src/reverse.ts +198 -128
  1000. package/src/root-error-boundary.tsx +42 -48
  1001. package/src/route-content-wrapper.tsx +10 -72
  1002. package/src/route-definition/dsl-helpers.ts +1116 -0
  1003. package/src/route-definition/helper-factories.ts +90 -0
  1004. package/src/route-definition/helpers-types.ts +506 -0
  1005. package/src/route-definition/index.ts +55 -0
  1006. package/src/route-definition/redirect.ts +135 -0
  1007. package/src/route-definition/resolve-handler-use.ts +161 -0
  1008. package/src/route-definition/use-item-types.ts +32 -0
  1009. package/src/route-definition.ts +1 -1450
  1010. package/src/route-map-builder.ts +82 -144
  1011. package/src/route-name.ts +53 -0
  1012. package/src/route-types.ts +72 -41
  1013. package/src/router/basename.ts +14 -0
  1014. package/src/router/content-negotiation.ts +215 -0
  1015. package/src/router/debug-manifest.ts +72 -0
  1016. package/src/router/error-handling.ts +22 -26
  1017. package/src/router/find-match.ts +181 -0
  1018. package/src/router/handler-context.ts +372 -125
  1019. package/src/router/instrument.ts +346 -0
  1020. package/src/router/intercept-resolution.ts +34 -27
  1021. package/src/router/lazy-includes.ts +200 -0
  1022. package/src/router/loader-resolution.ts +381 -147
  1023. package/src/router/logging.ts +106 -6
  1024. package/src/router/manifest.ts +88 -51
  1025. package/src/router/match-api.ts +162 -245
  1026. package/src/router/match-context.ts +4 -24
  1027. package/src/router/match-handlers.ts +440 -0
  1028. package/src/router/match-middleware/background-revalidation.ts +90 -89
  1029. package/src/router/match-middleware/cache-lookup.ts +297 -150
  1030. package/src/router/match-middleware/cache-store.ts +75 -33
  1031. package/src/router/match-middleware/intercept-resolution.ts +44 -43
  1032. package/src/router/match-middleware/segment-resolution.ts +64 -22
  1033. package/src/router/match-pipelines.ts +11 -87
  1034. package/src/router/match-result.ts +121 -50
  1035. package/src/router/metrics.ts +219 -28
  1036. package/src/router/middleware-types.ts +93 -0
  1037. package/src/router/middleware.ts +421 -413
  1038. package/src/router/navigation-snapshot.ts +131 -0
  1039. package/src/router/params-util.ts +23 -0
  1040. package/src/router/pattern-matching.ts +263 -79
  1041. package/src/router/prerender-match.ts +541 -0
  1042. package/src/router/preview-match.ts +100 -0
  1043. package/src/router/request-classification.ts +276 -0
  1044. package/src/router/revalidation.ts +143 -44
  1045. package/src/router/route-snapshot.ts +244 -0
  1046. package/src/router/router-context.ts +41 -47
  1047. package/src/router/router-interfaces.ts +525 -0
  1048. package/src/router/router-options.ts +726 -0
  1049. package/src/router/router-registry.ts +21 -0
  1050. package/src/router/segment-resolution/fresh.ts +747 -0
  1051. package/src/router/segment-resolution/helpers.ts +314 -0
  1052. package/src/router/segment-resolution/loader-cache.ts +207 -0
  1053. package/src/router/segment-resolution/revalidation.ts +1322 -0
  1054. package/src/router/segment-resolution/static-store.ts +81 -0
  1055. package/src/router/segment-resolution/streamed-handler-telemetry.ts +52 -0
  1056. package/src/router/segment-resolution/view-transition-default.ts +36 -0
  1057. package/src/router/segment-resolution.ts +24 -1354
  1058. package/src/router/segment-wrappers.ts +289 -0
  1059. package/src/router/state-cookie-name.ts +33 -0
  1060. package/src/router/substitute-pattern-params.ts +56 -0
  1061. package/src/router/telemetry-otel.ts +261 -0
  1062. package/src/router/telemetry.ts +377 -0
  1063. package/src/router/timeout.ts +128 -0
  1064. package/src/router/tracing.ts +206 -0
  1065. package/src/router/trie-matching.ts +172 -60
  1066. package/src/router/types.ts +23 -70
  1067. package/src/router/url-params.ts +44 -0
  1068. package/src/router.ts +748 -2376
  1069. package/src/rsc/handler-context.ts +46 -0
  1070. package/src/rsc/handler.ts +861 -1141
  1071. package/src/rsc/helpers.ts +269 -19
  1072. package/src/rsc/index.ts +1 -21
  1073. package/src/rsc/json-route-result.ts +38 -0
  1074. package/src/rsc/loader-fetch.ts +235 -0
  1075. package/src/rsc/manifest-init.ts +77 -0
  1076. package/src/rsc/nonce.ts +14 -0
  1077. package/src/rsc/origin-guard.ts +155 -0
  1078. package/src/rsc/progressive-enhancement.ts +413 -0
  1079. package/src/rsc/redirect-guard.ts +99 -0
  1080. package/src/rsc/response-error.ts +104 -0
  1081. package/src/rsc/response-route-handler.ts +374 -0
  1082. package/src/rsc/rsc-rendering.ts +261 -0
  1083. package/src/rsc/runtime-warnings.ts +55 -0
  1084. package/src/rsc/server-action.ts +376 -0
  1085. package/src/rsc/ssr-setup.ts +144 -0
  1086. package/src/rsc/types.ts +58 -12
  1087. package/src/runtime-env.ts +18 -0
  1088. package/src/search-params.ts +70 -74
  1089. package/src/segment-content-promise.ts +67 -0
  1090. package/src/segment-loader-promise.ts +134 -0
  1091. package/src/segment-system.tsx +292 -134
  1092. package/src/serialize.ts +243 -0
  1093. package/src/server/context.ts +439 -85
  1094. package/src/server/cookie-store.ts +265 -0
  1095. package/src/server/fetchable-loader-store.ts +11 -6
  1096. package/src/server/handle-store.ts +112 -31
  1097. package/src/server/loader-registry.ts +23 -82
  1098. package/src/server/request-context.ts +724 -143
  1099. package/src/server.ts +26 -164
  1100. package/src/ssr/index.tsx +113 -36
  1101. package/src/static-handler.ts +45 -18
  1102. package/src/testing/cache-status.ts +162 -0
  1103. package/src/testing/collect-handle.ts +40 -0
  1104. package/src/testing/dispatch.ts +618 -0
  1105. package/src/testing/dom.entry.ts +22 -0
  1106. package/src/testing/e2e/fixture.ts +188 -0
  1107. package/src/testing/e2e/index.ts +128 -0
  1108. package/src/testing/e2e/matchers.ts +35 -0
  1109. package/src/testing/e2e/page-helpers.ts +272 -0
  1110. package/src/testing/e2e/parity.ts +387 -0
  1111. package/src/testing/e2e/server.ts +195 -0
  1112. package/src/testing/flight-matchers.ts +97 -0
  1113. package/src/testing/flight-normalize.ts +11 -0
  1114. package/src/testing/flight-runtime.d.ts +57 -0
  1115. package/src/testing/flight-tree.ts +682 -0
  1116. package/src/testing/flight.entry.ts +52 -0
  1117. package/src/testing/flight.ts +232 -0
  1118. package/src/testing/generated-routes.ts +183 -0
  1119. package/src/testing/index.ts +99 -0
  1120. package/src/testing/internal/context.ts +348 -0
  1121. package/src/testing/internal/flight-client-globals.ts +30 -0
  1122. package/src/testing/internal/seed-vars.ts +54 -0
  1123. package/src/testing/render-handler.ts +330 -0
  1124. package/src/testing/render-route.tsx +566 -0
  1125. package/src/testing/run-loader.ts +378 -0
  1126. package/src/testing/run-middleware.ts +205 -0
  1127. package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
  1128. package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
  1129. package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
  1130. package/src/testing/vitest-stubs/version.ts +5 -0
  1131. package/src/testing/vitest.ts +305 -0
  1132. package/src/theme/ThemeProvider.tsx +21 -67
  1133. package/src/theme/ThemeScript.tsx +5 -11
  1134. package/src/theme/constants.ts +5 -14
  1135. package/src/theme/index.ts +3 -20
  1136. package/src/theme/theme-context.ts +5 -35
  1137. package/src/theme/theme-script.ts +21 -32
  1138. package/src/theme/use-theme.ts +0 -3
  1139. package/src/types/boundaries.ts +123 -0
  1140. package/src/types/cache-types.ts +207 -0
  1141. package/src/types/error-types.ts +132 -0
  1142. package/src/types/global-namespace.ts +113 -0
  1143. package/src/types/handler-context.ts +839 -0
  1144. package/src/types/index.ts +79 -0
  1145. package/src/types/loader-types.ts +212 -0
  1146. package/src/types/request-scope.ts +107 -0
  1147. package/src/types/route-config.ts +126 -0
  1148. package/src/types/route-entry.ts +114 -0
  1149. package/src/types/segments.ts +171 -0
  1150. package/src/types.ts +1 -1795
  1151. package/src/urls/include-helper.ts +160 -0
  1152. package/src/urls/index.ts +43 -0
  1153. package/src/urls/path-helper-types.ts +386 -0
  1154. package/src/urls/path-helper.ts +275 -0
  1155. package/src/urls/pattern-types.ts +124 -0
  1156. package/src/urls/response-types.ts +109 -0
  1157. package/src/urls/type-extraction.ts +291 -0
  1158. package/src/urls/urls-function.ts +81 -0
  1159. package/src/urls.ts +1 -1323
  1160. package/src/use-loader.tsx +406 -141
  1161. package/src/vite/debug.ts +185 -0
  1162. package/src/vite/discovery/bundle-postprocess.ts +182 -0
  1163. package/src/vite/discovery/discover-routers.ts +389 -0
  1164. package/src/vite/discovery/discovery-errors.ts +194 -0
  1165. package/src/vite/discovery/gate-state.ts +171 -0
  1166. package/src/vite/discovery/prerender-collection.ts +467 -0
  1167. package/src/vite/discovery/route-types-writer.ts +214 -0
  1168. package/src/vite/discovery/self-gen-tracking.ts +73 -0
  1169. package/src/vite/discovery/state.ts +161 -0
  1170. package/src/vite/discovery/virtual-module-codegen.ts +183 -0
  1171. package/src/vite/index.ts +17 -2259
  1172. package/src/vite/plugin-types.ts +221 -0
  1173. package/src/vite/plugins/cjs-to-esm.ts +83 -0
  1174. package/src/vite/plugins/client-ref-dedup.ts +120 -0
  1175. package/src/vite/plugins/client-ref-hashing.ts +118 -0
  1176. package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
  1177. package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  1178. package/src/vite/plugins/cloudflare-protocol-stub.ts +194 -0
  1179. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +88 -110
  1180. package/src/vite/{expose-id-utils.ts → plugins/expose-id-utils.ts} +13 -87
  1181. package/src/vite/plugins/expose-ids/export-analysis.ts +338 -0
  1182. package/src/vite/plugins/expose-ids/handler-transform.ts +141 -0
  1183. package/src/vite/plugins/expose-ids/loader-transform.ts +57 -0
  1184. package/src/vite/plugins/expose-ids/router-transform.ts +124 -0
  1185. package/src/vite/plugins/expose-ids/types.ts +45 -0
  1186. package/src/vite/plugins/expose-internal-ids.ts +806 -0
  1187. package/src/vite/plugins/performance-tracks.ts +89 -0
  1188. package/src/vite/plugins/refresh-cmd.ts +127 -0
  1189. package/src/vite/plugins/use-cache-transform.ts +313 -0
  1190. package/src/vite/plugins/version-injector.ts +79 -0
  1191. package/src/vite/plugins/version-plugin.ts +275 -0
  1192. package/src/vite/plugins/virtual-entries.ts +108 -0
  1193. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  1194. package/src/vite/rango.ts +444 -0
  1195. package/src/vite/router-discovery.ts +1581 -0
  1196. package/src/vite/{ast-handler-extract.ts → utils/ast-handler-extract.ts} +193 -37
  1197. package/src/vite/utils/banner.ts +36 -0
  1198. package/src/vite/utils/bundle-analysis.ts +132 -0
  1199. package/src/vite/utils/client-chunks.ts +184 -0
  1200. package/src/vite/utils/forward-user-plugins.ts +171 -0
  1201. package/src/vite/utils/manifest-utils.ts +15 -0
  1202. package/src/vite/utils/package-resolution.ts +89 -0
  1203. package/src/vite/utils/prerender-utils.ts +223 -0
  1204. package/src/vite/utils/shared-utils.ts +219 -0
  1205. package/CLAUDE.md +0 -43
  1206. package/src/browser/lru-cache.ts +0 -69
  1207. package/src/browser/request-controller.ts +0 -164
  1208. package/src/browser/shallow.ts +0 -35
  1209. package/src/cache/memory-store.ts +0 -253
  1210. package/src/router.gen.ts +0 -6
  1211. package/src/static-handler.gen.ts +0 -5
  1212. package/src/urls.gen.ts +0 -8
  1213. package/src/vite/expose-internal-ids.ts +0 -1167
  1214. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -1,5 +1,5 @@
1
1
  /// <reference types="@vitejs/plugin-rsc/types" />
2
- /// <reference path="../vite/version.d.ts" />
2
+ /// <reference path="../vite/plugins/version.d.ts" />
3
3
  /**
4
4
  * RSC Request Handler
5
5
  *
@@ -8,72 +8,93 @@
8
8
  */
9
9
 
10
10
  import { createElement } from "react";
11
- import { renderSegments } from "../segment-system.js";
12
- import { RouteNotFoundError, RouterError } from "../errors.js";
13
- import type { ResponseError } from "../urls.js";
14
- import { getLoaderLazy } from "../server/loader-registry.js";
15
- import {
16
- matchMiddleware,
17
- executeMiddleware,
18
- executeLoaderMiddleware,
19
- } from "../router/middleware.js";
11
+ import { isRouteNotFoundError } from "../errors.js";
12
+ import { matchMiddleware, executeMiddleware } from "../router/middleware.js";
20
13
  import {
21
14
  runWithRequestContext,
22
15
  setRequestContextParams,
23
16
  requireRequestContext,
17
+ getRequestContext,
18
+ _getRequestContext,
24
19
  createRequestContext,
25
- type ExecutionContext,
26
20
  } from "../server/request-context.js";
27
21
  import * as rscDeps from "@vitejs/plugin-rsc/rsc";
28
-
29
22
  import type {
30
23
  RscPayload,
31
- ReactFormState,
32
24
  CreateRSCHandlerOptions,
25
+ LoadSSRModule,
26
+ SSRModule,
33
27
  } from "./types.js";
34
- import { hasBodyContent, createResponseWithMergedHeaders } from "./helpers.js";
35
- import { generateNonce } from "./nonce.js";
28
+ import {
29
+ createResponseWithMergedHeaders,
30
+ finalizeResponse,
31
+ interceptRedirectForPartial,
32
+ buildRouteMiddlewareEntries,
33
+ } from "./helpers.js";
34
+ import { guardOutgoingRedirect } from "./redirect-guard.js";
35
+ import { isWebSocketUpgradeResponse } from "../response-utils.js";
36
+ import {
37
+ handleResponseRoute,
38
+ type ResponseRouteMatch,
39
+ } from "./response-route-handler.js";
40
+ import { generateNonce, nonce as nonceToken } from "./nonce.js";
36
41
  import { VERSION } from "@rangojs/router:version";
37
42
  import type { ErrorPhase } from "../types.js";
43
+ import type { RouterRequestInput } from "../router/router-interfaces.js";
38
44
  import { invokeOnError } from "../router/error-handling.js";
39
45
  import {
40
- getGlobalRouteMap,
46
+ createReverseFunction,
47
+ stripInternalParams,
48
+ } from "../router/handler-context.js";
49
+ import { contextSet } from "../context-var.js";
50
+ import {
41
51
  hasCachedManifest,
42
- setCachedManifest,
43
52
  getRouteTrie,
44
- setRouteTrie,
45
53
  getPrecomputedEntries,
46
54
  waitForManifestReady,
47
55
  getRouterManifest,
48
56
  getRouterTrie,
49
- setRouterManifest,
50
- setRouterTrie,
51
57
  } from "../route-map-builder.js";
52
-
53
- /**
54
- * Build a ResponseError payload from a caught error.
55
- * RouterError messages are always exposed (developer-crafted).
56
- * Standard Error messages are hidden in production.
57
- */
58
- function createResponseErrorPayload(error: unknown, isDev: boolean): ResponseError {
59
- if (error instanceof RouterError) {
60
- return {
61
- message: error.message,
62
- code: error.code,
63
- ...(error.type ? { type: error.type } : {}),
64
- ...(isDev && error.stack ? { stack: error.stack } : {}),
65
- };
66
- }
67
- if (error instanceof Error) {
68
- return {
69
- message: isDev ? error.message : "Internal Server Error",
70
- ...(isDev && error.stack ? { stack: error.stack } : {}),
71
- };
72
- }
73
- return {
74
- message: isDev ? String(error) : "Internal Server Error",
75
- };
76
- }
58
+ import type { HandlerContext } from "./handler-context.js";
59
+ import type { CacheErrorCategory } from "../cache/cache-error.js";
60
+ import type { SegmentCacheStore } from "../cache/types.js";
61
+ import { buildRouterTrieFromUrlpatterns } from "./manifest-init.js";
62
+ import { handleProgressiveEnhancement } from "./progressive-enhancement.js";
63
+ import {
64
+ executeServerAction,
65
+ revalidateAfterAction,
66
+ type ActionContinuation,
67
+ } from "./server-action.js";
68
+ import { handleLoaderFetch } from "./loader-fetch.js";
69
+ import {
70
+ checkRequestOrigin,
71
+ ORIGIN_CHECK_PHASE_BY_MODE,
72
+ } from "./origin-guard.js";
73
+ import { handleRscRendering } from "./rsc-rendering.js";
74
+ import {
75
+ withTimeout,
76
+ RouterTimeoutError,
77
+ createDefaultTimeoutResponse,
78
+ type TimeoutPhase,
79
+ } from "../router/timeout.js";
80
+ import {
81
+ createMetricsStore,
82
+ appendMetric,
83
+ buildMetricsTiming,
84
+ } from "../router/metrics.js";
85
+ import { observePhase, observeEvent, PHASES } from "../router/instrument.js";
86
+ import {
87
+ startSSRSetup,
88
+ getSSRSetup,
89
+ mayNeedSSR,
90
+ isRscRequest,
91
+ SSR_SETUP_VAR,
92
+ } from "./ssr-setup.js";
93
+ import {
94
+ classifyRequest,
95
+ type RequestPlan,
96
+ type ExecutableRequestPlan,
97
+ } from "../router/request-classification.js";
77
98
 
78
99
  /**
79
100
  * Create an RSC request handler.
@@ -108,12 +129,35 @@ function createResponseErrorPayload(error: unknown, isDev: boolean): ResponseErr
108
129
  * });
109
130
  * ```
110
131
  */
132
+
133
+ /**
134
+ * Response that tells the client to do a full document navigation. Shared by
135
+ * the terminal reload plans (version-mismatch and app-switch): an empty 200
136
+ * carrying X-RSC-Reload, which the client turns into window.location.href.
137
+ */
138
+ function createReloadResponse(reloadUrl: string) {
139
+ return createResponseWithMergedHeaders(null, {
140
+ status: 200,
141
+ headers: {
142
+ "X-RSC-Reload": reloadUrl,
143
+ "content-type": "text/x-component;charset=utf-8",
144
+ },
145
+ });
146
+ }
147
+
111
148
  export function createRSCHandler<
112
149
  TEnv = unknown,
113
150
  TRoutes extends Record<string, string> = Record<string, string>,
114
151
  >(options: CreateRSCHandlerOptions<TEnv, TRoutes>) {
115
152
  const { router, version = VERSION, nonce: nonceProvider } = options;
116
153
 
154
+ // Handler-owned registry of explicit per-scope stores from cache({ store }).
155
+ // Lives in the closure so it is scoped per handler (multi-router deployments
156
+ // get separate registries) and accumulates every explicit store this handler
157
+ // resolves across requests. updateTag()/revalidateTag() iterate it to reach
158
+ // stores not covered by the app-level ctx._cacheStore.
159
+ const explicitTaggedStores = new Set<SegmentCacheStore>();
160
+
117
161
  // Use provided deps or default to @vitejs/plugin-rsc/rsc exports
118
162
  const deps = options.deps ?? rscDeps;
119
163
  const {
@@ -125,33 +169,175 @@ export function createRSCHandler<
125
169
  decodeFormState,
126
170
  } = deps;
127
171
 
128
- // Use provided loadSSRModule or default to vite RSC module loader
129
- const loadSSRModule =
172
+ // Use provided loadSSRModule or default to vite RSC module loader.
173
+ // In production the SSR module is stable across requests, so memoize
174
+ // the dynamic import to avoid repeated module resolution overhead.
175
+ // In dev mode Vite may hot-reload the module, so skip memoization.
176
+ const rawLoadSSRModule: LoadSSRModule =
130
177
  options.loadSSRModule ??
131
178
  (() => import.meta.viteRsc.loadModule("ssr", "index"));
179
+ let _ssrModulePromise: Promise<SSRModule> | undefined;
180
+ const loadSSRModule: LoadSSRModule =
181
+ process.env.NODE_ENV === "production"
182
+ ? () =>
183
+ (_ssrModulePromise ??= rawLoadSSRModule().catch((err) => {
184
+ _ssrModulePromise = undefined;
185
+ throw err;
186
+ }))
187
+ : rawLoadSSRModule;
132
188
 
133
189
  /**
134
- * Wrapper for invokeOnError that binds the router's onError callback.
135
- * Uses the shared utility from router/error-handling.ts for consistent behavior.
190
+ * Per-request error reporter that deduplicates via the ALS request context.
191
+ *
192
+ * Uses the same _reportedErrors WeakSet as the router layer so errors
193
+ * that propagate across layers are only reported once per request.
136
194
  */
137
195
  function callOnError(
138
196
  error: unknown,
139
197
  phase: ErrorPhase,
140
198
  context: Parameters<typeof invokeOnError<TEnv>>[3],
141
199
  ): void {
200
+ // Guard: abort signal handlers fire asynchronously outside the ALS
201
+ // request scope, so the context may be gone. Skip dedup in that
202
+ // case — the error is from a cancelled stream, not a real failure.
203
+ const reqCtx = _getRequestContext();
204
+ if (error != null && typeof error === "object" && reqCtx) {
205
+ if (reqCtx._reportedErrors.has(error)) return;
206
+ reqCtx._reportedErrors.add(error);
207
+ }
142
208
  invokeOnError(router.onError, error, phase, context, "RSC");
143
209
  }
144
210
 
145
- return async function handler(
211
+ function getRequiredRouteMap(): Record<string, string> {
212
+ const routeMap = getRouterManifest(router.id);
213
+ if (!routeMap) {
214
+ throw new Error(
215
+ `Route manifest for router "${router.id}" is not available.`,
216
+ );
217
+ }
218
+ return routeMap;
219
+ }
220
+
221
+ /**
222
+ * Handle a timeout by reporting the error, emitting telemetry,
223
+ * and returning either the custom onTimeout response or a default 504.
224
+ */
225
+ async function handleTimeoutResponse(
146
226
  request: Request,
147
- env: TEnv & { ctx?: ExecutionContext } = {} as TEnv & {
148
- ctx?: ExecutionContext;
227
+ env: TEnv,
228
+ url: URL,
229
+ phase: TimeoutPhase,
230
+ durationMs: number,
231
+ routeKey?: string,
232
+ actionId?: string,
233
+ ): Promise<Response> {
234
+ const timeoutError = new RouterTimeoutError(phase, durationMs);
235
+
236
+ callOnError(timeoutError, phase === "action" ? "action" : "handler", {
237
+ request,
238
+ url,
239
+ env,
240
+ routeKey,
241
+ actionId,
242
+ handledByBoundary: false,
243
+ metadata: { timeout: true, phase, durationMs },
244
+ });
245
+
246
+ observeEvent({
247
+ type: "request.timeout",
248
+ timestamp: performance.now(),
249
+ phase,
250
+ pathname: url.pathname,
251
+ routeKey,
252
+ actionId,
253
+ durationMs,
254
+ customHandler: !!router.onTimeout,
255
+ });
256
+
257
+ if (router.onTimeout) {
258
+ try {
259
+ return await router.onTimeout({
260
+ phase,
261
+ request,
262
+ url,
263
+ env,
264
+ routeKey,
265
+ actionId,
266
+ durationMs,
267
+ });
268
+ } catch (e) {
269
+ if (process.env.NODE_ENV !== "production") {
270
+ console.error("[RSC] onTimeout callback error:", e);
271
+ }
272
+ return createDefaultTimeoutResponse(phase);
273
+ }
274
+ }
275
+
276
+ return createDefaultTimeoutResponse(phase);
277
+ }
278
+
279
+ /**
280
+ * Build a 200 Flight response that carries a redirect URL and optional state.
281
+ * Used when a partial/action request results in a redirect -- fetch
282
+ * auto-follows 3xx so we send the redirect as payload metadata instead.
283
+ */
284
+ function createRedirectFlightResponse(
285
+ redirectUrl: string,
286
+ locationState?: Record<string, unknown>,
287
+ external?: boolean,
288
+ ): Response {
289
+ const redirectPayload: RscPayload = {
290
+ metadata: {
291
+ pathname: redirectUrl,
292
+ segments: [],
293
+ redirect: { url: redirectUrl, ...(external && { external: true }) },
294
+ ...(locationState && { locationState }),
295
+ },
296
+ };
297
+ const rscStream = renderToReadableStream<RscPayload>(redirectPayload);
298
+ return createResponseWithMergedHeaders(rscStream, {
299
+ status: 200,
300
+ headers: { "content-type": "text/x-component;charset=utf-8" },
301
+ });
302
+ }
303
+
304
+ // Bundle shared dependencies for extracted handler functions.
305
+ // callOnError reads from ALS so it's inherently per-request scoped.
306
+ const handlerCtx: HandlerContext<TEnv> = {
307
+ router,
308
+ version,
309
+ renderToReadableStream,
310
+ decodeReply,
311
+ createTemporaryReferenceSet,
312
+ loadServerAction,
313
+ decodeAction,
314
+ decodeFormState,
315
+ loadSSRModule,
316
+ callOnError,
317
+ getRequiredRouteMap,
318
+ createRedirectFlightResponse,
319
+ resolveStreamMode: async (request, env, url) => {
320
+ const resolver = router.ssr?.resolveStreaming;
321
+ if (!resolver) return "stream";
322
+ return resolver({ request, env, url });
149
323
  },
324
+ };
325
+
326
+ return async function handler(
327
+ request: Request,
328
+ input: RouterRequestInput<TEnv> = {},
150
329
  ): Promise<Response> {
151
330
  const handlerStart = performance.now();
331
+ // Create the metrics store at handler start so handler:total has startTime=0
332
+ // and all metrics are relative to the request entry point.
333
+ const earlyMetricsStore = router.debugPerformance
334
+ ? createMetricsStore(true, handlerStart)
335
+ : undefined;
336
+
337
+ const { env = {} as TEnv, vars: initialVars, ctx: executionCtx } = input;
152
338
 
153
339
  // Connection warmup: return 204 immediately before any processing
154
- if (router.warmupEnabled && request.method === "HEAD") {
340
+ if (router?.warmupEnabled && request.method === "HEAD") {
155
341
  const warmupUrl = new URL(request.url);
156
342
  if (warmupUrl.searchParams.has("_rsc_warmup")) {
157
343
  return new Response(null, { status: 204 });
@@ -175,24 +361,27 @@ export function createRSCHandler<
175
361
  const mwMatchDur = performance.now() - mwMatchStart;
176
362
 
177
363
  // Shared variables between middleware and route handlers
178
- // Initialize from env.Variables if provided (allows pre-seeding from worker entry)
179
- const variables: Record<string, any> = {
180
- ...((env as any)?.Variables ?? {}),
181
- };
364
+ // Initialize from input.vars if provided (allows pre-seeding from worker entry)
365
+ const variables: Record<string, any> = initialVars
366
+ ? { ...initialVars }
367
+ : {};
182
368
 
183
- // Store nonce in variables so middleware can access via ctx.get('nonce')
369
+ // Store nonce via ContextVar token and string key for backward compat
184
370
  if (nonce) {
371
+ contextSet(variables, nonceToken, nonce);
185
372
  variables.nonce = nonce;
186
373
  }
187
374
 
188
375
  // Resolve cache store configuration
189
376
  // Priority: options.cache (handler override) > router.cache (router default)
190
377
  // Store is enabled only if: config provided, enabled, and no ?__no_cache query param
191
- let cacheStore = undefined;
378
+ let cacheStore: SegmentCacheStore | undefined;
192
379
  const cacheOption = options.cache ?? router.cache;
193
380
  if (cacheOption && !url.searchParams.has("__no_cache")) {
194
381
  const cacheConfig =
195
- typeof cacheOption === "function" ? cacheOption(env) : cacheOption;
382
+ typeof cacheOption === "function"
383
+ ? cacheOption(env, executionCtx)
384
+ : cacheOption;
196
385
 
197
386
  if (cacheConfig.enabled !== false) {
198
387
  cacheStore = cacheConfig.store;
@@ -223,56 +412,7 @@ export function createRSCHandler<
223
412
  // Cloudflare dev: generate manifest inline for this router.
224
413
  // Each router generates its own manifest independently so
225
414
  // multi-router setups (host routing) work correctly.
226
- const { generateManifest } =
227
- await import("../build/generate-manifest.js");
228
- const generated = generateManifest(router.urlpatterns);
229
- if (
230
- generated._routeAncestry &&
231
- Object.keys(generated._routeAncestry).length > 0
232
- ) {
233
- const { buildRouteTrie } = await import("../build/route-trie.js");
234
- // Map each route to its include() staticPrefix so the trie
235
- // returns the correct sp for lazy entry lookup in findMatch.
236
- const routeToStaticPrefix: Record<string, string> = {};
237
- for (const name of Object.keys(generated.routeManifest)) {
238
- routeToStaticPrefix[name] = "";
239
- }
240
- // Override with prefix from include() entries so the trie
241
- // returns the correct sp for lazy entry lookup in findMatch.
242
- // Walk recursively to include routes in nested includes.
243
- if (generated.prefixTree) {
244
- const visitPrefixNode = (node: any): void => {
245
- const sp = node.staticPrefix || "";
246
- for (const route of (node.routes || [])) {
247
- routeToStaticPrefix[route] = sp;
248
- }
249
- for (const child of Object.values(node.children || {})) {
250
- visitPrefixNode(child);
251
- }
252
- };
253
- for (const node of Object.values(generated.prefixTree)) {
254
- visitPrefixNode(node);
255
- }
256
- }
257
- const trie = buildRouteTrie(
258
- generated.routeManifest,
259
- generated._routeAncestry,
260
- routeToStaticPrefix,
261
- generated.routeTrailingSlash,
262
- generated.prerenderRoutes ? new Set(generated.prerenderRoutes) : undefined,
263
- generated.passthroughRoutes ? new Set(generated.passthroughRoutes) : undefined,
264
- generated.responseTypeRoutes,
265
- );
266
- setRouterTrie(router.id, trie);
267
- // Set global trie only if not already set by another router
268
- if (!getRouteTrie()) {
269
- setRouteTrie(trie);
270
- }
271
- }
272
- setRouterManifest(router.id, generated.routeManifest);
273
- // Merge into global manifest (needed for reverse/href across routers)
274
- const existing = hasCachedManifest() ? getGlobalRouteMap() : {};
275
- setCachedManifest({ ...existing, ...generated.routeManifest });
415
+ await buildRouterTrieFromUrlpatterns(router);
276
416
  }
277
417
  if (!getRouterManifest(router.id) && !hasCachedManifest()) {
278
418
  throw new Error(
@@ -280,10 +420,17 @@ export function createRSCHandler<
280
420
  );
281
421
  }
282
422
  }
283
- const manifestCacheDur = performance.now() - manifestCacheStart;
284
423
 
285
- // Note: Route map for useHref() is loaded lazily via getGlobalRouteMap()
286
- // This allows it to include all routes from lazy includes after manifest loading
424
+ // Rebuild the trie when the manifest exists but the per-router trie is
425
+ // missing. This happens in dev mode after HMR: the virtual module sets
426
+ // the manifest (from fresh gen files) but the trie is intentionally not
427
+ // injected to avoid stale discovery-time data. Without the trie, route
428
+ // matching falls back to regex iteration which does not handle wildcard
429
+ // priority correctly (catch-all patterns match before specific routes).
430
+ if (!getRouterTrie(router.id) && router.urlpatterns) {
431
+ await buildRouterTrieFromUrlpatterns(router);
432
+ }
433
+ const manifestCacheDur = performance.now() - manifestCacheStart;
287
434
 
288
435
  // Create unified request context with all methods
289
436
  // Includes: stub response, handle store, loader memoization, use(), cookies, headers, cache store
@@ -295,9 +442,30 @@ export function createRSCHandler<
295
442
  url,
296
443
  variables,
297
444
  cacheStore,
298
- executionContext: env.ctx,
445
+ explicitTaggedStores,
446
+ cacheProfiles: router.cacheProfiles,
447
+ executionContext: executionCtx,
299
448
  themeConfig: router.themeConfig,
449
+ stateCookieName: router.resolvedStateCookieName,
450
+ version,
300
451
  });
452
+ if (earlyMetricsStore) {
453
+ requestContext._debugPerformance = true;
454
+ requestContext._metricsStore = earlyMetricsStore;
455
+ }
456
+ // Wire background error reporting so "use cache" and other subsystems
457
+ // can surface non-fatal errors through the router's onError callback.
458
+ requestContext._reportBackgroundError = (
459
+ error: unknown,
460
+ category: CacheErrorCategory,
461
+ ) => {
462
+ callOnError(error, "cache", {
463
+ request,
464
+ url,
465
+ metadata: { category },
466
+ });
467
+ };
468
+
301
469
  const ctxCreateDur = performance.now() - ctxCreateStart;
302
470
 
303
471
  // Accumulate handler-level timing for Server-Timing header
@@ -319,296 +487,132 @@ export function createRSCHandler<
319
487
  // - Server components during rendering
320
488
  // - Error boundaries
321
489
  // - Streaming
322
- return runWithRequestContext(requestContext, async () => {
323
- // Core handler logic (wrapped by middleware)
324
- const coreHandler = async (): Promise<Response> => {
325
- return coreRequestHandler(request, env, url, variables, nonce);
326
- };
327
-
328
- // Execute middleware chain if any, otherwise call core handler directly
329
- if (matchedMiddleware.length > 0) {
330
- return executeMiddleware(
331
- matchedMiddleware,
332
- request,
333
- env,
334
- variables,
335
- coreHandler,
336
- );
337
- }
338
-
339
- return coreHandler();
340
- });
341
- };
342
-
343
- // Core request handling logic (separated for middleware wrapping)
344
- async function coreRequestHandler(
345
- request: Request,
346
- env: TEnv,
347
- url: URL,
348
- variables: Record<string, any>,
349
- nonce: string | undefined,
350
- ): Promise<Response> {
351
- // First, check for route-level middleware
352
- const previewStart = performance.now();
353
- const preview = await router.previewMatch(request, env);
354
- const previewDur = performance.now() - previewStart;
355
- const handlerTiming: string[] = variables.__handlerTiming || [];
356
- handlerTiming.push(`handler-preview-match;dur=${previewDur.toFixed(2)}`);
357
- // Response route short-circuit: skip entire RSC pipeline
358
- if (preview?.responseType && preview.handler) {
359
- const isPartial = url.searchParams.has("_rsc_partial");
360
-
361
- // Partial requests (client-side navigation) to response routes
362
- // get X-RSC-Reload to trigger hard navigation in the browser
363
- if (isPartial) {
364
- const cleanUrl = new URL(url);
365
- cleanUrl.searchParams.delete("_rsc_partial");
366
- cleanUrl.searchParams.delete("_rsc_segments");
367
- cleanUrl.searchParams.delete("_rsc_v");
368
- cleanUrl.searchParams.delete("_rsc_stale");
369
- cleanUrl.searchParams.delete("_rsc_action");
370
- cleanUrl.searchParams.delete("_rsc_prev");
371
-
372
- return createResponseWithMergedHeaders(null, {
373
- status: 200,
374
- headers: {
375
- "X-RSC-Reload": cleanUrl.toString(),
376
- "content-type": "text/x-component;charset=utf-8",
377
- },
378
- });
379
- }
490
+ // Store basename on request context (scoped per-request via existing ALS)
491
+ requestContext._basename = router.basename;
492
+
493
+ // Resolved span tracing for this request (read at each traced phase).
494
+ requestContext._tracing = router.tracing;
495
+
496
+ // The "rango.request" span is opened inside the request context so the
497
+ // Cloudflare runner can read executionContext.tracing, and so every nested
498
+ // phase span (and the platform's automatic KV/D1/fetch spans) nests under
499
+ // it. Construction-bound: the span ends when the Response is built, never
500
+ // wrapping the streamed body. metric:false — handler:total is metered
501
+ // directly below (a grand total incl. the pre-context bootstrap timings).
502
+ // When tracing is off this is a direct pass-through.
503
+ return runWithRequestContext(requestContext, () =>
504
+ observePhase(PHASES.request, async (span) => {
505
+ span.setAttribute("http.method", request.method);
506
+ // The matched route template is not known until match() runs later, so
507
+ // emit the concrete path as url.path (low-level), NOT http.route — the
508
+ // latter is reserved for the low-cardinality template (OTel convention).
509
+ span.setAttribute("url.path", url.pathname);
510
+
511
+ // Core handler logic (wrapped by middleware)
512
+ const coreHandler = async (): Promise<Response> => {
513
+ return coreRequestHandler(request, env, url, variables, nonce);
514
+ };
380
515
 
381
- // Build lightweight context for response handler
382
- const bindings = (env as any)?.Bindings ?? env;
383
- const reqCtx = requireRequestContext();
384
- const responseHandlerCtx = {
385
- request,
386
- params: preview.params || {},
387
- env: bindings,
388
- searchParams: url.searchParams,
389
- url,
390
- pathname: url.pathname,
391
- href: (name: string, hrefParams?: Record<string, string>) => {
392
- if (name.startsWith("/")) {
393
- if (!hrefParams) return name;
394
- return name.replace(/:([^/]+)/g, (_, key) => {
395
- const value = hrefParams[key];
396
- if (value === undefined) throw new Error(`Missing param "${key}" for path "${name}"`);
397
- return encodeURIComponent(value);
398
- });
399
- }
400
- return name;
401
- },
402
- get: (key: string) => variables[key],
403
- header: (name: string, value: string) => reqCtx.header(name, value),
404
- setCookie: (name: string, value: string, options?: any) => reqCtx.setCookie(name, value, options),
405
- };
516
+ // Execute middleware chain if any, otherwise call core handler directly
517
+ let response: Response;
518
+ if (matchedMiddleware.length > 0) {
519
+ const mwResponse = await executeMiddleware(
520
+ matchedMiddleware,
521
+ request,
522
+ env,
523
+ variables,
524
+ coreHandler,
525
+ createReverseFunction(getRequiredRouteMap()),
526
+ );
406
527
 
407
- // Call handler directly, wrapped by route middleware if present
408
- const callHandler = async () => {
409
- // JSON response routes: wrap in { data } / { error } envelope
410
- if (preview.responseType === "json") {
411
- const errorCtx = { request, url, env };
412
- try {
413
- const result = await (preview.handler as Function)(responseHandlerCtx);
414
- if (result instanceof Response) {
415
- const mergedHeaders: Record<string, string> = {};
416
- result.headers.forEach((value, key) => {
417
- mergedHeaders[key] = value;
418
- });
419
- return createResponseWithMergedHeaders(result.body, {
420
- status: result.status,
421
- headers: mergedHeaders,
422
- });
423
- }
424
- return createResponseWithMergedHeaders(
425
- JSON.stringify({ data: result }),
426
- { status: 200, headers: { "content-type": "application/json;charset=utf-8" } },
427
- );
428
- } catch (error) {
429
- callOnError(error, "handler", errorCtx);
430
- const isDev = process.env.NODE_ENV !== "production";
431
- const status = error instanceof RouterError ? error.status : 500;
432
- return createResponseWithMergedHeaders(
433
- JSON.stringify({ error: createResponseErrorPayload(error, isDev) }),
434
- { status, headers: { "content-type": "application/json;charset=utf-8" } },
528
+ if (
529
+ url.searchParams.has("_rsc_partial") ||
530
+ url.searchParams.has("_rsc_action")
531
+ ) {
532
+ const intercepted = interceptRedirectForPartial(
533
+ mwResponse,
534
+ createRedirectFlightResponse,
435
535
  );
536
+ response = intercepted ?? finalizeResponse(mwResponse);
537
+ } else {
538
+ response = finalizeResponse(mwResponse);
436
539
  }
540
+ } else {
541
+ response = await coreHandler();
437
542
  }
438
543
 
439
- // Non-JSON response routes: catch errors and return plain Response
440
- const errorCtx = { request, url, env };
441
- try {
442
- const result = await (preview.handler as Function)(responseHandlerCtx);
443
-
444
- if (result instanceof Response) {
445
- // Handler returned a Response directly -- pass through
446
- const mergedHeaders: Record<string, string> = {};
447
- result.headers.forEach((value, key) => {
448
- mergedHeaders[key] = value;
449
- });
450
- return createResponseWithMergedHeaders(result.body, {
451
- status: result.status,
452
- headers: mergedHeaders,
453
- });
454
- }
455
-
456
- // Auto-wrap based on response type tag
457
- switch (preview.responseType) {
458
- case "text":
459
- return createResponseWithMergedHeaders(
460
- String(result),
461
- { status: 200, headers: { "content-type": "text/plain;charset=utf-8" } },
462
- );
463
- case "html":
464
- return createResponseWithMergedHeaders(
465
- String(result),
466
- { status: 200, headers: { "content-type": "text/html;charset=utf-8" } },
467
- );
468
- case "xml":
469
- return createResponseWithMergedHeaders(
470
- String(result),
471
- { status: 200, headers: { "content-type": "application/xml;charset=utf-8" } },
472
- );
473
- case "md":
474
- return createResponseWithMergedHeaders(
475
- String(result),
476
- { status: 200, headers: { "content-type": "text/markdown;charset=utf-8" } },
477
- );
478
- default:
479
- // image, stream, any -- must return Response
480
- throw new Error(
481
- `Response route handler for "${preview.responseType}" must return a Response object, got ${typeof result}`
482
- );
483
- }
484
- } catch (error) {
485
- callOnError(error, "handler", errorCtx);
486
- const isDev = process.env.NODE_ENV !== "production";
487
- const status = error instanceof RouterError ? error.status : 500;
488
- const message = error instanceof RouterError
489
- ? error.message
490
- : isDev && error instanceof Error
491
- ? error.message
492
- : "Internal Server Error";
493
- return createResponseWithMergedHeaders(message, {
494
- status,
495
- headers: { "content-type": "text/plain;charset=utf-8" },
496
- });
544
+ // Finalize metrics after all middleware (including post-next work)
545
+ // has completed so :post spans are captured in the timeline.
546
+ // Handler timing parts are always emitted (even without debug metrics)
547
+ // so non-debug requests still get bootstrap Server-Timing entries.
548
+ const handlerTimingArr: string[] = variables.__handlerTiming || [];
549
+ // Preserve any existing Server-Timing set by response routes or middleware
550
+ const existingTiming = response.headers.get("Server-Timing");
551
+ const timingParts = existingTiming
552
+ ? [existingTiming, ...handlerTimingArr]
553
+ : [...handlerTimingArr];
554
+
555
+ const metricsStore = requestContext._metricsStore;
556
+ if (metricsStore) {
557
+ // When the store was created at handler start (earlyMetricsStore),
558
+ // handler:total covers the full request. When ctx.debugPerformance()
559
+ // created the store mid-request, use its requestStart to avoid a
560
+ // negative startTime offset.
561
+ const totalStart = earlyMetricsStore
562
+ ? handlerStart
563
+ : metricsStore.requestStart;
564
+ appendMetric(
565
+ metricsStore,
566
+ "handler:total",
567
+ totalStart,
568
+ performance.now() - totalStart,
569
+ );
570
+ const metricsTiming = buildMetricsTiming(
571
+ request.method,
572
+ url.pathname,
573
+ metricsStore,
574
+ );
575
+ if (metricsTiming) timingParts.push(metricsTiming);
497
576
  }
498
- };
499
577
 
500
- // Wrap callHandler to append Vary: Accept on content-negotiated responses
501
- const callHandlerWithVary = async () => {
502
- const response = await callHandler();
503
- if (preview.negotiated) {
504
- response.headers.append("Vary", "Accept");
578
+ const fullTiming = timingParts.join(", ");
579
+ if (fullTiming && !isWebSocketUpgradeResponse(response)) {
580
+ response.headers.set("Server-Timing", fullTiming);
505
581
  }
506
- return response;
507
- };
508
-
509
- if (preview.routeMiddleware && preview.routeMiddleware.length > 0) {
510
- const middlewareEntries = preview.routeMiddleware.map((mw) => ({
511
- entry: {
512
- pattern: null,
513
- regex: null,
514
- paramNames: [],
515
- handler: mw.handler,
516
- mountPrefix: null,
517
- },
518
- params: mw.params,
519
- }));
520
- return executeMiddleware(middlewareEntries, request, env, variables, callHandlerWithVary);
521
- }
522
-
523
- return callHandlerWithVary();
524
- }
525
-
526
- // Wrap RSC handler to append Vary: Accept on content-negotiated routes
527
- const rscHandler = async () => {
528
- const response = await coreRequestHandlerInner(request, env, url, variables, nonce);
529
- if (preview?.negotiated) {
530
- response.headers.append("Vary", "Accept");
531
- }
532
- return response;
533
- };
534
582
 
535
- if (preview?.routeMiddleware && preview.routeMiddleware.length > 0) {
536
- // Convert route middleware to app middleware format for execution
537
- const middlewareEntries = preview.routeMiddleware.map((mw) => ({
538
- entry: {
539
- pattern: null,
540
- regex: null,
541
- paramNames: [],
542
- handler: mw.handler,
543
- mountPrefix: null,
544
- },
545
- params: mw.params,
546
- }));
547
-
548
- // Execute route middleware wrapping the actual request handling
549
- return executeMiddleware(middlewareEntries, request, env, variables, rscHandler);
550
- }
551
-
552
- // No route middleware, proceed directly
553
- return rscHandler();
554
- }
583
+ // Single open-redirect chokepoint: every response (PE, full-page,
584
+ // middleware short-circuit, response-route) funnels through here, so
585
+ // guarding browser-followed (3xx) redirects once covers them all and any
586
+ // future redirect exit. Soft SPA/Flight redirects are 200/204 and pass
587
+ // through untouched (validated client-side instead).
588
+ return guardOutgoingRedirect(response, url.origin, router.basename);
589
+ }),
590
+ );
591
+ };
555
592
 
556
- // Inner request handler (actual RSC logic, wrapped by route middleware if any)
557
- async function coreRequestHandlerInner(
593
+ // Core request handling logic (separated for middleware wrapping).
594
+ // Uses the classify → execute model: classifyRequest produces a RequestPlan,
595
+ // then execution dispatches on the plan mode.
596
+ async function coreRequestHandler(
558
597
  request: Request,
559
598
  env: TEnv,
560
599
  url: URL,
561
600
  variables: Record<string, any>,
562
601
  nonce: string | undefined,
563
602
  ): Promise<Response> {
564
- const isPartial = url.searchParams.has("_rsc_partial");
565
- const isAction =
566
- request.headers.has("rsc-action") || url.searchParams.has("_rsc_action");
567
- const actionId =
568
- request.headers.get("rsc-action") || url.searchParams.get("_rsc_action");
569
-
570
- // Version mismatch detection - client may have stale code after HMR/deployment
571
- // If versions don't match, tell the client to reload
572
- const clientVersion = url.searchParams.get("_rsc_v");
573
- if (version && clientVersion && clientVersion !== version) {
574
- console.log(
575
- `[RSC] Version mismatch: client=${clientVersion}, server=${version}. Forcing reload.`,
576
- );
603
+ const handlerTiming: string[] = variables.__handlerTiming || [];
577
604
 
578
- // Clean URL by removing RSC params
579
- const cleanUrl = new URL(url);
580
- cleanUrl.searchParams.delete("_rsc_partial");
581
- cleanUrl.searchParams.delete("_rsc_segments");
582
- cleanUrl.searchParams.delete("_rsc_v");
583
- cleanUrl.searchParams.delete("_rsc_stale");
584
- cleanUrl.searchParams.delete("_rsc_action");
585
- cleanUrl.searchParams.delete("_rsc_prev");
586
-
587
- // For actions, reload current page (referer)
588
- // For navigation, load the target URL
589
- const reloadUrl = isAction
590
- ? request.headers.get("referer") || cleanUrl.toString()
591
- : cleanUrl.toString();
592
-
593
- // Return special response that tells client to reload
594
- return createResponseWithMergedHeaders(null, {
595
- status: 200,
596
- headers: {
597
- "X-RSC-Reload": reloadUrl,
598
- "content-type": "text/x-component;charset=utf-8",
599
- },
600
- });
601
- }
602
- // Debug manifest endpoint: ?__debug_manifest on any route.
603
- // Always available in dev, requires allowDebugManifest option in production.
605
+ // Debug manifest endpoint: handled before classification since it
606
+ // doesn't need a route match and needs trie access from the closure.
604
607
  const isDev = process.env.NODE_ENV !== "production";
605
608
  if (
606
609
  url.searchParams.has("__debug_manifest") &&
607
610
  (isDev || router.allowDebugManifest)
608
611
  ) {
609
612
  const trie = getRouterTrie(router.id) ?? getRouteTrie();
610
- const routeManifest = getRouterManifest(router.id) ?? getGlobalRouteMap();
611
- const { extractAncestryFromTrie } = await import("../build/route-trie.js");
613
+ const routeManifest = getRequiredRouteMap();
614
+ const { extractAncestryFromTrie } =
615
+ await import("../build/route-trie.js");
612
616
  return new Response(
613
617
  JSON.stringify(
614
618
  {
@@ -627,851 +631,567 @@ export function createRSCHandler<
627
631
  );
628
632
  }
629
633
 
630
- // Get handle store from request context (created at start of request)
631
- const handleStore = requireRequestContext()._handleStore;
632
-
634
+ // ---- 1. Classify ----
635
+ // classifyRequest may throw RouteNotFoundError for unknown routes.
636
+ // In that case, fall through to a full-render plan so the pipeline
637
+ // can render the 404 page via the existing error handling path.
638
+ const classifyStart = performance.now();
639
+ let plan: RequestPlan<TEnv>;
633
640
  try {
634
- // ============================================================================
635
- // PROGRESSIVE ENHANCEMENT: No-JS Form Submissions
636
- // ============================================================================
637
- const progressiveResult = await handleProgressiveEnhancement(
638
- request,
639
- env,
640
- url,
641
- isAction,
642
- handleStore,
643
- nonce,
644
- );
645
- if (progressiveResult) {
646
- return progressiveResult;
641
+ plan = await classifyRequest<TEnv>(request, url, {
642
+ findMatch: router.findMatch,
643
+ routerVersion: version,
644
+ routerId: router.id,
645
+ });
646
+ } catch (error) {
647
+ if (isRouteNotFoundError(error)) {
648
+ // Let the render path handle 404 — match()/matchPartial() will
649
+ // re-throw RouteNotFoundError and the catch block in
650
+ // executeRenderWithMiddleware renders the not-found page.
651
+ plan = {
652
+ mode: "full-render",
653
+ route: {
654
+ matched: null as any,
655
+ manifestEntry: null as any,
656
+ entries: [],
657
+ routeKey: "",
658
+ localRouteName: "",
659
+ params: {},
660
+ routeMiddleware: [],
661
+ cacheScope: null,
662
+ isPassthrough: false,
663
+ },
664
+ negotiated: false,
665
+ };
666
+ } else {
667
+ throw error;
647
668
  }
648
-
649
- // ============================================================================
650
- // SERVER ACTION EXECUTION (JavaScript-enabled client)
651
- // ============================================================================
652
- if (isAction && actionId) {
653
- return handleServerAction(request, env, url, actionId, handleStore);
669
+ }
670
+ const classifyDur = performance.now() - classifyStart;
671
+ handlerTiming.push(`handler-classify;dur=${classifyDur.toFixed(2)}`);
672
+
673
+ // ---- 2. Terminal plans (no execution needed) ----
674
+ if (plan.mode === "redirect") {
675
+ // Redirects are handled by the pipeline (match/matchPartial),
676
+ // but for partial requests we short-circuit with a Flight redirect.
677
+ if (url.searchParams.has("_rsc_partial")) {
678
+ return createRedirectFlightResponse(plan.redirectUrl);
654
679
  }
680
+ // Full requests: let the pipeline handle the redirect via match()
681
+ // which returns { redirect: url }. Fall through to full-render.
682
+ }
655
683
 
656
- // ============================================================================
657
- // LOADER FETCH EXECUTION (data fetching with RSC serialization)
658
- // ============================================================================
659
- const isLoaderRequest = url.searchParams.has("_rsc_loader");
660
- if (isLoaderRequest) {
661
- return handleLoaderFetch(request, env, url, variables);
662
- }
684
+ if (plan.mode === "version-mismatch") {
685
+ console.log(
686
+ `[RSC] Version mismatch: client=${url.searchParams.get("_rsc_v")}, server=${version}. Forcing reload.`,
687
+ );
688
+ return createReloadResponse(plan.reloadUrl);
689
+ }
690
+
691
+ if (plan.mode === "app-switch") {
692
+ // Cross-app SPA navigation crossed a host-router app boundary. Force a
693
+ // real document navigation so the target app's document is re-established
694
+ // (stylesheets, theme, warmup, prefetch-TTL). See request-classification.
695
+ return createReloadResponse(plan.reloadUrl);
696
+ }
663
697
 
664
- // ============================================================================
665
- // REGULAR RSC RENDERING (Navigation)
666
- // ============================================================================
667
- // Note: Must use "return await" for try/catch to catch async rejections
668
- return await handleRscRendering(
698
+ // ---- 3. Origin guard (gate for action/loader/PE modes) ----
699
+ const originPhase = ORIGIN_CHECK_PHASE_BY_MODE[plan.mode];
700
+ if (originPhase) {
701
+ const originResult = await checkRequestOrigin(
669
702
  request,
670
- env,
671
703
  url,
672
- isPartial,
673
- handleStore,
674
- nonce,
704
+ router.originCheck,
705
+ env,
706
+ router.id,
707
+ originPhase,
675
708
  );
676
- } catch (error) {
677
- // Check if middleware/handler returned Response
678
- if (error instanceof Response) {
679
- // During partial (client-side navigation), a 200 Response from a handler
680
- // means the route serves raw content (JSON, text, etc.), not JSX.
681
- // Signal the browser to hard-navigate so it renders the raw response.
682
- // Only for 200 — redirects (3xx) work already because the browser follows
683
- // them automatically to a URL that serves Flight data.
684
- if (isPartial && error.status === 200) {
685
- console.warn(
686
- `[RSC] Route handler at ${url.pathname} returned a Response during client-side navigation. ` +
687
- `Falling back to hard navigation. Use data-external on the <Link> to avoid the extra round-trip.`,
688
- );
689
- const cleanUrl = new URL(url);
690
- cleanUrl.searchParams.delete("_rsc_partial");
691
- cleanUrl.searchParams.delete("_rsc_segments");
692
- cleanUrl.searchParams.delete("_rsc_v");
693
- cleanUrl.searchParams.delete("_rsc_stale");
694
- cleanUrl.searchParams.delete("_rsc_action");
695
- cleanUrl.searchParams.delete("_rsc_prev");
696
- return createResponseWithMergedHeaders(null, {
697
- status: 200,
698
- headers: {
699
- "X-RSC-Reload": cleanUrl.toString(),
700
- "content-type": "text/x-component;charset=utf-8",
701
- },
702
- });
703
- }
704
- return error;
705
- }
709
+ if (originResult) {
710
+ const originError = new Error(
711
+ `Origin check rejected: ${request.headers.get("origin") ?? "none"} vs ${request.headers.get("host") ?? "none"}`,
712
+ );
713
+ originError.name = "OriginCheckError";
706
714
 
707
- // Render 404 page for unmatched routes
708
- // Check both instanceof and error.name for cross-bundle compatibility
709
- const isRouteNotFound =
710
- error instanceof RouteNotFoundError ||
711
- (error instanceof Error && error.name === "RouteNotFoundError");
712
- if (isRouteNotFound) {
713
- callOnError(error, "routing", {
715
+ callOnError(originError, "origin", {
714
716
  request,
715
717
  url,
716
718
  env,
717
- handledByBoundary: true, // Handled by notFound component
718
- });
719
-
720
- // Get notFound component from router options or use default
721
- const notFoundOption = router.notFound;
722
- const notFoundComponent =
723
- typeof notFoundOption === "function"
724
- ? notFoundOption({ pathname: url.pathname })
725
- : (notFoundOption ?? createElement("h1", null, "Not Found"));
726
-
727
- // Create a simple segment for the 404 page
728
- const notFoundSegment = {
729
- id: "notFound",
730
- namespace: "notFound",
731
- type: "route" as const,
732
- index: 0,
733
- component: notFoundComponent,
734
- params: {},
735
- };
736
-
737
- // Render with rootLayout to maintain app shell
738
- const root = await renderSegments([notFoundSegment], {
739
- rootLayout: router.rootLayout,
740
- // No routeName for not-found routes
741
- });
742
-
743
- const payload: RscPayload = {
744
- root,
719
+ handledByBoundary: false,
745
720
  metadata: {
746
- pathname: url.pathname,
747
- segments: [notFoundSegment],
748
- matched: [],
749
- diff: [],
750
- isPartial: false,
751
- handles: handleStore.stream(),
752
- version,
753
- themeConfig: router.themeConfig,
754
- warmupEnabled: router.warmupEnabled,
755
- initialTheme: requireRequestContext().theme,
756
- // No routeName for not-found routes
721
+ phase: originPhase,
722
+ origin: request.headers.get("origin"),
723
+ host: request.headers.get("host"),
757
724
  },
758
- };
759
-
760
- const rscStream = renderToReadableStream(payload);
761
-
762
- // Determine if this is an RSC request or HTML request
763
- const isRscRequest =
764
- (!request.headers.get("accept")?.includes("text/html") &&
765
- !url.searchParams.has("__html")) ||
766
- url.searchParams.has("__rsc");
767
-
768
- if (isRscRequest) {
769
- return createResponseWithMergedHeaders(rscStream, {
770
- status: 404,
771
- headers: { "content-type": "text/x-component;charset=utf-8" },
772
- });
773
- }
774
-
775
- // Delegate to SSR for HTML response
776
- const ssrModule = await loadSSRModule();
777
- const htmlStream = await ssrModule.renderHTML(rscStream, { nonce });
725
+ });
778
726
 
779
- return createResponseWithMergedHeaders(htmlStream, {
780
- status: 404,
781
- headers: { "content-type": "text/html;charset=utf-8" },
727
+ observeEvent({
728
+ type: "request.origin-rejected",
729
+ timestamp: performance.now(),
730
+ method: request.method,
731
+ pathname: url.pathname,
732
+ phase: originPhase,
733
+ origin: request.headers.get("origin"),
734
+ host: request.headers.get("host"),
782
735
  });
783
- }
784
736
 
785
- // Report unhandled errors
786
- callOnError(error, "routing", {
787
- request,
788
- url,
789
- env,
790
- handledByBoundary: false,
791
- });
792
- console.error(`[RSC] Error:`, error);
793
- throw error;
737
+ return originResult;
738
+ }
794
739
  }
740
+
741
+ // ---- 4. Execute ----
742
+ return executeRequest(
743
+ plan as ExecutableRequestPlan<TEnv>,
744
+ request,
745
+ env,
746
+ url,
747
+ variables,
748
+ nonce,
749
+ );
795
750
  }
796
751
 
797
- // ============================================================================
798
- // PROGRESSIVE ENHANCEMENT HANDLER
799
- // When JavaScript is disabled, React renders forms with hidden fields
800
- // ($ACTION_REF_*, $ACTION_KEY) containing the action reference.
801
- // We detect these and return HTML instead of RSC stream.
802
- // ============================================================================
803
- async function handleProgressiveEnhancement(
752
+ // Execute a classified request plan. Dispatches to the appropriate handler
753
+ // based on plan.mode. Lives in the createRSCHandler closure for access to
754
+ // handlerCtx, router, callOnError, etc.
755
+ // Only receives executable plans (version-mismatch is handled above).
756
+ async function executeRequest(
757
+ plan: ExecutableRequestPlan<TEnv>,
804
758
  request: Request,
805
759
  env: TEnv,
806
760
  url: URL,
807
- isAction: boolean,
808
- handleStore: ReturnType<typeof requireRequestContext>["_handleStore"],
761
+ variables: Record<string, any>,
809
762
  nonce: string | undefined,
810
- ): Promise<Response | null> {
811
- const contentType = request.headers.get("content-type") || "";
812
- const isFormSubmission =
813
- contentType.includes("multipart/form-data") ||
814
- contentType.includes("application/x-www-form-urlencoded");
815
-
816
- if (request.method !== "POST" || isAction || !isFormSubmission) {
817
- return null;
818
- }
819
-
820
- // Clone the request to read FormData without consuming it
821
- const formData = await request.clone().formData();
822
-
823
- // Look for React's progressive enhancement hidden fields
824
- let isDirectAction = false;
825
- let isUseActionState = false;
826
- let directActionId: string | null = null;
763
+ ): Promise<Response> {
764
+ // Common setup
765
+ const handleStore = requireRequestContext()._handleStore;
827
766
 
828
- formData.forEach((_value, key) => {
829
- if (key.startsWith("$ACTION_ID_")) {
830
- isDirectAction = true;
831
- directActionId = key.slice("$ACTION_ID_".length);
832
- } else if (key.startsWith("$ACTION_REF_")) {
833
- isUseActionState = true;
834
- }
835
- });
767
+ // Wire up error reporting for late streaming-handle failures
768
+ handleStore.onError = (error: Error) => {
769
+ const reqCtx = requireRequestContext();
770
+ callOnError(error, "handler", {
771
+ request,
772
+ url,
773
+ routeKey: reqCtx._routeName,
774
+ params: reqCtx.params as Record<string, string>,
775
+ handledByBoundary: true,
776
+ });
777
+ observeEvent({
778
+ type: "handler.error",
779
+ timestamp: performance.now(),
780
+ error,
781
+ handledByBoundary: true,
782
+ pathname: url.pathname,
783
+ routeKey: reqCtx._routeName,
784
+ params: reqCtx.params as Record<string, string>,
785
+ });
786
+ };
836
787
 
837
- if (!isDirectAction && !isUseActionState) {
838
- return null;
788
+ // Set route params early so all execution paths can access ctx.params.
789
+ // Also store the classified snapshot so match/matchPartial can reuse it
790
+ // instead of calling resolveRoute again.
791
+ if (plan.mode !== "redirect") {
792
+ setRequestContextParams(plan.route.params, plan.route.routeKey);
793
+ requireRequestContext()._classifiedRoute = plan.route;
839
794
  }
840
795
 
841
- // Execute action and return HTML
842
- let actionResult: unknown = undefined;
843
- let reactFormState: ReactFormState | null = null;
844
-
845
- if (isUseActionState) {
846
- try {
847
- const boundAction = await decodeAction(formData);
848
- actionResult = await boundAction();
849
- } catch (error) {
850
- callOnError(error, "action", {
796
+ const routeReverse = createReverseFunction(getRequiredRouteMap());
797
+
798
+ // ---- Response route: skip entire RSC pipeline ----
799
+ if (plan.mode === "response") {
800
+ // Build ResponseRouteMatch from plan fields. handleResponseRoute
801
+ // expects a flat object with params at the top level.
802
+ const responseMatch: ResponseRouteMatch = {
803
+ responseType: plan.responseType,
804
+ handler: plan.handler,
805
+ params: plan.route.params,
806
+ negotiated: plan.negotiated,
807
+ manifestEntry: plan.manifestEntry,
808
+ routeMiddleware: plan.routeMiddleware,
809
+ };
810
+ const responseOutcome = await withTimeout(
811
+ handleResponseRoute(
812
+ handlerCtx,
813
+ responseMatch,
851
814
  request,
852
- url,
853
815
  env,
854
- handledByBoundary: false,
855
- });
856
- console.error("[RSC] Progressive enhancement action error:", error);
857
- }
858
- } else if (isDirectAction && directActionId) {
859
- const temporaryReferences = createTemporaryReferenceSet();
860
-
861
- let args: unknown[] = [];
862
- try {
863
- args = await decodeReply(formData, { temporaryReferences });
864
- } catch {
865
- args = [formData];
866
- }
867
-
868
- try {
869
- const loadedAction = await loadServerAction(directActionId);
870
- actionResult = await loadedAction.apply(null, args);
871
- } catch (error) {
872
- callOnError(error, "action", {
873
- request,
874
816
  url,
817
+ variables,
818
+ ),
819
+ router.timeouts.renderStartMs,
820
+ "render-start",
821
+ );
822
+ if (responseOutcome.timedOut) {
823
+ return handleTimeoutResponse(
824
+ request,
875
825
  env,
876
- actionId: directActionId,
877
- handledByBoundary: false,
878
- });
879
- console.error("[RSC] Progressive enhancement action error:", error);
826
+ url,
827
+ "render-start",
828
+ responseOutcome.durationMs,
829
+ plan.route.routeKey,
830
+ );
880
831
  }
832
+ const response = responseOutcome.result;
833
+ if (plan.negotiated && !isWebSocketUpgradeResponse(response)) {
834
+ response.headers.append("Vary", "Accept");
835
+ }
836
+ return response;
881
837
  }
882
838
 
883
- // Decode form state for useActionState progressive enhancement
884
- try {
885
- reactFormState = await decodeFormState(actionResult, formData);
886
- } catch (error) {
887
- callOnError(error, "action", {
839
+ // SSR setup: kick off in parallel for modes that need HTML rendering.
840
+ // Placed after response-route short-circuit so response/mime routes
841
+ // never pay for SSR work.
842
+ if (plan.mode !== "loader" && mayNeedSSR(request, url)) {
843
+ variables[SSR_SETUP_VAR] = startSSRSetup(
844
+ handlerCtx,
888
845
  request,
889
- url,
890
846
  env,
891
- handledByBoundary: false,
892
- });
893
- console.error("[RSC] Failed to decode form state:", error);
894
- }
895
-
896
- // Re-render the page and return HTML
897
- const renderRequest = new Request(url.toString(), {
898
- method: "GET",
899
- headers: new Headers({ accept: "text/html" }),
900
- });
901
-
902
- const match = await router.match(renderRequest, env);
903
-
904
- if (match.redirect) {
905
- return new Response(null, {
906
- status: 308,
907
- headers: { Location: match.redirect },
908
- });
847
+ url,
848
+ router.debugPerformance
849
+ ? () => requireRequestContext()._metricsStore
850
+ : undefined,
851
+ );
909
852
  }
910
853
 
911
- const root = renderSegments(match.segments, {
912
- rootLayout: router.rootLayout,
913
- });
914
-
915
- const payload: RscPayload = {
916
- root,
917
- metadata: {
918
- pathname: url.pathname,
919
- segments: match.segments,
920
- matched: match.matched,
921
- diff: match.diff,
922
- isPartial: false,
923
- rootLayout: router.rootLayout,
924
- handles: handleStore.stream(),
925
- version,
926
- themeConfig: router.themeConfig,
927
- warmupEnabled: router.warmupEnabled,
928
- initialTheme: requireRequestContext().theme,
929
- },
930
- formState: actionResult,
931
- };
932
-
933
- const rscStream = renderToReadableStream<RscPayload>(payload);
934
- const ssrModule = await loadSSRModule();
935
- const htmlStream = await ssrModule.renderHTML(rscStream, {
936
- formState: reactFormState,
937
- nonce,
938
- });
939
-
940
- return new Response(htmlStream, {
941
- headers: { "content-type": "text/html;charset=utf-8" },
942
- });
943
- }
944
-
945
- // ============================================================================
946
- // SERVER ACTION HANDLER
947
- // ============================================================================
948
- async function handleServerAction(
949
- request: Request,
950
- env: TEnv,
951
- url: URL,
952
- actionId: string,
953
- handleStore: ReturnType<typeof requireRequestContext>["_handleStore"],
954
- ): Promise<Response> {
955
- const temporaryReferences = createTemporaryReferenceSet();
956
-
957
- // Decode action arguments from request body
958
- const contentType = request.headers.get("content-type") || "";
959
- let args: unknown[] = [];
960
- let actionFormData: FormData | undefined;
961
-
962
- try {
963
- const body = contentType.includes("multipart/form-data")
964
- ? await request.formData()
965
- : await request.text();
966
-
967
- if (body instanceof FormData) {
968
- actionFormData = body;
969
- }
970
-
971
- if (hasBodyContent(body)) {
972
- args = await decodeReply(body, { temporaryReferences });
973
- }
974
- } catch (error) {
975
- callOnError(error, "action", {
854
+ // ---- Loader fetch ----
855
+ if (plan.mode === "loader") {
856
+ return handleLoaderFetch(
857
+ handlerCtx,
976
858
  request,
977
- url,
978
859
  env,
979
- actionId,
980
- handledByBoundary: false,
981
- });
982
- throw new Error(`Failed to decode action arguments: ${error}`, {
983
- cause: error,
984
- });
860
+ url,
861
+ variables,
862
+ plan.route.params,
863
+ );
985
864
  }
986
865
 
987
- // Execute the server action
988
- let returnValue: { ok: boolean; data: unknown };
989
- let actionStatus = 200;
990
- let loadedAction: Function | undefined;
991
-
992
- try {
993
- loadedAction = await loadServerAction(actionId);
994
- const data = await loadedAction!.apply(null, args);
995
- returnValue = { ok: true, data };
996
- } catch (error) {
997
- returnValue = { ok: false, data: error };
998
- actionStatus = 500;
999
-
1000
- // Try to render error boundary
1001
- const errorResult = await router.matchError(request, env, error, "route");
1002
-
1003
- // Report the action error (handledByBoundary indicates if error boundary will render)
1004
- callOnError(error, "action", {
866
+ // ---- Progressive enhancement ----
867
+ if (plan.mode === "pe-render") {
868
+ const peResult = await handleProgressiveEnhancement(
869
+ handlerCtx,
1005
870
  request,
1006
- url,
1007
871
  env,
1008
- actionId,
1009
- handledByBoundary: !!errorResult,
1010
- });
1011
-
1012
- if (errorResult) {
1013
- setRequestContextParams(errorResult.params);
1014
-
1015
- const payload: RscPayload = {
1016
- root: null,
1017
- metadata: {
1018
- pathname: url.pathname,
1019
- segments: errorResult.segments,
1020
- isPartial: true,
1021
- matched: errorResult.matched,
1022
- diff: errorResult.diff,
1023
- isError: true,
1024
- handles: handleStore.stream(),
1025
- version,
1026
- },
1027
- returnValue,
1028
- };
1029
-
1030
- const rscStream = renderToReadableStream<RscPayload>(payload, {
1031
- temporaryReferences,
1032
- });
1033
-
1034
- return createResponseWithMergedHeaders(rscStream, {
1035
- status: actionStatus,
1036
- headers: { "content-type": "text/x-component;charset=utf-8" },
1037
- });
1038
- }
872
+ url,
873
+ false, // isAction = false for PE
874
+ handleStore,
875
+ nonce,
876
+ {
877
+ routeMiddleware: plan.route.routeMiddleware,
878
+ variables,
879
+ routeReverse,
880
+ },
881
+ );
882
+ if (peResult) return peResult;
883
+ // PE handler returned null (not a PE form) — fall through to render
1039
884
  }
1040
885
 
1041
- // Revalidate after action
1042
- const resolvedActionId =
1043
- (loadedAction as { $id?: string; $$id?: string } | undefined)?.$id ??
1044
- (loadedAction as { $$id?: string } | undefined)?.$$id ??
1045
- actionId;
1046
- const actionContext = {
1047
- actionId: resolvedActionId,
1048
- actionUrl: new URL(request.url),
1049
- actionResult: returnValue.data,
1050
- formData: actionFormData,
1051
- };
1052
-
1053
- const matchResult = await router.matchPartial(request, env, actionContext);
1054
-
1055
- if (!matchResult) {
1056
- // Fall back to full render
1057
- const fullMatch = await router.match(request, env);
1058
- setRequestContextParams(fullMatch.params);
1059
-
1060
- if (fullMatch.redirect) {
1061
- return createResponseWithMergedHeaders(null, {
1062
- status: 308,
1063
- headers: { Location: fullMatch.redirect },
886
+ // ---- Action: execute action, then revalidate wrapped in route middleware ----
887
+ if (plan.mode === "action") {
888
+ let actionContinuation: ActionContinuation | undefined;
889
+ try {
890
+ // Instrument the action execution as its own phase (action:<actionId> +
891
+ // rango.action), so a POST shows the mutation time AND which action ran,
892
+ // not just the downstream revalidation render. The action's own
893
+ // loaders/fetches nest under rango.action.
894
+ const actionOutcome = await withTimeout(
895
+ observePhase(PHASES.action(plan.actionId), () =>
896
+ executeServerAction(
897
+ handlerCtx,
898
+ request,
899
+ env,
900
+ url,
901
+ plan.actionId,
902
+ handleStore,
903
+ ),
904
+ ),
905
+ router.timeouts.actionMs,
906
+ "action",
907
+ );
908
+ if (actionOutcome.timedOut) {
909
+ return handleTimeoutResponse(
910
+ request,
911
+ env,
912
+ url,
913
+ "action",
914
+ actionOutcome.durationMs,
915
+ plan.route.routeKey,
916
+ plan.actionId,
917
+ );
918
+ }
919
+ const result = actionOutcome.result;
920
+ // Response means redirect or error boundary — done.
921
+ if (result instanceof Response) return result;
922
+ actionContinuation = result;
923
+ } catch (error) {
924
+ callOnError(error, "action", {
925
+ request,
926
+ url,
927
+ env,
928
+ actionId: plan.actionId,
929
+ handledByBoundary: false,
1064
930
  });
931
+ console.error(`[RSC] Action error:`, error);
932
+ throw error;
1065
933
  }
1066
934
 
1067
- const renderStart = performance.now();
1068
- const root = renderSegments(fullMatch.segments, {
1069
- rootLayout: router.rootLayout,
1070
- isAction: true,
1071
- });
1072
- const renderDuration = performance.now() - renderStart;
1073
- const serverTiming = fullMatch.serverTiming
1074
- ? `${fullMatch.serverTiming}, rendering;dur=${renderDuration.toFixed(2)}`
1075
- : `rendering;dur=${renderDuration.toFixed(2)}`;
1076
-
1077
- const payload: RscPayload = {
1078
- root,
1079
- metadata: {
1080
- pathname: url.pathname,
1081
- segments: fullMatch.segments,
1082
- matched: fullMatch.matched,
1083
- diff: fullMatch.diff,
1084
- handles: handleStore.stream(),
1085
- version,
1086
- },
1087
- returnValue,
1088
- };
1089
-
1090
- const rscStream = renderToReadableStream<RscPayload>(payload, {
1091
- temporaryReferences,
1092
- });
1093
-
1094
- const headers: Record<string, string> = {
1095
- "content-type": "text/x-component;charset=utf-8",
1096
- };
1097
- if (serverTiming) {
1098
- headers["Server-Timing"] = serverTiming;
1099
- }
1100
-
1101
- return createResponseWithMergedHeaders(rscStream, {
1102
- status: actionStatus,
1103
- headers,
1104
- });
1105
- }
1106
-
1107
- // Return updated segments
1108
- setRequestContextParams(matchResult.params);
1109
-
1110
- const renderStart = performance.now();
1111
-
1112
- const renderDuration = performance.now() - renderStart;
1113
- const serverTiming = matchResult.serverTiming
1114
- ? `${matchResult.serverTiming}, rendering;dur=${renderDuration.toFixed(2)}`
1115
- : `rendering;dur=${renderDuration.toFixed(2)}`;
1116
-
1117
- const payload: RscPayload = {
1118
- root: null,
1119
- metadata: {
1120
- pathname: url.pathname,
1121
- segments: matchResult.segments,
1122
- isPartial: true,
1123
- matched: matchResult.matched,
1124
- diff: matchResult.diff,
1125
- slots: matchResult.slots,
1126
- handles: handleStore.stream(),
1127
- version,
1128
- },
1129
- returnValue,
1130
- };
1131
-
1132
- const rscStream = renderToReadableStream<RscPayload>(payload, {
1133
- temporaryReferences,
1134
- });
1135
-
1136
- const actionHeaders: Record<string, string> = {
1137
- "content-type": "text/x-component;charset=utf-8",
1138
- };
1139
- if (serverTiming) {
1140
- actionHeaders["Server-Timing"] = serverTiming;
935
+ // Revalidation render wrapped in route middleware.
936
+ // Actions from client-side navigation include _rsc_partial — preserve
937
+ // the partial flag so the revalidation returns a Flight stream, not HTML.
938
+ // App-switch is already excluded by classifyRequest (would be full-render).
939
+ const isPartialAction = url.searchParams.has("_rsc_partial");
940
+ return executeRenderWithMiddleware(
941
+ plan.route.routeMiddleware,
942
+ plan.negotiated,
943
+ plan.route.routeKey,
944
+ routeReverse,
945
+ request,
946
+ env,
947
+ url,
948
+ variables,
949
+ nonce,
950
+ handleStore,
951
+ isPartialAction,
952
+ actionContinuation,
953
+ );
1141
954
  }
1142
955
 
1143
- return createResponseWithMergedHeaders(rscStream, {
1144
- status: actionStatus,
1145
- headers: actionHeaders,
1146
- });
956
+ // Full render, partial render, fallen-through PE, and full-page redirect all
957
+ // render through the same middleware-wrapped path. Only full/partial-render
958
+ // carry negotiation + the partial flag; pe/redirect render plainly.
959
+ const isPartial = plan.mode === "partial-render";
960
+ const negotiated =
961
+ plan.mode === "full-render" || plan.mode === "partial-render"
962
+ ? plan.negotiated
963
+ : false;
964
+ return executeRenderWithMiddleware(
965
+ plan.route.routeMiddleware,
966
+ negotiated,
967
+ plan.route.routeKey,
968
+ routeReverse,
969
+ request,
970
+ env,
971
+ url,
972
+ variables,
973
+ nonce,
974
+ handleStore,
975
+ isPartial,
976
+ );
1147
977
  }
1148
978
 
1149
- // ============================================================================
1150
- // LOADER FETCH HANDLER
1151
- // Supports GET (params in query string) and POST/PUT/PATCH/DELETE (JSON body)
1152
- // ============================================================================
1153
- async function handleLoaderFetch(
979
+ // Shared render execution: wraps handleRscRendering (or revalidateAfterAction)
980
+ // in route middleware and timeout handling. Consolidates the pattern used by
981
+ // action-revalidate, full-render, and partial-render modes.
982
+ async function executeRenderWithMiddleware(
983
+ routeMiddleware: import("../router/middleware-types.js").CollectedMiddleware[],
984
+ negotiated: boolean,
985
+ routeKey: string,
986
+ routeReverse: ReturnType<typeof createReverseFunction>,
1154
987
  request: Request,
1155
988
  env: TEnv,
1156
989
  url: URL,
1157
990
  variables: Record<string, any>,
991
+ nonce: string | undefined,
992
+ handleStore: ReturnType<typeof requireRequestContext>["_handleStore"],
993
+ isPartial: boolean,
994
+ actionContinuation?: ActionContinuation,
1158
995
  ): Promise<Response> {
1159
- const loaderId = url.searchParams.get("_rsc_loader");
996
+ const renderHandler = async (): Promise<Response> => {
997
+ try {
998
+ let response: Response;
999
+ if (actionContinuation) {
1000
+ response = await revalidateAfterAction(
1001
+ handlerCtx,
1002
+ request,
1003
+ env,
1004
+ url,
1005
+ handleStore,
1006
+ actionContinuation,
1007
+ );
1008
+ } else {
1009
+ response = await handleRscRendering(
1010
+ handlerCtx,
1011
+ request,
1012
+ env,
1013
+ url,
1014
+ isPartial,
1015
+ handleStore,
1016
+ nonce,
1017
+ );
1018
+ }
1019
+ if (negotiated && !isWebSocketUpgradeResponse(response)) {
1020
+ response.headers.append("Vary", "Accept");
1021
+ }
1022
+ return response;
1023
+ } catch (error) {
1024
+ // Check if middleware/handler returned Response
1025
+ if (error instanceof Response) {
1026
+ // An action revalidation render is delivered to the client over the
1027
+ // same Flight-parsing path as a partial navigation, so a Response
1028
+ // thrown during it must be converted exactly like a partial one
1029
+ // (raw 200 -> hard-nav hint, 3xx -> Flight redirect). Without this,
1030
+ // the no-middleware path returns the raw Response (the with-middleware
1031
+ // path is already covered by the isPartial || actionContinuation
1032
+ // guard below).
1033
+ const treatAsPartial = isPartial || actionContinuation != null;
1034
+
1035
+ // During partial (client-side navigation), a 200 Response from a handler
1036
+ // means the route serves raw content (JSON, text, etc.), not JSX.
1037
+ // Signal the browser to hard-navigate so it renders the raw response.
1038
+ if (treatAsPartial && error.status === 200) {
1039
+ console.warn(
1040
+ `[RSC] Route handler at ${url.pathname} returned a Response during client-side navigation. ` +
1041
+ `Falling back to hard navigation. Use data-external on the <Link> to avoid the extra round-trip.`,
1042
+ );
1043
+ return createResponseWithMergedHeaders(null, {
1044
+ status: 200,
1045
+ headers: {
1046
+ "X-RSC-Reload": stripInternalParams(url).toString(),
1047
+ "content-type": "text/x-component;charset=utf-8",
1048
+ },
1049
+ });
1050
+ }
1160
1051
 
1161
- if (!loaderId) {
1162
- return createResponseWithMergedHeaders("Missing _rsc_loader parameter", {
1163
- status: 400,
1164
- });
1165
- }
1052
+ if (treatAsPartial) {
1053
+ const intercepted = interceptRedirectForPartial(
1054
+ error,
1055
+ createRedirectFlightResponse,
1056
+ );
1057
+ if (intercepted) return intercepted;
1058
+ }
1166
1059
 
1167
- // Look up loader lazily
1168
- const registeredLoader = await getLoaderLazy(loaderId);
1169
- if (!registeredLoader) {
1170
- return createResponseWithMergedHeaders(
1171
- `Loader "${loaderId}" not found in registry`,
1172
- { status: 404 },
1173
- );
1174
- }
1060
+ return error;
1061
+ }
1175
1062
 
1176
- // Parse params and body based on request method
1177
- let loaderParams: Record<string, string> = {};
1178
- let loaderBody: unknown = undefined;
1179
- const isBodyMethod = request.method !== "GET" && request.method !== "HEAD";
1063
+ // Render 404 page for unmatched routes
1064
+ if (isRouteNotFoundError(error)) {
1065
+ callOnError(error, "routing", {
1066
+ request,
1067
+ url,
1068
+ env,
1069
+ handledByBoundary: true,
1070
+ });
1180
1071
 
1181
- if (isBodyMethod) {
1182
- try {
1183
- const contentType = request.headers.get("content-type") || "";
1184
- if (contentType.includes("application/json")) {
1185
- const jsonBody = (await request.json()) as {
1186
- params?: Record<string, string>;
1187
- body?: unknown;
1072
+ const notFoundOption = router.notFound;
1073
+ const notFoundComponent =
1074
+ typeof notFoundOption === "function"
1075
+ ? notFoundOption({ pathname: url.pathname })
1076
+ : (notFoundOption ?? createElement("h1", null, "Not Found"));
1077
+
1078
+ const notFoundSegment = {
1079
+ id: "notFound",
1080
+ namespace: "notFound",
1081
+ type: "route" as const,
1082
+ index: 0,
1083
+ component: notFoundComponent,
1084
+ params: {},
1188
1085
  };
1189
- loaderParams = jsonBody.params ?? {};
1190
- loaderBody = jsonBody.body;
1191
- }
1192
- } catch {
1193
- return createResponseWithMergedHeaders("Invalid JSON body", {
1194
- status: 400,
1195
- });
1196
- }
1197
- } else {
1198
- const loaderParamsJson = url.searchParams.get("_rsc_loader_params");
1199
- if (loaderParamsJson) {
1200
- try {
1201
- loaderParams = JSON.parse(loaderParamsJson);
1202
- } catch {
1203
- return createResponseWithMergedHeaders(
1204
- "Invalid _rsc_loader_params JSON",
1205
- { status: 400 },
1206
- );
1207
- }
1208
- }
1209
- }
1210
-
1211
- // Execute the loader with middleware
1212
- try {
1213
- const { fn, middleware } = registeredLoader;
1214
1086
 
1215
- return await executeLoaderMiddleware(
1216
- middleware,
1217
- request,
1218
- env,
1219
- loaderParams,
1220
- variables,
1221
- async () => {
1222
- const ctx = requireRequestContext();
1223
- const loaderCtx: any = {
1224
- ...ctx,
1225
- params: loaderParams,
1226
- body: loaderBody,
1087
+ const payload: RscPayload = {
1088
+ metadata: {
1089
+ pathname: url.pathname,
1090
+ routerId: router.id,
1091
+ basename: router.basename,
1092
+ segments: [notFoundSegment],
1093
+ matched: [],
1094
+ diff: [],
1095
+ isPartial: false,
1096
+ rootLayout: router.rootLayout,
1097
+ handles: handleStore.stream(),
1098
+ version,
1099
+ stateCookieName: router.resolvedStateCookieName,
1100
+ themeConfig: router.themeConfig,
1101
+ warmupEnabled: router.warmupEnabled,
1102
+ initialTheme: requireRequestContext().theme,
1103
+ },
1227
1104
  };
1228
1105
 
1229
- const result = await fn(loaderCtx);
1106
+ const rscStream = renderToReadableStream(payload, {
1107
+ onError: (error: unknown) => {
1108
+ callOnError(error, "rendering", { request, url, env });
1109
+ },
1110
+ });
1230
1111
 
1231
- interface LoaderPayload {
1232
- loaderResult: unknown;
1112
+ if (isRscRequest(request, url, isPartial)) {
1113
+ return createResponseWithMergedHeaders(rscStream, {
1114
+ status: 404,
1115
+ headers: {
1116
+ "content-type": "text/x-component;charset=utf-8",
1117
+ // Router identity for the client's pre-decode integrity check; a
1118
+ // same-app 404 matches and applies in place. See response-adapter.
1119
+ "X-RSC-Router-Id": router.id,
1120
+ },
1121
+ });
1233
1122
  }
1234
- const loaderPayload: LoaderPayload = { loaderResult: result };
1235
- const rscStream =
1236
- renderToReadableStream<LoaderPayload>(loaderPayload);
1237
1123
 
1238
- return createResponseWithMergedHeaders(rscStream, {
1239
- headers: { "content-type": "text/x-component;charset=utf-8" },
1124
+ const [ssrModule, streamMode] = await getSSRSetup(
1125
+ handlerCtx,
1126
+ request,
1127
+ env,
1128
+ url,
1129
+ requireRequestContext()._metricsStore,
1130
+ );
1131
+ const htmlStream = await ssrModule.renderHTML(rscStream, {
1132
+ nonce,
1133
+ streamMode,
1240
1134
  });
1241
- },
1242
- );
1243
- } catch (error) {
1244
- const err = error instanceof Error ? error : new Error(String(error));
1245
- const isDev = process.env.NODE_ENV !== "production";
1246
-
1247
- console.error("[RSC] Loader error:", error);
1248
-
1249
- callOnError(error, "loader", {
1250
- request,
1251
- url,
1252
- env,
1253
- loaderName: loaderId,
1254
- handledByBoundary: false,
1255
- });
1256
-
1257
- const errorPayload = {
1258
- loaderResult: null,
1259
- loaderError: {
1260
- message: isDev ? err.message : "An error occurred",
1261
- name: err.name,
1262
- },
1263
- };
1264
- const rscStream = renderToReadableStream(errorPayload);
1265
-
1266
- return createResponseWithMergedHeaders(rscStream, {
1267
- status: 500,
1268
- headers: { "content-type": "text/x-component;charset=utf-8" },
1269
- });
1270
- }
1271
- }
1272
1135
 
1273
- // ============================================================================
1274
- // RSC RENDERING HANDLER (Navigation)
1275
- // ============================================================================
1276
- async function handleRscRendering(
1277
- request: Request,
1278
- env: TEnv,
1279
- url: URL,
1280
- isPartial: boolean,
1281
- handleStore: ReturnType<typeof requireRequestContext>["_handleStore"],
1282
- nonce: string | undefined,
1283
- ): Promise<Response> {
1284
- // Retrieve handler-level timing from variables
1285
- const reqCtx = requireRequestContext();
1286
- const handlerTimingArr: string[] = reqCtx.var.__handlerTiming || [];
1287
- const handlerStart: number = reqCtx.var.__handlerStart || 0;
1288
-
1289
- let payload: RscPayload;
1290
- let serverTiming: string | undefined;
1291
-
1292
- if (isPartial) {
1293
- // Partial render (navigation)
1294
- const result = await router.matchPartial(request, env);
1295
-
1296
- if (!result) {
1297
- // Fall back to full render
1298
- const match = await router.match(request, env);
1299
- setRequestContextParams(match.params);
1300
-
1301
- if (match.redirect) {
1302
- return createResponseWithMergedHeaders(null, {
1303
- status: 308,
1304
- headers: { Location: match.redirect },
1136
+ return createResponseWithMergedHeaders(htmlStream, {
1137
+ status: 404,
1138
+ headers: { "content-type": "text/html;charset=utf-8" },
1305
1139
  });
1306
1140
  }
1307
1141
 
1308
- const renderStart = performance.now();
1309
- const root = renderSegments(match.segments, {
1310
- rootLayout: router.rootLayout,
1311
- });
1312
- const renderDuration = performance.now() - renderStart;
1313
- serverTiming = match.serverTiming
1314
- ? `${match.serverTiming}, rendering;dur=${renderDuration.toFixed(2)}`
1315
- : `rendering;dur=${renderDuration.toFixed(2)}`;
1316
-
1317
- payload = {
1318
- root,
1319
- metadata: {
1320
- pathname: url.pathname,
1321
- segments: match.segments,
1322
- matched: match.matched,
1323
- diff: match.diff,
1324
- isPartial: false,
1325
- handles: handleStore.stream(),
1326
- version,
1327
- themeConfig: router.themeConfig,
1328
- initialTheme: reqCtx.theme,
1329
- },
1330
- };
1331
- } else {
1332
- setRequestContextParams(result.params);
1333
- serverTiming = result.serverTiming;
1334
-
1335
- payload = {
1336
- root: null,
1337
- metadata: {
1338
- pathname: url.pathname,
1339
- segments: result.segments,
1340
- matched: result.matched,
1341
- diff: result.diff,
1342
- isPartial: true,
1343
- slots: result.slots,
1344
- handles: handleStore.stream(),
1345
- version,
1346
- },
1347
- };
1348
- }
1349
- } else {
1350
- // Full render (initial page load)
1351
- const match = await router.match(request, env);
1352
- setRequestContextParams(match.params);
1353
-
1354
- if (match.redirect) {
1355
- return createResponseWithMergedHeaders(null, {
1356
- status: 308,
1357
- headers: { Location: match.redirect },
1142
+ // Report unhandled errors
1143
+ callOnError(error, "routing", {
1144
+ request,
1145
+ url,
1146
+ env,
1147
+ handledByBoundary: false,
1358
1148
  });
1149
+ console.error(`[RSC] Error:`, error);
1150
+ throw error;
1359
1151
  }
1152
+ };
1360
1153
 
1361
- // Caching is now handled in router.match() via cache provider in request context
1362
- // match.segments already contains cached or fresh segments as appropriate
1363
-
1364
- if (url.searchParams.has("__prerender_collect")) {
1365
- // Build-time prerender collection: serialize segments and handle data
1366
- // to JSON for storage as build artifacts. At runtime the worker
1367
- // deserializes these and feeds them through the normal segment pipeline.
1368
- const nonLoaderSegments = match.segments.filter((s) => s.type !== "loader");
1369
- await handleStore.settled;
1370
- const { serializeSegments } = await import("../cache/cache-scope.js");
1371
- const serializedSegments = await serializeSegments(nonLoaderSegments);
1372
- const handles: Record<string, Record<string, unknown[]>> = {};
1373
- for (const seg of nonLoaderSegments) {
1374
- const segHandles = handleStore.getDataForSegment(seg.id);
1375
- if (Object.keys(segHandles).length > 0) {
1376
- handles[seg.id] = segHandles;
1377
- }
1378
- }
1379
- return new Response(
1380
- JSON.stringify({
1381
- segments: serializedSegments,
1382
- handles,
1383
- routeName: match.routeName,
1384
- params: match.params,
1385
- }),
1386
- { headers: { "Content-Type": "application/json" } },
1154
+ // Wrap the render path in a renderStartMs timeout
1155
+ const executeRender = async (): Promise<Response> => {
1156
+ if (routeMiddleware.length > 0) {
1157
+ const mwResponse = await executeMiddleware(
1158
+ buildRouteMiddlewareEntries<TEnv>(routeMiddleware),
1159
+ request,
1160
+ env,
1161
+ variables,
1162
+ renderHandler,
1163
+ routeReverse,
1387
1164
  );
1388
- } else {
1389
- const renderStart = performance.now();
1390
- const root = renderSegments(match.segments, {
1391
- rootLayout: router.rootLayout,
1392
- });
1393
- const renderDuration = performance.now() - renderStart;
1394
- serverTiming = match.serverTiming
1395
- ? `${match.serverTiming}, rendering;dur=${renderDuration.toFixed(2)}`
1396
- : `rendering;dur=${renderDuration.toFixed(2)}`;
1397
-
1398
- payload = {
1399
- root,
1400
- metadata: {
1401
- pathname: url.pathname,
1402
- segments: match.segments,
1403
- matched: match.matched,
1404
- diff: match.diff,
1405
- isPartial: false,
1406
- rootLayout: router.rootLayout,
1407
- handles: handleStore.stream(),
1408
- version,
1409
- themeConfig: router.themeConfig,
1410
- initialTheme: reqCtx.theme,
1411
- },
1412
- };
1413
- }
1414
- }
1415
1165
 
1416
- // Serialize to RSC stream
1417
- const rscSerializeStart = performance.now();
1418
- const rscStream = renderToReadableStream<RscPayload>(payload);
1419
- const rscSerializeDur = performance.now() - rscSerializeStart;
1420
-
1421
- // Determine if this is an RSC request or HTML request
1422
- const isRscRequest =
1423
- (!request.headers.get("accept")?.includes("text/html") &&
1424
- !url.searchParams.has("__html")) ||
1425
- url.searchParams.has("__rsc");
1426
-
1427
- // Build complete Server-Timing: handler phases + match/manifest + rendering + RSC serialize
1428
- const timingParts: string[] = [...handlerTimingArr];
1429
- if (serverTiming) {
1430
- timingParts.push(serverTiming);
1431
- }
1432
- timingParts.push(`rsc-serialize;dur=${rscSerializeDur.toFixed(2)}`);
1166
+ if (isPartial || actionContinuation) {
1167
+ const intercepted = interceptRedirectForPartial(
1168
+ mwResponse,
1169
+ createRedirectFlightResponse,
1170
+ );
1171
+ if (intercepted) return intercepted;
1172
+ }
1433
1173
 
1434
- if (isRscRequest) {
1435
- const fullTiming = timingParts.join(", ");
1436
- const rscHeaders: Record<string, string> = {
1437
- "content-type": "text/x-component;charset=utf-8",
1438
- vary: "accept",
1439
- };
1440
- if (fullTiming) {
1441
- rscHeaders["Server-Timing"] = fullTiming;
1174
+ return finalizeResponse(mwResponse);
1442
1175
  }
1443
- return createResponseWithMergedHeaders(rscStream, {
1444
- headers: rscHeaders,
1445
- });
1446
- }
1447
-
1448
- // Delegate to SSR for HTML response
1449
- const ssrModuleStart = performance.now();
1450
- const ssrModule = await loadSSRModule();
1451
- const ssrModuleDur = performance.now() - ssrModuleStart;
1452
- timingParts.push(`ssr-module-load;dur=${ssrModuleDur.toFixed(2)}`);
1453
-
1454
- const ssrRenderStart = performance.now();
1455
- const htmlStream = await ssrModule.renderHTML(rscStream, { nonce });
1456
- const ssrRenderDur = performance.now() - ssrRenderStart;
1457
- timingParts.push(`ssr-render-html;dur=${ssrRenderDur.toFixed(2)}`);
1458
-
1459
- // Add total handler duration
1460
- if (handlerStart) {
1461
- const totalHandler = performance.now() - handlerStart;
1462
- timingParts.push(`handler-total;dur=${totalHandler.toFixed(2)}`);
1463
- }
1464
1176
 
1465
- const fullTiming = timingParts.join(", ");
1466
- const htmlHeaders: Record<string, string> = {
1467
- "content-type": "text/html;charset=utf-8",
1177
+ return renderHandler();
1468
1178
  };
1469
- if (fullTiming) {
1470
- htmlHeaders["Server-Timing"] = fullTiming;
1471
- }
1472
1179
 
1473
- return createResponseWithMergedHeaders(htmlStream, {
1474
- headers: htmlHeaders,
1475
- });
1180
+ const renderOutcome = await withTimeout(
1181
+ executeRender(),
1182
+ router.timeouts.renderStartMs,
1183
+ "render-start",
1184
+ );
1185
+ if (renderOutcome.timedOut) {
1186
+ return handleTimeoutResponse(
1187
+ request,
1188
+ env,
1189
+ url,
1190
+ "render-start",
1191
+ renderOutcome.durationMs,
1192
+ routeKey,
1193
+ );
1194
+ }
1195
+ return renderOutcome.result;
1476
1196
  }
1477
1197
  }