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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (376) hide show
  1. package/AGENTS.md +4 -0
  2. package/README.md +198 -44
  3. package/dist/bin/rango.js +287 -105
  4. package/dist/testing/vitest.js +82 -0
  5. package/dist/vite/index.js +3248 -1117
  6. package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  7. package/package.json +73 -21
  8. package/skills/api-client/SKILL.md +211 -0
  9. package/skills/breadcrumbs/SKILL.md +107 -1
  10. package/skills/bundle-analysis/SKILL.md +159 -0
  11. package/skills/cache-guide/SKILL.md +245 -21
  12. package/skills/caching/SKILL.md +302 -6
  13. package/skills/composability/SKILL.md +27 -2
  14. package/skills/css/SKILL.md +76 -0
  15. package/skills/document-cache/SKILL.md +78 -55
  16. package/skills/handler-use/SKILL.md +364 -0
  17. package/skills/hooks/SKILL.md +270 -30
  18. package/skills/host-router/SKILL.md +82 -22
  19. package/skills/i18n/SKILL.md +276 -0
  20. package/skills/intercept/SKILL.md +49 -5
  21. package/skills/layout/SKILL.md +35 -9
  22. package/skills/links/SKILL.md +249 -17
  23. package/skills/loader/SKILL.md +294 -30
  24. package/skills/middleware/SKILL.md +52 -13
  25. package/skills/migrate-nextjs/SKILL.md +584 -0
  26. package/skills/migrate-react-router/SKILL.md +769 -0
  27. package/skills/mime-routes/SKILL.md +27 -0
  28. package/skills/observability/SKILL.md +137 -0
  29. package/skills/parallel/SKILL.md +203 -7
  30. package/skills/prerender/SKILL.md +123 -100
  31. package/skills/rango/SKILL.md +250 -22
  32. package/skills/react-compiler/SKILL.md +168 -0
  33. package/skills/response-routes/SKILL.md +122 -47
  34. package/skills/route/SKILL.md +97 -5
  35. package/skills/router-setup/SKILL.md +90 -5
  36. package/skills/server-actions/SKILL.md +775 -0
  37. package/skills/streams-and-websockets/SKILL.md +283 -0
  38. package/skills/tailwind/SKILL.md +27 -3
  39. package/skills/testing/SKILL.md +129 -0
  40. package/skills/testing/bindings.md +89 -0
  41. package/skills/testing/cache-prerender.md +124 -0
  42. package/skills/testing/client-components.md +122 -0
  43. package/skills/testing/e2e-parity.md +125 -0
  44. package/skills/testing/flight.md +92 -0
  45. package/skills/testing/handles.md +129 -0
  46. package/skills/testing/loader.md +128 -0
  47. package/skills/testing/middleware.md +99 -0
  48. package/skills/testing/render-handler.md +121 -0
  49. package/skills/testing/response-routes.md +95 -0
  50. package/skills/testing/reverse-and-types.md +84 -0
  51. package/skills/testing/server-actions.md +107 -0
  52. package/skills/testing/server-tree.md +128 -0
  53. package/skills/testing/setup.md +120 -0
  54. package/skills/typesafety/SKILL.md +329 -27
  55. package/skills/use-cache/SKILL.md +36 -5
  56. package/skills/view-transitions/SKILL.md +294 -0
  57. package/src/__augment-tests__/augment.ts +81 -0
  58. package/src/__augment-tests__/augmented.check.ts +116 -0
  59. package/src/__internal.ts +67 -40
  60. package/src/browser/action-coordinator.ts +53 -36
  61. package/src/browser/action-fence.ts +47 -0
  62. package/src/browser/app-shell.ts +39 -0
  63. package/src/browser/app-version.ts +14 -0
  64. package/src/browser/cookie-name.ts +140 -0
  65. package/src/browser/event-controller.ts +86 -147
  66. package/src/browser/history-state.ts +21 -0
  67. package/src/browser/index.ts +3 -3
  68. package/src/browser/invalidate-client-cache.ts +52 -0
  69. package/src/browser/link-interceptor.ts +4 -0
  70. package/src/browser/navigation-bridge.ts +148 -19
  71. package/src/browser/navigation-client.ts +187 -67
  72. package/src/browser/navigation-store-handle.ts +38 -0
  73. package/src/browser/navigation-store.ts +76 -67
  74. package/src/browser/navigation-transaction.ts +18 -66
  75. package/src/browser/partial-update.ts +123 -94
  76. package/src/browser/prefetch/cache.ts +214 -36
  77. package/src/browser/prefetch/fetch.ts +260 -38
  78. package/src/browser/prefetch/policy.ts +6 -0
  79. package/src/browser/prefetch/queue.ts +126 -20
  80. package/src/browser/prefetch/resource-ready.ts +77 -0
  81. package/src/browser/rango-state.ts +158 -76
  82. package/src/browser/react/Link.tsx +93 -11
  83. package/src/browser/react/NavigationProvider.tsx +115 -34
  84. package/src/browser/react/ScrollRestoration.tsx +10 -6
  85. package/src/browser/react/context.ts +7 -2
  86. package/src/browser/react/filter-segment-order.ts +49 -7
  87. package/src/browser/react/index.ts +0 -48
  88. package/src/browser/react/location-state-shared.ts +166 -8
  89. package/src/browser/react/location-state.ts +39 -14
  90. package/src/browser/react/use-action.ts +6 -15
  91. package/src/browser/react/use-handle.ts +23 -69
  92. package/src/browser/react/use-link-status.ts +0 -4
  93. package/src/browser/react/use-navigation.ts +22 -5
  94. package/src/browser/react/use-params.ts +20 -10
  95. package/src/browser/react/use-reverse.ts +106 -0
  96. package/src/browser/react/use-router.ts +46 -11
  97. package/src/browser/react/use-search-params.ts +0 -5
  98. package/src/browser/react/use-segments.ts +11 -21
  99. package/src/browser/response-adapter.ts +52 -1
  100. package/src/browser/rsc-router.tsx +215 -76
  101. package/src/browser/scroll-restoration.ts +46 -39
  102. package/src/browser/segment-reconciler.ts +36 -9
  103. package/src/browser/segment-structure-assert.ts +2 -2
  104. package/src/browser/server-action-bridge.ts +176 -50
  105. package/src/browser/types.ts +95 -11
  106. package/src/browser/validate-redirect-origin.ts +43 -16
  107. package/src/build/collect-fallback-refs.ts +107 -0
  108. package/src/build/generate-manifest.ts +65 -40
  109. package/src/build/generate-route-types.ts +5 -0
  110. package/src/build/index.ts +8 -2
  111. package/src/build/prefix-tree-utils.ts +123 -0
  112. package/src/build/route-trie.ts +137 -32
  113. package/src/build/route-types/codegen.ts +4 -4
  114. package/src/build/route-types/include-resolution.ts +9 -2
  115. package/src/build/route-types/param-extraction.ts +6 -3
  116. package/src/build/route-types/per-module-writer.ts +7 -4
  117. package/src/build/route-types/router-processing.ts +278 -96
  118. package/src/build/route-types/scan-filter.ts +9 -2
  119. package/src/build/route-types/source-scan.ts +118 -0
  120. package/src/build/runtime-discovery.ts +9 -20
  121. package/src/cache/cache-error.ts +104 -0
  122. package/src/cache/cache-policy.ts +68 -28
  123. package/src/cache/cache-runtime.ts +149 -43
  124. package/src/cache/cache-scope.ts +148 -81
  125. package/src/cache/cache-tag.ts +98 -0
  126. package/src/cache/cf/cf-cache-store.ts +2550 -93
  127. package/src/cache/cf/index.ts +11 -17
  128. package/src/cache/document-cache.ts +78 -27
  129. package/src/cache/handle-snapshot.ts +63 -0
  130. package/src/cache/index.ts +23 -20
  131. package/src/cache/memory-segment-store.ts +136 -37
  132. package/src/cache/profile-registry.ts +6 -30
  133. package/src/cache/read-through-swr.ts +41 -11
  134. package/src/cache/segment-codec.ts +0 -16
  135. package/src/cache/tag-invalidation.ts +230 -0
  136. package/src/cache/taint.ts +55 -0
  137. package/src/cache/types.ts +33 -100
  138. package/src/cache/vercel/index.ts +11 -0
  139. package/src/cache/vercel/vercel-cache-store.ts +799 -0
  140. package/src/client.rsc.tsx +6 -21
  141. package/src/client.tsx +108 -290
  142. package/src/component-utils.ts +19 -0
  143. package/src/context-var.ts +84 -2
  144. package/src/debug.ts +2 -2
  145. package/src/decode-loader-results.ts +36 -0
  146. package/src/defer.ts +196 -0
  147. package/src/deps/ssr.ts +0 -1
  148. package/src/errors.ts +30 -4
  149. package/src/handle.ts +70 -22
  150. package/src/handles/MetaTags.tsx +0 -14
  151. package/src/handles/breadcrumbs.ts +16 -5
  152. package/src/handles/meta.ts +0 -39
  153. package/src/host/cookie-handler.ts +0 -36
  154. package/src/host/errors.ts +0 -24
  155. package/src/host/index.ts +8 -2
  156. package/src/host/pattern-matcher.ts +7 -50
  157. package/src/host/router.ts +107 -99
  158. package/src/host/testing.ts +40 -27
  159. package/src/host/types.ts +37 -4
  160. package/src/host/utils.ts +1 -1
  161. package/src/href-client.ts +137 -22
  162. package/src/index.rsc.ts +52 -26
  163. package/src/index.ts +100 -38
  164. package/src/internal-debug.ts +2 -4
  165. package/src/loader-store.ts +500 -0
  166. package/src/loader.rsc.ts +20 -13
  167. package/src/loader.ts +12 -11
  168. package/src/missing-id-error.ts +68 -0
  169. package/src/network-error-thrower.tsx +1 -6
  170. package/src/outlet-context.ts +1 -1
  171. package/src/outlet-provider.tsx +1 -5
  172. package/src/prerender/param-hash.ts +10 -11
  173. package/src/prerender/store.ts +37 -41
  174. package/src/prerender.ts +198 -82
  175. package/src/redirect-origin.ts +100 -0
  176. package/src/response-utils.ts +37 -0
  177. package/src/reverse.ts +65 -15
  178. package/src/root-error-boundary.tsx +1 -19
  179. package/src/route-content-wrapper.tsx +7 -72
  180. package/src/route-definition/dsl-helpers.ts +437 -274
  181. package/src/route-definition/helper-factories.ts +29 -139
  182. package/src/route-definition/helpers-types.ts +113 -37
  183. package/src/route-definition/index.ts +3 -0
  184. package/src/route-definition/redirect.ts +52 -10
  185. package/src/route-definition/resolve-handler-use.ts +161 -0
  186. package/src/route-definition/use-item-types.ts +32 -0
  187. package/src/route-map-builder.ts +7 -17
  188. package/src/route-types.ts +37 -41
  189. package/src/router/basename.ts +14 -0
  190. package/src/router/content-negotiation.ts +108 -9
  191. package/src/router/error-handling.ts +13 -17
  192. package/src/router/find-match.ts +45 -22
  193. package/src/router/handler-context.ts +83 -41
  194. package/src/router/intercept-resolution.ts +25 -23
  195. package/src/router/lazy-includes.ts +19 -53
  196. package/src/router/loader-resolution.ts +213 -30
  197. package/src/router/logging.ts +5 -8
  198. package/src/router/manifest.ts +49 -45
  199. package/src/router/match-api.ts +120 -204
  200. package/src/router/match-context.ts +0 -22
  201. package/src/router/match-handlers.ts +58 -58
  202. package/src/router/match-middleware/background-revalidation.ts +27 -6
  203. package/src/router/match-middleware/cache-lookup.ts +205 -249
  204. package/src/router/match-middleware/cache-store.ts +45 -32
  205. package/src/router/match-middleware/intercept-resolution.ts +8 -28
  206. package/src/router/match-middleware/segment-resolution.ts +52 -18
  207. package/src/router/match-pipelines.ts +1 -42
  208. package/src/router/match-result.ts +104 -40
  209. package/src/router/metrics.ts +5 -34
  210. package/src/router/middleware-types.ts +13 -142
  211. package/src/router/middleware.ts +173 -143
  212. package/src/router/navigation-snapshot.ts +131 -0
  213. package/src/router/params-util.ts +23 -0
  214. package/src/router/pattern-matching.ts +109 -63
  215. package/src/router/prerender-match.ts +190 -54
  216. package/src/router/preview-match.ts +32 -102
  217. package/src/router/request-classification.ts +276 -0
  218. package/src/router/revalidation.ts +63 -55
  219. package/src/router/route-snapshot.ts +244 -0
  220. package/src/router/router-context.ts +6 -28
  221. package/src/router/router-interfaces.ts +100 -35
  222. package/src/router/router-options.ts +91 -11
  223. package/src/router/router-registry.ts +2 -5
  224. package/src/router/segment-resolution/fresh.ts +242 -75
  225. package/src/router/segment-resolution/helpers.ts +63 -24
  226. package/src/router/segment-resolution/loader-cache.ts +41 -37
  227. package/src/router/segment-resolution/revalidation.ts +456 -372
  228. package/src/router/segment-resolution/static-store.ts +19 -5
  229. package/src/router/segment-resolution/streamed-handler-telemetry.ts +52 -0
  230. package/src/router/segment-resolution/view-transition-default.ts +36 -0
  231. package/src/router/segment-resolution.ts +4 -1
  232. package/src/router/segment-wrappers.ts +2 -3
  233. package/src/router/state-cookie-name.ts +33 -0
  234. package/src/router/substitute-pattern-params.ts +56 -0
  235. package/src/router/telemetry-otel.ts +0 -20
  236. package/src/router/telemetry.ts +96 -19
  237. package/src/router/timeout.ts +0 -20
  238. package/src/router/trie-matching.ts +91 -46
  239. package/src/router/types.ts +10 -63
  240. package/src/router/url-params.ts +44 -0
  241. package/src/router.ts +134 -43
  242. package/src/rsc/handler-context.ts +3 -2
  243. package/src/rsc/handler.ts +492 -383
  244. package/src/rsc/helpers.ts +162 -46
  245. package/src/rsc/index.ts +1 -1
  246. package/src/rsc/json-route-result.ts +38 -0
  247. package/src/rsc/loader-fetch.ts +23 -3
  248. package/src/rsc/manifest-init.ts +33 -42
  249. package/src/rsc/origin-guard.ts +39 -25
  250. package/src/rsc/progressive-enhancement.ts +30 -3
  251. package/src/rsc/redirect-guard.ts +99 -0
  252. package/src/rsc/response-error.ts +79 -12
  253. package/src/rsc/response-route-handler.ts +90 -63
  254. package/src/rsc/rsc-rendering.ts +56 -54
  255. package/src/rsc/runtime-warnings.ts +23 -10
  256. package/src/rsc/server-action.ts +74 -67
  257. package/src/rsc/ssr-setup.ts +18 -2
  258. package/src/rsc/types.ts +25 -6
  259. package/src/runtime-env.ts +18 -0
  260. package/src/search-params.ts +4 -20
  261. package/src/segment-content-promise.ts +67 -0
  262. package/src/segment-loader-promise.ts +134 -0
  263. package/src/segment-system.tsx +272 -129
  264. package/src/serialize.ts +243 -0
  265. package/src/server/context.ts +309 -61
  266. package/src/server/cookie-store.ts +80 -5
  267. package/src/server/handle-store.ts +26 -24
  268. package/src/server/loader-registry.ts +10 -28
  269. package/src/server/request-context.ts +338 -126
  270. package/src/ssr/index.tsx +23 -15
  271. package/src/static-handler.ts +27 -18
  272. package/src/testing/cache-status.ts +162 -0
  273. package/src/testing/collect-handle.ts +40 -0
  274. package/src/testing/dispatch.ts +618 -0
  275. package/src/testing/dom.entry.ts +22 -0
  276. package/src/testing/e2e/fixture.ts +188 -0
  277. package/src/testing/e2e/index.ts +128 -0
  278. package/src/testing/e2e/matchers.ts +35 -0
  279. package/src/testing/e2e/page-helpers.ts +272 -0
  280. package/src/testing/e2e/parity.ts +387 -0
  281. package/src/testing/e2e/server.ts +195 -0
  282. package/src/testing/flight-matchers.ts +97 -0
  283. package/src/testing/flight-normalize.ts +11 -0
  284. package/src/testing/flight-runtime.d.ts +57 -0
  285. package/src/testing/flight-tree.ts +682 -0
  286. package/src/testing/flight.entry.ts +52 -0
  287. package/src/testing/flight.ts +232 -0
  288. package/src/testing/generated-routes.ts +183 -0
  289. package/src/testing/index.ts +99 -0
  290. package/src/testing/internal/context.ts +348 -0
  291. package/src/testing/internal/flight-client-globals.ts +30 -0
  292. package/src/testing/internal/seed-vars.ts +54 -0
  293. package/src/testing/render-handler.ts +330 -0
  294. package/src/testing/render-route.tsx +566 -0
  295. package/src/testing/run-loader.ts +378 -0
  296. package/src/testing/run-middleware.ts +205 -0
  297. package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
  298. package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
  299. package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
  300. package/src/testing/vitest-stubs/version.ts +5 -0
  301. package/src/testing/vitest.ts +305 -0
  302. package/src/theme/ThemeProvider.tsx +0 -52
  303. package/src/theme/ThemeScript.tsx +0 -6
  304. package/src/theme/constants.ts +0 -12
  305. package/src/theme/index.ts +0 -7
  306. package/src/theme/theme-context.ts +1 -5
  307. package/src/theme/theme-script.ts +0 -14
  308. package/src/theme/use-theme.ts +0 -3
  309. package/src/types/boundaries.ts +0 -35
  310. package/src/types/cache-types.ts +17 -8
  311. package/src/types/error-types.ts +30 -90
  312. package/src/types/global-namespace.ts +54 -41
  313. package/src/types/handler-context.ts +233 -81
  314. package/src/types/index.ts +1 -10
  315. package/src/types/loader-types.ts +44 -15
  316. package/src/types/request-scope.ts +107 -0
  317. package/src/types/route-config.ts +6 -50
  318. package/src/types/route-entry.ts +19 -7
  319. package/src/types/segments.ts +37 -14
  320. package/src/urls/include-helper.ts +33 -70
  321. package/src/urls/index.ts +1 -11
  322. package/src/urls/path-helper-types.ts +58 -11
  323. package/src/urls/path-helper.ts +57 -111
  324. package/src/urls/pattern-types.ts +48 -19
  325. package/src/urls/response-types.ts +25 -22
  326. package/src/urls/type-extraction.ts +58 -139
  327. package/src/urls/urls-function.ts +1 -18
  328. package/src/use-loader.tsx +346 -89
  329. package/src/vite/debug.ts +185 -0
  330. package/src/vite/discovery/bundle-postprocess.ts +36 -38
  331. package/src/vite/discovery/discover-routers.ts +130 -85
  332. package/src/vite/discovery/discovery-errors.ts +194 -0
  333. package/src/vite/discovery/gate-state.ts +171 -0
  334. package/src/vite/discovery/prerender-collection.ts +192 -99
  335. package/src/vite/discovery/route-types-writer.ts +40 -84
  336. package/src/vite/discovery/self-gen-tracking.ts +27 -1
  337. package/src/vite/discovery/state.ts +51 -6
  338. package/src/vite/discovery/virtual-module-codegen.ts +14 -34
  339. package/src/vite/index.ts +8 -0
  340. package/src/vite/plugin-types.ts +187 -69
  341. package/src/vite/plugins/cjs-to-esm.ts +8 -18
  342. package/src/vite/plugins/client-ref-dedup.ts +16 -11
  343. package/src/vite/plugins/client-ref-hashing.ts +28 -15
  344. package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
  345. package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  346. package/src/vite/plugins/cloudflare-protocol-stub.ts +194 -0
  347. package/src/vite/plugins/expose-action-id.ts +49 -98
  348. package/src/vite/plugins/expose-id-utils.ts +11 -50
  349. package/src/vite/plugins/expose-ids/export-analysis.ts +76 -34
  350. package/src/vite/plugins/expose-ids/handler-transform.ts +10 -48
  351. package/src/vite/plugins/expose-ids/loader-transform.ts +3 -20
  352. package/src/vite/plugins/expose-ids/router-transform.ts +20 -16
  353. package/src/vite/plugins/expose-internal-ids.ts +554 -317
  354. package/src/vite/plugins/performance-tracks.ts +89 -0
  355. package/src/vite/plugins/refresh-cmd.ts +89 -27
  356. package/src/vite/plugins/use-cache-transform.ts +73 -83
  357. package/src/vite/plugins/vercel-output.ts +258 -0
  358. package/src/vite/plugins/version-injector.ts +21 -25
  359. package/src/vite/plugins/version-plugin.ts +41 -20
  360. package/src/vite/plugins/virtual-entries.ts +2 -17
  361. package/src/vite/rango.ts +257 -289
  362. package/src/vite/router-discovery.ts +930 -140
  363. package/src/vite/utils/ast-handler-extract.ts +15 -31
  364. package/src/vite/utils/banner.ts +4 -4
  365. package/src/vite/utils/bundle-analysis.ts +10 -15
  366. package/src/vite/utils/client-chunks.ts +184 -0
  367. package/src/vite/utils/forward-user-plugins.ts +171 -0
  368. package/src/vite/utils/manifest-utils.ts +4 -59
  369. package/src/vite/utils/package-resolution.ts +20 -52
  370. package/src/vite/utils/prerender-utils.ts +27 -29
  371. package/src/vite/utils/shared-utils.ts +92 -42
  372. package/src/browser/action-response-classifier.ts +0 -99
  373. package/src/browser/react/use-client-cache.ts +0 -58
  374. package/src/browser/shallow.ts +0 -40
  375. package/src/handles/index.ts +0 -7
  376. package/src/router/middleware-cookies.ts +0 -55
