@rangojs/router 0.0.0-experimental.10 → 0.0.0-experimental.100

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 (329) hide show
  1. package/AGENTS.md +9 -0
  2. package/README.md +1037 -4
  3. package/dist/bin/rango.js +1619 -157
  4. package/dist/vite/index.js +5762 -2301
  5. package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  6. package/package.json +71 -63
  7. package/skills/breadcrumbs/SKILL.md +252 -0
  8. package/skills/cache-guide/SKILL.md +294 -0
  9. package/skills/caching/SKILL.md +93 -23
  10. package/skills/composability/SKILL.md +172 -0
  11. package/skills/debug-manifest/SKILL.md +12 -8
  12. package/skills/document-cache/SKILL.md +18 -16
  13. package/skills/fonts/SKILL.md +6 -4
  14. package/skills/handler-use/SKILL.md +364 -0
  15. package/skills/hooks/SKILL.md +367 -71
  16. package/skills/host-router/SKILL.md +218 -0
  17. package/skills/i18n/SKILL.md +276 -0
  18. package/skills/intercept/SKILL.md +176 -8
  19. package/skills/layout/SKILL.md +124 -3
  20. package/skills/links/SKILL.md +304 -25
  21. package/skills/loader/SKILL.md +474 -47
  22. package/skills/middleware/SKILL.md +207 -37
  23. package/skills/migrate-nextjs/SKILL.md +562 -0
  24. package/skills/migrate-react-router/SKILL.md +769 -0
  25. package/skills/mime-routes/SKILL.md +15 -11
  26. package/skills/parallel/SKILL.md +272 -1
  27. package/skills/prerender/SKILL.md +467 -65
  28. package/skills/rango/SKILL.md +89 -21
  29. package/skills/response-routes/SKILL.md +152 -91
  30. package/skills/route/SKILL.md +305 -14
  31. package/skills/router-setup/SKILL.md +210 -32
  32. package/skills/server-actions/SKILL.md +739 -0
  33. package/skills/streams-and-websockets/SKILL.md +283 -0
  34. package/skills/theme/SKILL.md +9 -8
  35. package/skills/typesafety/SKILL.md +333 -86
  36. package/skills/use-cache/SKILL.md +324 -0
  37. package/skills/view-transitions/SKILL.md +212 -0
  38. package/src/__internal.ts +102 -4
  39. package/src/bin/rango.ts +312 -15
  40. package/src/browser/action-coordinator.ts +97 -0
  41. package/src/browser/action-response-classifier.ts +99 -0
  42. package/src/browser/app-shell.ts +52 -0
  43. package/src/browser/app-version.ts +14 -0
  44. package/src/browser/event-controller.ts +136 -68
  45. package/src/browser/history-state.ts +80 -0
  46. package/src/browser/intercept-utils.ts +52 -0
  47. package/src/browser/link-interceptor.ts +24 -4
  48. package/src/browser/logging.ts +55 -0
  49. package/src/browser/merge-segment-loaders.ts +20 -12
  50. package/src/browser/navigation-bridge.ts +374 -561
  51. package/src/browser/navigation-client.ts +228 -70
  52. package/src/browser/navigation-store.ts +97 -55
  53. package/src/browser/navigation-transaction.ts +297 -0
  54. package/src/browser/network-error-handler.ts +61 -0
  55. package/src/browser/partial-update.ts +376 -315
  56. package/src/browser/prefetch/cache.ts +314 -0
  57. package/src/browser/prefetch/fetch.ts +282 -0
  58. package/src/browser/prefetch/observer.ts +65 -0
  59. package/src/browser/prefetch/policy.ts +48 -0
  60. package/src/browser/prefetch/queue.ts +191 -0
  61. package/src/browser/prefetch/resource-ready.ts +77 -0
  62. package/src/browser/rango-state.ts +152 -0
  63. package/src/browser/react/Link.tsx +255 -71
  64. package/src/browser/react/NavigationProvider.tsx +152 -24
  65. package/src/browser/react/context.ts +11 -0
  66. package/src/browser/react/filter-segment-order.ts +55 -0
  67. package/src/browser/react/index.ts +15 -12
  68. package/src/browser/react/location-state-shared.ts +95 -53
  69. package/src/browser/react/location-state.ts +60 -15
  70. package/src/browser/react/mount-context.ts +6 -1
  71. package/src/browser/react/nonce-context.ts +23 -0
  72. package/src/browser/react/shallow-equal.ts +27 -0
  73. package/src/browser/react/use-action.ts +29 -51
  74. package/src/browser/react/use-client-cache.ts +5 -3
  75. package/src/browser/react/use-handle.ts +30 -120
  76. package/src/browser/react/use-link-status.ts +6 -5
  77. package/src/browser/react/use-navigation.ts +44 -65
  78. package/src/browser/react/use-params.ts +78 -0
  79. package/src/browser/react/use-pathname.ts +47 -0
  80. package/src/browser/react/use-reverse.ts +99 -0
  81. package/src/browser/react/use-router.ts +83 -0
  82. package/src/browser/react/use-search-params.ts +56 -0
  83. package/src/browser/react/use-segments.ts +85 -99
  84. package/src/browser/response-adapter.ts +73 -0
  85. package/src/browser/rsc-router.tsx +246 -64
  86. package/src/browser/scroll-restoration.ts +127 -52
  87. package/src/browser/segment-reconciler.ts +243 -0
  88. package/src/browser/segment-structure-assert.ts +16 -0
  89. package/src/browser/server-action-bridge.ts +510 -603
  90. package/src/browser/shallow.ts +6 -1
  91. package/src/browser/types.ts +158 -48
  92. package/src/browser/validate-redirect-origin.ts +29 -0
  93. package/src/build/generate-manifest.ts +84 -23
  94. package/src/build/generate-route-types.ts +39 -828
  95. package/src/build/index.ts +4 -5
  96. package/src/build/route-trie.ts +85 -32
  97. package/src/build/route-types/ast-helpers.ts +25 -0
  98. package/src/build/route-types/ast-route-extraction.ts +98 -0
  99. package/src/build/route-types/codegen.ts +102 -0
  100. package/src/build/route-types/include-resolution.ts +418 -0
  101. package/src/build/route-types/param-extraction.ts +48 -0
  102. package/src/build/route-types/per-module-writer.ts +128 -0
  103. package/src/build/route-types/router-processing.ts +618 -0
  104. package/src/build/route-types/scan-filter.ts +85 -0
  105. package/src/build/runtime-discovery.ts +231 -0
  106. package/src/cache/background-task.ts +34 -0
  107. package/src/cache/cache-key-utils.ts +44 -0
  108. package/src/cache/cache-policy.ts +125 -0
  109. package/src/cache/cache-runtime.ts +342 -0
  110. package/src/cache/cache-scope.ts +167 -307
  111. package/src/cache/cf/cf-cache-store.ts +573 -21
  112. package/src/cache/cf/index.ts +13 -3
  113. package/src/cache/document-cache.ts +116 -77
  114. package/src/cache/handle-capture.ts +81 -0
  115. package/src/cache/handle-snapshot.ts +41 -0
  116. package/src/cache/index.ts +1 -15
  117. package/src/cache/memory-segment-store.ts +191 -13
  118. package/src/cache/profile-registry.ts +73 -0
  119. package/src/cache/read-through-swr.ts +134 -0
  120. package/src/cache/segment-codec.ts +256 -0
  121. package/src/cache/taint.ts +153 -0
  122. package/src/cache/types.ts +72 -122
  123. package/src/client.rsc.tsx +6 -1
  124. package/src/client.tsx +118 -302
  125. package/src/component-utils.ts +4 -4
  126. package/src/components/DefaultDocument.tsx +5 -1
  127. package/src/context-var.ts +156 -0
  128. package/src/debug.ts +19 -9
  129. package/src/errors.ts +77 -7
  130. package/src/handle.ts +55 -10
  131. package/src/handles/MetaTags.tsx +73 -20
  132. package/src/handles/breadcrumbs.ts +66 -0
  133. package/src/handles/index.ts +1 -0
  134. package/src/handles/meta.ts +30 -13
  135. package/src/host/cookie-handler.ts +21 -15
  136. package/src/host/errors.ts +8 -8
  137. package/src/host/index.ts +4 -7
  138. package/src/host/pattern-matcher.ts +27 -27
  139. package/src/host/router.ts +61 -39
  140. package/src/host/testing.ts +8 -8
  141. package/src/host/types.ts +15 -7
  142. package/src/host/utils.ts +1 -1
  143. package/src/href-client.ts +65 -45
  144. package/src/index.rsc.ts +138 -21
  145. package/src/index.ts +206 -51
  146. package/src/internal-debug.ts +11 -0
  147. package/src/loader.rsc.ts +25 -143
  148. package/src/loader.ts +27 -10
  149. package/src/network-error-thrower.tsx +3 -1
  150. package/src/outlet-context.ts +1 -1
  151. package/src/outlet-provider.tsx +45 -0
  152. package/src/prerender/param-hash.ts +4 -2
  153. package/src/prerender/store.ts +159 -13
  154. package/src/prerender.ts +397 -29
  155. package/src/response-utils.ts +28 -0
  156. package/src/reverse.ts +231 -121
  157. package/src/root-error-boundary.tsx +41 -29
  158. package/src/route-content-wrapper.tsx +7 -4
  159. package/src/route-definition/dsl-helpers.ts +1134 -0
  160. package/src/route-definition/helper-factories.ts +200 -0
  161. package/src/route-definition/helpers-types.ts +483 -0
  162. package/src/route-definition/index.ts +55 -0
  163. package/src/route-definition/redirect.ts +101 -0
  164. package/src/route-definition/resolve-handler-use.ts +155 -0
  165. package/src/route-definition.ts +1 -1431
  166. package/src/route-map-builder.ts +162 -123
  167. package/src/route-name.ts +53 -0
  168. package/src/route-types.ts +66 -9
  169. package/src/router/content-negotiation.ts +215 -0
  170. package/src/router/debug-manifest.ts +72 -0
  171. package/src/router/error-handling.ts +9 -9
  172. package/src/router/find-match.ts +160 -0
  173. package/src/router/handler-context.ts +418 -86
  174. package/src/router/intercept-resolution.ts +35 -20
  175. package/src/router/lazy-includes.ts +237 -0
  176. package/src/router/loader-resolution.ts +359 -128
  177. package/src/router/logging.ts +251 -0
  178. package/src/router/manifest.ts +98 -32
  179. package/src/router/match-api.ts +196 -261
  180. package/src/router/match-context.ts +4 -2
  181. package/src/router/match-handlers.ts +441 -0
  182. package/src/router/match-middleware/background-revalidation.ts +108 -93
  183. package/src/router/match-middleware/cache-lookup.ts +415 -86
  184. package/src/router/match-middleware/cache-store.ts +91 -29
  185. package/src/router/match-middleware/intercept-resolution.ts +48 -21
  186. package/src/router/match-middleware/segment-resolution.ts +73 -9
  187. package/src/router/match-pipelines.ts +10 -45
  188. package/src/router/match-result.ts +154 -35
  189. package/src/router/metrics.ts +240 -15
  190. package/src/router/middleware-cookies.ts +55 -0
  191. package/src/router/middleware-types.ts +209 -0
  192. package/src/router/middleware.ts +373 -371
  193. package/src/router/navigation-snapshot.ts +182 -0
  194. package/src/router/pattern-matching.ts +292 -52
  195. package/src/router/prerender-match.ts +502 -0
  196. package/src/router/preview-match.ts +98 -0
  197. package/src/router/request-classification.ts +310 -0
  198. package/src/router/revalidation.ts +152 -39
  199. package/src/router/route-snapshot.ts +245 -0
  200. package/src/router/router-context.ts +41 -21
  201. package/src/router/router-interfaces.ts +484 -0
  202. package/src/router/router-options.ts +618 -0
  203. package/src/router/router-registry.ts +24 -0
  204. package/src/router/segment-resolution/fresh.ts +756 -0
  205. package/src/router/segment-resolution/helpers.ts +268 -0
  206. package/src/router/segment-resolution/loader-cache.ts +199 -0
  207. package/src/router/segment-resolution/revalidation.ts +1407 -0
  208. package/src/router/segment-resolution/static-store.ts +67 -0
  209. package/src/router/segment-resolution.ts +21 -1315
  210. package/src/router/segment-wrappers.ts +291 -0
  211. package/src/router/substitute-pattern-params.ts +56 -0
  212. package/src/router/telemetry-otel.ts +299 -0
  213. package/src/router/telemetry.ts +300 -0
  214. package/src/router/timeout.ts +148 -0
  215. package/src/router/trie-matching.ts +111 -39
  216. package/src/router/types.ts +17 -9
  217. package/src/router/url-params.ts +49 -0
  218. package/src/router.ts +642 -2011
  219. package/src/rsc/handler-context.ts +45 -0
  220. package/src/rsc/handler.ts +864 -1114
  221. package/src/rsc/helpers.ts +181 -19
  222. package/src/rsc/index.ts +0 -20
  223. package/src/rsc/loader-fetch.ts +229 -0
  224. package/src/rsc/manifest-init.ts +90 -0
  225. package/src/rsc/nonce.ts +14 -0
  226. package/src/rsc/origin-guard.ts +141 -0
  227. package/src/rsc/progressive-enhancement.ts +395 -0
  228. package/src/rsc/response-error.ts +37 -0
  229. package/src/rsc/response-route-handler.ts +360 -0
  230. package/src/rsc/rsc-rendering.ts +256 -0
  231. package/src/rsc/runtime-warnings.ts +42 -0
  232. package/src/rsc/server-action.ts +360 -0
  233. package/src/rsc/ssr-setup.ts +128 -0
  234. package/src/rsc/types.ts +52 -11
  235. package/src/search-params.ts +230 -0
  236. package/src/segment-content-promise.ts +67 -0
  237. package/src/segment-loader-promise.ts +122 -0
  238. package/src/segment-system.tsx +187 -38
  239. package/src/server/context.ts +333 -59
  240. package/src/server/cookie-store.ts +190 -0
  241. package/src/server/fetchable-loader-store.ts +37 -0
  242. package/src/server/handle-store.ts +113 -15
  243. package/src/server/loader-registry.ts +24 -64
  244. package/src/server/request-context.ts +603 -109
  245. package/src/server.ts +35 -155
  246. package/src/ssr/index.tsx +107 -30
  247. package/src/static-handler.ts +126 -0
  248. package/src/theme/ThemeProvider.tsx +21 -15
  249. package/src/theme/ThemeScript.tsx +5 -5
  250. package/src/theme/constants.ts +5 -2
  251. package/src/theme/index.ts +4 -14
  252. package/src/theme/theme-context.ts +4 -30
  253. package/src/theme/theme-script.ts +21 -18
  254. package/src/types/boundaries.ts +158 -0
  255. package/src/types/cache-types.ts +198 -0
  256. package/src/types/error-types.ts +192 -0
  257. package/src/types/global-namespace.ts +100 -0
  258. package/src/types/handler-context.ts +764 -0
  259. package/src/types/index.ts +88 -0
  260. package/src/types/loader-types.ts +209 -0
  261. package/src/types/request-scope.ts +126 -0
  262. package/src/types/route-config.ts +170 -0
  263. package/src/types/route-entry.ts +120 -0
  264. package/src/types/segments.ts +167 -0
  265. package/src/types.ts +1 -1757
  266. package/src/urls/include-helper.ts +207 -0
  267. package/src/urls/index.ts +53 -0
  268. package/src/urls/path-helper-types.ts +372 -0
  269. package/src/urls/path-helper.ts +364 -0
  270. package/src/urls/pattern-types.ts +107 -0
  271. package/src/urls/response-types.ts +108 -0
  272. package/src/urls/type-extraction.ts +372 -0
  273. package/src/urls/urls-function.ts +98 -0
  274. package/src/urls.ts +1 -1282
  275. package/src/use-loader.tsx +161 -81
  276. package/src/vite/debug.ts +184 -0
  277. package/src/vite/discovery/bundle-postprocess.ts +181 -0
  278. package/src/vite/discovery/discover-routers.ts +376 -0
  279. package/src/vite/discovery/gate-state.ts +171 -0
  280. package/src/vite/discovery/prerender-collection.ts +486 -0
  281. package/src/vite/discovery/route-types-writer.ts +258 -0
  282. package/src/vite/discovery/self-gen-tracking.ts +73 -0
  283. package/src/vite/discovery/state.ts +117 -0
  284. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  285. package/src/vite/index.ts +15 -2063
  286. package/src/vite/plugin-types.ts +103 -0
  287. package/src/vite/plugins/cjs-to-esm.ts +98 -0
  288. package/src/vite/plugins/client-ref-dedup.ts +131 -0
  289. package/src/vite/plugins/client-ref-hashing.ts +117 -0
  290. package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
  291. package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  292. package/src/vite/plugins/cloudflare-protocol-stub.ts +214 -0
  293. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +107 -64
  294. package/src/vite/plugins/expose-id-utils.ts +299 -0
  295. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  296. package/src/vite/plugins/expose-ids/handler-transform.ts +209 -0
  297. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  298. package/src/vite/plugins/expose-ids/router-transform.ts +127 -0
  299. package/src/vite/plugins/expose-ids/types.ts +45 -0
  300. package/src/vite/plugins/expose-internal-ids.ts +816 -0
  301. package/src/vite/plugins/performance-tracks.ts +96 -0
  302. package/src/vite/plugins/refresh-cmd.ts +127 -0
  303. package/src/vite/plugins/use-cache-transform.ts +336 -0
  304. package/src/vite/plugins/version-injector.ts +109 -0
  305. package/src/vite/plugins/version-plugin.ts +266 -0
  306. package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
  307. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  308. package/src/vite/rango.ts +497 -0
  309. package/src/vite/router-discovery.ts +1423 -0
  310. package/src/vite/utils/ast-handler-extract.ts +517 -0
  311. package/src/vite/utils/banner.ts +36 -0
  312. package/src/vite/utils/bundle-analysis.ts +137 -0
  313. package/src/vite/utils/manifest-utils.ts +70 -0
  314. package/src/vite/utils/package-resolution.ts +161 -0
  315. package/src/vite/utils/prerender-utils.ts +222 -0
  316. package/src/vite/utils/shared-utils.ts +170 -0
  317. package/CLAUDE.md +0 -43
  318. package/src/browser/lru-cache.ts +0 -69
  319. package/src/browser/request-controller.ts +0 -164
  320. package/src/cache/memory-store.ts +0 -253
  321. package/src/href-context.ts +0 -33
  322. package/src/router.gen.ts +0 -6
  323. package/src/urls.gen.ts +0 -8
  324. package/src/vite/expose-handle-id.ts +0 -209
  325. package/src/vite/expose-loader-id.ts +0 -426
  326. package/src/vite/expose-location-state-id.ts +0 -177
  327. package/src/vite/expose-prerender-handler-id.ts +0 -429
  328. package/src/vite/package-resolution.ts +0 -125
  329. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -5,57 +5,180 @@
