@rangojs/router 0.0.0-experimental.8 → 0.0.0-experimental.80

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 (312) hide show
  1. package/AGENTS.md +9 -0
  2. package/README.md +942 -4
  3. package/dist/bin/rango.js +1689 -0
  4. package/dist/vite/index.js +4960 -935
  5. package/package.json +70 -60
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +294 -0
  8. package/skills/caching/SKILL.md +93 -23
  9. package/skills/composability/SKILL.md +172 -0
  10. package/skills/debug-manifest/SKILL.md +12 -8
  11. package/skills/document-cache/SKILL.md +18 -16
  12. package/skills/fonts/SKILL.md +167 -0
  13. package/skills/handler-use/SKILL.md +362 -0
  14. package/skills/hooks/SKILL.md +334 -72
  15. package/skills/host-router/SKILL.md +218 -0
  16. package/skills/intercept/SKILL.md +151 -8
  17. package/skills/layout/SKILL.md +122 -3
  18. package/skills/links/SKILL.md +92 -31
  19. package/skills/loader/SKILL.md +404 -44
  20. package/skills/middleware/SKILL.md +205 -37
  21. package/skills/migrate-nextjs/SKILL.md +560 -0
  22. package/skills/migrate-react-router/SKILL.md +764 -0
  23. package/skills/mime-routes/SKILL.md +128 -0
  24. package/skills/parallel/SKILL.md +263 -1
  25. package/skills/prerender/SKILL.md +685 -0
  26. package/skills/rango/SKILL.md +87 -16
  27. package/skills/response-routes/SKILL.md +411 -0
  28. package/skills/route/SKILL.md +281 -14
  29. package/skills/router-setup/SKILL.md +210 -32
  30. package/skills/tailwind/SKILL.md +129 -0
  31. package/skills/theme/SKILL.md +9 -8
  32. package/skills/typesafety/SKILL.md +328 -89
  33. package/skills/use-cache/SKILL.md +324 -0
  34. package/src/__internal.ts +102 -4
  35. package/src/bin/rango.ts +321 -0
  36. package/src/browser/action-coordinator.ts +97 -0
  37. package/src/browser/action-response-classifier.ts +99 -0
  38. package/src/browser/app-version.ts +14 -0
  39. package/src/browser/event-controller.ts +92 -64
  40. package/src/browser/history-state.ts +80 -0
  41. package/src/browser/intercept-utils.ts +52 -0
  42. package/src/browser/link-interceptor.ts +24 -4
  43. package/src/browser/logging.ts +55 -0
  44. package/src/browser/merge-segment-loaders.ts +20 -12
  45. package/src/browser/navigation-bridge.ts +317 -560
  46. package/src/browser/navigation-client.ts +206 -68
  47. package/src/browser/navigation-store.ts +73 -55
  48. package/src/browser/navigation-transaction.ts +297 -0
  49. package/src/browser/network-error-handler.ts +61 -0
  50. package/src/browser/partial-update.ts +343 -316
  51. package/src/browser/prefetch/cache.ts +216 -0
  52. package/src/browser/prefetch/fetch.ts +206 -0
  53. package/src/browser/prefetch/observer.ts +65 -0
  54. package/src/browser/prefetch/policy.ts +48 -0
  55. package/src/browser/prefetch/queue.ts +160 -0
  56. package/src/browser/prefetch/resource-ready.ts +77 -0
  57. package/src/browser/rango-state.ts +112 -0
  58. package/src/browser/react/Link.tsx +253 -74
  59. package/src/browser/react/NavigationProvider.tsx +87 -11
  60. package/src/browser/react/context.ts +11 -0
  61. package/src/browser/react/filter-segment-order.ts +11 -0
  62. package/src/browser/react/index.ts +12 -12
  63. package/src/browser/react/location-state-shared.ts +95 -53
  64. package/src/browser/react/location-state.ts +60 -15
  65. package/src/browser/react/mount-context.ts +6 -1
  66. package/src/browser/react/nonce-context.ts +23 -0
  67. package/src/browser/react/shallow-equal.ts +27 -0
  68. package/src/browser/react/use-action.ts +29 -51
  69. package/src/browser/react/use-client-cache.ts +5 -3
  70. package/src/browser/react/use-handle.ts +30 -126
  71. package/src/browser/react/use-href.tsx +2 -2
  72. package/src/browser/react/use-link-status.ts +6 -5
  73. package/src/browser/react/use-navigation.ts +44 -65
  74. package/src/browser/react/use-params.ts +65 -0
  75. package/src/browser/react/use-pathname.ts +47 -0
  76. package/src/browser/react/use-router.ts +76 -0
  77. package/src/browser/react/use-search-params.ts +56 -0
  78. package/src/browser/react/use-segments.ts +80 -97
  79. package/src/browser/response-adapter.ts +73 -0
  80. package/src/browser/rsc-router.tsx +214 -58
  81. package/src/browser/scroll-restoration.ts +127 -52
  82. package/src/browser/segment-reconciler.ts +243 -0
  83. package/src/browser/segment-structure-assert.ts +16 -0
  84. package/src/browser/server-action-bridge.ts +510 -603
  85. package/src/browser/shallow.ts +6 -1
  86. package/src/browser/types.ts +141 -48
  87. package/src/browser/validate-redirect-origin.ts +29 -0
  88. package/src/build/generate-manifest.ts +235 -24
  89. package/src/build/generate-route-types.ts +39 -0
  90. package/src/build/index.ts +13 -0
  91. package/src/build/route-trie.ts +291 -0
  92. package/src/build/route-types/ast-helpers.ts +25 -0
  93. package/src/build/route-types/ast-route-extraction.ts +98 -0
  94. package/src/build/route-types/codegen.ts +102 -0
  95. package/src/build/route-types/include-resolution.ts +418 -0
  96. package/src/build/route-types/param-extraction.ts +48 -0
  97. package/src/build/route-types/per-module-writer.ts +128 -0
  98. package/src/build/route-types/router-processing.ts +618 -0
  99. package/src/build/route-types/scan-filter.ts +85 -0
  100. package/src/build/runtime-discovery.ts +231 -0
  101. package/src/cache/background-task.ts +34 -0
  102. package/src/cache/cache-key-utils.ts +44 -0
  103. package/src/cache/cache-policy.ts +125 -0
  104. package/src/cache/cache-runtime.ts +342 -0
  105. package/src/cache/cache-scope.ts +167 -309
  106. package/src/cache/cf/cf-cache-store.ts +571 -17
  107. package/src/cache/cf/index.ts +13 -3
  108. package/src/cache/document-cache.ts +116 -77
  109. package/src/cache/handle-capture.ts +81 -0
  110. package/src/cache/handle-snapshot.ts +41 -0
  111. package/src/cache/index.ts +1 -15
  112. package/src/cache/memory-segment-store.ts +191 -13
  113. package/src/cache/profile-registry.ts +73 -0
  114. package/src/cache/read-through-swr.ts +134 -0
  115. package/src/cache/segment-codec.ts +256 -0
  116. package/src/cache/taint.ts +153 -0
  117. package/src/cache/types.ts +72 -122
  118. package/src/client.rsc.tsx +3 -1
  119. package/src/client.tsx +135 -301
  120. package/src/component-utils.ts +4 -4
  121. package/src/components/DefaultDocument.tsx +5 -1
  122. package/src/context-var.ts +156 -0
  123. package/src/debug.ts +19 -9
  124. package/src/errors.ts +108 -2
  125. package/src/handle.ts +55 -29
  126. package/src/handles/MetaTags.tsx +73 -20
  127. package/src/handles/breadcrumbs.ts +66 -0
  128. package/src/handles/index.ts +1 -0
  129. package/src/handles/meta.ts +30 -13
  130. package/src/host/cookie-handler.ts +21 -15
  131. package/src/host/errors.ts +8 -8
  132. package/src/host/index.ts +4 -7
  133. package/src/host/pattern-matcher.ts +27 -27
  134. package/src/host/router.ts +61 -39
  135. package/src/host/testing.ts +8 -8
  136. package/src/host/types.ts +15 -7
  137. package/src/host/utils.ts +1 -1
  138. package/src/href-client.ts +119 -29
  139. package/src/index.rsc.ts +155 -19
  140. package/src/index.ts +251 -30
  141. package/src/internal-debug.ts +11 -0
  142. package/src/loader.rsc.ts +26 -157
  143. package/src/loader.ts +27 -10
  144. package/src/network-error-thrower.tsx +3 -1
  145. package/src/outlet-provider.tsx +45 -0
  146. package/src/prerender/param-hash.ts +37 -0
  147. package/src/prerender/store.ts +186 -0
  148. package/src/prerender.ts +524 -0
  149. package/src/reverse.ts +354 -0
  150. package/src/root-error-boundary.tsx +41 -29
  151. package/src/route-content-wrapper.tsx +7 -4
  152. package/src/route-definition/dsl-helpers.ts +1121 -0
  153. package/src/route-definition/helper-factories.ts +200 -0
  154. package/src/route-definition/helpers-types.ts +478 -0
  155. package/src/route-definition/index.ts +55 -0
  156. package/src/route-definition/redirect.ts +101 -0
  157. package/src/route-definition/resolve-handler-use.ts +149 -0
  158. package/src/route-definition.ts +1 -1428
  159. package/src/route-map-builder.ts +217 -123
  160. package/src/route-name.ts +53 -0
  161. package/src/route-types.ts +77 -8
  162. package/src/router/content-negotiation.ts +215 -0
  163. package/src/router/debug-manifest.ts +72 -0
  164. package/src/router/error-handling.ts +9 -9
  165. package/src/router/find-match.ts +160 -0
  166. package/src/router/handler-context.ts +438 -86
  167. package/src/router/intercept-resolution.ts +402 -0
  168. package/src/router/lazy-includes.ts +237 -0
  169. package/src/router/loader-resolution.ts +356 -128
  170. package/src/router/logging.ts +251 -0
  171. package/src/router/manifest.ts +163 -35
  172. package/src/router/match-api.ts +555 -0
  173. package/src/router/match-context.ts +5 -3
  174. package/src/router/match-handlers.ts +440 -0
  175. package/src/router/match-middleware/background-revalidation.ts +108 -93
  176. package/src/router/match-middleware/cache-lookup.ts +460 -10
  177. package/src/router/match-middleware/cache-store.ts +98 -26
  178. package/src/router/match-middleware/intercept-resolution.ts +57 -17
  179. package/src/router/match-middleware/segment-resolution.ts +80 -6
  180. package/src/router/match-pipelines.ts +10 -45
  181. package/src/router/match-result.ts +135 -35
  182. package/src/router/metrics.ts +240 -15
  183. package/src/router/middleware-cookies.ts +55 -0
  184. package/src/router/middleware-types.ts +220 -0
  185. package/src/router/middleware.ts +324 -369
  186. package/src/router/navigation-snapshot.ts +182 -0
  187. package/src/router/pattern-matching.ts +211 -43
  188. package/src/router/prerender-match.ts +502 -0
  189. package/src/router/preview-match.ts +98 -0
  190. package/src/router/request-classification.ts +310 -0
  191. package/src/router/revalidation.ts +137 -38
  192. package/src/router/route-snapshot.ts +245 -0
  193. package/src/router/router-context.ts +41 -21
  194. package/src/router/router-interfaces.ts +484 -0
  195. package/src/router/router-options.ts +618 -0
  196. package/src/router/router-registry.ts +24 -0
  197. package/src/router/segment-resolution/fresh.ts +748 -0
  198. package/src/router/segment-resolution/helpers.ts +268 -0
  199. package/src/router/segment-resolution/loader-cache.ts +199 -0
  200. package/src/router/segment-resolution/revalidation.ts +1379 -0
  201. package/src/router/segment-resolution/static-store.ts +67 -0
  202. package/src/router/segment-resolution.ts +21 -0
  203. package/src/router/segment-wrappers.ts +291 -0
  204. package/src/router/telemetry-otel.ts +299 -0
  205. package/src/router/telemetry.ts +300 -0
  206. package/src/router/timeout.ts +148 -0
  207. package/src/router/trie-matching.ts +239 -0
  208. package/src/router/types.ts +78 -3
  209. package/src/router.ts +740 -4252
  210. package/src/rsc/handler-context.ts +45 -0
  211. package/src/rsc/handler.ts +907 -797
  212. package/src/rsc/helpers.ts +140 -6
  213. package/src/rsc/index.ts +0 -20
  214. package/src/rsc/loader-fetch.ts +229 -0
  215. package/src/rsc/manifest-init.ts +90 -0
  216. package/src/rsc/nonce.ts +14 -0
  217. package/src/rsc/origin-guard.ts +141 -0
  218. package/src/rsc/progressive-enhancement.ts +391 -0
  219. package/src/rsc/response-error.ts +37 -0
  220. package/src/rsc/response-route-handler.ts +347 -0
  221. package/src/rsc/rsc-rendering.ts +246 -0
  222. package/src/rsc/runtime-warnings.ts +42 -0
  223. package/src/rsc/server-action.ts +356 -0
  224. package/src/rsc/ssr-setup.ts +128 -0
  225. package/src/rsc/types.ts +46 -11
  226. package/src/search-params.ts +230 -0
  227. package/src/segment-content-promise.ts +67 -0
  228. package/src/segment-loader-promise.ts +122 -0
  229. package/src/segment-system.tsx +134 -36
  230. package/src/server/context.ts +341 -61
  231. package/src/server/cookie-store.ts +190 -0
  232. package/src/server/fetchable-loader-store.ts +37 -0
  233. package/src/server/handle-store.ts +113 -15
  234. package/src/server/loader-registry.ts +24 -64
  235. package/src/server/request-context.ts +607 -81
  236. package/src/server.ts +35 -130
  237. package/src/ssr/index.tsx +103 -30
  238. package/src/static-handler.ts +126 -0
  239. package/src/theme/ThemeProvider.tsx +21 -15
  240. package/src/theme/ThemeScript.tsx +5 -5
  241. package/src/theme/constants.ts +5 -2
  242. package/src/theme/index.ts +4 -14
  243. package/src/theme/theme-context.ts +4 -30
  244. package/src/theme/theme-script.ts +21 -18
  245. package/src/types/boundaries.ts +158 -0
  246. package/src/types/cache-types.ts +198 -0
  247. package/src/types/error-types.ts +192 -0
  248. package/src/types/global-namespace.ts +100 -0
  249. package/src/types/handler-context.ts +791 -0
  250. package/src/types/index.ts +88 -0
  251. package/src/types/loader-types.ts +210 -0
  252. package/src/types/route-config.ts +170 -0
  253. package/src/types/route-entry.ts +120 -0
  254. package/src/types/segments.ts +150 -0
  255. package/src/types.ts +1 -1623
  256. package/src/urls/include-helper.ts +207 -0
  257. package/src/urls/index.ts +53 -0
  258. package/src/urls/path-helper-types.ts +372 -0
  259. package/src/urls/path-helper.ts +364 -0
  260. package/src/urls/pattern-types.ts +107 -0
  261. package/src/urls/response-types.ts +116 -0
  262. package/src/urls/type-extraction.ts +372 -0
  263. package/src/urls/urls-function.ts +98 -0
  264. package/src/urls.ts +1 -802
  265. package/src/use-loader.tsx +161 -81
  266. package/src/vite/discovery/bundle-postprocess.ts +181 -0
  267. package/src/vite/discovery/discover-routers.ts +348 -0
  268. package/src/vite/discovery/prerender-collection.ts +439 -0
  269. package/src/vite/discovery/route-types-writer.ts +258 -0
  270. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  271. package/src/vite/discovery/state.ts +117 -0
  272. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  273. package/src/vite/index.ts +15 -1133
  274. package/src/vite/plugin-types.ts +103 -0
  275. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  276. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  277. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  278. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -53
  279. package/src/vite/plugins/expose-id-utils.ts +299 -0
  280. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  281. package/src/vite/plugins/expose-ids/handler-transform.ts +209 -0
  282. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  283. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  284. package/src/vite/plugins/expose-ids/types.ts +45 -0
  285. package/src/vite/plugins/expose-internal-ids.ts +786 -0
  286. package/src/vite/plugins/performance-tracks.ts +88 -0
  287. package/src/vite/plugins/refresh-cmd.ts +127 -0
  288. package/src/vite/plugins/use-cache-transform.ts +323 -0
  289. package/src/vite/plugins/version-injector.ts +83 -0
  290. package/src/vite/plugins/version-plugin.ts +266 -0
  291. package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
  292. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  293. package/src/vite/rango.ts +462 -0
  294. package/src/vite/router-discovery.ts +918 -0
  295. package/src/vite/utils/ast-handler-extract.ts +517 -0
  296. package/src/vite/utils/banner.ts +36 -0
  297. package/src/vite/utils/bundle-analysis.ts +137 -0
  298. package/src/vite/utils/manifest-utils.ts +70 -0
  299. package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
  300. package/src/vite/utils/prerender-utils.ts +221 -0
  301. package/src/vite/utils/shared-utils.ts +170 -0
  302. package/CLAUDE.md +0 -43
  303. package/src/browser/lru-cache.ts +0 -69
  304. package/src/browser/request-controller.ts +0 -164
  305. package/src/cache/memory-store.ts +0 -253
  306. package/src/href-context.ts +0 -33
  307. package/src/href.ts +0 -255
  308. package/src/server/route-manifest-cache.ts +0 -173
  309. package/src/vite/expose-handle-id.ts +0 -209
  310. package/src/vite/expose-loader-id.ts +0 -426
  311. package/src/vite/expose-location-state-id.ts +0 -177
  312. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -15,6 +15,7 @@
