@rangojs/router 0.0.0-experimental.7 → 0.0.0-experimental.71

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 (307) 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 +4951 -930
  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/hooks/SKILL.md +334 -72
  14. package/skills/host-router/SKILL.md +218 -0
  15. package/skills/intercept/SKILL.md +131 -8
  16. package/skills/layout/SKILL.md +100 -3
  17. package/skills/links/SKILL.md +92 -31
  18. package/skills/loader/SKILL.md +404 -44
  19. package/skills/middleware/SKILL.md +173 -34
  20. package/skills/mime-routes/SKILL.md +128 -0
  21. package/skills/parallel/SKILL.md +204 -1
  22. package/skills/prerender/SKILL.md +685 -0
  23. package/skills/rango/SKILL.md +85 -16
  24. package/skills/response-routes/SKILL.md +411 -0
  25. package/skills/route/SKILL.md +257 -14
  26. package/skills/router-setup/SKILL.md +210 -32
  27. package/skills/tailwind/SKILL.md +129 -0
  28. package/skills/theme/SKILL.md +9 -8
  29. package/skills/typesafety/SKILL.md +328 -89
  30. package/skills/use-cache/SKILL.md +324 -0
  31. package/src/__internal.ts +102 -4
  32. package/src/bin/rango.ts +321 -0
  33. package/src/browser/action-coordinator.ts +97 -0
  34. package/src/browser/action-response-classifier.ts +99 -0
  35. package/src/browser/app-version.ts +14 -0
  36. package/src/browser/event-controller.ts +92 -64
  37. package/src/browser/history-state.ts +80 -0
  38. package/src/browser/intercept-utils.ts +52 -0
  39. package/src/browser/link-interceptor.ts +24 -4
  40. package/src/browser/logging.ts +55 -0
  41. package/src/browser/merge-segment-loaders.ts +20 -12
  42. package/src/browser/navigation-bridge.ts +296 -558
  43. package/src/browser/navigation-client.ts +179 -69
  44. package/src/browser/navigation-store.ts +73 -55
  45. package/src/browser/navigation-transaction.ts +297 -0
  46. package/src/browser/network-error-handler.ts +61 -0
  47. package/src/browser/partial-update.ts +328 -313
  48. package/src/browser/prefetch/cache.ts +206 -0
  49. package/src/browser/prefetch/fetch.ts +150 -0
  50. package/src/browser/prefetch/observer.ts +65 -0
  51. package/src/browser/prefetch/policy.ts +48 -0
  52. package/src/browser/prefetch/queue.ts +160 -0
  53. package/src/browser/prefetch/resource-ready.ts +77 -0
  54. package/src/browser/rango-state.ts +112 -0
  55. package/src/browser/react/Link.tsx +230 -74
  56. package/src/browser/react/NavigationProvider.tsx +87 -11
  57. package/src/browser/react/context.ts +11 -0
  58. package/src/browser/react/filter-segment-order.ts +11 -0
  59. package/src/browser/react/index.ts +12 -12
  60. package/src/browser/react/location-state-shared.ts +95 -53
  61. package/src/browser/react/location-state.ts +60 -15
  62. package/src/browser/react/mount-context.ts +6 -1
  63. package/src/browser/react/nonce-context.ts +23 -0
  64. package/src/browser/react/shallow-equal.ts +27 -0
  65. package/src/browser/react/use-action.ts +29 -51
  66. package/src/browser/react/use-client-cache.ts +5 -3
  67. package/src/browser/react/use-handle.ts +30 -126
  68. package/src/browser/react/use-href.tsx +2 -2
  69. package/src/browser/react/use-link-status.ts +6 -5
  70. package/src/browser/react/use-navigation.ts +22 -63
  71. package/src/browser/react/use-params.ts +65 -0
  72. package/src/browser/react/use-pathname.ts +47 -0
  73. package/src/browser/react/use-router.ts +76 -0
  74. package/src/browser/react/use-search-params.ts +56 -0
  75. package/src/browser/react/use-segments.ts +80 -97
  76. package/src/browser/response-adapter.ts +73 -0
  77. package/src/browser/rsc-router.tsx +214 -58
  78. package/src/browser/scroll-restoration.ts +127 -52
  79. package/src/browser/segment-reconciler.ts +221 -0
  80. package/src/browser/segment-structure-assert.ts +16 -0
  81. package/src/browser/server-action-bridge.ts +510 -603
  82. package/src/browser/shallow.ts +6 -1
  83. package/src/browser/types.ts +141 -48
  84. package/src/browser/validate-redirect-origin.ts +29 -0
  85. package/src/build/generate-manifest.ts +235 -24
  86. package/src/build/generate-route-types.ts +39 -0
  87. package/src/build/index.ts +13 -0
  88. package/src/build/route-trie.ts +265 -0
  89. package/src/build/route-types/ast-helpers.ts +25 -0
  90. package/src/build/route-types/ast-route-extraction.ts +98 -0
  91. package/src/build/route-types/codegen.ts +102 -0
  92. package/src/build/route-types/include-resolution.ts +418 -0
  93. package/src/build/route-types/param-extraction.ts +48 -0
  94. package/src/build/route-types/per-module-writer.ts +128 -0
  95. package/src/build/route-types/router-processing.ts +618 -0
  96. package/src/build/route-types/scan-filter.ts +85 -0
  97. package/src/build/runtime-discovery.ts +231 -0
  98. package/src/cache/background-task.ts +34 -0
  99. package/src/cache/cache-key-utils.ts +44 -0
  100. package/src/cache/cache-policy.ts +125 -0
  101. package/src/cache/cache-runtime.ts +342 -0
  102. package/src/cache/cache-scope.ts +167 -309
  103. package/src/cache/cf/cf-cache-store.ts +571 -17
  104. package/src/cache/cf/index.ts +13 -3
  105. package/src/cache/document-cache.ts +116 -77
  106. package/src/cache/handle-capture.ts +81 -0
  107. package/src/cache/handle-snapshot.ts +41 -0
  108. package/src/cache/index.ts +1 -15
  109. package/src/cache/memory-segment-store.ts +191 -13
  110. package/src/cache/profile-registry.ts +73 -0
  111. package/src/cache/read-through-swr.ts +134 -0
  112. package/src/cache/segment-codec.ts +256 -0
  113. package/src/cache/taint.ts +153 -0
  114. package/src/cache/types.ts +72 -122
  115. package/src/client.rsc.tsx +3 -1
  116. package/src/client.tsx +105 -179
  117. package/src/component-utils.ts +4 -4
  118. package/src/components/DefaultDocument.tsx +5 -1
  119. package/src/context-var.ts +156 -0
  120. package/src/debug.ts +19 -9
  121. package/src/errors.ts +108 -2
  122. package/src/handle.ts +55 -29
  123. package/src/handles/MetaTags.tsx +73 -20
  124. package/src/handles/breadcrumbs.ts +66 -0
  125. package/src/handles/index.ts +1 -0
  126. package/src/handles/meta.ts +30 -13
  127. package/src/host/cookie-handler.ts +21 -15
  128. package/src/host/errors.ts +8 -8
  129. package/src/host/index.ts +4 -7
  130. package/src/host/pattern-matcher.ts +27 -27
  131. package/src/host/router.ts +61 -39
  132. package/src/host/testing.ts +8 -8
  133. package/src/host/types.ts +15 -7
  134. package/src/host/utils.ts +1 -1
  135. package/src/href-client.ts +119 -29
  136. package/src/index.rsc.ts +155 -19
  137. package/src/index.ts +223 -30
  138. package/src/internal-debug.ts +11 -0
  139. package/src/loader.rsc.ts +26 -157
  140. package/src/loader.ts +27 -10
  141. package/src/network-error-thrower.tsx +3 -1
  142. package/src/outlet-provider.tsx +45 -0
  143. package/src/prerender/param-hash.ts +37 -0
  144. package/src/prerender/store.ts +186 -0
  145. package/src/prerender.ts +524 -0
  146. package/src/reverse.ts +351 -0
  147. package/src/root-error-boundary.tsx +41 -29
  148. package/src/route-content-wrapper.tsx +7 -4
  149. package/src/route-definition/dsl-helpers.ts +982 -0
  150. package/src/route-definition/helper-factories.ts +200 -0
  151. package/src/route-definition/helpers-types.ts +434 -0
  152. package/src/route-definition/index.ts +55 -0
  153. package/src/route-definition/redirect.ts +101 -0
  154. package/src/route-definition/resolve-handler-use.ts +149 -0
  155. package/src/route-definition.ts +1 -1428
  156. package/src/route-map-builder.ts +217 -123
  157. package/src/route-name.ts +53 -0
  158. package/src/route-types.ts +70 -8
  159. package/src/router/content-negotiation.ts +215 -0
  160. package/src/router/debug-manifest.ts +72 -0
  161. package/src/router/error-handling.ts +9 -9
  162. package/src/router/find-match.ts +160 -0
  163. package/src/router/handler-context.ts +435 -86
  164. package/src/router/intercept-resolution.ts +402 -0
  165. package/src/router/lazy-includes.ts +237 -0
  166. package/src/router/loader-resolution.ts +356 -128
  167. package/src/router/logging.ts +251 -0
  168. package/src/router/manifest.ts +154 -35
  169. package/src/router/match-api.ts +555 -0
  170. package/src/router/match-context.ts +5 -3
  171. package/src/router/match-handlers.ts +440 -0
  172. package/src/router/match-middleware/background-revalidation.ts +108 -93
  173. package/src/router/match-middleware/cache-lookup.ts +459 -10
  174. package/src/router/match-middleware/cache-store.ts +98 -26
  175. package/src/router/match-middleware/intercept-resolution.ts +57 -17
  176. package/src/router/match-middleware/segment-resolution.ts +80 -6
  177. package/src/router/match-pipelines.ts +10 -45
  178. package/src/router/match-result.ts +135 -35
  179. package/src/router/metrics.ts +240 -15
  180. package/src/router/middleware-cookies.ts +55 -0
  181. package/src/router/middleware-types.ts +220 -0
  182. package/src/router/middleware.ts +324 -369
  183. package/src/router/navigation-snapshot.ts +182 -0
  184. package/src/router/pattern-matching.ts +211 -43
  185. package/src/router/prerender-match.ts +502 -0
  186. package/src/router/preview-match.ts +98 -0
  187. package/src/router/request-classification.ts +310 -0
  188. package/src/router/revalidation.ts +137 -38
  189. package/src/router/route-snapshot.ts +245 -0
  190. package/src/router/router-context.ts +41 -21
  191. package/src/router/router-interfaces.ts +484 -0
  192. package/src/router/router-options.ts +618 -0
  193. package/src/router/router-registry.ts +24 -0
  194. package/src/router/segment-resolution/fresh.ts +748 -0
  195. package/src/router/segment-resolution/helpers.ts +268 -0
  196. package/src/router/segment-resolution/loader-cache.ts +199 -0
  197. package/src/router/segment-resolution/revalidation.ts +1379 -0
  198. package/src/router/segment-resolution/static-store.ts +67 -0
  199. package/src/router/segment-resolution.ts +21 -0
  200. package/src/router/segment-wrappers.ts +291 -0
  201. package/src/router/telemetry-otel.ts +299 -0
  202. package/src/router/telemetry.ts +300 -0
  203. package/src/router/timeout.ts +148 -0
  204. package/src/router/trie-matching.ts +239 -0
  205. package/src/router/types.ts +78 -3
  206. package/src/router.ts +740 -4252
  207. package/src/rsc/handler-context.ts +45 -0
  208. package/src/rsc/handler.ts +907 -797
  209. package/src/rsc/helpers.ts +140 -6
  210. package/src/rsc/index.ts +0 -20
  211. package/src/rsc/loader-fetch.ts +229 -0
  212. package/src/rsc/manifest-init.ts +90 -0
  213. package/src/rsc/nonce.ts +14 -0
  214. package/src/rsc/origin-guard.ts +141 -0
  215. package/src/rsc/progressive-enhancement.ts +391 -0
  216. package/src/rsc/response-error.ts +37 -0
  217. package/src/rsc/response-route-handler.ts +347 -0
  218. package/src/rsc/rsc-rendering.ts +246 -0
  219. package/src/rsc/runtime-warnings.ts +42 -0
  220. package/src/rsc/server-action.ts +356 -0
  221. package/src/rsc/ssr-setup.ts +128 -0
  222. package/src/rsc/types.ts +46 -11
  223. package/src/search-params.ts +230 -0
  224. package/src/segment-system.tsx +165 -17
  225. package/src/server/context.ts +315 -58
  226. package/src/server/cookie-store.ts +190 -0
  227. package/src/server/fetchable-loader-store.ts +37 -0
  228. package/src/server/handle-store.ts +113 -15
  229. package/src/server/loader-registry.ts +24 -64
  230. package/src/server/request-context.ts +607 -81
  231. package/src/server.ts +35 -130
  232. package/src/ssr/index.tsx +103 -30
  233. package/src/static-handler.ts +126 -0
  234. package/src/theme/ThemeProvider.tsx +21 -15
  235. package/src/theme/ThemeScript.tsx +5 -5
  236. package/src/theme/constants.ts +5 -2
  237. package/src/theme/index.ts +4 -14
  238. package/src/theme/theme-context.ts +4 -30
  239. package/src/theme/theme-script.ts +21 -18
  240. package/src/types/boundaries.ts +158 -0
  241. package/src/types/cache-types.ts +198 -0
  242. package/src/types/error-types.ts +192 -0
  243. package/src/types/global-namespace.ts +100 -0
  244. package/src/types/handler-context.ts +791 -0
  245. package/src/types/index.ts +88 -0
  246. package/src/types/loader-types.ts +210 -0
  247. package/src/types/route-config.ts +170 -0
  248. package/src/types/route-entry.ts +109 -0
  249. package/src/types/segments.ts +151 -0
  250. package/src/types.ts +1 -1623
  251. package/src/urls/include-helper.ts +197 -0
  252. package/src/urls/index.ts +53 -0
  253. package/src/urls/path-helper-types.ts +346 -0
  254. package/src/urls/path-helper.ts +364 -0
  255. package/src/urls/pattern-types.ts +107 -0
  256. package/src/urls/response-types.ts +116 -0
  257. package/src/urls/type-extraction.ts +372 -0
  258. package/src/urls/urls-function.ts +98 -0
  259. package/src/urls.ts +1 -802
  260. package/src/use-loader.tsx +161 -81
  261. package/src/vite/discovery/bundle-postprocess.ts +181 -0
  262. package/src/vite/discovery/discover-routers.ts +348 -0
  263. package/src/vite/discovery/prerender-collection.ts +439 -0
  264. package/src/vite/discovery/route-types-writer.ts +258 -0
  265. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  266. package/src/vite/discovery/state.ts +117 -0
  267. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  268. package/src/vite/index.ts +15 -1129
  269. package/src/vite/plugin-types.ts +103 -0
  270. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  271. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  272. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  273. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -53
  274. package/src/vite/plugins/expose-id-utils.ts +299 -0
  275. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  276. package/src/vite/plugins/expose-ids/handler-transform.ts +209 -0
  277. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  278. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  279. package/src/vite/plugins/expose-ids/types.ts +45 -0
  280. package/src/vite/plugins/expose-internal-ids.ts +786 -0
  281. package/src/vite/plugins/performance-tracks.ts +88 -0
  282. package/src/vite/plugins/refresh-cmd.ts +127 -0
  283. package/src/vite/plugins/use-cache-transform.ts +323 -0
  284. package/src/vite/plugins/version-injector.ts +83 -0
  285. package/src/vite/plugins/version-plugin.ts +266 -0
  286. package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
  287. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  288. package/src/vite/rango.ts +462 -0
  289. package/src/vite/router-discovery.ts +918 -0
  290. package/src/vite/utils/ast-handler-extract.ts +517 -0
  291. package/src/vite/utils/banner.ts +36 -0
  292. package/src/vite/utils/bundle-analysis.ts +137 -0
  293. package/src/vite/utils/manifest-utils.ts +70 -0
  294. package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
  295. package/src/vite/utils/prerender-utils.ts +207 -0
  296. package/src/vite/utils/shared-utils.ts +170 -0
  297. package/CLAUDE.md +0 -43
  298. package/src/browser/lru-cache.ts +0 -69
  299. package/src/browser/request-controller.ts +0 -164
  300. package/src/cache/memory-store.ts +0 -253
  301. package/src/href-context.ts +0 -33
  302. package/src/href.ts +0 -255
  303. package/src/server/route-manifest-cache.ts +0 -173
  304. package/src/vite/expose-handle-id.ts +0 -209
  305. package/src/vite/expose-loader-id.ts +0 -426
  306. package/src/vite/expose-location-state-id.ts +0 -177
  307. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
