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

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 +55 -33
  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 +743 -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 +1373 -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 +150 -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/server.ts CHANGED
@@ -1,146 +1,51 @@
1
1
  /**
2
- * rsc-router/server
2
+ * @rangojs/router/server — Internal subpath
3
3
  *
4
- * Server-only exports for route definition and building
5
- * These should only be imported in server-side handler files
4
+ * This module is NOT user-facing. Import from "@rangojs/router" instead.
5
+ *
6
+ * Exports here are consumed by the Vite plugin (discovery, manifest injection,
7
+ * virtual modules) and the RSC handler internals. They are not part of the
8
+ * public API and may change without notice.
6
9
  */
7
10
 
8
- // Route definition helpers (server-only)
9
- export {
10
- createLoader,
11
- redirect,
12
- type RouteHelpers,
13
- type RouteHandlers,
14
- } from "./route-definition.js";
11
+ // Router registry (used by Vite plugin for build-time discovery)
12
+ export { RSC_ROUTER_BRAND, RouterRegistry } from "./router.js";
15
13
 
16
- // Django-style URL patterns (server-only)
14
+ // Host router registry (used by Vite plugin for host-router lazy discovery)
17
15
  export {
18
- urls,
19
- type PathHelpers,
20
- type PathOptions,
21
- type UrlPatterns,
22
- type IncludeOptions,
23
- } from "./urls.js";
24
-
25
- // Re-export IncludeItem from route-types
26
- export type { IncludeItem } from "./route-types.js";
16
+ HostRouterRegistry,
17
+ type HostRouterRegistryEntry,
18
+ } from "./host/router.js";
27
19
 
28
- // Core router (server-only)
20
+ // Route map builder (Vite plugin injects these via virtual modules)
29
21
  export {
30
- createRouter,
31
- RSC_ROUTER_BRAND,
32
- RouterRegistry,
33
- type RSCRouter,
34
- type RSCRouterOptions,
35
- type RootLayoutProps,
36
- } from "./router.js";
22
+ registerRouteMap,
23
+ setCachedManifest,
24
+ clearCachedManifest,
25
+ clearAllRouterData,
26
+ getGlobalRouteMap,
27
+ getRouterManifest,
28
+ setPrecomputedEntries,
29
+ setRouteTrie,
30
+ setManifestReadyPromise,
31
+ setRouterManifest,
32
+ setRouterTrie,
33
+ setRouterPrecomputedEntries,
34
+ registerRouterManifestLoader,
35
+ ensureRouterManifest,
36
+ } from "./route-map-builder.js";
37
37
 
38
- // Type-safe href utilities
38
+ // Loader registry (Vite plugin registers lazy loader imports)
39
39
  export {
40
- createHref,
41
- type HrefFunction,
42
- type PrefixedRoutes,
43
- type PrefixRoutePatterns,
44
- type ParamsFor,
45
- type SanitizePrefix,
46
- type MergeRoutes,
47
- } from "./href.js";
48
-
49
- // Segment system (server-only)
50
- export { renderSegments } from "./segment-system.js";
51
-
52
- // Performance tracking (server-only)
53
- export { track } from "./server/context.js";
54
-
55
- // Handle API (works in both server and client contexts)
56
- export { createHandle, isHandle, type Handle } from "./handle.js";
57
-
58
- // Built-in handles
59
- export { Meta } from "./handles/meta.js";
60
-
61
- // Loader registry (for GET-based loader fetching)
62
- export { registerLoaderById, setLoaderImports } from "./server/loader-registry.js";
63
-
64
- // Route map builder (for build-time manifest registration)
65
- export { registerRouteMap, setCachedManifest } from "./route-map-builder.js";
40
+ registerLoaderById,
41
+ setLoaderImports,
42
+ } from "./server/loader-registry.js";
66
43
 
67
- // Request context (for accessing request data in server components/actions)
44
+ // Request context creation (used by RSC handler, not user-facing)
68
45
  export {
69
- getRequestContext,
70
- requireRequestContext,
71
46
  createRequestContext,
72
- type RequestContext,
73
47
  type CreateRequestContextOptions,
74
48
  } from "./server/request-context.js";
