@rangojs/router 0.0.0-experimental.8 → 0.0.0-experimental.8a4d0430

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 (300) hide show
  1. package/AGENTS.md +5 -0
  2. package/README.md +884 -4
  3. package/dist/bin/rango.js +1601 -0
  4. package/dist/vite/index.js +4474 -867
  5. package/package.json +60 -51
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +262 -0
  8. package/skills/caching/SKILL.md +50 -21
  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 +89 -30
  18. package/skills/loader/SKILL.md +388 -38
  19. package/skills/middleware/SKILL.md +171 -34
  20. package/skills/mime-routes/SKILL.md +128 -0
  21. package/skills/parallel/SKILL.md +78 -1
  22. package/skills/prerender/SKILL.md +643 -0
  23. package/skills/rango/SKILL.md +85 -16
  24. package/skills/response-routes/SKILL.md +411 -0
  25. package/skills/route/SKILL.md +226 -14
  26. package/skills/router-setup/SKILL.md +123 -30
  27. package/skills/tailwind/SKILL.md +129 -0
  28. package/skills/theme/SKILL.md +9 -8
  29. package/skills/typesafety/SKILL.md +318 -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/event-controller.ts +87 -64
  36. package/src/browser/history-state.ts +80 -0
  37. package/src/browser/intercept-utils.ts +52 -0
  38. package/src/browser/link-interceptor.ts +24 -4
  39. package/src/browser/logging.ts +55 -0
  40. package/src/browser/merge-segment-loaders.ts +20 -12
  41. package/src/browser/navigation-bridge.ts +285 -553
  42. package/src/browser/navigation-client.ts +124 -71
  43. package/src/browser/navigation-store.ts +33 -50
  44. package/src/browser/navigation-transaction.ts +295 -0
  45. package/src/browser/network-error-handler.ts +61 -0
  46. package/src/browser/partial-update.ts +258 -308
  47. package/src/browser/prefetch/cache.ts +146 -0
  48. package/src/browser/prefetch/fetch.ts +135 -0
  49. package/src/browser/prefetch/observer.ts +65 -0
  50. package/src/browser/prefetch/policy.ts +42 -0
  51. package/src/browser/prefetch/queue.ts +88 -0
  52. package/src/browser/rango-state.ts +112 -0
  53. package/src/browser/react/Link.tsx +185 -73
  54. package/src/browser/react/NavigationProvider.tsx +51 -11
  55. package/src/browser/react/context.ts +6 -0
  56. package/src/browser/react/filter-segment-order.ts +11 -0
  57. package/src/browser/react/index.ts +12 -12
  58. package/src/browser/react/location-state-shared.ts +95 -53
  59. package/src/browser/react/location-state.ts +60 -15
  60. package/src/browser/react/mount-context.ts +6 -1
  61. package/src/browser/react/nonce-context.ts +23 -0
  62. package/src/browser/react/shallow-equal.ts +27 -0
  63. package/src/browser/react/use-action.ts +29 -51
  64. package/src/browser/react/use-client-cache.ts +5 -3
  65. package/src/browser/react/use-handle.ts +32 -79
  66. package/src/browser/react/use-href.tsx +2 -2
  67. package/src/browser/react/use-link-status.ts +6 -5
  68. package/src/browser/react/use-navigation.ts +22 -63
  69. package/src/browser/react/use-params.ts +65 -0
  70. package/src/browser/react/use-pathname.ts +47 -0
  71. package/src/browser/react/use-router.ts +63 -0
  72. package/src/browser/react/use-search-params.ts +56 -0
  73. package/src/browser/react/use-segments.ts +80 -97
  74. package/src/browser/response-adapter.ts +73 -0
  75. package/src/browser/rsc-router.tsx +107 -26
  76. package/src/browser/scroll-restoration.ts +92 -16
  77. package/src/browser/segment-reconciler.ts +216 -0
  78. package/src/browser/segment-structure-assert.ts +16 -0
  79. package/src/browser/server-action-bridge.ts +504 -599
  80. package/src/browser/shallow.ts +6 -1
  81. package/src/browser/types.ts +109 -47
  82. package/src/browser/validate-redirect-origin.ts +29 -0
  83. package/src/build/generate-manifest.ts +235 -24
  84. package/src/build/generate-route-types.ts +36 -0
  85. package/src/build/index.ts +13 -0
  86. package/src/build/route-trie.ts +265 -0
  87. package/src/build/route-types/ast-helpers.ts +25 -0
  88. package/src/build/route-types/ast-route-extraction.ts +98 -0
  89. package/src/build/route-types/codegen.ts +102 -0
  90. package/src/build/route-types/include-resolution.ts +411 -0
  91. package/src/build/route-types/param-extraction.ts +48 -0
  92. package/src/build/route-types/per-module-writer.ts +128 -0
  93. package/src/build/route-types/router-processing.ts +469 -0
  94. package/src/build/route-types/scan-filter.ts +78 -0
  95. package/src/build/runtime-discovery.ts +231 -0
  96. package/src/cache/background-task.ts +34 -0
  97. package/src/cache/cache-key-utils.ts +44 -0
  98. package/src/cache/cache-policy.ts +125 -0
  99. package/src/cache/cache-runtime.ts +338 -0
  100. package/src/cache/cache-scope.ts +120 -303
  101. package/src/cache/cf/cf-cache-store.ts +119 -7
  102. package/src/cache/cf/index.ts +8 -2
  103. package/src/cache/document-cache.ts +101 -72
  104. package/src/cache/handle-capture.ts +81 -0
  105. package/src/cache/handle-snapshot.ts +41 -0
  106. package/src/cache/index.ts +0 -15
  107. package/src/cache/memory-segment-store.ts +191 -13
  108. package/src/cache/profile-registry.ts +73 -0
  109. package/src/cache/read-through-swr.ts +134 -0
  110. package/src/cache/segment-codec.ts +256 -0
  111. package/src/cache/taint.ts +98 -0
  112. package/src/cache/types.ts +72 -122
  113. package/src/client.rsc.tsx +3 -1
  114. package/src/client.tsx +106 -126
  115. package/src/component-utils.ts +4 -4
  116. package/src/components/DefaultDocument.tsx +5 -1
  117. package/src/context-var.ts +86 -0
  118. package/src/debug.ts +17 -7
  119. package/src/errors.ts +108 -2
  120. package/src/handle.ts +15 -29
  121. package/src/handles/MetaTags.tsx +73 -20
  122. package/src/handles/breadcrumbs.ts +66 -0
  123. package/src/handles/index.ts +1 -0
  124. package/src/handles/meta.ts +30 -13
  125. package/src/host/cookie-handler.ts +21 -15
  126. package/src/host/errors.ts +8 -8
  127. package/src/host/index.ts +4 -7
  128. package/src/host/pattern-matcher.ts +27 -27
  129. package/src/host/router.ts +61 -39
  130. package/src/host/testing.ts +8 -8
  131. package/src/host/types.ts +15 -7
  132. package/src/host/utils.ts +1 -1
  133. package/src/href-client.ts +119 -29
  134. package/src/index.rsc.ts +153 -19
  135. package/src/index.ts +211 -30
  136. package/src/internal-debug.ts +11 -0
  137. package/src/loader.rsc.ts +26 -157
  138. package/src/loader.ts +27 -10
  139. package/src/network-error-thrower.tsx +3 -1
  140. package/src/outlet-provider.tsx +45 -0
  141. package/src/prerender/param-hash.ts +37 -0
  142. package/src/prerender/store.ts +185 -0
  143. package/src/prerender.ts +463 -0
  144. package/src/reverse.ts +330 -0
  145. package/src/root-error-boundary.tsx +41 -29
  146. package/src/route-content-wrapper.tsx +7 -4
  147. package/src/route-definition/dsl-helpers.ts +934 -0
  148. package/src/route-definition/helper-factories.ts +200 -0
  149. package/src/route-definition/helpers-types.ts +430 -0
  150. package/src/route-definition/index.ts +52 -0
  151. package/src/route-definition/redirect.ts +93 -0
  152. package/src/route-definition.ts +1 -1428
  153. package/src/route-map-builder.ts +211 -123
  154. package/src/route-name.ts +53 -0
  155. package/src/route-types.ts +59 -8
  156. package/src/router/content-negotiation.ts +116 -0
  157. package/src/router/debug-manifest.ts +72 -0
  158. package/src/router/error-handling.ts +9 -9
  159. package/src/router/find-match.ts +158 -0
  160. package/src/router/handler-context.ts +374 -81
  161. package/src/router/intercept-resolution.ts +395 -0
  162. package/src/router/lazy-includes.ts +234 -0
  163. package/src/router/loader-resolution.ts +215 -122
  164. package/src/router/logging.ts +248 -0
  165. package/src/router/manifest.ts +148 -35
  166. package/src/router/match-api.ts +620 -0
  167. package/src/router/match-context.ts +5 -3
  168. package/src/router/match-handlers.ts +440 -0
  169. package/src/router/match-middleware/background-revalidation.ts +80 -93
  170. package/src/router/match-middleware/cache-lookup.ts +382 -9
  171. package/src/router/match-middleware/cache-store.ts +51 -22
  172. package/src/router/match-middleware/intercept-resolution.ts +55 -17
  173. package/src/router/match-middleware/segment-resolution.ts +24 -6
  174. package/src/router/match-pipelines.ts +10 -45
  175. package/src/router/match-result.ts +34 -28
  176. package/src/router/metrics.ts +235 -15
  177. package/src/router/middleware-cookies.ts +55 -0
  178. package/src/router/middleware-types.ts +222 -0
  179. package/src/router/middleware.ts +324 -367
  180. package/src/router/pattern-matching.ts +211 -43
  181. package/src/router/prerender-match.ts +402 -0
  182. package/src/router/preview-match.ts +170 -0
  183. package/src/router/revalidation.ts +137 -38
  184. package/src/router/router-context.ts +36 -21
  185. package/src/router/router-interfaces.ts +452 -0
  186. package/src/router/router-options.ts +592 -0
  187. package/src/router/router-registry.ts +24 -0
  188. package/src/router/segment-resolution/fresh.ts +570 -0
  189. package/src/router/segment-resolution/helpers.ts +263 -0
  190. package/src/router/segment-resolution/loader-cache.ts +198 -0
  191. package/src/router/segment-resolution/revalidation.ts +1241 -0
  192. package/src/router/segment-resolution/static-store.ts +67 -0
  193. package/src/router/segment-resolution.ts +21 -0
  194. package/src/router/segment-wrappers.ts +289 -0
  195. package/src/router/telemetry-otel.ts +299 -0
  196. package/src/router/telemetry.ts +300 -0
  197. package/src/router/timeout.ts +148 -0
  198. package/src/router/trie-matching.ts +239 -0
  199. package/src/router/types.ts +77 -3
  200. package/src/router.ts +692 -4257
  201. package/src/rsc/handler-context.ts +45 -0
  202. package/src/rsc/handler.ts +764 -754
  203. package/src/rsc/helpers.ts +140 -6
  204. package/src/rsc/index.ts +0 -20
  205. package/src/rsc/loader-fetch.ts +209 -0
  206. package/src/rsc/manifest-init.ts +86 -0
  207. package/src/rsc/nonce.ts +14 -0
  208. package/src/rsc/origin-guard.ts +141 -0
  209. package/src/rsc/progressive-enhancement.ts +379 -0
  210. package/src/rsc/response-error.ts +37 -0
  211. package/src/rsc/response-route-handler.ts +347 -0
  212. package/src/rsc/rsc-rendering.ts +235 -0
  213. package/src/rsc/runtime-warnings.ts +42 -0
  214. package/src/rsc/server-action.ts +348 -0
  215. package/src/rsc/ssr-setup.ts +128 -0
  216. package/src/rsc/types.ts +38 -11
  217. package/src/search-params.ts +230 -0
  218. package/src/segment-system.tsx +25 -13
  219. package/src/server/context.ts +182 -51
  220. package/src/server/cookie-store.ts +190 -0
  221. package/src/server/fetchable-loader-store.ts +37 -0
  222. package/src/server/handle-store.ts +94 -15
  223. package/src/server/loader-registry.ts +15 -56
  224. package/src/server/request-context.ts +430 -70
  225. package/src/server.ts +35 -130
  226. package/src/ssr/index.tsx +100 -31
  227. package/src/static-handler.ts +114 -0
  228. package/src/theme/ThemeProvider.tsx +21 -15
  229. package/src/theme/ThemeScript.tsx +5 -5
  230. package/src/theme/constants.ts +5 -2
  231. package/src/theme/index.ts +4 -14
  232. package/src/theme/theme-context.ts +4 -30
  233. package/src/theme/theme-script.ts +21 -18
  234. package/src/types/boundaries.ts +158 -0
  235. package/src/types/cache-types.ts +198 -0
  236. package/src/types/error-types.ts +192 -0
  237. package/src/types/global-namespace.ts +100 -0
  238. package/src/types/handler-context.ts +687 -0
  239. package/src/types/index.ts +88 -0
  240. package/src/types/loader-types.ts +183 -0
  241. package/src/types/route-config.ts +170 -0
  242. package/src/types/route-entry.ts +102 -0
  243. package/src/types/segments.ts +148 -0
  244. package/src/types.ts +1 -1623
  245. package/src/urls/include-helper.ts +197 -0
  246. package/src/urls/index.ts +53 -0
  247. package/src/urls/path-helper-types.ts +339 -0
  248. package/src/urls/path-helper.ts +329 -0
  249. package/src/urls/pattern-types.ts +95 -0
  250. package/src/urls/response-types.ts +106 -0
  251. package/src/urls/type-extraction.ts +372 -0
  252. package/src/urls/urls-function.ts +98 -0
  253. package/src/urls.ts +1 -802
  254. package/src/use-loader.tsx +85 -77
  255. package/src/vite/discovery/bundle-postprocess.ts +184 -0
  256. package/src/vite/discovery/discover-routers.ts +344 -0
  257. package/src/vite/discovery/prerender-collection.ts +385 -0
  258. package/src/vite/discovery/route-types-writer.ts +258 -0
  259. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  260. package/src/vite/discovery/state.ts +110 -0
  261. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  262. package/src/vite/index.ts +11 -1133
  263. package/src/vite/plugin-types.ts +131 -0
  264. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  265. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  266. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  267. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -51
  268. package/src/vite/plugins/expose-id-utils.ts +287 -0
  269. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  270. package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
  271. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  272. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  273. package/src/vite/plugins/expose-ids/types.ts +45 -0
  274. package/src/vite/plugins/expose-internal-ids.ts +569 -0
  275. package/src/vite/plugins/refresh-cmd.ts +65 -0
  276. package/src/vite/plugins/use-cache-transform.ts +323 -0
  277. package/src/vite/plugins/version-injector.ts +83 -0
  278. package/src/vite/plugins/version-plugin.ts +254 -0
  279. package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
  280. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  281. package/src/vite/rango.ts +510 -0
  282. package/src/vite/router-discovery.ts +785 -0
  283. package/src/vite/utils/ast-handler-extract.ts +517 -0
  284. package/src/vite/utils/banner.ts +36 -0
  285. package/src/vite/utils/bundle-analysis.ts +137 -0
  286. package/src/vite/utils/manifest-utils.ts +70 -0
  287. package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
  288. package/src/vite/utils/prerender-utils.ts +189 -0
  289. package/src/vite/utils/shared-utils.ts +169 -0
  290. package/CLAUDE.md +0 -43
  291. package/src/browser/lru-cache.ts +0 -69
  292. package/src/browser/request-controller.ts +0 -164
  293. package/src/cache/memory-store.ts +0 -253
  294. package/src/href-context.ts +0 -33
  295. package/src/href.ts +0 -255
  296. package/src/server/route-manifest-cache.ts +0 -173
  297. package/src/vite/expose-handle-id.ts +0 -209
  298. package/src/vite/expose-loader-id.ts +0 -426
  299. package/src/vite/expose-location-state-id.ts +0 -177
  300. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -5,15 +5,83 @@
