@rangojs/router 0.0.0-experimental.b9cb8739 → 0.0.0-experimental.bd6e11bc

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 (285) hide show
  1. package/README.md +196 -43
  2. package/dist/bin/rango.js +277 -99
  3. package/dist/testing/vitest.js +48 -0
  4. package/dist/vite/index.js +2779 -1064
  5. package/dist/vite/index.js.bak +5448 -0
  6. package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  7. package/package.json +57 -11
  8. package/skills/breadcrumbs/SKILL.md +3 -1
  9. package/skills/bundle-analysis/SKILL.md +159 -0
  10. package/skills/cache-guide/SKILL.md +243 -21
  11. package/skills/caching/SKILL.md +155 -6
  12. package/skills/composability/SKILL.md +27 -2
  13. package/skills/document-cache/SKILL.md +78 -55
  14. package/skills/handler-use/SKILL.md +364 -0
  15. package/skills/hooks/SKILL.md +229 -20
  16. package/skills/host-router/SKILL.md +45 -20
  17. package/skills/i18n/SKILL.md +276 -0
  18. package/skills/intercept/SKILL.md +46 -4
  19. package/skills/layout/SKILL.md +28 -7
  20. package/skills/links/SKILL.md +249 -17
  21. package/skills/loader/SKILL.md +273 -53
  22. package/skills/middleware/SKILL.md +49 -12
  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 +27 -0
  26. package/skills/observability/SKILL.md +137 -0
  27. package/skills/parallel/SKILL.md +197 -6
  28. package/skills/prerender/SKILL.md +123 -100
  29. package/skills/rango/SKILL.md +242 -22
  30. package/skills/react-compiler/SKILL.md +168 -0
  31. package/skills/response-routes/SKILL.md +66 -9
  32. package/skills/route/SKILL.md +88 -4
  33. package/skills/router-setup/SKILL.md +90 -5
  34. package/skills/server-actions/SKILL.md +751 -0
  35. package/skills/streams-and-websockets/SKILL.md +283 -0
  36. package/skills/testing/SKILL.md +716 -0
  37. package/skills/typesafety/SKILL.md +329 -27
  38. package/skills/use-cache/SKILL.md +34 -5
  39. package/skills/view-transitions/SKILL.md +294 -0
  40. package/src/__augment-tests__/augment.ts +81 -0
  41. package/src/__augment-tests__/augmented.check.ts +117 -0
  42. package/src/__internal.ts +1 -1
  43. package/src/browser/action-coordinator.ts +53 -36
  44. package/src/browser/app-shell.ts +52 -0
  45. package/src/browser/app-version.ts +14 -0
  46. package/src/browser/event-controller.ts +91 -70
  47. package/src/browser/history-state.ts +21 -0
  48. package/src/browser/index.ts +3 -3
  49. package/src/browser/navigation-bridge.ts +102 -16
  50. package/src/browser/navigation-client.ts +164 -59
  51. package/src/browser/navigation-store.ts +75 -17
  52. package/src/browser/navigation-transaction.ts +21 -37
  53. package/src/browser/partial-update.ts +139 -38
  54. package/src/browser/prefetch/cache.ts +175 -15
  55. package/src/browser/prefetch/fetch.ts +180 -33
  56. package/src/browser/prefetch/queue.ts +123 -20
  57. package/src/browser/prefetch/resource-ready.ts +77 -0
  58. package/src/browser/rango-state.ts +53 -13
  59. package/src/browser/react/Link.tsx +81 -9
  60. package/src/browser/react/NavigationProvider.tsx +110 -33
  61. package/src/browser/react/context.ts +7 -2
  62. package/src/browser/react/filter-segment-order.ts +51 -7
  63. package/src/browser/react/index.ts +3 -0
  64. package/src/browser/react/location-state-shared.ts +175 -4
  65. package/src/browser/react/location-state.ts +39 -13
  66. package/src/browser/react/use-handle.ts +23 -64
  67. package/src/browser/react/use-navigation.ts +22 -2
  68. package/src/browser/react/use-params.ts +20 -8
  69. package/src/browser/react/use-reverse.ts +106 -0
  70. package/src/browser/react/use-router.ts +43 -10
  71. package/src/browser/react/use-segments.ts +11 -8
  72. package/src/browser/response-adapter.ts +25 -0
  73. package/src/browser/rsc-router.tsx +191 -74
  74. package/src/browser/scroll-restoration.ts +41 -14
  75. package/src/browser/segment-reconciler.ts +36 -9
  76. package/src/browser/segment-structure-assert.ts +2 -2
  77. package/src/browser/server-action-bridge.ts +31 -36
  78. package/src/browser/types.ts +57 -5
  79. package/src/build/collect-fallback-refs.ts +107 -0
  80. package/src/build/generate-manifest.ts +65 -40
  81. package/src/build/generate-route-types.ts +5 -0
  82. package/src/build/index.ts +2 -0
  83. package/src/build/route-trie.ts +52 -25
  84. package/src/build/route-types/codegen.ts +4 -4
  85. package/src/build/route-types/include-resolution.ts +9 -2
  86. package/src/build/route-types/per-module-writer.ts +7 -4
  87. package/src/build/route-types/router-processing.ts +278 -88
  88. package/src/build/route-types/scan-filter.ts +9 -2
  89. package/src/build/route-types/source-scan.ts +118 -0
  90. package/src/build/runtime-discovery.ts +9 -20
  91. package/src/cache/cache-runtime.ts +15 -11
  92. package/src/cache/cache-scope.ts +76 -49
  93. package/src/cache/cf/cf-cache-store.ts +501 -18
  94. package/src/cache/cf/index.ts +5 -1
  95. package/src/cache/document-cache.ts +17 -7
  96. package/src/cache/index.ts +1 -0
  97. package/src/cache/taint.ts +55 -0
  98. package/src/client.rsc.tsx +3 -0
  99. package/src/client.tsx +94 -238
  100. package/src/context-var.ts +72 -2
  101. package/src/debug.ts +2 -2
  102. package/src/decode-loader-results.ts +36 -0
  103. package/src/errors.ts +30 -1
  104. package/src/handle.ts +65 -12
  105. package/src/host/index.ts +2 -2
  106. package/src/host/router.ts +129 -57
  107. package/src/host/types.ts +31 -2
  108. package/src/host/utils.ts +1 -1
  109. package/src/href-client.ts +140 -20
  110. package/src/index.rsc.ts +12 -5
  111. package/src/index.ts +61 -11
  112. package/src/loader-store.ts +500 -0
  113. package/src/loader.rsc.ts +2 -5
  114. package/src/loader.ts +3 -10
  115. package/src/missing-id-error.ts +68 -0
  116. package/src/outlet-context.ts +1 -1
  117. package/src/prerender/store.ts +5 -4
  118. package/src/prerender.ts +141 -80
  119. package/src/response-utils.ts +37 -0
  120. package/src/reverse.ts +65 -15
  121. package/src/route-content-wrapper.tsx +6 -28
  122. package/src/route-definition/dsl-helpers.ts +435 -260
  123. package/src/route-definition/helper-factories.ts +29 -139
  124. package/src/route-definition/helpers-types.ts +110 -34
  125. package/src/route-definition/index.ts +3 -0
  126. package/src/route-definition/redirect.ts +11 -3
  127. package/src/route-definition/resolve-handler-use.ts +155 -0
  128. package/src/route-definition/use-item-types.ts +32 -0
  129. package/src/route-map-builder.ts +7 -1
  130. package/src/route-types.ts +37 -41
  131. package/src/router/basename.ts +14 -0
  132. package/src/router/content-negotiation.ts +113 -1
  133. package/src/router/error-handling.ts +1 -1
  134. package/src/router/find-match.ts +4 -2
  135. package/src/router/handler-context.ts +77 -38
  136. package/src/router/intercept-resolution.ts +15 -22
  137. package/src/router/lazy-includes.ts +12 -9
  138. package/src/router/loader-resolution.ts +174 -22
  139. package/src/router/logging.ts +5 -2
  140. package/src/router/manifest.ts +31 -16
  141. package/src/router/match-api.ts +128 -192
  142. package/src/router/match-handlers.ts +63 -20
  143. package/src/router/match-middleware/background-revalidation.ts +30 -2
  144. package/src/router/match-middleware/cache-lookup.ts +136 -106
  145. package/src/router/match-middleware/cache-store.ts +54 -10
  146. package/src/router/match-middleware/intercept-resolution.ts +9 -7
  147. package/src/router/match-middleware/segment-resolution.ts +61 -5
  148. package/src/router/match-result.ts +125 -10
  149. package/src/router/metrics.ts +7 -2
  150. package/src/router/middleware-types.ts +21 -34
  151. package/src/router/middleware.ts +103 -90
  152. package/src/router/navigation-snapshot.ts +182 -0
  153. package/src/router/pattern-matching.ts +101 -17
  154. package/src/router/prerender-match.ts +110 -10
  155. package/src/router/preview-match.ts +32 -102
  156. package/src/router/request-classification.ts +286 -0
  157. package/src/router/revalidation.ts +58 -2
  158. package/src/router/route-snapshot.ts +245 -0
  159. package/src/router/router-context.ts +6 -1
  160. package/src/router/router-interfaces.ts +77 -28
  161. package/src/router/router-options.ts +76 -11
  162. package/src/router/router-registry.ts +2 -5
  163. package/src/router/segment-resolution/fresh.ts +223 -24
  164. package/src/router/segment-resolution/helpers.ts +29 -24
  165. package/src/router/segment-resolution/loader-cache.ts +1 -0
  166. package/src/router/segment-resolution/revalidation.ts +466 -285
  167. package/src/router/segment-resolution/view-transition-default.ts +36 -0
  168. package/src/router/segment-wrappers.ts +2 -0
  169. package/src/router/substitute-pattern-params.ts +56 -0
  170. package/src/router/telemetry.ts +99 -0
  171. package/src/router/trie-matching.ts +18 -13
  172. package/src/router/types.ts +9 -0
  173. package/src/router/url-params.ts +49 -0
  174. package/src/router.ts +91 -23
  175. package/src/rsc/handler-context.ts +2 -2
  176. package/src/rsc/handler.ts +440 -381
  177. package/src/rsc/helpers.ts +91 -43
  178. package/src/rsc/index.ts +1 -1
  179. package/src/rsc/loader-fetch.ts +23 -3
  180. package/src/rsc/manifest-init.ts +5 -1
  181. package/src/rsc/origin-guard.ts +28 -10
  182. package/src/rsc/progressive-enhancement.ts +18 -2
  183. package/src/rsc/response-route-handler.ts +46 -53
  184. package/src/rsc/rsc-rendering.ts +41 -48
  185. package/src/rsc/runtime-warnings.ts +9 -10
  186. package/src/rsc/server-action.ts +25 -37
  187. package/src/rsc/ssr-setup.ts +18 -2
  188. package/src/rsc/types.ts +17 -3
  189. package/src/search-params.ts +4 -4
  190. package/src/segment-content-promise.ts +67 -0
  191. package/src/segment-loader-promise.ts +122 -0
  192. package/src/segment-system.tsx +219 -67
  193. package/src/serialize.ts +243 -0
  194. package/src/server/context.ts +277 -61
  195. package/src/server/cookie-store.ts +28 -4
  196. package/src/server/handle-store.ts +19 -0
  197. package/src/server/loader-registry.ts +9 -8
  198. package/src/server/request-context.ts +204 -60
  199. package/src/ssr/index.tsx +9 -1
  200. package/src/static-handler.ts +19 -7
  201. package/src/testing/cache-status.ts +166 -0
  202. package/src/testing/collect-handle.ts +63 -0
  203. package/src/testing/dispatch.ts +440 -0
  204. package/src/testing/dom.entry.ts +22 -0
  205. package/src/testing/e2e/fixture.ts +154 -0
  206. package/src/testing/e2e/index.ts +149 -0
  207. package/src/testing/e2e/matchers.ts +51 -0
  208. package/src/testing/e2e/page-helpers.ts +272 -0
  209. package/src/testing/e2e/parity.ts +306 -0
  210. package/src/testing/e2e/server.ts +183 -0
  211. package/src/testing/flight-matchers.ts +104 -0
  212. package/src/testing/flight-runtime.d.ts +21 -0
  213. package/src/testing/flight.entry.ts +22 -0
  214. package/src/testing/flight.ts +182 -0
  215. package/src/testing/generated-routes.ts +223 -0
  216. package/src/testing/index.ts +106 -0
  217. package/src/testing/internal/context.ts +255 -0
  218. package/src/testing/render-route.tsx +565 -0
  219. package/src/testing/run-loader.ts +296 -0
  220. package/src/testing/run-middleware.ts +179 -0
  221. package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
  222. package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
  223. package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
  224. package/src/testing/vitest-stubs/version.ts +5 -0
  225. package/src/testing/vitest.ts +183 -0
  226. package/src/types/cache-types.ts +4 -4
  227. package/src/types/global-namespace.ts +39 -26
  228. package/src/types/handler-context.ts +194 -72
  229. package/src/types/index.ts +1 -0
  230. package/src/types/loader-types.ts +41 -15
  231. package/src/types/request-scope.ts +126 -0
  232. package/src/types/route-entry.ts +19 -1
  233. package/src/types/segments.ts +37 -1
  234. package/src/urls/include-helper.ts +34 -67
  235. package/src/urls/index.ts +0 -3
  236. package/src/urls/path-helper-types.ts +50 -9
  237. package/src/urls/path-helper.ts +63 -63
  238. package/src/urls/pattern-types.ts +48 -19
  239. package/src/urls/response-types.ts +25 -22
  240. package/src/urls/type-extraction.ts +26 -116
  241. package/src/urls/urls-function.ts +1 -5
  242. package/src/use-loader.tsx +487 -44
  243. package/src/vite/debug.ts +185 -0
  244. package/src/vite/discovery/bundle-postprocess.ts +34 -37
  245. package/src/vite/discovery/discover-routers.ts +105 -51
  246. package/src/vite/discovery/discovery-errors.ts +194 -0
  247. package/src/vite/discovery/gate-state.ts +171 -0
  248. package/src/vite/discovery/prerender-collection.ts +188 -93
  249. package/src/vite/discovery/route-types-writer.ts +40 -84
  250. package/src/vite/discovery/self-gen-tracking.ts +27 -1
  251. package/src/vite/discovery/state.ts +46 -6
  252. package/src/vite/discovery/virtual-module-codegen.ts +13 -23
  253. package/src/vite/index.ts +6 -0
  254. package/src/vite/plugin-types.ts +111 -72
  255. package/src/vite/plugins/cjs-to-esm.ts +8 -7
  256. package/src/vite/plugins/client-ref-dedup.ts +16 -0
  257. package/src/vite/plugins/client-ref-hashing.ts +28 -5
  258. package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
  259. package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  260. package/src/vite/plugins/cloudflare-protocol-stub.ts +214 -0
  261. package/src/vite/plugins/expose-action-id.ts +55 -33
  262. package/src/vite/plugins/expose-id-utils.ts +24 -8
  263. package/src/vite/plugins/expose-ids/export-analysis.ts +100 -20
  264. package/src/vite/plugins/expose-ids/handler-transform.ts +12 -35
  265. package/src/vite/plugins/expose-ids/loader-transform.ts +3 -5
  266. package/src/vite/plugins/expose-ids/router-transform.ts +20 -3
  267. package/src/vite/plugins/expose-internal-ids.ts +544 -317
  268. package/src/vite/plugins/performance-tracks.ts +92 -0
  269. package/src/vite/plugins/refresh-cmd.ts +88 -26
  270. package/src/vite/plugins/use-cache-transform.ts +65 -50
  271. package/src/vite/plugins/version-injector.ts +39 -23
  272. package/src/vite/plugins/version-plugin.ts +72 -3
  273. package/src/vite/plugins/virtual-entries.ts +2 -2
  274. package/src/vite/rango.ts +265 -226
  275. package/src/vite/router-discovery.ts +920 -137
  276. package/src/vite/utils/ast-handler-extract.ts +15 -15
  277. package/src/vite/utils/banner.ts +4 -4
  278. package/src/vite/utils/bundle-analysis.ts +4 -2
  279. package/src/vite/utils/client-chunks.ts +190 -0
  280. package/src/vite/utils/forward-user-plugins.ts +193 -0
  281. package/src/vite/utils/manifest-utils.ts +21 -5
  282. package/src/vite/utils/package-resolution.ts +41 -1
  283. package/src/vite/utils/prerender-utils.ts +38 -5
  284. package/src/vite/utils/shared-utils.ts +109 -27
  285. package/src/browser/action-response-classifier.ts +0 -99