75
49
 
76
- // Meta types
77
- export type { MetaDescriptor, MetaDescriptorBase } from "./router/types.js";
78
-
79
- // Middleware context types (Middleware type is exported from types.ts)
80
- export type {
81
- MiddlewareContext,
82
- CookieOptions,
83
- } from "./router/middleware.js";
84
-
85
- // Error classes and utilities
86
- export {
87
- RouteNotFoundError,
88
- DataNotFoundError,
89
- notFound,
90
- MiddlewareError,
91
- HandlerError,
92
- BuildError,
93
- InvalidHandlerError,
94
- sanitizeError,
95
- } from "./errors.js";
96
-
97
- // Component utilities
98
- export {
99
- isClientComponent,
100
- assertClientComponent,
101
- } from "./component-utils.js";
102
-
103
- // Debug utilities for route matching (development only)
104
- export {
105
- enableMatchDebug,
106
- getMatchDebugStats,
107
- } from "./router/pattern-matching.js";
108
-
109
- // Types (re-exported for convenience - user-facing only)
110
- export type {
111
- // Configuration types
112
- RouterEnv,
113
- DefaultEnv,
114
- RouteDefinition,
115
- RouteConfig,
116
- RouteDefinitionOptions,
117
- TrailingSlashMode,
118
- // Handler types
119
- Handler, // Supports params object, path pattern, or route name
120
- HandlerContext,
121
- ExtractParams,
122
- GenericParams,
123
- // Middleware types (also exported from router/middleware.js above)
124
- Middleware, // Supports env type and optional route name for params
125
- // Revalidation types
126
- RevalidateParams,
127
- Revalidate,
128
- RouteKeys,
129
- // Loader types
130
- LoaderDefinition,
131
- LoaderFn,
132
- LoaderContext,
133
- // Error boundary types
134
- ErrorInfo,
135
- ErrorBoundaryFallbackProps,
136
- ErrorBoundaryHandler,
137
- ClientErrorBoundaryFallbackProps,
138
- // NotFound boundary types
139
- NotFoundInfo,
140
- NotFoundBoundaryFallbackProps,
141
- NotFoundBoundaryHandler,
142
- // Error handling callback types
143
- ErrorPhase,
144
- OnErrorContext,
145
- OnErrorCallback,
146
- } from "./types.js";
50
+ // Component utilities (used internally for server/client boundary checks)
51
+ export { isClientComponent, assertClientComponent } from "./component-utils.js";
package/src/ssr/index.tsx CHANGED
@@ -1,14 +1,18 @@
1
1
  import React from "react";
2
- import { initHandleDataSync } from "../browser/react/use-handle.js";
3
- import { initSegmentsSync } from "../browser/react/use-segments.js";
4
- import { initThemeConfigSync } from "../theme/theme-context.js";
2
+ import { renderSegments } from "../segment-system.js";
3
+ import { filterSegmentOrder } from "../browser/react/filter-segment-order.js";
5
4
  import { ThemeProvider } from "../theme/ThemeProvider.js";
5
+ import { NonceContext } from "../browser/react/nonce-context.js";
6
6
  import { NavigationStoreContext } from "../browser/react/context.js";
7
7
  import type { NavigationStoreContextValue } from "../browser/react/context.js";
8
8
  import type { HandleData } from "../browser/types.js";
9
9
  import type { ErrorPhase } from "../types.js";
10
+ import type { ResolvedSegment } from "../types.js";
10
11
  import type { ResolvedThemeConfig, Theme } from "../theme/types.js";
11
- import type { EventController, DerivedNavigationState } from "../browser/event-controller.js";
12
+ import type {
13
+ EventController,
14
+ DerivedNavigationState,
15
+ } from "../browser/event-controller.js";
12
16
 
13
17
  /**
14
18
  * Options for injectRSCPayload
@@ -29,6 +33,13 @@ interface RenderToReadableStreamOptions {
29
33
  formState?: unknown;
30
34
  }
31
35
 
36
+ /**
37
+ * ReadableStream with the allReady promise added by react-dom/server.edge.
38
+ */
39
+ interface ReactDOMReadableStream extends ReadableStream<Uint8Array> {
40
+ allReady: Promise<void>;
41
+ }
42
+
32
43
  /**
33
44
  * Options for the renderHTML function
34
45
  */