5
5
  */
6
6
 
7
7
  import type { HandlerContext, InternalHandlerContext } from "../types";
8
- import { getRequestContext } from "../server/request-context.js";
8
+ import { _getRequestContext } from "../server/request-context.js";
9
+ import { getSearchSchema, isRouteRootScoped } from "../route-map-builder.js";
10
+ import { parseSearchParams, serializeSearchParams } from "../search-params.js";
11
+ import {
12
+ contextGet,
13
+ contextSet,
14
+ isNonCacheable,
15
+ type ContextSetOptions,
16
+ } from "../context-var.js";
17
+ import { isInsideCacheScope } from "../server/context.js";
18
+ import { NOCACHE_SYMBOL, assertNotInsideCacheExec } from "../cache/taint.js";
19
+ import { isAutoGeneratedRouteName } from "../route-name.js";
20
+ import { PRERENDER_PASSTHROUGH } from "../prerender.js";
21
+ import { substitutePatternParams } from "./substitute-pattern-params.js";
22
+ import { fireAndForgetWaitUntil } from "../types/request-scope.js";
23
+
24
+ /**
25
+ * Strip internal _rsc* query params from a URL.
26
+ * Returns a new URL with only user-facing params.
27
+ */
28
+ export function stripInternalParams(url: URL): URL {
29
+ const clean = new URL(url);
30
+ for (const key of [...clean.searchParams.keys()]) {
31
+ if (key.startsWith("_rsc")) {
32
+ clean.searchParams.delete(key);
33
+ }
34
+ }
35
+ return clean;
36
+ }
9
37
 
