@rangojs/router 0.0.0-experimental.31 → 0.0.0-experimental.3232cd17

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 (376) hide show
  1. package/AGENTS.md +4 -0
  2. package/README.md +198 -44
  3. package/dist/bin/rango.js +287 -105
  4. package/dist/testing/vitest.js +82 -0
  5. package/dist/vite/index.js +3248 -1117
  6. package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  7. package/package.json +73 -21
  8. package/skills/api-client/SKILL.md +211 -0
  9. package/skills/breadcrumbs/SKILL.md +107 -1
  10. package/skills/bundle-analysis/SKILL.md +159 -0
  11. package/skills/cache-guide/SKILL.md +245 -21
  12. package/skills/caching/SKILL.md +302 -6
  13. package/skills/composability/SKILL.md +27 -2
  14. package/skills/css/SKILL.md +76 -0
  15. package/skills/document-cache/SKILL.md +78 -55
  16. package/skills/handler-use/SKILL.md +364 -0
  17. package/skills/hooks/SKILL.md +270 -30
  18. package/skills/host-router/SKILL.md +82 -22
  19. package/skills/i18n/SKILL.md +276 -0
  20. package/skills/intercept/SKILL.md +49 -5
  21. package/skills/layout/SKILL.md +35 -9
  22. package/skills/links/SKILL.md +249 -17
  23. package/skills/loader/SKILL.md +294 -30
  24. package/skills/middleware/SKILL.md +52 -13
  25. package/skills/migrate-nextjs/SKILL.md +584 -0
  26. package/skills/migrate-react-router/SKILL.md +769 -0
  27. package/skills/mime-routes/SKILL.md +27 -0
  28. package/skills/observability/SKILL.md +137 -0
  29. package/skills/parallel/SKILL.md +203 -7
  30. package/skills/prerender/SKILL.md +123 -100
  31. package/skills/rango/SKILL.md +250 -22
  32. package/skills/react-compiler/SKILL.md +168 -0
  33. package/skills/response-routes/SKILL.md +122 -47
  34. package/skills/route/SKILL.md +97 -5
  35. package/skills/router-setup/SKILL.md +90 -5
  36. package/skills/server-actions/SKILL.md +775 -0
  37. package/skills/streams-and-websockets/SKILL.md +283 -0
  38. package/skills/tailwind/SKILL.md +27 -3
  39. package/skills/testing/SKILL.md +129 -0
  40. package/skills/testing/bindings.md +89 -0
  41. package/skills/testing/cache-prerender.md +124 -0
  42. package/skills/testing/client-components.md +122 -0
  43. package/skills/testing/e2e-parity.md +125 -0
  44. package/skills/testing/flight.md +92 -0
  45. package/skills/testing/handles.md +129 -0
  46. package/skills/testing/loader.md +128 -0
  47. package/skills/testing/middleware.md +99 -0
  48. package/skills/testing/render-handler.md +121 -0
  49. package/skills/testing/response-routes.md +95 -0
  50. package/skills/testing/reverse-and-types.md +84 -0
  51. package/skills/testing/server-actions.md +107 -0
  52. package/skills/testing/server-tree.md +128 -0
  53. package/skills/testing/setup.md +120 -0
  54. package/skills/typesafety/SKILL.md +329 -27
  55. package/skills/use-cache/SKILL.md +36 -5
  56. package/skills/view-transitions/SKILL.md +294 -0
  57. package/src/__augment-tests__/augment.ts +81 -0
  58. package/src/__augment-tests__/augmented.check.ts +116 -0
  59. package/src/__internal.ts +67 -40
  60. package/src/browser/action-coordinator.ts +53 -36
  61. package/src/browser/action-fence.ts +47 -0
  62. package/src/browser/app-shell.ts +39 -0
  63. package/src/browser/app-version.ts +14 -0
  64. package/src/browser/cookie-name.ts +140 -0
  65. package/src/browser/event-controller.ts +86 -147
  66. package/src/browser/history-state.ts +21 -0
  67. package/src/browser/index.ts +3 -3
  68. package/src/browser/invalidate-client-cache.ts +52 -0
  69. package/src/browser/link-interceptor.ts +4 -0
  70. package/src/browser/navigation-bridge.ts +148 -19
  71. package/src/browser/navigation-client.ts +187 -67
  72. package/src/browser/navigation-store-handle.ts +38 -0
  73. package/src/browser/navigation-store.ts +76 -67
  74. package/src/browser/navigation-transaction.ts +18 -66
  75. package/src/browser/partial-update.ts +123 -94
  76. package/src/browser/prefetch/cache.ts +214 -36
  77. package/src/browser/prefetch/fetch.ts +260 -38
  78. package/src/browser/prefetch/policy.ts +6 -0
  79. package/src/browser/prefetch/queue.ts +126 -20
  80. package/src/browser/prefetch/resource-ready.ts +77 -0
  81. package/src/browser/rango-state.ts +158 -76
  82. package/src/browser/react/Link.tsx +93 -11
  83. package/src/browser/react/NavigationProvider.tsx +115 -34
  84. package/src/browser/react/ScrollRestoration.tsx +10 -6
  85. package/src/browser/react/context.ts +7 -2
  86. package/src/browser/react/filter-segment-order.ts +49 -7
  87. package/src/browser/react/index.ts +0 -48
  88. package/src/browser/react/location-state-shared.ts +166 -8
  89. package/src/browser/react/location-state.ts +39 -14
  90. package/src/browser/react/use-action.ts +6 -15
  91. package/src/browser/react/use-handle.ts +23 -69
  92. package/src/browser/react/use-link-status.ts +0 -4
  93. package/src/browser/react/use-navigation.ts +22 -5
  94. package/src/browser/react/use-params.ts +20 -10
  95. package/src/browser/react/use-reverse.ts +106 -0
  96. package/src/browser/react/use-router.ts +46 -11
  97. package/src/browser/react/use-search-params.ts +0 -5
  98. package/src/browser/react/use-segments.ts +11 -21
  99. package/src/browser/response-adapter.ts +52 -1
  100. package/src/browser/rsc-router.tsx +215 -76
  101. package/src/browser/scroll-restoration.ts +46 -39
  102. package/src/browser/segment-reconciler.ts +36 -9
  103. package/src/browser/segment-structure-assert.ts +2 -2
  104. package/src/browser/server-action-bridge.ts +176 -50
  105. package/src/browser/types.ts +95 -11
  106. package/src/browser/validate-redirect-origin.ts +43 -16
  107. package/src/build/collect-fallback-refs.ts +107 -0
  108. package/src/build/generate-manifest.ts +65 -40
  109. package/src/build/generate-route-types.ts +5 -0
  110. package/src/build/index.ts +8 -2
  111. package/src/build/prefix-tree-utils.ts +123 -0
  112. package/src/build/route-trie.ts +137 -32
  113. package/src/build/route-types/codegen.ts +4 -4
  114. package/src/build/route-types/include-resolution.ts +9 -2
  115. package/src/build/route-types/param-extraction.ts +6 -3
  116. package/src/build/route-types/per-module-writer.ts +7 -4
  117. package/src/build/route-types/router-processing.ts +278 -96
  118. package/src/build/route-types/scan-filter.ts +9 -2
  119. package/src/build/route-types/source-scan.ts +118 -0
  120. package/src/build/runtime-discovery.ts +9 -20
  121. package/src/cache/cache-error.ts +104 -0
  122. package/src/cache/cache-policy.ts +68 -28
  123. package/src/cache/cache-runtime.ts +149 -43
  124. package/src/cache/cache-scope.ts +148 -81
  125. package/src/cache/cache-tag.ts +98 -0
  126. package/src/cache/cf/cf-cache-store.ts +2550 -93
  127. package/src/cache/cf/index.ts +11 -17
  128. package/src/cache/document-cache.ts +78 -27
  129. package/src/cache/handle-snapshot.ts +63 -0
  130. package/src/cache/index.ts +23 -20
  131. package/src/cache/memory-segment-store.ts +136 -37
  132. package/src/cache/profile-registry.ts +6 -30
  133. package/src/cache/read-through-swr.ts +41 -11
  134. package/src/cache/segment-codec.ts +0 -16
  135. package/src/cache/tag-invalidation.ts +230 -0
  136. package/src/cache/taint.ts +55 -0
  137. package/src/cache/types.ts +33 -100
  138. package/src/cache/vercel/index.ts +11 -0
  139. package/src/cache/vercel/vercel-cache-store.ts +799 -0
  140. package/src/client.rsc.tsx +6 -21
  141. package/src/client.tsx +108 -290
  142. package/src/component-utils.ts +19 -0
  143. package/src/context-var.ts +84 -2
  144. package/src/debug.ts +2 -2
  145. package/src/decode-loader-results.ts +36 -0
  146. package/src/defer.ts +196 -0
  147. package/src/deps/ssr.ts +0 -1
  148. package/src/errors.ts +30 -4
  149. package/src/handle.ts +70 -22
  150. package/src/handles/MetaTags.tsx +0 -14
  151. package/src/handles/breadcrumbs.ts +16 -5
  152. package/src/handles/meta.ts +0 -39
  153. package/src/host/cookie-handler.ts +0 -36
  154. package/src/host/errors.ts +0 -24
  155. package/src/host/index.ts +8 -2
  156. package/src/host/pattern-matcher.ts +7 -50
  157. package/src/host/router.ts +107 -99
  158. package/src/host/testing.ts +40 -27
  159. package/src/host/types.ts +37 -4
  160. package/src/host/utils.ts +1 -1
  161. package/src/href-client.ts +137 -22
  162. package/src/index.rsc.ts +52 -26
  163. package/src/index.ts +100 -38
  164. package/src/internal-debug.ts +2 -4
  165. package/src/loader-store.ts +500 -0
  166. package/src/loader.rsc.ts +20 -13
  167. package/src/loader.ts +12 -11
  168. package/src/missing-id-error.ts +68 -0
  169. package/src/network-error-thrower.tsx +1 -6
  170. package/src/outlet-context.ts +1 -1
  171. package/src/outlet-provider.tsx +1 -5
  172. package/src/prerender/param-hash.ts +10 -11
  173. package/src/prerender/store.ts +37 -41
  174. package/src/prerender.ts +198 -82
  175. package/src/redirect-origin.ts +100 -0
  176. package/src/response-utils.ts +37 -0
  177. package/src/reverse.ts +65 -15
  178. package/src/root-error-boundary.tsx +1 -19
  179. package/src/route-content-wrapper.tsx +7 -72
  180. package/src/route-definition/dsl-helpers.ts +437 -274
  181. package/src/route-definition/helper-factories.ts +29 -139
  182. package/src/route-definition/helpers-types.ts +113 -37
  183. package/src/route-definition/index.ts +3 -0
  184. package/src/route-definition/redirect.ts +52 -10
  185. package/src/route-definition/resolve-handler-use.ts +161 -0
  186. package/src/route-definition/use-item-types.ts +32 -0
  187. package/src/route-map-builder.ts +7 -17
  188. package/src/route-types.ts +37 -41
  189. package/src/router/basename.ts +14 -0
  190. package/src/router/content-negotiation.ts +108 -9
  191. package/src/router/error-handling.ts +13 -17
  192. package/src/router/find-match.ts +45 -22
  193. package/src/router/handler-context.ts +83 -41
  194. package/src/router/intercept-resolution.ts +25 -23
  195. package/src/router/lazy-includes.ts +19 -53
  196. package/src/router/loader-resolution.ts +213 -30
  197. package/src/router/logging.ts +5 -8
  198. package/src/router/manifest.ts +49 -45
  199. package/src/router/match-api.ts +121 -205
  200. package/src/router/match-context.ts +0 -22
  201. package/src/router/match-handlers.ts +58 -58
  202. package/src/router/match-middleware/background-revalidation.ts +27 -6
  203. package/src/router/match-middleware/cache-lookup.ts +205 -249
  204. package/src/router/match-middleware/cache-store.ts +45 -32
  205. package/src/router/match-middleware/intercept-resolution.ts +8 -28
  206. package/src/router/match-middleware/segment-resolution.ts +52 -18
  207. package/src/router/match-pipelines.ts +1 -42
  208. package/src/router/match-result.ts +104 -40
  209. package/src/router/metrics.ts +5 -34
  210. package/src/router/middleware-types.ts +13 -142
  211. package/src/router/middleware.ts +173 -143
  212. package/src/router/navigation-snapshot.ts +131 -0
  213. package/src/router/params-util.ts +23 -0
  214. package/src/router/pattern-matching.ts +109 -63
  215. package/src/router/prerender-match.ts +192 -54
  216. package/src/router/preview-match.ts +32 -102
  217. package/src/router/request-classification.ts +276 -0
  218. package/src/router/revalidation.ts +63 -55
  219. package/src/router/route-snapshot.ts +244 -0
  220. package/src/router/router-context.ts +6 -28
  221. package/src/router/router-interfaces.ts +100 -35
  222. package/src/router/router-options.ts +91 -11
  223. package/src/router/router-registry.ts +2 -5
  224. package/src/router/segment-resolution/fresh.ts +242 -75
  225. package/src/router/segment-resolution/helpers.ts +64 -25
  226. package/src/router/segment-resolution/loader-cache.ts +41 -37
  227. package/src/router/segment-resolution/revalidation.ts +456 -372
  228. package/src/router/segment-resolution/static-store.ts +19 -5
  229. package/src/router/segment-resolution/streamed-handler-telemetry.ts +52 -0
  230. package/src/router/segment-resolution/view-transition-default.ts +36 -0
  231. package/src/router/segment-resolution.ts +4 -1
  232. package/src/router/segment-wrappers.ts +2 -3
  233. package/src/router/state-cookie-name.ts +33 -0
  234. package/src/router/substitute-pattern-params.ts +56 -0
  235. package/src/router/telemetry-otel.ts +0 -20
  236. package/src/router/telemetry.ts +96 -19
  237. package/src/router/timeout.ts +0 -20
  238. package/src/router/trie-matching.ts +91 -46
  239. package/src/router/types.ts +10 -63
  240. package/src/router/url-params.ts +44 -0
  241. package/src/router.ts +134 -43
  242. package/src/rsc/handler-context.ts +3 -2
  243. package/src/rsc/handler.ts +492 -383
  244. package/src/rsc/helpers.ts +162 -46
  245. package/src/rsc/index.ts +1 -1
  246. package/src/rsc/json-route-result.ts +38 -0
  247. package/src/rsc/loader-fetch.ts +23 -3
  248. package/src/rsc/manifest-init.ts +33 -42
  249. package/src/rsc/origin-guard.ts +39 -25
  250. package/src/rsc/progressive-enhancement.ts +30 -3
  251. package/src/rsc/redirect-guard.ts +99 -0
  252. package/src/rsc/response-error.ts +79 -12
  253. package/src/rsc/response-route-handler.ts +90 -63
  254. package/src/rsc/rsc-rendering.ts +56 -54
  255. package/src/rsc/runtime-warnings.ts +23 -10
  256. package/src/rsc/server-action.ts +74 -67
  257. package/src/rsc/ssr-setup.ts +18 -2
  258. package/src/rsc/types.ts +25 -6
  259. package/src/runtime-env.ts +18 -0
  260. package/src/search-params.ts +4 -20
  261. package/src/segment-content-promise.ts +67 -0
  262. package/src/segment-loader-promise.ts +134 -0
  263. package/src/segment-system.tsx +272 -129
  264. package/src/serialize.ts +243 -0
  265. package/src/server/context.ts +309 -61
  266. package/src/server/cookie-store.ts +80 -5
  267. package/src/server/handle-store.ts +26 -24
  268. package/src/server/loader-registry.ts +10 -28
  269. package/src/server/request-context.ts +348 -128
  270. package/src/ssr/index.tsx +23 -15
  271. package/src/static-handler.ts +27 -18
  272. package/src/testing/cache-status.ts +162 -0
  273. package/src/testing/collect-handle.ts +40 -0
  274. package/src/testing/dispatch.ts +618 -0
  275. package/src/testing/dom.entry.ts +22 -0
  276. package/src/testing/e2e/fixture.ts +188 -0
  277. package/src/testing/e2e/index.ts +128 -0
  278. package/src/testing/e2e/matchers.ts +35 -0
  279. package/src/testing/e2e/page-helpers.ts +272 -0
  280. package/src/testing/e2e/parity.ts +387 -0
  281. package/src/testing/e2e/server.ts +195 -0
  282. package/src/testing/flight-matchers.ts +97 -0
  283. package/src/testing/flight-normalize.ts +11 -0
  284. package/src/testing/flight-runtime.d.ts +57 -0
  285. package/src/testing/flight-tree.ts +682 -0
  286. package/src/testing/flight.entry.ts +52 -0
  287. package/src/testing/flight.ts +232 -0
  288. package/src/testing/generated-routes.ts +183 -0
  289. package/src/testing/index.ts +99 -0
  290. package/src/testing/internal/context.ts +348 -0
  291. package/src/testing/internal/flight-client-globals.ts +30 -0
  292. package/src/testing/internal/seed-vars.ts +54 -0
  293. package/src/testing/render-handler.ts +330 -0
  294. package/src/testing/render-route.tsx +566 -0
  295. package/src/testing/run-loader.ts +378 -0
  296. package/src/testing/run-middleware.ts +205 -0
  297. package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
  298. package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
  299. package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
  300. package/src/testing/vitest-stubs/version.ts +5 -0
  301. package/src/testing/vitest.ts +305 -0
  302. package/src/theme/ThemeProvider.tsx +0 -52
  303. package/src/theme/ThemeScript.tsx +0 -6
  304. package/src/theme/constants.ts +0 -12
  305. package/src/theme/index.ts +0 -7
  306. package/src/theme/theme-context.ts +1 -5
  307. package/src/theme/theme-script.ts +0 -14
  308. package/src/theme/use-theme.ts +0 -3
  309. package/src/types/boundaries.ts +0 -35
  310. package/src/types/cache-types.ts +17 -8
  311. package/src/types/error-types.ts +30 -90
  312. package/src/types/global-namespace.ts +54 -41
  313. package/src/types/handler-context.ts +233 -81
  314. package/src/types/index.ts +1 -10
  315. package/src/types/loader-types.ts +44 -15
  316. package/src/types/request-scope.ts +107 -0
  317. package/src/types/route-config.ts +6 -50
  318. package/src/types/route-entry.ts +19 -7
  319. package/src/types/segments.ts +37 -14
  320. package/src/urls/include-helper.ts +33 -70
  321. package/src/urls/index.ts +1 -11
  322. package/src/urls/path-helper-types.ts +58 -11
  323. package/src/urls/path-helper.ts +57 -111
  324. package/src/urls/pattern-types.ts +48 -19
  325. package/src/urls/response-types.ts +25 -22
  326. package/src/urls/type-extraction.ts +58 -139
  327. package/src/urls/urls-function.ts +1 -18
  328. package/src/use-loader.tsx +346 -89
  329. package/src/vite/debug.ts +185 -0
  330. package/src/vite/discovery/bundle-postprocess.ts +36 -38
  331. package/src/vite/discovery/discover-routers.ts +130 -85
  332. package/src/vite/discovery/discovery-errors.ts +194 -0
  333. package/src/vite/discovery/gate-state.ts +171 -0
  334. package/src/vite/discovery/prerender-collection.ts +192 -99
  335. package/src/vite/discovery/route-types-writer.ts +40 -84
  336. package/src/vite/discovery/self-gen-tracking.ts +27 -1
  337. package/src/vite/discovery/state.ts +51 -6
  338. package/src/vite/discovery/virtual-module-codegen.ts +14 -34
  339. package/src/vite/index.ts +8 -0
  340. package/src/vite/plugin-types.ts +187 -69
  341. package/src/vite/plugins/cjs-to-esm.ts +8 -18
  342. package/src/vite/plugins/client-ref-dedup.ts +16 -11
  343. package/src/vite/plugins/client-ref-hashing.ts +28 -15
  344. package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
  345. package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  346. package/src/vite/plugins/cloudflare-protocol-stub.ts +194 -0
  347. package/src/vite/plugins/expose-action-id.ts +49 -98
  348. package/src/vite/plugins/expose-id-utils.ts +11 -50
  349. package/src/vite/plugins/expose-ids/export-analysis.ts +76 -34
  350. package/src/vite/plugins/expose-ids/handler-transform.ts +10 -48
  351. package/src/vite/plugins/expose-ids/loader-transform.ts +3 -20
  352. package/src/vite/plugins/expose-ids/router-transform.ts +20 -16
  353. package/src/vite/plugins/expose-internal-ids.ts +554 -317
  354. package/src/vite/plugins/performance-tracks.ts +89 -0
  355. package/src/vite/plugins/refresh-cmd.ts +89 -27
  356. package/src/vite/plugins/use-cache-transform.ts +73 -83
  357. package/src/vite/plugins/vercel-output.ts +258 -0
  358. package/src/vite/plugins/version-injector.ts +21 -25
  359. package/src/vite/plugins/version-plugin.ts +41 -20
  360. package/src/vite/plugins/virtual-entries.ts +2 -17
  361. package/src/vite/rango.ts +257 -289
  362. package/src/vite/router-discovery.ts +930 -140
  363. package/src/vite/utils/ast-handler-extract.ts +15 -31
  364. package/src/vite/utils/banner.ts +4 -4
  365. package/src/vite/utils/bundle-analysis.ts +10 -15
  366. package/src/vite/utils/client-chunks.ts +184 -0
  367. package/src/vite/utils/forward-user-plugins.ts +171 -0
  368. package/src/vite/utils/manifest-utils.ts +4 -59
  369. package/src/vite/utils/package-resolution.ts +20 -52
  370. package/src/vite/utils/prerender-utils.ts +27 -29
  371. package/src/vite/utils/shared-utils.ts +92 -42
  372. package/src/browser/action-response-classifier.ts +0 -99
  373. package/src/browser/react/use-client-cache.ts +0 -58
  374. package/src/browser/shallow.ts +0 -40
  375. package/src/handles/index.ts +0 -7
  376. package/src/router/middleware-cookies.ts +0 -55