@@ -1,54 +1,160 @@
1
1
  /**
2
2
  * Prefetch Cache
3
3
  *
4
- * In-memory cache storing prefetch Response objects for instant cache hits
5
- * on subsequent navigation. Cache key is source-dependent (includes the
6
- * current page URL) because the server's diff-based response depends on
7
- * where the user navigates from.
4
+ * In-memory cache storing eagerly-decoded prefetch payloads for instant,
5
+ * already-warm cache hits on subsequent navigation. A prefetch fetches the
6
+ * RSC partial AND decodes it (createFromFetch) up front decoding the Flight
7
+ * stream resolves the route's client references, so the route's JS chunks are
8
+ * imported during prefetch rather than on click. The decoded payload is reused
9
+ * verbatim by navigation, so a prefetched click loads no new code. Two key
10
+ * scopes are in play:
11
+ * - Wildcard (default): built by `buildPrefetchKey(rangoState, target)` —
12
+ * shape `rangoState\0/target?...`. Shared across all source pages and
13
+ * invalidated automatically when Rango state bumps (deploy or
14
+ * server-action invalidation).
15
+ * - Source-scoped: built by `buildSourceKey(rangoState, sourceHref, target)`
16
+ * — shape `rangoState\0sourceHref\0/target?...`. Embeds the Rango state
17
+ * (so rotation invalidates source-scoped entries too) plus the source
18
+ * href (so each originating page gets its own slot). Populated when the
19
+ * server tags a response with `X-RSC-Prefetch-Scope: source` (intercept
20
+ * modals etc.), OR when a Link opts in with `prefetchKey=":source"` — in
21
+ * both cases so source-sensitive responses cannot bleed into navigations
22
+ * from other pages.
23
+ *
24
+ * Also tracks in-flight prefetch promises. Each promise resolves to the
25
+ * decoded prefetch entry (or null), letting navigation adopt a
26
+ * still-downloading prefetch without issuing a duplicate request. A
27
+ * single promise can be registered under multiple alias keys (see
28
+ * `setInflightPromiseWithAliases`) so same-source navigations adopt via
29
+ * their source key while cross-source ones fall through to the wildcard
30
+ * alias — with consume/clear atomically removing every alias.
8
31
  *
9
32
  * Replaces the previous browser HTTP cache approach which was unreliable
10
33
  * due to response draining race conditions and browser inconsistencies.
34
+ *
35
+ * State here lives in module-level singletons (cache, inflight, generation,
36
+ * cacheTTL, etc.) rather than a per-instance factory. This is correct because
37
+ * exactly one router is live per document — an SPA navigation crossing a
38
+ * host-router boundary forces a full document reload — so the singletons are
39
+ * effectively per-document. Unit tests reset them via clearPrefetchCache().
11
40
  */
