@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
@@ -8,16 +8,17 @@
8
8
  */
9
9
 
10
10
  import { createElement } from "react";
11
- import { RouteNotFoundError } from "../errors.js";
11
+ import { isRouteNotFoundError } from "../errors.js";
12
12
  import { matchMiddleware, executeMiddleware } from "../router/middleware.js";
13
13
  import {
14
14
  runWithRequestContext,
15
15
  setRequestContextParams,
16
16
  requireRequestContext,
17
+ getRequestContext,
18
+ _getRequestContext,
17
19
  createRequestContext,
18
20
  } from "../server/request-context.js";
19
21
  import * as rscDeps from "@vitejs/plugin-rsc/rsc";
20
-
21
22
  import type {
22
23
  RscPayload,
23
24
  CreateRSCHandlerOptions,
@@ -30,6 +31,8 @@ import {
30
31
  interceptRedirectForPartial,
31
32
  buildRouteMiddlewareEntries,
32
33
  } from "./helpers.js";
34
+ import { guardOutgoingRedirect } from "./redirect-guard.js";
35
+ import { isWebSocketUpgradeResponse } from "../response-utils.js";
33
36
  import {
34
37
  handleResponseRoute,
35
38
  type ResponseRouteMatch,
@@ -55,6 +58,8 @@ import {
55
58
  getRouterTrie,
56
59
  } from "../route-map-builder.js";
57
60
  import type { HandlerContext } from "./handler-context.js";
61
+ import type { CacheErrorCategory } from "../cache/cache-error.js";
62
+ import type { SegmentCacheStore } from "../cache/types.js";
58
63
  import { buildRouterTrieFromUrlpatterns } from "./manifest-init.js";
59
64
  import { handleProgressiveEnhancement } from "./progressive-enhancement.js";
60
65
  import {
@@ -63,7 +68,10 @@ import {
63
68
  type ActionContinuation,
64
69
  } from "./server-action.js";
65
70
  import { handleLoaderFetch } from "./loader-fetch.js";
66
- import { checkRequestOrigin, type OriginCheckPhase } from "./origin-guard.js";
71
+ import {
72
+ checkRequestOrigin,
73
+ ORIGIN_CHECK_PHASE_BY_MODE,
74
+ } from "./origin-guard.js";
67
75
  import { handleRscRendering } from "./rsc-rendering.js";
68
76
  import {
69
77
  withTimeout,
@@ -80,8 +88,14 @@ import {
80
88
  startSSRSetup,
81
89
  getSSRSetup,
82
90
  mayNeedSSR,
91
+ isRscRequest,
83
92
  SSR_SETUP_VAR,
84
93
  } from "./ssr-setup.js";
94
+ import {
95
+ classifyRequest,
96
+ type RequestPlan,
97
+ type ExecutableRequestPlan,
98
+ } from "../router/request-classification.js";
85
99
 
86
100
  /**
87
101
  * Create an RSC request handler.
@@ -116,12 +130,35 @@ import {
116
130
  * });
117
131
  * ```
118
132
  */
133
+
134
+ /**
135
+ * Response that tells the client to do a full document navigation. Shared by
136
+ * the terminal reload plans (version-mismatch and app-switch): an empty 200
137
+ * carrying X-RSC-Reload, which the client turns into window.location.href.
138
+ */
139
+ function createReloadResponse(reloadUrl: string) {
140
+ return createResponseWithMergedHeaders(null, {
141
+ status: 200,
142
+ headers: {
143
+ "X-RSC-Reload": reloadUrl,
144
+ "content-type": "text/x-component;charset=utf-8",
145
+ },
146
+ });
147
+ }
148
+
119
149
  export function createRSCHandler<
120
150
  TEnv = unknown,
121
151
  TRoutes extends Record<string, string> = Record<string, string>,
122
152
  >(options: CreateRSCHandlerOptions<TEnv, TRoutes>) {
123
153
  const { router, version = VERSION, nonce: nonceProvider } = options;
124
154
 
155
+ // Handler-owned registry of explicit per-scope stores from cache({ store }).
156
+ // Lives in the closure so it is scoped per handler (multi-router deployments
157
+ // get separate registries) and accumulates every explicit store this handler
158
+ // resolves across requests. updateTag()/revalidateTag() iterate it to reach
159
+ // stores not covered by the app-level ctx._cacheStore.
160
+ const explicitTaggedStores = new Set<SegmentCacheStore>();
161
+
125
162
  // Use provided deps or default to @vitejs/plugin-rsc/rsc exports
126
163
  const deps = options.deps ?? rscDeps;
127
164
  const {
@@ -161,10 +198,13 @@ export function createRSCHandler<
161
198
  phase: ErrorPhase,
162
199
  context: Parameters<typeof invokeOnError<TEnv>>[3],
163
200
  ): void {
164
- if (error != null && typeof error === "object") {
165
- const reportedErrors = requireRequestContext()._reportedErrors;
166
- if (reportedErrors.has(error)) return;
167
- reportedErrors.add(error);
201
+ // Guard: abort signal handlers fire asynchronously outside the ALS
202
+ // request scope, so the context may be gone. Skip dedup in that
203
+ // case — the error is from a cancelled stream, not a real failure.
204
+ const reqCtx = _getRequestContext();
205
+ if (error != null && typeof error === "object" && reqCtx) {
206
+ if (reqCtx._reportedErrors.has(error)) return;
207
+ reqCtx._reportedErrors.add(error);
168
208
  }
169
209
  invokeOnError(router.onError, error, phase, context, "RSC");
170
210
  }
@@ -253,12 +293,13 @@ export function createRSCHandler<
253
293
  function createRedirectFlightResponse(
254
294
  redirectUrl: string,
255
295
  locationState?: Record<string, unknown>,
296
+ external?: boolean,
256
297
  ): Response {
257
298
  const redirectPayload: RscPayload = {
258
299
  metadata: {
259
300
  pathname: redirectUrl,
260
301
  segments: [],
261
- redirect: { url: redirectUrl },
302
+ redirect: { url: redirectUrl, ...(external && { external: true }) },
262
303
  ...(locationState && { locationState }),
263
304
  },
264
305
  };
@@ -343,7 +384,7 @@ export function createRSCHandler<
343
384
  // Resolve cache store configuration
344
385
  // Priority: options.cache (handler override) > router.cache (router default)
345
386
  // Store is enabled only if: config provided, enabled, and no ?__no_cache query param
346
- let cacheStore = undefined;
387
+ let cacheStore: SegmentCacheStore | undefined;
347
388
  const cacheOption = options.cache ?? router.cache;
348
389
  if (cacheOption && !url.searchParams.has("__no_cache")) {
349
390
  const cacheConfig =
@@ -410,9 +451,12 @@ export function createRSCHandler<
410
451
  url,
411
452
  variables,
412
453
  cacheStore,
454
+ explicitTaggedStores,
413
455
  cacheProfiles: router.cacheProfiles,
414
456
  executionContext: executionCtx,
415
457
  themeConfig: router.themeConfig,
458
+ stateCookieName: router.resolvedStateCookieName,
459
+ version,
416
460
  });
417
461
  if (earlyMetricsStore) {
418
462
  requestContext._debugPerformance = true;
@@ -422,7 +466,7 @@ export function createRSCHandler<
422
466
  // can surface non-fatal errors through the router's onError callback.
423
467
  requestContext._reportBackgroundError = (
424
468
  error: unknown,
425
- category: string,
469
+ category: CacheErrorCategory,
426
470
  ) => {
427
471
  callOnError(error, "cache", {
428
472
  request,
@@ -452,6 +496,9 @@ export function createRSCHandler<
452
496
  // - Server components during rendering
453
497
  // - Error boundaries
454
498
  // - Streaming
499
+ // Store basename on request context (scoped per-request via existing ALS)
500
+ requestContext._basename = router.basename;
501
+
455
502
  return runWithRequestContext(requestContext, async () => {
456
503
  // Core handler logic (wrapped by middleware)
457
504
  const coreHandler = async (): Promise<Response> => {
@@ -521,13 +568,22 @@ export function createRSCHandler<
521
568
  }
522
569
 
523
570
  const fullTiming = timingParts.join(", ");
524
- if (fullTiming) response.headers.set("Server-Timing", fullTiming);
571
+ if (fullTiming && !isWebSocketUpgradeResponse(response)) {
572
+ response.headers.set("Server-Timing", fullTiming);
573
+ }
525
574
 
526
- return response;
575
+ // Single open-redirect chokepoint: every response (PE, full-page,
576
+ // middleware short-circuit, response-route) funnels through here, so
577
+ // guarding browser-followed (3xx) redirects once covers them all and any
578
+ // future redirect exit. Soft SPA/Flight redirects are 200/204 and pass
579
+ // through untouched (validated client-side instead).
580
+ return guardOutgoingRedirect(response, url.origin, router.basename);
527
581
  });
528
582
  };
529
583
 
530
- // Core request handling logic (separated for middleware wrapping)
584
+ // Core request handling logic (separated for middleware wrapping).
585
+ // Uses the classify → execute model: classifyRequest produces a RequestPlan,
586
+ // then execution dispatches on the plan mode.
531
587
  async function coreRequestHandler(
532
588
  request: Request,
533
589
  env: TEnv,
@@ -535,71 +591,103 @@ export function createRSCHandler<
535
591
  variables: Record<string, any>,
536
592
  nonce: string | undefined,
537
593
  ): Promise<Response> {
538
- const previewStart = performance.now();
539
- const preview = await router.previewMatch(request, { env });
540
- const previewDur = performance.now() - previewStart;
541
594
  const handlerTiming: string[] = variables.__handlerTiming || [];
542
- handlerTiming.push(`handler-preview-match;dur=${previewDur.toFixed(2)}`);
543
- // Response route short-circuit: skip entire RSC pipeline
544
- if (preview?.responseType && preview.handler) {
545
- const responseOutcome = await withTimeout(
546
- handleResponseRoute(
547
- handlerCtx,
548
- preview as ResponseRouteMatch,
549
- request,
550
- env,
551
- url,
552
- variables,
595
+
596
+ // Debug manifest endpoint: handled before classification since it
597
+ // doesn't need a route match and needs trie access from the closure.
598
+ const isDev = process.env.NODE_ENV !== "production";
599
+ if (
600
+ url.searchParams.has("__debug_manifest") &&
601
+ (isDev || router.allowDebugManifest)
602
+ ) {
603
+ const trie = getRouterTrie(router.id) ?? getRouteTrie();
604
+ const routeManifest = getRequiredRouteMap();
605
+ const { extractAncestryFromTrie } =
606
+ await import("../build/route-trie.js");
607
+ return new Response(
608
+ JSON.stringify(
609
+ {
610
+ routerId: router.id,
611
+ routeManifest,
612
+ routeAncestry: trie ? extractAncestryFromTrie(trie) : {},
613
+ routeTrie: trie,
614
+ precomputedEntries: getPrecomputedEntries(),
615
+ },
616
+ null,
617
+ 2,
553
618
  ),
554
- router.timeouts.renderStartMs,
555
- "render-start",
619
+ {
620
+ headers: { "Content-Type": "application/json" },
621
+ },
556
622
  );
557
- if (responseOutcome.timedOut) {
558
- return handleTimeoutResponse(
559
- request,
560
- env,
561
- url,
562
- "render-start",
563
- responseOutcome.durationMs,
564
- preview?.routeKey,
565
- );
623
+ }
624
+
625
+ // ---- 1. Classify ----
626
+ // classifyRequest may throw RouteNotFoundError for unknown routes.
627
+ // In that case, fall through to a full-render plan so the pipeline
628
+ // can render the 404 page via the existing error handling path.
629
+ const classifyStart = performance.now();
630
+ let plan: RequestPlan<TEnv>;
631
+ try {
632
+ plan = await classifyRequest<TEnv>(request, url, {
633
+ findMatch: router.findMatch,
634
+ routerVersion: version,
635
+ routerId: router.id,
636
+ });
637
+ } catch (error) {
638
+ if (isRouteNotFoundError(error)) {
639
+ // Let the render path handle 404 — match()/matchPartial() will
640
+ // re-throw RouteNotFoundError and the catch block in
641
+ // executeRenderWithMiddleware renders the not-found page.
642
+ plan = {
643
+ mode: "full-render",
644
+ route: {
645
+ matched: null as any,
646
+ manifestEntry: null as any,
647
+ entries: [],
648
+ routeKey: "",
649
+ localRouteName: "",
650
+ params: {},
651
+ routeMiddleware: [],
652
+ cacheScope: null,
653
+ isPassthrough: false,
654
+ },
655
+ negotiated: false,
656
+ };
657
+ } else {
658
+ throw error;
566
659
  }
567
- return responseOutcome.result;
660
+ }
661
+ const classifyDur = performance.now() - classifyStart;
662
+ handlerTiming.push(`handler-classify;dur=${classifyDur.toFixed(2)}`);
663
+
664
+ // ---- 2. Terminal plans (no execution needed) ----
665
+ if (plan.mode === "redirect") {
666
+ // Redirects are handled by the pipeline (match/matchPartial),
667
+ // but for partial requests we short-circuit with a Flight redirect.
668
+ if (url.searchParams.has("_rsc_partial")) {
669
+ return createRedirectFlightResponse(plan.redirectUrl);
670
+ }
671
+ // Full requests: let the pipeline handle the redirect via match()
672
+ // which returns { redirect: url }. Fall through to full-render.
568
673
  }
569
674
 
570
- // Kick off SSR module loading + stream mode resolution in parallel with
571
- // segment resolution. Placed after the response-route short-circuit so
572
- // response/mime routes never pay for SSR work.
573
- if (mayNeedSSR(request, url)) {
574
- variables[SSR_SETUP_VAR] = startSSRSetup(
575
- handlerCtx,
576
- request,
577
- env,
578
- url,
579
- router.debugPerformance
580
- ? () => requireRequestContext()._metricsStore
581
- : undefined,
675
+ if (plan.mode === "version-mismatch") {
676
+ console.log(
677
+ `[RSC] Version mismatch: client=${url.searchParams.get("_rsc_v")}, server=${version}. Forcing reload.`,
582
678
  );
679
+ return createReloadResponse(plan.reloadUrl);
583
680
  }
584
681
 
585
- const routeReverse = createReverseFunction(getRequiredRouteMap());
682
+ if (plan.mode === "app-switch") {
683
+ // Cross-app SPA navigation crossed a host-router app boundary. Force a
684
+ // real document navigation so the target app's document is re-established
685
+ // (stylesheets, theme, warmup, prefetch-TTL). See request-classification.
686
+ return createReloadResponse(plan.reloadUrl);
687
+ }
586
688
 
587
- const isAction =
588
- request.headers.has("rsc-action") || url.searchParams.has("_rsc_action");
589
- const isLoaderFetch = url.searchParams.has("_rsc_loader");
590
- const actionId =
591
- request.headers.get("rsc-action") || url.searchParams.get("_rsc_action");
592
-
593
- // Origin guard: reject cross-origin actions, loader fetches, and
594
- // PE form submissions before any execution. Regular page navigations
595
- // (GET without _rsc_loader/_rsc_action) are not affected.
596
- const originPhase: OriginCheckPhase | null = isAction
597
- ? "action"
598
- : isLoaderFetch
599
- ? "loader"
600
- : request.method === "POST"
601
- ? "pe-form"
602
- : null;
689
+ // ---- 3. Origin guard (gate for action/loader/PE modes) ----
690
+ const originPhase = ORIGIN_CHECK_PHASE_BY_MODE[plan.mode];
603
691
  if (originPhase) {
604
692
  const originResult = await checkRequestOrigin(
605
693
  request,
@@ -649,13 +737,33 @@ export function createRSCHandler<
649
737
  }
650
738
  }
651
739
 
652
- // Get handle store from request context
740
+ // ---- 4. Execute ----
741
+ return executeRequest(
742
+ plan as ExecutableRequestPlan<TEnv>,
743
+ request,
744
+ env,
745
+ url,
746
+ variables,
747
+ nonce,
748
+ );
749
+ }
750
+
751
+ // Execute a classified request plan. Dispatches to the appropriate handler
752
+ // based on plan.mode. Lives in the createRSCHandler closure for access to
753
+ // handlerCtx, router, callOnError, etc.
754
+ // Only receives executable plans (version-mismatch is handled above).
755
+ async function executeRequest(
756
+ plan: ExecutableRequestPlan<TEnv>,
757
+ request: Request,
758
+ env: TEnv,
759
+ url: URL,
760
+ variables: Record<string, any>,
761
+ nonce: string | undefined,
762
+ ): Promise<Response> {
763
+ // Common setup
653
764
  const handleStore = requireRequestContext()._handleStore;
654
765
 
655
766
  // Wire up error reporting for late streaming-handle failures
656
- // (LateHandlePushError: handle pushed after stream completion).
657
- // Without this, these errors are only caught by React's error boundary
658
- // and never reach the router's onError callback or telemetry.
659
767
  handleStore.onError = (error: Error) => {
660
768
  const reqCtx = requireRequestContext();
661
769
  callOnError(error, "handler", {
@@ -685,37 +793,106 @@ export function createRSCHandler<
685
793
  };
686
794
 
687
795
  // Set route params early so all execution paths can access ctx.params.
688
- if (preview?.params) {
689
- setRequestContextParams(preview.params, preview.routeKey);
796
+ // Also store the classified snapshot so match/matchPartial can reuse it
797
+ // instead of calling resolveRoute again.
798
+ if (plan.mode !== "redirect") {
799
+ setRequestContextParams(plan.route.params, plan.route.routeKey);
800
+ requireRequestContext()._classifiedRoute = plan.route;
690
801
  }
691
802
 
692
- // Progressive enhancement runs before the normal action/render paths.
693
- // Route middleware wraps the PE re-render so handlers see the same
694
- // context variables regardless of JS/no-JS transport.
695
- const progressiveResult = await handleProgressiveEnhancement(
696
- handlerCtx,
697
- request,
698
- env,
699
- url,
700
- isAction,
701
- handleStore,
702
- nonce,
703
- {
704
- routeMiddleware: preview?.routeMiddleware,
803
+ const routeReverse = createReverseFunction(getRequiredRouteMap());
804
+
805
+ // ---- Response route: skip entire RSC pipeline ----
806
+ if (plan.mode === "response") {
807
+ // Build ResponseRouteMatch from plan fields. handleResponseRoute
808
+ // expects a flat object with params at the top level.
809
+ const responseMatch: ResponseRouteMatch = {
810
+ responseType: plan.responseType,
811
+ handler: plan.handler,
812
+ params: plan.route.params,
813
+ negotiated: plan.negotiated,
814
+ manifestEntry: plan.manifestEntry,
815
+ routeMiddleware: plan.routeMiddleware,
816
+ };
817
+ const responseOutcome = await withTimeout(
818
+ handleResponseRoute(
819
+ handlerCtx,
820
+ responseMatch,
821
+ request,
822
+ env,
823
+ url,
824
+ variables,
825
+ ),
826
+ router.timeouts.renderStartMs,
827
+ "render-start",
828
+ );
829
+ if (responseOutcome.timedOut) {
830
+ return handleTimeoutResponse(
831
+ request,
832
+ env,
833
+ url,
834
+ "render-start",
835
+ responseOutcome.durationMs,
836
+ plan.route.routeKey,
837
+ );
838
+ }
839
+ const response = responseOutcome.result;
840
+ if (plan.negotiated && !isWebSocketUpgradeResponse(response)) {
841
+ response.headers.append("Vary", "Accept");
842
+ }
843
+ return response;
844
+ }
845
+
846
+ // SSR setup: kick off in parallel for modes that need HTML rendering.
847
+ // Placed after response-route short-circuit so response/mime routes
848
+ // never pay for SSR work.
849
+ if (plan.mode !== "loader" && mayNeedSSR(request, url)) {
850
+ variables[SSR_SETUP_VAR] = startSSRSetup(
851
+ handlerCtx,
852
+ request,
853
+ env,
854
+ url,
855
+ router.debugPerformance
856
+ ? () => requireRequestContext()._metricsStore
857
+ : undefined,
858
+ );
859
+ }
860
+
861
+ // ---- Loader fetch ----
862
+ if (plan.mode === "loader") {
863
+ return handleLoaderFetch(
864
+ handlerCtx,
865
+ request,
866
+ env,
867
+ url,
705
868
  variables,
706
- routeReverse,
707
- },
708
- );
709
- if (progressiveResult) {
710
- return progressiveResult;
869
+ plan.route.params,
870
+ );
711
871
  }
712
872
 
713
- // --- Action execution: runs BEFORE route middleware ---
714
- // Route middleware wraps rendering only. For actions, the action runs
715
- // first in the global middleware context, then route middleware wraps
716
- // the revalidation pass (identical to a normal render).
717
- let actionContinuation: ActionContinuation | undefined;
718
- if (isAction && actionId) {
873
+ // ---- Progressive enhancement ----
874
+ if (plan.mode === "pe-render") {
875
+ const peResult = await handleProgressiveEnhancement(
876
+ handlerCtx,
877
+ request,
878
+ env,
879
+ url,
880
+ false, // isAction = false for PE
881
+ handleStore,
882
+ nonce,
883
+ {
884
+ routeMiddleware: plan.route.routeMiddleware,
885
+ variables,
886
+ routeReverse,
887
+ },
888
+ );
889
+ if (peResult) return peResult;
890
+ // PE handler returned null (not a PE form) — fall through to render
891
+ }
892
+
893
+ // ---- Action: execute action, then revalidate wrapped in route middleware ----
894
+ if (plan.mode === "action") {
895
+ let actionContinuation: ActionContinuation | undefined;
719
896
  try {
720
897
  const actionOutcome = await withTimeout(
721
898
  executeServerAction(
@@ -723,7 +900,7 @@ export function createRSCHandler<
723
900
  request,
724
901
  env,
725
902
  url,
726
- actionId,
903
+ plan.actionId,
727
904
  handleStore,
728
905
  ),
729
906
  router.timeouts.actionMs,
@@ -736,8 +913,8 @@ export function createRSCHandler<
736
913
  url,
737
914
  "action",
738
915
  actionOutcome.durationMs,
739
- preview?.routeKey,
740
- actionId,
916
+ plan.route.routeKey,
917
+ plan.actionId,
741
918
  );
742
919
  }
743
920
  const result = actionOutcome.result;
@@ -749,40 +926,237 @@ export function createRSCHandler<
749
926
  request,
750
927
  url,
751
928
  env,
752
- actionId,
929
+ actionId: plan.actionId,
753
930
  handledByBoundary: false,
754
931
  });
755
932
  console.error(`[RSC] Action error:`, error);
756
933
  throw error;
757
934
  }
758
- }
759
935
 
760
- // --- Rendering (action revalidation or navigation) ---
761
- // Route middleware wraps this same code path for both cases.
762
- const renderHandler = async () => {
763
- const response = await coreRequestHandlerInner(
936
+ // Revalidation render wrapped in route middleware.
937
+ // Actions from client-side navigation include _rsc_partial preserve
938
+ // the partial flag so the revalidation returns a Flight stream, not HTML.
939
+ // App-switch is already excluded by classifyRequest (would be full-render).
940
+ const isPartialAction = url.searchParams.has("_rsc_partial");
941
+ return executeRenderWithMiddleware(
942
+ plan.route.routeMiddleware,
943
+ plan.negotiated,
944
+ plan.route.routeKey,
945
+ routeReverse,
764
946
  request,
765
947
  env,
766
948
  url,
767
949
  variables,
768
950
  nonce,
769
- preview?.params,
770
- preview?.routeKey,
771
951
  handleStore,
952
+ isPartialAction,
772
953
  actionContinuation,
773
954
  );
774
- if (preview?.negotiated) {
775
- response.headers.append("Vary", "Accept");
955
+ }
956
+
957
+ // Full render, partial render, fallen-through PE, and full-page redirect all
958
+ // render through the same middleware-wrapped path. Only full/partial-render
959
+ // carry negotiation + the partial flag; pe/redirect render plainly.
960
+ const isPartial = plan.mode === "partial-render";
961
+ const negotiated =
962
+ plan.mode === "full-render" || plan.mode === "partial-render"
963
+ ? plan.negotiated
964
+ : false;
965
+ return executeRenderWithMiddleware(
966
+ plan.route.routeMiddleware,
967
+ negotiated,
968
+ plan.route.routeKey,
969
+ routeReverse,
970
+ request,
971
+ env,
972
+ url,
973
+ variables,
974
+ nonce,
975
+ handleStore,
976
+ isPartial,
977
+ );
978
+ }
979
+
980
+ // Shared render execution: wraps handleRscRendering (or revalidateAfterAction)
981
+ // in route middleware and timeout handling. Consolidates the pattern used by
982
+ // action-revalidate, full-render, and partial-render modes.
983
+ async function executeRenderWithMiddleware(
984
+ routeMiddleware: import("../router/middleware-types.js").CollectedMiddleware[],
985
+ negotiated: boolean,
986
+ routeKey: string,
987
+ routeReverse: ReturnType<typeof createReverseFunction>,
988
+ request: Request,
989
+ env: TEnv,
990
+ url: URL,
991
+ variables: Record<string, any>,
992
+ nonce: string | undefined,
993
+ handleStore: ReturnType<typeof requireRequestContext>["_handleStore"],
994
+ isPartial: boolean,
995
+ actionContinuation?: ActionContinuation,
996
+ ): Promise<Response> {
997
+ const renderHandler = async (): Promise<Response> => {
998
+ try {
999
+ let response: Response;
1000
+ if (actionContinuation) {
1001
+ response = await revalidateAfterAction(
1002
+ handlerCtx,
1003
+ request,
1004
+ env,
1005
+ url,
1006
+ handleStore,
1007
+ actionContinuation,
1008
+ );
1009
+ } else {
1010
+ response = await handleRscRendering(
1011
+ handlerCtx,
1012
+ request,
1013
+ env,
1014
+ url,
1015
+ isPartial,
1016
+ handleStore,
1017
+ nonce,
1018
+ );
1019
+ }
1020
+ if (negotiated && !isWebSocketUpgradeResponse(response)) {
1021
+ response.headers.append("Vary", "Accept");
1022
+ }
1023
+ return response;
1024
+ } catch (error) {
1025
+ // Check if middleware/handler returned Response
1026
+ if (error instanceof Response) {
1027
+ // An action revalidation render is delivered to the client over the
1028
+ // same Flight-parsing path as a partial navigation, so a Response
1029
+ // thrown during it must be converted exactly like a partial one
1030
+ // (raw 200 -> hard-nav hint, 3xx -> Flight redirect). Without this,
1031
+ // the no-middleware path returns the raw Response (the with-middleware
1032
+ // path is already covered by the isPartial || actionContinuation
1033
+ // guard below).
1034
+ const treatAsPartial = isPartial || actionContinuation != null;
1035
+
1036
+ // During partial (client-side navigation), a 200 Response from a handler
1037
+ // means the route serves raw content (JSON, text, etc.), not JSX.
1038
+ // Signal the browser to hard-navigate so it renders the raw response.
1039
+ if (treatAsPartial && error.status === 200) {
1040
+ console.warn(
1041
+ `[RSC] Route handler at ${url.pathname} returned a Response during client-side navigation. ` +
1042
+ `Falling back to hard navigation. Use data-external on the <Link> to avoid the extra round-trip.`,
1043
+ );
1044
+ return createResponseWithMergedHeaders(null, {
1045
+ status: 200,
1046
+ headers: {
1047
+ "X-RSC-Reload": stripInternalParams(url).toString(),
1048
+ "content-type": "text/x-component;charset=utf-8",
1049
+ },
1050
+ });
1051
+ }
1052
+
1053
+ if (treatAsPartial) {
1054
+ const intercepted = interceptRedirectForPartial(
1055
+ error,
1056
+ createRedirectFlightResponse,
1057
+ );
1058
+ if (intercepted) return intercepted;
1059
+ }
1060
+
1061
+ return error;
1062
+ }
1063
+
1064
+ // Render 404 page for unmatched routes
1065
+ if (isRouteNotFoundError(error)) {
1066
+ callOnError(error, "routing", {
1067
+ request,
1068
+ url,
1069
+ env,
1070
+ handledByBoundary: true,
1071
+ });
1072
+
1073
+ const notFoundOption = router.notFound;
1074
+ const notFoundComponent =
1075
+ typeof notFoundOption === "function"
1076
+ ? notFoundOption({ pathname: url.pathname })
1077
+ : (notFoundOption ?? createElement("h1", null, "Not Found"));
1078
+
1079
+ const notFoundSegment = {
1080
+ id: "notFound",
1081
+ namespace: "notFound",
1082
+ type: "route" as const,
1083
+ index: 0,
1084
+ component: notFoundComponent,
1085
+ params: {},
1086
+ };
1087
+
1088
+ const payload: RscPayload = {
1089
+ metadata: {
1090
+ pathname: url.pathname,
1091
+ routerId: router.id,
1092
+ basename: router.basename,
1093
+ segments: [notFoundSegment],
1094
+ matched: [],
1095
+ diff: [],
1096
+ isPartial: false,
1097
+ rootLayout: router.rootLayout,
1098
+ handles: handleStore.stream(),
1099
+ version,
1100
+ stateCookieName: router.resolvedStateCookieName,
1101
+ themeConfig: router.themeConfig,
1102
+ warmupEnabled: router.warmupEnabled,
1103
+ initialTheme: requireRequestContext().theme,
1104
+ },
1105
+ };
1106
+
1107
+ const rscStream = renderToReadableStream(payload, {
1108
+ onError: (error: unknown) => {
1109
+ callOnError(error, "rendering", { request, url, env });
1110
+ },
1111
+ });
1112
+
1113
+ if (isRscRequest(request, url, isPartial)) {
1114
+ return createResponseWithMergedHeaders(rscStream, {
1115
+ status: 404,
1116
+ headers: {
1117
+ "content-type": "text/x-component;charset=utf-8",
1118
+ // Router identity for the client's pre-decode integrity check; a
1119
+ // same-app 404 matches and applies in place. See response-adapter.
1120
+ "X-RSC-Router-Id": router.id,
1121
+ },
1122
+ });
1123
+ }
1124
+
1125
+ const [ssrModule, streamMode] = await getSSRSetup(
1126
+ handlerCtx,
1127
+ request,
1128
+ env,
1129
+ url,
1130
+ requireRequestContext()._metricsStore,
1131
+ );
1132
+ const htmlStream = await ssrModule.renderHTML(rscStream, {
1133
+ nonce,
1134
+ streamMode,
1135
+ });
1136
+
1137
+ return createResponseWithMergedHeaders(htmlStream, {
1138
+ status: 404,
1139
+ headers: { "content-type": "text/html;charset=utf-8" },
1140
+ });
1141
+ }
1142
+
1143
+ // Report unhandled errors
1144
+ callOnError(error, "routing", {
1145
+ request,
1146
+ url,
1147
+ env,
1148
+ handledByBoundary: false,
1149
+ });
1150
+ console.error(`[RSC] Error:`, error);
1151
+ throw error;
776
1152
  }
777
- return response;
778
1153
  };
779
1154
 
780
- // Wrap the render path (with or without route middleware) in a
781
- // renderStartMs timeout so slow renders are caught before output.
1155
+ // Wrap the render path in a renderStartMs timeout
782
1156
  const executeRender = async (): Promise<Response> => {
783
- if (preview?.routeMiddleware && preview.routeMiddleware.length > 0) {
1157
+ if (routeMiddleware.length > 0) {
784
1158
  const mwResponse = await executeMiddleware(
785
- buildRouteMiddlewareEntries<TEnv>(preview.routeMiddleware),
1159
+ buildRouteMiddlewareEntries<TEnv>(routeMiddleware),
786
1160
  request,
787
1161
  env,
788
1162
  variables,
@@ -790,10 +1164,7 @@ export function createRSCHandler<
790
1164
  routeReverse,
791
1165
  );
792
1166
 
793
- if (
794
- url.searchParams.has("_rsc_partial") ||
795
- url.searchParams.has("_rsc_action")
796
- ) {
1167
+ if (isPartial || actionContinuation) {
797
1168
  const intercepted = interceptRedirectForPartial(
798
1169
  mwResponse,
799
1170
  createRedirectFlightResponse,
@@ -804,7 +1175,6 @@ export function createRSCHandler<
804
1175
  return finalizeResponse(mwResponse);
805
1176
  }
806
1177
 
807
- // No route middleware, proceed directly
808
1178
  return renderHandler();
809
1179
  };
810
1180
 
@@ -820,270 +1190,9 @@ export function createRSCHandler<
820
1190
  url,
821
1191
  "render-start",
822
1192
  renderOutcome.durationMs,
823
- preview?.routeKey,
1193
+ routeKey,
824
1194
  );
825
1195
  }
826
1196
  return renderOutcome.result;
827
1197
  }
828
-
829
- // Inner request handler: rendering logic wrapped by route middleware.
830
- // Handles action revalidation (when actionContinuation is present),
831
- // loader fetches, and regular RSC rendering.
832
- async function coreRequestHandlerInner(
833
- request: Request,
834
- env: TEnv,
835
- url: URL,
836
- variables: Record<string, any>,
837
- nonce: string | undefined,
838
- routeParams?: Record<string, string>,
839
- routeKey?: string,
840
- handleStore?: ReturnType<typeof requireRequestContext>["_handleStore"],
841
- actionContinuation?: ActionContinuation,
842
- ): Promise<Response> {
843
- const isPartial = url.searchParams.has("_rsc_partial");
844
- const isAction =
845
- request.headers.has("rsc-action") || url.searchParams.has("_rsc_action");
846
-
847
- // Version mismatch detection - client may have stale code after HMR/deployment
848
- // If versions don't match, tell the client to reload
849
- const clientVersion = url.searchParams.get("_rsc_v");
850
- if (version && clientVersion && clientVersion !== version) {
851
- console.log(
852
- `[RSC] Version mismatch: client=${clientVersion}, server=${version}. Forcing reload.`,
853
- );
854
-
855
- // For actions, reload current page (referer) if same origin.
856
- // For navigation, load the target URL.
857
- // Validate referer origin to prevent open redirect via crafted header.
858
- let reloadUrl = stripInternalParams(url).toString();
859
- if (isAction) {
860
- const referer = request.headers.get("referer");
861
- if (referer) {
862
- try {
863
- const refererUrl = new URL(referer);
864
- if (refererUrl.origin === url.origin) {
865
- reloadUrl = referer;
866
- }
867
- } catch {
868
- // Malformed referer, fall back to cleanUrl
869
- }
870
- }
871
- }
872
-
873
- // Return special response that tells client to reload
874
- return createResponseWithMergedHeaders(null, {
875
- status: 200,
876
- headers: {
877
- "X-RSC-Reload": reloadUrl,
878
- "content-type": "text/x-component;charset=utf-8",
879
- },
880
- });
881
- }
882
- // Debug manifest endpoint: ?__debug_manifest on any route.
883
- // Always available in dev, requires allowDebugManifest option in production.
884
- const isDev = process.env.NODE_ENV !== "production";
885
- if (
886
- url.searchParams.has("__debug_manifest") &&
887
- (isDev || router.allowDebugManifest)
888
- ) {
889
- const trie = getRouterTrie(router.id) ?? getRouteTrie();
890
- const routeManifest = getRequiredRouteMap();
891
- const { extractAncestryFromTrie } =
892
- await import("../build/route-trie.js");
893
- return new Response(
894
- JSON.stringify(
895
- {
896
- routerId: router.id,
897
- routeManifest,
898
- routeAncestry: trie ? extractAncestryFromTrie(trie) : {},
899
- routeTrie: trie,
900
- precomputedEntries: getPrecomputedEntries(),
901
- },
902
- null,
903
- 2,
904
- ),
905
- {
906
- headers: { "Content-Type": "application/json" },
907
- },
908
- );
909
- }
910
-
911
- const store = handleStore ?? requireRequestContext()._handleStore;
912
-
913
- try {
914
- // Route params were already set in coreRequestHandler, but set again
915
- // for callers that enter coreRequestHandlerInner directly.
916
- if (routeParams) {
917
- setRequestContextParams(routeParams, routeKey);
918
- }
919
-
920
- // ============================================================================
921
- // ACTION REVALIDATION (action already executed, revalidate segments)
922
- // ============================================================================
923
- if (actionContinuation) {
924
- return await revalidateAfterAction(
925
- handlerCtx,
926
- request,
927
- env,
928
- url,
929
- store,
930
- actionContinuation,
931
- );
932
- }
933
-
934
- // ============================================================================
935
- // LOADER FETCH EXECUTION (data fetching with RSC serialization)
936
- // ============================================================================
937
- const isLoaderRequest = url.searchParams.has("_rsc_loader");
938
- if (isLoaderRequest) {
939
- return handleLoaderFetch(
940
- handlerCtx,
941
- request,
942
- env,
943
- url,
944
- variables,
945
- routeParams,
946
- );
947
- }
948
-
949
- // ============================================================================
950
- // REGULAR RSC RENDERING (Navigation)
951
- // ============================================================================
952
- // Note: Must use "return await" for try/catch to catch async rejections
953
- return await handleRscRendering(
954
- handlerCtx,
955
- request,
956
- env,
957
- url,
958
- isPartial,
959
- store,
960
- nonce,
961
- );
962
- } catch (error) {
963
- // Check if middleware/handler returned Response
964
- if (error instanceof Response) {
965
- // During partial (client-side navigation), a 200 Response from a handler
966
- // means the route serves raw content (JSON, text, etc.), not JSX.
967
- // Signal the browser to hard-navigate so it renders the raw response.
968
- // Only for 200 — redirects (3xx) work already because the browser follows
969
- // them automatically to a URL that serves Flight data.
970
- if (isPartial && error.status === 200) {
971
- console.warn(
972
- `[RSC] Route handler at ${url.pathname} returned a Response during client-side navigation. ` +
973
- `Falling back to hard navigation. Use data-external on the <Link> to avoid the extra round-trip.`,
974
- );
975
- return createResponseWithMergedHeaders(null, {
976
- status: 200,
977
- headers: {
978
- "X-RSC-Reload": stripInternalParams(url).toString(),
979
- "content-type": "text/x-component;charset=utf-8",
980
- },
981
- });
982
- }
983
-
984
- if (isPartial) {
985
- const intercepted = interceptRedirectForPartial(
986
- error,
987
- createRedirectFlightResponse,
988
- );
989
- if (intercepted) return intercepted;
990
- }
991
-
992
- return error;
993
- }
994
-
995
- // Render 404 page for unmatched routes
996
- // Check both instanceof and error.name for cross-bundle compatibility
997
- const isRouteNotFound =
998
- error instanceof RouteNotFoundError ||
999
- (error instanceof Error && error.name === "RouteNotFoundError");
1000
- if (isRouteNotFound) {
1001
- callOnError(error, "routing", {
1002
- request,
1003
- url,
1004
- env,
1005
- handledByBoundary: true, // Handled by notFound component
1006
- });
1007
-
1008
- // Get notFound component from router options or use default
1009
- const notFoundOption = router.notFound;
1010
- const notFoundComponent =
1011
- typeof notFoundOption === "function"
1012
- ? notFoundOption({ pathname: url.pathname })
1013
- : (notFoundOption ?? createElement("h1", null, "Not Found"));
1014
-
1015
- // Create a simple segment for the 404 page
1016
- const notFoundSegment = {
1017
- id: "notFound",
1018
- namespace: "notFound",
1019
- type: "route" as const,
1020
- index: 0,
1021
- component: notFoundComponent,
1022
- params: {},
1023
- };
1024
-
1025
- const payload: RscPayload = {
1026
- metadata: {
1027
- pathname: url.pathname,
1028
- segments: [notFoundSegment],
1029
- matched: [],
1030
- diff: [],
1031
- isPartial: false,
1032
- rootLayout: router.rootLayout,
1033
- handles: store.stream(),
1034
- version,
1035
- themeConfig: router.themeConfig,
1036
- warmupEnabled: router.warmupEnabled,
1037
- initialTheme: requireRequestContext().theme,
1038
- // No routeName for not-found routes
1039
- },
1040
- };
1041
-
1042
- const rscStream = renderToReadableStream(payload);
1043
-
1044
- // Determine if this is an RSC request or HTML request.
1045
- // Partial requests are always RSC (see main isRscRequest comment).
1046
- const isRscRequest =
1047
- isPartial ||
1048
- (!request.headers.get("accept")?.includes("text/html") &&
1049
- !url.searchParams.has("__html")) ||
1050
- url.searchParams.has("__rsc");
1051
-
1052
- if (isRscRequest) {
1053
- return createResponseWithMergedHeaders(rscStream, {
1054
- status: 404,
1055
- headers: { "content-type": "text/x-component;charset=utf-8" },
1056
- });
1057
- }
1058
-
1059
- // Delegate to SSR for HTML response (reuse early setup if available)
1060
- const [ssrModule, streamMode] = await getSSRSetup(
1061
- handlerCtx,
1062
- request,
1063
- env,
1064
- url,
1065
- requireRequestContext()._metricsStore,
1066
- );
1067
- const htmlStream = await ssrModule.renderHTML(rscStream, {
1068
- nonce,
1069
- streamMode,
1070
- });
1071
-
1072
- return createResponseWithMergedHeaders(htmlStream, {
1073
- status: 404,
1074
- headers: { "content-type": "text/html;charset=utf-8" },
1075
- });
1076
- }
1077
-
1078
- // Report unhandled errors
1079
- callOnError(error, "routing", {
1080
- request,
1081
- url,
1082
- env,
1083
- handledByBoundary: false,
1084
- });
1085
- console.error(`[RSC] Error:`, error);
1086
- throw error;
1087
- }
1088
- }
1089
1198
  }