@rangojs/router 0.0.0-experimental.19 → 0.0.0-experimental.1fa245e2

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 (160) hide show
  1. package/{CLAUDE.md → AGENTS.md} +4 -0
  2. package/README.md +122 -30
  3. package/dist/bin/rango.js +245 -63
  4. package/dist/vite/index.js +859 -418
  5. package/package.json +3 -3
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +32 -0
  8. package/skills/caching/SKILL.md +49 -8
  9. package/skills/document-cache/SKILL.md +2 -2
  10. package/skills/hooks/SKILL.md +33 -31
  11. package/skills/host-router/SKILL.md +218 -0
  12. package/skills/links/SKILL.md +3 -1
  13. package/skills/loader/SKILL.md +72 -22
  14. package/skills/middleware/SKILL.md +2 -0
  15. package/skills/parallel/SKILL.md +126 -0
  16. package/skills/prerender/SKILL.md +112 -70
  17. package/skills/rango/SKILL.md +0 -1
  18. package/skills/route/SKILL.md +34 -4
  19. package/skills/router-setup/SKILL.md +95 -5
  20. package/skills/typesafety/SKILL.md +35 -23
  21. package/src/__internal.ts +92 -0
  22. package/src/bin/rango.ts +18 -0
  23. package/src/browser/app-version.ts +14 -0
  24. package/src/browser/event-controller.ts +5 -0
  25. package/src/browser/link-interceptor.ts +4 -0
  26. package/src/browser/navigation-bridge.ts +114 -18
  27. package/src/browser/navigation-client.ts +126 -44
  28. package/src/browser/navigation-store.ts +43 -8
  29. package/src/browser/navigation-transaction.ts +11 -9
  30. package/src/browser/partial-update.ts +80 -15
  31. package/src/browser/prefetch/cache.ts +166 -27
  32. package/src/browser/prefetch/fetch.ts +52 -39
  33. package/src/browser/prefetch/policy.ts +6 -0
  34. package/src/browser/prefetch/queue.ts +92 -20
  35. package/src/browser/prefetch/resource-ready.ts +77 -0
  36. package/src/browser/react/Link.tsx +70 -14
  37. package/src/browser/react/NavigationProvider.tsx +40 -4
  38. package/src/browser/react/context.ts +7 -2
  39. package/src/browser/react/use-handle.ts +9 -58
  40. package/src/browser/react/use-router.ts +21 -8
  41. package/src/browser/rsc-router.tsx +143 -59
  42. package/src/browser/scroll-restoration.ts +41 -42
  43. package/src/browser/segment-reconciler.ts +6 -1
  44. package/src/browser/server-action-bridge.ts +454 -436
  45. package/src/browser/types.ts +60 -5
  46. package/src/build/generate-manifest.ts +6 -6
  47. package/src/build/generate-route-types.ts +5 -0
  48. package/src/build/route-trie.ts +19 -3
  49. package/src/build/route-types/include-resolution.ts +8 -1
  50. package/src/build/route-types/router-processing.ts +346 -87
  51. package/src/build/route-types/scan-filter.ts +8 -1
  52. package/src/cache/cache-runtime.ts +15 -11
  53. package/src/cache/cache-scope.ts +48 -7
  54. package/src/cache/cf/cf-cache-store.ts +453 -11
  55. package/src/cache/cf/index.ts +5 -1
  56. package/src/cache/document-cache.ts +17 -7
  57. package/src/cache/index.ts +1 -0
  58. package/src/cache/taint.ts +55 -0
  59. package/src/client.rsc.tsx +2 -1
  60. package/src/client.tsx +3 -102
  61. package/src/context-var.ts +72 -2
  62. package/src/debug.ts +2 -2
  63. package/src/handle.ts +40 -0
  64. package/src/handles/breadcrumbs.ts +66 -0
  65. package/src/handles/index.ts +1 -0
  66. package/src/host/index.ts +0 -3
  67. package/src/index.rsc.ts +8 -37
  68. package/src/index.ts +40 -66
  69. package/src/prerender/store.ts +57 -15
  70. package/src/prerender.ts +138 -77
  71. package/src/reverse.ts +22 -1
  72. package/src/route-definition/dsl-helpers.ts +73 -25
  73. package/src/route-definition/helpers-types.ts +10 -6
  74. package/src/route-definition/index.ts +3 -3
  75. package/src/route-definition/redirect.ts +11 -3
  76. package/src/route-definition/resolve-handler-use.ts +149 -0
  77. package/src/route-map-builder.ts +7 -1
  78. package/src/route-types.ts +11 -0
  79. package/src/router/content-negotiation.ts +100 -1
  80. package/src/router/find-match.ts +4 -2
  81. package/src/router/handler-context.ts +108 -25
  82. package/src/router/intercept-resolution.ts +11 -4
  83. package/src/router/lazy-includes.ts +4 -1
  84. package/src/router/loader-resolution.ts +123 -11
  85. package/src/router/logging.ts +5 -2
  86. package/src/router/manifest.ts +9 -3
  87. package/src/router/match-api.ts +125 -190
  88. package/src/router/match-middleware/background-revalidation.ts +30 -2
  89. package/src/router/match-middleware/cache-lookup.ts +88 -16
  90. package/src/router/match-middleware/cache-store.ts +53 -10
  91. package/src/router/match-middleware/intercept-resolution.ts +9 -7
  92. package/src/router/match-middleware/segment-resolution.ts +61 -5
  93. package/src/router/match-result.ts +22 -15
  94. package/src/router/metrics.ts +238 -13
  95. package/src/router/middleware-types.ts +53 -12
  96. package/src/router/middleware.ts +172 -85
  97. package/src/router/navigation-snapshot.ts +182 -0
  98. package/src/router/pattern-matching.ts +20 -5
  99. package/src/router/prerender-match.ts +114 -10
  100. package/src/router/preview-match.ts +30 -102
  101. package/src/router/request-classification.ts +310 -0
  102. package/src/router/revalidation.ts +27 -7
  103. package/src/router/route-snapshot.ts +245 -0
  104. package/src/router/router-context.ts +6 -1
  105. package/src/router/router-interfaces.ts +50 -5
  106. package/src/router/router-options.ts +50 -19
  107. package/src/router/segment-resolution/fresh.ts +200 -19
  108. package/src/router/segment-resolution/helpers.ts +30 -25
  109. package/src/router/segment-resolution/loader-cache.ts +1 -0
  110. package/src/router/segment-resolution/revalidation.ts +429 -301
  111. package/src/router/segment-wrappers.ts +2 -0
  112. package/src/router/trie-matching.ts +20 -2
  113. package/src/router/types.ts +1 -0
  114. package/src/router.ts +88 -15
  115. package/src/rsc/handler.ts +546 -359
  116. package/src/rsc/index.ts +0 -20
  117. package/src/rsc/manifest-init.ts +5 -1
  118. package/src/rsc/progressive-enhancement.ts +25 -8
  119. package/src/rsc/rsc-rendering.ts +35 -43
  120. package/src/rsc/server-action.ts +16 -10
  121. package/src/rsc/ssr-setup.ts +128 -0
  122. package/src/rsc/types.ts +10 -1
  123. package/src/search-params.ts +16 -13
  124. package/src/segment-system.tsx +140 -4
  125. package/src/server/context.ts +148 -16
  126. package/src/server/loader-registry.ts +9 -8
  127. package/src/server/request-context.ts +182 -34
  128. package/src/server.ts +6 -0
  129. package/src/ssr/index.tsx +4 -0
  130. package/src/static-handler.ts +18 -6
  131. package/src/theme/index.ts +4 -13
  132. package/src/types/cache-types.ts +4 -4
  133. package/src/types/handler-context.ts +149 -49
  134. package/src/types/loader-types.ts +36 -9
  135. package/src/types/route-config.ts +17 -8
  136. package/src/types/route-entry.ts +8 -1
  137. package/src/types/segments.ts +2 -5
  138. package/src/urls/path-helper-types.ts +9 -2
  139. package/src/urls/path-helper.ts +48 -13
  140. package/src/urls/pattern-types.ts +12 -0
  141. package/src/urls/response-types.ts +16 -6
  142. package/src/use-loader.tsx +73 -4
  143. package/src/vite/discovery/bundle-postprocess.ts +61 -89
  144. package/src/vite/discovery/discover-routers.ts +23 -5
  145. package/src/vite/discovery/prerender-collection.ts +48 -15
  146. package/src/vite/discovery/state.ts +17 -13
  147. package/src/vite/index.ts +8 -3
  148. package/src/vite/plugin-types.ts +51 -79
  149. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  150. package/src/vite/plugins/expose-action-id.ts +1 -3
  151. package/src/vite/plugins/performance-tracks.ts +88 -0
  152. package/src/vite/plugins/refresh-cmd.ts +127 -0
  153. package/src/vite/plugins/version-plugin.ts +13 -1
  154. package/src/vite/rango.ts +174 -211
  155. package/src/vite/router-discovery.ts +169 -42
  156. package/src/vite/utils/banner.ts +3 -3
  157. package/src/vite/utils/prerender-utils.ts +78 -0
  158. package/src/vite/utils/shared-utils.ts +3 -2
  159. package/skills/testing/SKILL.md +0 -226
  160. package/src/route-definition/route-function.ts +0 -119