@@ -15,6 +15,7 @@
15
15
  */
16
16
 
17
17
  import type { GetRegisteredRoutes } from "./types.js";
18
+ import type { JsonSerialize } from "./serialize.js";
18
19
  import type { ResponseEnvelope } from "./urls.js";
19
20
 
20
21
  /**
@@ -103,29 +104,75 @@ type NameForPattern<TPattern extends string, TRoutes = GetRegisteredRoutes> = {
103
104
  }[keyof TRoutes];
104
105
 
105
106
  /**
106
- * Look up the response data type for a route pattern from RegisteredRoutes.
107
- *
108
- * Works by reverse-looking up the route name for the given pattern,
109
- * then extracting the response type from the route entry.
107
+ * Strip a query (`?…`) and/or hash (`#…`) suffix before matching, so a concrete
108
+ * URL like `/api/health?ts=1` still resolves to its route's response. Removes
109
+ * from the earliest of `?`/`#`: a `#` before the first `?` (the query is part of
110
+ * a fragment, e.g. `/health#top?x=1`) is handled, as is a `/:` that only appears
111
+ * inside the query (e.g. `/health?next=/:id`).
112
+ */
113
+ type StripPathSuffix<T extends string> = T extends `${infer Base}?${string}`
114
+ ? Base extends `${infer Frag}#${string}`
115
+ ? Frag
116
+ : Base
117
+ : T extends `${infer Base}#${string}`
118
+ ? Base
119
+ : T;
120
+
121
+ /** Extract a route entry's response payload (or `never` for RSC routes). */
122
+ type ResponsePayloadOf<TRoutes, K extends keyof TRoutes> = TRoutes[K] extends {
123
+ readonly response: infer R;
124
+ }
125
+ ? Exclude<R, Response>
126
+ : never;
127
+
128
+ /**
129
+ * Look up the response payload for a route, keyed by either a route pattern
130
+ * (`/api/products/:id`) or a concrete path (`/api/products/123`). The same type
131
+ * serves a pattern lookup and a typed `fetch` wrapper that forwards a concrete
132
+ * `Rango.Path`:
110
133
  *
111
- * For static routes (no params), pattern === path:
112
- * PathResponse<"/api/health"> → { status: string; timestamp: number }
134
+ * PathResponse<"/api/products/:id"> Product // by pattern
135
+ * PathResponse<"/api/products/123"> → Product // by concrete path
113
136
  *
114
- * For dynamic routes, use the pattern:
115
- * PathResponse<"/api/products/:id"> Product
137
+ * The query/hash suffix is stripped first; the stripped key is then treated as a
138
+ * pattern when it contains a `/:param` segment and matched exactly (precise even
139
+ * for nested dynamic routes), otherwise as a concrete path matched against each
140
+ * route's `PatternToPath` template. Because those holes are `${string}`
141
+ * (slash-greedy), a concrete path under a *nested* dynamic route can match several
142
+ * patterns and union their responses — pattern lookups do not have this
143
+ * looseness. RSC routes (no response) and unmatched keys resolve to `never`.
144
+ */
145
+ type ResponsePayloadFor<
146
+ TPath extends string,
147
+ TRoutes = GetRegisteredRoutes,
148
+ > = ResponsePayloadForKey<StripPathSuffix<TPath>, TRoutes>;
149
+
150
+ type ResponsePayloadForKey<
151
+ TKey extends string,
152
+ TRoutes,
153
+ > = TKey extends `${string}/:${string}`
154
+ ? {
155
+ [K in keyof TRoutes]: RoutePattern<TRoutes, K> extends TKey
156
+ ? ResponsePayloadOf<TRoutes, K>
157
+ : never;
158
+ }[keyof TRoutes]
159
+ : {
160
+ [K in keyof TRoutes]: TKey extends PatternToPath<RoutePattern<TRoutes, K>>
161
+ ? ResponsePayloadOf<TRoutes, K>
162
+ : never;
163
+ }[keyof TRoutes];
164
+
165
+ /**
166
+ * Public response type for a route, keyed by pattern or concrete path. The
167
+ * payload is wrapped in `JsonSerialize` so it describes the JSON **wire** value a
168
+ * consumer receives from `fetch().then(r => r.json())`, not the handler's raw
169
+ * return type — e.g. a handler returning `{ createdAt: Date }` resolves here to
170
+ * `ResponseEnvelope<{ createdAt: string }>`.
116
171
  */