5
5
  * Uses globalThis to survive HMR in development.
6
6
  */
7
7
 
8
- import type { SegmentCacheStore, CachedEntryData, CacheDefaults, CacheGetResult } from "./types.js";
8
+ import type {
9
+ SegmentCacheStore,
10
+ CachedEntryData,
11
+ CacheDefaults,
12
+ CacheGetResult,
13
+ CacheItemResult,
14
+ CacheItemOptions,
15
+ SegmentHandleData,
16
+ } from "./types.js";
9
17
  import type { RequestContext } from "../server/request-context.js";
18
+ import {
19
+ resolveTtl,
20
+ resolveSwrWindow,
21
+ computeExpiration,
22
+ DEFAULT_FUNCTION_TTL,
23
+ } from "./cache-policy.js";
10
24
 
11
- const CACHE_GLOBAL_KEY = "__rsc_router_segment_cache_store__";
25
+ const CACHE_REGISTRY_KEY = "__rsc_router_segment_cache_registry__";
26
+ const RESPONSE_CACHE_REGISTRY_KEY = "__rsc_router_response_cache_registry__";
27
+ const ITEM_CACHE_REGISTRY_KEY = "__rsc_router_item_cache_registry__";
28
+
29
+ /**
30
+ * Get or create a named Map from a globalThis-backed registry.
31
+ * The registry survives HMR; individual stores are keyed by name.
32
+ */
33
+ function getNamedMap<V>(registryKey: string, name: string): Map<string, V> {
34
+ let registry = (globalThis as any)[registryKey] as
35
+ | Map<string, Map<string, V>>
36
+ | undefined;
37
+ if (!registry) {
38
+ registry = new Map();
39
+ (globalThis as any)[registryKey] = registry;
40
+ }
41
+ let map = registry.get(name);
42
+ if (!map) {
43
+ map = new Map<string, V>();
44
+ registry.set(name, map);
45
+ }
46
+ return map;
47
+ }
48
+
49
+ interface CachedResponseEntry {
50
+ body: ArrayBuffer;
51
+ status: number;
52
+ headers: [string, string][];
53
+ expiresAt: number;
54
+ staleAt: number;
55
+ }
56
+
57
+ interface CachedItemEntry {
58
+ value: string;
59
+ handles?: Record<string, SegmentHandleData>;
60
+ expiresAt: number;
61
+ staleAt: number;
62
+ }
12
63
 