package/src/index.ts CHANGED
@@ -1,18 +1,15 @@
1
1
  /**
2
- * rsc-router
2
+ * @rangojs/router
3
3
  *
4
- * Universal exports - types and utilities safe for both server and client
4
+ * Single user-facing entrypoint for all router APIs.
5
5
  *
6
- * For server-only exports (urls, createRouter, createLoader, etc.):
7
- * import from "rsc-router/server"
6
+ * The "react-server" export condition selects index.rsc.ts (real implementations)
7
+ * vs this file (client stubs for server-only functions).
8
8
  *
9
- * For client-only exports (Outlet, useOutlet, etc.):
10
- * import from "rsc-router/client"
9
+ * For client-only exports (Outlet, useOutlet, hooks, components):
10
+ * import from "@rangojs/router/client"
11
11
  */
12
12
 
13
- // Universal rendering utilities (work on both server and client)
14
- export { renderSegments } from "./segment-system.js";
15
-
16
13
  // Error classes (can be used on both server and client)
17
14
  export {
18
15
  RouteNotFoundError,
@@ -22,20 +19,22 @@ export {
22
19
  HandlerError,
23
20
  BuildError,
24
21
  InvalidHandlerError,
25
- NetworkError,
26
- isNetworkError,
27
- sanitizeError,
22
+ RouterError,
23
+ Skip,
24
+ isSkip,
28
25
  } from "./errors.js";
29
26
 
30
27
  // Types (safe to import anywhere - no runtime code)
31
28
  export type {
32
29
  // Configuration types
33
30
  DocumentProps,
34
- RouterEnv,
35
31
  DefaultEnv,
36
32
  RouteDefinition,
33
+ RouteConfig,
34
+ RouteDefinitionOptions,
35
+ TrailingSlashMode,
37
36
  // Handler types
38
- Handler, // Supports params object, path pattern, or route name
37
+ Handler, // Supports params object, path pattern, or route name
39
38
  HandlerContext,
40
39
  ExtractParams,
41
40
  GenericParams,
@@ -51,9 +50,6 @@ export type {
51
50
  LoaderContext,
52
51
  FetchableLoaderOptions,
53
52
  LoadOptions,
54
- LoaderActionContext,
55
- LoaderAction,
56
- LoaderMiddlewareFn,
57
53
  // Error boundary types
58
54
  ErrorInfo,
59
55
  ErrorBoundaryFallbackProps,
@@ -63,34 +59,231 @@ export type {
63
59
  NotFoundInfo,
64
60
  NotFoundBoundaryFallbackProps,
65
61
  NotFoundBoundaryHandler,
62
+ // Error handling callback types
63
+ ErrorPhase,
64
+ OnErrorContext,
65
+ OnErrorCallback,
66
66
  } from "./types.js";
67
67
 
68
+ // Search params schema types
69
+ export type {
70
+ SearchSchema,
71
+ SearchSchemaValue,
72
+ ResolveSearchSchema,
73
+ RouteSearchParams,
74
+ RouteParams,
75
+ } from "./search-params.js";
76
+
68
77
  // Client-safe createLoader - only stores the $$id, function is not included
69
78
  // Use this when defining loaders that will be imported by client components
70
79
  export { createLoader } from "./loader.js";
71
80
 
81
+ // Route definition types (safe to import anywhere)
82
+ export type { RouteHelpers, RouteHandlers } from "./route-definition.js";
83
+ export type { TransitionConfig, ViewTransitionClass } from "./types.js";
84
+
85
+ // Composition types for reusable callback factories
86
+ export type {
87
+ RouteUseItem,
88
+ LayoutUseItem,
89
+ AllUseItems,
90
+ UseItems,
91
+ HandlerUseItem,
92
+ } from "./route-types.js";
93
+
94
+ // Response route types (usable in both server and client contexts)
95
+ export type {
96
+ ResponseHandler,
97
+ ResponseHandlerContext,
98
+ JsonResponseHandler,
99
+ TextResponseHandler,
100
+ JsonValue,
101
+ ResponsePathFn,
102
+ JsonResponsePathFn,
103
+ TextResponsePathFn,
104
+ RouteResponse,
105
+ ResponseError,
106
+ ResponseEnvelope,
107
+ } from "./urls.js";
108
+
109
+ // Middleware context types
110
+ export type { MiddlewareContext, CookieOptions } from "./router/middleware.js";
111
+
112
+ function serverOnlyStubError(name: string): Error {
113
+ return new Error(
114
+ `${name}() is only available from "@rangojs/router" in a react-server/RSC environment. ` +
115
+ `For client hooks and components, import from "@rangojs/router/client".`,
116
+ );
117
+ }
118
+
72
119
  /**
73
120
  * Error-throwing stub for server-only `urls` function.
74
- * Import from "@rangojs/router/server" or use within RSC context instead.
75
121
  */
76
122
  export function urls(): never {
77
- throw new Error(
78
- 'urls() is server-only. Import from "@rangojs/router/server" instead, or ensure you\'re using it in a server component.'
79
- );
123
+ throw serverOnlyStubError("urls");
80
124
  }
81
125
 
82
126
  /**
83
127
  * Error-throwing stub for server-only `createRouter` function.
84
- * Import from "@rangojs/router/server" instead.
85
128
  */
86
129
  export function createRouter(): never {
87
- throw new Error(
88
- 'createRouter() is server-only. Import from "@rangojs/router/server" instead.'
89
- );
130
+ throw serverOnlyStubError("createRouter");
131
+ }
132
+
133
+ /**
134
+ * Error-throwing stub for server-only `redirect` function.
135
+ */
136
+ export function redirect(): never {
137
+ throw serverOnlyStubError("redirect");
138
+ }
139
+
140
+ // Handle API (universal - works on both server and client)
141
+ export { createHandle, isHandle, type Handle } from "./handle.js";
142
+
143
+ // Context variable API (typed ctx.set/ctx.get tokens)
144
+ export { createVar, type ContextVar } from "./context-var.js";
145
+
146
+ // CSP nonce token (use with ctx.get(nonce) in middleware/handlers)
147
+ export { nonce } from "./rsc/nonce.js";
148
+
149
+ /**
150
+ * Error-throwing stub for server-only `Prerender` function.
151
+ */
152
+ export function Prerender(): never {
153
+ throw serverOnlyStubError("Prerender");
154
+ }
155
+
156
+ /**
157
+ * Error-throwing stub for server-only `Passthrough` function.
158
+ */
159
+ export function Passthrough(): never {
160
+ throw serverOnlyStubError("Passthrough");
90
161
  }
91
162
 
92
- // Href type utilities for type-safe URL generation
93
- // ScopedHrefFunction is used with scopedHref<typeof patterns>() for composable modules
94
- export type { ScopedHrefFunction, HrefFunction, ExtractLocalRoutes } from "./href.js";
95
- // scopedHref() helper for handlers to get locally-typed href
96
- export { scopedHref } from "./href.js";
163
+ /**
164
+ * Error-throwing stub for server-only `Static` function.
165
+ */
166
+ export function Static(): never {
167
+ throw serverOnlyStubError("Static");
168
+ }
169
+
170
+ /**
171
+ * Error-throwing stub for server-only `getRequestContext` function.
172
+ */
173
+ export function getRequestContext(): never {
174
+ throw serverOnlyStubError("getRequestContext");
175
+ }
176
+
177
+ /**
178
+ * Error-throwing stub for server-only `cookies` function.
179
+ */
180
+ export function cookies(): never {
181
+ throw serverOnlyStubError("cookies");
182
+ }
183
+
184
+ /**
185
+ * Error-throwing stub for server-only `headers` function.
186
+ */
187
+ export function headers(): never {
188
+ throw serverOnlyStubError("headers");
189
+ }
190
+
191
+ /**
192
+ * Error-throwing stub for server-only `createReverse` function.
193
+ */
194
+ export function createReverse(): never {
195
+ throw serverOnlyStubError("createReverse");
196
+ }
197
+
198
+ // Error-throwing stubs for server-only route helpers
199
+ export function layout(): never {
200
+ throw serverOnlyStubError("layout");
201
+ }
202
+ export function cache(): never {
203
+ throw serverOnlyStubError("cache");
204
+ }
205
+ export function middleware(): never {
206
+ throw serverOnlyStubError("middleware");
207
+ }
208
+ export function revalidate(): never {
209
+ throw serverOnlyStubError("revalidate");
210
+ }
211
+ export function loader(): never {
212
+ throw serverOnlyStubError("loader");
213
+ }
214
+ export function loading(): never {
215
+ throw serverOnlyStubError("loading");
216
+ }
217
+ export function parallel(): never {
218
+ throw serverOnlyStubError("parallel");
219
+ }
220
+ export function intercept(): never {
221
+ throw serverOnlyStubError("intercept");
222
+ }
223
+ export function when(): never {
224
+ throw serverOnlyStubError("when");
225
+ }
226
+ export function errorBoundary(): never {
227
+ throw serverOnlyStubError("errorBoundary");
228
+ }
229
+ export function notFoundBoundary(): never {
230
+ throw serverOnlyStubError("notFoundBoundary");
231
+ }
232
+ export function transition(): never {
233
+ throw serverOnlyStubError("transition");
234
+ }
235
+
236
+ // Request context type (safe for client)
237
+ export type { PublicRequestContext as RequestContext } from "./server/request-context.js";
238
+
239
+ // Cookie store types (safe for client)
240
+ export type {
241
+ CookieStore,
242
+ Cookie,
243
+ ReadonlyHeaders,
244
+ } from "./server/cookie-store.js";
245
+
246
+ // Built-in handles (universal — work on both server and client)
247
+ export { Meta } from "./handles/meta.js";
248
+ export { Breadcrumbs } from "./handles/breadcrumbs.js";
249
+
250
+ // Meta types
251
+ export type { MetaDescriptor, MetaDescriptorBase } from "./router/types.js";
252
+
253
+ // Breadcrumb types
254
+ export type { BreadcrumbItem } from "./handles/breadcrumbs.js";
255
+
256
+ // Reverse type utilities for type-safe URL generation (Django-style URL reversal)
257
+ export type {
258
+ ScopedReverseFunction,
259
+ ReverseFunction,
260
+ ExtractLocalRoutes,
261
+ ParamsFor,
262
+ } from "./reverse.js";
263
+ // scopedReverse() helper for handlers to get locally-typed reverse
264
+ export { scopedReverse } from "./reverse.js";
265
+
266
+ // Location state (universal - works on both server and client)
267
+ export {
268
+ createLocationState,
269
+ type LocationStateDefinition,
270
+ type LocationStateEntry,
271
+ type LocationStateOptions,
272
+ } from "./browser/react/location-state-shared.js";
273
+
274
+ // Path-based response type lookup from RegisteredRoutes
275
+ export type { PathResponse } from "./href-client.js";
276
+
277
+ // Telemetry sink
278
+ export { createConsoleSink } from "./router/telemetry.js";
279
+ export { createOTelSink } from "./router/telemetry-otel.js";
280
+ export type { OTelTracer, OTelSpan } from "./router/telemetry-otel.js";
281
+ export type { TelemetrySink, TelemetryEvent } from "./router/telemetry.js";
282
+
283
+ // Timeout types and error class
284
+ export { RouterTimeoutError } from "./router/timeout.js";
285
+ export type {
286
+ RouterTimeouts,
287
+ TimeoutPhase,
288
+ TimeoutContext,
289
+ } from "./router/timeout.js";
@@ -0,0 +1,11 @@
1
+ // Internal debug gate. Enable with INTERNAL_RANGO_DEBUG=1 in the environment.
2
+ // Uses a Vite define (__RANGO_DEBUG__) for compile-time injection so it works
3
+ // in all runtimes including Cloudflare Workers where process.env is unavailable.
4
+ // Falls back to process.env for non-Vite contexts (tests, direct Node usage).
5
+ export const INTERNAL_RANGO_DEBUG: boolean =
6
+ typeof __RANGO_DEBUG__ !== "undefined"
7
+ ? __RANGO_DEBUG__
8
+ : typeof process !== "undefined" &&
9
+ Boolean((process as any).env?.INTERNAL_RANGO_DEBUG);
10
+
11
+ declare const __RANGO_DEBUG__: boolean;
package/src/loader.rsc.ts CHANGED
@@ -5,9 +5,9 @@
5
5
  * Only used in react-server context via export conditions.
6
6
  *
7
7
  * For non-fetchable loaders: returns a loader definition with fn included
8
- * For fetchable loaders: stores fn in registry and returns a serializable loader with action
8
+ * For fetchable loaders: stores fn in registry and returns a serializable loader
9
9
  *
10
- * The $$id is injected by the Vite exposeLoaderId plugin as a hidden parameter.
10
+ * The $$id is injected by the Vite exposeInternalIds plugin as a hidden parameter.
11
11
  * Users don't need to pass any name - IDs are auto-generated from file path.
12
12
  */
13
13
 
@@ -17,96 +17,58 @@ import type {
17
17
  LoaderFn,
18
18
  } from "./types.js";
19
19
  import type { MiddlewareFn } from "./router/middleware.js";
20
- import { getRequestContext } from "./server/request-context.js";
20
+ import {
21
+ registerFetchableLoader,
22
+ getFetchableLoader,
23
+ } from "./server/fetchable-loader-store.js";
21
24
 
22
- // Internal registry for fetchable loaders (server-side only)
23
- // Maps loader $$id to its function and middleware
24
- //
25
- // WHY TWO REGISTRIES?
26
- // This registry (fetchableLoaderRegistry) is populated immediately when createLoader() runs.
27
- // The other registry in loader-registry.ts (loaderRegistry) is a cache used by the RSC handler
28
- // for GET-based fetching. The RSC handler calls getFetchableLoader() from here to populate
29
- // its cache. This separation allows:
30
- // 1. Server actions to look up loaders directly without going through lazy loading
31
- // 2. The RSC handler to use lazy loading for production builds
32
- // 3. Both to share the same source of truth (this registry)
33
- const fetchableLoaderRegistry = new Map<
34
- string,
35
- { fn: LoaderFn<any, any, any>; middleware: MiddlewareFn[] }
36
- >();
37
-
38
- /**
39
- * Register a fetchable loader's function internally
40
- * Called during module initialization with the $$id
41
- */
42
- function registerFetchableLoader(
43
- id: string,
44
- fn: LoaderFn<any, any, any>,
45
- middleware: MiddlewareFn[]
46
- ): void {
47
- fetchableLoaderRegistry.set(id, { fn, middleware });
48
- }
49
-
50
- /**
51
- * Get a fetchable loader's function from the internal registry by $$id
52
- *
53
- * This is used internally by:
54
- * - Server actions (loaderAction) to execute loader functions
55
- * - loader-registry.ts to populate the main registry for GET-based fetching
56
- *
57
- * Loaders are registered here when createLoader() is called with fetchable: true.
58
- * The $$id is injected by the Vite exposeLoaderId plugin.
59
- *
60
- * @param id - The loader's $$id (auto-generated from file path + export name)
61
- * @returns The loader function and middleware, or undefined if not found
62
- *
63
- * @internal This is primarily for internal use by the router infrastructure
64
- */
65
- export function getFetchableLoader(
66
- id: string
67
- ): { fn: LoaderFn<any, any, any>; middleware: MiddlewareFn[] } | undefined {
68
- return fetchableLoaderRegistry.get(id);
69
- }
25
+ export { getFetchableLoader };
70
26
 
71
27
  // Overload 1: With function only (not fetchable)
72
28
  export function createLoader<T>(
73
- fn: LoaderFn<T, Record<string, string | undefined>, any>
29
+ fn: LoaderFn<T, Record<string, string | undefined>, any>,
74
30
  ): LoaderDefinition<Awaited<T>, Record<string, string | undefined>>;
75
31
 
76
32
  // Overload 2: Fetchable with `true` (no middleware)
77
33
  export function createLoader<T>(
78
34
  fn: LoaderFn<T, Record<string, string | undefined>, any>,
79
- fetchable: true
35
+ fetchable: true,
80
36
  ): LoaderDefinition<Awaited<T>, Record<string, string | undefined>>;
81
37
 
82
38
  // Overload 3: Fetchable with middleware options
83
39
  export function createLoader<T>(
84
40
  fn: LoaderFn<T, Record<string, string | undefined>, any>,
85
- options: FetchableLoaderOptions
41
+ options: FetchableLoaderOptions,
86
42
  ): LoaderDefinition<Awaited<T>, Record<string, string | undefined>>;
87
43
 
88
44
  // Implementation - the $$id parameter is injected by Vite plugin, not user-provided
89
45
  export function createLoader<T>(
90
46
  fn: LoaderFn<T, Record<string, string | undefined>, any>,
91
47
  fetchable?: true | FetchableLoaderOptions,
92
- // Hidden parameter injected by Vite exposeLoaderId plugin
93
- __injectedId?: string
48
+ // Hidden parameter injected by Vite exposeInternalIds plugin
49
+ __injectedId?: string,
94
50
  ): LoaderDefinition<Awaited<T>, Record<string, string | undefined>> {
95
51
  // The $$id will be set on the returned object by Vite plugin
96
52
  // For fetchable loaders, __injectedId is also passed as a parameter
97
53
  const loaderId = __injectedId || "";
98
54
 
99
- // Serializes to just { __brand, $$id } when passed as a prop to client components.
100
- // RSC Flight protocol calls toJSON when serializing objects.
101
- const toJSON = () => ({ __brand: "loader" as const, $$id: loaderId });
55
+ if (!loaderId && process.env.NODE_ENV === "development") {
56
+ throw new Error(
57
+ "[rsc-router] Loader is missing $$id. " +
58
+ "Make sure the exposeInternalIds Vite plugin is enabled and " +
59
+ "the loader is exported with: export const MyLoader = createLoader(...)",
60
+ );
61
+ }
102
62
 
103
- // If not fetchable, return a simple stub with fn included
63
+ // If not fetchable, store fn in registry (for SSR ctx.use() resolution)
64
+ // but mark fetchable=false so the _rsc_loader endpoint rejects it.
104
65
  if (fetchable === undefined) {
66
+ if (fn && loaderId) {
67
+ registerFetchableLoader(loaderId, fn, [], false);
68
+ }
105
69
  return {
106
70
  __brand: "loader",
107
71
  $$id: loaderId,
108
- fn: fn as LoaderFn<Awaited<T>, Record<string, string | undefined>, any>,
109
- toJSON,
110
72
  };
111
73
  }
112
74
 
@@ -115,106 +77,13 @@ export function createLoader<T>(
115
77
  fetchable === true ? [] : fetchable?.middleware || [];
116
78
 
117
79
  // Register the function in the internal registry by $$id (server-side only)
118
- // The server action will look it up by $$id when executed
80
+ // The loader fetch handler looks it up by $$id when load() is called from the client.
119
81
  if (fn && loaderId) {
120
- registerFetchableLoader(loaderId, fn, middleware);
82
+ registerFetchableLoader(loaderId, fn, middleware, true);
121
83
  }
122
84
 
123
- // Create server action for form-based fetching
124
- // This action is serializable and can be passed to client components
125
- // The loaderId is captured in closure (it's a primitive string)
126
- //
127
- // IMPORTANT: The signature must be (prevState, formData) for useActionState compatibility.
128
- // When used with useActionState, React passes the previous state as the first argument.
129
- // The prevState is ignored here since loaders are stateless data fetchers.
130
- async function loaderAction(
131
- _prevState: Awaited<T> | null,
132
- formData: FormData
133
- ): Promise<Awaited<T>> {
134
- "use server";
135
-
136
- // Look up the loader from registry by $$id
137
- const registered = fetchableLoaderRegistry.get(loaderId);
138
- if (!registered) {
139
- throw new Error(`Loader "${loaderId}" not found in registry`);
140
- }
141
-
142
- // Get request context (env, request, url, variables) from the RSC handler
143
- // This is set by runWithRequestContext in rsc/index.ts when executing actions
144
- const requestCtx = getRequestContext();
145
-
146
- // Convert FormData to params object
147
- const params: Record<string, string> = {};
148
- formData.forEach((value, key) => {
149
- if (typeof value === "string") {
150
- params[key] = value;
151
- }
152
- });
153
-
154
- // Use real request/url from context, or fall back to synthetic for edge cases
155
- const actionUrl = requestCtx?.url ?? new URL("http://localhost/");
156
- const actionRequest = requestCtx?.request ?? new Request(actionUrl, { method: "POST" });
157
- const env = requestCtx?.env ?? {};
158
-
159
- // Merge variables from request context (app-level middleware) with loader-specific variables
160
- // requestCtx.var is the shared variables object from the handler
161
- const variables: Record<string, any> = { ...requestCtx?.var };
162
-
163
- // Execute middleware for auth checks, headers, cookies
164
- // Headers/cookies set on ctx.res will be merged into the final response
165
- if (registered.middleware.length > 0 && requestCtx?.res) {
166
- const { executeServerActionMiddleware } = await import(
167
- "./router/middleware.js"
168
- );
169
- await executeServerActionMiddleware(
170
- registered.middleware,
171
- actionRequest,
172
- env,
173
- params,
174
- variables,
175
- requestCtx.res
176
- );
177
- }
178
-
179
- // Build context using createHandlerContext for consistency with route handlers
180
- // Variables are now accessed from request context via getRequestContext()
181
- const { createHandlerContext } = await import("./router/handler-context.js");
182
- const baseCtx = createHandlerContext(
183
- params,
184
- actionRequest,
185
- actionUrl.searchParams,
186
- actionUrl.pathname,
187
- actionUrl,
188
- env
189
- );
190
-
191
- // Extend with server action specific properties
192
- const ctx: any = {
193
- ...baseCtx,
194
- method: "POST",
195
- formData,
196
- };
197
-
198
- // Execute and return result
199
- return registered.fn(ctx);
200
- }
201
-
202
- // For fetchable loaders, include action in toJSON so it survives RSC serialization.
203
- // loaderAction has "use server" so RSC Flight serializes it as a server action reference.
204
- // Without this, passing a loader as a prop to a client component would lose the action,
205
- // breaking useActionState() which needs loader.action for progressive enhancement.
206
- const fetchableToJSON = () => ({
207
- __brand: "loader" as const,
208
- $$id: loaderId,
209
- action: loaderAction,
210
- });
211
-
212
- // Return a loader object with action for form-based fetching
213
- // The exposeLoaderId plugin will set $$id on this object
214
85
  return {
215
86
  __brand: "loader",
216
87
  $$id: loaderId,
217
- action: loaderAction,
218
- toJSON: fetchableToJSON,
219
88
  };
220
89
  }
package/src/loader.ts CHANGED
@@ -2,10 +2,15 @@
2
2
  * rsc-router/loader (client version)
3
3
  *
4
4
  * Client-only stub for createLoader. Returns a minimal loader definition
5
- * that can be passed to hooks like useLoader. The actual loader function
6
- * is not included - it only exists on the server.
5
+ * ({ __brand, $$id }) that can be passed to hooks like useLoader.
6
+ * The actual loader function is not included -- it only exists on the server.
7
7
  *
8
- * The $$id is injected by the Vite exposeLoaderId plugin.
8
+ * For export-only loader files, the Vite plugin replaces the entire file with
9
+ * object literals (bypassing this function). Those stubs only contain
10
+ * { __brand, $$id }.
11
+ * This function only runs when loaders are in mixed files (not export-only).
12
+ *
13
+ * The $$id is injected by the Vite exposeInternalIds plugin.
9
14
  */
10
15
 
11
16
  import type {
@@ -16,32 +21,44 @@ import type {
16
21
 
17
22
  // Overload 1: With function only (not fetchable)
18
23
  export function createLoader<T>(
19
- fn: LoaderFn<T, Record<string, string | undefined>, any>
24
+ fn: LoaderFn<T, Record<string, string | undefined>, any>,
20
25
  ): LoaderDefinition<Awaited<T>, Record<string, string | undefined>>;
21
26
 
22
27
  // Overload 2: Fetchable with `true` (no middleware)
23
28
  export function createLoader<T>(
24
29
  fn: LoaderFn<T, Record<string, string | undefined>, any>,
25
- fetchable: true
30
+ fetchable: true,
26
31
  ): LoaderDefinition<Awaited<T>, Record<string, string | undefined>>;
27
32
 
28
33
  // Overload 3: Fetchable with middleware options
29
34
  export function createLoader<T>(
30
35
  fn: LoaderFn<T, Record<string, string | undefined>, any>,
31
- options: FetchableLoaderOptions
36
+ options: FetchableLoaderOptions,
32
37
  ): LoaderDefinition<Awaited<T>, Record<string, string | undefined>>;
33
38
 
34
39
  // Implementation - client stub that just returns the loader definition
35
40
  // The $$id parameter is injected by Vite plugin, not user-provided
41
+ //
42
+ // NOTE: For export-only loader files, the Vite plugin replaces the entire
43
+ // file with object literals (bypassing this function). This function only
44
+ // runs when loaders are in mixed files (not export-only).
36
45
  export function createLoader<T>(
37
46
  _fn: LoaderFn<T, Record<string, string | undefined>, any>,
38
47
  _fetchable?: true | FetchableLoaderOptions,
39
- __injectedId?: string
48
+ __injectedId?: string,
40
49
  ): LoaderDefinition<Awaited<T>, Record<string, string | undefined>> {
41
- // Client only needs the $$id for identification
42
- // The actual loader function is only used on the server
50
+ const loaderId = __injectedId || "";
51
+
52
+ if (!loaderId && process.env.NODE_ENV === "development") {
53
+ throw new Error(
54
+ "[rsc-router] Loader is missing $$id. " +
55
+ "Make sure the exposeInternalIds Vite plugin is enabled and " +
56
+ "the loader is exported with: export const MyLoader = createLoader(...)",
57
+ );
58
+ }
59
+
43
60
  return {
44
61
  __brand: "loader",
45
- $$id: __injectedId || "",
62
+ $$id: loaderId,
46
63
  };
47
64
  }
@@ -16,6 +16,8 @@ interface NetworkErrorThrowerProps {
16
16
  * 1. Errors must be thrown during React's render phase to be caught by error boundaries
17
17
  * 2. The error occurs in async code (fetch), so we need to propagate it to React's render
18
18
  */
19
- export function NetworkErrorThrower({ error }: NetworkErrorThrowerProps): ReactNode {
19
+ export function NetworkErrorThrower({
20
+ error,
21
+ }: NetworkErrorThrowerProps): ReactNode {
20
22
  throw error;
21
23
  }
@@ -0,0 +1,45 @@
1
+ "use client";
2
+
3
+ import { useContext, useMemo, type ReactNode } from "react";
4
+ import { OutletContext, type OutletContextValue } from "./outlet-context.js";
5
+ import type { ResolvedSegment } from "./types.js";
6
+
7
+ /**
8
+ * Provider for outlet content - used internally by renderSegments
9
+ *
10
+ * Stores a reference to parent context so useLoader can walk up the chain
11
+ * to find loader data from parent layouts. If this segment defines a loading
12
+ * component, Outlet will wrap content with Suspense using that as fallback.
13
+ */
14
+ export function OutletProvider({
15
+ content,
16
+ parallel,
17
+ segment,
18
+ loaderData,
19
+ children,
20
+ }: {
21
+ content: ReactNode;
22
+ parallel?: ResolvedSegment[];
23
+ segment?: ResolvedSegment;
24
+ loaderData?: Record<string, any>;
25
+ children: ReactNode;
26
+ }): ReactNode {
27
+ // Get parent context to enable walking up the chain for loader lookups
28
+ const parentContext = useContext(OutletContext);
29
+
30
+ const value = useMemo(
31
+ () => ({
32
+ content,
33
+ parallel,
34
+ segment,
35
+ loaderData,
36
+ parent: parentContext,
37
+ loading: segment?.loading,
38
+ }),
39
+ [content, parallel, segment, loaderData, parentContext],
40
+ );
41
+
42
+ return (
43
+ <OutletContext.Provider value={value}>{children}</OutletContext.Provider>
44
+ );
45
+ }