@rangojs/router 0.0.0-experimental.7 → 0.0.0-experimental.70

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 (307) hide show
  1. package/AGENTS.md +9 -0
  2. package/README.md +942 -4
  3. package/dist/bin/rango.js +1689 -0
  4. package/dist/vite/index.js +4951 -930
  5. package/package.json +70 -60
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +294 -0
  8. package/skills/caching/SKILL.md +93 -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 +167 -0
  13. package/skills/hooks/SKILL.md +334 -72
  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 +92 -31
  18. package/skills/loader/SKILL.md +404 -44
  19. package/skills/middleware/SKILL.md +173 -34
  20. package/skills/mime-routes/SKILL.md +128 -0
  21. package/skills/parallel/SKILL.md +204 -1
  22. package/skills/prerender/SKILL.md +685 -0
  23. package/skills/rango/SKILL.md +85 -16
  24. package/skills/response-routes/SKILL.md +411 -0
  25. package/skills/route/SKILL.md +257 -14
  26. package/skills/router-setup/SKILL.md +210 -32
  27. package/skills/tailwind/SKILL.md +129 -0
  28. package/skills/theme/SKILL.md +9 -8
  29. package/skills/typesafety/SKILL.md +328 -89
  30. package/skills/use-cache/SKILL.md +324 -0
  31. package/src/__internal.ts +102 -4
  32. package/src/bin/rango.ts +321 -0
  33. package/src/browser/action-coordinator.ts +97 -0
  34. package/src/browser/action-response-classifier.ts +99 -0
  35. package/src/browser/app-version.ts +14 -0
  36. package/src/browser/event-controller.ts +92 -64
  37. package/src/browser/history-state.ts +80 -0
  38. package/src/browser/intercept-utils.ts +52 -0
  39. package/src/browser/link-interceptor.ts +24 -4
  40. package/src/browser/logging.ts +55 -0
  41. package/src/browser/merge-segment-loaders.ts +20 -12
  42. package/src/browser/navigation-bridge.ts +296 -558
  43. package/src/browser/navigation-client.ts +179 -69
  44. package/src/browser/navigation-store.ts +73 -55
  45. package/src/browser/navigation-transaction.ts +297 -0
  46. package/src/browser/network-error-handler.ts +61 -0
  47. package/src/browser/partial-update.ts +328 -313
  48. package/src/browser/prefetch/cache.ts +206 -0
  49. package/src/browser/prefetch/fetch.ts +150 -0
  50. package/src/browser/prefetch/observer.ts +65 -0
  51. package/src/browser/prefetch/policy.ts +48 -0
  52. package/src/browser/prefetch/queue.ts +160 -0
  53. package/src/browser/prefetch/resource-ready.ts +77 -0
  54. package/src/browser/rango-state.ts +112 -0
  55. package/src/browser/react/Link.tsx +230 -74
  56. package/src/browser/react/NavigationProvider.tsx +87 -11
  57. package/src/browser/react/context.ts +11 -0
  58. package/src/browser/react/filter-segment-order.ts +11 -0
  59. package/src/browser/react/index.ts +12 -12
  60. package/src/browser/react/location-state-shared.ts +95 -53
  61. package/src/browser/react/location-state.ts +60 -15
  62. package/src/browser/react/mount-context.ts +6 -1
  63. package/src/browser/react/nonce-context.ts +23 -0
  64. package/src/browser/react/shallow-equal.ts +27 -0
  65. package/src/browser/react/use-action.ts +29 -51
  66. package/src/browser/react/use-client-cache.ts +5 -3
  67. package/src/browser/react/use-handle.ts +30 -126
  68. package/src/browser/react/use-href.tsx +2 -2
  69. package/src/browser/react/use-link-status.ts +6 -5
  70. package/src/browser/react/use-navigation.ts +22 -63
  71. package/src/browser/react/use-params.ts +65 -0
  72. package/src/browser/react/use-pathname.ts +47 -0
  73. package/src/browser/react/use-router.ts +76 -0
  74. package/src/browser/react/use-search-params.ts +56 -0
  75. package/src/browser/react/use-segments.ts +80 -97
  76. package/src/browser/response-adapter.ts +73 -0
  77. package/src/browser/rsc-router.tsx +214 -58
  78. package/src/browser/scroll-restoration.ts +127 -52
  79. package/src/browser/segment-reconciler.ts +221 -0
  80. package/src/browser/segment-structure-assert.ts +16 -0
  81. package/src/browser/server-action-bridge.ts +510 -603
  82. package/src/browser/shallow.ts +6 -1
  83. package/src/browser/types.ts +141 -48
  84. package/src/browser/validate-redirect-origin.ts +29 -0
  85. package/src/build/generate-manifest.ts +235 -24
  86. package/src/build/generate-route-types.ts +39 -0
  87. package/src/build/index.ts +13 -0
  88. package/src/build/route-trie.ts +265 -0
  89. package/src/build/route-types/ast-helpers.ts +25 -0
  90. package/src/build/route-types/ast-route-extraction.ts +98 -0
  91. package/src/build/route-types/codegen.ts +102 -0
  92. package/src/build/route-types/include-resolution.ts +418 -0
  93. package/src/build/route-types/param-extraction.ts +48 -0
  94. package/src/build/route-types/per-module-writer.ts +128 -0
  95. package/src/build/route-types/router-processing.ts +618 -0
  96. package/src/build/route-types/scan-filter.ts +85 -0
  97. package/src/build/runtime-discovery.ts +231 -0
  98. package/src/cache/background-task.ts +34 -0
  99. package/src/cache/cache-key-utils.ts +44 -0
  100. package/src/cache/cache-policy.ts +125 -0
  101. package/src/cache/cache-runtime.ts +342 -0
  102. package/src/cache/cache-scope.ts +167 -309
  103. package/src/cache/cf/cf-cache-store.ts +571 -17
  104. package/src/cache/cf/index.ts +13 -3
  105. package/src/cache/document-cache.ts +116 -77
  106. package/src/cache/handle-capture.ts +81 -0
  107. package/src/cache/handle-snapshot.ts +41 -0
  108. package/src/cache/index.ts +1 -15
  109. package/src/cache/memory-segment-store.ts +191 -13
  110. package/src/cache/profile-registry.ts +73 -0
  111. package/src/cache/read-through-swr.ts +134 -0
  112. package/src/cache/segment-codec.ts +256 -0
  113. package/src/cache/taint.ts +153 -0
  114. package/src/cache/types.ts +72 -122
  115. package/src/client.rsc.tsx +3 -1
  116. package/src/client.tsx +105 -179
  117. package/src/component-utils.ts +4 -4
  118. package/src/components/DefaultDocument.tsx +5 -1
  119. package/src/context-var.ts +156 -0
  120. package/src/debug.ts +19 -9
  121. package/src/errors.ts +108 -2
  122. package/src/handle.ts +55 -29
  123. package/src/handles/MetaTags.tsx +73 -20
  124. package/src/handles/breadcrumbs.ts +66 -0
  125. package/src/handles/index.ts +1 -0
  126. package/src/handles/meta.ts +30 -13
  127. package/src/host/cookie-handler.ts +21 -15
  128. package/src/host/errors.ts +8 -8
  129. package/src/host/index.ts +4 -7
  130. package/src/host/pattern-matcher.ts +27 -27
  131. package/src/host/router.ts +61 -39
  132. package/src/host/testing.ts +8 -8
  133. package/src/host/types.ts +15 -7
  134. package/src/host/utils.ts +1 -1
  135. package/src/href-client.ts +119 -29
  136. package/src/index.rsc.ts +155 -19
  137. package/src/index.ts +223 -30
  138. package/src/internal-debug.ts +11 -0
  139. package/src/loader.rsc.ts +26 -157
  140. package/src/loader.ts +27 -10
  141. package/src/network-error-thrower.tsx +3 -1
  142. package/src/outlet-provider.tsx +45 -0
  143. package/src/prerender/param-hash.ts +37 -0
  144. package/src/prerender/store.ts +186 -0
  145. package/src/prerender.ts +524 -0
  146. package/src/reverse.ts +351 -0
  147. package/src/root-error-boundary.tsx +41 -29
  148. package/src/route-content-wrapper.tsx +7 -4
  149. package/src/route-definition/dsl-helpers.ts +982 -0
  150. package/src/route-definition/helper-factories.ts +200 -0
  151. package/src/route-definition/helpers-types.ts +434 -0
  152. package/src/route-definition/index.ts +55 -0
  153. package/src/route-definition/redirect.ts +101 -0
  154. package/src/route-definition/resolve-handler-use.ts +149 -0
  155. package/src/route-definition.ts +1 -1428
  156. package/src/route-map-builder.ts +217 -123
  157. package/src/route-name.ts +53 -0
  158. package/src/route-types.ts +70 -8
  159. package/src/router/content-negotiation.ts +215 -0
  160. package/src/router/debug-manifest.ts +72 -0
  161. package/src/router/error-handling.ts +9 -9
  162. package/src/router/find-match.ts +160 -0
  163. package/src/router/handler-context.ts +435 -86
  164. package/src/router/intercept-resolution.ts +402 -0
  165. package/src/router/lazy-includes.ts +237 -0
  166. package/src/router/loader-resolution.ts +356 -128
  167. package/src/router/logging.ts +251 -0
  168. package/src/router/manifest.ts +154 -35
  169. package/src/router/match-api.ts +555 -0
  170. package/src/router/match-context.ts +5 -3
  171. package/src/router/match-handlers.ts +440 -0
  172. package/src/router/match-middleware/background-revalidation.ts +108 -93
  173. package/src/router/match-middleware/cache-lookup.ts +459 -10
  174. package/src/router/match-middleware/cache-store.ts +98 -26
  175. package/src/router/match-middleware/intercept-resolution.ts +57 -17
  176. package/src/router/match-middleware/segment-resolution.ts +80 -6
  177. package/src/router/match-pipelines.ts +10 -45
  178. package/src/router/match-result.ts +55 -33
  179. package/src/router/metrics.ts +240 -15
  180. package/src/router/middleware-cookies.ts +55 -0
  181. package/src/router/middleware-types.ts +220 -0
  182. package/src/router/middleware.ts +324 -369
  183. package/src/router/navigation-snapshot.ts +182 -0
  184. package/src/router/pattern-matching.ts +211 -43
  185. package/src/router/prerender-match.ts +502 -0
  186. package/src/router/preview-match.ts +98 -0
  187. package/src/router/request-classification.ts +310 -0
  188. package/src/router/revalidation.ts +137 -38
  189. package/src/router/route-snapshot.ts +245 -0
  190. package/src/router/router-context.ts +41 -21
  191. package/src/router/router-interfaces.ts +484 -0
  192. package/src/router/router-options.ts +618 -0
  193. package/src/router/router-registry.ts +24 -0
  194. package/src/router/segment-resolution/fresh.ts +743 -0
  195. package/src/router/segment-resolution/helpers.ts +268 -0
  196. package/src/router/segment-resolution/loader-cache.ts +199 -0
  197. package/src/router/segment-resolution/revalidation.ts +1373 -0
  198. package/src/router/segment-resolution/static-store.ts +67 -0
  199. package/src/router/segment-resolution.ts +21 -0
  200. package/src/router/segment-wrappers.ts +291 -0
  201. package/src/router/telemetry-otel.ts +299 -0
  202. package/src/router/telemetry.ts +300 -0
  203. package/src/router/timeout.ts +148 -0
  204. package/src/router/trie-matching.ts +239 -0
  205. package/src/router/types.ts +78 -3
  206. package/src/router.ts +740 -4252
  207. package/src/rsc/handler-context.ts +45 -0
  208. package/src/rsc/handler.ts +907 -797
  209. package/src/rsc/helpers.ts +140 -6
  210. package/src/rsc/index.ts +0 -20
  211. package/src/rsc/loader-fetch.ts +229 -0
  212. package/src/rsc/manifest-init.ts +90 -0
  213. package/src/rsc/nonce.ts +14 -0
  214. package/src/rsc/origin-guard.ts +141 -0
  215. package/src/rsc/progressive-enhancement.ts +391 -0
  216. package/src/rsc/response-error.ts +37 -0
  217. package/src/rsc/response-route-handler.ts +347 -0
  218. package/src/rsc/rsc-rendering.ts +246 -0
  219. package/src/rsc/runtime-warnings.ts +42 -0
  220. package/src/rsc/server-action.ts +356 -0
  221. package/src/rsc/ssr-setup.ts +128 -0
  222. package/src/rsc/types.ts +46 -11
  223. package/src/search-params.ts +230 -0
  224. package/src/segment-system.tsx +165 -17
  225. package/src/server/context.ts +315 -58
  226. package/src/server/cookie-store.ts +190 -0
  227. package/src/server/fetchable-loader-store.ts +37 -0
  228. package/src/server/handle-store.ts +113 -15
  229. package/src/server/loader-registry.ts +24 -64
  230. package/src/server/request-context.ts +607 -81
  231. package/src/server.ts +35 -130
  232. package/src/ssr/index.tsx +103 -30
  233. package/src/static-handler.ts +126 -0
  234. package/src/theme/ThemeProvider.tsx +21 -15
  235. package/src/theme/ThemeScript.tsx +5 -5
  236. package/src/theme/constants.ts +5 -2
  237. package/src/theme/index.ts +4 -14
  238. package/src/theme/theme-context.ts +4 -30
  239. package/src/theme/theme-script.ts +21 -18
  240. package/src/types/boundaries.ts +158 -0
  241. package/src/types/cache-types.ts +198 -0
  242. package/src/types/error-types.ts +192 -0
  243. package/src/types/global-namespace.ts +100 -0
  244. package/src/types/handler-context.ts +791 -0
  245. package/src/types/index.ts +88 -0
  246. package/src/types/loader-types.ts +210 -0
  247. package/src/types/route-config.ts +170 -0
  248. package/src/types/route-entry.ts +109 -0
  249. package/src/types/segments.ts +150 -0
  250. package/src/types.ts +1 -1623
  251. package/src/urls/include-helper.ts +197 -0
  252. package/src/urls/index.ts +53 -0
  253. package/src/urls/path-helper-types.ts +346 -0
  254. package/src/urls/path-helper.ts +364 -0
  255. package/src/urls/pattern-types.ts +107 -0
  256. package/src/urls/response-types.ts +116 -0
  257. package/src/urls/type-extraction.ts +372 -0
  258. package/src/urls/urls-function.ts +98 -0
  259. package/src/urls.ts +1 -802
  260. package/src/use-loader.tsx +161 -81
  261. package/src/vite/discovery/bundle-postprocess.ts +181 -0
  262. package/src/vite/discovery/discover-routers.ts +348 -0
  263. package/src/vite/discovery/prerender-collection.ts +439 -0
  264. package/src/vite/discovery/route-types-writer.ts +258 -0
  265. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  266. package/src/vite/discovery/state.ts +117 -0
  267. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  268. package/src/vite/index.ts +15 -1129
  269. package/src/vite/plugin-types.ts +103 -0
  270. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  271. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  272. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  273. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -53
  274. package/src/vite/plugins/expose-id-utils.ts +299 -0
  275. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  276. package/src/vite/plugins/expose-ids/handler-transform.ts +209 -0
  277. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  278. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  279. package/src/vite/plugins/expose-ids/types.ts +45 -0
  280. package/src/vite/plugins/expose-internal-ids.ts +786 -0
  281. package/src/vite/plugins/performance-tracks.ts +88 -0
  282. package/src/vite/plugins/refresh-cmd.ts +127 -0
  283. package/src/vite/plugins/use-cache-transform.ts +323 -0
  284. package/src/vite/plugins/version-injector.ts +83 -0
  285. package/src/vite/plugins/version-plugin.ts +266 -0
  286. package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
  287. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  288. package/src/vite/rango.ts +462 -0
  289. package/src/vite/router-discovery.ts +918 -0
  290. package/src/vite/utils/ast-handler-extract.ts +517 -0
  291. package/src/vite/utils/banner.ts +36 -0
  292. package/src/vite/utils/bundle-analysis.ts +137 -0
  293. package/src/vite/utils/manifest-utils.ts +70 -0
  294. package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
  295. package/src/vite/utils/prerender-utils.ts +207 -0
  296. package/src/vite/utils/shared-utils.ts +170 -0
  297. package/CLAUDE.md +0 -43
  298. package/src/browser/lru-cache.ts +0 -69
  299. package/src/browser/request-controller.ts +0 -164
  300. package/src/cache/memory-store.ts +0 -253
  301. package/src/href-context.ts +0 -33
  302. package/src/href.ts +0 -255
  303. package/src/server/route-manifest-cache.ts +0 -173
  304. package/src/vite/expose-handle-id.ts +0 -209
  305. package/src/vite/expose-loader-id.ts +0 -426
  306. package/src/vite/expose-location-state-id.ts +0 -177
  307. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -9,169 +9,55 @@
