@rangojs/router 0.0.0-experimental.31 → 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 +121 -205
  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 +192 -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 +64 -25
  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 +348 -128
  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
@@ -0,0 +1,107 @@
1
+ import type { DefaultEnv } from "./global-namespace.js";
2
+
3
+ /**
4
+ * Minimal subset of Cloudflare Workers' ExecutionContext that the router
5
+ * uses. Defined locally so the package does not depend on
6
+ * `@cloudflare/workers-types`. Consumers that want the full type can cast.
7
+ *
8
+ * On non-Cloudflare runtimes (Node, dev server, tests), this is undefined
9
+ * — portable apps should prefer `ctx.waitUntil(...)`, which degrades
10
+ * gracefully. `ctx.executionContext` is the escape hatch for libraries
11
+ * (MCP, Durable Object routing, etc.) that type their arguments as the
12
+ * raw ExecutionContext.
13
+ */
14
+ export interface ExecutionContext {
15
+ waitUntil(promise: Promise<any>): void;
16
+ passThroughOnException(): void;
17
+ }
18
+
19
+ /**
20
+ * Fallback `waitUntil` body used when no Cloudflare `ExecutionContext`
21
+ * is available (Node, dev, tests). Runs the work fire-and-forget and
22
+ * logs errors so they don't silently swallow.
23
+ *
24
+ * Exported so every `waitUntil` call site degrades identically instead
25
+ * of inventing its own fallback policy.
26
+ */
27
+ export function fireAndForgetWaitUntil(fn: () => Promise<void>): void {
28
+ fn().catch((err) =>
29
+ console.error("[waitUntil] Background task failed:", err),
30
+ );
31
+ }
32
+
33
+ /**
34
+ * Fields present on every user-facing request context.
35
+ *
36
+ * @template TEnv - Platform bindings type (Cloudflare env, etc.).
37
+ */
38
+ export interface RequestScope<TEnv = DefaultEnv> {
39
+ /**
40
+ * The original incoming Request object (transport URL intact).
41
+ * Use `url` / `searchParams` for application logic — those have
42
+ * internal `_rsc*` params stripped. `request` preserves the raw URL
43
+ * when you need original headers, method, or body.
44
+ */
45
+ request: Request;
46
+
47
+ /**
48
+ * The request URL with internal `_rsc*` transport params stripped.
49
+ * Use this for routing, link generation, and display.
50
+ */
51
+ url: URL;
52
+
53
+ /**
54
+ * The original request URL with all parameters intact, including
55
+ * internal `_rsc*` transport params. Use `url` for application logic
56
+ * — this is only needed for advanced cases like debugging or custom
57
+ * cache keying.
58
+ */
59
+ originalUrl: URL;
60
+
61
+ /** URL pathname (same as `url.pathname`). */
62
+ pathname: string;
63
+
64
+ /**
65
+ * Query parameters from the URL (system params like `_rsc*` are
66
+ * filtered). Always a standard `URLSearchParams` instance.
67
+ */
68
+ searchParams: URLSearchParams;
69
+
70
+ /**
71
+ * Platform bindings (DB, KV, secrets, etc.). On Cloudflare Workers
72
+ * these are the `env` object passed to the Worker's `fetch()` handler.
73
+ */
74
+ env: TEnv;
75
+
76
+ /**
77
+ * Schedule work to run after the response is sent.
78
+ * On Cloudflare Workers, delegates to `executionContext.waitUntil()`.
79
+ * On Node / dev / tests, runs as fire-and-forget with error logging.
80
+ *
81
+ * @example
82
+ * ```typescript
83
+ * ctx.waitUntil(async () => {
84
+ * await cacheStore.set(key, data, ttl);
85
+ * });
86
+ * ```
87
+ */
88
+ waitUntil(fn: () => Promise<void>): void;
89
+
90
+ /**
91
+ * Raw Cloudflare Workers `ExecutionContext`, when running on a
92
+ * Cloudflare-compatible runtime. Undefined elsewhere.
93
+ *
94
+ * Escape hatch for libraries that type their arguments as
95
+ * `ExecutionContext` (MCP `fetch`, `routeAgentRequest`, etc.).
96
+ * For the common "do work after the response" case, prefer
97
+ * `ctx.waitUntil(...)` — it is platform-neutral.
98
+ *
99
+ * @example
100
+ * ```typescript
101
+ * path.any("/mcp", (ctx) =>
102
+ * emailMcp.fetch(ctx.request, ctx.env, ctx.executionContext!),
103
+ * );
104
+ * ```
105
+ */
106
+ executionContext?: ExecutionContext;
107
+ }
@@ -7,47 +7,24 @@ export type DocumentProps = {
7
7
  children: ReactNode;
8
8
  };