13
64
  /**
14
65
  * Options for MemorySegmentCacheStore
15
66
  */
16
67
  export interface MemorySegmentCacheStoreOptions<TEnv = unknown> {
68
+ /**
69
+ * Optional name for this store instance. Named stores persist their
70
+ * backing Map on globalThis so data survives Vite HMR module reloads.
71
+ * Stores with different names get separate Maps.
72
+ *
73
+ * When omitted, the store uses a plain instance-level Map with no
74
+ * globalThis sharing, which is the safest default for isolation.
75
+ *
76
+ * @example
77
+ * ```typescript
78
+ * // Two named stores are isolated from each other
79
+ * const fast = new MemorySegmentCacheStore({ name: "fast", defaults: { ttl: 10 } });
80
+ * const slow = new MemorySegmentCacheStore({ name: "slow", defaults: { ttl: 300 } });
81
+ * ```
82
+ */
83
+ name?: string;
84
+
17
85
  /**
18
86
  * Default cache options for cache() boundaries.
19
87
  * When cache() is called without explicit ttl/swr,
@@ -35,14 +103,14 @@ export interface MemorySegmentCacheStoreOptions<TEnv = unknown> {
35
103
  * @example
36
104
  * ```typescript
37
105
  * keyGenerator: (ctx, defaultKey) => {
38
- * const locale = ctx.cookie('locale') || 'en';
106
+ * const locale = cookies().get('locale')?.value || 'en';
39
107
  * return `${locale}:${defaultKey}`;
40
108
  * }
41
109
  * ```
42
110
  */
43
111
  keyGenerator?: (
44
112
  ctx: RequestContext<TEnv>,
45
- defaultKey: string
113
+ defaultKey: string,
46
114
  ) => string | Promise<string>;
47
115
  }