9
9
  * - Forgiving API: if middleware doesn't return, original response is used
10
10
  */
11
11
 
12
- import type { RouterEnv } from "../types.js";
13
-
14
- /**
15
- * Helper type to extract Variables from RouterEnv
16
- * Uses 0 extends 1 & TEnv to detect `any` type and fall back to Record<string, unknown>
17
- */
18
- type ExtractVariables<TEnv> = 0 extends 1 & TEnv
19
- ? Record<string, unknown> // TEnv is any
20
- : TEnv extends RouterEnv<unknown, infer V>
21
- ? V
22
- : Record<string, unknown>;
23
-
24
- /**
25
- * Get variable function type
26
- */
27
- type GetVariableFn<TEnv> = <K extends keyof ExtractVariables<TEnv>>(
28
- key: K
29
- ) => ExtractVariables<TEnv>[K];
30
-
31
- /**
32
- * Set variable function type
33
- */
34
- type SetVariableFn<TEnv> = <K extends keyof ExtractVariables<TEnv>>(
35
- key: K,
36
- value: ExtractVariables<TEnv>[K]
37
- ) => void;
12
+ import { contextGet, contextSet } from "../context-var.js";
13
+ import type {
14
+ CollectedMiddleware,
15
+ MiddlewareCollectableEntry,
16
+ MiddlewareContext,
17
+ MiddlewareEntry,
18
+ MiddlewareFn,
19
+ ResponseHolder,
20
+ } from "./middleware-types.js";
21
+ import { _getRequestContext } from "../server/request-context.js";
22
+ import { isAutoGeneratedRouteName } from "../route-name.js";
23
+ import { appendMetric, createMetricsStore } from "./metrics.js";
24
+ import { stripInternalParams } from "./handler-context.js";
25
+
26
+ // Re-export types and cookie utilities for backward compatibility
27
+ export type {
28
+ CookieOptions,
29
+ CollectedMiddleware,
30
+ MiddlewareCollectableEntry,
31
+ MiddlewareContext,
32
+ MiddlewareEntry,
33
+ MiddlewareFn,
34
+ ResponseHolder,
35
+ } from "./middleware-types.js";
36
+ export { parseCookies, serializeCookie } from "./middleware-cookies.js";
37
+
38
+ const MIDDLEWARE_METRIC_DEPTH = 1;
39
+ /** Ignore post-next() durations below this threshold (measurement noise). */
40
+ const POST_METRIC_MIN_DURATION_MS = 0.01;
41
+
42
+ function getMiddlewareMetricBase<TEnv>(
43
+ entry: MiddlewareEntry<TEnv>,
44
+ ordinal: number,
45
+ ): string {
46
+ const handlerName = entry.handler.name?.trim();
47
+ const scope = entry.pattern ?? "*";
38
48
 
39
- /**
40
- * Cookie options for setting cookies
41
- */
42
- export interface CookieOptions {
43
- domain?: string;
44
- path?: string;
45
- maxAge?: number;
46
- expires?: Date;
47
- httpOnly?: boolean;
48
- secure?: boolean;
49
- sameSite?: "strict" | "lax" | "none";
50
- }
49
+ if (handlerName) {
50
+ return `${handlerName}@${scope}`;
51
+ }
51
52
 
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, Record<string, string> for global middleware)
57
- */
58
- export interface MiddlewareContext<
59
- TEnv = any,
60
- TParams = Record<string, string>,
61
- > {
62
- /** Original request */
63
- request: Request;
64
-
65
- /** Parsed URL */
66
- url: URL;
67
-
68
- /** URL pathname */
69
- pathname: string;
70
-
71
- /** URL search params */
72
- searchParams: URLSearchParams;
73
-
74
- /** Platform bindings (Cloudflare, etc.) */
75
- env: TEnv extends RouterEnv<infer B, unknown> ? B : {};
76
-
77
- /** URL params extracted from route/middleware pattern */
78
- params: TParams;
79
-
80
- /**
81
- * Response object - available immediately via stub, real response after `await next()`
82
- *
83
- * Headers set before `next()` are merged into the final response.
84
- * Can be used to modify headers directly like Hono's `c.res`.
85
- *
86
- * @example
87
- * ```typescript
88
- * middleware(async (ctx, next) => {
89
- * // Set headers BEFORE next() - will be merged into final response
90
- * ctx.res.headers.set('X-Request-Id', generateId());
91
- *
92
- * await next();
93
- *
94
- * // Set headers AFTER next() - applied directly
95
- * ctx.res.headers.set('X-Custom', 'value');
96
- * // No return needed!
97
- * });
98
- * ```
99
- */
100
- res: Response;
101
-
102
- /** Get a cookie value */
103
- cookie(name: string): string | undefined;
104
-
105
- /** Get all cookies as object */
106
- cookies(): Record<string, string>;
107
-
108
- /** Set a cookie on the response */
109
- setCookie(name: string, value: string, options?: CookieOptions): void;
110
-
111
- /** Delete a cookie */
112
- deleteCookie(
113
- name: string,
114
- options?: Pick<CookieOptions, "domain" | "path">
115
- ): void;
116
-
117
- /** Get a context variable (shared with route handlers) */
118
- get: GetVariableFn<TEnv>;
119
-
120
- /** Set a context variable (shared with route handlers) */
121
- set: SetVariableFn<TEnv>;
122
-
123
- /**
124
- * Set a response header - can be called before or after `next()`
125
- *
126
- * When called before `next()`, headers are queued and merged into the final response.
127
- * When called after `next()`, headers are set directly on the response.
128
- * Shorthand for `ctx.res.headers.set()`.
129
- */
130
- header(name: string, value: string): void;
53
+ return `${scope}#${ordinal + 1}`;
131
54
  }
