@rangojs/router 0.0.0-experimental.13 → 0.0.0-experimental.13221847

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 (298) hide show
  1. package/AGENTS.md +9 -0
  2. package/README.md +884 -4
  3. package/dist/bin/rango.js +1531 -212
  4. package/dist/vite/index.js +3995 -2489
  5. package/package.json +57 -52
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +262 -0
  8. package/skills/caching/SKILL.md +85 -23
  9. package/skills/composability/SKILL.md +172 -0
  10. package/skills/debug-manifest/SKILL.md +12 -8
  11. package/skills/document-cache/SKILL.md +18 -16
  12. package/skills/fonts/SKILL.md +6 -4
  13. package/skills/hooks/SKILL.md +328 -70
  14. package/skills/host-router/SKILL.md +218 -0
  15. package/skills/intercept/SKILL.md +131 -8
  16. package/skills/layout/SKILL.md +100 -3
  17. package/skills/links/SKILL.md +62 -15
  18. package/skills/loader/SKILL.md +368 -42
  19. package/skills/middleware/SKILL.md +171 -34
  20. package/skills/mime-routes/SKILL.md +14 -10
  21. package/skills/parallel/SKILL.md +137 -1
  22. package/skills/prerender/SKILL.md +366 -28
  23. package/skills/rango/SKILL.md +85 -21
  24. package/skills/response-routes/SKILL.md +136 -83
  25. package/skills/route/SKILL.md +195 -21
  26. package/skills/router-setup/SKILL.md +123 -30
  27. package/skills/theme/SKILL.md +9 -8
  28. package/skills/typesafety/SKILL.md +240 -102
  29. package/skills/use-cache/SKILL.md +324 -0
  30. package/src/__internal.ts +102 -4
  31. package/src/bin/rango.ts +312 -15
  32. package/src/browser/action-coordinator.ts +97 -0
  33. package/src/browser/action-response-classifier.ts +99 -0
  34. package/src/browser/event-controller.ts +92 -64
  35. package/src/browser/history-state.ts +80 -0
  36. package/src/browser/intercept-utils.ts +52 -0
  37. package/src/browser/link-interceptor.ts +24 -4
  38. package/src/browser/logging.ts +11 -0
  39. package/src/browser/merge-segment-loaders.ts +20 -12
  40. package/src/browser/navigation-bridge.ts +266 -558
  41. package/src/browser/navigation-client.ts +132 -75
  42. package/src/browser/navigation-store.ts +33 -50
  43. package/src/browser/navigation-transaction.ts +297 -0
  44. package/src/browser/network-error-handler.ts +61 -0
  45. package/src/browser/partial-update.ts +303 -309
  46. package/src/browser/prefetch/cache.ts +206 -0
  47. package/src/browser/prefetch/fetch.ts +144 -0
  48. package/src/browser/prefetch/observer.ts +65 -0
  49. package/src/browser/prefetch/policy.ts +48 -0
  50. package/src/browser/prefetch/queue.ts +128 -0
  51. package/src/browser/rango-state.ts +112 -0
  52. package/src/browser/react/Link.tsx +190 -70
  53. package/src/browser/react/NavigationProvider.tsx +78 -11
  54. package/src/browser/react/context.ts +6 -0
  55. package/src/browser/react/filter-segment-order.ts +11 -0
  56. package/src/browser/react/index.ts +12 -12
  57. package/src/browser/react/location-state-shared.ts +95 -53
  58. package/src/browser/react/location-state.ts +60 -15
  59. package/src/browser/react/mount-context.ts +6 -1
  60. package/src/browser/react/nonce-context.ts +23 -0
  61. package/src/browser/react/shallow-equal.ts +27 -0
  62. package/src/browser/react/use-action.ts +29 -51
  63. package/src/browser/react/use-client-cache.ts +5 -3
  64. package/src/browser/react/use-handle.ts +29 -70
  65. package/src/browser/react/use-link-status.ts +6 -5
  66. package/src/browser/react/use-navigation.ts +22 -63
  67. package/src/browser/react/use-params.ts +65 -0
  68. package/src/browser/react/use-pathname.ts +47 -0
  69. package/src/browser/react/use-router.ts +63 -0
  70. package/src/browser/react/use-search-params.ts +56 -0
  71. package/src/browser/react/use-segments.ts +80 -97
  72. package/src/browser/response-adapter.ts +73 -0
  73. package/src/browser/rsc-router.tsx +188 -57
  74. package/src/browser/scroll-restoration.ts +117 -44
  75. package/src/browser/segment-reconciler.ts +221 -0
  76. package/src/browser/segment-structure-assert.ts +16 -0
  77. package/src/browser/server-action-bridge.ts +488 -606
  78. package/src/browser/shallow.ts +6 -1
  79. package/src/browser/types.ts +116 -47
  80. package/src/browser/validate-redirect-origin.ts +29 -0
  81. package/src/build/generate-manifest.ts +63 -21
  82. package/src/build/generate-route-types.ts +36 -1038
  83. package/src/build/index.ts +2 -5
  84. package/src/build/route-trie.ts +38 -12
  85. package/src/build/route-types/ast-helpers.ts +25 -0
  86. package/src/build/route-types/ast-route-extraction.ts +98 -0
  87. package/src/build/route-types/codegen.ts +102 -0
  88. package/src/build/route-types/include-resolution.ts +411 -0
  89. package/src/build/route-types/param-extraction.ts +48 -0
  90. package/src/build/route-types/per-module-writer.ts +128 -0
  91. package/src/build/route-types/router-processing.ts +479 -0
  92. package/src/build/route-types/scan-filter.ts +78 -0
  93. package/src/build/runtime-discovery.ts +231 -0
  94. package/src/cache/background-task.ts +34 -0
  95. package/src/cache/cache-key-utils.ts +44 -0
  96. package/src/cache/cache-policy.ts +125 -0
  97. package/src/cache/cache-runtime.ts +342 -0
  98. package/src/cache/cache-scope.ts +122 -303
  99. package/src/cache/cf/cf-cache-store.ts +571 -17
  100. package/src/cache/cf/index.ts +13 -3
  101. package/src/cache/document-cache.ts +116 -77
  102. package/src/cache/handle-capture.ts +81 -0
  103. package/src/cache/handle-snapshot.ts +41 -0
  104. package/src/cache/index.ts +1 -15
  105. package/src/cache/memory-segment-store.ts +191 -13
  106. package/src/cache/profile-registry.ts +73 -0
  107. package/src/cache/read-through-swr.ts +134 -0
  108. package/src/cache/segment-codec.ts +256 -0
  109. package/src/cache/taint.ts +98 -0
  110. package/src/cache/types.ts +72 -122
  111. package/src/client.rsc.tsx +3 -1
  112. package/src/client.tsx +84 -126
  113. package/src/component-utils.ts +4 -4
  114. package/src/components/DefaultDocument.tsx +5 -1
  115. package/src/context-var.ts +86 -0
  116. package/src/debug.ts +19 -9
  117. package/src/errors.ts +77 -7
  118. package/src/handle.ts +12 -7
  119. package/src/handles/MetaTags.tsx +73 -20
  120. package/src/handles/breadcrumbs.ts +66 -0
  121. package/src/handles/index.ts +1 -0
  122. package/src/handles/meta.ts +30 -13
  123. package/src/host/cookie-handler.ts +21 -15
  124. package/src/host/errors.ts +8 -8
  125. package/src/host/index.ts +4 -7
  126. package/src/host/pattern-matcher.ts +27 -27
  127. package/src/host/router.ts +61 -39
  128. package/src/host/testing.ts +8 -8
  129. package/src/host/types.ts +15 -7
  130. package/src/host/utils.ts +1 -1
  131. package/src/href-client.ts +65 -45
  132. package/src/index.rsc.ts +104 -40
  133. package/src/index.ts +122 -67
  134. package/src/internal-debug.ts +9 -3
  135. package/src/loader.rsc.ts +18 -93
  136. package/src/loader.ts +26 -9
  137. package/src/network-error-thrower.tsx +3 -1
  138. package/src/outlet-provider.tsx +45 -0
  139. package/src/prerender/param-hash.ts +4 -2
  140. package/src/prerender/store.ts +121 -17
  141. package/src/prerender.ts +325 -20
  142. package/src/reverse.ts +144 -124
  143. package/src/root-error-boundary.tsx +41 -29
  144. package/src/route-content-wrapper.tsx +7 -4
  145. package/src/route-definition/dsl-helpers.ts +959 -0
  146. package/src/route-definition/helper-factories.ts +200 -0
  147. package/src/route-definition/helpers-types.ts +430 -0
  148. package/src/route-definition/index.ts +52 -0
  149. package/src/route-definition/redirect.ts +93 -0
  150. package/src/route-definition.ts +1 -1450
  151. package/src/route-map-builder.ts +87 -133
  152. package/src/route-name.ts +53 -0
  153. package/src/route-types.ts +41 -6
  154. package/src/router/content-negotiation.ts +116 -0
  155. package/src/router/debug-manifest.ts +72 -0
  156. package/src/router/error-handling.ts +9 -9
  157. package/src/router/find-match.ts +160 -0
  158. package/src/router/handler-context.ts +324 -116
  159. package/src/router/intercept-resolution.ts +11 -4
  160. package/src/router/lazy-includes.ts +237 -0
  161. package/src/router/loader-resolution.ts +179 -133
  162. package/src/router/logging.ts +112 -6
  163. package/src/router/manifest.ts +58 -19
  164. package/src/router/match-api.ts +89 -88
  165. package/src/router/match-context.ts +4 -2
  166. package/src/router/match-handlers.ts +440 -0
  167. package/src/router/match-middleware/background-revalidation.ts +86 -89
  168. package/src/router/match-middleware/cache-lookup.ts +295 -49
  169. package/src/router/match-middleware/cache-store.ts +56 -13
  170. package/src/router/match-middleware/intercept-resolution.ts +45 -22
  171. package/src/router/match-middleware/segment-resolution.ts +20 -9
  172. package/src/router/match-pipelines.ts +10 -45
  173. package/src/router/match-result.ts +44 -21
  174. package/src/router/metrics.ts +240 -15
  175. package/src/router/middleware-cookies.ts +55 -0
  176. package/src/router/middleware-types.ts +222 -0
  177. package/src/router/middleware.ts +327 -369
  178. package/src/router/pattern-matching.ts +169 -31
  179. package/src/router/prerender-match.ts +402 -0
  180. package/src/router/preview-match.ts +170 -0
  181. package/src/router/revalidation.ts +105 -14
  182. package/src/router/router-context.ts +40 -21
  183. package/src/router/router-interfaces.ts +452 -0
  184. package/src/router/router-options.ts +592 -0
  185. package/src/router/router-registry.ts +24 -0
  186. package/src/router/segment-resolution/fresh.ts +677 -0
  187. package/src/router/segment-resolution/helpers.ts +263 -0
  188. package/src/router/segment-resolution/loader-cache.ts +199 -0
  189. package/src/router/segment-resolution/revalidation.ts +1296 -0
  190. package/src/router/segment-resolution/static-store.ts +67 -0
  191. package/src/router/segment-resolution.ts +21 -1354
  192. package/src/router/segment-wrappers.ts +291 -0
  193. package/src/router/telemetry-otel.ts +299 -0
  194. package/src/router/telemetry.ts +300 -0
  195. package/src/router/timeout.ts +148 -0
  196. package/src/router/trie-matching.ts +96 -29
  197. package/src/router/types.ts +15 -9
  198. package/src/router.ts +642 -2366
  199. package/src/rsc/handler-context.ts +45 -0
  200. package/src/rsc/handler.ts +639 -1027
  201. package/src/rsc/helpers.ts +140 -6
  202. package/src/rsc/index.ts +0 -20
  203. package/src/rsc/loader-fetch.ts +209 -0
  204. package/src/rsc/manifest-init.ts +86 -0
  205. package/src/rsc/nonce.ts +14 -0
  206. package/src/rsc/origin-guard.ts +141 -0
  207. package/src/rsc/progressive-enhancement.ts +379 -0
  208. package/src/rsc/response-error.ts +37 -0
  209. package/src/rsc/response-route-handler.ts +347 -0
  210. package/src/rsc/rsc-rendering.ts +237 -0
  211. package/src/rsc/runtime-warnings.ts +42 -0
  212. package/src/rsc/server-action.ts +348 -0
  213. package/src/rsc/ssr-setup.ts +128 -0
  214. package/src/rsc/types.ts +38 -11
  215. package/src/search-params.ts +66 -54
  216. package/src/segment-system.tsx +165 -17
  217. package/src/server/context.ts +237 -54
  218. package/src/server/cookie-store.ts +190 -0
  219. package/src/server/fetchable-loader-store.ts +11 -6
  220. package/src/server/handle-store.ts +94 -15
  221. package/src/server/loader-registry.ts +15 -56
  222. package/src/server/request-context.ts +438 -71
  223. package/src/server.ts +26 -164
  224. package/src/ssr/index.tsx +101 -31
  225. package/src/static-handler.ts +22 -4
  226. package/src/theme/ThemeProvider.tsx +21 -15
  227. package/src/theme/ThemeScript.tsx +5 -5
  228. package/src/theme/constants.ts +5 -2
  229. package/src/theme/index.ts +4 -14
  230. package/src/theme/theme-context.ts +4 -30
  231. package/src/theme/theme-script.ts +21 -18
  232. package/src/types/boundaries.ts +158 -0
  233. package/src/types/cache-types.ts +198 -0
  234. package/src/types/error-types.ts +192 -0
  235. package/src/types/global-namespace.ts +100 -0
  236. package/src/types/handler-context.ts +773 -0
  237. package/src/types/index.ts +88 -0
  238. package/src/types/loader-types.ts +183 -0
  239. package/src/types/route-config.ts +170 -0
  240. package/src/types/route-entry.ts +109 -0
  241. package/src/types/segments.ts +150 -0
  242. package/src/types.ts +1 -1795
  243. package/src/urls/include-helper.ts +197 -0
  244. package/src/urls/index.ts +53 -0
  245. package/src/urls/path-helper-types.ts +339 -0
  246. package/src/urls/path-helper.ts +329 -0
  247. package/src/urls/pattern-types.ts +95 -0
  248. package/src/urls/response-types.ts +106 -0
  249. package/src/urls/type-extraction.ts +372 -0
  250. package/src/urls/urls-function.ts +98 -0
  251. package/src/urls.ts +1 -1323
  252. package/src/use-loader.tsx +85 -77
  253. package/src/vite/discovery/bundle-postprocess.ts +184 -0
  254. package/src/vite/discovery/discover-routers.ts +344 -0
  255. package/src/vite/discovery/prerender-collection.ts +385 -0
  256. package/src/vite/discovery/route-types-writer.ts +258 -0
  257. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  258. package/src/vite/discovery/state.ts +108 -0
  259. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  260. package/src/vite/index.ts +11 -2259
  261. package/src/vite/plugin-types.ts +48 -0
  262. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  263. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  264. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  265. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -47
  266. package/src/vite/{expose-id-utils.ts → plugins/expose-id-utils.ts} +8 -43
  267. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  268. package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
  269. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  270. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  271. package/src/vite/plugins/expose-ids/types.ts +45 -0
  272. package/src/vite/plugins/expose-internal-ids.ts +569 -0
  273. package/src/vite/plugins/refresh-cmd.ts +65 -0
  274. package/src/vite/plugins/use-cache-transform.ts +323 -0
  275. package/src/vite/plugins/version-injector.ts +83 -0
  276. package/src/vite/plugins/version-plugin.ts +266 -0
  277. package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
  278. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  279. package/src/vite/rango.ts +445 -0
  280. package/src/vite/router-discovery.ts +777 -0
  281. package/src/vite/{ast-handler-extract.ts → utils/ast-handler-extract.ts} +181 -9
  282. package/src/vite/utils/banner.ts +36 -0
  283. package/src/vite/utils/bundle-analysis.ts +137 -0
  284. package/src/vite/utils/manifest-utils.ts +70 -0
  285. package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
  286. package/src/vite/utils/prerender-utils.ts +189 -0
  287. package/src/vite/utils/shared-utils.ts +169 -0
  288. package/CLAUDE.md +0 -43
  289. package/dist/vite/index.named-routes.gen.ts +0 -103
  290. package/src/browser/lru-cache.ts +0 -69
  291. package/src/browser/request-controller.ts +0 -164
  292. package/src/cache/memory-store.ts +0 -253
  293. package/src/href-context.ts +0 -33
  294. package/src/router.gen.ts +0 -6
  295. package/src/static-handler.gen.ts +0 -5
  296. package/src/urls.gen.ts +0 -8
  297. package/src/vite/expose-internal-ids.ts +0 -1167
  298. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -0,0 +1,677 @@
