@rangojs/router 0.0.0-experimental.0f44aca1

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 (305) hide show
  1. package/AGENTS.md +5 -0
  2. package/README.md +899 -0
  3. package/dist/bin/rango.js +1601 -0
  4. package/dist/vite/index.js +5214 -0
  5. package/package.json +176 -0
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +262 -0
  8. package/skills/caching/SKILL.md +220 -0
  9. package/skills/composability/SKILL.md +172 -0
  10. package/skills/debug-manifest/SKILL.md +112 -0
  11. package/skills/document-cache/SKILL.md +182 -0
  12. package/skills/fonts/SKILL.md +167 -0
  13. package/skills/hooks/SKILL.md +704 -0
  14. package/skills/host-router/SKILL.md +218 -0
  15. package/skills/intercept/SKILL.md +313 -0
  16. package/skills/layout/SKILL.md +310 -0
  17. package/skills/links/SKILL.md +239 -0
  18. package/skills/loader/SKILL.md +596 -0
  19. package/skills/middleware/SKILL.md +339 -0
  20. package/skills/mime-routes/SKILL.md +128 -0
  21. package/skills/parallel/SKILL.md +305 -0
  22. package/skills/prerender/SKILL.md +643 -0
  23. package/skills/rango/SKILL.md +118 -0
  24. package/skills/response-routes/SKILL.md +411 -0
  25. package/skills/route/SKILL.md +385 -0
  26. package/skills/router-setup/SKILL.md +439 -0
  27. package/skills/tailwind/SKILL.md +129 -0
  28. package/skills/theme/SKILL.md +79 -0
  29. package/skills/typesafety/SKILL.md +623 -0
  30. package/skills/use-cache/SKILL.md +324 -0
  31. package/src/__internal.ts +273 -0
  32. package/src/bin/rango.ts +321 -0
  33. package/src/browser/action-coordinator.ts +97 -0
  34. package/src/browser/action-response-classifier.ts +99 -0
  35. package/src/browser/event-controller.ts +899 -0
  36. package/src/browser/history-state.ts +80 -0
  37. package/src/browser/index.ts +18 -0
  38. package/src/browser/intercept-utils.ts +52 -0
  39. package/src/browser/link-interceptor.ts +141 -0
  40. package/src/browser/logging.ts +55 -0
  41. package/src/browser/merge-segment-loaders.ts +134 -0
  42. package/src/browser/navigation-bridge.ts +645 -0
  43. package/src/browser/navigation-client.ts +215 -0
  44. package/src/browser/navigation-store.ts +806 -0
  45. package/src/browser/navigation-transaction.ts +295 -0
  46. package/src/browser/network-error-handler.ts +61 -0
  47. package/src/browser/partial-update.ts +550 -0
  48. package/src/browser/prefetch/cache.ts +146 -0
  49. package/src/browser/prefetch/fetch.ts +135 -0
  50. package/src/browser/prefetch/observer.ts +65 -0
  51. package/src/browser/prefetch/policy.ts +42 -0
  52. package/src/browser/prefetch/queue.ts +88 -0
  53. package/src/browser/rango-state.ts +112 -0
  54. package/src/browser/react/Link.tsx +360 -0
  55. package/src/browser/react/NavigationProvider.tsx +386 -0
  56. package/src/browser/react/ScrollRestoration.tsx +94 -0
  57. package/src/browser/react/context.ts +59 -0
  58. package/src/browser/react/filter-segment-order.ts +11 -0
  59. package/src/browser/react/index.ts +52 -0
  60. package/src/browser/react/location-state-shared.ts +162 -0
  61. package/src/browser/react/location-state.ts +107 -0
  62. package/src/browser/react/mount-context.ts +37 -0
  63. package/src/browser/react/nonce-context.ts +23 -0
  64. package/src/browser/react/shallow-equal.ts +27 -0
  65. package/src/browser/react/use-action.ts +218 -0
  66. package/src/browser/react/use-client-cache.ts +58 -0
  67. package/src/browser/react/use-handle.ts +162 -0
  68. package/src/browser/react/use-href.tsx +40 -0
  69. package/src/browser/react/use-link-status.ts +135 -0
  70. package/src/browser/react/use-mount.ts +31 -0
  71. package/src/browser/react/use-navigation.ts +99 -0
  72. package/src/browser/react/use-params.ts +65 -0
  73. package/src/browser/react/use-pathname.ts +47 -0
  74. package/src/browser/react/use-router.ts +63 -0
  75. package/src/browser/react/use-search-params.ts +56 -0
  76. package/src/browser/react/use-segments.ts +171 -0
  77. package/src/browser/response-adapter.ts +73 -0
  78. package/src/browser/rsc-router.tsx +431 -0
  79. package/src/browser/scroll-restoration.ts +400 -0
  80. package/src/browser/segment-reconciler.ts +216 -0
  81. package/src/browser/segment-structure-assert.ts +83 -0
  82. package/src/browser/server-action-bridge.ts +667 -0
  83. package/src/browser/shallow.ts +40 -0
  84. package/src/browser/types.ts +538 -0
  85. package/src/browser/validate-redirect-origin.ts +29 -0
  86. package/src/build/generate-manifest.ts +438 -0
  87. package/src/build/generate-route-types.ts +36 -0
  88. package/src/build/index.ts +35 -0
  89. package/src/build/route-trie.ts +265 -0
  90. package/src/build/route-types/ast-helpers.ts +25 -0
  91. package/src/build/route-types/ast-route-extraction.ts +98 -0
  92. package/src/build/route-types/codegen.ts +102 -0
  93. package/src/build/route-types/include-resolution.ts +411 -0
  94. package/src/build/route-types/param-extraction.ts +48 -0
  95. package/src/build/route-types/per-module-writer.ts +128 -0
  96. package/src/build/route-types/router-processing.ts +469 -0
  97. package/src/build/route-types/scan-filter.ts +78 -0
  98. package/src/build/runtime-discovery.ts +231 -0
  99. package/src/cache/background-task.ts +34 -0
  100. package/src/cache/cache-key-utils.ts +44 -0
  101. package/src/cache/cache-policy.ts +125 -0
  102. package/src/cache/cache-runtime.ts +338 -0
  103. package/src/cache/cache-scope.ts +382 -0
  104. package/src/cache/cf/cf-cache-store.ts +540 -0
  105. package/src/cache/cf/index.ts +25 -0
  106. package/src/cache/document-cache.ts +369 -0
  107. package/src/cache/handle-capture.ts +81 -0
  108. package/src/cache/handle-snapshot.ts +41 -0
  109. package/src/cache/index.ts +43 -0
  110. package/src/cache/memory-segment-store.ts +328 -0
  111. package/src/cache/profile-registry.ts +73 -0
  112. package/src/cache/read-through-swr.ts +134 -0
  113. package/src/cache/segment-codec.ts +256 -0
  114. package/src/cache/taint.ts +98 -0
  115. package/src/cache/types.ts +342 -0
  116. package/src/client.rsc.tsx +85 -0
  117. package/src/client.tsx +601 -0
  118. package/src/component-utils.ts +76 -0
  119. package/src/components/DefaultDocument.tsx +27 -0
  120. package/src/context-var.ts +86 -0
  121. package/src/debug.ts +243 -0
  122. package/src/default-error-boundary.tsx +88 -0
  123. package/src/deps/browser.ts +8 -0
  124. package/src/deps/html-stream-client.ts +2 -0
  125. package/src/deps/html-stream-server.ts +2 -0
  126. package/src/deps/rsc.ts +10 -0
  127. package/src/deps/ssr.ts +2 -0
  128. package/src/errors.ts +365 -0
  129. package/src/handle.ts +135 -0
  130. package/src/handles/MetaTags.tsx +246 -0
  131. package/src/handles/breadcrumbs.ts +66 -0
  132. package/src/handles/index.ts +7 -0
  133. package/src/handles/meta.ts +264 -0
  134. package/src/host/cookie-handler.ts +165 -0
  135. package/src/host/errors.ts +97 -0
  136. package/src/host/index.ts +53 -0
  137. package/src/host/pattern-matcher.ts +214 -0
  138. package/src/host/router.ts +352 -0
  139. package/src/host/testing.ts +79 -0
  140. package/src/host/types.ts +146 -0
  141. package/src/host/utils.ts +25 -0
  142. package/src/href-client.ts +222 -0
  143. package/src/index.rsc.ts +233 -0
  144. package/src/index.ts +277 -0
  145. package/src/internal-debug.ts +11 -0
  146. package/src/loader.rsc.ts +89 -0
  147. package/src/loader.ts +64 -0
  148. package/src/network-error-thrower.tsx +23 -0
  149. package/src/outlet-context.ts +15 -0
  150. package/src/outlet-provider.tsx +45 -0
  151. package/src/prerender/param-hash.ts +37 -0
  152. package/src/prerender/store.ts +185 -0
  153. package/src/prerender.ts +463 -0
  154. package/src/reverse.ts +330 -0
  155. package/src/root-error-boundary.tsx +289 -0
  156. package/src/route-content-wrapper.tsx +196 -0
  157. package/src/route-definition/dsl-helpers.ts +934 -0
  158. package/src/route-definition/helper-factories.ts +200 -0
  159. package/src/route-definition/helpers-types.ts +430 -0
  160. package/src/route-definition/index.ts +52 -0
  161. package/src/route-definition/redirect.ts +93 -0
  162. package/src/route-definition.ts +1 -0
  163. package/src/route-map-builder.ts +275 -0
  164. package/src/route-name.ts +53 -0
  165. package/src/route-types.ts +259 -0
  166. package/src/router/content-negotiation.ts +116 -0
  167. package/src/router/debug-manifest.ts +72 -0
  168. package/src/router/error-handling.ts +287 -0
  169. package/src/router/find-match.ts +158 -0
  170. package/src/router/handler-context.ts +451 -0
  171. package/src/router/intercept-resolution.ts +395 -0
  172. package/src/router/lazy-includes.ts +234 -0
  173. package/src/router/loader-resolution.ts +420 -0
  174. package/src/router/logging.ts +248 -0
  175. package/src/router/manifest.ts +267 -0
  176. package/src/router/match-api.ts +620 -0
  177. package/src/router/match-context.ts +266 -0
  178. package/src/router/match-handlers.ts +440 -0
  179. package/src/router/match-middleware/background-revalidation.ts +223 -0
  180. package/src/router/match-middleware/cache-lookup.ts +634 -0
  181. package/src/router/match-middleware/cache-store.ts +295 -0
  182. package/src/router/match-middleware/index.ts +81 -0
  183. package/src/router/match-middleware/intercept-resolution.ts +306 -0
  184. package/src/router/match-middleware/segment-resolution.ts +192 -0
  185. package/src/router/match-pipelines.ts +179 -0
  186. package/src/router/match-result.ts +219 -0
  187. package/src/router/metrics.ts +282 -0
  188. package/src/router/middleware-cookies.ts +55 -0
  189. package/src/router/middleware-types.ts +222 -0
  190. package/src/router/middleware.ts +748 -0
  191. package/src/router/pattern-matching.ts +563 -0
  192. package/src/router/prerender-match.ts +402 -0
  193. package/src/router/preview-match.ts +170 -0
  194. package/src/router/revalidation.ts +289 -0
  195. package/src/router/router-context.ts +316 -0
  196. package/src/router/router-interfaces.ts +452 -0
  197. package/src/router/router-options.ts +592 -0
  198. package/src/router/router-registry.ts +24 -0
  199. package/src/router/segment-resolution/fresh.ts +570 -0
  200. package/src/router/segment-resolution/helpers.ts +263 -0
  201. package/src/router/segment-resolution/loader-cache.ts +198 -0
  202. package/src/router/segment-resolution/revalidation.ts +1239 -0
  203. package/src/router/segment-resolution/static-store.ts +67 -0
  204. package/src/router/segment-resolution.ts +21 -0
  205. package/src/router/segment-wrappers.ts +289 -0
  206. package/src/router/telemetry-otel.ts +299 -0
  207. package/src/router/telemetry.ts +300 -0
  208. package/src/router/timeout.ts +148 -0
  209. package/src/router/trie-matching.ts +239 -0
  210. package/src/router/types.ts +170 -0
  211. package/src/router.ts +1002 -0
  212. package/src/rsc/handler-context.ts +45 -0
  213. package/src/rsc/handler.ts +1089 -0
  214. package/src/rsc/helpers.ts +198 -0
  215. package/src/rsc/index.ts +36 -0
  216. package/src/rsc/loader-fetch.ts +209 -0
  217. package/src/rsc/manifest-init.ts +86 -0
  218. package/src/rsc/nonce.ts +32 -0
  219. package/src/rsc/origin-guard.ts +141 -0
  220. package/src/rsc/progressive-enhancement.ts +379 -0
  221. package/src/rsc/response-error.ts +37 -0
  222. package/src/rsc/response-route-handler.ts +347 -0
  223. package/src/rsc/rsc-rendering.ts +235 -0
  224. package/src/rsc/runtime-warnings.ts +42 -0
  225. package/src/rsc/server-action.ts +348 -0
  226. package/src/rsc/ssr-setup.ts +128 -0
  227. package/src/rsc/types.ts +263 -0
  228. package/src/search-params.ts +230 -0
  229. package/src/segment-system.tsx +454 -0
  230. package/src/server/context.ts +591 -0
  231. package/src/server/cookie-store.ts +190 -0
  232. package/src/server/fetchable-loader-store.ts +37 -0
  233. package/src/server/handle-store.ts +308 -0
  234. package/src/server/loader-registry.ts +133 -0
  235. package/src/server/request-context.ts +914 -0
  236. package/src/server/root-layout.tsx +10 -0
  237. package/src/server/tsconfig.json +14 -0
  238. package/src/server.ts +51 -0
  239. package/src/ssr/index.tsx +365 -0
  240. package/src/static-handler.ts +114 -0
  241. package/src/theme/ThemeProvider.tsx +297 -0
  242. package/src/theme/ThemeScript.tsx +61 -0
  243. package/src/theme/constants.ts +62 -0
  244. package/src/theme/index.ts +48 -0
  245. package/src/theme/theme-context.ts +44 -0
  246. package/src/theme/theme-script.ts +155 -0
  247. package/src/theme/types.ts +182 -0
  248. package/src/theme/use-theme.ts +44 -0
  249. package/src/types/boundaries.ts +158 -0
  250. package/src/types/cache-types.ts +198 -0
  251. package/src/types/error-types.ts +192 -0
  252. package/src/types/global-namespace.ts +100 -0
  253. package/src/types/handler-context.ts +687 -0
  254. package/src/types/index.ts +88 -0
  255. package/src/types/loader-types.ts +183 -0
  256. package/src/types/route-config.ts +170 -0
  257. package/src/types/route-entry.ts +102 -0
  258. package/src/types/segments.ts +148 -0
  259. package/src/types.ts +1 -0
  260. package/src/urls/include-helper.ts +197 -0
  261. package/src/urls/index.ts +53 -0
  262. package/src/urls/path-helper-types.ts +339 -0
  263. package/src/urls/path-helper.ts +329 -0
  264. package/src/urls/pattern-types.ts +95 -0
  265. package/src/urls/response-types.ts +106 -0
  266. package/src/urls/type-extraction.ts +372 -0
  267. package/src/urls/urls-function.ts +98 -0
  268. package/src/urls.ts +1 -0
  269. package/src/use-loader.tsx +354 -0
  270. package/src/vite/discovery/bundle-postprocess.ts +184 -0
  271. package/src/vite/discovery/discover-routers.ts +344 -0
  272. package/src/vite/discovery/prerender-collection.ts +385 -0
  273. package/src/vite/discovery/route-types-writer.ts +258 -0
  274. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  275. package/src/vite/discovery/state.ts +110 -0
  276. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  277. package/src/vite/index.ts +16 -0
  278. package/src/vite/plugin-types.ts +131 -0
  279. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  280. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  281. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  282. package/src/vite/plugins/expose-action-id.ts +365 -0
  283. package/src/vite/plugins/expose-id-utils.ts +287 -0
  284. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  285. package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
  286. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  287. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  288. package/src/vite/plugins/expose-ids/types.ts +45 -0
  289. package/src/vite/plugins/expose-internal-ids.ts +569 -0
  290. package/src/vite/plugins/refresh-cmd.ts +65 -0
  291. package/src/vite/plugins/use-cache-transform.ts +323 -0
  292. package/src/vite/plugins/version-injector.ts +83 -0
  293. package/src/vite/plugins/version-plugin.ts +254 -0
  294. package/src/vite/plugins/version.d.ts +12 -0
  295. package/src/vite/plugins/virtual-entries.ts +123 -0
  296. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  297. package/src/vite/rango.ts +510 -0
  298. package/src/vite/router-discovery.ts +785 -0
  299. package/src/vite/utils/ast-handler-extract.ts +517 -0
  300. package/src/vite/utils/banner.ts +36 -0
  301. package/src/vite/utils/bundle-analysis.ts +137 -0
  302. package/src/vite/utils/manifest-utils.ts +70 -0
  303. package/src/vite/utils/package-resolution.ts +121 -0
  304. package/src/vite/utils/prerender-utils.ts +189 -0
  305. package/src/vite/utils/shared-utils.ts +169 -0