12
41
 
13
- import { cancelAllPrefetches } from "./queue.js";
42
+ import { abortAllPrefetches } from "./queue.js";
14
43
  import { invalidateRangoState } from "../rango-state.js";
44
+ import type { RscPayload } from "../types.js";
45
+
46
+ /**
47
+ * A prefetch that has been fetched AND eagerly decoded. Storing the decoded
48
+ * payload (not the raw Response) is what makes a prefetched navigation "warm":
49
+ * decoding the Flight stream during prefetch pulls the route's client chunks,
50
+ * so the click reuses ready elements and loads no new JS.
51
+ */
52
+ export interface DecodedPrefetch {
53
+ /** The eagerly-decoded RSC payload. Reused verbatim by navigation. */
54
+ payload: Promise<RscPayload>;
55
+ /**
56
+ * Resolves when the underlying RSC stream finishes draining. Navigation
57
+ * forwards this as its streamComplete so scroll/revalidation gating is
58
+ * unchanged from the fresh-fetch path.
59
+ */
60
+ streamComplete: Promise<void>;
61
+ /**
62
+ * Prefetch scope as tagged by the server via `X-RSC-Prefetch-Scope`.
63
+ * `"source"` means the response is source-page-sensitive and must not be
64
+ * reused by a navigation from a different page — navigation enforces this
65
+ * when it adopted an inflight entry through the wildcard key.
66
+ */
67
+ scope: "source" | "wildcard";
68
+ }
15
69
 