1
+ /**
2
+ * Fresh Path Segment Resolution
3
+ *
4
+ * Functions for resolving segments during a full (non-revalidation) request.
5
+ * Handles loaders, layouts, routes, parallels, orphan layouts, and error boundaries.
6
+ */
7
+
8
+ import type { ReactNode } from "react";
9
+ import { invariant } from "../../errors";
10
+ import {
11
+ getParallelEntries,
12
+ getParallelSlotEntries,
13
+ type EntryData,
14
+ } from "../../server/context";
15
+ import type {
16
+ HandlerContext,
17
+ InternalHandlerContext,
18
+ ResolvedSegment,
19
+ } from "../../types";
20
+ import type { SegmentResolutionDeps } from "../types.js";
21
+ import { resolveLoaderData } from "./loader-cache.js";
22
+ import { _getRequestContext } from "../../server/request-context.js";
23
+ import { appendMetric } from "../metrics.js";
24
+ import {
25
+ handleHandlerResult,
26
+ tryStaticHandler,
27
+ tryStaticSlot,
28
+ resolveLayoutComponent,
29
+ resolveWithErrorBoundary,
30
+ } from "./helpers.js";
31
+ import { getRouterContext } from "../router-context.js";
32
+ import { resolveSink, safeEmit } from "../telemetry.js";
33
+ import { track } from "../../server/context.js";
34
+
35
+ // ---------------------------------------------------------------------------
36
+ // Streamed handler telemetry
37
+ // ---------------------------------------------------------------------------
38
+
39
+ /**
40
+ * Attach a fire-and-forget rejection observer to a streamed handler promise.
41
+ * React catches the actual error via its error boundary; this only emits
42
+ * the handler.error telemetry event.
43
+ */
44
+ function observeStreamedHandler(
45
+ promise: Promise<ReactNode>,
46
+ segmentId: string,
47
+ segmentType: string,
48
+ pathname?: string,
49
+ routeKey?: string,
50
+ params?: Record<string, string>,
51
+ ): void {
52
+ let routerCtx;
53
+ try {
54
+ routerCtx = getRouterContext();
55
+ } catch {
56
+ return;
57
+ }
58
+ if (!routerCtx?.telemetry) return;
59
+ const sink = resolveSink(routerCtx.telemetry);
60
+ const reqId = routerCtx.requestId;
61
+ promise.catch((err: unknown) => {
62
+ const errorObj = err instanceof Error ? err : new Error(String(err));
63
+ safeEmit(sink, {
64
+ type: "handler.error",
65
+ timestamp: performance.now(),
66
+ requestId: reqId,
67
+ segmentId,
68
+ segmentType,
69
+ error: errorObj,
70
+ handledByBoundary: true,
71
+ pathname,
72
+ routeKey,
73
+ params,
74
+ });
75
+ });
76
+ }
77
+
78
+ // ---------------------------------------------------------------------------
79
+ // Fresh path (full match, no revalidation)
80
+ // ---------------------------------------------------------------------------
81
+
82
+ /**
83
+ * Resolve loaders for an entry and emit segments.
84
+ * Loaders are run lazily via ctx.use() and memoized for parallel execution.
85
+ */
86
+ export async function resolveLoaders<TEnv>(
87
+ entry: EntryData,
88
+ ctx: HandlerContext<any, TEnv>,
89
+ belongsToRoute: boolean,
90
+ deps: SegmentResolutionDeps<TEnv>,
91
+ shortCodeOverride?: string,
92
+ ): Promise<ResolvedSegment[]> {
93
+ const loaderEntries = entry.loader ?? [];
94
+ if (loaderEntries.length === 0) return [];
95
+
96
+ const shortCode = shortCodeOverride ?? entry.shortCode;
97
+ const hasLoading = "loading" in entry && entry.loading !== undefined;
98
+ const loadingDisabled = hasLoading && entry.loading === false;
99
+ const ms = _getRequestContext()?._metricsStore;
100
+
101
+ if (!loadingDisabled) {
102
+ // Streaming loaders: promises kick off now, settle during RSC serialization.
103
+ // No per-loader timing here — settlement happens asynchronously during
104
+ // RSC/SSR stream consumption, after the perf timeline is logged.
105
+ return loaderEntries.map((loaderEntry, i) => {
106
+ const { loader } = loaderEntry;
107
+ const segmentId = `${shortCode}D${i}.${loader.$$id}`;
108
+ return {
109
+ id: segmentId,
110
+ namespace: entry.id,
111
+ type: "loader" as const,
112
+ index: i,
113
+ component: null,
114
+ params: ctx.params,
115
+ loaderId: loader.$$id,
116
+ loaderData: deps.wrapLoaderPromise(
117
+ resolveLoaderData(loaderEntry, ctx, ctx.pathname),
118
+ entry,
119
+ segmentId,
120
+ ctx.pathname,
121
+ ),
122
+ belongsToRoute,
123
+ };
124
+ });
125
+ }
126
+
127
+ // Loading disabled: still start all loaders in parallel, but only emit
128
+ // settled promises so handlers don't stream loading placeholders.
129
+ // We can measure actual execution time here since we await all loaders.
130
+ const pendingLoaderData = loaderEntries.map((loaderEntry) => {
131
+ const start = performance.now();
132
+ const promise = resolveLoaderData(loaderEntry, ctx, ctx.pathname);
133
+ return { promise, start, loaderId: loaderEntry.loader.$$id };
134
+ });
135
+ await Promise.all(pendingLoaderData.map((p) => p.promise));
136
+
137
+ return loaderEntries.map((loaderEntry, i) => {
138
+ const { loader } = loaderEntry;
139
+ const segmentId = `${shortCode}D${i}.${loader.$$id}`;
140
+ const pending = pendingLoaderData[i]!;
141
+ if (ms && !ms.metrics.some((m) => m.label === `loader:${loader.$$id}`)) {
142
+ // All loaders ran in parallel via Promise.all — each span covers
143
+ // from its own kickoff to the batch settlement, giving a ceiling
144
+ // on that loader's contribution to the overall wait.
145
+ const batchEnd = performance.now();
146
+ appendMetric(
147
+ ms,
148
+ `loader:${loader.$$id}`,
149
+ pending.start,
150
+ batchEnd - pending.start,
151
+ 2,
152
+ );
153
+ }
154
+ return {
155
+ id: segmentId,
156
+ namespace: entry.id,
157
+ type: "loader" as const,
158
+ index: i,
159
+ component: null,
160
+ params: ctx.params,
161
+ loaderId: loader.$$id,
162
+ loaderData: deps.wrapLoaderPromise(
163
+ pending.promise,
164
+ entry,
165
+ segmentId,
166
+ ctx.pathname,
167
+ ),
168
+ belongsToRoute,
169
+ };
170
+ });
171
+ }
172
+
173
+ /**
174
+ * Options for segment resolution.
175
+ */
176
+ export interface ResolveSegmentOptions {
177
+ /** When true, skip resolveLoaders() calls (used for pre-rendering) */
178
+ skipLoaders?: boolean;
179
+ }
180
+
181
+ /**
182
+ * Resolve segments from EntryData.
183
+ * Executes middlewares, loaders, parallels, and handlers in correct order.
184
+ * Returns array: [main segment, ...orphan layout segments]
185
+ */
186
+ export async function resolveSegment<TEnv>(
187
+ entry: EntryData,
188
+ routeKey: string,
189
+ params: Record<string, string>,
190
+ context: HandlerContext<any, TEnv>,
191
+ loaderPromises: Map<string, Promise<any>>,
192
+ deps: SegmentResolutionDeps<TEnv>,
193
+ isRouteEntry: boolean = false,
194
+ options?: ResolveSegmentOptions,
195
+ ): Promise<ResolvedSegment[]> {
196
+ const segments: ResolvedSegment[] = [];
197
+
198
+ if (entry.type === "layout" || entry.type === "cache") {
199
+ if (!options?.skipLoaders) {
200
+ const loaderSegments = await resolveLoaders(entry, context, false, deps);
201
+ segments.push(...loaderSegments);
202
+ }
203
+
204
+ // Handler-first: layout handler executes before its parallels and orphan
205
+ // layouts so that ctx.set() values are visible to all children.
206
+ (context as InternalHandlerContext<any, TEnv>)._currentSegmentId =
207
+ entry.shortCode;
208
+
209
+ const doneLayoutHandler = track(`handler:${entry.id}`, 2);
210
+ const component = await resolveLayoutComponent(entry, context);
211
+ doneLayoutHandler();
212
+
213
+ segments.push({
214
+ id: entry.shortCode,
215
+ namespace: entry.id,
216
+ type: "layout",
217
+ index: 0,
218
+ component,
219
+ loading: entry.loading === false ? null : entry.loading,
220
+ transition: entry.transition,
221
+ params,
222
+ belongsToRoute: false,
223
+ layoutName: entry.id,
224
+ ...(entry.mountPath ? { mountPath: entry.mountPath } : {}),
225
+ });
226
+
227
+ const resolvedParallelEntries = new Set<string>();
228
+ for (const { slot, entry: parallelEntry } of getParallelSlotEntries(
229
+ entry.parallel,
230
+ )) {
231
+ const parallelSegments = await resolveParallelEntry(
232
+ parallelEntry,
233
+ params,
234
+ context,
235
+ false,
236
+ entry.shortCode,
237
+ deps,
238
+ options,
239
+ routeKey,
240
+ [slot],
241
+ !resolvedParallelEntries.has(parallelEntry.id),
242
+ );
243
+ segments.push(...parallelSegments);
244
+ resolvedParallelEntries.add(parallelEntry.id);
245
+ }
246
+
247
+ for (const orphan of entry.layout) {
248
+ const orphanSegments = await resolveOrphanLayout(
249
+ orphan,
250
+ params,
251
+ context,
252
+ loaderPromises,
253
+ false,
254
+ deps,
255
+ options,
256
+ routeKey,
257
+ );
258
+ segments.push(...orphanSegments);
259
+ }
260
+ } else if (entry.type === "route") {
261
+ if (!options?.skipLoaders) {
262
+ const loaderSegments = await resolveLoaders(entry, context, true, deps);
263
+ segments.push(...loaderSegments);
264
+ }
265
+
266
+ // Route handler EXECUTES before its children (orphan layouts, parallels).
267
+ // This lets the handler set() context variables that children can read
268
+ // via get(). Caching wraps all segments together (per-route, not
269
+ // per-segment), so either all run or none do -- no partial scenarios.
270
+ //
271
+ // The handler's segment is PUSHED after orphans/parallels to preserve
272
+ // the correct tree composition order (layouts wrap the route content).
273
+ (context as InternalHandlerContext<any, TEnv>)._currentSegmentId =
274
+ entry.shortCode;
275
+ let component: ReactNode | undefined = await tryStaticHandler(
276
+ entry,
277
+ entry.shortCode,
278
+ );
279
+ if (component === undefined) {
280
+ const doneRouteHandler = track(`handler:${entry.id}`, 2);
281
+ if (entry.loading) {
282
+ const result = handleHandlerResult(entry.handler(context));
283
+ if (result instanceof Promise) {
284
+ result.finally(doneRouteHandler).catch(() => {});
285
+ const tracked = deps.trackHandler(result, {
286
+ segmentId: entry.shortCode,
287
+ segmentType: entry.type,
288
+ });
289
+ observeStreamedHandler(
290
+ tracked,
291
+ entry.shortCode,
292
+ entry.type,
293
+ context.pathname,
294
+ routeKey,
295
+ params,
296
+ );
297
+ component = tracked;
298
+ } else {
299
+ doneRouteHandler();
300
+ component = result;
301
+ }
302
+ } else {
303
+ component = handleHandlerResult(await entry.handler(context));
304
+ doneRouteHandler();
305
+ }
306
+ }
307
+
308
+ for (const orphan of entry.layout) {
309
+ const orphanSegments = await resolveOrphanLayout(
310
+ orphan,
311
+ params,
312
+ context,
313
+ loaderPromises,
314
+ true,
315
+ deps,
316
+ options,
317
+ routeKey,
318
+ );
319
+ segments.push(...orphanSegments);
320
+ }
321
+
322
+ const resolvedParallelEntries = new Set<string>();
323
+ for (const { slot, entry: parallelEntry } of getParallelSlotEntries(
324
+ entry.parallel,
325
+ )) {
326
+ const parallelSegments = await resolveParallelEntry(
327
+ parallelEntry,
328
+ params,
329
+ context,
330
+ true,
331
+ entry.shortCode,
332
+ deps,
333
+ options,
334
+ routeKey,
335
+ [slot],
336
+ !resolvedParallelEntries.has(parallelEntry.id),
337
+ );
338
+ segments.push(...parallelSegments);
339
+ resolvedParallelEntries.add(parallelEntry.id);
340
+ }
341
+
342
+ segments.push({
343
+ id: entry.shortCode,
344
+ namespace: entry.id,
345
+ type: "route",
346
+ index: 0,
347
+ component: component ?? null,
348
+ loading: entry.loading === false ? null : entry.loading,
349
+ transition: entry.transition,
350
+ params,
351
+ belongsToRoute: true,
352
+ ...(entry.mountPath ? { mountPath: entry.mountPath } : {}),
353
+ });
354
+ } else {
355
+ throw new Error(`Unknown entry type: ${(entry as any).type}`);
356
+ }
357
+
358
+ return segments;
359
+ }
360
+
361
+ /**
362
+ * Resolve orphan layout with its middlewares, loaders, and parallels.
363
+ */
364
+ export async function resolveOrphanLayout<TEnv>(
365
+ orphan: EntryData,
366
+ params: Record<string, string>,
367
+ context: HandlerContext<any, TEnv>,
368
+ loaderPromises: Map<string, Promise<any>>,
369
+ belongsToRoute: boolean,
370
+ deps: SegmentResolutionDeps<TEnv>,
371
+ options?: ResolveSegmentOptions,
372
+ routeKey?: string,
373
+ ): Promise<ResolvedSegment[]> {
374
+ invariant(
375
+ orphan.type === "layout" || orphan.type === "cache",
376
+ `Expected orphan to be a layout or cache, got: ${orphan.type}`,
377
+ );
378
+
379
+ const segments: ResolvedSegment[] = [];
380
+ if (!options?.skipLoaders) {
381
+ const loaderSegments = await resolveLoaders(
382
+ orphan,
383
+ context,
384
+ belongsToRoute,
385
+ deps,
386
+ );
387
+ segments.push(...loaderSegments);
388
+ }
389
+
390
+ // Handler-first: orphan layout handler executes before its parallels
391
+ // so that ctx.set() values are visible to parallel children.
392
+ const doneOrphanHandler = track(`handler:${orphan.id}`, 2);
393
+ const component = await resolveLayoutComponent(orphan, context);
394
+ doneOrphanHandler();
395
+
396
+ segments.push({
397
+ id: orphan.shortCode,
398
+ namespace: orphan.id,
399
+ type: "layout",
400
+ index: 0,
401
+ component,
402
+ params,
403
+ belongsToRoute,
404
+ layoutName: orphan.id,
405
+ loading: orphan.loading === false ? null : orphan.loading,
406
+ transition: orphan.transition,
407
+ ...(orphan.mountPath ? { mountPath: orphan.mountPath } : {}),
408
+ });
409
+
410
+ const resolvedParallelEntries = new Set<string>();
411
+ for (const { slot, entry: parallelEntry } of getParallelSlotEntries(
412
+ orphan.parallel,
413
+ )) {
414
+ const parallelSegments = await resolveParallelEntry(
415
+ parallelEntry,
416
+ params,
417
+ context,
418
+ belongsToRoute,
419
+ orphan.shortCode,
420
+ deps,
421
+ options,
422
+ routeKey,
423
+ [slot],
424
+ !resolvedParallelEntries.has(parallelEntry.id),
425
+ );
426
+ segments.push(...parallelSegments);
427
+ resolvedParallelEntries.add(parallelEntry.id);
428
+ }
429
+
430
+ return segments;
431
+ }
432
+
433
+ /**
434
+ * Resolve parallel EntryData with its loaders and slot handlers.
435
+ */
436
+ export async function resolveParallelEntry<TEnv>(
437
+ parallelEntry: EntryData,
438
+ params: Record<string, string>,
439
+ context: HandlerContext<any, TEnv>,
440
+ belongsToRoute: boolean,
441
+ parentShortCode: string,
442
+ deps: SegmentResolutionDeps<TEnv>,
443
+ options?: ResolveSegmentOptions,
444
+ routeKey?: string,
445
+ slotNames?: `@${string}`[],
446
+ includeLoaders: boolean = true,
447
+ ): Promise<ResolvedSegment[]> {
448
+ invariant(
449
+ parallelEntry.type === "parallel",
450
+ `Expected parallel entry, got: ${parallelEntry.type}`,
451
+ );
452
+
453
+ const segments: ResolvedSegment[] = [];
454
+
455
+ const slots = parallelEntry.handler as Record<
456
+ `@${string}`,
457
+ | ((ctx: HandlerContext<any, TEnv>) => ReactNode | Promise<ReactNode>)
458
+ | ReactNode
459
+ >;
460
+
461
+ const slotsToResolve = slotNames ?? (Object.keys(slots) as `@${string}`[]);
462
+
463
+ for (const slot of slotsToResolve) {
464
+ // Try static lookup first — in production, handler bodies are evicted
465
+ // and replaced with stubs that have no .handler property (undefined).
466
+ // The static store holds the pre-rendered component for these slots.
467
+ let component: ReactNode | undefined = await tryStaticSlot(
468
+ parallelEntry,
469
+ slot,
470
+ `${parentShortCode}.${slot}`,
471
+ );
472
+
473
+ if (component === undefined) {
474
+ const handler = slots[slot];
475
+ if (handler === undefined) {
476
+ continue;
477
+ }
478
+ const doneParallelHandler = track(
479
+ `handler:${parallelEntry.id}.${slot}`,
480
+ 2,
481
+ );
482
+ const hasLoadingFallback =
483
+ parallelEntry.loading !== undefined && parallelEntry.loading !== false;
484
+ if (hasLoadingFallback) {
485
+ const result =
486
+ typeof handler === "function" ? handler(context) : handler;
487
+ if (result instanceof Promise) {
488
+ result.finally(doneParallelHandler).catch(() => {});
489
+ const tracked = deps.trackHandler(result, {
490
+ segmentId: `${parentShortCode}.${slot}`,
491
+ segmentType: "parallel",
492
+ });
493
+ observeStreamedHandler(
494
+ tracked,
495
+ `${parentShortCode}.${slot}`,
496
+ "parallel",
497
+ context.pathname,
498
+ routeKey,
499
+ params,
500
+ );
501
+ component = tracked as ReactNode;
502
+ } else {
503
+ doneParallelHandler();
504
+ component = result as ReactNode;
505
+ }
506
+ } else {
507
+ component =
508
+ typeof handler === "function" ? await handler(context) : handler;
509
+ doneParallelHandler();
510
+ }
511
+ }
512
+
513
+ segments.push({
514
+ id: `${parentShortCode}.${slot}`,
515
+ namespace: parallelEntry.id,
516
+ type: "parallel",
517
+ index: 0,
518
+ component,
519
+ loading: parallelEntry.loading === false ? null : parallelEntry.loading,
520
+ transition: parallelEntry.transition,
521
+ params,
522
+ slot,
523
+ belongsToRoute,
524
+ parallelName: `${parallelEntry.id}.${slot}`,
525
+ ...(parallelEntry.mountPath
526
+ ? { mountPath: parallelEntry.mountPath }
527
+ : {}),
528
+ });
529
+ }
530
+
531
+ if (!options?.skipLoaders && includeLoaders) {
532
+ const loaderSegments = await resolveLoaders(
533
+ parallelEntry,
534
+ context,
535
+ belongsToRoute,
536
+ deps,
537
+ parentShortCode,
538
+ );
539
+ // Tag parallel-owned loaders so renderSegments can stream them
540
+ // using the parallel's loading() instead of awaiting on the layout
541
+ const parallelLoading =
542
+ parallelEntry.loading === false ? undefined : parallelEntry.loading;
543
+ if (parallelLoading) {
544
+ for (const seg of loaderSegments) {
545
+ seg.parallelLoading = parallelLoading;
546
+ }
547
+ }
548
+ segments.push(...loaderSegments);
549
+ }
550
+
551
+ return segments;
552
+ }
553
+
554
+ /**
555
+ * Resolve all segments for a route (used for single-cache-per-request pattern).
556
+ */
557
+ export async function resolveAllSegments<TEnv>(
558
+ entries: EntryData[],
559
+ routeKey: string,
560
+ params: Record<string, string>,
561
+ context: HandlerContext<any, TEnv>,
562
+ loaderPromises: Map<string, Promise<any>>,
563
+ deps: SegmentResolutionDeps<TEnv>,
564
+ options?: ResolveSegmentOptions,
565
+ ): Promise<ResolvedSegment[]> {
566
+ const allSegments: ResolvedSegment[] = [];
567
+ const seenIds = new Set<string>();
568
+
569
+ // Safe request access: during build-time prerendering, context.request
570
+ // is a throwing getter. Use undefined when unavailable.
571
+ let safeRequest: Request | undefined;
572
+ try {
573
+ safeRequest = context.request;
574
+ } catch {}
575
+
576
+ // Get telemetry sink from RouterContext (may not exist during prerendering)
577
+ let telemetry;
578
+ try {
579
+ telemetry = getRouterContext()?.telemetry;
580
+ } catch {}
581
+
582
+ for (const entry of entries) {
583
+ const doneEntry = track(`segment:${entry.id}`, 1);
584
+ const resolvedSegments = await resolveWithErrorBoundary(
585
+ entry,
586
+ params,
587
+ () =>
588
+ resolveSegment(
589
+ entry,
590
+ routeKey,
591
+ params,
592
+ context,
593
+ loaderPromises,
594
+ deps,
595
+ false,
596
+ options,
597
+ ),
598
+ (seg) => [seg],
599
+ deps,
600
+ { request: safeRequest, url: context.url, routeKey, telemetry },
601
+ context.pathname,
602
+ );
603
+ doneEntry();
604
+ // Deduplicate by segment ID. include() scopes can produce entries that
605
+ // resolve the same shared layout/loader segment. Duplicates in the segment
606
+ // array propagate to the client's matched[] and change the React tree depth.
607
+ for (const seg of resolvedSegments) {
608
+ if (!seenIds.has(seg.id)) {
609
+ seenIds.add(seg.id);
610
+ allSegments.push(seg);
611
+ }
612
+ }
613
+ }
614
+
615
+ return allSegments;
616
+ }
617
+
618
+ /**
619
+ * Resolve only loader segments for all entries (used when serving cached non-loader segments).
620
+ */
621
+ export async function resolveLoadersOnly<TEnv>(
622
+ entries: EntryData[],
623
+ context: HandlerContext<any, TEnv>,
624
+ deps: SegmentResolutionDeps<TEnv>,
625
+ ): Promise<ResolvedSegment[]> {
626
+ const loaderSegments: ResolvedSegment[] = [];
627
+ const seenIds = new Set<string>();
628
+
629
+ async function collectEntryLoaders(
630
+ entry: EntryData,
631
+ belongsToRoute: boolean,
632
+ shortCodeOverride?: string,
633
+ ): Promise<void> {
634
+ // Skip if all loaders from this entry have already been resolved
635
+ // via a parent (e.g., cache boundary wrapping a layout with shared loaders).
636
+ const entryLoaders = entry.loader ?? [];
637
+ const sc = shortCodeOverride ?? entry.shortCode;
638
+ const allAlreadySeen =
639
+ entryLoaders.length > 0 &&
640
+ entryLoaders.every((le, i) =>
641
+ seenIds.has(`${sc}D${i}.${le.loader.$$id}`),
642
+ );
643
+ if (!allAlreadySeen) {
644
+ const segments = await resolveLoaders(
645
+ entry,
646
+ context,
647
+ belongsToRoute,
648
+ deps,
649
+ shortCodeOverride,
650
+ );
651
+ for (const seg of segments) {
652
+ if (!seenIds.has(seg.id)) {
653
+ seenIds.add(seg.id);
654
+ loaderSegments.push(seg);
655
+ }
656
+ }
657
+ }
658
+
659
+ const seenParallelEntryIds = new Set<string>();
660
+ for (const parallelEntry of getParallelEntries(entry.parallel)) {
661
+ if (seenParallelEntryIds.has(parallelEntry.id)) continue;
662
+ seenParallelEntryIds.add(parallelEntry.id);
663
+ await collectEntryLoaders(parallelEntry, belongsToRoute, entry.shortCode);
664
+ }
665
+
666
+ const childBelongsToRoute = belongsToRoute || entry.type === "route";
667
+ for (const layoutEntry of entry.layout) {
668
+ await collectEntryLoaders(layoutEntry, childBelongsToRoute);
669
+ }
670
+ }
671
+
672
+ for (const entry of entries) {
673
+ await collectEntryLoaders(entry, entry.type === "route");
674
+ }
675
+
676
+ return loaderSegments;
677
+ }