package/src/ssr/index.tsx CHANGED
@@ -71,7 +71,7 @@ export interface SSRRenderOptions {
71
71
  */
72
72
  export interface SSRDependencies<TEnv = unknown> {
73
73
  /**
74
- * createFromReadableStream from @vitejs/plugin-rsc/ssr
74
+ * createFromReadableStream from @rangojs/router/internal/deps/ssr
75
75
  */
76
76
  createFromReadableStream: <T>(
77
77
  stream: ReadableStream<Uint8Array>,
@@ -86,7 +86,7 @@ export interface SSRDependencies<TEnv = unknown> {
86
86
  ) => Promise<ReactDOMReadableStream>;
87
87
 
88
88
  /**
89
- * injectRSCPayload from rsc-html-stream/server
89
+ * injectRSCPayload from @rangojs/router/internal/deps/html-stream-server
90
90
  */
91
91
  injectRSCPayload: (
92
92
  rscStream: ReadableStream<Uint8Array>,
@@ -129,6 +129,7 @@ interface RscPayload {
129
129
  matched?: string[];
130
130
  pathname?: string;
131
131
  params?: Record<string, string>;
132
+ basename?: string;
132
133
  themeConfig?: ResolvedThemeConfig | null;
133
134
  initialTheme?: Theme;
134
135
  version?: string;
@@ -161,13 +162,18 @@ function createSsrEventController(opts: {
161
162
  }): EventController {
162
163
  const location = new URL(opts.pathname, "http://localhost");
163
164
  let params = opts.params ?? {};
165
+ const rawMatched = opts.matched ?? [];
164
166
  const handleState = {
165
167
  data: opts.handleData ?? {},
166
- segmentOrder: filterSegmentOrder(opts.matched ?? []),
168
+ segmentOrder: filterSegmentOrder(rawMatched),
169
+ routeSegmentIds: rawMatched.filter(
170
+ (id) => !id.includes(".@") && !/D\d+\./.test(id),
171
+ ),
167
172
  };
168
173
  const state: DerivedNavigationState = {
169
174
  state: "idle",
170
175
  isStreaming: false,
176
+ isNavigating: false,
171
177
  location,
172
178
  pendingUrl: null,
173
179
  inflightActions: [],
@@ -212,10 +218,10 @@ function createSsrEventController(opts: {
212
218
  *
213
219
  * @example
214
220
  * ```tsx
215
- * import { createSSRHandler } from "rsc-router/ssr";
216
- * import { createFromReadableStream } from "@vitejs/plugin-rsc/ssr";
221
+ * import { createSSRHandler } from "@rangojs/router/ssr";
222
+ * import { createFromReadableStream } from "@rangojs/router/internal/deps/ssr";
217
223
  * import { renderToReadableStream } from "react-dom/server.edge";
218
- * import { injectRSCPayload } from "rsc-html-stream/server";
224
+ * import { injectRSCPayload } from "@rangojs/router/internal/deps/html-stream-server";
219
225
  *
220
226
  * export const renderHTML = createSSRHandler({
221
227
  * createFromReadableStream,
@@ -257,9 +263,11 @@ export function createSSRHandler<TEnv = unknown>(deps: SSRDependencies<TEnv>) {
257
263
  let payload: Promise<RscPayload> | undefined;
258
264
  let handlesPromise: Promise<HandleData> | undefined;
259
265
  let ssrContextValue: NavigationStoreContextValue | undefined;
266
+ let rootPromise: Promise<React.ReactNode> | undefined;
260
267
  function SsrRoot() {
261
268
  payload ??= createFromReadableStream<RscPayload>(rscStream1);
262
269
  const resolved = React.use(payload);
270
+
263
271
  const themeConfig = resolved.metadata?.themeConfig ?? null;
264
272
  const pathname = resolved.metadata?.pathname ?? "/";
265
273
 
@@ -285,20 +293,20 @@ export function createSSRHandler<TEnv = unknown>(deps: SSRDependencies<TEnv>) {
285
293
  navigate: async () => {},
286
294
  refresh: async () => {},
287
295
  version: resolved.metadata?.version,
296
+ basename: resolved.metadata?.basename,
288
297
  };
289
298
 
290
299
  // Build content tree from segments.
291
- // Order must match NavigationProvider: NavigationStoreContext > ThemeProvider > content
292
- const reconstructedRoot = renderSegments(
293
- resolved.metadata?.segments ?? [],
294
- {
300
+ // Order must match NavigationProvider: NavigationStoreContext > NonceContext > ThemeProvider > content
301
+ // Memoize like payload/handles above: renderSegments is async, so
302
+ // React.use() on a fresh promise suspends and replays SsrRoot, which
303
+ // would re-run the entire segment-tree build on every initial render.
304
+ rootPromise ??= Promise.resolve(
305
+ renderSegments(resolved.metadata?.segments ?? [], {
295
306
  rootLayout: resolved.metadata?.rootLayout,
296
- },
307
+ }),
297
308
  );
298
- let content: React.ReactNode =
299
- reconstructedRoot instanceof Promise
300
- ? React.use(reconstructedRoot)
301
- : reconstructedRoot;
309
+ let content: React.ReactNode = React.use(rootPromise);
302
310
 
303
311
  // Wrap content with ThemeProvider if theme is enabled
304
312
  if (themeConfig) {
@@ -32,10 +32,19 @@
32
32
  */
33
33
  import type { ReactNode } from "react";
34
34
  import type { Handler } from "./types.js";
35
- import type { PrerenderOptions, StaticBuildContext } from "./prerender.js";
35
+ import type { StaticBuildContext } from "./prerender.js";
36
+ import type { UseItems, HandlerUseItem } from "./route-types.js";
36
37
  import { isCachedFunction } from "./cache/taint.js";
38
+ import { isUnderTestRunner } from "./runtime-env.js";
37
39
 
38
- // -- Types ------------------------------------------------------------------
40
+ export interface StaticHandlerOptions {
41
+ /**
42
+ * Keep handler in server bundle for live fallback (default: false).
43
+ * false: handler replaced with stub, source-only APIs excluded from bundle.
44
+ * true: handler stays in bundle, renders live at request time.
45
+ */
46
+ passthrough?: boolean;
47
+ }
39
48
 
40
49
  export interface StaticHandlerDefinition<
41
50
  TParams extends Record<string, any> = any,
@@ -46,22 +55,24 @@ export interface StaticHandlerDefinition<
46
55
  /** In dev mode, the actual handler function that layout/path/parallel can call. */
47
56
  handler: Handler<TParams>;
48
57
  /** Static handler options (passthrough support). */
49
- options?: PrerenderOptions;
58
+ options?: StaticHandlerOptions;
59
+ /** Composable default DSL items merged when the handler is mounted. */
60
+ use?: () => UseItems<HandlerUseItem>;
50
61
  }
51
62
 
52
- // -- Function ---------------------------------------------------------------
63
+ // Process-stable fallback id counter (mirrors createHandle/createLoader/Prerender).
64
+ // Only assigned in bare unit tests where the Vite plugin did not inject an id.
65
+ let runtimeStaticIdCounter = 0;
53
66
 
54
67
  export function Static<TParams extends Record<string, any> = {}>(
55
68
  handler: (ctx: StaticBuildContext) => ReactNode | Promise<ReactNode>,
56
- options?: PrerenderOptions,
69
+ options?: StaticHandlerOptions,
57
70
  __injectedId?: string,
58
71
  ): StaticHandlerDefinition<TParams>;
59
72
 
60
- // -- Implementation ---------------------------------------------------------
61
-
62
73
  export function Static<TParams extends Record<string, any>>(
63
74
  handler: Function,
64
- optionsOrId?: PrerenderOptions | string,
75
+ optionsOrId?: StaticHandlerOptions | string,
65
76
  maybeId?: string,
66
77
  ): StaticHandlerDefinition<TParams> {
67
78
  if (isCachedFunction(handler)) {
@@ -72,22 +83,25 @@ export function Static<TParams extends Record<string, any>>(
72
83
  );
73
84
  }
74
85
 
75
- let options: PrerenderOptions | undefined;
86
+ let options: StaticHandlerOptions | undefined;
76
87
  let id: string;
77
88
 
78
89
  if (typeof optionsOrId === "string") {
79
90
  id = optionsOrId;
80
91
  } else {
81
- options = optionsOrId as PrerenderOptions | undefined;
92
+ options = optionsOrId as StaticHandlerOptions | undefined;
82
93
  id = maybeId ?? "";
83
94
  }
84
95
 
85
- if (!id) {
96
+ if (!id && !isUnderTestRunner()) {
86
97
  throw new Error(
87
- "[rsc-router] Static: missing $$id. " +
88
- "Ensure the exposeInternalIds Vite plugin is configured.",
98
+ "[rango] Static: missing $$id. Use `export const X = Static(...)` and " +
99
+ "ensure the exposeInternalIds Vite plugin is configured.",
89
100
  );
90
101
  }
102
+ if (!id) {
103
+ id = `__rango_runtime_static_${runtimeStaticIdCounter++}`;
104
+ }
91
105
 
92
106
  return {
93
107
  __brand: "staticHandler" as const,
@@ -97,11 +111,6 @@ export function Static<TParams extends Record<string, any>>(
97
111
  };
98
112
  }
99
113
 
100
- // -- Type guard -------------------------------------------------------------
101
-
102
- /**
103
- * Type guard to check if a value is a StaticHandlerDefinition.
104
- */
105
114
  export function isStaticHandler(
106
115
  value: unknown,
107
116
  ): value is StaticHandlerDefinition {
@@ -0,0 +1,162 @@
1
+ /**
2
+ * Cache-status testing primitives for @rangojs/router consumers.
3
+ *
4
+ * Two complementary paths, both DEVELOPMENT/TEST ONLY:
5
+ *
6
+ * 1. Header path — `parseCacheHeader` / `assertCacheStatus` read the
7
+ * `X-Rango-Cache` response header. The header is emitted only when the
8
+ * router's debug cache signal gate is on (the `debugCacheSignal` option or
9
+ * `RANGO_TEST_SIGNALS=1`). With the gate off there is no header and these
10
+ * helpers throw a clear "header missing" error.
11
+ *
12
+ * 2. Telemetry path — `createCacheSink` returns a `{ sink, events }` pair the
13
+ * consumer wires via `createRouter({ telemetry: sink })`. This has ZERO
14
+ * production surface: no header, just structured `cache.decision` events
15
+ * (which carry the same coarse `segments` cache signal). Assert with
16
+ * `assertCacheDecision(events, routeKey, expected)` (the one-call counterpart
17
+ * of `assertCacheStatus`) or filter raw via `filterCacheDecisions`.
18
+ *
19
+ * Both paths report the SAME coarse route-level signal — pick by TRANSPORT, not
20
+ * by meaning: the header is the only signal a black-box Playwright `Response`
21
+ * carries (needs the debug gate ON); the sink is the only zero-production-surface
22
+ * option and the only one exposing per-segment `shouldRevalidate`.
23
+ *
24
+ * v1 cache status is COARSE (route-level): the router reports a single entry
25
+ * keyed by the route key (the route NAME), not per individual segment.
26
+ *
27
+ * Import path: from a Vitest unit/integration test use `@rangojs/router/testing`;
28
+ * from a Playwright e2e use `@rangojs/router/testing/e2e` (the barrel pulls a
29
+ * build-only virtual that does not resolve in a plain Playwright runner).
30
+ */
31
+
32
+ import type {
33
+ CacheDecisionEvent,
34
+ CacheSegmentStatus,
35
+ TelemetryEvent,
36
+ TelemetrySink,
37
+ } from "../router/telemetry.js";
38
+
39
+ const CACHE_HEADER = "X-Rango-Cache";
40
+
41
+ /** Expected cache status passed to assertCacheStatus. */
42
+ export type ExpectedCacheStatus = CacheSegmentStatus;
43
+
44
+ /** A target carrying response headers (a Response or a `{ headers }` object). */
45
+ export type CacheStatusTarget = Response | { headers: Headers };
46
+
47
+ export function parseCacheHeader(
48
+ headerValue: string | null | undefined,
49
+ ): Record<string, string> {
50
+ const result: Record<string, string> = {};
51
+ if (!headerValue) return result;
52
+ for (const rawEntry of headerValue.split(",")) {
53
+ const entry = rawEntry.trim();
54
+ if (entry.length === 0) continue;
55
+ const eq = entry.indexOf("=");
56
+ if (eq === -1) continue;
57
+ const id = entry.slice(0, eq).trim();
58
+ const status = entry.slice(eq + 1).trim();
59
+ if (id.length === 0 || status.length === 0) continue;
60
+ result[id] = status;
61
+ }
62
+ return result;
63
+ }
64
+
65
+ function getHeaders(target: CacheStatusTarget): Headers {
66
+ return target.headers;
67
+ }
68
+
69
+ export function assertCacheStatus(
70
+ target: CacheStatusTarget,
71
+ segment: string,
72
+ expected: ExpectedCacheStatus,
73
+ ): void {
74
+ const headerValue = getHeaders(target).get(CACHE_HEADER);
75
+ if (headerValue === null) {
76
+ throw new Error(
77
+ `assertCacheStatus: response has no ${CACHE_HEADER} header. ` +
78
+ `Enable the debug cache signal via createRouter({ debugCacheSignal: true }) ` +
79
+ `or RANGO_TEST_SIGNALS=1.`,
80
+ );
81
+ }
82
+ const map = parseCacheHeader(headerValue);
83
+ const actual = map[segment];
84
+ if (actual === undefined) {
85
+ const known = Object.keys(map);
86
+ throw new Error(
87
+ `assertCacheStatus: segment "${segment}" not found in ${CACHE_HEADER} ` +
88
+ `("${headerValue}"). Known segments: ${
89
+ known.length > 0 ? known.join(", ") : "(none)"
90
+ }.`,
91
+ );
92
+ }
93
+ if (actual !== expected) {
94
+ throw new Error(
95
+ `assertCacheStatus: segment "${segment}" expected "${expected}" but got "${actual}".`,
96
+ );
97
+ }
98
+ }
99
+
100
+ /**
101
+ * A telemetry sink paired with the array it records events into.
102
+ */
103
+ export interface CacheSink {
104
+ /** Wire into `createRouter({ telemetry: sink })`. */
105
+ sink: TelemetrySink;
106
+ /** All telemetry events captured so far, in emit order. */
107
+ events: TelemetryEvent[];
108
+ }
109
+
110
+ export function createCacheSink(): CacheSink {
111
+ const events: TelemetryEvent[] = [];
112
+ const sink: TelemetrySink = {
113
+ emit(event: TelemetryEvent): void {
114
+ events.push(event);
115
+ },
116
+ };
117
+ return { sink, events };
118
+ }
119
+
120
+ export function filterCacheDecisions(
121
+ events: readonly TelemetryEvent[],
122
+ ): CacheDecisionEvent[] {
123
+ return events.filter(
124
+ (e): e is CacheDecisionEvent => e.type === "cache.decision",
125
+ );
126
+ }
127
+
128
+ /**
129
+ * Telemetry-path counterpart of {@link assertCacheStatus}: assert a captured
130
+ * `cache.decision` event reported `expected` for the segment keyed by `routeKey`
131
+ * (the route NAME, the same coarse key the header path uses). Throws an
132
+ * actionable error when no matching segment was captured, or on a mismatch.
133
+ *
134
+ * Pairs with {@link createCacheSink}: wire `createRouter({ telemetry: sink })`,
135
+ * drive an RSC request, then assert against the recorded `events`. This is the
136
+ * zero-production-surface path (no header to enable). NOTE: `events` accumulates
137
+ * across requests, so the FIRST matching segment wins — slice or recreate the
138
+ * sink between requests for the same `routeKey`.
139
+ */
140
+ export function assertCacheDecision(
141
+ events: readonly TelemetryEvent[],
142
+ routeKey: string,
143
+ expected: ExpectedCacheStatus,
144
+ ): void {
145
+ const segments = filterCacheDecisions(events).flatMap(
146
+ (d) => d.segments ?? [],
147
+ );
148
+ const seg = segments.find((s) => s.id === routeKey);
149
+ if (seg === undefined) {
150
+ const known = segments.map((s) => s.id);
151
+ throw new Error(
152
+ `assertCacheDecision: no cache.decision segment for routeKey "${routeKey}". ` +
153
+ `Seen: ${known.length > 0 ? known.join(", ") : "(none)"}. Wire ` +
154
+ `createRouter({ telemetry: createCacheSink().sink }) and drive an RSC request.`,
155
+ );
156
+ }
157
+ if (seg.cacheStatus !== expected) {
158
+ throw new Error(
159
+ `assertCacheDecision: routeKey "${routeKey}" expected "${expected}" but got "${seg.cacheStatus}".`,
160
+ );
161
+ }
162
+ }
@@ -0,0 +1,40 @@
1
+ /**
2
+ * collectHandle — unit-test a handle's `collect`/accumulator function directly.
3
+ *
4
+ * A handle's collect function (the `createHandle(collect)` argument that maps the
5
+ * per-segment pushed values into the accumulated result) is otherwise not
6
+ * directly reachable: createHandle keeps it in a private registry keyed by the
7
+ * handle's `$$id` and returns only `{ __brand, $$id }`. This primitive runs that
8
+ * REAL registered collect on per-segment values you provide and returns the
9
+ * accumulated result — so the mapper/accumulator is unit-testable without a full
10
+ * route match.
11
+ *
12
+ * It relies on createHandle registering the collect even in a bare test (it
13
+ * assigns a runtime fallback id when the Vite plugin did not inject one). If a
14
+ * handle's module was never imported (so createHandle never ran), the collect is
15
+ * unregistered and this falls back to a flat array with a warning.
16
+ */
17
+
18
+ import { getCollectFn, type Handle } from "../handle.js";
19
+
20
+ export function collectHandle<TData, TAccumulated>(
21
+ handle: Handle<TData, TAccumulated>,
22
+ segments: ReadonlyArray<ReadonlyArray<TData>>,
23
+ ): TAccumulated {
24
+ const collectFn = getCollectFn(handle.$$id) as
25
+ | ((segments: TData[][]) => TAccumulated)
26
+ | undefined;
27
+
28
+ if (!collectFn) {
29
+ console.warn(
30
+ `[rango] collectHandle: handle "${handle.$$id}" has no registered collect ` +
31
+ `function. Import the handle's module so createHandle() runs. Falling ` +
32
+ `back to a flat array.`,
33
+ );
34
+ return segments.flat() as unknown as TAccumulated;
35
+ }
36
+
37
+ // Drop empty arrays matching production behavior (segment count/indices).
38
+ const nonEmpty = segments.filter((seg) => seg.length > 0) as TData[][];
39
+ return collectFn(nonEmpty);
40
+ }