@rangojs/router 0.0.0-experimental.122 → 0.0.0-experimental.125

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 (260) hide show
  1. package/dist/bin/rango.js +10 -6
  2. package/dist/testing/vitest.js +82 -0
  3. package/dist/vite/index.js +55 -48
  4. package/package.json +61 -21
  5. package/skills/caching/SKILL.md +2 -1
  6. package/skills/hooks/SKILL.md +40 -29
  7. package/skills/host-router/SKILL.md +16 -2
  8. package/skills/intercept/SKILL.md +4 -2
  9. package/skills/layout/SKILL.md +11 -6
  10. package/skills/loader/SKILL.md +6 -2
  11. package/skills/middleware/SKILL.md +4 -2
  12. package/skills/migrate-nextjs/SKILL.md +3 -1
  13. package/skills/parallel/SKILL.md +9 -4
  14. package/skills/rango/SKILL.md +12 -0
  15. package/skills/route/SKILL.md +10 -2
  16. package/skills/testing/SKILL.md +129 -0
  17. package/skills/testing/bindings.md +89 -0
  18. package/skills/testing/cache-prerender.md +98 -0
  19. package/skills/testing/client-components.md +122 -0
  20. package/skills/testing/e2e-parity.md +125 -0
  21. package/skills/testing/flight.md +89 -0
  22. package/skills/testing/handles.md +129 -0
  23. package/skills/testing/loader.md +128 -0
  24. package/skills/testing/middleware.md +99 -0
  25. package/skills/testing/render-handler.md +118 -0
  26. package/skills/testing/response-routes.md +95 -0
  27. package/skills/testing/reverse-and-types.md +84 -0
  28. package/skills/testing/server-actions.md +107 -0
  29. package/skills/testing/server-tree.md +128 -0
  30. package/skills/testing/setup.md +120 -0
  31. package/src/__internal.ts +0 -65
  32. package/src/browser/action-coordinator.ts +1 -1
  33. package/src/browser/action-fence.ts +47 -0
  34. package/src/browser/cookie-name.ts +140 -0
  35. package/src/browser/event-controller.ts +1 -83
  36. package/src/browser/invalidate-client-cache.ts +52 -0
  37. package/src/browser/navigation-bridge.ts +14 -1
  38. package/src/browser/navigation-client.ts +14 -1
  39. package/src/browser/navigation-store-handle.ts +38 -0
  40. package/src/browser/navigation-store.ts +26 -51
  41. package/src/browser/navigation-transaction.ts +0 -32
  42. package/src/browser/partial-update.ts +1 -83
  43. package/src/browser/prefetch/cache.ts +6 -45
  44. package/src/browser/prefetch/fetch.ts +7 -0
  45. package/src/browser/prefetch/queue.ts +6 -3
  46. package/src/browser/rango-state.ts +157 -99
  47. package/src/browser/react/Link.tsx +0 -2
  48. package/src/browser/react/NavigationProvider.tsx +2 -1
  49. package/src/browser/react/ScrollRestoration.tsx +10 -6
  50. package/src/browser/react/filter-segment-order.ts +0 -2
  51. package/src/browser/react/index.ts +0 -51
  52. package/src/browser/react/location-state-shared.ts +0 -13
  53. package/src/browser/react/location-state.ts +0 -1
  54. package/src/browser/react/use-action.ts +6 -15
  55. package/src/browser/react/use-handle.ts +0 -5
  56. package/src/browser/react/use-link-status.ts +0 -4
  57. package/src/browser/react/use-navigation.ts +0 -3
  58. package/src/browser/react/use-params.ts +0 -2
  59. package/src/browser/react/use-search-params.ts +0 -5
  60. package/src/browser/react/use-segments.ts +0 -13
  61. package/src/browser/rsc-router.tsx +12 -4
  62. package/src/browser/server-action-bridge.ts +77 -15
  63. package/src/browser/types.ts +7 -2
  64. package/src/browser/validate-redirect-origin.ts +4 -5
  65. package/src/build/route-trie.ts +3 -0
  66. package/src/build/route-types/param-extraction.ts +6 -3
  67. package/src/build/route-types/router-processing.ts +0 -8
  68. package/src/cache/cache-policy.ts +0 -54
  69. package/src/cache/cache-runtime.ts +27 -24
  70. package/src/cache/cache-scope.ts +0 -27
  71. package/src/cache/cache-tag.ts +0 -37
  72. package/src/cache/cf/cf-cache-store.ts +94 -46
  73. package/src/cache/cf/index.ts +0 -24
  74. package/src/cache/document-cache.ts +11 -36
  75. package/src/cache/handle-snapshot.ts +0 -40
  76. package/src/cache/index.ts +0 -27
  77. package/src/cache/memory-segment-store.ts +2 -48
  78. package/src/cache/profile-registry.ts +7 -3
  79. package/src/cache/read-through-swr.ts +41 -11
  80. package/src/cache/segment-codec.ts +0 -16
  81. package/src/cache/types.ts +0 -98
  82. package/src/client.rsc.tsx +1 -22
  83. package/src/client.tsx +14 -38
  84. package/src/component-utils.ts +19 -0
  85. package/src/deps/ssr.ts +0 -1
  86. package/src/handle.ts +28 -18
  87. package/src/handles/MetaTags.tsx +0 -14
  88. package/src/handles/meta.ts +0 -39
  89. package/src/host/cookie-handler.ts +0 -36
  90. package/src/host/errors.ts +0 -24
  91. package/src/host/index.ts +6 -0
  92. package/src/host/pattern-matcher.ts +7 -50
  93. package/src/host/router.ts +1 -65
  94. package/src/host/testing.ts +40 -27
  95. package/src/host/types.ts +6 -2
  96. package/src/href-client.ts +0 -4
  97. package/src/index.rsc.ts +42 -3
  98. package/src/index.ts +31 -1
  99. package/src/internal-debug.ts +2 -4
  100. package/src/loader.rsc.ts +19 -9
  101. package/src/loader.ts +12 -4
  102. package/src/network-error-thrower.tsx +1 -6
  103. package/src/outlet-provider.tsx +1 -5
  104. package/src/prerender/param-hash.ts +10 -11
  105. package/src/prerender/store.ts +23 -30
  106. package/src/prerender.ts +58 -3
  107. package/src/root-error-boundary.tsx +1 -19
  108. package/src/route-content-wrapper.tsx +1 -44
  109. package/src/route-definition/dsl-helpers.ts +7 -19
  110. package/src/route-definition/helpers-types.ts +3 -3
  111. package/src/route-definition/redirect.ts +11 -1
  112. package/src/route-map-builder.ts +0 -16
  113. package/src/router/basename.ts +14 -0
  114. package/src/router/content-negotiation.ts +0 -13
  115. package/src/router/error-handling.ts +12 -16
  116. package/src/router/find-match.ts +4 -30
  117. package/src/router/intercept-resolution.ts +10 -1
  118. package/src/router/lazy-includes.ts +1 -57
  119. package/src/router/loader-resolution.ts +3 -2
  120. package/src/router/logging.ts +0 -6
  121. package/src/router/manifest.ts +1 -25
  122. package/src/router/match-api.ts +0 -20
  123. package/src/router/match-context.ts +0 -22
  124. package/src/router/match-handlers.ts +57 -58
  125. package/src/router/match-middleware/background-revalidation.ts +0 -7
  126. package/src/router/match-middleware/cache-lookup.ts +1 -54
  127. package/src/router/match-middleware/cache-store.ts +0 -31
  128. package/src/router/match-middleware/intercept-resolution.ts +0 -22
  129. package/src/router/match-middleware/segment-resolution.ts +0 -21
  130. package/src/router/match-pipelines.ts +1 -42
  131. package/src/router/match-result.ts +1 -52
  132. package/src/router/metrics.ts +0 -34
  133. package/src/router/middleware-cookies.ts +0 -13
  134. package/src/router/middleware-types.ts +0 -115
  135. package/src/router/middleware.ts +7 -30
  136. package/src/router/navigation-snapshot.ts +0 -51
  137. package/src/router/params-util.ts +23 -0
  138. package/src/router/pattern-matching.ts +1 -33
  139. package/src/router/prerender-match.ts +33 -45
  140. package/src/router/request-classification.ts +1 -38
  141. package/src/router/revalidation.ts +5 -58
  142. package/src/router/router-context.ts +0 -26
  143. package/src/router/router-interfaces.ts +7 -0
  144. package/src/router/router-options.ts +30 -0
  145. package/src/router/segment-resolution/fresh.ts +25 -57
  146. package/src/router/segment-resolution/helpers.ts +34 -0
  147. package/src/router/segment-resolution/loader-cache.ts +10 -13
  148. package/src/router/segment-resolution/revalidation.ts +5 -42
  149. package/src/router/segment-resolution/streamed-handler-telemetry.ts +52 -0
  150. package/src/router/segment-resolution.ts +4 -1
  151. package/src/router/state-cookie-name.ts +33 -0
  152. package/src/router/telemetry-otel.ts +0 -20
  153. package/src/router/telemetry.ts +96 -19
  154. package/src/router/timeout.ts +0 -20
  155. package/src/router/trie-matching.ts +63 -40
  156. package/src/router/types.ts +1 -63
  157. package/src/router/url-params.ts +0 -5
  158. package/src/router.ts +40 -9
  159. package/src/rsc/handler.ts +14 -2
  160. package/src/rsc/helpers.ts +34 -0
  161. package/src/rsc/origin-guard.ts +0 -12
  162. package/src/rsc/progressive-enhancement.ts +4 -1
  163. package/src/rsc/rsc-rendering.ts +4 -7
  164. package/src/rsc/runtime-warnings.ts +14 -0
  165. package/src/rsc/server-action.ts +30 -28
  166. package/src/rsc/types.ts +2 -1
  167. package/src/runtime-env.ts +18 -0
  168. package/src/search-params.ts +0 -16
  169. package/src/segment-loader-promise.ts +14 -2
  170. package/src/segment-system.tsx +79 -88
  171. package/src/server/cookie-store.ts +52 -1
  172. package/src/server/handle-store.ts +7 -24
  173. package/src/server/loader-registry.ts +5 -24
  174. package/src/server/request-context.ts +74 -77
  175. package/src/ssr/index.tsx +14 -14
  176. package/src/static-handler.ts +10 -13
  177. package/src/testing/cache-status.ts +119 -0
  178. package/src/testing/collect-handle.ts +40 -0
  179. package/src/testing/dispatch.ts +581 -0
  180. package/src/testing/dom.entry.ts +22 -0
  181. package/src/testing/e2e/fixture.ts +188 -0
  182. package/src/testing/e2e/index.ts +127 -0
  183. package/src/testing/e2e/matchers.ts +35 -0
  184. package/src/testing/e2e/page-helpers.ts +272 -0
  185. package/src/testing/e2e/parity.ts +387 -0
  186. package/src/testing/e2e/server.ts +195 -0
  187. package/src/testing/flight-matchers.ts +97 -0
  188. package/src/testing/flight-normalize.ts +11 -0
  189. package/src/testing/flight-runtime.d.ts +57 -0
  190. package/src/testing/flight-tree.ts +682 -0
  191. package/src/testing/flight.entry.ts +52 -0
  192. package/src/testing/flight.ts +186 -0
  193. package/src/testing/generated-routes.ts +183 -0
  194. package/src/testing/index.ts +98 -0
  195. package/src/testing/internal/context.ts +348 -0
  196. package/src/testing/internal/flight-client-globals.ts +30 -0
  197. package/src/testing/internal/seed-vars.ts +54 -0
  198. package/src/testing/render-handler.ts +311 -0
  199. package/src/testing/render-route.tsx +504 -0
  200. package/src/testing/run-loader.ts +378 -0
  201. package/src/testing/run-middleware.ts +205 -0
  202. package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
  203. package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
  204. package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
  205. package/src/testing/vitest-stubs/version.ts +5 -0
  206. package/src/testing/vitest.ts +305 -0
  207. package/src/theme/ThemeProvider.tsx +0 -52
  208. package/src/theme/ThemeScript.tsx +0 -6
  209. package/src/theme/constants.ts +0 -12
  210. package/src/theme/index.ts +0 -7
  211. package/src/theme/theme-context.ts +1 -5
  212. package/src/theme/theme-script.ts +0 -14
  213. package/src/theme/use-theme.ts +0 -3
  214. package/src/types/boundaries.ts +0 -35
  215. package/src/types/error-types.ts +25 -89
  216. package/src/types/global-namespace.ts +15 -15
  217. package/src/types/handler-context.ts +16 -13
  218. package/src/types/index.ts +0 -10
  219. package/src/types/request-scope.ts +0 -19
  220. package/src/types/route-config.ts +6 -50
  221. package/src/types/route-entry.ts +0 -6
  222. package/src/types/segments.ts +0 -13
  223. package/src/urls/include-helper.ts +0 -4
  224. package/src/urls/index.ts +0 -6
  225. package/src/urls/path-helper-types.ts +2 -2
  226. package/src/urls/path-helper.ts +0 -54
  227. package/src/urls/urls-function.ts +0 -13
  228. package/src/use-loader.tsx +0 -186
  229. package/src/vite/discovery/bundle-postprocess.ts +2 -1
  230. package/src/vite/discovery/discover-routers.ts +6 -7
  231. package/src/vite/discovery/virtual-module-codegen.ts +1 -11
  232. package/src/vite/plugin-types.ts +3 -1
  233. package/src/vite/plugins/cjs-to-esm.ts +0 -11
  234. package/src/vite/plugins/client-ref-dedup.ts +0 -11
  235. package/src/vite/plugins/client-ref-hashing.ts +0 -10
  236. package/src/vite/plugins/cloudflare-protocol-stub.ts +0 -20
  237. package/src/vite/plugins/expose-action-id.ts +2 -73
  238. package/src/vite/plugins/expose-id-utils.ts +0 -55
  239. package/src/vite/plugins/expose-ids/export-analysis.ts +0 -38
  240. package/src/vite/plugins/expose-ids/handler-transform.ts +0 -15
  241. package/src/vite/plugins/expose-ids/loader-transform.ts +0 -15
  242. package/src/vite/plugins/expose-ids/router-transform.ts +0 -13
  243. package/src/vite/plugins/expose-internal-ids.ts +10 -0
  244. package/src/vite/plugins/performance-tracks.ts +0 -3
  245. package/src/vite/plugins/use-cache-transform.ts +0 -36
  246. package/src/vite/plugins/version-injector.ts +0 -20
  247. package/src/vite/plugins/version-plugin.ts +1 -49
  248. package/src/vite/plugins/virtual-entries.ts +0 -15
  249. package/src/vite/rango.ts +1 -108
  250. package/src/vite/router-discovery.ts +2 -1
  251. package/src/vite/utils/ast-handler-extract.ts +0 -16
  252. package/src/vite/utils/bundle-analysis.ts +6 -13
  253. package/src/vite/utils/client-chunks.ts +0 -6
  254. package/src/vite/utils/forward-user-plugins.ts +0 -22
  255. package/src/vite/utils/manifest-utils.ts +0 -4
  256. package/src/vite/utils/package-resolution.ts +1 -73
  257. package/src/vite/utils/prerender-utils.ts +0 -35
  258. package/src/vite/utils/shared-utils.ts +3 -35
  259. package/src/browser/react/use-client-cache.ts +0 -58
  260. package/src/browser/shallow.ts +0 -40