117
172
  export type PathResponse<
118
- TPattern extends string,
173
+ TPath extends string,
119
174
  TRoutes = GetRegisteredRoutes,
120
- > = ResponseEnvelope<
121
- {
122
- [K in keyof TRoutes]: RoutePattern<TRoutes, K> extends TPattern
123
- ? TRoutes[K] extends { readonly response: infer R }
124
- ? Exclude<R, Response>
125
- : never
126
- : never;
127
- }[keyof TRoutes]
128
- >;
175
+ > = ResponseEnvelope<JsonSerialize<ResponsePayloadFor<TPath, TRoutes>>>;
129
176
 
130
177
  /**
131
178
  * Strip trailing slash from a path (e.g., "/blog/" -> "/blog" | "/blog/")
@@ -140,7 +187,7 @@ type OptionalTrailingSlash<T extends string> = T extends `${infer Base}/`
140
187
  /**
141
188
  * Union of all valid paths from registered routes
142
189
  *
143
- * Generated from RSCRouter.RegisteredRoutes via module augmentation.
190
+ * Generated from Rango.RegisteredRoutes via module augmentation.
144
191
  * Allows optional query strings and hash fragments.
145
192
  */
146
193
  export type ValidPaths<TRoutes = GetRegisteredRoutes> =
@@ -154,6 +201,76 @@ export type ValidPaths<TRoutes = GetRegisteredRoutes> =
154
201
  }[keyof TRoutes]