16
- // Default TTL: 5 minutes. Overridden by initPrefetchCache() with
17
- // the server-configured prefetchCacheTTL from router options.
18
- // 0 disables the in-memory cache entirely.
19
70
  let cacheTTL = 300_000;
20
71
 
21
72
  /**
22
73
  * Initialize the prefetch cache with the configured TTL.
23
74
  * Called once at app startup with the value from server metadata.
24
- * A TTL of 0 disables the in-memory cache.
75
+ * A TTL of 0 disables the in-memory cache and all prefetching.
25
76
  */
26
77
  export function initPrefetchCache(ttlMs: number): void {
27
78
  cacheTTL = ttlMs;
28
79
  }
80
+
81
+ /**
82
+ * Check if the prefetch cache is disabled (TTL <= 0).
83
+ * When disabled, no prefetch requests should be issued.
84
+ */
85
+ export function isPrefetchCacheDisabled(): boolean {
86
+ return cacheTTL <= 0;
87
+ }
29
88
  const MAX_PREFETCH_CACHE_SIZE = 50;
30
89
 
31
90
  interface PrefetchCacheEntry {
32
- response: Response;
91
+ entry: DecodedPrefetch;
33
92
  timestamp: number;
34
93
  }
35
94
 
36
95
  const cache = new Map<string, PrefetchCacheEntry>();