9
9
 
10
- /**
11
- * Parse constraint values into a union type
12
- * "a|b|c" -> "a" | "b" | "c"
13
- */
14
10
  type ParseConstraint<T extends string> =
15
11
  T extends `${infer First}|${infer Rest}` ? First | ParseConstraint<Rest> : T;
16
12
 
17
- /**
18
- * Extract param info from a param segment
19
- *
20
- * Handles:
21
- * - :param -> { name: "param", optional: false, type: string }
22
- * - :param? -> { name: "param", optional: true, type: string }
23
- * - :param(a|b) -> { name: "param", optional: false, type: "a" | "b" }
24
- * - :param(a|b)? -> { name: "param", optional: true, type: "a" | "b" }
25
- */
26
13
  type ExtractParamInfo<T extends string> =
27
- // Optional + constrained (with optional suffix): :param(a|b)?suffix
28
14
  T extends `${infer Name}(${infer Constraint})?${string}`
29
15
  ? { name: Name; optional: true; type: ParseConstraint<Constraint> }
30
- : // Constrained (with optional suffix): :param(a|b)suffix
31
- T extends `${infer Name}(${infer Constraint})${string}`
16
+ : T extends `${infer Name}(${infer Constraint})${string}`
32
17
  ? { name: Name; optional: false; type: ParseConstraint<Constraint> }
33
- : // Optional (with optional suffix): :param?suffix
34
- T extends `${infer Name}?${string}`
18
+ : T extends `${infer Name}?${string}`
35
19
  ? { name: Name; optional: true; type: string }
36
- : // Param with dot-suffix: :param.html
37
- T extends `${infer Name}.${string}`
20
+ : T extends `${infer Name}.${string}`
38
21
  ? { name: Name; optional: false; type: string }
39
- : // Param with dash-suffix: :param-slug
40
- T extends `${infer Name}-${string}`
22
+ : T extends `${infer Name}-${string}`
41
23
  ? { name: Name; optional: false; type: string }
42
- : // Param with tilde-suffix: :param~v2
43
- T extends `${infer Name}~${string}`
24
+ : T extends `${infer Name}~${string}`
44
25
  ? { name: Name; optional: false; type: string }
45
- : // Required: :param (no suffix)
46
- { name: T; optional: false; type: string };
26
+ : { name: T; optional: false; type: string };
47
27
 