155
202
  >;
156
203
 
204
+ // Module-scoped alias so the ambient `Rango.PathResponse` below can reference
205
+ // the module-level `PathResponse` without the global namespace shadowing the
206
+ // name when both are called `PathResponse`.
207
+ type GlobalPathResponse<
208
+ TPattern extends string,
209
+ TRoutes = GetRegisteredRoutes,
210
+ > = PathResponse<TPattern, TRoutes>;
211
+
212
+ /**
213
+ * Ambient path types on the `Rango` namespace.
214
+ *
215
+ * These live on the same global namespace consumers already augment for
216
+ * `Rango.Env` / `Rango.Vars`, so they are reachable with no import wherever the
217
+ * router's types are in scope. They are the public, recommended surface for
218
+ * typing anything that wraps `href()`. `ValidPaths` / `PathResponse` stay as the
219
+ * internal building blocks behind them.
220
+ */
221
+ declare global {
222
+ namespace Rango {
223
+ /**
224
+ * Union of every valid route path accepted by `href()`.
225
+ *
226
+ * Type a wrapper's path parameter as `Rango.Path` so it shares `href()`'s
227
+ * compile-time validation against the registered routes:
228
+ *
229
+ * ```ts
230
+ * import { href } from "@rangojs/router/client";
231
+ *
232
+ * export const appHref = (path: Rango.Path) => href(path);
233
+ * ```
234
+ *
235
+ * Resolves from `Rango.RegisteredRoutes` when augmented, otherwise the
236
+ * auto-generated `Rango.GeneratedRouteMap`, otherwise a permissive
237
+ * `/${string}` fallback.
238
+ */
239
+ type Path<TRoutes = GetRegisteredRoutes> = ValidPaths<TRoutes>;
240
+
241
+ /**
242
+ * Response payload for a route, looked up from the global route map by
243
+ * either a route pattern (`/api/products/:id`) or a concrete path
244
+ * (`/api/products/123`). Because it accepts a concrete `Rango.Path`, it
245
+ * doubles as the return type of a typed `fetch` wrapper:
246
+ *
247
+ * ```ts
248
+ * type Product = Rango.PathResponse<"/api/products/:id">; // by pattern
249
+ * type Same = Rango.PathResponse<"/api/products/42">; // by concrete path
250
+ *
251
+ * const get = async <T extends Rango.Path>(
252
+ * path: T,
253
+ * ): Promise<Rango.PathResponse<T>> =>
254
+ * fetch(href(path)).then((r) => r.json());
255
+ * ```
256
+ *
257
+ * The payload is the JSON **wire** shape (via `Rango.JsonSerialize`), not the
258
+ * handler's raw return — a handler returning `{ createdAt: Date }` resolves
259
+ * here to `ResponseEnvelope<{ createdAt: string }>`, matching what
260
+ * `fetch().then(r => r.json())` actually yields.
261
+ *
262
+ * Only resolves once `Rango.RegisteredRoutes` carries response metadata (the
263
+ * generated map has paths and search but no payloads). Pass an explicit route
264
+ * map as the second argument to look up against a non-global map (rarely
265
+ * needed in app code).
266
+ */
267
+ type PathResponse<
268
+ TPath extends string,
269
+ TRoutes = GetRegisteredRoutes,
270
+ > = GlobalPathResponse<TPath, TRoutes>;
271
+ }
272
+ }
273
+
157
274
  /**
158
275
  * Type-safe href function for client-side use
159
276
  *
@@ -186,7 +303,10 @@ export function href<T extends ValidPaths>(path: T, mount?: string): string {
186
303
  const normalizedMount = mount.endsWith("/") ? mount.slice(0, -1) : mount;
187
304
  return normalizedMount + path;
188
305
  }
189
- return path;
306
+ // ValidPaths is built from template literals so T does extend string at
307
+ // runtime, but the inference can fail past a certain route-union complexity
308
+ // and TypeScript reports T as not assignable to string.
309
+ return path as string;
190
310
  }
191
311
 
192
312
  /**
package/src/index.rsc.ts CHANGED
@@ -18,6 +18,7 @@ export {
18
18
  MiddlewareError,
19
19
  HandlerError,
20
20
  BuildError,
21
+ DslContextError,
21
22
  InvalidHandlerError,
22
23
  RouterError,
23
24
  Skip,
@@ -43,6 +44,7 @@ export type {
43
44
  // Revalidation types
44
45
  RevalidateParams,
45
46
  Revalidate,
47
+ ActionRef,
46
48
  RouteKeys,
47
49
  // Loader types
48
50
  LoaderDefinition,
@@ -67,7 +69,7 @@ export type {
67
69
 
68
70
  // Router options type (server-only, so import directly)
69
71
  export type {
70
- RSCRouterOptions,
72
+ RangoOptions,
71
73
  SSRStreamMode,
72
74
  SSROptions,
73
75
  ResolveStreamingContext,
@@ -100,6 +102,7 @@ export type {
100
102
  LayoutUseItem,
101
103
  AllUseItems,
102
104
  UseItems,
105
+ HandlerUseItem,
103
106
  } from "./route-types.js";
104
107
 
105
108
  // Handle API
@@ -114,8 +117,9 @@ export { nonce } from "./rsc/nonce.js";
114
117
  // Pre-render handler API
115
118
  export {
116
119
  Prerender,
120
+ Passthrough,
117
121
  type PrerenderHandlerDefinition,
118
- type PrerenderPassthroughContext,
122
+ type PassthroughHandlerDefinition,
119
123
  type PrerenderOptions,
120
124
  type BuildContext,
121
125
  type StaticBuildContext,
@@ -149,7 +153,7 @@ export {
149
153
  // Core router (server-side)
150
154
  export {
151
155
  createRouter,
152
- type RSCRouter,
156
+ type Rango,
153
157
  type RootLayoutProps,
154
158
  type RouterRequestInput,
155
159
  } from "./router.js";
@@ -170,6 +174,9 @@ export type { PublicRequestContext as RequestContext } from "./server/request-co
170
174
  import type { PublicRequestContext } from "./server/request-context.js";
171
175
  import type { DefaultEnv } from "./types/global-namespace.js";
172
176
 
177
+ // Shared base for every user-facing request context (mirrors index.ts).
178
+ export type { RequestScope, ExecutionContext } from "./types/request-scope.js";
179
+
173
180
  export const getRequestContext: <
174
181
  TEnv = DefaultEnv,
175
182
  >() => PublicRequestContext<TEnv> = _getRequestContextInternal;
@@ -215,8 +222,8 @@ export {
215
222
  type LocationStateOptions,
216
223
  } from "./browser/react/location-state-shared.js";
217
224
 
218
- // Path-based response type lookup from RegisteredRoutes
219
- export type { PathResponse } from "./href-client.js";
225
+ // Path and response types are ambient on the `Rango` namespace (`Rango.Path`,
226
+ // `Rango.PathResponse`, declared in href-client.ts) — no import needed.
220
227
 
221
228
  // Telemetry sink
222
229
  export { createConsoleSink } from "./router/telemetry.js";
package/src/index.ts CHANGED
@@ -18,6 +18,7 @@ export {
18
18
  MiddlewareError,
19
19
  HandlerError,
20
20
  BuildError,
21
+ DslContextError,
21
22
  InvalidHandlerError,
22
23
  RouterError,
23
24
  Skip,
@@ -43,6 +44,7 @@ export type {
43
44
  // Revalidation types
44
45
  RevalidateParams,
45
46
  Revalidate,
47
+ ActionRef,
46
48
  RouteKeys,
47
49
  // Loader types
48
50
  LoaderDefinition,
@@ -88,6 +90,7 @@ export type {
88
90
  LayoutUseItem,
89
91
  AllUseItems,
90
92
  UseItems,
93
+ HandlerUseItem,
91
94
  } from "./route-types.js";
92
95
 
93
96
  // Response route types (usable in both server and client contexts)
@@ -146,17 +149,52 @@ export { createVar, type ContextVar } from "./context-var.js";
146
149
  export { nonce } from "./rsc/nonce.js";
147
150
 
148
151
  /**
149
- * Error-throwing stub for server-only `Prerender` function.
152
+ * SSR/client stub for server-only `Prerender` function.
153
+ *
154
+ * Returns a lightweight stub object instead of throwing so that the
155
+ * production SSR build can safely bundle the RSC entry chunk — the SSR
156
+ * bundler resolves `@rangojs/router` to this (SSR) entry, so Prerender
157
+ * calls in RSC code must not crash at module-evaluation time.
158
+ */
159
+ export function Prerender(
160
+ _handler?: any,
161
+ _optionsOrId?: any,
162
+ __injectedId?: string,
163
+ ): any {
164
+ const id =
165
+ typeof _optionsOrId === "string" ? _optionsOrId : __injectedId || "";
166
+ return { __brand: "prerenderHandler" as const, $$id: id };
167
+ }
168
+
169
+ /**
170
+ * SSR/client stub for server-only `Passthrough` function.
150
171
  */