132
55
 
133
- /**
134
- * Middleware function signature
135
- *
136
- * @template TEnv - Environment type - defaults to any for internal flexibility
137
- * @template TParams - URL params type (typed for route middleware)
138
- *
139
- * When using middleware with global augmentation (RSCRouter.Env), explicitly
140
- * annotate your middleware functions, or the types will be inferred from context:
141
- *
142
- * @example
143
- * ```typescript
144
- * // With explicit annotation (recommended for reusable middleware)
145
- * const authMiddleware: MiddlewareFn<AppEnv> = async (ctx, next) => {...}
146
- *
147
- * // Types inferred from router.use() call
148
- * router.use((ctx, next) => {...}) // ctx is typed from router's TEnv
149
- * ```
150
- */
151
- export type MiddlewareFn<TEnv = any, TParams = Record<string, string>> = (
152
- ctx: MiddlewareContext<TEnv, TParams>,
153
- next: () => Promise<Response>
154
- ) => Response | void | Promise<Response | void>;
155
-
156
- /**
157
- * Stored middleware entry with pattern matching info
158
- * @internal - uses any for internal flexibility
159
- */
160
- export interface MiddlewareEntry<TEnv = any> {
161
- /** Original pattern string */
162
- pattern: string | null;
163
-
164
- /** Compiled regex for matching */
165
- regex: RegExp | null;
166
-
167
- /** Param names extracted from pattern */
168
- paramNames: string[];
169
-
170
- /** The middleware function */
171
- handler: MiddlewareFn<TEnv>;
172
-
173
- /** Mount prefix this middleware is scoped to (null = global) */
174
- mountPrefix: string | null;
56
+ function getMiddlewareMetricLabel<TEnv>(
57
+ entry: MiddlewareEntry<TEnv>,
58
+ ordinal: number,
59
+ ): string {
60
+ return `middleware:${getMiddlewareMetricBase(entry, ordinal)}`;
175
61
  }
176
62
 
177
63
  /**
@@ -231,7 +117,7 @@ function escapeRegex(str: string): string {
231
117
  export function extractParams(
232
118
  pathname: string,
233
119
  regex: RegExp,
234
- paramNames: string[]
120
+ paramNames: string[],
235
121
  ): Record<string, string> {
236
122
  const match = pathname.match(regex);
237
123
  if (!match) return {};
@@ -243,151 +129,134 @@ export function extractParams(
243
129
  return params;
244
130
  }
245
131
 
246
- /**
247
- * Parse cookies from Cookie header
248
- */
249
- export function parseCookies(
250
- cookieHeader: string | null
251
- ): Record<string, string> {
252
- if (!cookieHeader) return {};
253
-
254
- const cookies: Record<string, string> = {};
255
- const pairs = cookieHeader.split(";");
256
-
257
- for (const pair of pairs) {
258
- const [name, ...rest] = pair.trim().split("=");
259
- if (name) {
260
- cookies[name] = decodeURIComponent(rest.join("="));
261
- }
262
- }
263
-
264
- return cookies;
265
- }
266
-
267
- /**
268
- * Serialize a cookie for Set-Cookie header
269
- */
270
- export function serializeCookie(
271
- name: string,
272
- value: string,
273
- options: CookieOptions = {}
274
- ): string {
275
- let cookie = `${encodeURIComponent(name)}=${encodeURIComponent(value)}`;
276
-
277
- if (options.domain) cookie += `; Domain=${options.domain}`;
278
- if (options.path) cookie += `; Path=${options.path}`;
279
- if (options.maxAge !== undefined) cookie += `; Max-Age=${options.maxAge}`;
280
- if (options.expires) cookie += `; Expires=${options.expires.toUTCString()}`;
281
- if (options.httpOnly) cookie += "; HttpOnly";
282
- if (options.secure) cookie += "; Secure";
283
- if (options.sameSite) cookie += `; SameSite=${options.sameSite}`;
284
-
285
- return cookie;
286
- }
287
-
288
- /**
289
- * Mutable response holder - allows ctx.res to be updated after next() is called
290
- */
291
- export interface ResponseHolder {
292
- response: Response | null;
293
- }
294
-
295
132
  /**
296
133
  * Create middleware context
297
134
  *
298
135
  * Note: The implementation uses runtime values while the interface provides
299
136
  * compile-time type safety. The env/get/set types are resolved at call sites
300
- * via conditional types based on TEnv extending RouterEnv.
137
+ * via conditional types based on TEnv from createRouter<TBindings>().
301
138
  */
302
139
  export function createMiddlewareContext<TEnv>(
303
140
  request: Request,
304
141
  env: TEnv,
305
142
  params: Record<string, string>,
306
143
  variables: Record<string, unknown>,
307
- responseHolder: ResponseHolder
144
+ responseHolder: ResponseHolder,
145
+ reverse?: (
146
+ name: string,
147
+ params?: Record<string, string>,
148
+ search?: Record<string, unknown>,
149
+ ) => string,
308
150
  ): MiddlewareContext<TEnv> {
309
- const url = new URL(request.url);
310
- const cookieHeader = request.headers.get("Cookie");
311
- let parsedCookies: Record<string, string> | null = null;
312
-
151
+ const url = stripInternalParams(new URL(request.url));
152
+
153
+ // Track the initial response to detect pre/post-next() phase.
154
+ // Before next(): responseHolder.response === initialResponse (the stub).
155
+ // After next(): responseHolder.response is the real downstream response.
156
+ const initialResponse = responseHolder.response;
157
+ const isPreNext = () => responseHolder.response === initialResponse;
158
+
159
+ // Delegation strategy for RequestContext (reqCtx):
160
+ // - res getter: before next() returns shared reqCtx stub; after next() returns
161
+ // the real downstream response.
162
+ // - header(): before next() delegates to reqCtx; after next() writes to the
163
+ // real downstream response.
164
+ // Cookie operations are handled by the standalone cookies() function which
165
+ // delegates to the shared RequestContext internally.
313
166
  // The runtime implementation - types are enforced at call sites via MiddlewareContext<TEnv>
167
+ // Internal helper: resolve the current response (stub before next(), real after).
168
+ // Not exposed on the public MiddlewareContext type — use ctx.headers instead.
169
+ const getResponse = (): Response => {
170
+ if (isPreNext()) {
171
+ const reqCtx = _getRequestContext();
172
+ if (reqCtx) return reqCtx.res;
173
+ }
174
+ if (!responseHolder.response) {
175
+ throw new Error(
176
+ "Response is not available - responseHolder was not initialized",
177
+ );
178
+ }
179
+ return responseHolder.response;
180
+ };
181
+
314
182
  return {
315
183
  request,
316
184
  url,
185
+ originalUrl: new URL(request.url),
317
186
  pathname: url.pathname,
318
187
  searchParams: url.searchParams,
319
188
  env: env as MiddlewareContext<TEnv>["env"],
320
189
  params,
321
-
322
- // res getter - returns the stub or real response (always available)
323
- get res(): Response {
324
- if (!responseHolder.response) {
325
- throw new Error(
326
- "ctx.res is not available - responseHolder was not initialized"
327
- );
328
- }
329
- return responseHolder.response;
190
+ // Getter: re-derives from request context on each access so that global
191
+ // middleware sees the matched route name after await next().
192
+ get routeName(): MiddlewareContext<TEnv>["routeName"] {
193
+ const reqCtx = _getRequestContext();
194
+ const raw = reqCtx?._routeName;
195
+ return (
196
+ raw && !isAutoGeneratedRouteName(raw) ? raw : undefined
197
+ ) as MiddlewareContext<TEnv>["routeName"];
330
198
  },
331
199
 
332
- // res setter - allows middleware to replace the response
333
- set res(response: Response) {
334
- responseHolder.response = response;
200
+ get headers(): Headers {
201
+ return getResponse().headers;
335
202
  },
336
203
 
337
- cookie(name: string): string | undefined {
338
- if (!parsedCookies) {
339
- parsedCookies = parseCookies(cookieHeader);
340
- }
341
- return parsedCookies[name];
342
- },
204
+ get: ((keyOrVar: any) =>
205
+ contextGet(variables, keyOrVar)) as MiddlewareContext<TEnv>["get"],
343
206
 
344
- cookies(): Record<string, string> {
345
- if (!parsedCookies) {
346
- parsedCookies = parseCookies(cookieHeader);
207
+ set: ((keyOrVar: any, value: unknown, options?: any) => {
208
+ contextSet(variables, keyOrVar, value, options);
209
+ }) as MiddlewareContext<TEnv>["set"],
210
+ header(name: string, value: string): void {
211
+ // Before next(): delegate to shared RequestContext stub
212
+ if (isPreNext()) {
213
+ const reqCtx = _getRequestContext();
214
+ if (reqCtx) {
215
+ reqCtx.header(name, value);
216
+ return;
217
+ }
347
218
  }
348
- return { ...parsedCookies };
349
- },
350
-
351
- setCookie(name: string, value: string, options?: CookieOptions): void {
219
+ // After next() or standalone: write to current response
352
220
  if (!responseHolder.response) {
353
221
  throw new Error(
354
- "ctx.setCookie() is not available - responseHolder was not initialized"
222
+ "ctx.header() is not available - responseHolder was not initialized",
355
223
  );
356
224
  }
357
- responseHolder.response.headers.append(
358
- "Set-Cookie",
359
- serializeCookie(name, value, options)
360
- );
225
+ responseHolder.response.headers.set(name, value);
361
226
  },
362
227
 
363
- deleteCookie(
364
- name: string,
365
- options?: Pick<CookieOptions, "domain" | "path">
366
- ): void {
367
- if (!responseHolder.response) {
228
+ get theme(): MiddlewareContext<TEnv>["theme"] {
229
+ return _getRequestContext()?.theme;
230
+ },
231
+
232
+ get setTheme(): MiddlewareContext<TEnv>["setTheme"] {
233
+ return _getRequestContext()?.setTheme;
234
+ },
235
+
236
+ setLocationState(entries) {
237
+ const reqCtx = _getRequestContext();
238
+ if (!reqCtx) {
368
239
  throw new Error(
369
- "ctx.deleteCookie() is not available - responseHolder was not initialized"
240
+ "setLocationState() is not available outside a request context",
370
241
  );
371
242
  }
372
- responseHolder.response.headers.append(
373
- "Set-Cookie",
374
- serializeCookie(name, "", { ...options, maxAge: 0 })
375
- );
243
+ reqCtx.setLocationState(entries);
376
244
  },
377
245
 
378
- get: ((key: string) => variables[key]) as MiddlewareContext<TEnv>["get"],
379
-
380
- set: ((key: string, value: unknown) => {
381
- variables[key] = value;
382
- }) as MiddlewareContext<TEnv>["set"],
383
-
384
- header(name: string, value: string): void {
385
- if (!responseHolder.response) {
246
+ reverse:
247
+ reverse ??
248
+ ((name: string) => {
386
249
  throw new Error(
387
- "ctx.header() is not available - responseHolder was not initialized"
250
+ `ctx.reverse() is not available - route map was not provided to middleware context`,
388
251
  );
252
+ }),
253
+
254
+ debugPerformance(): void {
255
+ const reqCtx = _getRequestContext();
256
+ if (reqCtx) {
257
+ reqCtx._debugPerformance = true;
258
+ reqCtx._metricsStore ??= createMetricsStore(true);
389
259
  }
390
- responseHolder.response.headers.set(name, value);
391
260
  },
392
261
  };
393
262
  }
@@ -398,7 +267,7 @@ export function createMiddlewareContext<TEnv>(
398
267
  */
399
268
  export function matchMiddleware<TEnv>(
400
269
  pathname: string,
401
- entries: MiddlewareEntry<TEnv>[]
270
+ entries: MiddlewareEntry<TEnv>[],
402
271
  ): Array<{ entry: MiddlewareEntry<TEnv>; params: Record<string, string> }> {
403
272
  const matches: Array<{
404
273
  entry: MiddlewareEntry<TEnv>;
@@ -427,9 +296,9 @@ export function matchMiddleware<TEnv>(
427
296
  *
428
297
  * Features:
429
298
  * - `await next()` returns actual Response
430
- * - `ctx.res` available after `await next()` (like Hono's `c.res`)
431
- * - `ctx.header()` shorthand for setting headers
432
- * - Forgiving: if middleware doesn't return, uses `ctx.res`
299
+ * - `ctx.headers` available before and after `await next()`
300
+ * - `ctx.header()` shorthand for setting a single header
301
+ * - Forgiving: if middleware doesn't return, uses the downstream response
433
302
  * - Short-circuit: return Response to stop chain
434
303
  * - Error catching: try/catch around `next()` works
435
304
  */
@@ -441,7 +310,12 @@ export async function executeMiddleware<TEnv>(
441
310
  request: Request,
442
311
  env: TEnv,
443
312
  variables: Record<string, any>,
444
- finalHandler: () => Promise<Response>
313
+ finalHandler: () => Promise<Response>,
314
+ reverse?: (
315
+ name: string,
316
+ params?: Record<string, string>,
317
+ search?: Record<string, unknown>,
318
+ ) => string,
445
319
  ): Promise<Response> {
446
320
  let index = 0;
447
321
 
@@ -455,8 +329,8 @@ export async function executeMiddleware<TEnv>(
455
329
  // End of chain - call actual RSC handler
456
330
  const response = await finalHandler();
457
331
 
458
- // Merge headers set on stub into the real response
459
- // Use append for Set-Cookie to preserve multiple cookies
332
+ // Merge headers set on stub into the real response.
333
+ // Use append for Set-Cookie to preserve multiple cookies.
460
334
  const mergedHeaders = new Headers(response.headers);
461
335
  stubResponse.headers.forEach((value, name) => {
462
336
  if (name.toLowerCase() === "set-cookie") {
@@ -465,6 +339,26 @@ export async function executeMiddleware<TEnv>(
465
339
  mergedHeaders.set(name, value);
466
340
  }
467
341
  });
342
+ // Also merge shared RequestContext stub (cookies written via cookies().set()).
343
+ // Dedup Set-Cookie: an inner executeMiddleware (route-level middleware)
344
+ // may have already merged the same reqCtx cookies into the response.
345
+ const reqCtx = _getRequestContext();
346
+ if (reqCtx) {
347
+ const stubCookies = reqCtx.res.headers.getSetCookie();
348
+ if (stubCookies.length > 0) {
349
+ const existing = new Set(mergedHeaders.getSetCookie());
350
+ for (const cookie of stubCookies) {
351
+ if (!existing.has(cookie)) {
352
+ mergedHeaders.append("set-cookie", cookie);
353
+ }
354
+ }
355
+ }
356
+ reqCtx.res.headers.forEach((value, name) => {
357
+ if (name !== "set-cookie" && !mergedHeaders.has(name)) {
358
+ mergedHeaders.set(name, value);
359
+ }
360
+ });
361
+ }
468
362
 
469
363
  // Clone response with merged headers (mutable for post-next() modifications)
470
364
  responseHolder.response = new Response(response.body, {
@@ -476,29 +370,122 @@ export async function executeMiddleware<TEnv>(
476
370
  return responseHolder.response;
477
371
  }
478
372
 
373
+ const middlewareOrdinal = index;
479
374
  const { entry, params } = middlewares[index++];
480
375
  const ctx = createMiddlewareContext(
481
376
  request,
482
377
  env,
483
378
  params,
484
379
  variables,
485
- responseHolder
380
+ responseHolder,
381
+ reverse,
486
382
  );
383
+ const metricStart = performance.now();
384
+ const metricLabel = getMiddlewareMetricLabel(entry, middlewareOrdinal);
385
+ let middlewareFinished = false;
386
+ const finishMiddleware = () => {
387
+ if (!middlewareFinished) {
388
+ middlewareFinished = true;
389
+ appendMetric(
390
+ _getRequestContext()?._metricsStore,
391
+ `${metricLabel}:pre`,
392
+ metricStart,
393
+ performance.now() - metricStart,
394
+ MIDDLEWARE_METRIC_DEPTH,
395
+ );
396
+ }
397
+ };
487
398
 
488
- // Track if next() was called and capture its Promise
489
- // This handles the case where middleware calls next() synchronously without await
399
+ // Track if next() was called and capture its Promise.
400
+ // Guard against double-calling: a second call would re-enter the
401
+ // downstream chain and overwrite responseHolder.response.
490
402
  let nextPromise: Promise<Response> | null = null;
403
+ let nextResolvedAt: number | undefined;
491
404
  const wrappedNext = (): Promise<Response> => {
492
- nextPromise = next();
405
+ if (nextPromise) {
406
+ throw new Error(
407
+ `[@rangojs/router] Middleware called next() more than once.`,
408
+ );
409
+ }
410
+ finishMiddleware();
411
+ const downstream = next();
412
+ nextPromise = downstream.then(
413
+ (res) => {
414
+ nextResolvedAt = performance.now();
415
+ return res;
416
+ },
417
+ (err) => {
418
+ nextResolvedAt = performance.now();
419
+ throw err;
420
+ },
421
+ );
493
422
  return nextPromise;
494
423
  };
495
424
 
496
- const result = await entry.handler(ctx, wrappedNext);
425
+ let result: Response | void;
426
+ try {
427
+ result = await entry.handler(ctx, wrappedNext);
428
+ } catch (error) {
429
+ finishMiddleware();
430
+ throw error;
431
+ }
432
+ finishMiddleware();
433
+
434
+ // Record post-next() processing time when middleware did work after
435
+ // the downstream chain resolved (e.g. adding headers, logging).
436
+ if (nextResolvedAt !== undefined) {
437
+ const postDur = performance.now() - nextResolvedAt;
438
+ if (postDur > POST_METRIC_MIN_DURATION_MS) {
439
+ appendMetric(
440
+ _getRequestContext()?._metricsStore,
441
+ `${metricLabel}:post`,
442
+ nextResolvedAt,
443
+ postDur,
444
+ MIDDLEWARE_METRIC_DEPTH,
445
+ );
446
+ }
447
+ }
497
448
 
498
- // Explicit return takes precedence
449
+ // Explicit return takes precedence (middleware short-circuit).
450
+ // Merge stub headers (from ctx.header before this point) and
451
+ // RequestContext stub headers (from ctx.setCookie) into the
452
+ // returned Response so they are not lost.
499
453
  if (result instanceof Response) {
500
- responseHolder.response = result;
501
- return result;
454
+ const mergedHeaders = new Headers(result.headers);
455
+ stubResponse.headers.forEach((value, name) => {
456
+ if (name.toLowerCase() === "set-cookie") {
457
+ mergedHeaders.append(name, value);
458
+ } else if (!mergedHeaders.has(name)) {
459
+ mergedHeaders.set(name, value);
460
+ }
461
+ });
462
+ // Also merge shared RequestContext stub (cookies written via setCookie).
463
+ // Dedup Set-Cookie: an inner executeMiddleware (route-level middleware)
464
+ // may have already merged the same reqCtx cookies into the response.
465
+ const reqCtx = _getRequestContext();
466
+ if (reqCtx) {
467
+ const stubCookies = reqCtx.res.headers.getSetCookie();
468
+ if (stubCookies.length > 0) {
469
+ const existing = new Set(mergedHeaders.getSetCookie());
470
+ for (const cookie of stubCookies) {
471
+ if (!existing.has(cookie)) {
472
+ mergedHeaders.append("set-cookie", cookie);
473
+ }
474
+ }
475
+ }
476
+ reqCtx.res.headers.forEach((value, name) => {
477
+ if (name !== "set-cookie" && !mergedHeaders.has(name)) {
478
+ mergedHeaders.set(name, value);
479
+ }
480
+ });
481
+ }
482
+ const merged = new Response(result.body, {
483
+ status: result.status,
484
+ statusText: result.statusText,
485
+ headers: mergedHeaders,
486
+ });
487
+ responseHolder.response = merged;
488
+ return merged;
502
489
  }
503
490
 
504
491
  // Warn about unexpected return values (non-Response, non-undefined)
@@ -507,7 +494,7 @@ export async function executeMiddleware<TEnv>(
507
494
  const fnName = entry.handler.name || "(anonymous)";
508
495
  console.warn(
509
496
  `[Middleware] "${fnName}" returned ${typeof result} instead of Response or undefined. ` +
510
- `This return value will be ignored. Did you mean to return a Response?`
497
+ `This return value will be ignored. Did you mean to return a Response?`,
511
498
  );
512
499
  }
513
500
 
@@ -524,7 +511,7 @@ export async function executeMiddleware<TEnv>(
524
511
  `Middleware must call next() or return a Response. ` +
525
512
  `Function: ${fnName}, Pattern: ${entry.pattern ?? "(all)"}
526
513
  Source: ${import.meta.env.DEV ? entry.handler.toString().slice(0, 200) : "(source hidden in production)"}`,
527
- { cause: { url: request.url, fn: entry.handler } }
514
+ { cause: { url: request.url, fn: entry.handler } },
528
515
  );
529
516
  };
530
517
 
@@ -536,64 +523,30 @@ export async function executeMiddleware<TEnv>(
536
523
  throw new Error("No response generated by middleware chain");
537
524
  }
538
525
 
539
- return finalResponse;
540
- }
541
-
542
- /**
543
- * Execute middleware for server actions
544
- *
545
- * Server actions can't return Response directly, but headers/cookies set
546
- * on ctx.res (from getRequestContext().res) will be merged into the final response.
547
- *
548
- * - Runs middleware for auth checks, variable setting, headers, cookies
549
- * - Throws if middleware returns Response (can't short-circuit server action)
550
- */
551
- export async function executeServerActionMiddleware<TEnv>(
552
- middlewares: MiddlewareFn<TEnv>[],
553
- request: Request,
554
- env: TEnv,
555
- params: Record<string, string>,
556
- variables: Record<string, any>,
557
- stubResponse: Response
558
- ): Promise<void> {
559
- if (middlewares.length === 0) {
560
- return;
561
- }
562
-
563
- let index = 0;
564
- const responseHolder: ResponseHolder = { response: stubResponse };
565
-
566
- const next = async (): Promise<Response> => {
567
- if (index >= middlewares.length) {
568
- return stubResponse;
569
- }
570
-
571
- const middleware = middlewares[index++];
572
- const ctx = createMiddlewareContext(
573
- request,
574
- env,
575
- params,
576
- variables,
577
- responseHolder
578
- );
579
-
580
- const result = await middleware(ctx, next);
581
-
582
- // If middleware returned a Response, throw an error
583
- // Server actions can't short-circuit with a Response
584
- if (result instanceof Response) {
585
- throw new Error(
586
- `Loader middleware returned a Response (status: ${result.status}). ` +
587
- `Server actions cannot return Response. ` +
588
- `Use GET-based loader fetching for redirects, or throw an error instead.`
589
- );
526
+ // Final re-merge: capture any RequestContext stub headers added after the
527
+ // last merge point (e.g. cookies().set() called after await next()).
528
+ // The reqCtx stub may have already been partially merged during finalHandler
529
+ // or early-return paths; only append *new* Set-Cookie entries to avoid dupes.
530
+ const reqCtx = _getRequestContext();
531
+ if (reqCtx) {
532
+ const stubCookies = reqCtx.res.headers.getSetCookie();
533
+ if (stubCookies.length > 0) {
534
+ const existingCookies = new Set(finalResponse.headers.getSetCookie());
535
+ for (const cookie of stubCookies) {
536
+ if (!existingCookies.has(cookie)) {
537
+ finalResponse.headers.append("set-cookie", cookie);
538
+ }
539
+ }
590
540
  }
541
+ // Fill in non-cookie headers that aren't already on the response
542
+ reqCtx.res.headers.forEach((value, name) => {
543
+ if (name !== "set-cookie" && !finalResponse.headers.has(name)) {
544
+ finalResponse.headers.set(name, value);
545
+ }
546
+ });
547
+ }
591
548
 
592
- return stubResponse;
593
- };
594
-
595
- await next();
596
- // Headers/cookies set on stubResponse will be merged by the caller
549
+ return finalResponse;
597
550
  }
598
551
 
599
552
  /**
@@ -617,7 +570,12 @@ export async function executeInterceptMiddleware<TEnv>(
617
570
  env: TEnv,
618
571
  params: Record<string, string>,
619
572
  variables: Record<string, any>,
620
- stubResponse: Response
573
+ stubResponse: Response,
574
+ reverse?: (
575
+ name: string,
576
+ params?: Record<string, string>,
577
+ search?: Record<string, unknown>,
578
+ ) => string,
621
579
  ): Promise<Response | null> {
622
580
  if (middlewares.length === 0) {
623
581
  return null;
@@ -640,22 +598,28 @@ export async function executeInterceptMiddleware<TEnv>(
640
598
  env,
641
599
  params,
642
600
  variables,
643
- responseHolder
601
+ responseHolder,
602
+ reverse,
644
603
  );
645
604
 
646
- const result = await middleware(ctx, next);
605
+ let nextCalled = false;
606
+ const guardedNext = (): Promise<Response> => {
607
+ if (nextCalled) {
608
+ throw new Error(
609
+ `[@rangojs/router] Intercept middleware called next() more than once.`,
610
+ );
611
+ }
612
+ nextCalled = true;
613
+ return next();
614
+ };
615
+
616
+ const result = await middleware(ctx, guardedNext);
647
617
 
648
618
  if (result instanceof Response) {
649
619
  earlyResponse = result;
650
620
  return result;
651
621
  }
652
622
 
653
- // Check if middleware replaced ctx.res with a different response
654
- if (responseHolder.response && responseHolder.response !== stubResponse) {
655
- earlyResponse = responseHolder.response;
656
- return earlyResponse;
657
- }
658
-
659
623
  return stubResponse;
660
624
  };
661
625
 
@@ -673,12 +637,14 @@ export async function executeInterceptMiddleware<TEnv>(
673
637
  });
674
638
 
675
639
  if (hasStubHeaders) {
676
- // Clone and merge headers from stub into early response
640
+ // Clone and merge headers from stub into early response.
641
+ // Only fill in missing headers — the returned Response's explicit
642
+ // headers take precedence, matching executeMiddleware behavior.
677
643
  const mergedHeaders = new Headers(response.headers);
678
644
  stubResponse.headers.forEach((value, name) => {
679
645
  if (name.toLowerCase() === "set-cookie") {
680
646
  mergedHeaders.append(name, value);
681
- } else {
647
+ } else if (!mergedHeaders.has(name)) {
682
648
  mergedHeaders.set(name, value);
683
649
  }
684
650
  });
@@ -708,7 +674,12 @@ export async function executeLoaderMiddleware<TEnv>(
708
674
  env: TEnv,
709
675
  params: Record<string, string>,
710
676
  variables: Record<string, any>,
711
- finalHandler: () => Promise<Response>
677
+ finalHandler: () => Promise<Response>,
678
+ reverse?: (
679
+ name: string,
680
+ params?: Record<string, string>,
681
+ search?: Record<string, unknown>,
682
+ ) => string,
712
683
  ): Promise<Response> {
713
684
  if (middlewares.length === 0) {
714
685
  return finalHandler();
@@ -731,27 +702,11 @@ export async function executeLoaderMiddleware<TEnv>(
731
702
  request,
732
703
  env,
733
704
  variables,
734
- finalHandler
705
+ finalHandler,
706
+ reverse,
735
707
  );
736
708
  }
737
709
 
738
- /**
739
- * Entry type for middleware collection
740
- * Matches the shape of EntryData used in router.ts
741
- */
742
- export interface MiddlewareCollectableEntry {
743
- middleware?: MiddlewareFn<any, any>[];
744
- layout?: MiddlewareCollectableEntry[];
745
- }
746
-
747
- /**
748
- * Collected route middleware with params
749
- */
750
- export interface CollectedMiddleware {
751
- handler: MiddlewareFn<any, any>;
752
- params: Record<string, string>;
753
- }
754
-
755
710
  /**
756
711
  * Collect route-level middleware from an entry tree
757
712
  *
@@ -764,7 +719,7 @@ export interface CollectedMiddleware {
764
719
  */
765
720
  export function collectRouteMiddleware(
766
721
  entries: Iterable<MiddlewareCollectableEntry>,
767
- params: Record<string, string>
722
+ params: Record<string, string>,
768
723
  ): CollectedMiddleware[] {
769
724
  const result: CollectedMiddleware[] = [];
770
725