15
15
  */
16
16
 
17
17
  import type { GetRegisteredRoutes } from "./types.js";
18
+ import type { ResponseEnvelope } from "./urls.js";
18
19
 
19
20
  /**
20
21
  * Parse constraint values into a union type for paths
@@ -44,30 +45,34 @@ type ParseConstraintPath<T extends string> =
44
45
  export type PatternToPath<T extends string> =
45
46
  // Optional + constrained param in middle: /:param(a|b)?/rest
46
47
  T extends `${infer Before}:${infer _Name}(${infer Constraint})?/${infer After}`
47
- ? PatternToPath<`${Before}${After}`> | `${Before}${ParseConstraintPath<Constraint>}/${PatternToPath<After>}`
48
- // Optional + constrained param at end: /path/:param(a|b)?
49
- : T extends `${infer Before}:${infer _Name}(${infer Constraint})?`
50
- ? Before | `${Before}${ParseConstraintPath<Constraint>}`
51
- // Constrained param in middle: /:param(a|b)/rest
52
- : T extends `${infer Before}:${infer _Name}(${infer Constraint})/${infer After}`
53
- ? `${Before}${ParseConstraintPath<Constraint>}/${PatternToPath<After>}`
54
- // Constrained param at end: /path/:param(a|b)
55
- : T extends `${infer Before}:${infer _Name}(${infer Constraint})`
56
- ? `${Before}${ParseConstraintPath<Constraint>}`
57
- // Optional param in middle: /:param?/rest
58
- : T extends `${infer Before}:${infer _Param}?/${infer After}`
59
- ? PatternToPath<`${Before}${After}`> | `${Before}${string}/${PatternToPath<After>}`
60
- // Optional param at end: /path/:param?
61
- : T extends `${infer Before}:${infer _Param}?`
62
- ? Before | `${Before}${string}`
63
- // Required param in middle: /:param/rest
64
- : T extends `${infer Before}:${infer _Param}/${infer After}`
65
- ? `${Before}${string}/${PatternToPath<After>}`
66
- // Required param at end: /path/:param
67
- : T extends `${infer Before}:${infer _Param}`
68
- ? `${Before}${string}`
69
- // Static path
70
- : T;
48
+ ?
49
+ | PatternToPath<`${Before}${After}`>
50
+ | `${Before}${ParseConstraintPath<Constraint>}/${PatternToPath<After>}`
51
+ : // Optional + constrained param at end: /path/:param(a|b)?
52
+ T extends `${infer Before}:${infer _Name}(${infer Constraint})?`
53
+ ? Before | `${Before}${ParseConstraintPath<Constraint>}`
54
+ : // Constrained param in middle: /:param(a|b)/rest
55
+ T extends `${infer Before}:${infer _Name}(${infer Constraint})/${infer After}`
56
+ ? `${Before}${ParseConstraintPath<Constraint>}/${PatternToPath<After>}`
57
+ : // Constrained param at end: /path/:param(a|b)
58
+ T extends `${infer Before}:${infer _Name}(${infer Constraint})`
59
+ ? `${Before}${ParseConstraintPath<Constraint>}`
60
+ : // Optional param in middle: /:param?/rest
61
+ T extends `${infer Before}:${infer _Param}?/${infer After}`
62
+ ?
63
+ | PatternToPath<`${Before}${After}`>
64
+ | `${Before}${string}/${PatternToPath<After>}`
65
+ : // Optional param at end: /path/:param?
66
+ T extends `${infer Before}:${infer _Param}?`
67
+ ? Before | `${Before}${string}`
68
+ : // Required param in middle: /:param/rest
69
+ T extends `${infer Before}:${infer _Param}/${infer After}`
70
+ ? `${Before}${string}/${PatternToPath<After>}`
71
+ : // Required param at end: /path/:param
72
+ T extends `${infer Before}:${infer _Param}`
73
+ ? `${Before}${string}`
74
+ : // Static path
75
+ T;
71
76
 
72
77
  /**
73
78
  * Allow optional query string (?...) and/or hash fragment (#...) suffix
@@ -82,10 +87,55 @@ type WithSuffix<T extends string> =
82
87
  | `${T}?${string}#${string}`;
83
88
 
84
89
  /**
85
- * Helper type to get pattern from routes, handling both Record and interface types
90
+ * Helper type to get pattern from routes, handling string values and { path, response } objects
91
+ */
92
+ type RoutePattern<TRoutes, K extends keyof TRoutes> = TRoutes[K] extends string
93
+ ? TRoutes[K]
94
+ : TRoutes[K] extends { readonly path: infer P extends string }
95
+ ? P
96
+ : string;
97
+
98
+ /**
99
+ * Reverse lookup: find route name where the pattern matches TPattern
100
+ */
101
+ type NameForPattern<TPattern extends string, TRoutes = GetRegisteredRoutes> = {
102
+ [K in keyof TRoutes]: RoutePattern<TRoutes, K> extends TPattern ? K : never;
103
+ }[keyof TRoutes];
104
+
105
+ /**
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.
110
+ *
111
+ * For static routes (no params), pattern === path:
112
+ * PathResponse<"/api/health"> → { status: string; timestamp: number }
113
+ *
114
+ * For dynamic routes, use the pattern:
115
+ * PathResponse<"/api/products/:id"> → Product
86
116
  */