151
- export function Prerender(): never {
152
- throw serverOnlyStubError("Prerender");
172
+ export function Passthrough(
173
+ _handler?: any,
174
+ _optionsOrId?: any,
175
+ __injectedId?: string,
176
+ ): any {
177
+ const id =
178
+ typeof _optionsOrId === "string" ? _optionsOrId : __injectedId || "";
179
+ return { __brand: "passthroughHandler" as const, $$id: id };
153
180
  }
154
181
 
155
182
  /**
156
- * Error-throwing stub for server-only `Static` function.
183
+ * SSR/client stub for server-only `Static` function.
184
+ *
185
+ * Returns a lightweight stub object instead of throwing so that the
186
+ * production SSR build can safely bundle the RSC entry chunk — the SSR
187
+ * bundler resolves `@rangojs/router` to this (SSR) entry, so Static
188
+ * calls in RSC code must not crash at module-evaluation time.
157
189
  */
158
- export function Static(): never {
159
- throw serverOnlyStubError("Static");
190
+ export function Static(
191
+ _handler?: any,
192
+ _optionsOrId?: any,
193
+ __injectedId?: string,
194
+ ): any {
195
+ const id =
196
+ typeof _optionsOrId === "string" ? _optionsOrId : __injectedId || "";
197
+ return { __brand: "staticHandler" as const, $$id: id };
160
198
  }
161
199
 
162
200
  /**
@@ -228,6 +266,9 @@ export function transition(): never {
228
266
  // Request context type (safe for client)
229
267
  export type { PublicRequestContext as RequestContext } from "./server/request-context.js";
230
268
 
269
+ // Shared base for every user-facing request context.
270
+ export type { RequestScope, ExecutionContext } from "./types/request-scope.js";
271
+
231
272
  // Cookie store types (safe for client)
232
273
  export type {
233
274
  CookieStore,
@@ -235,6 +276,10 @@ export type {
235
276
  ReadonlyHeaders,
236
277
  } from "./server/cookie-store.js";
237
278
 
279
+ // Built-in handles (universal — work on both server and client)
280
+ export { Meta } from "./handles/meta.js";
281
+ export { Breadcrumbs } from "./handles/breadcrumbs.js";
282
+
238
283
  // Meta types
239
284
  export type { MetaDescriptor, MetaDescriptorBase } from "./router/types.js";
240
285
 
@@ -259,12 +304,17 @@ export {
259
304
  type LocationStateOptions,
260
305
  } from "./browser/react/location-state-shared.js";
261
306
 
262
- // Path-based response type lookup from RegisteredRoutes
263
- export type { PathResponse } from "./href-client.js";
307
+ // Path and response types are ambient on the `Rango` namespace (`Rango.Path`,
308
+ // `Rango.PathResponse`, declared in href-client.ts) — no import needed.
264
309
 
265
- // Telemetry sink
266
- export { createConsoleSink } from "./router/telemetry.js";
267
- export { createOTelSink } from "./router/telemetry-otel.js";
310
+ // Telemetry types only — the createConsoleSink/createOTelSink values are
311
+ // server-only and live in index.rsc.ts (the `react-server` condition of the
312
+ // bare `@rangojs/router` import). Re-exporting them as values from this
313
+ // (default/client) entry would pull telemetry.ts and telemetry-otel.ts into
314
+ // the client module graph; both tree-shake to zero bytes but still appear in
315
+ // bundle analysis output and slow build-time module resolution. Consumers
316
+ // who need the values in non-RSC contexts can import from
317
+ // `@rangojs/router/server`.
268
318
  export type { OTelTracer, OTelSpan } from "./router/telemetry-otel.js";
269
319
  export type { TelemetrySink, TelemetryEvent } from "./router/telemetry.js";
270
320