@@ -45,6 +56,14 @@ export interface SSRRenderOptions {
45
56
  * Nonce for Content Security Policy (CSP)
46
57
  */
47
58
  nonce?: string;
59
+
60
+ /**
61
+ * SSR stream mode.
62
+ *
63
+ * - `"stream"` (default) — start flushing HTML immediately.
64
+ * - `"allReady"` — await `stream.allReady` before returning.
65
+ */
66
+ streamMode?: import("../router/router-options.js").SSRStreamMode;
48
67
  }
49
68
 
50
69
  /**
@@ -54,22 +73,24 @@ export interface SSRDependencies<TEnv = unknown> {
54
73
  /**
55
74
  * createFromReadableStream from @vitejs/plugin-rsc/ssr
56
75
  */
57
- createFromReadableStream: <T>(stream: ReadableStream<Uint8Array>) => Promise<T>;
76
+ createFromReadableStream: <T>(
77
+ stream: ReadableStream<Uint8Array>,
78
+ ) => Promise<T>;
58
79
 
59
80
  /**
60
81
  * renderToReadableStream from react-dom/server.edge
61
82
  */
62
83
  renderToReadableStream: (
63
84
  element: React.ReactNode,
64
- options?: RenderToReadableStreamOptions
65
- ) => Promise<ReadableStream<Uint8Array>>;
85
+ options?: RenderToReadableStreamOptions,
86
+ ) => Promise<ReactDOMReadableStream>;
66
87
 
67
88
  /**
68
89
  * injectRSCPayload from rsc-html-stream/server
69
90
  */
70
91
  injectRSCPayload: (
71
92
  rscStream: ReadableStream<Uint8Array>,
72
- options?: InjectRSCPayloadOptions
93
+ options?: InjectRSCPayloadOptions,
73
94
  ) => TransformStream<Uint8Array, Uint8Array>;
74
95
 
75
96
  /**
@@ -101,13 +122,17 @@ export interface SSRDependencies<TEnv = unknown> {
101
122
  * RSC payload type (minimal interface for SSR)
102
123
  */
103
124
  interface RscPayload {
104
- root: React.ReactNode;
105
125
  metadata?: {
126
+ segments?: ResolvedSegment[];
127
+ rootLayout?: React.ComponentType<{ children: React.ReactNode }>;
106
128
  handles?: AsyncGenerator<HandleData, void, unknown>;
107
129
  matched?: string[];
108
130
  pathname?: string;
131
+ params?: Record<string, string>;
132
+ basename?: string;
109
133
  themeConfig?: ResolvedThemeConfig | null;
110
134
  initialTheme?: Theme;
135
+ version?: string;
111
136
  };
112
137
  }
113
138
 