87
- type RoutePattern<TRoutes, K extends keyof TRoutes> =
88
- TRoutes[K] extends string ? TRoutes[K] : string;
117
+ export type PathResponse<
118
+ TPattern extends string,
119
+ 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
+ >;
129
+
130
+ /**
131
+ * Strip trailing slash from a path (e.g., "/blog/" -> "/blog" | "/blog/")
132
+ * Allows navigation to include() prefixes without requiring trailing slash
133
+ */
134
+ type OptionalTrailingSlash<T extends string> = T extends `${infer Base}/`
135
+ ? Base extends ""
136
+ ? T
137
+ : Base | T
138
+ : T;
89
139
 
90
140
  /**
91
141
  * Union of all valid paths from registered routes
@@ -96,7 +146,13 @@ type RoutePattern<TRoutes, K extends keyof TRoutes> =
96
146
  export type ValidPaths<TRoutes = GetRegisteredRoutes> =
97
147
  keyof TRoutes extends never
98
148
  ? `/${string}` // Fallback when no routes are registered
99
- : WithSuffix<PatternToPath<RoutePattern<TRoutes, keyof TRoutes>>>;
149
+ : WithSuffix<
150
+ {
151
+ [K in keyof TRoutes]: OptionalTrailingSlash<
152
+ PatternToPath<RoutePattern<TRoutes, K>>
153
+ >;
154
+ }[keyof TRoutes]
155
+ >;
100
156
 
101
157
  /**
102
158
  * Type-safe href function for client-side use
@@ -126,7 +182,41 @@ export type ValidPaths<TRoutes = GetRegisteredRoutes> =
126
182
  */