package/src/reverse.ts CHANGED
@@ -305,8 +305,22 @@ export function createReverse<TRoutes extends Record<string, string>>(
305
305
  if (params) {
306
306
  // Replace :param placeholders with actual values
307
307
  // Strip constraint syntax: :param(a|b) -> use "param" as key
308
+ // Optional params (:param?) are omitted when not provided
309
+ let hadOmittedOptional = false;
308
310
  result = result.replace(
309
- /:([a-zA-Z_][a-zA-Z0-9_]*)(\([^)]*\))?\??/g,
311
+ /:([a-zA-Z_][a-zA-Z0-9_]*)(\([^)]*\))?(\?)/g,
312
+ (_, key, _constraint, optional) => {
313
+ const value = params[key];
314
+ if (value === undefined) {
315
+ hadOmittedOptional = true;
316
+ return "";
317
+ }
318
+ return encodeURIComponent(value);
319
+ },
320
+ );
321
+ // Second pass: required params (no trailing ?)
322
+ result = result.replace(
323
+ /:([a-zA-Z_][a-zA-Z0-9_]*)(\([^)]*\))?(?!\?)/g,
310
324
  (_, key) => {
311
325
  const value = params[key];
312
326
  if (value === undefined) {
@@ -315,6 +329,13 @@ export function createReverse<TRoutes extends Record<string, string>>(
315
329
  return encodeURIComponent(value);
316
330
  },
317
331
  );
332
+ // Clean up slashes only when an optional param was actually omitted,
333
+ // so intentional trailing-slash patterns like "/blog/" are preserved.
334
+ if (hadOmittedOptional) {
335
+ const hadTrailingSlash = pattern.length > 1 && pattern.endsWith("/");
336
+ result = result.replace(/\/\/+/g, "/").replace(/\/+$/, "") || "/";
337
+ if (hadTrailingSlash && !result.endsWith("/")) result += "/";
338
+ }
318
339
  }