@@ -116,7 +141,7 @@ interface RscPayload {
116
141
  * Used for SSR where we need to await all handle data before rendering.
117
142
  */
118
143
  async function consumeAsyncGenerator(
119
- generator: AsyncGenerator<HandleData, void, unknown>
144
+ generator: AsyncGenerator<HandleData, void, unknown>,
120
145
  ): Promise<HandleData> {
121
146
  let lastData: HandleData = {};
122
147
  for await (const data of generator) {
@@ -129,11 +154,22 @@ async function consumeAsyncGenerator(
129
154
  * Create a minimal event controller for SSR.
130
155
  * This provides the correct pathname so useNavigation returns the right value during SSR.
131
156
  */
132
- function createSsrEventController(pathname: string): EventController {
133
- const location = new URL(pathname, "http://localhost");
157
+ function createSsrEventController(opts: {
158
+ pathname: string;
159
+ params?: Record<string, string>;
160
+ handleData?: HandleData;
161
+ matched?: string[];
162
+ }): EventController {
163
+ const location = new URL(opts.pathname, "http://localhost");
164
+ let params = opts.params ?? {};
165
+ const handleState = {
166
+ data: opts.handleData ?? {},
167
+ segmentOrder: filterSegmentOrder(opts.matched ?? []),
168
+ };
134
169
  const state: DerivedNavigationState = {
135
170
  state: "idle",
136
171
  isStreaming: false,
172
+ isNavigating: false,
137
173
  location,
138
174
  pendingUrl: null,
139
175
  inflightActions: [],
@@ -141,6 +177,7 @@ function createSsrEventController(pathname: string): EventController {
141
177
 
142
178
  return {
143
179
  getState: () => state,
180
+ getLocation: () => location,
144
181
  subscribe: () => () => {},
145
182
  getActionState: () => ({
146
183
  state: "idle",
@@ -152,7 +189,11 @@ function createSsrEventController(pathname: string): EventController {
152
189
  subscribeToAction: () => () => {},
153
190
  subscribeToHandles: () => () => {},
154
191
  setHandleData: () => {},
155
- getHandleState: () => ({ data: {}, segmentOrder: [] }),
192
+ getHandleState: () => handleState,
193
+ setParams: (nextParams) => {
194
+ params = nextParams;
195
+ },
196
+ getParams: () => params,
156
197
  setLocation: () => {},
157
198
  startNavigation: () => {
158
199
  throw new Error("Navigation not supported during SSR");
@@ -164,6 +205,7 @@ function createSsrEventController(pathname: string): EventController {
164
205
  abortAllActions: () => {},
165
206
  getCurrentNavigation: () => null,
166
207
  getInflightActions: () => new Map(),
208
+ hadAnyConcurrentActions: () => false,
167
209
  };
168
210
  }
169
211
 
@@ -203,9 +245,9 @@ export function createSSRHandler<TEnv = unknown>(deps: SSRDependencies<TEnv>) {
203
245
  */
204
246
  return async function renderHTML(
205
247
  rscStream: ReadableStream<Uint8Array>,
206
- options?: SSRRenderOptions
248
+ options?: SSRRenderOptions,
207
249
  ): Promise<ReadableStream<Uint8Array>> {
208
- const { nonce, formState } = options ?? {};
250
+ const { nonce, formState, streamMode } = options ?? {};
209
251
 
210
252
  try {
211
253
  // Tee the stream:
@@ -221,46 +263,69 @@ export function createSSRHandler<TEnv = unknown>(deps: SSRDependencies<TEnv>) {
221
263
  payload ??= createFromReadableStream<RscPayload>(rscStream1);
222
264
  const resolved = React.use(payload);
223
265
 
224
- // Initialize segments state before children render (for useSegments hook)
225
- initSegmentsSync(resolved.metadata?.matched, resolved.metadata?.pathname);
226
-
227
- // Initialize theme config for MetaTags to render theme script
228
266
  const themeConfig = resolved.metadata?.themeConfig ?? null;
229
- initThemeConfigSync(themeConfig);
267
+ const pathname = resolved.metadata?.pathname ?? "/";
230
268
 
231
- // Await handles and initialize state before children render
269
+ // Await handles before creating SSR event controller so hooks can
270
+ // read request-local handle data via NavigationStoreContext.
232
271
  // The handles property is an async generator that yields on each push
233
272
  // Memoize the promise since async generators can only be iterated once
273
+ let handleData: HandleData = {};
234
274
  if (resolved.metadata?.handles) {
235
275
  handlesPromise ??= consumeAsyncGenerator(resolved.metadata.handles);
236
- const handleData = React.use(handlesPromise);
237
- initHandleDataSync(handleData, resolved.metadata.matched);
276
+ handleData = React.use(handlesPromise);
238
277
  }
239
278
 
240
- // Create SSR context with correct pathname for useNavigation
279
+ // Create SSR context with request-local pathname/params/handles.
241
280
  ssrContextValue ??= {
242
281
  store: null as any,
243
- eventController: createSsrEventController(resolved.metadata?.pathname ?? "/"),
282
+ eventController: createSsrEventController({
283
+ pathname,
284
+ params: resolved.metadata?.params,
285
+ handleData,
286
+ matched: resolved.metadata?.matched,
287
+ }),
244
288
  navigate: async () => {},
245
289
  refresh: async () => {},
290
+ version: resolved.metadata?.version,
291
+ basename: resolved.metadata?.basename,
246
292
  };
247
293
 
248
- // Build content tree with all necessary providers
294
+ // Build content tree from segments.
249
295
  // Order must match NavigationProvider: NavigationStoreContext > ThemeProvider > content
250
- let content: React.ReactNode = resolved.root;
296
+ const reconstructedRoot = renderSegments(
297
+ resolved.metadata?.segments ?? [],
298
+ {
299
+ rootLayout: resolved.metadata?.rootLayout,
300
+ },
301
+ );
302
+ let content: React.ReactNode =
303
+ reconstructedRoot instanceof Promise
304
+ ? React.use(reconstructedRoot)
305
+ : reconstructedRoot;
251
306
 
252
307
  // Wrap content with ThemeProvider if theme is enabled
253
308
  if (themeConfig) {
254
309
  content = (
255
- <ThemeProvider config={themeConfig} initialTheme={resolved.metadata?.initialTheme}>
310
+ <ThemeProvider
311
+ config={themeConfig}
312
+ initialTheme={resolved.metadata?.initialTheme}
313
+ >
256
314
  {content}
257
315
  </ThemeProvider>
258
316
  );
259
317
  }
260
318
 
319
+ // Wrap with NonceContext so client components (e.g. MetaTags) can
320
+ // apply CSP nonces to inline scripts during SSR. Always present to
321
+ // match the browser-side NavigationProvider tree shape for hydration.
322
+ content = (
323
+ <NonceContext.Provider value={nonce}>{content}</NonceContext.Provider>
324
+ );
325
+
261
326
  // Wrap with NavigationStoreContext for useNavigation hook
262
327
  return (
263
- <NavigationStoreContext.Provider value={ssrContextValue}>
328
+ <NavigationStoreContext.Provider value={ssrContextValue!}>
264
329
  {content}
265
330
  </NavigationStoreContext.Provider>
266
331
  );
@@ -278,12 +343,20 @@ export function createSSRHandler<TEnv = unknown>(deps: SSRDependencies<TEnv>) {
278
343
  nonce,
279
344
  });
280
345
 
346
+ // Wait for all Suspense boundaries to resolve when streamMode is "allReady".
347
+ // This buffers the entire HTML before flushing — used for bots that
348
+ // cannot process streamed HTML.
349
+ if (streamMode === "allReady") {
350
+ await htmlStream.allReady;
351
+ }
352
+
281
353
  // Inject RSC payload into HTML as <script nonce="...">__FLIGHT_DATA__</script>
282
354
  return htmlStream.pipeThrough(injectRSCPayload(rscStream2, { nonce }));
283
355
  } catch (error) {
284
356
  // Invoke onError callback if provided
285
357
  if (onError) {
286
- const errorObj = error instanceof Error ? error : new Error(String(error));
358
+ const errorObj =
359
+ error instanceof Error ? error : new Error(String(error));
287
360
  try {
288
361
  onError(errorObj, { phase: "rendering" });
289
362
  } catch (callbackError) {
@@ -0,0 +1,126 @@
1
+ /**
2
+ * Static handler definition for build-time rendering of individual segments.
3
+ *
4
+ * Static wraps a handler so that in production the segment is
5
+ * rendered once at build time. The handler is then replaced with a static
6
+ * asset import -- no runtime store lookup needed.
7
+ *
8
+ * In dev mode, Static behaves as a normal handler: the wrapped
9
+ * function runs on every request, identical to a regular layout/path handler.
10
+ *
11
+ * The $$id is auto-generated by the Vite exposeInternalIds plugin based on
12
+ * file path and export name. No manual naming required.
13
+ *
14
+ * Key difference from Prerender:
15
+ * - Prerender: route-scoped, produces URLs via getParams, renders subtree per-params
16
+ * - Static: segment-scoped, renders once, no URLs, no params
17
+ *
18
+ * Works on: layout(), parallel(), and path().
19
+ *
20
+ * @example
21
+ * ```ts
22
+ * export const DocsNav = Static((ctx) => <Nav docs={readDocsSync()} />);
23
+ * export const DocShell = Static((ctx) => <Shell />);
24
+ *
25
+ * urls(({ path, layout }) => [
26
+ * layout(DocsNav, () => [
27
+ * path("/getting-started", DocShell, { name: "doc.gs" }),
28
+ * path("/:slug", Prerender(getParams, DocPageHandler)),
29
+ * ]),
30
+ * ]);
31
+ * ```
32
+ */
33
+ import type { ReactNode } from "react";
34
+ import type { Handler } from "./types.js";
35
+ import type { StaticBuildContext } from "./prerender.js";
36
+ import type { UseItems, HandlerUseItem } from "./route-types.js";
37
+ import { isCachedFunction } from "./cache/taint.js";
38
+
39
+ // -- Types ------------------------------------------------------------------
40
+
41
+ export interface StaticHandlerOptions {
42
+ /**
43
+ * Keep handler in server bundle for live fallback (default: false).
44
+ * false: handler replaced with stub, source-only APIs excluded from bundle.
45
+ * true: handler stays in bundle, renders live at request time.
46
+ */
47
+ passthrough?: boolean;
48
+ }
49
+
50
+ export interface StaticHandlerDefinition<
51
+ TParams extends Record<string, any> = any,
52
+ > {
53
+ readonly __brand: "staticHandler";
54
+ /** Auto-generated unique ID (injected by Vite plugin). */
55
+ $$id: string;
56
+ /** In dev mode, the actual handler function that layout/path/parallel can call. */
57
+ handler: Handler<TParams>;
58
+ /** Static handler options (passthrough support). */
59
+ options?: StaticHandlerOptions;
60
+ /** Composable default DSL items merged when the handler is mounted. */
61
+ use?: () => UseItems<HandlerUseItem>;
62
+ }
63
+
64
+ // -- Function ---------------------------------------------------------------
65
+
66
+ export function Static<TParams extends Record<string, any> = {}>(
67
+ handler: (ctx: StaticBuildContext) => ReactNode | Promise<ReactNode>,
68
+ options?: StaticHandlerOptions,
69
+ __injectedId?: string,
70
+ ): StaticHandlerDefinition<TParams>;
71
+
72
+ // -- Implementation ---------------------------------------------------------
73
+
74
+ export function Static<TParams extends Record<string, any>>(
75
+ handler: Function,
76
+ optionsOrId?: StaticHandlerOptions | string,
77
+ maybeId?: string,
78
+ ): StaticHandlerDefinition<TParams> {
79
+ if (isCachedFunction(handler)) {
80
+ throw new Error(
81
+ 'A "use cache" function cannot be used as a Static() handler. ' +
82
+ "Static handlers are rendered once at build time. Remove the " +
83
+ '"use cache" directive — Static already provides caching.',
84
+ );
85
+ }
86
+
87
+ let options: StaticHandlerOptions | undefined;
88
+ let id: string;
89
+
90
+ if (typeof optionsOrId === "string") {
91
+ id = optionsOrId;
92
+ } else {
93
+ options = optionsOrId as StaticHandlerOptions | undefined;
94
+ id = maybeId ?? "";
95
+ }
96
+
97
+ if (!id) {
98
+ throw new Error(
99
+ "[rsc-router] Static: missing $$id. " +
100
+ "Ensure the exposeInternalIds Vite plugin is configured.",
101
+ );
102
+ }
103
+
104
+ return {
105
+ __brand: "staticHandler" as const,
106
+ $$id: id,
107
+ handler: handler as Handler<TParams>,
108
+ ...(options ? { options } : {}),
109
+ };
110
+ }
111
+
112
+ // -- Type guard -------------------------------------------------------------
113
+
114
+ /**
115
+ * Type guard to check if a value is a StaticHandlerDefinition.
116
+ */
117
+ export function isStaticHandler(
118
+ value: unknown,
119
+ ): value is StaticHandlerDefinition {
120
+ return (
121
+ typeof value === "object" &&
122
+ value !== null &&
123
+ "__brand" in value &&
124
+ (value as { __brand: unknown }).__brand === "staticHandler"
125
+ );
126
+ }
@@ -11,7 +11,13 @@
11
11
  * - Handles SSR hydration by deferring system theme detection
12
12
  */
13
13
 
14
- import React, { useCallback, useEffect, useMemo, useState, useRef } from "react";
14
+ import React, {
15
+ useCallback,
16
+ useEffect,
17
+ useMemo,
18
+ useState,
19
+ useRef,
20
+ } from "react";
15
21
  import { ThemeContext } from "./theme-context.js";
16
22
  import type {
17
23
  ResolvedTheme,
@@ -44,7 +50,12 @@ function readThemeFromCookie(storageKey: string): string | null {
44
50
  for (const cookie of cookies) {
45
51
  const [name, ...rest] = cookie.trim().split("=");
46
52
  if (name === storageKey) {
47
- return decodeURIComponent(rest.join("="));
53
+ const raw = rest.join("=");
54
+ try {
55
+ return decodeURIComponent(raw);
56
+ } catch {
57
+ return raw;
58
+ }
48
59
  }
49
60
  }
50
61
  return null;
@@ -90,15 +101,13 @@ function writeThemeToStorage(storageKey: string, theme: Theme): void {
90
101
  /**
91
102
  * Apply theme to HTML element
92
103
  */
93
- function applyThemeToDocument(
94
- theme: Theme,
95
- config: ResolvedThemeConfig
96
- ): void {
104
+ function applyThemeToDocument(theme: Theme, config: ResolvedThemeConfig): void {
97
105
  if (typeof document === "undefined") return;
98
106
 
99
- const resolved = theme === "system" && config.enableSystem
100
- ? getSystemTheme()
101
- : (theme as ResolvedTheme);
107
+ const resolved =
108
+ theme === "system" && config.enableSystem
109
+ ? getSystemTheme()
110
+ : (theme as ResolvedTheme);
102
111
 
103
112
  const value = config.value[resolved] || resolved;
104
113
  const el = document.documentElement;
@@ -188,7 +197,7 @@ export function ThemeProvider({
188
197
  writeThemeToStorage(config.storageKey, newTheme);
189
198
  applyThemeToDocument(newTheme, config);
190
199
  },
191
- [config]
200
+ [config],
192
201
  );
193
202
 
194
203
  // Listen for system preference changes
@@ -227,10 +236,7 @@ export function ThemeProvider({
227
236
  if (!newTheme) return;
228
237
 
229
238
  // Validate and apply
230
- if (
231
- newTheme === "system" ||
232
- config.themes.includes(newTheme)
233
- ) {
239
+ if (newTheme === "system" || config.themes.includes(newTheme)) {
234
240
  setThemeState(newTheme as Theme);
235
241
  applyThemeToDocument(newTheme as Theme, config);
236
242
  }
@@ -280,7 +286,7 @@ export function ThemeProvider({
280
286
  themes,
281
287
  config,
282
288
  }),
283
- [theme, setTheme, resolvedTheme, systemTheme, themes, config, mounted]
289
+ [theme, setTheme, resolvedTheme, systemTheme, themes, config, mounted],
284
290
  );
285
291
 
286
292
  return (
@@ -49,13 +49,13 @@ export interface ThemeScriptProps {
49
49
  * This renders a synchronous inline script that applies the theme
50
50
  * to the HTML element before React hydration, preventing FOUC.
51
51
  */
52
- export function ThemeScript({ config, nonce }: ThemeScriptProps): React.ReactNode {
52
+ export function ThemeScript({
53
+ config,
54
+ nonce,
55
+ }: ThemeScriptProps): React.ReactNode {
53
56
  const scriptContent = generateThemeScript(config);
54
57
 
55
58
  return (
56
- <script
57
- nonce={nonce}
58
- dangerouslySetInnerHTML={{ __html: scriptContent }}
59
- />
59
+ <script nonce={nonce} dangerouslySetInnerHTML={{ __html: scriptContent }} />
60
60
  );
61
61
  }