127
183
  export function href<T extends ValidPaths>(path: T, mount?: string): string {
128
184
  if (mount && mount !== "/") {
129
- return mount + path;
185
+ // Strip trailing slash from mount to avoid double-slash when joining
186
+ const normalizedMount = mount.endsWith("/") ? mount.slice(0, -1) : mount;
187
+ return normalizedMount + path;
130
188
  }
131
189
  return path;
132
190
  }
191
+
192
+ /**
193
+ * Props shape returned by href.json() etc. for spreading on <Link>.
194
+ * Sets data-external to trigger hard navigation (skips RSC fetch).
195
+ */
196
+ export interface ResponseHrefProps {
197
+ to: string;
198
+ "data-external": "";
199
+ }
200
+
201
+ type ResponseHrefFn = <T extends ValidPaths>(
202
+ path: T,
203
+ mount?: string,
204
+ ) => ResponseHrefProps;
205
+
206
+ function createResponseHrefTag(): ResponseHrefFn {
207
+ return (path, mount) => ({
208
+ to: href(path, mount),
209
+ "data-external": "" as const,
210
+ });
211
+ }
212
+
213
+ export namespace href {
214
+ export const json: ResponseHrefFn = createResponseHrefTag();
215
+ export const text: ResponseHrefFn = createResponseHrefTag();
216
+ export const html: ResponseHrefFn = createResponseHrefTag();
217
+ export const xml: ResponseHrefFn = createResponseHrefTag();
218
+ export const md: ResponseHrefFn = createResponseHrefTag();
219
+ export const image: ResponseHrefFn = createResponseHrefTag();
220
+ export const stream: ResponseHrefFn = createResponseHrefTag();
221
+ export const any: ResponseHrefFn = createResponseHrefTag();
222
+ }
package/src/index.rsc.ts CHANGED
@@ -1,9 +1,9 @@
1
1
  /**
2
- * rsc-router (react-server environment)
2
+ * @rangojs/router (react-server environment)
3
3
  *
4
- * This file is used when importing "rsc-router" from RSC (server components).
4
+ * This file is used when importing "@rangojs/router" from RSC (server components).
5
5
  * It re-exports everything from the universal index.ts plus adds server-side
6
- * createLoader that includes the actual loader function.
6
+ * implementations that replace the client-side error stubs.
7
7
  *
8
8
  * The bundler uses the "react-server" export condition to select this file
9
9
  * in RSC context, while the regular index.ts is used in client components.
@@ -11,8 +11,6 @@
11
11
 
12
12
  // Re-export all universal exports from index.ts
13
13
  export {
14
- // Universal rendering utilities
15
- renderSegments,
16
14
  // Error classes
17
15
  RouteNotFoundError,
18
16
  DataNotFoundError,
@@ -21,18 +19,20 @@ export {
21
19
  HandlerError,
22
20
  BuildError,
23
21
  InvalidHandlerError,
24
- NetworkError,
25
- isNetworkError,
26
- sanitizeError,
22
+ RouterError,
23
+ Skip,
24
+ isSkip,
27
25
  } from "./index.js";
28
26
 
29
27
  // Re-export all types from types.ts (user-facing types only)
30
28
  export type {
31
29
  // Configuration types
32
30
  DocumentProps,
33
- RouterEnv,
34
31
  DefaultEnv,
35
32
  RouteDefinition,
33
+ RouteConfig,
34
+ RouteDefinitionOptions,
35
+ TrailingSlashMode,
36
36
  // Handler types
37
37
  Handler,
38
38
  HandlerContext,
@@ -50,9 +50,6 @@ export type {
50
50
  LoaderContext,
51
51
  FetchableLoaderOptions,
52
52
  LoadOptions,
53
- LoaderActionContext,
54
- LoaderAction,
55
- LoaderMiddlewareFn,
56
53
  // Error boundary types
57
54
  ErrorInfo,
58
55
  ErrorBoundaryFallbackProps,
@@ -62,14 +59,73 @@ export type {
62
59
  NotFoundInfo,
63
60
  NotFoundBoundaryFallbackProps,
64
61
  NotFoundBoundaryHandler,
62
+ // Error handling callback types
63
+ ErrorPhase,
64
+ OnErrorContext,
65
+ OnErrorCallback,
65
66
  } from "./types.js";
66
67
 
67
68
  // Router options type (server-only, so import directly)
68
- export type { RSCRouterOptions } from "./router.js";
69
+ export type {
70
+ RSCRouterOptions,
71
+ SSRStreamMode,
72
+ SSROptions,
73
+ ResolveStreamingContext,
74
+ } from "./router.js";
75
+
76
+ // Server-side createLoader and redirect
77
+ export {
78
+ createLoader,
79
+ redirect,
80
+ type RouteHelpers,
81
+ type RouteHandlers,
82
+ // Globally importable route helpers for composition
83
+ layout,
84
+ cache,
85
+ middleware,
86
+ revalidate,
87
+ loader,
88
+ loading,
89
+ parallel,
90
+ intercept,
91
+ when,
92
+ errorBoundary,
93
+ notFoundBoundary,
94
+ transition,
95
+ } from "./route-definition.js";
96
+
97
+ // Composition types for reusable callback factories
98
+ export type {
99
+ RouteUseItem,
100
+ LayoutUseItem,
101
+ AllUseItems,
102
+ UseItems,
103
+ HandlerUseItem,
104
+ } from "./route-types.js";
105
+
106
+ // Handle API
107
+ export { createHandle, isHandle, type Handle } from "./handle.js";
69
108
 
70
- // Server-side createLoader - includes the actual loader function
71
- // This is the key addition for RSC context
72
- export { createLoader } from "./route-definition.js";
109
+ // Context variable API (typed ctx.set/ctx.get tokens)
110
+ export { createVar, type ContextVar } from "./context-var.js";
111
+
112
+ // CSP nonce token (use with ctx.get(nonce) in middleware/handlers)
113
+ export { nonce } from "./rsc/nonce.js";
114
+
115
+ // Pre-render handler API
116
+ export {
117
+ Prerender,
118
+ Passthrough,
119
+ type PrerenderHandlerDefinition,
120
+ type PassthroughHandlerDefinition,
121
+ type PrerenderOptions,
122
+ type BuildContext,
123
+ type StaticBuildContext,
124
+ type GetParamsContext,
125
+ } from "./prerender.js";
126
+
127
+ // Static handler API
128
+ export { Static, type StaticHandlerDefinition } from "./static-handler.js";
73
129
 
74
130
  // Django-style URL patterns (RSC/server context)
75
131
  export {
@@ -79,6 +135,17 @@ export {
79
135
  type UrlPatterns,
80
136
  type IncludeOptions,
81
137
  type IncludeItem,
138
+ type RouteResponse,
139
+ type ResponseError,
140
+ type ResponseEnvelope,
141
+ type ResponseHandler,
142
+ type ResponseHandlerContext,
143
+ type JsonResponseHandler,
144
+ type TextResponseHandler,
145
+ type JsonValue,
146
+ type ResponsePathFn,
147
+ type JsonResponsePathFn,
148
+ type TextResponsePathFn,
82
149
  } from "./urls.js";
83
150
 
84
151
  // Core router (server-side)
@@ -86,6 +153,7 @@ export {
86
153
  createRouter,
87
154
  type RSCRouter,
88
155
  type RootLayoutProps,
156
+ type RouterRequestInput,
89
157
  } from "./router.js";
90
158
 
91
159
  // RSC handler types (server-side)
@@ -93,7 +161,75 @@ export type { HandlerCacheConfig } from "./rsc/types.js";
93
161
 
94
162
  // Built-in handles (server-side)
95
163
  export { Meta } from "./handles/meta.js";
164
+ export { Breadcrumbs, type BreadcrumbItem } from "./handles/breadcrumbs.js";
165
+
166
+ // Request context (for accessing request data in server actions/components).
167
+ // Re-exported with a narrowed return type so that public consumers only see
168
+ // public members. Internal code imports from "./server/request-context.js"
169
+ // directly and gets the full type.
170
+ import { getRequestContext as _getRequestContextInternal } from "./server/request-context.js";
171
+ export type { PublicRequestContext as RequestContext } from "./server/request-context.js";
172
+ import type { PublicRequestContext } from "./server/request-context.js";
173
+ import type { DefaultEnv } from "./types/global-namespace.js";
174
+
175
+ export const getRequestContext: <
176
+ TEnv = DefaultEnv,
177
+ >() => PublicRequestContext<TEnv> = _getRequestContextInternal;
96
178
 
97
- // Href type utilities for type-safe URL generation
98
- export type { ScopedHrefFunction, HrefFunction, ExtractLocalRoutes } from "./href.js";
99
- export { scopedHref } from "./href.js";
179
+ // Request-scoped shorthands
180
+ export {
181
+ cookies,
182
+ headers,
183
+ type CookieStore,
184
+ type Cookie,
185
+ type ReadonlyHeaders,
186
+ } from "./server/cookie-store.js";
187
+
188
+ // Meta types
189
+ export type { MetaDescriptor, MetaDescriptorBase } from "./router/types.js";
190
+
191
+ // Middleware context types
192
+ export type { MiddlewareContext, CookieOptions } from "./router/middleware.js";
193
+
194
+ // Reverse type utilities for type-safe URL generation (Django-style URL reversal)
195
+ export type {
196
+ ScopedReverseFunction,
197
+ ReverseFunction,
198
+ ExtractLocalRoutes,
199
+ ParamsFor,
200
+ } from "./reverse.js";
201
+ export { scopedReverse, createReverse } from "./reverse.js";
202
+
203
+ // Search params schema types
204
+ export type {
205
+ SearchSchema,
206
+ SearchSchemaValue,
207
+ ResolveSearchSchema,
208
+ RouteSearchParams,
209
+ RouteParams,
210
+ } from "./search-params.js";
211
+
212
+ // Location state (universal)
213
+ export {
214
+ createLocationState,
215
+ type LocationStateDefinition,
216
+ type LocationStateEntry,
217
+ type LocationStateOptions,
218
+ } from "./browser/react/location-state-shared.js";
219
+
220
+ // Path-based response type lookup from RegisteredRoutes
221
+ export type { PathResponse } from "./href-client.js";
222
+
223
+ // Telemetry sink
224
+ export { createConsoleSink } from "./router/telemetry.js";
225
+ export { createOTelSink } from "./router/telemetry-otel.js";
226
+ export type { OTelTracer, OTelSpan } from "./router/telemetry-otel.js";
227
+ export type { TelemetrySink, TelemetryEvent } from "./router/telemetry.js";
228
+
229
+ // Timeout types and error class
230
+ export { RouterTimeoutError } from "./router/timeout.js";
231
+ export type {
232
+ RouterTimeouts,
233
+ TimeoutPhase,
234
+ TimeoutContext,
235
+ } from "./router/timeout.js";