10
38
  /**
11
39
  * Resolve route name with namespace prefix support.
12
- * Supports local names, absolute names (dot notation), and path-based URLs.
40
+ * Supports local names (dot-prefixed) and absolute names (global lookup).
41
+ *
42
+ * @param rootScoped - Explicit override for root-scope check. When undefined,
43
+ * falls back to the global scope registry, then to a heuristic.
13
44
  */
14
45
  function resolveRouteName(
15
46
  name: string,
16
47
  routeMap: Record<string, string>,
17
- currentRoutePrefix?: string
48
+ currentRoutePrefix?: string,
49
+ rootScoped?: boolean,
18
50
  ): string | undefined {
19
- // 1. Path-based - starts with /
20
- if (name.startsWith("/")) {
21
- return name;
22
- }
23
-
24
- // 2. Absolute name - already has a dot (e.g., "shop.cart")
25
- if (name.includes(".")) {
26
- return routeMap[name];
27
- }
51
+ // 1. Dot-prefixed (".article", ".author.posts") local resolution only.
52
+ // Resolves within the current include() scope using the mount prefix.
53
+ if (name.startsWith(".")) {
54
+ const lookupName = name.slice(1);
55
+ if (!currentRoutePrefix) return undefined;
28
56
 
29
- // 3. Local name - try with current prefix first, then fall back to direct lookup
30
- if (currentRoutePrefix) {
31
- // Extract the prefix from current route name
32
- // e.g., "blog.posts.detail" → prefix is "blog.posts"
57
+ // Extract the include prefix from current route name
58
+ // e.g., "magazine.author" -> prefix is "magazine"
33
59
  const lastDot = currentRoutePrefix.lastIndexOf(".");
34
- const prefix = lastDot > 0 ? currentRoutePrefix.substring(0, lastDot) : currentRoutePrefix;
60
+ const prefix =
61
+ lastDot > 0
62
+ ? currentRoutePrefix.substring(0, lastDot)
63
+ : currentRoutePrefix;
35
64
 
36
- // Try prefixed name
37
- const prefixedName = `${prefix}.${name}`;
65
+ // Try prefixed name at current level
66
+ const prefixedName = `${prefix}.${lookupName}`;
38
67
  if (routeMap[prefixedName] !== undefined) {
39
68
  return routeMap[prefixedName];
40
69
  }
41
70
 
42
- // If current route is a nested include, try parent prefixes
43
- // e.g., for "blog.posts.detail", try "blog.posts.index", then "blog.index"
71
+ // Walk up parent prefixes for nested includes
44
72
  let currentPrefix = prefix;
45
73
  while (currentPrefix.includes(".")) {
46
74
  const parentDot = currentPrefix.lastIndexOf(".");
47
75
  currentPrefix = currentPrefix.substring(0, parentDot);
48
- const parentPrefixedName = `${currentPrefix}.${name}`;
76
+ const parentPrefixedName = `${currentPrefix}.${lookupName}`;
49
77
  if (routeMap[parentPrefixedName] !== undefined) {
50
78
  return routeMap[parentPrefixedName];
51
79
  }
52
80
  }
81
+
82
+ // Fallback: try bare name at root scope only.
83
+ // Routes inside { name: "" } mounts are at root scope — their dot-local
84
+ // names can fall back to bare names (e.g., "sub.detail" reaching "flatIndex").
85
+ // Routes inside named mounts (e.g., { name: "magazine" }) are NOT at root
86
+ // scope — dot-local must not leak into unrelated global names.
87
+ //
88
+ // Resolution order: explicit param > scope registry > heuristic.
89
+ const isRootScoped =
90
+ rootScoped ??
91
+ isRouteRootScoped(currentRoutePrefix) ??
92
+ !currentRoutePrefix.includes(".");
93
+ if (isRootScoped) {
94
+ if (routeMap[lookupName] !== undefined) {
95
+ return routeMap[lookupName];
96
+ }
97
+ }
98
+
99
+ return undefined;
53
100
  }
54
101
 
55
- // Fall back to direct lookup (route without prefix)
102
+ // 2. Unprefixed ("magazine.index", "blog.post") global resolution only.
103
+ // Direct lookup in the full named-routes map.
56
104
  return routeMap[name];
57
105
  }
58
106
 
107
+ function createPrerenderPassthroughFn(
108
+ build: boolean,
109
+ isPassthroughRoute: boolean,
110
+ ): () => typeof PRERENDER_PASSTHROUGH {
111
+ return () => {
112
+ if (!build) {
113
+ throw new Error(
114
+ "ctx.passthrough() can only be called during build-time prerendering.",
115
+ );
116
+ }
117
+ if (!isPassthroughRoute) {
118
+ throw new Error(
119
+ "ctx.passthrough() is only available on routes wrapped with " +
120
+ "Passthrough(). Remove the passthrough() call or wrap the " +
121
+ "Prerender definition with Passthrough(prerenderDef, liveHandler).",
122
+ );
123
+ }
124
+ return PRERENDER_PASSTHROUGH;
125
+ };
126
+ }
127
+
128
+ /**
129
+ * Create a reverse function for URL generation from route names.
130
+ * Used by both HandlerContext and MiddlewareContext.
131
+ *
132
+ * When currentParams is provided, those params are used as defaults for URL
133
+ * generation. This enables auto-filling mount params from include() prefixes:
134
+ * inner handlers can call ctx.reverse(".sibling") without explicitly passing
135
+ * params that are already known from the current URL match.
136
+ * Explicitly passed hrefParams take priority over currentParams.
137
+ */
138
+ export function createReverseFunction(
139
+ routeMap: Record<string, string>,
140
+ currentRoutePrefix?: string,
141
+ currentParams?: Record<string, string>,
142
+ rootScoped?: boolean,
143
+ ): (
144
+ name: string,
145
+ hrefParams?: Record<string, string>,
146
+ search?: Record<string, unknown>,
147
+ ) => string {
148
+ return (name, hrefParams, search) => {
149
+ // Resolve route name with namespace support
150
+ const pattern = resolveRouteName(
151
+ name,
152
+ routeMap,
153
+ currentRoutePrefix,
154
+ rootScoped,
155
+ );
156
+
157
+ if (pattern === undefined) {
158
+ throw new Error(
159
+ `Unknown route: "${name}"${currentRoutePrefix ? ` (current route: ${currentRoutePrefix})` : ""}`,
160
+ );
161
+ }
162
+
163
+ // Merge current request params as defaults, explicit params override
164
+ const effectiveParams = currentParams
165
+ ? { ...currentParams, ...hrefParams }
166
+ : hrefParams;
167
+
168
+ let result = effectiveParams
169
+ ? substitutePatternParams(pattern, effectiveParams, name)
170
+ : pattern;
171
+
172
+ // Append search params as query string
173
+ if (search) {
174
+ const qs = serializeSearchParams(search);
175
+ if (qs) result += `?${qs}`;
176
+ }
177
+
178
+ return result;
179
+ };
180
+ }
181
+
59
182
  /**
60
183
  * Create HandlerContext with typed env/var/get/set
61
184
  */
@@ -65,49 +188,91 @@ export function createHandlerContext<TEnv>(
65
188
  searchParams: URLSearchParams,
66
189
  pathname: string,
67
190
  url: URL,
68
- bindings: any = {},
191
+ bindings: TEnv = {} as TEnv,
69
192
  routeMap: Record<string, string> = {},
70
- routeName?: string
193
+ routeName?: string,
194
+ responseType?: string,
195
+ isPassthroughRoute: boolean = false,
71
196
  ): InternalHandlerContext<any, TEnv> {
72
197
  // Get variables from request context - this is the unified context
73
198
  // shared between middleware and route handlers
74
- const requestContext = getRequestContext();
75
- const variables: any = requestContext?.var ?? {};
76
-
77
- // Filter system parameters (starting with _rsc) from searchParams
78
- // This ensures handlers only see user-facing query params
79
- const cleanSearchParams = new URLSearchParams();
80
- searchParams.forEach((value, key) => {
81
- if (!key.startsWith("_rsc")) {
82
- cleanSearchParams.set(key, value);
83
- }
84
- });
199
+ const requestContext = _getRequestContext();
200
+ const variables: any = requestContext?._variables ?? {};
85
201
 
86
- // Create clean URL without system params
87
- const cleanUrl = new URL(url);
88
- cleanUrl.search = cleanSearchParams.toString();
202
+ // If route has a search schema, parse URLSearchParams into typed object
203
+ const searchSchema = routeName ? getSearchSchema(routeName) : undefined;
204
+ const resolvedSearchParams = searchSchema
205
+ ? parseSearchParams(searchParams, searchSchema)
206
+ : searchParams;
89
207
 
90
208
  // Get stub response from request context for setting headers
91
- const stubResponse = requestContext?.res ?? new Response(null, { status: 200 });
209
+ const stubResponse =
210
+ requestContext?.res ?? new Response(null, { status: 200 });
92
211
 
93
- return {
212
+ // Guard mutating Headers methods so they throw inside "use cache" or cache() scope.
213
+ // Uses lazy `ctx` reference (assigned below) — only the specific handler ctx
214
+ // is stamped by cache-runtime, not the shared request context.
215
+ const MUTATING_HEADERS_METHODS = new Set(["set", "append", "delete"]);
216
+ let ctx: InternalHandlerContext<any, TEnv>;
217
+ const guardedHeaders = new Proxy(stubResponse.headers, {
218
+ get(target, prop, receiver) {
219
+ const value = Reflect.get(target, prop, receiver);
220
+ if (typeof value === "function") {
221
+ if (MUTATING_HEADERS_METHODS.has(prop as string)) {
222
+ return (...args: any[]) => {
223
+ assertNotInsideCacheExec(ctx, "headers");
224
+ if (isInsideCacheScope()) {
225
+ throw new Error(
226
+ `ctx.headers.${String(prop)}() cannot be called inside a cache() boundary. ` +
227
+ `On cache hit the handler is skipped, so this side effect would be lost. ` +
228
+ `Move header mutations to a middleware or layout outside the cache() scope.`,
229
+ );
230
+ }
231
+ return value.apply(target, args);
232
+ };
233
+ }
234
+ return value.bind(target);
235
+ }
236
+ return value;
237
+ },
238
+ });
239
+
240
+ ctx = {
94
241
  params,
242
+ build: false,
243
+ dev: false,
95
244
  request,
96
- searchParams: cleanSearchParams, // Filtered params
245
+ searchParams,
246
+ search: searchSchema ? resolvedSearchParams : {},
97
247
  pathname,
98
- url: cleanUrl, // Clean URL
248
+ url,
249
+ originalUrl: requestContext?.originalUrl ?? new URL(request.url),
99
250
  env: bindings,
100
- var: variables,
101
- get: ((key: string) => variables[key]) as HandlerContext<
102
- any,
103
- TEnv
104
- >["get"],
105
- set: ((key: string, value: any) => {
106
- variables[key] = value;
251
+ waitUntil: requestContext
252
+ ? requestContext.waitUntil.bind(requestContext)
253
+ : fireAndForgetWaitUntil,
254
+ executionContext: requestContext?.executionContext,
255
+ _variables: variables,
256
+ get: ((keyOrVar: any) => {
257
+ // Read-time guard: non-cacheable var inside cache() → throw.
258
+ // Works for both ContextVar tokens and string keys.
259
+ if (isNonCacheable(variables, keyOrVar) && isInsideCacheScope()) {
260
+ throw new Error(
261
+ `ctx.get() for a non-cacheable variable cannot be called inside a cache() boundary. ` +
262
+ `The variable was created with { cache: false } or set with { cache: false }, ` +
263
+ `and its value would be stale on cache hit. Move the read outside the cached scope.`,
264
+ );
265
+ }
266
+ return contextGet(variables, keyOrVar);
267
+ }) as HandlerContext<any, TEnv>["get"],
268
+ set: ((keyOrVar: any, value: any, options?: ContextSetOptions) => {
269
+ assertNotInsideCacheExec(ctx, "set");
270
+ // Write is dumb: store value + non-cacheable metadata.
271
+ // Enforcement happens at read time via ctx.get().
272
+ contextSet(variables, keyOrVar, value, options);
107
273
  }) as HandlerContext<any, TEnv>["set"],
108
- _originalRequest: request, // Raw request for advanced use
109
274
  res: stubResponse, // Stub response for setting headers
110
- headers: stubResponse.headers, // Shorthand for res.headers
275
+ headers: guardedHeaders, // Guarded shorthand for res.headers
111
276
  // Placeholder use() - will be replaced with actual implementation during request
112
277
  use: () => {
113
278
  throw new Error("ctx.use() called before loaders were initialized");
@@ -115,44 +280,211 @@ export function createHandlerContext<TEnv>(
115
280
  // Theme support (when enabled via router config)
116
281
  theme: requestContext?.theme,
117
282
  setTheme: requestContext?.setTheme,
118
- // Scoped reverse for URL generation
119
- reverse: (name: string, hrefParams?: Record<string, string>) => {
120
- // Path-based - return directly (optionally with param substitution)
121
- if (name.startsWith("/")) {
122
- if (hrefParams) {
123
- return name.replace(/:([^/]+)/g, (_, key) => {
124
- const value = hrefParams[key];
125
- if (value === undefined) {
126
- throw new Error(`Missing param "${key}" for path "${name}"`);
127
- }
128
- return encodeURIComponent(value);
129
- });
130
- }
131
- return name;
132
- }
133
-
134
- // Resolve route name with namespace support
135
- const pattern = resolveRouteName(name, routeMap, routeName);
136
-
137
- if (pattern === undefined) {
283
+ // Location state support (delegates to request context)
284
+ setLocationState(entries) {
285
+ if (!requestContext) {
138
286
  throw new Error(
139
- `Unknown route: "${name}"${routeName ? ` (current route: ${routeName})` : ""}`
287
+ "setLocationState() is not available outside a request context",
140
288
  );
141
289
  }
290
+ requestContext.setLocationState(entries);
291
+ },
292
+ routeName: (routeName && !isAutoGeneratedRouteName(routeName)
293
+ ? routeName
294
+ : undefined) as HandlerContext["routeName"],
295
+ // Scoped reverse for URL generation (auto-fills current request params).
296
+ // Resolve rootScoped eagerly so the reverse function is self-contained
297
+ // and does not depend on the global rootScopeRoutes registry at call time.
298
+ reverse: createReverseFunction(
299
+ routeMap,
300
+ routeName,
301
+ params,
302
+ routeName ? isRouteRootScoped(routeName) : undefined,
303
+ ),
304
+ passthrough: createPrerenderPassthroughFn(false, isPassthroughRoute),
305
+ _responseType: responseType,
306
+ _routeName: routeName,
307
+ };
308
+ // Brand with taint symbol so "use cache" excludes ctx from cache keys
309
+ (ctx as any)[NOCACHE_SYMBOL] = true;
310
+ return ctx;
311
+ }
142
312
 
143
- // If no params, return pattern directly
144
- if (!hrefParams) {
145
- return pattern;
146
- }
313
+ /**
314
+ * Create a PrerenderContext for Prerender() handlers at build time.
315
+ *
316
+ * Returns an InternalHandlerContext where params, pathname, url, searchParams,
317
+ * search, reverse, and use(handle) work. Request-time properties
318
+ * (request, env, headers, cookies, get, set, res) throw with a clear error.
319
+ */
320
+ export function createPrerenderContext<TEnv>(
321
+ params: Record<string, string>,
322
+ pathname: string,
323
+ routeMap: Record<string, string>,
324
+ routeName?: string,
325
+ buildVars?: Record<string, any>,
326
+ isPassthroughRoute?: boolean,
327
+ buildEnv?: TEnv,
328
+ devMode?: boolean,
329
+ ): InternalHandlerContext<any, TEnv> {
330
+ const syntheticUrl = new URL(`http://prerender${pathname}`);
331
+ const variables = buildVars ?? {};
147
332
 
148
- // Substitute params
149
- return pattern.replace(/:([^/]+)/g, (_, key) => {
150
- const value = hrefParams[key];
151
- if (value === undefined) {
152
- throw new Error(`Missing param "${key}" for route "${name}"`);
153
- }
154
- return encodeURIComponent(value);
155
- });
333
+ function throwUnavailable(prop: string): never {
334
+ throw new Error(
335
+ `Property "${prop}" is not available during pre-rendering. ` +
336
+ `Fetch data directly in the handler or use a passthrough prerender handler.`,
337
+ );
338
+ }
339
+
340
+ return {
341
+ params,
342
+ build: true,
343
+ dev: devMode ?? false,
344
+ get request(): Request {
345
+ return throwUnavailable("request");
156
346
  },
157
- };
347
+ searchParams: syntheticUrl.searchParams,
348
+ search: {},
349
+ pathname,
350
+ url: syntheticUrl,
351
+ originalUrl: syntheticUrl,
352
+ get env(): TEnv {
353
+ if (buildEnv !== undefined) return buildEnv;
354
+ throw new Error(
355
+ "ctx.env is not available during pre-rendering. " +
356
+ "Configure buildEnv in your rango() plugin options to enable build-time env access.",
357
+ );
358
+ },
359
+ // Build-time prerender has no live request. waitUntil is a true no-op
360
+ // (running fn() here would fire side effects during build, which is
361
+ // incorrect — these are meant to outlive the live response).
362
+ // executionContext is absent for the same reason.
363
+ waitUntil: () => {},
364
+ executionContext: undefined,
365
+ _variables: variables,
366
+ get: ((keyOrVar: any) => contextGet(variables, keyOrVar)) as any,
367
+ set: ((keyOrVar: any, value: any) => {
368
+ contextSet(variables, keyOrVar, value);
369
+ }) as any,
370
+ get res(): Response {
371
+ return throwUnavailable("res");
372
+ },
373
+ get headers(): Headers {
374
+ return throwUnavailable("headers");
375
+ },
376
+ // Placeholder use() - replaced by setupBuildUse
377
+ use: () => {
378
+ throw new Error("ctx.use() called before build context was initialized");
379
+ },
380
+ theme: undefined,
381
+ setTheme: undefined,
382
+ routeName: (routeName && !isAutoGeneratedRouteName(routeName)
383
+ ? routeName
384
+ : undefined) as HandlerContext["routeName"],
385
+ setLocationState: () => {
386
+ throwUnavailable("setLocationState");
387
+ },
388
+ reverse: createReverseFunction(
389
+ routeMap,
390
+ routeName,
391
+ params,
392
+ routeName ? isRouteRootScoped(routeName) : undefined,
393
+ ),
394
+ passthrough: createPrerenderPassthroughFn(
395
+ true,
396
+ isPassthroughRoute === true,
397
+ ),
398
+ _routeName: routeName,
399
+ } as InternalHandlerContext<any, TEnv>;
400
+ }
401
+
402
+ /**
403
+ * Create a StaticContext for Static() handlers at build time.
404
+ *
405
+ * Returns an InternalHandlerContext where only reverse and use(handle) work.
406
+ * Static handlers have no URL, no params, no pathname — everything else throws.
407
+ */
408
+ export function createStaticContext<TEnv>(
409
+ routeMap: Record<string, string>,
410
+ routeName?: string,
411
+ buildEnv?: TEnv,
412
+ devMode?: boolean,
413
+ ): InternalHandlerContext<any, TEnv> {
414
+ const variables: Record<string, any> = {};
415
+
416
+ function throwUnavailable(prop: string): never {
417
+ throw new Error(
418
+ `Property "${prop}" is not available in Static() handlers. ` +
419
+ `Static handlers render content without request context.`,
420
+ );
421
+ }
422
+
423
+ return {
424
+ get params(): any {
425
+ return throwUnavailable("params");
426
+ },
427
+ build: true,
428
+ dev: devMode ?? false,
429
+ get request(): Request {
430
+ return throwUnavailable("request");
431
+ },
432
+ get searchParams(): URLSearchParams {
433
+ return throwUnavailable("searchParams");
434
+ },
435
+ get search(): any {
436
+ return throwUnavailable("search");
437
+ },
438
+ get pathname(): string {
439
+ return throwUnavailable("pathname");
440
+ },
441
+ get url(): URL {
442
+ return throwUnavailable("url");
443
+ },
444
+ get originalUrl(): URL {
445
+ return throwUnavailable("originalUrl");
446
+ },
447
+ get env(): TEnv {
448
+ if (buildEnv !== undefined) return buildEnv;
449
+ throw new Error(
450
+ "ctx.env is not available in Static() handlers. " +
451
+ "Configure buildEnv in your rango() plugin options to enable build-time env access.",
452
+ );
453
+ },
454
+ // Static() handlers have no live request. waitUntil is a true no-op
455
+ // (running fn() here would fire side effects during build, which is
456
+ // incorrect). executionContext is absent for the same reason.
457
+ waitUntil: () => {},
458
+ executionContext: undefined,
459
+ _variables: variables,
460
+ get: ((keyOrVar: any) => contextGet(variables, keyOrVar)) as any,
461
+ set: ((keyOrVar: any, value: any) => {
462
+ contextSet(variables, keyOrVar, value);
463
+ }) as any,
464
+ get res(): Response {
465
+ return throwUnavailable("res");
466
+ },
467
+ get headers(): Headers {
468
+ return throwUnavailable("headers");
469
+ },
470
+ // Placeholder use() - replaced by setupBuildUse
471
+ use: () => {
472
+ throw new Error("ctx.use() called before build context was initialized");
473
+ },
474
+ theme: undefined,
475
+ setTheme: undefined,
476
+ routeName: (routeName && !isAutoGeneratedRouteName(routeName)
477
+ ? routeName
478
+ : undefined) as HandlerContext["routeName"],
479
+ setLocationState: () => {
480
+ throwUnavailable("setLocationState");
481
+ },
482
+ reverse: createReverseFunction(
483
+ routeMap,
484
+ routeName,
485
+ undefined,
486
+ routeName ? isRouteRootScoped(routeName) : undefined,
487
+ ),
488
+ _routeName: routeName,
489
+ } as InternalHandlerContext<any, TEnv>;
158
490
  }
@@ -11,12 +11,20 @@ import type {
11
11
  InterceptEntry,
12
12
  InterceptSelectorContext,
13
13
  } from "../server/context";
14
- import type { HandlerContext, ResolvedSegment } from "../types";
14
+ import type {
15
+ HandlerContext,
16
+ InternalHandlerContext,
17
+ ResolvedSegment,
18
+ } from "../types";
15
19
  import { evaluateRevalidation } from "./revalidation.js";
16
20
  import { getRequestContext } from "../server/request-context.js";
17
21
  import { executeInterceptMiddleware } from "./middleware.js";
22
+ import { createReverseFunction } from "./handler-context.js";
23
+ import { getGlobalRouteMap } from "../route-map-builder.js";
18
24
  import { handleHandlerResult } from "./segment-resolution.js";
19
25
  import type { SegmentResolutionDeps } from "./types.js";
26
+ import { debugLog } from "./logging.js";
27
+ import { runInsideLoaderScope } from "../server/context.js";
20
28
 
21
29
  /**
22
30
  * Check if an intercept's when conditions are satisfied.
@@ -130,8 +138,9 @@ export async function resolveInterceptEntry<TEnv>(
130
138
  context.request,
131
139
  context.env,
132
140
  params,
133
- context.var as Record<string, any>,
141
+ (context as InternalHandlerContext<any, TEnv>)._variables,
134
142
  requestCtx.res,
143
+ createReverseFunction(getGlobalRouteMap()),
135
144
  );
136
145
  if (middlewareResponse) throw middlewareResponse;
137
146
  }
@@ -184,24 +193,26 @@ export async function resolveInterceptEntry<TEnv>(
184
193
  context,
185
194
  actionContext,
186
195
  stale,
196
+ traceSource: "intercept-loader",
187
197
  });
188
198
 
189
199
  if (!shouldRevalidate) {
190
- console.log(
191
- `[Router] Intercept loader ${loader.$$id} skipped (revalidation=false)`,
192
- );
200
+ debugLog("intercept.loader", "skipped revalidation", {
201
+ loaderId: loader.$$id,
202
+ });
193
203
  continue;
194
204
  }
195
- console.log(
196
- `[Router] Intercept loader ${loader.$$id} revalidating (stale=${stale})`,
197
- );
205
+ debugLog("intercept.loader", "revalidating", {
206
+ loaderId: loader.$$id,
207
+ stale,
208
+ });
198
209
  }
199
210
  }
200
211
 
201
212
  loaderIds.push(loader.$$id);
202
213
  loaderPromises.push(
203
214
  deps.wrapLoaderPromise(
204
- context.use(loader),
215
+ runInsideLoaderScope(() => context.use(loader)),
205
216
  parentEntry,
206
217
  segmentId,
207
218
  context.pathname,
@@ -350,23 +361,25 @@ export async function resolveInterceptLoadersOnly<TEnv>(
350
361
  context,
351
362
  actionContext,
352
363
  stale,
364
+ traceSource: "intercept-loader",
353
365
  });
354
366
 
355
367
  if (!shouldRevalidate) {
356
- console.log(
357
- `[Router] Intercept loader ${loader.$$id} skipped (cache hit, revalidation=false)`,
358
- );
368
+ debugLog("intercept.loader", "skipped on cache hit", {
369
+ loaderId: loader.$$id,
370
+ });
359
371
  continue;
360
372
  }
361
- console.log(
362
- `[Router] Intercept loader ${loader.$$id} revalidating on cache hit (stale=${stale})`,
363
- );
373
+ debugLog("intercept.loader", "revalidating on cache hit", {
374
+ loaderId: loader.$$id,
375
+ stale,
376
+ });
364
377
  }
365
378
 
366
379
  loaderIds.push(loader.$$id);
367
380
  loaderPromises.push(
368
381
  deps.wrapLoaderPromise(
369
- context.use(loader),
382
+ runInsideLoaderScope(() => context.use(loader)),
370
383
  parentEntry,
371
384
  segmentId,
372
385
  context.pathname,
@@ -378,10 +391,12 @@ export async function resolveInterceptLoadersOnly<TEnv>(
378
391
  return null;
379
392
  }
380
393
 
381
- const loaderDataPromise =
382
- interceptEntry.loading !== undefined
383
- ? Promise.all(loaderPromises)
384
- : await Promise.all(loaderPromises);
394
+ // Match fresh-path semantics: only defer (no await) when loading is truthy.
395
+ // `loading: false` means "no loading UI, await loaders before render" —
396
+ // same as the fresh path's `if (interceptEntry.loading && ...)` check.
397
+ const loaderDataPromise = interceptEntry.loading
398
+ ? Promise.all(loaderPromises)
399
+ : await Promise.all(loaderPromises);
385
400
 
386
401
  return { loaderDataPromise, loaderIds };
387
402
  }