319
340
 
320
341
  // Append search params as query string
@@ -37,6 +37,7 @@ import type {
37
37
  UseItems,
38
38
  } from "../route-types.js";
39
39
  import type { RouteHelpers } from "./helpers-types.js";
40
+ import { resolveHandlerUse, mergeHandlerUse } from "./resolve-handler-use.js";
40
41
 
41
42
  /**
42
43
  * Check if an item contains routes (directly or inside nested structures like cache).
@@ -282,7 +283,7 @@ const cache: RouteHelpers<any, any>["cache"] = (
282
283
  errorBoundary: [],
283
284
  notFoundBoundary: [],
284
285
  layout: [],
285
- parallel: [],
286
+ parallel: {},
286
287
  intercept: [],
287
288
  loader: [],
288
289
  ...(cacheUrlPrefix ? { mountPath: cacheUrlPrefix } : {}),
@@ -320,7 +321,7 @@ const cache: RouteHelpers<any, any>["cache"] = (
320
321
  errorBoundary: [],
321
322
  notFoundBoundary: [],
322
323
  layout: [],
323
- parallel: [],
324
+ parallel: {},
324
325
  intercept: [],
325
326
  loader: [],
326
327
  ...(cacheUrlPrefix2 ? { mountPath: cacheUrlPrefix2 } : {}),
@@ -393,6 +394,8 @@ const parallel: RouteHelpers<any, any>["parallel"] = (slots, use) => {
393
394
  "parallel() cannot be nested inside another parallel()",
394
395
  );
395
396
 
397
+ const slotNames = Object.keys(slots as Record<string, any>) as `@${string}`[];
398
+
396
399
  const namespace = `${ctx.namespace}.$${store.getNextIndex("parallel")}`;
397
400
 
398
401
  // Unwrap any static handler definitions in parallel slots
@@ -431,7 +434,7 @@ const parallel: RouteHelpers<any, any>["parallel"] = (slots, use) => {
431
434
  errorBoundary: [],
432
435
  notFoundBoundary: [],
433
436
  layout: [],
434
- parallel: [],
437
+ parallel: {},
435
438
  intercept: [],
436
439
  loader: [],
437
440
  ...(parallelUrlPrefix ? { mountPath: parallelUrlPrefix } : {}),
@@ -445,16 +448,46 @@ const parallel: RouteHelpers<any, any>["parallel"] = (slots, use) => {
445
448
  : {}),
446
449
  } satisfies EntryData;
447
450
 
448
- // Run use callback if provided to collect loaders, revalidate, loading
449
- if (use && typeof use === "function") {
450
- const result = store.run(namespace, entry, use)?.flat(3);
451
- invariant(
452
- Array.isArray(result) && result.every((item) => isValidUseItem(item)),
453
- `parallel() use() callback must return an array of use items [${namespace}]`,
454
- );
455
- }
451
+ for (const slotName of slotNames) {
452
+ const slotEntry = {
453
+ ...entry,
454
+ handler: { [slotName]: unwrappedSlots[slotName]! },
455
+ middleware: [...entry.middleware],
456
+ revalidate: [...entry.revalidate],
457
+ errorBoundary: [...entry.errorBoundary],
458
+ notFoundBoundary: [...entry.notFoundBoundary],
459
+ layout: [...entry.layout],
460
+ parallel: { ...entry.parallel },
461
+ intercept: [...entry.intercept],
462
+ loader: [...entry.loader],
463
+ ...(entry.staticHandlerIds?.[slotName]
464
+ ? {
465
+ isStaticPrerender: true as const,
466
+ staticHandlerIds: { [slotName]: entry.staticHandlerIds[slotName]! },
467
+ }
468
+ : {
469
+ isStaticPrerender: undefined,
470
+ staticHandlerIds: undefined,
471
+ }),
472
+ } satisfies EntryData;
473
+
474
+ // Per-slot: handler.use defaults first, then explicit use second.
475
+ // This matches the "defaults first, overrides second" rule used by
476
+ // path(), layout(), and intercept(). Each slot's handler.use is
477
+ // scoped to its own entry (no cross-slot bleed).
478
+ const slotHandler = (slots as Record<string, any>)[slotName];
479
+ const slotHandlerUse = resolveHandlerUse(slotHandler);
480
+ const slotMergedUse = mergeHandlerUse(slotHandlerUse, use, "parallel");
481
+ if (slotMergedUse) {
482
+ const result = store.run(namespace, slotEntry, slotMergedUse)?.flat(3);
483
+ invariant(
484
+ Array.isArray(result) && result.every((item) => isValidUseItem(item)),
485
+ `parallel() use() callback must return an array of use items [${namespace}]`,
486
+ );
487
+ }
456
488
 
457
- ctx.parent.parallel.push(entry);
489
+ ctx.parent.parallel[slotName] = slotEntry;
490
+ }
458
491
  return { name: namespace, type: "parallel" } as ParallelItem;
459
492
  };
460
493
 
@@ -502,8 +535,12 @@ const intercept = (
502
535
  when: [], // Selector conditions for conditional interception
503
536
  };
504
537
 
505
- // Run use callback if provided to collect loaders, revalidate, middleware, etc.
506
- if (use && typeof use === "function") {
538
+ // Merge handler.use defaults with explicit use
539
+ const handlerUseFn = resolveHandlerUse(handler);
540
+ const mergedUse = mergeHandlerUse(handlerUseFn, use, "intercept");
541
+
542
+ // Run merged use callback to collect loaders, revalidate, middleware, etc.
543
+ if (mergedUse) {
507
544
  // Create a temporary parent context for the use() callback
508
545
  // so that middleware, loader, revalidate attach to the intercept entry
509
546
  const originalParent = ctx.parent;
@@ -530,7 +567,7 @@ const intercept = (
530
567
  };
531
568
  ctx.parent = tempParent as EntryData;
532
569
 
533
- const result = use()?.flat(3);
570
+ const result = mergedUse()?.flat(3);
534
571
 
535
572
  // Restore original parent
536
573
  ctx.parent = originalParent;
@@ -627,11 +664,15 @@ const loadingFn: RouteHelpers<any, any>["loading"] = (component, options) => {
627
664
  invariant(false, "No parent entry available for loading()");
628
665
  }
629
666
 
667
+ // Unwrap function form: loading(() => <Skeleton />) → loading(<Skeleton />)
668
+ const resolved =
669
+ typeof component === "function" ? (component as () => any)() : component;
670
+
630
671
  // If ssr: false and we're in SSR, set loading to false
631
672
  if (options?.ssr === false && ctx.isSSR) {
632
673
  parent.loading = false;
633
674
  } else {
634
- parent.loading = component;
675
+ parent.loading = resolved;
635
676
  }
636
677
 
637
678
  const name = `$${store.getNextIndex("loading")}`;
@@ -687,7 +728,7 @@ const transitionFn = (
687
728
  errorBoundary: [],
688
729
  notFoundBoundary: [],
689
730
  layout: [],
690
- parallel: [],
731
+ parallel: {},
691
732
  intercept: [],
692
733
  loader: [],
693
734
  } as EntryData;
@@ -727,14 +768,14 @@ const routeFn: RouteHelpers<any, any>["route"] = (name, handler, use) => {
727
768
  shortCode: store.getShortCode("route"),
728
769
  type: "route",
729
770
  parent: ctx.parent,
730
- handler,
771
+ handler: handler as unknown as Handler<any, any, any>,
731
772
  loading: undefined, // Allow loading() to attach loading state
732
773
  middleware: [],
733
774
  revalidate: [],
734
775
  errorBoundary: [],
735
776
  notFoundBoundary: [],
736
777
  layout: [],
737
- parallel: [],
778
+ parallel: {},
738
779
  intercept: [],
739
780
  loader: [],
740
781
  } satisfies EntryData;
@@ -746,9 +787,12 @@ const routeFn: RouteHelpers<any, any>["route"] = (name, handler, use) => {
746
787
  );
747
788
  /* Register route entry */