@@ -1,9 +1,3 @@
1
- /**
2
- * Router Metrics Utilities
3
- *
4
- * Performance metrics collection and reporting for Rango.
5
- */
6
-
7
1
  import type { MetricsStore, PerformanceMetric } from "../server/context";
8
2
 
9
3
  const BASE_INDENT = 2;
@@ -16,7 +10,6 @@ function formatMs(value: number): string {
16
10
 
17
11
  function sortMetrics(metrics: PerformanceMetric[]): PerformanceMetric[] {
18
12
  return [...metrics].sort((a, b) => {
19
- // handler:total always goes last (it wraps everything)
20
13
  if (a.label === "handler:total") return 1;
21
14
  if (b.label === "handler:total") return -1;
22
15
  return a.startTime - b.startTime;
@@ -68,11 +61,6 @@ function createTimelineAxis(total: number): string {
68
61
  )}${totalLabel}`;
69
62
  }
70
63
 
71
- /**
72
- * Create a metrics store for the request if debugPerformance is enabled.
73
- * An optional `requestStart` timestamp can anchor the store to an earlier
74
- * point (e.g. handler start) so that handler:total has startTime=0.
75
- */
76
64
  export function createMetricsStore(
77
65
  debugPerformance: boolean,
78
66
  requestStart?: number,
@@ -85,9 +73,6 @@ export function createMetricsStore(
85
73
  };
86
74
  }
87
75
 
88
- /**
89
- * Append a metric to the request store using an absolute start timestamp.
90
- */
91
76
  export function appendMetric(
92
77
  metricsStore: MetricsStore | undefined,
93
78
  label: string,
@@ -104,9 +89,6 @@ export function appendMetric(
104
89
  });
105
90
  }
106
91
 
107
- /**
108
- * Log the current request metrics and return the corresponding Server-Timing value.
109
- */
110
92
  export function buildMetricsTiming(
111
93
  method: string,
112
94
  pathname: string,
@@ -117,7 +99,6 @@ export function buildMetricsTiming(
117
99
  return generateServerTiming(metricsStore) || undefined;
118
100
  }
119
101
 
120
- /** Display row produced by merging :pre/:post metric pairs. */
121
102
  interface DisplayRow {
122
103
  label: string;
123
104
  startTime: number;
@@ -126,12 +107,7 @@ interface DisplayRow {
126
107
  spans: Span[];
127
108
  }
128
109
 
129
- /**
130
- * Build display rows from sorted metrics, merging :pre/:post pairs into
131
- * a single row with disjoint timeline segments.
132
- */
133
110
  function buildDisplayRows(sorted: PerformanceMetric[]): DisplayRow[] {
134
- // Index :pre and :post metrics by their base label
135
111
  const preMap = new Map<string, PerformanceMetric>();
136
112
  const postMap = new Map<string, PerformanceMetric>();
137
113
  const consumed = new Set<PerformanceMetric>();
@@ -211,11 +187,6 @@ function buildDisplayRows(sorted: PerformanceMetric[]): DisplayRow[] {
211
187
  return rows;
212
188
  }
213
189
 
214
- /**
215
- * Log metrics to console in a formatted way.
216
- * Uses a shared-axis timeline so overlapping work stays visible.
217
- * Merges :pre/:post pairs onto one row with disjoint timeline segments.
218
- */
219
190
  export function logMetrics(
220
191
  method: string,
221
192
  pathname: string,
@@ -267,11 +238,6 @@ export function logMetrics(
267
238
  }
268
239
  }
269
240
 
270
- /**
271
- * Generate Server-Timing header value from metrics
272
- * Format: metric-name;dur=X.XX
273
- * Depth is encoded as a "d{N}-" prefix for nested metrics.
274
- */
275
241
  export function generateServerTiming(metricsStore: MetricsStore): string {
276
242
  return metricsStore.metrics
277
243
  .map((m) => {
@@ -1,14 +1,5 @@
1
- /**
2
- * Cookie Utilities
3
- *
4
- * Parsing and serialization for HTTP cookies used by middleware context.
5
- */
6
-
7
1
  import type { CookieOptions } from "./middleware-types.js";
8
2
 
9
- /**
10
- * Parse cookies from Cookie header
11
- */
12
3
  export function parseCookies(
13
4
  cookieHeader: string | null,
14
5
  ): Record<string, string> {
@@ -24,7 +15,6 @@ export function parseCookies(
24
15
  try {
25
16
  cookies[name] = decodeURIComponent(raw);
26
17
  } catch {
27
- // Malformed percent-encoded value (e.g. %zz) - fall back to raw value
28
18
  cookies[name] = raw;
29
19
  }
30
20
  }
@@ -33,9 +23,6 @@ export function parseCookies(
33
23
  return cookies;
34
24
  }
35
25
 
36
- /**
37
- * Serialize a cookie for Set-Cookie header
38
- */
39
26
  export function serializeCookie(
40
27
  name: string,
41
28
  value: string,
@@ -1,10 +1,3 @@
1
- /**
2
- * Middleware Types
3
- *
4
- * Type definitions and interfaces for the middleware system.
5
- * Separated from execution logic for cleaner imports.
6
- */
7
-
8
1
  import type { ContextVar } from "../context-var.js";
9
2
  import type {
10
3
  DefaultReverseRouteMap,
@@ -16,17 +9,11 @@ import type { Theme } from "../theme/types.js";
16
9
  import type { LocationStateEntry } from "../browser/react/location-state-shared.js";
17
10
  import type { RequestScope } from "../types/request-scope.js";
18
11
 
19
- /**
20
- * Get variable function type
21
- */
22
12
  type GetVariableFn = {
23
13
  <T>(contextVar: ContextVar<T>): T | undefined;
24
14
  <K extends keyof DefaultVars>(key: K): DefaultVars[K];
25
15
  };
26
16
 
27
- /**
28
- * Set variable function type
29
- */
30
17
  type SetVariableFn = {
31
18
  <T>(contextVar: ContextVar<T>, value: T, options?: { cache?: boolean }): void;
32
19
  <K extends keyof DefaultVars>(
@@ -36,9 +23,6 @@ type SetVariableFn = {
36
23
  ): void;
37
24
  };
38
25
 
39
- /**
40
- * Cookie options for setting cookies
41
- */
42
26
  export interface CookieOptions {
43
27
  domain?: string;
44
28
  path?: string;
@@ -49,109 +33,36 @@ export interface CookieOptions {
49
33
  sameSite?: "strict" | "lax" | "none";
50
34
  }
51
35
 
52
- /**
53
- * Context passed to middleware
54
- *
55
- * @template TEnv - Environment type (bindings, variables) - defaults to any for internal flexibility
56
- * @template TParams - URL params type (typed for route middleware,
57
- * `Record<string, string | undefined>` for global middleware — absent
58
- * optional segments are omitted from the params record at runtime, so
59
- * the index signature must include `undefined`)
60
- */
61
36
  export interface MiddlewareContext<
62
37
  TEnv = any,
63
38
  TParams = Record<string, string | undefined>,
64
39
  > extends RequestScope<TEnv> {
65
- /** URL params extracted from route/middleware pattern */
66
40
  params: TParams;
67
41
 
68
- /**
69
- * Response headers.
70
- * Before `next()`, returns headers from the shared response stub.
71
- * After `next()`, returns headers from the downstream response.
72
- */
73
42
  readonly headers: Headers;
74
43
 
75
- /** Get a context variable (shared with route handlers) */
76
44
  get: GetVariableFn;
77
45
 
78
- /** Set a context variable (shared with route handlers) */
79
46
  set: SetVariableFn;
80
47
 
81
- /**
82
- * Set a response header - can be called before or after `next()`.
83
- *
84
- * When called before `next()`, headers are queued and merged into the final response.
85
- * When called after `next()`, headers are set directly on the response.
86
- */
87
48
  header(name: string, value: string): void;
88
49
 
89
- /**
90
- * The matched route name, if available and the route has an explicit name.
91
- * Undefined for global middleware (runs before route matching) or unnamed routes.
92
- */
93
50
  routeName?: DefaultRouteName;
94
51
 
95
- /**
96
- * Enable performance metrics for this request.
97
- * When called, granular timing breakdown is logged to console and
98
- * included in the Server-Timing response header, regardless of the
99
- * router-level `debugPerformance` option.
100
- *
101
- * Call **before** `await next()` so the metrics store exists when
102
- * downstream phases (route matching, rendering, SSR) record their
103
- * spans. Calling after `next()` returns still emits `handler:total`
104
- * but misses all upstream metrics.
105
- */
106
52
  debugPerformance(): void;
107
53
 
108
- /**
109
- * Current theme (from cookie or default).
110
- * Only available when theme is enabled in router config.
111
- */
112
54
  theme?: Theme;
113
55
 
114
- /**
115
- * Set the theme (only available when theme is enabled in router config).
116
- * Sets a cookie with the new theme value.
117
- */
118
56
  setTheme?: (theme: Theme) => void;
119
57
 
120
- /**
121
- * Attach location state entries to this response.
122
- * State is delivered to the client via history.pushState and accessible
123
- * through the useLocationState() hook.
124
- */
125
58
  setLocationState(entries: LocationStateEntry | LocationStateEntry[]): void;
126
59
 
127
- /**
128
- * Generate URLs from route names.
129
- * - `name` — global route, from the named-routes definition
130
- */
131
60
  reverse: ScopedReverseFunction<
132
61
  Record<string, string>,
133
62
  DefaultReverseRouteMap
134
63
  >;
135
64
  }
136
65
 
137
- /**
138
- * Middleware function signature
139
- *
140
- * @template TEnv - Environment type - defaults to any for internal flexibility
141
- * @template TParams - URL params type (typed for route middleware)
142
- *
143
- * When using middleware with global augmentation (Rango.Env), explicitly
144
- * annotate your middleware functions, or the types will be inferred from context:
145
- *
146
- * @example
147
- * ```typescript
148
- * // With explicit annotation (recommended for reusable middleware)
149
- * const authMiddleware: MiddlewareFn<AppEnv> = async (ctx, next) => {...}
150
- *
151
- * // Types inferred from router.use() call
152
- * router.use((ctx, next) => {...}) // ctx is typed from router's TEnv
153
- * ```
154
- */
155
66
  export type MiddlewareFn<
156
67
  TEnv = any,
157
68
  TParams = Record<string, string | undefined>,
@@ -160,50 +71,24 @@ export type MiddlewareFn<
160
71
  next: () => Promise<Response>,
161
72
  ) => Response | void | Promise<Response | void>;
162
73
 
163
- /**
164
- * Stored middleware entry with pattern matching info
165
- * @internal - uses any for internal flexibility
166
- */
167
74
  export interface MiddlewareEntry<TEnv = any> {
168
- /** Original pattern string */
169
75
  pattern: string | null;
170
-
171
- /** Compiled regex for matching */
172
76
  regex: RegExp | null;
173
-
174
- /** Param names extracted from pattern */
175
77
  paramNames: string[];
176
-
177
- /** The middleware function */
178
78
  handler: MiddlewareFn<TEnv>;
179
-
180
- /** Mount prefix this middleware is scoped to (null = global) */
181
79
  mountPrefix: string | null;
182
80
  }
183
81
 
184
- /**
185
- * Mutable response holder - tracks the current response through the middleware chain.
186
- */
187
82
  export interface ResponseHolder {
188
83
  response: Response | null;
189
84
  }
190
85
 
191
- /**
192
- * Entry type for middleware collection
193
- * Matches the shape of EntryData used in router.ts
194
- */
195
86
  export interface MiddlewareCollectableEntry {
196
87
  middleware?: MiddlewareFn<any, any>[];
197
88
  layout?: MiddlewareCollectableEntry[];
198
89
  }
199
90
 
200
- /**
201
- * Collected route middleware with params
202
- */
203
91
  export interface CollectedMiddleware {
204
92
  handler: MiddlewareFn<any, any>;
205
- // Internal shape only. The user-facing `MiddlewareContext.params` is
206
- // typed `Record<string, string | undefined>` to reflect that absent
207
- // optional segments are omitted from the params record at runtime.
208
93
  params: Record<string, string>;
209
94
  }
@@ -1,13 +1,4 @@
1
1
  /// <reference types="vite/types/importMeta.d.ts" />
2
- /**
3
- * Middleware Execution
4
- *
5
- * True middleware that wraps the entire RSC handler.
6
- * - `await next()` returns actual Response
7
- * - Can modify response headers
8
- * - Can catch errors from RSC rendering
9
- * - Forgiving API: if middleware doesn't return, original response is used
10
- */
11
2
 
12
3
  import { contextGet, contextSet } from "../context-var.js";
13
4
  import { safeDecodeURIComponent } from "./url-params.js";
@@ -26,23 +17,20 @@ import { appendMetric, createMetricsStore } from "./metrics.js";
26
17
  import { stripInternalParams } from "./handler-context.js";
27
18
  import { isWebSocketUpgradeResponse } from "../response-utils.js";
28
19
 
29
- // Re-export types and cookie utilities for backward compatibility
20
+ // Re-export types and cookie utilities consumed through this module's path.
30
21
  export type {
31
22
  CookieOptions,
32
- CollectedMiddleware,
33
23
  MiddlewareCollectableEntry,
34
24
  MiddlewareContext,
35
25
  MiddlewareEntry,
36
26
  MiddlewareFn,
37
- ResponseHolder,
38
27
  } from "./middleware-types.js";
39
28
  export { parseCookies, serializeCookie } from "./middleware-cookies.js";
40
29
 
41
30
  const MIDDLEWARE_METRIC_DEPTH = 1;
42
- /** Ignore post-next() durations below this threshold (measurement noise). */
43
31
  const POST_METRIC_MIN_DURATION_MS = 0.01;
44
32
 
45
- function getMiddlewareMetricBase<TEnv>(
33
+ function getMiddlewareMetricLabel<TEnv>(
46
34
  entry: MiddlewareEntry<TEnv>,
47
35
  ordinal: number,
48
36
  ): string {
@@ -50,23 +38,12 @@ function getMiddlewareMetricBase<TEnv>(
50
38
  const scope = entry.pattern ?? "*";
51
39
 
52
40
  if (handlerName) {
53
- return `${handlerName}@${scope}`;
41
+ return `middleware:${handlerName}@${scope}`;
54
42
  }
55
43
 
56
- return `${scope}#${ordinal + 1}`;
44
+ return `middleware:${scope}#${ordinal + 1}`;
57
45
  }
58
46
 
59
- function getMiddlewareMetricLabel<TEnv>(
60
- entry: MiddlewareEntry<TEnv>,
61
- ordinal: number,
62
- ): string {
63
- return `middleware:${getMiddlewareMetricBase(entry, ordinal)}`;
64
- }
65
-
66
- /**
67
- * Parse a route pattern into regex and param names
68
- * Supports: *, /path, /path/*, /path/:param, /path/:param/*
69
- */
70
47
  export function parsePattern(pattern: string): {
71
48
  regex: RegExp;
72
49
  paramNames: string[];
@@ -355,7 +332,7 @@ function mergeReqCtxStub(
355
332
  * - `ctx.headers` available before and after `await next()`
356
333
  * - `ctx.header()` shorthand for setting a single header
357
334
  * - Forgiving: if middleware doesn't return, uses the downstream response
358
- * - Short-circuit: return Response to stop chain
335
+ * - Short-circuit: return OR throw a Response to stop chain
359
336
  * - Error catching: try/catch around `next()` works
360
337
  */
361
338
  export async function executeMiddleware<TEnv>(
@@ -490,7 +467,7 @@ export async function executeMiddleware<TEnv>(
490
467
 
491
468
  // Explicit return takes precedence (middleware short-circuit).
492
469
  // Merge stub headers (from ctx.header before this point) and
493
- // RequestContext stub headers (from ctx.setCookie) into the
470
+ // RequestContext stub headers (from cookies().set()) into the
494
471
  // returned Response so they are not lost.
495
472
  if (result instanceof Response) {
496
473
  if (isWebSocketUpgradeResponse(result)) {
@@ -564,7 +541,7 @@ export async function executeMiddleware<TEnv>(
564
541
  *
565
542
  * Intercepts use a shared stubResponse from the request context. This function:
566
543
  * - Runs middleware in sequence with a simple next() chain
567
- * - Returns Response if any middleware short-circuits (returns Response or redirects BEFORE next())
544
+ * - Returns Response if any middleware short-circuits (returns OR throws a Response, or redirects, BEFORE next())
568
545
  * - Returns null if all middleware calls next() - headers set after next() remain on stubResponse
569
546
  *
570
547
  * @param middlewares - Array of middleware functions
@@ -1,58 +1,26 @@
1
- /**
2
- * Navigation Snapshot
3
- *
4
- * Pure data type representing the navigation-specific state for partial requests.
5
- * Consolidates the header parsing, previous-route matching, intercept-context
6
- * detection, and segment ID filtering that previously lived inline in
7
- * createMatchContextForPartial (match-api.ts).
8
- *
9
- * resolveNavigation() is the factory: given a request + URL + current route key,
10
- * it returns a NavigationSnapshot (or null if no previous URL).
11
- */
12
-
13
1
  import type { RouteMatchResult } from "./pattern-matching.js";
14
2
 
15
- /**
16
- * Snapshot of navigation state for a partial (navigation/action) request.
17
- *
18
- * Contains the "where are we coming from?" data: previous route, intercept
19
- * source, client segment state, and derived flags.
20
- */
21
3
  export interface NavigationSnapshot {
22
- /** Previous page URL (from X-RSC-Router-Client-Path or Referer) */
23
4
  prevUrl: URL;
24
- /** Params from the previous route match */
25
5
  prevParams: Record<string, string>;
26
- /** Previous route match result (null if prev URL doesn't match any route) */
27
6
  prevMatch: RouteMatchResult | null;
28
7
 
29
- /** URL used as intercept context source */
30
8
  interceptContextUrl: URL;
31
- /** Route match for the intercept context URL */
32
9
  interceptContextMatch: RouteMatchResult | null;
33
10
 
34
- /** Raw segment IDs the client currently has */
35
11
  clientSegmentIds: string[];
36
- /** Set version for O(1) lookup */
37
12
  clientSegmentSet: Set<string>;
38
- /** Segment IDs filtered to remove parallel (.@) and loader (D\d+.) entries */
39
13
  filteredSegmentIds: string[];
40
14
 
41
- /** Whether client considers its cache stale */
42
15
  stale: boolean;
43
16
 
44
- /** Whether the intercept context route is the same as the current route */
45
17
  isSameRouteNavigation: boolean;
46
18
 
47
- /** Effective "from" URL (intercept source URL when present, else prevUrl) */
48
19
  effectiveFromUrl: URL;
49
- /** Effective "from" match (intercept source match when present, else prevMatch) */
50
20
  effectiveFromMatch: RouteMatchResult | null;
51
21
 
52
- /** Whether an intercept source header was present */
53
22
  hasInterceptSource: boolean;
54
23
 
55
- /** Whether an HMR request header was present */
56
24
  isHmr: boolean;
57
25
  }
58
26
 
@@ -60,23 +28,12 @@ export interface ResolveNavigationDeps {
60
28
  findMatch: (pathname: string) => RouteMatchResult | null;
61
29
  }
62
30
 
63
- /**
64
- * Resolve navigation state from a partial request.
65
- *
66
- * Returns null if no previous URL is available (required for partial navigation).
67
- *
68
- * @param request - The incoming HTTP request
69
- * @param url - Parsed URL of the request
70
- * @param currentRouteKey - Route key of the current (target) route match
71
- * @param deps - Dependencies (findMatch)
72
- */
73
31
  export function resolveNavigation(
74
32
  request: Request,
75
33
  url: URL,
76
34
  currentRouteKey: string,
77
35
  deps: ResolveNavigationDeps,
78
36
  ): NavigationSnapshot | null {
79
- // Parse client state from RSC request params/headers
80
37
  const clientSegmentIds =
81
38
  url.searchParams.get("_rsc_segments")?.split(",").filter(Boolean) || [];
82
39
  const stale = url.searchParams.get("_rsc_stale") === "true";
@@ -92,7 +49,6 @@ export function resolveNavigation(
92
49
  return null;
93
50
  }
94
51
 
95
- // Parse previous URL
96
52
  let prevUrl: URL;
97
53
  try {
98
54
  prevUrl = new URL(previousUrl, url.origin);
@@ -100,7 +56,6 @@ export function resolveNavigation(
100
56
  return null;
101
57
  }
102
58
 
103
- // Parse intercept context URL
104
59
  let interceptContextUrl: URL;
105
60
  try {
106
61
  interceptContextUrl = interceptSourceUrl
@@ -110,14 +65,12 @@ export function resolveNavigation(
110
65
  interceptContextUrl = prevUrl;
111
66
  }
112
67
 
113
- // Match previous and intercept context routes
114
68
  const prevMatch = deps.findMatch(prevUrl.pathname);
115
69
  const prevParams = prevMatch?.params || {};
116
70
  const interceptContextMatch = interceptSourceUrl
117
71
  ? deps.findMatch(interceptContextUrl.pathname)
118
72
  : prevMatch;
119
73
 
120
- // Derived state
121
74
  const isSameRouteNavigation = !!(
122
75
  interceptContextMatch && interceptContextMatch.routeKey === currentRouteKey
123
76
  );
@@ -128,7 +81,6 @@ export function resolveNavigation(
128
81
  ? interceptContextMatch
129
82
  : prevMatch;
130
83
 
131
- // Filter segment IDs: remove parallel (.@) and loader (D\d+.) entries
132
84
  const filteredSegmentIds = clientSegmentIds.filter((id) => {
133
85
  if (id.includes(".@")) return false;
134
86
  if (/D\d+\./.test(id)) return false;
@@ -155,9 +107,6 @@ export function resolveNavigation(
155
107
  };
156
108
  }
157
109
 
158
- /**
159
- * Test helper: create a NavigationSnapshot with sensible defaults and overrides.
160
- */
161
110
  export function createNavigationSnapshot(
162
111
  overrides?: Partial<NavigationSnapshot>,
163
112
  ): NavigationSnapshot {
@@ -0,0 +1,23 @@
1
+ /**
2
+ * Shared route-param comparison helpers.
3
+ */
4
+
5
+ /**
6
+ * Shallow equality for two route-param records. Same-reference is a fast path;
7
+ * otherwise compares key count then each value.
8
+ */
9
+ export function paramsEqual(
10
+ a: Record<string, string>,
11
+ b: Record<string, string>,
12
+ ): boolean {
13
+ if (a === b) return true;
14
+
15
+ const keysA = Object.keys(a);
16
+ if (keysA.length !== Object.keys(b).length) return false;
17
+
18
+ for (const key of keysA) {
19
+ if (a[key] !== b[key]) return false;
20
+ }
21
+
22
+ return true;
23
+ }