37
96
  const inflight = new Set<string>();
38
97
 
39
- // Generation counter incremented on each clearPrefetchCache(). Fetches that
40
- // started before a clear carry a stale generation and must not store their
41
- // response (the data may be stale due to a server action invalidation).
98
+ const inflightPromises = new Map<string, Promise<DecodedPrefetch | null>>();
99
+
100
+ const inflightAliases = new Map<string, string[]>();
101
+
102
+ const adoptedKeys = new Set<string>();
103
+
42
104
  let generation = 0;
43
105
 
44
106
  /**
45
- * Build a source-dependent cache key.
46
- * Includes the source page href so the same target prefetched from
47
- * different pages gets separate entries the server response varies
48
- * based on the source page context (diff-based rendering).
107
+ * Build a cache key by combining a scope prefix with the target URL.
108
+ *
109
+ * Low-level primitive callers that want a specific scope should use
110
+ * one of:
111
+ * - Wildcard (source-agnostic): prefix is the Rango state value from
112
+ * `getRangoState()`. Shared across all source pages. Invalidated
113
+ * automatically when Rango state bumps (deploy or server-action).
114
+ * Key shape: `rangoState\0/target?...`.
115
+ * - Source-scoped: use `buildSourceKey()`. Key shape:
116
+ * `rangoState\0sourceHref\0/target?...` — embeds the Rango state so
117
+ * rotation invalidates source-scoped entries alongside wildcard ones,
118
+ * plus the source page href so the key is unique per originating page.
119
+ * Populated either when the server tags a response with
120
+ * `X-RSC-Prefetch-Scope: source` (intercept modals, etc.) or when a
121
+ * Link opts in via `prefetchKey=":source"`.
122
+ *
123
+ * The `_rsc_segments` query param that travels in the target URL means
124
+ * clients with different mounted segment trees naturally get different
125
+ * keys — so segment-level diffs remain consistent across both scopes.
126
+ */
127
+ export function buildPrefetchKey(prefix: string, targetUrl: URL): string {
128
+ return prefix + "\0" + targetUrl.pathname + targetUrl.search;
129
+ }
130
+
131
+ /**
132
+ * Build a source-scoped cache key. Key shape:
133
+ * `rangoState\0sourceHref\0/target?...`.
134
+ *
135
+ * - `rangoState` is included so state rotation invalidates source-scoped
136
+ * entries alongside wildcard ones.
137
+ * - `sourceHref` makes the key unique per originating page.
138
+ */
139
+ export function buildSourceKey(
140
+ rangoState: string,
141
+ sourceHref: string,
142
+ targetUrl: URL,
143
+ ): string {
144
+ return buildPrefetchKey(rangoState + "\0" + sourceHref, targetUrl);
145
+ }
146
+
147
+ /**
148
+ * Walk an inflight key plus any sibling aliases registered via
149
+ * `setInflightPromiseWithAliases`, invoking `fn` for each.
49
150
  */