@@ -0,0 +1,667 @@
1
+ import type {
2
+ ServerActionBridge,
3
+ ServerActionBridgeConfig,
4
+ RscPayload,
5
+ } from "./types.js";
6
+ import { createPartialUpdater } from "./partial-update.js";
7
+ import { createNavigationTransaction } from "./navigation-transaction.js";
8
+ import {
9
+ reconcileSegments,
10
+ reconcileErrorSegments,
11
+ } from "./segment-reconciler.js";
12
+ import { startTransition } from "react";
13
+ import type { EventController } from "./event-controller.js";
14
+ import {
15
+ toNetworkError,
16
+ emitNetworkError,
17
+ isBackgroundSuppressible,
18
+ } from "./network-error-handler.js";
19
+ import {
20
+ browserDebugLog,
21
+ isBrowserDebugEnabled,
22
+ startBrowserTransaction,
23
+ } from "./logging.js";
24
+ import { validateRedirectOrigin } from "./validate-redirect-origin.js";
25
+ import {
26
+ extractRscHeaderUrl,
27
+ emptyResponse,
28
+ teeWithCompletion,
29
+ } from "./response-adapter.js";
30
+ import { mergeLocationState } from "./history-state.js";
31
+ import { classifyActionOutcome } from "./action-coordinator.js";
32
+
33
+ // Polyfill Symbol.dispose/asyncDispose for Safari and older browsers
34
+ if (typeof Symbol.dispose === "undefined") {
35
+ (Symbol as any).dispose = Symbol("Symbol.dispose");
36
+ }
37
+ if (typeof Symbol.asyncDispose === "undefined") {
38
+ (Symbol as any).asyncDispose = Symbol("Symbol.asyncDispose");
39
+ }
40
+
41
+ /**
42
+ * Extended configuration for server action bridge with event controller
43
+ */
44
+ export interface ServerActionBridgeConfigWithController extends ServerActionBridgeConfig {
45
+ eventController: EventController;
46
+ /** RSC version from initial payload metadata */
47
+ version?: string;
48
+ /** Callback to trigger SPA navigation (for action redirects) */
49
+ onNavigate?: (
50
+ url: string,
51
+ options?: { state?: unknown; replace?: boolean; _skipCache?: boolean },
52
+ ) => Promise<void>;
53
+ }
54
+
55
+ /**
56
+ * Create a server action bridge for handling RSC server actions
57
+ *
58
+ * The bridge registers a callback with the RSC runtime that handles:
59
+ * - Encoding action arguments
60
+ * - Sending action requests to the server
61
+ * - Processing responses and updating UI
62
+ * - Managing concurrent action requests via event controller
63
+ * - HMR resilience (refetching if segments are missing)
64
+ *
65
+ * @param config - Bridge configuration
66
+ * @returns ServerActionBridge instance
67
+ */
68
+ export function createServerActionBridge(
69
+ config: ServerActionBridgeConfigWithController,
70
+ ): ServerActionBridge {
71
+ const {
72
+ store,
73
+ client,
74
+ eventController,
75
+ deps,
76
+ onUpdate,
77
+ renderSegments,
78
+ version,
79
+ onNavigate,
80
+ } = config;
81
+
82
+ let isRegistered = false;
83
+
84
+ const fetchPartialUpdate = createPartialUpdater({
85
+ store,
86
+ client,
87
+ onUpdate,
88
+ renderSegments,
89
+ version,
90
+ });
91
+
92
+ /**
93
+ * Refetch current route via a navigation transaction.
94
+ * Encapsulates the repeated pattern of creating a navTx + fetchPartialUpdate
95
+ * used by navigated-away, hmr-missing, and consolidation-needed scenarios.
96
+ */
97
+ async function refetchRoute(opts?: {
98
+ segments?: string[];
99
+ interceptSourceUrl?: string | null;
100
+ }): Promise<void> {
101
+ const src = opts?.interceptSourceUrl ?? null;
102
+ const navTx = createNavigationTransaction(
103
+ store,
104
+ eventController,
105
+ window.location.href,
106
+ { replace: true, skipLoadingState: true },
107
+ );
108
+ try {
109
+ await fetchPartialUpdate(
110
+ window.location.href,
111
+ opts?.segments ?? [],
112
+ false,
113
+ navTx.handle.signal,
114
+ navTx.with({
115
+ url: window.location.href,
116
+ storeOnly: true,
117
+ ...(src ? { intercept: true, interceptSourceUrl: src } : {}),
118
+ }),
119
+ {
120
+ type: "action" as const,
121
+ ...(src ? { interceptSourceUrl: src } : {}),
122
+ },
123
+ );
124
+ } finally {
125
+ navTx[Symbol.dispose]();
126
+ }
127
+ }
128
+
129
+ /**
130
+ * Server action callback handler
131
+ */
132
+ async function handleServerAction(id: string, args: any[]): Promise<unknown> {
133
+ const tx = isBrowserDebugEnabled()
134
+ ? startBrowserTransaction("action")
135
+ : null;
136
+ const log = (msg: string, details?: Record<string, unknown>) => {
137
+ if (tx) browserDebugLog(tx, msg, details);
138
+ };
139
+
140
+ const locationKey = window.history.state?.key;
141
+ log("action start", { id, argsCount: args.length });
142
+
143
+ // Start action in event controller - handles lifecycle tracking
144
+ const handle = eventController.startAction(id, args);
145
+ try {
146
+ const segmentState = store.getSegmentState();
147
+
148
+ // Mark cache as stale immediately when action starts
149
+ // This ensures SWR pattern kicks in if user navigates away during action
150
+ store.markCacheAsStaleAndBroadcast();
151
+
152
+ // Create temporary references for serialization
153
+ const temporaryReferences = deps.createTemporaryReferenceSet();
154
+
155
+ // Capture URL pathname at action start to detect navigation during action
156
+ // Must use window.location (not store.path) because intercepts change URL
157
+ // without changing store.path (e.g., /kanban -> /kanban/card/1)
158
+ const actionStartPathname = window.location.pathname;
159
+
160
+ // Build action request URL with current segments
161
+ const url = new URL(window.location.href);
162
+ url.searchParams.set("_rsc_action", id);
163
+ url.searchParams.set(
164
+ "_rsc_segments",
165
+ segmentState.currentSegmentIds.join(","),
166
+ );
167
+ // Add version param for version mismatch detection
168
+ if (version) {
169
+ url.searchParams.set("_rsc_v", version);
170
+ }
171
+
172
+ // Encode arguments
173
+ const encodedBody = await deps.encodeReply(args, { temporaryReferences });
174
+
175
+ log("sending action request", {
176
+ url: url.href,
177
+ bodyType: typeof encodedBody,
178
+ isFormData: encodedBody instanceof FormData,
179
+ segmentCount: segmentState.currentSegmentIds.length,
180
+ });
181
+
182
+ // Track when the stream completes
183
+ let resolveStreamComplete: () => void;
184
+ const streamComplete = new Promise<void>((resolve) => {
185
+ resolveStreamComplete = resolve;
186
+ });
187
+
188
+ // Get intercept source URL if in intercept context
189
+ const interceptSourceUrl = store.getInterceptSourceUrl();
190
+
191
+ // Track streaming token - will be set when response arrives
192
+ let streamingToken: { end(): void } | null = null;
193
+
194
+ // Use a dedicated abort controller for the fetch so we can cancel network
195
+ // I/O without disrupting the Flight stream once the response has arrived.
196
+ // Aborting a response mid-stream causes React's Flight decoder to throw
197
+ // asynchronous unhandled errors (BodyStreamBuffer was aborted).
198
+ const fetchAbort = new AbortController();
199
+ const onHandleAbort = () => fetchAbort.abort();
200
+ handle.signal.addEventListener("abort", onHandleAbort, { once: true });
201
+
202
+ // Send action request with stream tracking
203
+ const responsePromise = fetch(url, {
204
+ method: "POST",
205
+ headers: {
206
+ "rsc-action": id,
207
+ "X-RSC-Router-Client-Path": segmentState.currentUrl,
208
+ ...(tx && { "X-RSC-Router-Request-Id": tx.requestId }),
209
+ // Send intercept source URL so server can maintain intercept context
210
+ ...(interceptSourceUrl && {
211
+ "X-RSC-Router-Intercept-Source": interceptSourceUrl,
212
+ }),
213
+ },
214
+ body: encodedBody,
215
+ signal: fetchAbort.signal,
216
+ }).then(async (response) => {
217
+ // Response arrived — disconnect fetch abort from handle abort so
218
+ // abortAllActions() doesn't disrupt the in-progress Flight stream.
219
+ handle.signal.removeEventListener("abort", onHandleAbort);
220
+
221
+ // Check for version mismatch - server wants us to reload
222
+ const reload = extractRscHeaderUrl(response, "X-RSC-Reload");
223
+ if (reload === "blocked") {
224
+ resolveStreamComplete();
225
+ return emptyResponse();
226
+ }
227
+ if (reload) {
228
+ log("version mismatch on action, reloading", {
229
+ reloadUrl: reload.url,
230
+ });
231
+ window.location.href = reload.url;
232
+ return new Promise<Response>(() => {});
233
+ }
234
+
235
+ // Simple redirect from action (no state, no RSC payload).
236
+ // Short-circuits before createFromFetch — no Flight deserialization needed.
237
+ // Check handle.signal.aborted to avoid redirecting from a stale action
238
+ // when the user has already navigated away.
239
+ const redirect = extractRscHeaderUrl(response, "X-RSC-Redirect");
240
+ if (redirect && redirect !== "blocked" && !handle.signal.aborted) {
241
+ log("action simple redirect", { url: redirect.url });
242
+ handle.complete(undefined);
243
+ if (onNavigate) {
244
+ await onNavigate(redirect.url, {
245
+ replace: true,
246
+ _skipCache: true,
247
+ });
248
+ } else {
249
+ window.location.href = redirect.url;
250
+ }
251
+ return new Promise<Response>(() => {});
252
+ }
253
+ if (redirect === "blocked") {
254
+ resolveStreamComplete();
255
+ return emptyResponse();
256
+ }
257
+
258
+ // Start streaming immediately when response arrives
259
+ if (!handle.signal.aborted) {
260
+ streamingToken = handle.startStreaming();
261
+ }
262
+
263
+ return teeWithCompletion(response, () => {
264
+ log("stream complete");
265
+ streamingToken?.end();
266
+ resolveStreamComplete();
267
+ });
268
+ });
269
+
270
+ // Deserialize response (MUST use same temporaryReferences)
271
+ let payload: RscPayload;
272
+ try {
273
+ payload = await deps.createFromFetch<RscPayload>(responsePromise, {
274
+ temporaryReferences,
275
+ });
276
+ } catch (error) {
277
+ // Clean up streaming token on error (may be null if fetch failed before .then() ran)
278
+ // The token is assigned in .then() callback which runs before this catch block,
279
+ // but TypeScript doesn't track cross-async assignments, so use type assertion
280
+ (streamingToken as { end(): void } | null)?.end();
281
+ // resolveStreamComplete is assigned in the Promise constructor so it's safe to call
282
+ resolveStreamComplete!();
283
+
284
+ // Silently swallow abort errors — the action was intentionally cancelled
285
+ // (e.g., user navigated away or abortAllActions was called).
286
+ // Return undefined instead of throwing to avoid surfacing as a page error.
287
+ // Check both DOMException AbortError and stream-level abort messages
288
+ // (BodyStreamBuffer was aborted) that propagate from the aborted fetch.
289
+ if (handle.signal.aborted) {
290
+ return undefined;
291
+ }
292
+
293
+ // Convert network-level errors to NetworkError for proper handling
294
+ const networkError = toNetworkError(error, {
295
+ url: url.toString(),
296
+ operation: "action",
297
+ });
298
+ if (networkError) {
299
+ handle.fail(networkError);
300
+ emitNetworkError(onUpdate, networkError, segmentState.currentUrl);
301
+ throw networkError;
302
+ }
303
+ throw error;
304
+ }
305
+
306
+ log("action response received", {
307
+ isPartial: payload.metadata?.isPartial,
308
+ isError: payload.metadata?.isError,
309
+ matchedCount: payload.metadata?.matched?.length ?? 0,
310
+ diffCount: payload.metadata?.diff?.length ?? 0,
311
+ });
312
+
313
+ // Guard: if the action was aborted while streaming (e.g., user navigated
314
+ // away or abortAllActions fired), bail out before any reconcile/render/cache
315
+ // writes to avoid overwriting the current UI with stale action results.
316
+ if (handle.signal.aborted) {
317
+ log("action aborted after response, skipping reconciliation");
318
+ return undefined;
319
+ }
320
+
321
+ // Process response
322
+ const { metadata, returnValue } = payload;
323
+
324
+ // Handle action redirect: server converted the redirect to a Flight payload
325
+ // so we can perform SPA navigation instead of a full page reload.
326
+ // Check handle.signal.aborted to avoid redirecting from a stale action
327
+ // when the user has already navigated away.
328
+ if (metadata?.redirect && !handle.signal.aborted) {
329
+ const redirectUrl = validateRedirectOrigin(
330
+ metadata.redirect.url,
331
+ window.location.origin,
332
+ );
333
+ if (!redirectUrl) {
334
+ log("blocked action redirect payload", {
335
+ url: metadata.redirect.url,
336
+ });
337
+ handle.complete(returnValue?.data);
338
+ return returnValue?.data;
339
+ }
340
+ const redirectState = metadata.locationState;
341
+ log("action redirect", { url: redirectUrl });
342
+ handle.complete(returnValue?.data);
343
+ if (onNavigate) {
344
+ await onNavigate(redirectUrl, {
345
+ state: redirectState,
346
+ replace: true,
347
+ _skipCache: true,
348
+ });
349
+ } else {
350
+ window.location.href = redirectUrl;
351
+ }
352
+ return returnValue?.data;
353
+ }
354
+
355
+ // Bail out if the action was aborted after deserialization (e.g. user
356
+ // navigated away or abortAllActions was called while the Flight stream
357
+ // was being consumed). Without this check the code below would mutate
358
+ // the store / UI for a stale action.
359
+ if (handle.signal.aborted) {
360
+ log("action aborted after deserialization, skipping mutations");
361
+ return returnValue?.data;
362
+ }
363
+
364
+ const { matched, diff, segments, isPartial, isError } = metadata || {};
365
+
366
+ // Log action result
367
+ if (returnValue && !returnValue.ok) {
368
+ console.error(`[Browser] Action failed:`, returnValue.data);
369
+ }
370
+
371
+ // Handle error responses with error boundary UI
372
+ if (isError && isPartial && segments && diff) {
373
+ log("processing error boundary response");
374
+
375
+ // Fail current handle BEFORE aborting all actions so the event controller
376
+ // records the error state (abortAllActions clears inflight entries)
377
+ if (returnValue && !returnValue.ok) {
378
+ handle.fail(returnValue.data);
379
+ }
380
+
381
+ // Abort all other pending action requests - error takes precedence
382
+ // This prevents other actions from completing and overwriting the error UI
383
+ eventController.abortAllActions();
384
+
385
+ // Clear concurrent action tracking - no consolidation needed when showing error
386
+ handle.clearConsolidation();
387
+
388
+ // Get current page's cached segments
389
+ const currentKey = store.getHistoryKey();
390
+ const cached = store.getCachedSegments(currentKey);
391
+ const cachedSegments = cached?.segments || [];
392
+
393
+ // Reconcile error segments with cached tree
394
+ const errorResult = reconcileErrorSegments(cachedSegments, segments);
395
+
396
+ // Render the full tree with error segment merged with parent layouts
397
+ const errorTree = await renderSegments(errorResult.mainSegments, {
398
+ isAction: true,
399
+ interceptSegments:
400
+ errorResult.interceptSegments.length > 0
401
+ ? errorResult.interceptSegments
402
+ : undefined,
403
+ });
404
+
405
+ // Re-check route stability after async renderSegments — user may have
406
+ // navigated away while the error tree was being prepared.
407
+ if (window.location.pathname !== actionStartPathname) {
408
+ log("user navigated during error render, skipping");
409
+ if (returnValue && !returnValue.ok) {
410
+ throw returnValue.data;
411
+ }
412
+ handle.complete(undefined);
413
+ return undefined;
414
+ }
415
+ const currentKeyNow = store.getHistoryKey();
416
+ if (currentKeyNow !== currentKey) {
417
+ log("history key changed during error render, skipping cache update");
418
+ if (returnValue && !returnValue.ok) {
419
+ throw returnValue.data;
420
+ }
421
+ handle.complete(undefined);
422
+ return undefined;
423
+ }
424
+
425
+ // Update UI with error boundary
426
+ startTransition(() => {
427
+ onUpdate({ root: errorTree, metadata: metadata! });
428
+ });
429
+
430
+ // Update segment tracking to exclude error segment IDs
431
+ const errorSegmentIds = new Set(diff);
432
+ const segmentIdsAfterError = segmentState.currentSegmentIds.filter(
433
+ (id) => !errorSegmentIds.has(id),
434
+ );
435
+
436
+ // Update store state
437
+ store.setSegmentIds(segmentIdsAfterError);
438
+ const currentHandleData = eventController.getHandleState().data;
439
+ store.cacheSegmentsForHistory(
440
+ currentKey,
441
+ errorResult.segments,
442
+ currentHandleData,
443
+ );
444
+
445
+ // Throw the error so the action promise rejects
446
+ if (returnValue && !returnValue.ok) {
447
+ throw returnValue.data;
448
+ }
449
+
450
+ // No error in returnValue (shouldn't happen with isError: true)
451
+ handle.complete(undefined);
452
+ return undefined;
453
+ }
454
+
455
+ if (!isPartial) {
456
+ // Protocol invariant: action revalidation responses MUST be partial.
457
+ // The server always sends isPartial: true for successful revalidation
458
+ // and isPartial: true + isError: true for error boundary responses.
459
+ // A non-partial payload here indicates a server-side bug.
460
+ throw new Error(
461
+ `[Browser] Action response missing isPartial — the server must ` +
462
+ `always send partial payloads for action revalidation.`,
463
+ );
464
+ }
465
+
466
+ log("processing partial update", {
467
+ serverSegments: segments?.length ?? 0,
468
+ diff: diff?.join(", ") ?? "",
469
+ matched: matched?.join(", ") ?? "",
470
+ });
471
+
472
+ // Record revalidated segments for concurrent action tracking
473
+ if (diff) {
474
+ handle.recordRevalidatedSegments(diff);
475
+ }
476
+
477
+ // Get current page's cached segments for merging
478
+ const currentKey = store.getHistoryKey();
479
+ const cached = store.getCachedSegments(currentKey);
480
+ const cachedSegments = cached?.segments || [];
481
+
482
+ if (!matched) {
483
+ throw new Error("No matched segments in response");
484
+ }
485
+
486
+ // Reconcile server segments with cached segments (single source of truth)
487
+ const reconciled = reconcileSegments({
488
+ actor: "action",
489
+ matched,
490
+ diff: diff || [],
491
+ serverSegments: segments || [],
492
+ cachedSegments,
493
+ });
494
+ const fullSegments = reconciled.segments;
495
+
496
+ const returnData = returnValue?.data;
497
+
498
+ if (returnValue && !returnValue.ok) {
499
+ handle.fail(returnValue.data);
500
+ throw returnValue.data;
501
+ }
502
+
503
+ // Classify the post-reconciliation scenario
504
+ const scenario = classifyActionOutcome({
505
+ handleId: handle.id,
506
+ inflightActions: eventController.getInflightActions(),
507
+ hadAnyConcurrentActions: eventController.hadAnyConcurrentActions(),
508
+ revalidatedSegments: handle.getRevalidatedSegments(),
509
+ actionStartPathname,
510
+ currentPathname: window.location.pathname,
511
+ actionStartLocationKey: locationKey,
512
+ currentLocationKey: window.history.state?.key,
513
+ reconciledSegmentCount: fullSegments.length,
514
+ matchedCount: matched.length,
515
+ currentInterceptSource: store.getInterceptSourceUrl(),
516
+ });
517
+
518
+ switch (scenario.type) {
519
+ case "navigated-away": {
520
+ log("user navigated away during action", {
521
+ from: actionStartPathname,
522
+ to: window.location.pathname,
523
+ historyKeyChanged: scenario.historyKeyChanged,
524
+ });
525
+ // Clear concurrent action tracking - don't consolidate for old route's segments
526
+ handle.clearConsolidation();
527
+
528
+ if (scenario.historyKeyChanged) {
529
+ if (!scenario.onInterceptRoute) {
530
+ store.markCacheAsStaleAndBroadcast();
531
+ refetchRoute().catch((error) => {
532
+ if (isBackgroundSuppressible(error)) return;
533
+ console.error(
534
+ "[Browser] Background revalidation failed:",
535
+ error,
536
+ );
537
+ });
538
+ }
539
+ break;
540
+ }
541
+
542
+ // Same history key but different pathname - safe to refetch current route
543
+ store.markCacheAsStaleAndBroadcast();
544
+ await refetchRoute({
545
+ interceptSourceUrl: store.getInterceptSourceUrl(),
546
+ });
547
+ break;
548
+ }
549
+
550
+ case "hmr-missing": {
551
+ console.warn(
552
+ `[Browser] Missing segments after action (HMR detected), refetching...`,
553
+ );
554
+ await refetchRoute({ interceptSourceUrl });
555
+ store.broadcastCacheInvalidation();
556
+ break;
557
+ }
558
+
559
+ case "consolidation-needed": {
560
+ log("consolidation fetch needed", {
561
+ segmentIds: scenario.segmentIds,
562
+ });
563
+ // Calculate segments to send (exclude the ones we want fresh)
564
+ const currentSegmentIds = store.getSegmentState().currentSegmentIds;
565
+ const segmentsToSend = currentSegmentIds.filter(
566
+ (sid) => !scenario.segmentIds.includes(sid),
567
+ );
568
+
569
+ // Clear consolidation tracking before fetch
570
+ handle.clearConsolidation();
571
+
572
+ await refetchRoute({
573
+ segments: segmentsToSend,
574
+ interceptSourceUrl,
575
+ });
576
+ store.broadcastCacheInvalidation();
577
+ break;
578
+ }
579
+
580
+ case "concurrent-skip": {
581
+ log("skipping UI update, other actions fetching", {
582
+ otherCount: scenario.otherFetchingCount,
583
+ });
584
+ // Only update store if history key hasn't changed (user didn't navigate away)
585
+ const currentKeyNow = store.getHistoryKey();
586
+ if (currentKeyNow === currentKey) {
587
+ store.setSegmentIds(matched);
588
+ const currentHandleData = eventController.getHandleState().data;
589
+ store.cacheSegmentsForHistory(
590
+ currentKey,
591
+ fullSegments,
592
+ currentHandleData,
593
+ );
594
+ }
595
+ break;
596
+ }
597
+
598
+ case "normal": {
599
+ // Prepare new tree (await loader data resolution)
600
+ const newTree = await renderSegments(reconciled.mainSegments, {
601
+ isAction: true,
602
+ interceptSegments:
603
+ reconciled.interceptSegments.length > 0
604
+ ? reconciled.interceptSegments
605
+ : undefined,
606
+ });
607
+
608
+ // Re-check if user navigated away (could happen during async renderSegments)
609
+ if (window.location.pathname !== actionStartPathname) {
610
+ log("user navigated during render, skipping");
611
+ break;
612
+ }
613
+
614
+ // Verify the store's current key still matches what we captured at action start
615
+ // If they differ, user navigated away and we should NOT cache under the old key
616
+ const currentKeyNow = store.getHistoryKey();
617
+ if (currentKeyNow !== currentKey) {
618
+ log("history key changed during action, skipping cache update");
619
+ break;
620
+ }
621
+
622
+ startTransition(() => {
623
+ onUpdate({ root: newTree, metadata: metadata! });
624
+ });
625
+
626
+ // Apply server-set location state to history.state (non-redirect flow)
627
+ const actionLocationState = metadata?.locationState;
628
+ if (actionLocationState) {
629
+ mergeLocationState(actionLocationState);
630
+ }
631
+
632
+ // Update store state
633
+ store.setSegmentIds(matched);
634
+ const currentHandleData = eventController.getHandleState().data;
635
+ store.cacheSegmentsForHistory(
636
+ currentKey,
637
+ fullSegments,
638
+ currentHandleData,
639
+ );
640
+ store.markCacheAsStaleAndBroadcast();
641
+ break;
642
+ }
643
+ }
644
+
645
+ handle.complete(returnData);
646
+ return returnData;
647
+ } finally {
648
+ handle[Symbol.dispose]();
649
+ }
650
+ }
651
+
652
+ return {
653
+ /**
654
+ * Register the server action callback with the RSC runtime
655
+ */
656
+ register(): void {
657
+ if (isRegistered) {
658
+ console.warn("[Browser] Server action bridge already registered");
659
+ return;
660
+ }
661
+ deps.setServerCallback(handleServerAction);
662
+ isRegistered = true;
663
+ },
664
+ };
665
+ }
666
+
667
+ export { createServerActionBridge as default };
@@ -0,0 +1,40 @@
1
+ /**
2
+ * Shallow comparison utility for selector optimization
3
+ *
4
+ * Used by useNavigation hook to prevent unnecessary re-renders
5
+ * when the selected value hasn't changed.
6
+ *
7
+ * @param a - First value
8
+ * @param b - Second value
9
+ * @returns true if values are shallowly equal
10
+ */
11
+ export function shallow<T>(a: T, b: T): boolean {
12
+ // Same reference or primitive equality
13
+ if (Object.is(a, b)) return true;
14
+
15
+ // Different types or non-objects
16
+ if (typeof a !== "object" || typeof b !== "object") return false;
17
+
18
+ // Null checks
19
+ if (a === null || b === null) return false;
20
+
21
+ // Compare object keys
22
+ const keysA = Object.keys(a);
23
+ const keysB = Object.keys(b);
24
+
25
+ if (keysA.length !== keysB.length) return false;
26
+
27
+ // Check each key's value with Object.is
28
+ for (const key of keysA) {
29
+ if (
30
+ !Object.is(
31
+ (a as Record<string, unknown>)[key],
32
+ (b as Record<string, unknown>)[key],
33
+ )
34
+ ) {
35
+ return false;
36
+ }
37
+ }
38
+
39
+ return true;
40
+ }