48
- /**
49
- * Build param object from info
50
- */
51
28
  type ParamFromInfo<Info> = Info extends {
52
29
  name: infer N extends string;
53
30
  optional: true;
@@ -62,10 +39,6 @@ type ParamFromInfo<Info> = Info extends {
62
39
  ? { [K in N]: V }
63
40
  : never;
64
41
 
65
- /**
66
- * Merge two param objects preserving optionality
67
- * Uses Pick to preserve the modifiers from source types
68
- */
69
42
  type MergeParams<A, B> = Pick<A, keyof A> & Pick<B, keyof B> extends infer O
70
43
  ? { [K in keyof O]: O[K] }
71
44
  : never;
@@ -109,17 +82,11 @@ export type ExtractParams<
109
82
  */
110
83
  export type TrailingSlashMode = "never" | "always" | "ignore";
111
84
 
112
- /**
113
- * Route configuration object (alternative to string path)
114
- */
115
85
  export type RouteConfig = {
116
86
  path: string;
117
87
  trailingSlash?: TrailingSlashMode;
118
88
  };
119
89
 
120
- /**
121
- * Route definition options (global defaults)
122
- */
123
90
  export type RouteDefinitionOptions = {
124
91
  trailingSlash?: TrailingSlashMode;
125
92
  };
@@ -128,11 +95,6 @@ export type RouteDefinition = {
128
95
  [key: string]: string | RouteConfig | RouteDefinition;
129
96
  };
130
97
 
131
- /**
132
- * Recursively flatten nested routes with depth limit to prevent infinite recursion
133
- * Transforms: { products: { detail: "/product/:slug" } } => { "products.detail": "/product/:slug" }
134
- * Also handles RouteConfig objects: { api: { path: "/api" } } => { "api": "/api" }
135
- */
136
98
  type FlattenRoutes<
137
99
  T extends RouteDefinition,
138
100
  Prefix extends string = "",
@@ -153,18 +115,12 @@ type FlattenRoutes<
153
115
  : never;
154
116
  }[keyof T];
155
117
 
156
- /**
157
- * Union to intersection helper
158
- */
159
118
  type UnionToIntersection<U> = (
160
119
  U extends unknown ? (k: U) => void : never
161
120
  ) extends (k: infer I) => void
162
121
  ? I
163
122
  : never;
164
123
 
165
- /**
166
- * Resolved route map - flattened route definitions with full paths
167
- */
168
124
  export type ResolvedRouteMap<T extends RouteDefinition> = UnionToIntersection<
169
125
  FlattenRoutes<T>
170
126
  >;
@@ -1,22 +1,27 @@
1
1
  import type { AllUseItems } from "../route-types.js";
2
2
  import type { TrailingSlashMode, ResolvedRouteMap } from "./route-config.js";
3
3
 
4
- /**
5
- * Context captured for lazy include evaluation
6
- */
7
4
  export interface LazyIncludeContext {
8
5
  urlPrefix: string;
9
6
  namePrefix: string | undefined;
10
7
  parent: unknown; // EntryData - avoid circular import
8
+ /** Counter snapshot from pattern extraction for consistent shortCode indices */
9
+ counters?: Record<string, number>;
11
10
  cacheProfiles?: Record<
12
11
  string,
13
12
  import("../cache/profile-registry.js").CacheProfile
14
13
  >;
14
+ /** Root scope flag for dot-local reverse resolution */
15
+ rootScoped?: boolean;
16
+ /**
17
+ * Positional include scope token composed from the parent scope plus this
18
+ * include's sibling index (`${parentScope}I${idx}`). Applied to direct-
19
+ * descendant shortCodes during lazy evaluation so routes inside the
20
+ * include cannot collide with siblings declared outside it.
21
+ */
22
+ includeScope?: string;
15
23
  }
16
24
 
17
- /**
18
- * Internal route entry stored in router
19
- */
20
25
  export interface RouteEntry<TEnv = any> {
21
26
  prefix: string;
22
27
  /**
@@ -55,6 +60,13 @@ export interface RouteEntry<TEnv = any> {
55
60
  | Promise<() => Array<AllUseItems>>;
56
61
  mountIndex: number;
57
62
 
63
+ /**
64
+ * Router ID that owns this entry. Used to namespace the manifest cache
65
+ * so multi-router setups (host routing) don't share cached EntryData
66
+ * across routers with overlapping mountIndex + routeKey combinations.
67
+ */
68
+ routerId?: string;
69
+
58
70
  /**
59
71
  * Route keys in this entry that have pre-render handlers.
60
72
  * Used by the non-trie match path to set the `pr` flag.
@@ -62,7 +74,7 @@ export interface RouteEntry<TEnv = any> {
62
74
  prerenderRouteKeys?: Set<string>;
63
75
 
64
76
  /**
65
- * Route keys in this entry that use `{ passthrough: true }`.
77
+ * Route keys in this entry that are wrapped with `Passthrough()`.
66
78
  * Used by the non-trie match path to set the `pt` flag.
67
79
  */
68
80
  passthroughRouteKeys?: Set<string>;
@@ -10,7 +10,10 @@ export type ViewTransitionClass = Record<string, string> | string;
10
10
 
11
11
  /**
12
12
  * Configuration for React's <ViewTransition> component.
13
- * Maps directly to ViewTransitionProps (minus children/ref/callbacks).
13
+ *
14
+ * The phase fields (enter/exit/update/share/default/name) map directly to
15
+ * ViewTransitionProps (minus children/ref/callbacks). The `viewTransition`
16
+ * field is router-specific and is stripped before the config reaches React.
14
17
  */
15
18
  export interface TransitionConfig {
16
19
  enter?: ViewTransitionClass;
@@ -19,19 +22,25 @@ export interface TransitionConfig {
19
22
  share?: ViewTransitionClass;
20
23
  default?: ViewTransitionClass;
21
24
  name?: string;
25
+ /**
26
+ * Whether the router wraps this segment's content in its own
27
+ * <ViewTransition> boundary.
28
+ *
29
+ * - "auto" (default): the router places the boundary, producing the
30
+ * router-owned cross-fade described by the phase fields above.
31
+ * - false: the router places no boundary. The navigation commit is still
32
+ * driven through startTransition (so loaders hold instead of flashing a
33
+ * skeleton, and consumer-placed <ViewTransition> elements still animate),
34
+ * but the router contributes no cross-fade of its own.
35
+ *
36
+ * When unset, inherits the createRouter({ viewTransition }) default.
37
+ */
38
+ viewTransition?: "auto" | false;
22
39
  }
23
40
 
24
41
  /**
25
42
  * Resolved segment with component
26
43
  *
27
- * Segment types:
28
- * - layout: Wraps child content via <Outlet />
29
- * - route: The leaf content for a URL
30
- * - parallel: Named slots rendered via <ParallelOutlet name="@slot" />
31
- * - loader: Data segment (no visual rendering, carries loaderData)
32
- * - error: Error fallback segment (replaces failed segment with error UI)
33
- * - notFound: Not found fallback segment (replaces segment when data not found)
34
- *
35
44
  * @internal This type is an implementation detail and may change without notice.
36
45
  */
37
46
  export interface ResolvedSegment {
@@ -50,7 +59,9 @@ export interface ResolvedSegment {
50
59
  parallelName?: string; // For parallels: the parallel group name (used to match with revalidations)
51
60
  // Loader-specific fields
52
61
  loaderId?: string; // For loaders: the loader $$id identifier
62
+ _inherited?: boolean; // For inherited loaders: dedup marker for buildMatchResult
53
63
  loaderData?: any; // For loaders: the resolved data from loader execution
64
+ parallelLoading?: ReactNode; // For parallel-owned loaders: the parallel's loading fallback
54
65
  // Intercept loader fields (for streaming loader data in parallel segments)
55
66
  loaderDataPromise?: Promise<any[]> | any[]; // Loader data promise or resolved array
56
67
  loaderIds?: string[]; // IDs ($$id) of loaders for this segment
@@ -60,13 +71,16 @@ export interface ResolvedSegment {
60
71
  notFoundInfo?: NotFoundInfo; // For notFound segments: the not found information
61
72
  // Mount path from include() scope, used for MountContext.Provider wrapping
62
73
  mountPath?: string;
74
+ /**
75
+ * @internal Server-side marker: true when the segment's handler actually ran
76
+ * this request (not skipped via the revalidate cache path). Used by
77
+ * match-result.ts to populate `MatchResult.resolvedIds` for client-side
78
+ * handle-bucket cleanup. Stripped from the wire payload before serialization
79
+ * — never reaches the client.
80
+ */
81
+ _handlerRan?: boolean;
63
82
  }
64
83
 
65
- /**
66
- * Segment metadata (without component)
67
- *
68
- * @internal This type is an implementation detail and may change without notice.
69
- */
70
84
  export interface SegmentMetadata {
71
85
  id: string;
72
86
  type: "layout" | "route" | "parallel" | "loader" | "error" | "notFound";
@@ -114,6 +128,15 @@ export interface MatchResult {
114
128
  segments: ResolvedSegment[];
115
129
  matched: string[];
116
130
  diff: string[];
131
+ /**
132
+ * Every segment id whose handler actually ran on the server this request,
133
+ * including ones with `component === null` that get filtered out of
134
+ * `segments`/`diff` to avoid wasted bytes. Drives the client's handle-
135
+ * cleanup pass — a slot that re-resolves and pushes nothing must clear
136
+ * its previous handle bucket, but `diff` doesn't carry it because the
137
+ * segment payload doesn't either. A superset of `diff`.
138
+ */
139
+ resolvedIds: string[];
117
140
  /**
118
141
  * Merged route params from all matched segments
119
142
  * Available for use by the handler after route matching
@@ -1,10 +1,8 @@
1
1
  import type { AllUseItems, IncludeItem } from "../route-types.js";
2
2
  import {
3
- getContext,
4
- runWithPrefixes,
5
3
  getUrlPrefix,
6
4
  getNamePrefix,
7
- getRootScoped,
5
+ requireDslContext,
8
6
  } from "../server/context";
9
7
  import {
10
8
  INTERNAL_INCLUDE_SCOPE_PREFIX,
@@ -27,28 +25,10 @@ function allocateInternalIncludeScopeId(
27
25
  }
28
26
 
29
27
  /**
30
- * Process an IncludeItem by executing its nested patterns with prefixes
31
- * This expands the include into actual route registrations
32
- */
33
- function processIncludeItem(item: IncludeItem): AllUseItems[] {
34
- const { prefix, patterns } = item;
35
- const namePrefix =
36
- (item as IncludeItem & { _lazyContext?: { namePrefix?: string } })
37
- ._lazyContext?.namePrefix ?? item.options?.name;
38
-
39
- // Execute the nested patterns' handler with URL and name prefixes
40
- // The urlPrefix being set tells nested urls() to skip RootLayout wrapping
41
- return runWithPrefixes(prefix, namePrefix, () => {
42
- // Call the nested patterns' handler - this registers routes with prefixed patterns/names
43
- return (patterns as UrlPatterns).handler();
44
- });
45
- }
46
-
47
- /**
48
- * Recursively process items, expanding any IncludeItems
49
- * Returns items with IncludeItems expanded into actual route items
28
+ * Recursively walk items, recursing into layout children.
50
29
  *
51
- * Lazy includes are kept as-is (not expanded) for the router to handle later.
30
+ * All includes are lazy and kept as-is; the router expands them on the first
31
+ * matching request.
52
32
  */
53
33
  export function processItems(items: readonly AllUseItems[]): AllUseItems[] {
54
34
  const result: AllUseItems[] = [];
@@ -57,28 +37,8 @@ export function processItems(items: readonly AllUseItems[]): AllUseItems[] {
57
37
  if (!item) continue;
58
38
 
59
39
  if (item.type === "include") {
60
- const includeItem = item as IncludeItem & {
61
- _expanded?: AllUseItems[];
62
- lazy?: boolean;
63
- };
64
-
65
- // Lazy includes are NOT expanded here - kept for router to handle
66
- if (includeItem.lazy) {
67
- result.push(item);
68
- continue;
69
- }
70
-
71
- // Eager includes are already expanded during include() call
72
- if (includeItem._expanded) {
73
- // Items were expanded immediately - just process them recursively
74
- result.push(...processItems(includeItem._expanded));
75
- } else {
76
- // Fallback for legacy include items without _expanded
77
- const expanded = processIncludeItem(item as IncludeItem);
78
- result.push(...processItems(expanded));
79
- }
40
+ result.push(item);
80
41
  } else if (item.type === "layout" && (item as any).uses) {
81
- // Process nested items in layout
82
42
  const layoutItem = item as any;
83
43
  layoutItem.uses = processItems(layoutItem.uses);
84
44
  result.push(layoutItem);
@@ -93,13 +53,9 @@ export function processItems(items: readonly AllUseItems[]): AllUseItems[] {
93
53
  /**
94
54
  * Create include() helper for composing URL patterns
95
55
  *
96
- * By default, include() IMMEDIATELY expands the nested patterns. This ensures
97
- * that routes from included patterns inherit the correct parent context
98
- * (the layout they're included in).
99
- *
100
- * With `lazy: true`, patterns are NOT expanded at definition time. Instead,
101
- * they're evaluated on first request that matches the prefix. This improves
102
- * cold start time for apps with many routes.
56
+ * All includes are lazy: the nested patterns are NOT expanded at definition
57
+ * time. Instead they are evaluated on the first request that matches the
58
+ * prefix, which improves cold start time for apps with many routes.
103
59
  */
104
60
  export function createIncludeHelper<TEnv>(): IncludeFn<TEnv> {
105
61
  return (
@@ -107,9 +63,7 @@ export function createIncludeHelper<TEnv>(): IncludeFn<TEnv> {
107
63
  patterns: UrlPatterns<TEnv>,
108
64
  options?: IncludeOptions,
109
65
  ): IncludeItem => {
110
- const store = getContext();
111
- const ctx = store.getStore();
112
- if (!ctx) throw new Error("include() must be called inside urls()");
66
+ const { ctx } = requireDslContext("include() must be called inside urls()");
113
67
 
114
68
  const explicitName = options?.name;
115
69
  const hasExplicitName = hasExplicitNameOption(options);
@@ -149,22 +103,32 @@ export function createIncludeHelper<TEnv>(): IncludeFn<TEnv> {
149
103
  });
150
104
  }
151
105
 
152
- // Snapshot parent's counters so lazy manifest generation starts
153
- // at the correct index, preventing shortCode collisions with
154
- // sibling entries (e.g., BlogLayout and ArticlesLayout under NavLayout).
155
- const capturedCounters = { ...ctx.counters };
156
-
157
- // Reserve a layout slot in the parent's counter so sibling lazy includes
158
- // produce different shortCode indices for their root layout.
159
- // Without this, consecutive include() calls capture identical counters
160
- // and their first child layouts get the same shortCode (e.g., both M0L0L0),
161
- // causing the client partial-update diff to see no changes on navigation.
106
+ // Allocate an include-scope token for this include() call. The token is
107
+ // appended to the parent's shortCode prefix whenever the include's
108
+ // direct-descendant shortCodes are generated (see getShortCode in
109
+ // context.ts), partitioning the parent's counter namespace so routes
110
+ // inside an include cannot collide with siblings declared outside it.
111
+ //
112
+ // Scopes compose: a nested include inside an outer include with scope
113
+ // "I0" allocates against the `${parent.shortCode}I0_include` counter
114
+ // and produces scope "I0I0", "I0I1", etc.
115
+ const parentScope = ctx.includeScope ?? "";
116
+ let includeScope = parentScope;
162
117
  if (capturedParent?.shortCode) {
163
- const layoutCounterKey = `${capturedParent.shortCode}_layout`;
164
- ctx.counters[layoutCounterKey] ??= 0;
165
- ctx.counters[layoutCounterKey]++;
118
+ const includeCounterKey = `${capturedParent.shortCode}${parentScope}_include`;
119
+ ctx.counters[includeCounterKey] ??= 0;
120
+ const includeIdx = ctx.counters[includeCounterKey];
121
+ ctx.counters[includeCounterKey] = includeIdx + 1;
122
+ includeScope = `${parentScope}I${includeIdx}`;
166
123
  }
167
124
 
125
+ // Snapshot parent's counters AFTER allocating the include scope so lazy
126
+ // manifest generation starts with the same counter state this include
127
+ // observed — its descendants still get fresh per-scope counters because
128
+ // they key off `${parent.shortCode}${includeScope}_*` (not shared with
129
+ // siblings outside the include).
130
+ const capturedCounters = { ...ctx.counters };
131
+
168
132
  // Compute rootScoped at capture time, mirroring the logic in runWithPrefixes.
169
133
  // This ensures lazy evaluation restores the correct scope state.
170
134
  const parentRootScoped = ctx.rootScoped;
@@ -175,8 +139,6 @@ export function createIncludeHelper<TEnv>(): IncludeFn<TEnv> {
175
139
  ? (parentRootScoped ?? false)
176
140
  : parentRootScoped;
177
141
 
178
- // All includes are lazy - patterns are evaluated on first matching request
179
- // This improves cold start time significantly for large route sets
180
142
  return {
181
143
  type: "include",
182
144
  name,
@@ -191,6 +153,7 @@ export function createIncludeHelper<TEnv>(): IncludeFn<TEnv> {
191
153
  counters: capturedCounters,
192
154
  cacheProfiles: ctx.cacheProfiles,
193
155
  rootScoped: capturedRootScoped,
156
+ includeScope,
194
157
  },
195
158
  } as IncludeItem;
196
159
  };
package/src/urls/index.ts CHANGED
@@ -1,4 +1,3 @@
1
- // Response types and symbols
2
1
  export {
3
2
  RESPONSE_TYPE,
4
3
  type ResponseHandler,
@@ -8,28 +7,21 @@ export {
8
7
  type ResponseHandlerContext,
9
8
  } from "./response-types.js";
10
9
 
11
- // Pattern types
12
10
  export type {
13
11
  UnnamedRoute,
14
12
  LocalOnlyInclude,
15
13
  PathOptions,
16
- PathDefinition,
17
14
  UrlPatterns,
18
15
  IncludeOptions,
19
16
  } from "./pattern-types.js";
20
17
 
21
- // Type extraction utilities
22
18
  export type {
23
19
  ExtractRoutes,
24
20
  ExtractResponses,
25
- ExtractRouteNames,
26
- ExtractPathParams,
27
- ResponseError,
28
- ResponseEnvelope,
21
+ ProblemDetails,
29
22
  RouteResponse,
30
23
  } from "./type-extraction.js";
31
24
 
32
- // Path helper types
33
25
  export type {
34
26
  PathFn,
35
27
  ResponsePathFn,
@@ -39,10 +31,8 @@ export type {
39
31
  PathHelpers,
40
32
  } from "./path-helper-types.js";
41
33
 
42
- // Main entry point
43
34
  export { urls } from "./urls-function.js";
44
35
 
45
- // Re-exports from route-types
46
36
  export type {
47
37
  AllUseItems,
48
38
  IncludeItem,