748
789
  ctx.manifest.set(name, entry);
790
+ /* Merge handler.use defaults with explicit use */
791
+ const handlerUseFn = resolveHandlerUse(handler);
792
+ const mergedUse = mergeHandlerUse(handlerUseFn, use, "route");
749
793
  /* Run use and attach handlers */
750
- if (use && typeof use === "function") {
751
- const result = store.run(namespace, entry, use)?.flat(3);
794
+ if (mergedUse) {
795
+ const result = store.run(namespace, entry, mergedUse)?.flat(3);
752
796
  invariant(
753
797
  Array.isArray(result) && result.every((item) => isValidUseItem(item)),
754
798
  `route() use() callback must return an array of use items [${namespace}]`,
@@ -791,7 +835,7 @@ const layout: RouteHelpers<any, any>["layout"] = (handler, use) => {
791
835
  revalidate: [],
792
836
  errorBoundary: [],
793
837
  notFoundBoundary: [],
794
- parallel: [],
838
+ parallel: {},
795
839
  intercept: [],
796
840
  layout: [],
797
841
  loader: [],
@@ -809,10 +853,14 @@ const layout: RouteHelpers<any, any>["layout"] = (handler, use) => {
809
853
  (handler as any).$$routePrefix = ctx.namePrefix;
810
854
  }
811
855
 
812
- // Run use callback if provided
856
+ // Merge handler.use defaults with explicit use
857
+ const handlerUseFn = resolveHandlerUse(handler);
858
+ const mergedUse = mergeHandlerUse(handlerUseFn, use, "layout");
859
+
860
+ // Run merged use callback if present
813
861
  let result: AllUseItems[] | undefined;
814
- if (use && typeof use === "function") {
815
- result = store.run(namespace, entry, use)?.flat(3);
862
+ if (mergedUse) {
863
+ result = store.run(namespace, entry, mergedUse)?.flat(3);
816
864
 
817
865
  invariant(
818
866
  Array.isArray(result) && result.every((item) => isValidUseItem(item)),
@@ -228,11 +228,12 @@ export type RouteHelpers<T extends RouteDefinition, TEnv> = {
228
228
  * revalidate(({ actionId }) => actionId?.includes("Cart") ?? false),
229
229
  * ])
230
230
  *
231
- * // Access loader data in handlers via ctx.use()
232
- * route("products.detail", async (ctx) => {
233
- * const product = await ctx.use(ProductLoader);
234
- * return <ProductPage product={product} />;
235
- * })
231
+ * // Consume in client components with useLoader()
232
+ * // (preferred — cache-safe, always fresh)
233
+ * function ProductDetails() {
234
+ * const { data } = useLoader(ProductLoader);
235
+ * return <div>{data.name}</div>;
236
+ * }
236
237
  * ```
237
238
  * @param loaderDef - Loader created with createLoader()
238
239
  * @param use - Optional callback for loader-specific revalidation rules
@@ -254,7 +255,10 @@ export type RouteHelpers<T extends RouteDefinition, TEnv> = {
254
255
  * @param options - Configuration options
255
256
  * @param options.ssr - If false, skip showing loading on document requests (SSR)
256
257
  */
257
- loading: (component: ReactNode, options?: { ssr?: boolean }) => LoadingItem;
258
+ loading: (
259
+ component: ReactNode | (() => ReactNode),
260
+ options?: { ssr?: boolean },
261
+ ) => LoadingItem;
258
262
  /**
259
263
  * Attach an error boundary to catch errors in this segment and children
260
264
  * ```typescript
@@ -1,6 +1,3 @@
1
- // Route definition
2
- export { route, type RouteDefinitionResult } from "./route-function.js";
3
-
4
1
  // Type definitions
5
2
  export type { RouteHelpers } from "./helpers-types.js";
6
3
  export type {
@@ -48,6 +45,9 @@ export {
48
45
  type RouteHandlers,
49
46
  } from "./helper-factories.js";
50
47
 
48
+ // Handler use resolver
49
+ export { resolveHandlerUse } from "./resolve-handler-use.js";
50
+
51
51
  // Redirect
52
52
  export { redirect } from "./redirect.js";
53
53
 
@@ -2,6 +2,7 @@ import type { LocationStateEntry } from "../browser/react/location-state-shared.
2
2
  import {
3
3
  requireRequestContext,
4
4
  getRequestContext,
5
+ _getRequestContext,
5
6
  } from "../server/request-context.js";
6
7
 
7
8
  /**
@@ -71,9 +72,9 @@ export function redirect(
71
72
  // actions both deliver state through Flight payloads, so suppress for those.
72
73
  if (
73
74
  reqCtx &&
74
- !reqCtx.url.searchParams.has("_rsc_partial") &&
75
+ !reqCtx.originalUrl.searchParams.has("_rsc_partial") &&
75
76
  !reqCtx.request.headers.has("rsc-action") &&
76
- !reqCtx.url.searchParams.has("_rsc_action")
77
+ !reqCtx.originalUrl.searchParams.has("_rsc_action")
77
78
  ) {
78
79
  console.warn(
79
80
  `[Router] redirect() with state during a full-page (SSR) request to "${url}". ` +
@@ -83,10 +84,17 @@ export function redirect(
83
84
  }
84
85
  }
85
86
 
87
+ // Auto-prefix root-relative URLs with basename for app-local redirects.
88
+ const bn = _getRequestContext()?._basename;
89
+ let resolvedUrl = url;
90
+ if (bn && url.startsWith("/") && !url.startsWith(bn + "/") && url !== bn) {
91
+ resolvedUrl = url === "/" ? bn : bn + url;
92
+ }
93
+
86
94
  return new Response(null, {
87
95
  status,
88
96
  headers: {
89
- Location: url,
97
+ Location: resolvedUrl,
90
98
  "X-RSC-Redirect": "soft",
91
99
  },
92
100
  });
@@ -0,0 +1,149 @@
1
+ import type { AllUseItems } from "../route-types.js";
2
+ import { isPrerenderHandler, isPassthroughHandler } from "../prerender.js";
3
+ import { isStaticHandler } from "../static-handler.js";
4
+
5
+ /**
6
+ * Extract the .use callback from any handler shape.
7
+ *
8
+ * Checks definition brands first (objects with __brand), then plain functions.
9
+ * ReactNode handlers return undefined (no .use possible).
10
+ */
11
+ export function resolveHandlerUse(handler: unknown): (() => any[]) | undefined {
12
+ if (handler == null) return undefined;
13
+
14
+ // Check branded definitions first — they're objects but also have typeof "object"
15
+ if (isPassthroughHandler(handler)) {
16
+ return (handler as any).use;
17
+ }
18
+ if (isPrerenderHandler(handler)) {
19
+ return (handler as any).use;
20
+ }
21
+ if (isStaticHandler(handler)) {
22
+ return (handler as any).use;
23
+ }
24
+ // Plain handler function
25
+ if (typeof handler === "function") {
26
+ return (handler as any).use;
27
+ }
28
+ // ReactNode or other — no .use
29
+ return undefined;
30
+ }
31
+
32
+ /**
33
+ * Allowed item types per mount site.
34
+ * Mirrors the RouteUseItem / ParallelUseItem / InterceptUseItem / LayoutUseItem unions
35
+ * from route-types.ts for runtime validation.
36
+ */
37
+ const MOUNT_SITE_ALLOWED_TYPES: Record<string, Set<string>> = {
38
+ path: new Set([
39
+ "layout",
40
+ "parallel",
41
+ "intercept",
42
+ "middleware",
43
+ "revalidate",
44
+ "loader",
45
+ "loading",
46
+ "errorBoundary",
47
+ "notFoundBoundary",
48
+ "cache",
49
+ "transition",
50
+ ]),
51
+ // Response routes (path.json, path.text, etc.) — mirrors ResponseRouteUseItem
52
+ response: new Set(["middleware", "cache"]),
53
+ route: new Set([
54
+ "layout",
55
+ "parallel",
56
+ "intercept",
57
+ "middleware",
58
+ "revalidate",
59
+ "loader",
60
+ "loading",
61
+ "errorBoundary",
62
+ "notFoundBoundary",
63
+ "cache",
64
+ "transition",
65
+ ]),
66
+ // layout allows AllUseItems — no validation needed, but included for completeness
67
+ layout: new Set([
68
+ "layout",
69
+ "route",
70
+ "middleware",
71
+ "revalidate",
72
+ "parallel",
73
+ "intercept",
74
+ "loader",
75
+ "loading",
76
+ "errorBoundary",
77
+ "notFoundBoundary",
78
+ "cache",
79
+ "transition",
80
+ "include",
81
+ ]),
82
+ parallel: new Set([
83
+ "revalidate",
84
+ "loader",
85
+ "loading",
86
+ "errorBoundary",
87
+ "notFoundBoundary",
88
+ "transition",
89
+ ]),
90
+ intercept: new Set([
91
+ "middleware",
92
+ "revalidate",
93
+ "loader",
94
+ "loading",
95
+ "errorBoundary",
96
+ "notFoundBoundary",
97
+ "layout",
98
+ "route",
99
+ "when",
100
+ "transition",
101
+ ]),
102
+ };
103
+
104
+ /**
105
+ * Validate that items from handler.use() are valid for the given mount site.
106
+ * Throws a descriptive error if any item is not allowed.
107
+ */
108
+ export function validateHandlerUseItems(
109
+ items: AllUseItems[],
110
+ mountSite: string,
111
+ ): void {
112
+ const allowed = MOUNT_SITE_ALLOWED_TYPES[mountSite];
113
+ if (!allowed) return;
114
+ for (const item of items) {
115
+ if (item == null) continue;
116
+ if (!allowed.has((item as any).type)) {
117
+ throw new Error(
118
+ `handler.use() returned ${(item as any).type}() which is not valid inside ${mountSite}(). ` +
119
+ `Allowed types: ${[...allowed].join(", ")}.`,
120
+ );
121
+ }
122
+ }
123
+ }
124
+
125
+ /**
126
+ * Create a merged use callback from handler.use and explicit use.
127
+ * handler.use items come first (defaults), explicit items second (overrides).
128
+ * Returns undefined if both are absent.
129
+ */
130
+ export function mergeHandlerUse(
131
+ handlerUse: (() => any[]) | undefined,
132
+ explicitUse: (() => any[]) | undefined,
133
+ mountSite: string,
134
+ ): (() => any[]) | undefined {
135
+ if (!handlerUse && !explicitUse) return undefined;
136
+ if (!handlerUse) return explicitUse;
137
+ if (!explicitUse) {
138
+ return () => {
139
+ const items = handlerUse().flat(3);
140
+ validateHandlerUseItems(items, mountSite);
141
+ return items;
142
+ };
143
+ }
144
+ return () => {
145
+ const hItems = handlerUse().flat(3);
146
+ validateHandlerUseItems(hItems, mountSite);
147
+ return [...hItems, ...explicitUse()];
148
+ };
149
+ }
@@ -199,7 +199,13 @@ export function registerRouterManifestLoader(
199
199
  }
200
200
 
201
201
  export async function ensureRouterManifest(routerId: string): Promise<void> {
202
- if (perRouterManifestMap.has(routerId)) return;
202
+ // Check both manifest AND trie. The virtual module's setRouterManifest()
203
+ // pre-sets the manifest at startup, but the per-router trie is only
204
+ // available from the lazy loader. Without this, the lazy loader never
205
+ // runs and findMatch falls back to the global merged trie — which
206
+ // contains routes from ALL routers and breaks multi-router setups.
207
+ if (perRouterManifestMap.has(routerId) && perRouterTrieMap.has(routerId))
208
+ return;
203
209
  const loader = routerManifestLoaders.get(routerId);
204
210
  if (loader) {
205
211
  const mod = await loader();
@@ -257,3 +257,14 @@ export type LoaderUseItem = RevalidateItem | CacheItem;
257
257
  * runtime via .flat(3).
258
258
  */
259
259
  export type UseItems<T> = (T | readonly T[])[];
260
+
261
+ /**
262
+ * Union of all items that handler.use() may return.
263
+ * A handler doesn't know its mount site at definition time, so the type
264
+ * is intentionally broad — validation happens per-mount-site at runtime.
265
+ */
266
+ export type HandlerUseItem =
267
+ | RouteUseItem
268
+ | LayoutUseItem
269
+ | ParallelUseItem
270
+ | InterceptUseItem;
@@ -2,10 +2,18 @@
2
2
  * Content Negotiation Utilities
3
3
  *
4
4
  * Pure functions for HTTP Accept header parsing and response type matching.
5
- * Used by createRouter's previewMatch for content negotiation between
5
+ * Used by previewMatch and classifyRequest for content negotiation between
6
6
  * RSC routes and response routes (JSON, text, image, stream, etc.).
7
7
  */
8
8
 
9
+ import type { EntryData } from "../server/context.js";
10
+ import type { CollectedMiddleware } from "./middleware-types.js";
11
+ import { collectRouteMiddleware } from "./middleware.js";
12
+ import { loadManifest } from "./manifest.js";
13
+ import { traverseBack } from "./pattern-matching.js";
14
+ import type { RouteMatchResult } from "./pattern-matching.js";
15
+ import type { RouteSnapshot } from "./route-snapshot.js";
16
+
9
17
  // Response type -> MIME type used for Accept header matching
10
18
  export const RESPONSE_TYPE_MIME: Record<string, string> = {
11
19
  json: "application/json",
@@ -114,3 +122,94 @@ export function pickNegotiateVariant(
114
122
  // No match -- use first candidate as default
115
123
  return candidates[0]!;
116
124
  }
125
+
126
+ /**
127
+ * Result of content negotiation for a route with negotiate variants.
128
+ */
129
+ export interface NegotiationResult {
130
+ /** The winning response type */
131
+ responseType: string;
132
+ /** Handler function for the winning variant */
133
+ handler: Function;
134
+ /** Manifest entry for the winning variant (may differ from primary) */
135
+ manifestEntry: EntryData;
136
+ /** Route middleware for the winning variant */
137
+ routeMiddleware: CollectedMiddleware[];
138
+ /** Always true — negotiation occurred */
139
+ negotiated: true;
140
+ }
141
+
142
+ /**
143
+ * Perform content negotiation for a route with negotiate variants.
144
+ *
145
+ * Returns a NegotiationResult when a response route wins negotiation.
146
+ * Returns null when RSC wins or no negotiation is needed.
147
+ *
148
+ * Shared by previewMatch and classifyRequest to avoid duplicating
149
+ * the candidate-building and variant-loading logic.
150
+ */
151
+ export async function negotiateRoute(
152
+ request: Request,
153
+ pathname: string,
154
+ snapshot: RouteSnapshot,
155
+ ): Promise<NegotiationResult | null> {
156
+ const { matched, manifestEntry, routeMiddleware, responseType } = snapshot;
157
+ if (!matched.negotiateVariants || matched.negotiateVariants.length === 0) {
158
+ return null;
159
+ }
160
+
161
+ const acceptEntries = parseAcceptTypes(request.headers.get("accept") || "");
162
+
163
+ // Build candidate list preserving definition order.
164
+ const variants = matched.negotiateVariants;
165
+ let candidates: Array<{ routeKey: string; responseType: string }>;
166
+ if (responseType) {
167
+ candidates = [...variants, { routeKey: matched.routeKey, responseType }];
168
+ } else {
169
+ const rscCandidate = {
170
+ routeKey: matched.routeKey,
171
+ responseType: RSC_RESPONSE_TYPE,
172
+ };
173
+ candidates = matched.rscFirst
174
+ ? [rscCandidate, ...variants]
175
+ : [...variants, rscCandidate];
176
+ }
177
+
178
+ const variant = pickNegotiateVariant(acceptEntries, candidates);
179
+
180
+ // RSC won negotiation
181
+ if (variant.responseType === RSC_RESPONSE_TYPE) {
182
+ return null;
183
+ }
184
+
185
+ // Primary response-type won — use existing manifest entry and middleware
186
+ if (responseType && variant.routeKey === matched.routeKey) {
187
+ return {
188
+ responseType,
189
+ handler: manifestEntry.handler as Function,
190
+ manifestEntry,
191
+ routeMiddleware,
192
+ negotiated: true,
193
+ };
194
+ }
195
+
196
+ // Different variant won — load its manifest entry
197
+ const negotiateEntry = await loadManifest(
198
+ matched.entry,
199
+ variant.routeKey,
200
+ pathname,
201
+ undefined,
202
+ false,
203
+ );
204
+ const variantMiddleware = collectRouteMiddleware(
205
+ traverseBack(negotiateEntry),
206
+ matched.params,
207
+ );
208
+ return {
209
+ responseType: variant.responseType,
210
+ handler: negotiateEntry.handler as Function,
211
+ manifestEntry: negotiateEntry,
212
+ routeMiddleware: variantMiddleware,
213
+ negotiated: true,
214
+ };
215
+ }
@@ -52,8 +52,10 @@ export function createFindMatch<TEnv = any>(
52
52
  : undefined;
53
53
 
54
54
  // Phase 1: Try trie match (O(path_length))
55
- // Prefer per-router trie (isolated) over global trie (merged).
56
- const routeTrie = getRouterTrie(deps.routerId) ?? getRouteTrie();
55
+ // Only use the per-router trie. The global trie merges routes from ALL
56
+ // routers and must not be used — in multi-router setups (host routing)
57
+ // overlapping paths like "/" would match the wrong app's route.
58
+ const routeTrie = getRouterTrie(deps.routerId);
57
59
  if (routeTrie) {
58
60
  const trieStart = performance.now();
59
61
  const trieResult = tryTrieMatch(routeTrie, pathname);