48
116
 
@@ -69,19 +137,40 @@ export interface MemorySegmentCacheStoreOptions<TEnv = unknown> {
69
137
  * })
70
138
  * ```
71
139
  */
72
- export class MemorySegmentCacheStore<TEnv = unknown> implements SegmentCacheStore<TEnv> {
140
+ export class MemorySegmentCacheStore<
141
+ TEnv = unknown,
142
+ > implements SegmentCacheStore<TEnv> {
73
143
  private cache: Map<string, CachedEntryData>;
144
+ private responseCache: Map<string, CachedResponseEntry>;
145
+ private itemCache: Map<string, CachedItemEntry>;
74
146
  readonly defaults?: CacheDefaults;
75
147
  readonly keyGenerator?: (
76
148
  ctx: RequestContext<TEnv>,
77
- defaultKey: string
149
+ defaultKey: string,
78
150
  ) => string | Promise<string>;
79
151
 
80
152
  constructor(options?: MemorySegmentCacheStoreOptions<TEnv>) {
81
- // Use globalThis to survive HMR in development
82
- this.cache =
83
- (globalThis as any)[CACHE_GLOBAL_KEY] ??
84
- ((globalThis as any)[CACHE_GLOBAL_KEY] = new Map<string, CachedEntryData>());
153
+ if (options?.name != null) {
154
+ // Named stores use the globalThis registry so data survives HMR.
155
+ // Each name gets its own isolated Map.
156
+ this.cache = getNamedMap<CachedEntryData>(
157
+ CACHE_REGISTRY_KEY,
158
+ options.name,
159
+ );
160
+ this.responseCache = getNamedMap<CachedResponseEntry>(
161
+ RESPONSE_CACHE_REGISTRY_KEY,
162
+ options.name,
163
+ );
164
+ this.itemCache = getNamedMap<CachedItemEntry>(
165
+ ITEM_CACHE_REGISTRY_KEY,
166
+ options.name,
167
+ );
168
+ } else {
169
+ // Unnamed stores get a plain instance-level Map (no globalThis sharing).
170
+ this.cache = new Map<string, CachedEntryData>();
171
+ this.responseCache = new Map<string, CachedResponseEntry>();
172
+ this.itemCache = new Map<string, CachedItemEntry>();
173
+ }
85
174
  this.defaults = options?.defaults;
86
175
  this.keyGenerator = options?.keyGenerator;
87
176
  }
@@ -103,7 +192,12 @@ export class MemorySegmentCacheStore<TEnv = unknown> implements SegmentCacheStor
103
192
  return { data: cached, shouldRevalidate: false };
104
193
  }
105
194
 
106
- async set(key: string, data: CachedEntryData, ttl: number, _swr?: number): Promise<void> {
195
+ async set(
196
+ key: string,
197
+ data: CachedEntryData,
198
+ ttl: number,
199
+ _swr?: number,
200
+ ): Promise<void> {
107
201
  // Note: Memory store doesn't implement SWR - entries just expire at TTL
108
202
  // For SWR support, use CFCacheStore or similar distributed cache
109
203
  const entry: CachedEntryData = {
@@ -119,6 +213,88 @@ export class MemorySegmentCacheStore<TEnv = unknown> implements SegmentCacheStor
119
213
 
120
214
  async clear(): Promise<void> {
121
215
  this.cache.clear();
216
+ this.responseCache.clear();
217
+ this.itemCache.clear();
218
+ }
219
+
220
+ async getResponse(
221
+ key: string,
222
+ ): Promise<{ response: Response; shouldRevalidate: boolean } | null> {
223
+ const cached = this.responseCache.get(key);
224
+ if (!cached) return null;
225
+
226
+ if (Date.now() > cached.expiresAt) {
227
+ this.responseCache.delete(key);
228
+ return null;
229
+ }
230
+
231
+ const isStale = cached.staleAt > 0 && Date.now() > cached.staleAt;
232
+ const headers = new Headers(cached.headers);
233
+ return {
234
+ response: new Response(cached.body, {
235
+ status: cached.status,
236
+ headers,
237
+ }),
238
+ shouldRevalidate: isStale,
239
+ };
240
+ }
241
+
242
+ async putResponse(
243
+ key: string,
244
+ response: Response,
245
+ ttl: number,
246
+ swr?: number,
247
+ ): Promise<void> {
248
+ const body = await response.clone().arrayBuffer();
249
+ const headers: [string, string][] = [];
250
+ response.headers.forEach((value, name) => {
251
+ headers.push([name, value]);
252
+ });
253
+
254
+ const swrWindow = resolveSwrWindow(swr, this.defaults);
255
+ const { staleAt, expiresAt } = computeExpiration(ttl, swrWindow);
256
+
257
+ this.responseCache.set(key, {
258
+ body,
259
+ status: response.status,
260
+ headers,
261
+ expiresAt,
262
+ staleAt,
263
+ });
264
+ }
265
+
266
+ async getItem(key: string): Promise<CacheItemResult | null> {
267
+ const cached = this.itemCache.get(key);
268
+ if (!cached) return null;
269
+
270
+ const now = Date.now();
271
+ if (now > cached.expiresAt) {
272
+ this.itemCache.delete(key);
273
+ return null;
274
+ }
275
+
276
+ const isStale = now > cached.staleAt;
277
+ return {
278
+ value: cached.value,
279
+ handles: cached.handles,
280
+ shouldRevalidate: isStale,
281
+ };
282
+ }
283
+
284
+ async setItem(
285
+ key: string,
286
+ value: string,
287
+ options?: CacheItemOptions,
288
+ ): Promise<void> {
289
+ const ttl = resolveTtl(options?.ttl, this.defaults, DEFAULT_FUNCTION_TTL);
290
+ const swrWindow = resolveSwrWindow(options?.swr, this.defaults);
291
+ const { staleAt, expiresAt } = computeExpiration(ttl, swrWindow);
292
+ this.itemCache.set(key, {
293
+ value,
294
+ handles: options?.handles,
295
+ expiresAt,
296
+ staleAt,
297
+ });
122
298
  }
123
299
 
124
300
  /**
@@ -133,7 +309,7 @@ export class MemorySegmentCacheStore<TEnv = unknown> implements SegmentCacheStor
133
309
  }
134
310
 
135
311
  /**
136
- * Reset the global cache state.
312
+ * Reset the global cache registry.
137
313
  * Useful for test isolation - call this in beforeEach to ensure
138
314
  * tests don't share cache state via globalThis.
139
315
  *
@@ -145,6 +321,8 @@ export class MemorySegmentCacheStore<TEnv = unknown> implements SegmentCacheStor
145
321
  * ```
