@rangojs/router 0.0.0-experimental.32 → 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 +120 -204
  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 +190 -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 +63 -24
  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 +338 -126
  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
@@ -0,0 +1,348 @@
1
+ /**
2
+ * Shared internals for the consumer testing primitives.
3
+ *
4
+ * Builds a real RequestContext via the same createRequestContext the RSC
5
+ * handler uses, with test-friendly defaults, so loaders and middleware run
6
+ * with production-fidelity context (cookies, headers, get/set, use, reverse)
7
+ * instead of a hand-rolled mock.
8
+ */
9
+
10
+ import {
11
+ createRequestContext,
12
+ runWithRequestContext,
13
+ type RequestContext,
14
+ } from "../../server/request-context.js";
15
+ import { resolveLocationStateEntries } from "../../browser/react/location-state-shared.js";
16
+ import { createReverseFunction } from "../../router/handler-context.js";
17
+ import { normalizeBasename } from "../../router/basename.js";
18
+ import {
19
+ seedVariables,
20
+ resolveSeededStateCookieName,
21
+ type VarsInit,
22
+ type StateCookieSeed,
23
+ } from "./seed-vars.js";
24
+ import type { ThemeConfig } from "../../theme/types.js";
25
+ import { resolveThemeConfig } from "../../theme/constants.js";
26
+ import type { SegmentCacheStore } from "../../cache/types.js";
27
+ import type { CacheProfile } from "../../cache/profile-registry.js";
28
+
29
+ const DEFAULT_ORIGIN = "http://localhost/";
30
+
31
+ export type { VarsInit, StateCookieSeed };
32
+ export { seedVariables };
33
+
34
+ /** Normalize a Request | string | undefined into a concrete Request. */
35
+ export function toRequest(
36
+ request: Request | string | undefined,
37
+ init?: RequestInit,
38
+ ): Request {
39
+ if (request instanceof Request) return request;
40
+ return typeof request === "string"
41
+ ? new Request(new URL(request, DEFAULT_ORIGIN), init)
42
+ : new Request(DEFAULT_ORIGIN, init);
43
+ }
44
+
45
+ export interface CreateTestContextOptions<TEnv> {
46
+ env?: TEnv;
47
+ request?: Request | string;
48
+ requestInit?: RequestInit;
49
+ /** Backing store for ctx.get()/ctx.set(); pre-seeded from `vars`. */
50
+ variables?: Record<string, unknown>;
51
+ /** Variables a prior middleware would have set (object or [key, value] list). */
52
+ vars?: VarsInit;
53
+ /** Route name -> pattern map enabling ctx.reverse() without global state. */
54
+ routeMap?: Record<string, string>;
55
+ routeName?: string;
56
+ params?: Record<string, string>;
57
+ /**
58
+ * Router basename for this request (what the RSC handler stores on the
59
+ * context). Drives redirect() prefixing. Normalized exactly like
60
+ * createRouter({ basename }) (leading slash forced, trailing stripped, bare
61
+ * "/" -> undefined) so passing the same value your router takes yields the
62
+ * same redirect Location. Defaults to undefined (no basename).
63
+ */
64
+ basename?: string;
65
+ /**
66
+ * Cache store backing `use cache` functions invoked during the test, the
67
+ * same shape `createRouter({ cache })` resolves. Without it,
68
+ * registerCachedFunction bypasses (it checks for a store FIRST), so a cached
69
+ * function runs uncached and its taint/profile guards never fire. Wire one
70
+ * (e.g. `new MemorySegmentCacheStore()`) to exercise real cache behavior.
71
+ */
72
+ cacheStore?: SegmentCacheStore;
73
+ /**
74
+ * Cache profiles in the `createRouter({ cacheProfiles })` shape. Required for
75
+ * a `use cache: "profileName"` function to resolve its profile (an unknown
76
+ * profile throws), once a `cacheStore` is wired.
77
+ */
78
+ cacheProfiles?: Record<string, CacheProfile>;
79
+ /**
80
+ * Theme config in the same shape `createRouter({ theme })` takes (resolved
81
+ * internally). Without it `ctx.theme`/`ctx.setTheme` are inert (undefined),
82
+ * mirroring an app with no theme configured. Pass one (e.g. `true`, or
83
+ * `{ themes: [...] }`) to exercise a handler that reads them.
84
+ */
85
+ theme?: ThemeConfig | true;
86
+ /**
87
+ * Customize the rango state cookie that `invalidateClientCache()` rotates.
88
+ * The name is ALWAYS seeded (default `rango-state_router_0`) so a call to
89
+ * `invalidateClientCache()` rotates and emits the `Set-Cookie` exactly as in
90
+ * production, rather than silently no-opping. Override `prefix`/`routerId` to
91
+ * match your `createRouter({ stateCookiePrefix, id })` so the test asserts the
92
+ * same name, or `version` (the build identifier prefixed to the rotated
93
+ * `{version}:{timestamp}` value, default `"0"`). Assert
94
+ * `response.headers.getSetCookie()` against the resolved `stateCookieName`
95
+ * (returned by `runInRequestContext`).
96
+ */
97
+ stateCookie?: StateCookieSeed;
98
+ }
99
+
100
+ /**
101
+ * The seeded RequestContext with its `reverse` RELAXED to accept any route NAME
102
+ * from the `routeMap` you passed, rather than the global `Rango.GeneratedRouteMap`
103
+ * union — so reversing a test-only route name is not a type error (it works at
104
+ * runtime; the names come from your `routeMap`). Mirrors runLoader's
105
+ * `TestLoaderContext.reverse`. Everything else is the real `RequestContext`.
106
+ */
107
+ export type TestRequestContextObject<TEnv> = Omit<
108
+ RequestContext<TEnv>,
109
+ "reverse"
110
+ > & {
111
+ reverse: (
112
+ name: string,
113
+ params?: Record<string, string>,
114
+ search?: Record<string, unknown>,
115
+ ) => string;
116
+ };
117
+
118
+ export interface TestRequestContext<TEnv> {
119
+ ctx: TestRequestContextObject<TEnv>;
120
+ request: Request;
121
+ url: URL;
122
+ variables: Record<string, unknown>;
123
+ /**
124
+ * The resolved rango state cookie name seeded into the context (default
125
+ * `rango-state_router_0`, or composed from `opts.stateCookie`). The name a
126
+ * call to `invalidateClientCache()` rotates.
127
+ */
128
+ stateCookieName: string;
129
+ }
130
+
131
+ /**
132
+ * Create a real RequestContext for unit-testing loaders/middleware.
133
+ *
134
+ * The returned `ctx` must be ENTERED before use — wrap your call in
135
+ * `runWithRequestContext(ctx, fn)` (re-exported from `@rangojs/router/testing`)
136
+ * so that cookie/header mutations and `getRequestContext()` resolve. For the
137
+ * common case prefer {@link runInRequestContext}, which builds AND enters the
138
+ * context in a single call.
139
+ */
140
+ export function createTestRequestContext<TEnv>(
141
+ opts: CreateTestContextOptions<TEnv> = {},
142
+ ): TestRequestContext<TEnv> {
143
+ const request = toRequest(opts.request, opts.requestInit);
144
+ const url = new URL(request.url);
145
+ const variables = seedVariables(opts.variables ?? {}, opts.vars);
146
+ const stateCookieName = resolveSeededStateCookieName(opts.stateCookie);
147
+ const ctx = createRequestContext<TEnv>({
148
+ env: (opts.env ?? {}) as TEnv,
149
+ request,
150
+ url,
151
+ variables,
152
+ themeConfig:
153
+ opts.theme === undefined ? undefined : resolveThemeConfig(opts.theme),
154
+ cacheStore: opts.cacheStore,
155
+ cacheProfiles: opts.cacheProfiles,
156
+ stateCookieName,
157
+ version: opts.stateCookie?.version,
158
+ });
159
+ if (opts.basename !== undefined)
160
+ ctx._basename = normalizeBasename(opts.basename);
161
+ if (opts.params) ctx.params = opts.params;
162
+ if (opts.routeMap) {
163
+ ctx._routeName = opts.routeName;
164
+ ctx.reverse = createReverseFunction(
165
+ opts.routeMap,
166
+ opts.routeName,
167
+ opts.params ?? {},
168
+ ) as RequestContext<TEnv>["reverse"];
169
+ }
170
+ return {
171
+ ctx: ctx as unknown as TestRequestContextObject<TEnv>,
172
+ request,
173
+ url,
174
+ variables,
175
+ stateCookieName,
176
+ };
177
+ }
178
+
179
+ /**
180
+ * What a run accumulated on the request context, surfaced as PUBLIC values so a
181
+ * test never has to cast through the `@internal` `ctx.res` / `ctx.cookies()` to
182
+ * assert what an action produced.
183
+ */
184
+ export interface RunInRequestContextResult<T> {
185
+ /**
186
+ * The value `fn` returned (awaited), or `undefined` if `fn` threw — in which
187
+ * case the thrown value is on {@link thrown}. The snapshot below is captured
188
+ * either way.
189
+ */
190
+ result: T | undefined;
191
+ /**
192
+ * The value `fn` threw, or `undefined` if it returned normally. Commonly a
193
+ * `Response` from `throw redirect(...)` / `throw notFound()` — the dominant
194
+ * cookie+flash case is an action that sets them then throws a redirect — so
195
+ * this (and the snapshot below) is observable WITHOUT wrapping the action in
196
+ * your own try/catch. NOTE: the value is captured, NOT re-thrown; assert on it
197
+ * for a throwing action.
198
+ */
199
+ thrown: unknown;
200
+ /**
201
+ * A Response carrying the status, headers, and Set-Cookie the run set (via
202
+ * `cookies().set()`, `ctx.header()`, etc.). Assert Set-Cookie with
203
+ * `response.headers.getSetCookie()`. When `fn` threw a `Response` (a redirect),
204
+ * THIS is that Response with the accumulated Set-Cookie/headers merged in
205
+ * (mirroring how the framework merges them in production), so a redirect's
206
+ * Location AND the cookies it set are both observable here.
207
+ */
208
+ response: Response;
209
+ /**
210
+ * The effective cookie view after the run: request cookies merged with
211
+ * anything the run set or deleted (last-write-wins), as `{ name: value }`.
212
+ */
213
+ cookies: Record<string, string>;
214
+ /**
215
+ * The response headers the run set (via `ctx.header(...)`, plus a thrown
216
+ * redirect's `Location`), as a plain `{ name: value }` object — the same view
217
+ * as `response.headers`, but assertable like `cookies`/`locationState`.
218
+ * EXCLUDES `set-cookie` (use `cookies`, or `response.headers.getSetCookie()`).
219
+ * Header names are lowercased (HTTP headers are case-insensitive).
220
+ */
221
+ headers: Record<string, string>;
222
+ /**
223
+ * Location state the run set via `ctx.setLocationState()` / `redirect({ state })`,
224
+ * resolved to the flat `{ key: value }` shape the client reads off
225
+ * `history.state` (empty object when none) — so a post-action flash ("Saved!")
226
+ * is assertable at the unit layer.
227
+ */
228
+ locationState: Record<string, unknown>;
229
+ /**
230
+ * The resolved rango state cookie name seeded into the run (default
231
+ * `rango-state_router_0`, or composed from `opts.stateCookie`). Assert an
232
+ * action's `invalidateClientCache()` rotation against it without recomputing:
233
+ * `response.headers.getSetCookie().some((c) => c.startsWith(stateCookieName + "="))`.
234
+ */
235
+ stateCookieName: string;
236
+ }
237
+
238
+ export function snapshotRunEffects<TEnv>(ctx: RequestContext<TEnv>): {
239
+ cookies: Record<string, string>;
240
+ locationState: Record<string, unknown>;
241
+ } {
242
+ return {
243
+ cookies: { ...ctx.cookies() },
244
+ locationState: resolveLocationStateEntries(ctx._locationState ?? []),
245
+ };
246
+ }
247
+
248
+ export function headersToObject(headers: Headers): Record<string, string> {
249
+ const out: Record<string, string> = {};
250
+ headers.forEach((value, name) => {
251
+ if (name.toLowerCase() === "set-cookie") return;
252
+ out[name] = value;
253
+ });
254
+ return out;
255
+ }
256
+
257
+ export function buildRunResponse<TEnv>(
258
+ ctx: RequestContext<TEnv>,
259
+ thrown: unknown,
260
+ ): Response {
261
+ const stub = ctx.res;
262
+ if (thrown instanceof Response) {
263
+ const headers = new Headers(thrown.headers);
264
+ for (const cookie of stub.headers.getSetCookie()) {
265
+ headers.append("set-cookie", cookie);
266
+ }
267
+ stub.headers.forEach((value, name) => {
268
+ if (name.toLowerCase() === "set-cookie") return;
269
+ if (!headers.has(name)) headers.set(name, value);
270
+ });
271
+ return new Response(null, { status: thrown.status, headers });
272
+ }
273
+ return new Response(null, { status: stub.status, headers: stub.headers });
274
+ }
275
+
276
+ export function buildRunSnapshot<TEnv>(
277
+ ctx: RequestContext<TEnv>,
278
+ thrown: unknown,
279
+ stateCookieName: string,
280
+ ): {
281
+ thrown: unknown;
282
+ response: Response;
283
+ cookies: Record<string, string>;
284
+ headers: Record<string, string>;
285
+ locationState: Record<string, unknown>;
286
+ stateCookieName: string;
287
+ } {
288
+ const { cookies, locationState } = snapshotRunEffects(ctx);
289
+ const response = buildRunResponse(ctx, thrown);
290
+ const headers = headersToObject(response.headers);
291
+ return { thrown, response, cookies, headers, locationState, stateCookieName };
292
+ }
293
+
294
+ /**
295
+ * Build a seeded RequestContext (via {@link createTestRequestContext}) and run
296
+ * `fn` inside it, so code under test that calls `getRequestContext()`,
297
+ * `cookies()`, or reads/mutates request headers resolves exactly as in
298
+ * production.
299
+ *
300
+ * This is the entry point for the advanced cases the unit wrappers
301
+ * (`runLoader` / `runMiddleware`) do not model — most notably a server ACTION
302
+ * that authenticates off the request cookie or sets a session cookie / flash:
303
+ * an action has no loader context, so `runLoader` is the wrong shape, yet it
304
+ * still needs a real request context to read the cookie and resolve
305
+ * `getRequestContext()`.
306
+ *
307
+ * Returns `{ result, thrown, response, cookies, headers, locationState }` so the
308
+ * action's OUTPUT (Set-Cookie, response headers, flash) is assertable without
309
+ * casting through the `@internal` `ctx.res` / `ctx.cookies()`. `fn` may be async — the context stays
310
+ * active across its awaits (AsyncLocalStorage), and the snapshot is captured
311
+ * whether `fn` returns OR throws. The throw path matters: the most common
312
+ * cookie+flash case is an auth action that sets a cookie + flash then
313
+ * `throw redirect(...)` on success — the thrown redirect is on `thrown` (NOT
314
+ * re-thrown) and its Location plus the cookies are on `response`/`cookies`.
315
+ *
316
+ * @example
317
+ * ```ts
318
+ * const { result, cookies, response, thrown } = await runInRequestContext(
319
+ * () => loginAction(input), // sets a session cookie, then `throw redirect("/app")`
320
+ * {
321
+ * env,
322
+ * request: new Request("https://app.test/", {
323
+ * headers: { Cookie: "sid=abc" },
324
+ * }),
325
+ * },
326
+ * );
327
+ * expect(cookies.session).toBe("new-token");
328
+ * expect(headers.location).toBe("/app"); // response headers as a plain object
329
+ * expect((thrown as Response).headers.get("Location")).toBe("/app");
330
+ * expect(response.headers.getSetCookie()).toContainEqual(
331
+ * expect.stringContaining("session="),
332
+ * );
333
+ * ```
334
+ */
335
+ export async function runInRequestContext<T, TEnv = unknown>(
336
+ fn: (ctx: RequestContext<TEnv>) => T | Promise<T>,
337
+ opts: CreateTestContextOptions<TEnv> = {},
338
+ ): Promise<RunInRequestContextResult<T>> {
339
+ const { ctx, stateCookieName } = createTestRequestContext<TEnv>(opts);
340
+ let result: T | undefined;
341
+ let thrown: unknown;
342
+ try {
343
+ result = (await runWithRequestContext(ctx, () => fn(ctx))) as T;
344
+ } catch (error) {
345
+ thrown = error;
346
+ }
347
+ return { result, ...buildRunSnapshot(ctx, thrown, stateCookieName) };
348
+ }
@@ -0,0 +1,30 @@
1
+ /**
2
+ * Side-effect module: define the webpack-style globals the vendored
3
+ * react-server-dom CLIENT deserializer reads at module-eval time.
4
+ *
5
+ * In a real app the plugin-rsc Vite plugin rewrites `__webpack_require__` ->
6
+ * `__vite_rsc_require__` and `__webpack_require__.u` -> `({}).u`
7
+ * (@vitejs/plugin-rsc `core/plugin.js`). That transform does NOT run in a bare
8
+ * Vitest process, so the vendored client's free `__webpack_require__` /
9
+ * `__webpack_chunk_load__` references would be undefined. We provide minimal
10
+ * shims: `__webpack_require__` routes to the loader installed via
11
+ * `setRequireModule`, and `__webpack_chunk_load__` is a no-op (renderServerTree
12
+ * serializes with empty `chunks`, so no chunk fetch ever happens).
13
+ *
14
+ * MUST be imported (for side effect) BEFORE `@vitejs/plugin-rsc/react/browser`,
15
+ * which is why flight-tree.ts lists it first.
16
+ */
17
+ const g = globalThis as unknown as {
18
+ __webpack_require__?: ((id: string) => unknown) & { u?: unknown };
19
+ __webpack_chunk_load__?: (chunkId: string) => Promise<unknown>;
20
+ __vite_rsc_client_require__?: (id: string) => unknown;
21
+ };
22
+
23
+ if (!g.__webpack_require__) {
24
+ g.__webpack_require__ = (id: string) => g.__vite_rsc_client_require__!(id);
25
+ }
26
+ if (!g.__webpack_chunk_load__) {
27
+ g.__webpack_chunk_load__ = async () => {};
28
+ }
29
+
30
+ export {};
@@ -0,0 +1,54 @@
1
+ /**
2
+ * Variable seeding shared by the node/DOM testing tier (internal/context.ts)
3
+ * AND the react-server Flight tier (flight.ts). Depends only on the
4
+ * dependency-free `context-var` module and the env-agnostic state-cookie-name
5
+ * composition (no window/document), so it is safe to import under the
6
+ * `react-server` condition (unlike internal/context.ts, which pulls
7
+ * client/browser modules).
8
+ */
9
+ import { contextSet, type ContextVar } from "../../context-var.js";
10
+ import { resolveStateCookieName } from "../../router/state-cookie-name.js";
11
+
12
+ export interface StateCookieSeed {
13
+ /**
14
+ * Cookie-name prefix, sanitized then composed with `routerId` exactly like
15
+ * `createRouter({ stateCookiePrefix })`. Defaults to `"rango-state"`.
16
+ */
17
+ prefix?: string;
18
+ /**
19
+ * Router id; the resolved name is `{sanitizedPrefix}_{sanitizedRouterId}`.
20
+ * Defaults to `"router_0"` (the name a single default router resolves to), so
21
+ * the default name is `rango-state_router_0`.
22
+ */
23
+ routerId?: string;
24
+ /**
25
+ * Build version used as the rotated value's prefix (`{version}:{timestamp}`).
26
+ * Defaults to `"0"` (resolved inside createRequestContext).
27
+ */
28
+ version?: string;
29
+ }
30
+
31
+ export function resolveSeededStateCookieName(seed?: StateCookieSeed): string {
32
+ return resolveStateCookieName(seed?.prefix, seed?.routerId ?? "router_0");
33
+ }
34
+
35
+ export type VarsInit =
36
+ | Record<string, unknown>
37
+ | ReadonlyArray<readonly [ContextVar<unknown> | string, unknown]>;
38
+
39
+ export function seedVariables(
40
+ variables: Record<string, unknown>,
41
+ vars?: VarsInit,
42
+ ): Record<string, unknown> {
43
+ if (!vars) return variables;
44
+ const entries: Iterable<readonly [ContextVar<unknown> | string, unknown]> =
45
+ Symbol.iterator in (vars as object)
46
+ ? (vars as ReadonlyArray<
47
+ readonly [ContextVar<unknown> | string, unknown]
48
+ >)
49
+ : Object.entries(vars as Record<string, unknown>);
50
+ for (const [key, value] of entries) {
51
+ contextSet(variables, key as ContextVar<unknown>, value);
52
+ }
53
+ return variables;
54
+ }