50
- export function buildPrefetchKey(sourceHref: string, targetUrl: URL): string {
51
- return sourceHref + "\0" + targetUrl.pathname + targetUrl.search;
151
+ function forEachAlias(key: string, fn: (k: string) => void): void {
152
+ const aliases = inflightAliases.get(key);
153
+ if (aliases) {
154
+ for (const k of aliases) fn(k);
155
+ } else {
156
+ fn(key);
157
+ }
52
158
  }
53
159
 
54
160
  /**
@@ -67,11 +173,14 @@ export function hasPrefetch(key: string): boolean {
67
173
  }
68
174
 
69
175
  /**
70
- * Consume a cached prefetch response. Returns null if not found or expired.
71
- * One-time consumption: the entry is deleted after retrieval.
176
+ * Consume a cached, eagerly-decoded prefetch. Returns null if not found or
177
+ * expired. One-time consumption: the entry is deleted after retrieval.
72
178
  * Returns null when caching is disabled (TTL <= 0).
179
+ *
180
+ * Does NOT check in-flight prefetches — use consumeInflightPrefetch()
181
+ * for that (returns a Promise instead of a resolved entry).
73
182
  */
74
- export function consumePrefetch(key: string): Response | null {
183
+ export function consumePrefetch(key: string): DecodedPrefetch | null {
75
184
  if (cacheTTL <= 0) return null;
76
185
  const entry = cache.get(key);
77
186
  if (!entry) return null;
@@ -80,29 +189,72 @@ export function consumePrefetch(key: string): Response | null {
80
189
  return null;
81
190
  }
82
191
  cache.delete(key);
83
- return entry.response;
192
+ return entry.entry;
84
193
  }
85
194
 
86
195
  /**
87
- * Store a prefetch response in the in-memory cache.
88
- * The response body must be fully buffered (e.g. via arrayBuffer()) before
89
- * storing, so the cached Response is self-contained and network-independent.
196
+ * Consume an in-flight prefetch promise. Returns null if no prefetch is
197
+ * in-flight for this key. The returned Promise resolves to the decoded
198
+ * prefetch entry (or null if the fetch failed/was aborted, or carried a
199
+ * control header the navigation must re-fetch to honor).
200
+ *
201
+ * One-time consumption: the promise entry is removed (along with any
202
+ * sibling aliases registered via `setInflightPromiseWithAliases`) so a
203
+ * second call on any alias returns null — only one caller can adopt the
204
+ * shared Response stream. The `inflight` set entry is intentionally
205
+ * kept so that `hasPrefetch()` continues to return true while the
206
+ * underlying fetch is still downloading — this prevents
207
+ * `prefetchDirect()` or other callers from starting a duplicate request
208
+ * during the handoff window. The inflight flag is cleaned up naturally
209
+ * by `clearPrefetchInflight()` in the fetch's `.finally()`.
210
+ */
211
+ export function consumeInflightPrefetch(
212
+ key: string,
213
+ ): Promise<DecodedPrefetch | null> | null {
214
+ const promise = inflightPromises.get(key);
215
+ if (!promise) return null;
216
+ // Remove the promise under every alias so a second consumer cannot
217
+ // adopt the same stream and race on the body, and mark every alias as
218
+ // adopted so the pending `storePrefetch` (which resolves later, after this
219
+ // adoption) does not leave the now-owned, single-use entry in the cache map.
220
+ // `inflightAliases` is intentionally preserved — `clearPrefetchInflight()` in
221
+ // the fetch's `.finally()` still needs it to clear every inflight flag and
222
+ // adopted marker; deleting here would strand the sibling's flag forever.
223
+ forEachAlias(key, (k) => {
224
+ inflightPromises.delete(k);
225
+ adoptedKeys.add(k);
226
+ });
227
+ return promise;
228
+ }
229
+
230
+ /**
231
+ * Store an eagerly-decoded prefetch in the in-memory cache.
90
232
  *
91
233
  * Skips storage if the generation has changed since the fetch started
92
234
  * (a server action invalidated the cache mid-flight).
93
235
  */
94
236
  export function storePrefetch(
95
237
  key: string,
96
- response: Response,
238
+ entry: DecodedPrefetch,
97
239
  fetchGeneration: number,
98
240
  ): void {
99
241
  if (cacheTTL <= 0) return;
100
242
  if (fetchGeneration !== generation) return;
101
243
 
244
+ // If a navigation already adopted this prefetch's in-flight promise, it owns
245
+ // the single-use entry (and has drained its handle generator). Do NOT also
246
+ // publish it to the cache map, or a later navigation would be served the
247
+ // exhausted entry and lose that route's handles. Clear the marker (under all
248
+ // aliases) now that the decision is made.
249
+ if (adoptedKeys.has(key)) {
250
+ forEachAlias(key, (k) => adoptedKeys.delete(k));
251
+ return;
252
+ }
253
+
102
254
  // Evict expired entries
103
255
  const now = Date.now();
104
- for (const [k, entry] of cache) {
105
- if (now - entry.timestamp > cacheTTL) {
256
+ for (const [k, cached] of cache) {
257
+ if (now - cached.timestamp > cacheTTL) {
106
258
  cache.delete(k);
107
259
  }
108
260
  }
@@ -113,7 +265,7 @@ export function storePrefetch(
113
265
  if (oldest) cache.delete(oldest);
114
266
  }
115
267
 
116
- cache.set(key, { response, timestamp: now });
268
+ cache.set(key, { entry, timestamp: now });
117
269
  }
118
270
 
119
271
  /**
@@ -128,19 +280,45 @@ export function markPrefetchInflight(key: string): void {
128
280
  inflight.add(key);
129
281
  }
130
282
 
131
- export function clearPrefetchInflight(key: string): void {
132
- inflight.delete(key);
283
+ export function setInflightPromise(
284
+ key: string,
285
+ promise: Promise<DecodedPrefetch | null>,
286
+ ): void {
287
+ inflightPromises.set(key, promise);
133
288
  }
134
289
 
135
290
  /**
136
- * Invalidate all prefetch state. Called when server actions mutate data.
137
- * Clears the in-memory cache, cancels in-flight prefetches, and rotates
138
- * the Rango state key so CDN-cached responses are also invalidated.
291
+ * Store the same in-flight Promise under multiple keys, recording them
292
+ * as sibling aliases. Consuming or clearing any one alias atomically
293
+ * removes every entry, guaranteeing the shared Response stream has a
294
+ * single consumer even when navigation looks up either key.
139
295
  */
296
+ export function setInflightPromiseWithAliases(
297
+ keys: string[],
298
+ promise: Promise<DecodedPrefetch | null>,
299
+ ): void {
300
+ for (const k of keys) {
301
+ inflightPromises.set(k, promise);
302
+ inflightAliases.set(k, keys);
303
+ }
304
+ }
305
+
306
+ export function clearPrefetchInflight(key: string): void {
307
+ forEachAlias(key, (k) => {
308
+ inflight.delete(k);
309
+ inflightPromises.delete(k);
310
+ inflightAliases.delete(k);
311
+ adoptedKeys.delete(k);
312
+ });
313
+ }
314
+
140
315
  export function clearPrefetchCache(): void {
141
316
  generation++;
142
317
  inflight.clear();
318
+ inflightPromises.clear();
319
+ inflightAliases.clear();
320
+ adoptedKeys.clear();
143
321
  cache.clear();
144
- cancelAllPrefetches();
322
+ abortAllPrefetches();
145
323
  invalidateRangoState();
146
324
  }