146
322
  */
147
323
  static resetGlobalCache(): void {
148
- delete (globalThis as any)[CACHE_GLOBAL_KEY];
324
+ delete (globalThis as any)[CACHE_REGISTRY_KEY];
325
+ delete (globalThis as any)[RESPONSE_CACHE_REGISTRY_KEY];
326
+ delete (globalThis as any)[ITEM_CACHE_REGISTRY_KEY];
149
327
  }
150
328
  }
@@ -0,0 +1,73 @@
1
+ /**
2
+ * Cache Profile Registry
3
+ *
4
+ * Named cache profiles for "use cache" directive.
5
+ * Profiles define TTL, SWR, and optional default tags.
6
+ * Set by createRouter() at startup, read by registerCachedFunction() at runtime.
7
+ */
8
+
9
+ export interface CacheProfile {
10
+ /** Time-to-live in seconds */
11
+ ttl: number;
12
+ /** Stale-while-revalidate window in seconds */
13
+ swr?: number;
14
+ /** Default cache tags for invalidation */
15
+ tags?: string[];
16
+ }
17
+
18
+ const DEFAULT_PROFILE: CacheProfile = { ttl: 900, swr: 1800 };
19
+
20
+ let _profiles: Record<string, CacheProfile> = {
21
+ default: DEFAULT_PROFILE,
22
+ };
23
+
24
+ const PROFILE_NAME_RE = /^[a-zA-Z0-9_-]+$/;
25
+
26
+ /**
27
+ * Validate and merge user profiles with the default profile.
28
+ * Returns a new object suitable for both DSL-time and request-scoped use.
29
+ *
30
+ * Used by createRouter() to compute the resolved profile map once,
31
+ * stored on the router instance and passed to every request context.
32
+ */
33
+ export function resolveCacheProfiles(
34
+ profiles?: Record<string, CacheProfile>,
35
+ ): Record<string, CacheProfile> {
36
+ const merged: Record<string, CacheProfile> = {
37
+ default: DEFAULT_PROFILE,
38
+ };
39
+ if (profiles) {
40
+ for (const name of Object.keys(profiles)) {
41
+ if (!PROFILE_NAME_RE.test(name)) {
42
+ throw new Error(
43
+ `Invalid cache profile name "${name}". ` +
44
+ `Profile names must match [a-zA-Z0-9_-]+.`,
45
+ );
46
+ }
47
+ merged[name] = profiles[name];
48
+ }
49
+ }
50
+ return merged;
51
+ }
52
+
53
+ /**
54
+ * Set all cache profiles in the global registry.
55
+ * Called by createRouter() at startup for DSL-time resolution
56
+ * (cache("profileName") reads from this during route definition).
57
+ *
58
+ * WARNING: This is global mutable state. It exists only for DSL-time
59
+ * reads. Runtime resolution (registerCachedFunction) uses request-scoped
60
+ * profiles and does NOT read from this registry.
61
+ */
62
+ export function setCacheProfiles(profiles: Record<string, CacheProfile>): void {
63
+ _profiles = resolveCacheProfiles(profiles);
64
+ }
65
+
66
+ /**
67
+ * Get a cache profile by name from the global registry.
68
+ * Used only at DSL-time (cache("profileName") inside urls() evaluation).
69
+ * Runtime code uses request-scoped profiles instead.
70
+ */
71
+ export function getCacheProfile(name: string): CacheProfile | undefined {
72
+ return _profiles[name];
73
+ }
@@ -0,0 +1,134 @@
1
+ /**
2
+ * SWR Read-Through Engine
3
+ *
4
+ * Generic read-through cache with stale-while-revalidate support
5
+ * for item-level caching (getItem/setItem).
6
+ *
7
+ * Flow:
8
+ * 1. Lookup cached item by key
9
+ * 2. Fresh hit → deserialize, return
10
+ * 3. Stale hit → deserialize, return, revalidate in background
11
+ * 4. Miss → execute, cache write (blocking when no waitUntil), return
12
+ */
13
+
14
+ import type { CacheItemResult, CacheItemOptions } from "./types.js";
15
+ import { runBackground } from "./background-task.js";
16
+
17
+ interface WaitUntilHost {
18
+ waitUntil?: (fn: () => Promise<void>) => void;
19
+ }
20
+
21
+ export interface ReadThroughItemConfig<T> {
22
+ /** Retrieve a cached item by key */
23
+ getItem: (key: string) => Promise<CacheItemResult | null>;
24
+ /** Store a serialized item by key */
25
+ setItem: (
26
+ key: string,
27
+ value: string,
28
+ options?: CacheItemOptions,
29
+ ) => Promise<void>;
30
+ /** Cache key */
31
+ key: string;
32
+ /** Execute the underlying function/loader on miss or revalidation */
33
+ execute: () => Promise<T>;
34
+ /** Serialize result for storage. Return null to skip caching. */
35
+ serialize: (data: T) => Promise<string | null>;
36
+ /** Deserialize cached value back to the original type */
37
+ deserialize: (value: string) => Promise<T>;
38
+ /** Options passed to setItem on cache write */
39
+ storeOptions: CacheItemOptions;
40
+ /** Called on fresh cache hit (before returning data) */
41
+ onHit?: (cached: CacheItemResult) => void;
42
+ /** Called on stale cache hit (before scheduling background revalidation) */
43
+ onStale?: (cached: CacheItemResult) => void;
44
+ /** Called on cache miss (before executing) */
45
+ onMiss?: () => void;
46
+ /** Called after successful cache write */
47
+ onCached?: () => void;
48
+ /** Host with optional waitUntil for background tasks */
49
+ host?: WaitUntilHost | null;
50
+ }
51
+
52
+ /**
53
+ * Read-through cache with SWR support for item-level caching.
54
+ *
55
+ * On fresh hit: returns deserialized cached data.
56
+ * On stale hit: returns stale data, schedules background revalidation.
57
+ * On miss: executes, writes to cache (blocking when no waitUntil), returns.
58
+ */
59
+ export async function readThroughItem<T>(
60
+ config: ReadThroughItemConfig<T>,
61
+ ): Promise<T> {
62
+ const {
63
+ getItem,
64
+ setItem,
65
+ key,
66
+ execute,
67
+ serialize,
68
+ deserialize,
69
+ storeOptions,
70
+ onHit,
71
+ onStale,
72
+ onMiss,
73
+ onCached,
74
+ host,
75
+ } = config;
76
+
77
+ // Cache lookup
78
+ try {
79
+ const cached = await getItem(key);
80
+
81
+ if (cached) {
82
+ const data = await deserialize(cached.value);
83
+
84
+ if (!cached.shouldRevalidate) {
85
+ onHit?.(cached);
86
+ return data;
87
+ }
88
+
89
+ // Stale hit — return stale data, revalidate in background
90
+ onStale?.(cached);
91
+ runBackground(
92
+ host,
93
+ async () => {
94
+ try {
95
+ const fresh = await execute();
96
+ const serialized = await serialize(fresh);
97
+ if (serialized !== null) {
98
+ await setItem(key, serialized, storeOptions);
99
+ }
100
+ } catch {
101
+ // Background revalidation failed silently
102
+ }
103
+ },
104
+ true,
105
+ );
106
+ return data;
107
+ }
108
+ } catch {
109
+ // Cache lookup failed, fall through to fresh execution
110
+ }
111
+
112
+ // Cache miss
113
+ onMiss?.();
114
+ const data = await execute();
115
+
116
+ // Non-blocking cache write (blocks when no waitUntil)
117
+ await runBackground(
118
+ host,
119
+ async () => {
120
+ try {
121
+ const serialized = await serialize(data);
122
+ if (serialized !== null) {
123
+ await setItem(key, serialized, storeOptions);
124
+ onCached?.();
125
+ }
126
+ } catch {
127
+ // Cache write failed silently
128
+ }
129
+ },
130
+ true,
131
+ );
132
+
133
+ return data;
134
+ }