@rangojs/router 0.0.0-experimental.10

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 (172) hide show
  1. package/CLAUDE.md +43 -0
  2. package/README.md +19 -0
  3. package/dist/bin/rango.js +227 -0
  4. package/dist/vite/index.js +3039 -0
  5. package/package.json +171 -0
  6. package/skills/caching/SKILL.md +191 -0
  7. package/skills/debug-manifest/SKILL.md +108 -0
  8. package/skills/document-cache/SKILL.md +180 -0
  9. package/skills/fonts/SKILL.md +165 -0
  10. package/skills/hooks/SKILL.md +442 -0
  11. package/skills/intercept/SKILL.md +190 -0
  12. package/skills/layout/SKILL.md +213 -0
  13. package/skills/links/SKILL.md +180 -0
  14. package/skills/loader/SKILL.md +246 -0
  15. package/skills/middleware/SKILL.md +202 -0
  16. package/skills/mime-routes/SKILL.md +124 -0
  17. package/skills/parallel/SKILL.md +228 -0
  18. package/skills/prerender/SKILL.md +283 -0
  19. package/skills/rango/SKILL.md +54 -0
  20. package/skills/response-routes/SKILL.md +358 -0
  21. package/skills/route/SKILL.md +173 -0
  22. package/skills/router-setup/SKILL.md +346 -0
  23. package/skills/tailwind/SKILL.md +129 -0
  24. package/skills/theme/SKILL.md +78 -0
  25. package/skills/typesafety/SKILL.md +394 -0
  26. package/src/__internal.ts +175 -0
  27. package/src/bin/rango.ts +24 -0
  28. package/src/browser/event-controller.ts +876 -0
  29. package/src/browser/index.ts +18 -0
  30. package/src/browser/link-interceptor.ts +121 -0
  31. package/src/browser/lru-cache.ts +69 -0
  32. package/src/browser/merge-segment-loaders.ts +126 -0
  33. package/src/browser/navigation-bridge.ts +913 -0
  34. package/src/browser/navigation-client.ts +165 -0
  35. package/src/browser/navigation-store.ts +823 -0
  36. package/src/browser/partial-update.ts +600 -0
  37. package/src/browser/react/Link.tsx +248 -0
  38. package/src/browser/react/NavigationProvider.tsx +346 -0
  39. package/src/browser/react/ScrollRestoration.tsx +94 -0
  40. package/src/browser/react/context.ts +53 -0
  41. package/src/browser/react/index.ts +52 -0
  42. package/src/browser/react/location-state-shared.ts +120 -0
  43. package/src/browser/react/location-state.ts +62 -0
  44. package/src/browser/react/mount-context.ts +32 -0
  45. package/src/browser/react/use-action.ts +240 -0
  46. package/src/browser/react/use-client-cache.ts +56 -0
  47. package/src/browser/react/use-handle.ts +203 -0
  48. package/src/browser/react/use-href.tsx +40 -0
  49. package/src/browser/react/use-link-status.ts +134 -0
  50. package/src/browser/react/use-mount.ts +31 -0
  51. package/src/browser/react/use-navigation.ts +140 -0
  52. package/src/browser/react/use-segments.ts +188 -0
  53. package/src/browser/request-controller.ts +164 -0
  54. package/src/browser/rsc-router.tsx +352 -0
  55. package/src/browser/scroll-restoration.ts +324 -0
  56. package/src/browser/segment-structure-assert.ts +67 -0
  57. package/src/browser/server-action-bridge.ts +762 -0
  58. package/src/browser/shallow.ts +35 -0
  59. package/src/browser/types.ts +478 -0
  60. package/src/build/generate-manifest.ts +377 -0
  61. package/src/build/generate-route-types.ts +828 -0
  62. package/src/build/index.ts +36 -0
  63. package/src/build/route-trie.ts +239 -0
  64. package/src/cache/cache-scope.ts +563 -0
  65. package/src/cache/cf/cf-cache-store.ts +428 -0
  66. package/src/cache/cf/index.ts +19 -0
  67. package/src/cache/document-cache.ts +340 -0
  68. package/src/cache/index.ts +58 -0
  69. package/src/cache/memory-segment-store.ts +150 -0
  70. package/src/cache/memory-store.ts +253 -0
  71. package/src/cache/types.ts +392 -0
  72. package/src/client.rsc.tsx +83 -0
  73. package/src/client.tsx +643 -0
  74. package/src/component-utils.ts +76 -0
  75. package/src/components/DefaultDocument.tsx +23 -0
  76. package/src/debug.ts +233 -0
  77. package/src/default-error-boundary.tsx +88 -0
  78. package/src/deps/browser.ts +8 -0
  79. package/src/deps/html-stream-client.ts +2 -0
  80. package/src/deps/html-stream-server.ts +2 -0
  81. package/src/deps/rsc.ts +10 -0
  82. package/src/deps/ssr.ts +2 -0
  83. package/src/errors.ts +295 -0
  84. package/src/handle.ts +130 -0
  85. package/src/handles/MetaTags.tsx +193 -0
  86. package/src/handles/index.ts +6 -0
  87. package/src/handles/meta.ts +247 -0
  88. package/src/host/cookie-handler.ts +159 -0
  89. package/src/host/errors.ts +97 -0
  90. package/src/host/index.ts +56 -0
  91. package/src/host/pattern-matcher.ts +214 -0
  92. package/src/host/router.ts +330 -0
  93. package/src/host/testing.ts +79 -0
  94. package/src/host/types.ts +138 -0
  95. package/src/host/utils.ts +25 -0
  96. package/src/href-client.ts +202 -0
  97. package/src/href-context.ts +33 -0
  98. package/src/index.rsc.ts +121 -0
  99. package/src/index.ts +165 -0
  100. package/src/loader.rsc.ts +207 -0
  101. package/src/loader.ts +47 -0
  102. package/src/network-error-thrower.tsx +21 -0
  103. package/src/outlet-context.ts +15 -0
  104. package/src/prerender/param-hash.ts +35 -0
  105. package/src/prerender/store.ts +40 -0
  106. package/src/prerender.ts +156 -0
  107. package/src/reverse.ts +267 -0
  108. package/src/root-error-boundary.tsx +277 -0
  109. package/src/route-content-wrapper.tsx +193 -0
  110. package/src/route-definition.ts +1431 -0
  111. package/src/route-map-builder.ts +242 -0
  112. package/src/route-types.ts +220 -0
  113. package/src/router/error-handling.ts +287 -0
  114. package/src/router/handler-context.ts +158 -0
  115. package/src/router/intercept-resolution.ts +387 -0
  116. package/src/router/loader-resolution.ts +327 -0
  117. package/src/router/manifest.ts +216 -0
  118. package/src/router/match-api.ts +621 -0
  119. package/src/router/match-context.ts +264 -0
  120. package/src/router/match-middleware/background-revalidation.ts +236 -0
  121. package/src/router/match-middleware/cache-lookup.ts +382 -0
  122. package/src/router/match-middleware/cache-store.ts +276 -0
  123. package/src/router/match-middleware/index.ts +81 -0
  124. package/src/router/match-middleware/intercept-resolution.ts +281 -0
  125. package/src/router/match-middleware/segment-resolution.ts +184 -0
  126. package/src/router/match-pipelines.ts +214 -0
  127. package/src/router/match-result.ts +213 -0
  128. package/src/router/metrics.ts +62 -0
  129. package/src/router/middleware.ts +791 -0
  130. package/src/router/pattern-matching.ts +407 -0
  131. package/src/router/revalidation.ts +190 -0
  132. package/src/router/router-context.ts +301 -0
  133. package/src/router/segment-resolution.ts +1315 -0
  134. package/src/router/trie-matching.ts +172 -0
  135. package/src/router/types.ts +163 -0
  136. package/src/router.gen.ts +6 -0
  137. package/src/router.ts +2423 -0
  138. package/src/rsc/handler.ts +1443 -0
  139. package/src/rsc/helpers.ts +64 -0
  140. package/src/rsc/index.ts +56 -0
  141. package/src/rsc/nonce.ts +18 -0
  142. package/src/rsc/types.ts +236 -0
  143. package/src/segment-system.tsx +442 -0
  144. package/src/server/context.ts +466 -0
  145. package/src/server/handle-store.ts +229 -0
  146. package/src/server/loader-registry.ts +174 -0
  147. package/src/server/request-context.ts +554 -0
  148. package/src/server/root-layout.tsx +10 -0
  149. package/src/server/tsconfig.json +14 -0
  150. package/src/server.ts +171 -0
  151. package/src/ssr/index.tsx +296 -0
  152. package/src/theme/ThemeProvider.tsx +291 -0
  153. package/src/theme/ThemeScript.tsx +61 -0
  154. package/src/theme/constants.ts +59 -0
  155. package/src/theme/index.ts +58 -0
  156. package/src/theme/theme-context.ts +70 -0
  157. package/src/theme/theme-script.ts +152 -0
  158. package/src/theme/types.ts +182 -0
  159. package/src/theme/use-theme.ts +44 -0
  160. package/src/types.ts +1757 -0
  161. package/src/urls.gen.ts +8 -0
  162. package/src/urls.ts +1282 -0
  163. package/src/use-loader.tsx +346 -0
  164. package/src/vite/expose-action-id.ts +344 -0
  165. package/src/vite/expose-handle-id.ts +209 -0
  166. package/src/vite/expose-loader-id.ts +426 -0
  167. package/src/vite/expose-location-state-id.ts +177 -0
  168. package/src/vite/expose-prerender-handler-id.ts +429 -0
  169. package/src/vite/index.ts +2068 -0
  170. package/src/vite/package-resolution.ts +125 -0
  171. package/src/vite/version.d.ts +12 -0
  172. package/src/vite/virtual-entries.ts +114 -0
@@ -0,0 +1,253 @@
1
+ /**
2
+ * In-Memory Cache Store
3
+ *
4
+ * Simple implementation for development and testing.
5
+ * Not suitable for production (no persistence, single-instance only).
6
+ *
7
+ * @internal This is reserved for future extensibility.
8
+ * For segment caching, use MemorySegmentCacheStore instead.
9
+ */
10
+
11
+ import type {
12
+ CacheStore,
13
+ CacheEntry,
14
+ CacheValue,
15
+ CachePutOptions,
16
+ CacheMetadata,
17
+ CacheValueType,
18
+ } from "./types.js";
19
+
20
+ // ============================================================================
21
+ // Constants
22
+ // ============================================================================
23
+
24
+ /** Default TTL when no explicit value is provided */
25
+ const DEFAULT_TTL_SECONDS = 60;
26
+
27
+ // ============================================================================
28
+ // Types
29
+ // ============================================================================
30
+
31
+ interface StoredEntry {
32
+ /** Stored value (streams/responses converted to ArrayBuffer) */
33
+ value: ArrayBuffer | string | object;
34
+ metadata: CacheMetadata;
35
+ }
36
+
37
+ /**
38
+ * In-memory cache store implementation
39
+ */
40
+ export class MemoryCacheStore implements CacheStore {
41
+ private cache = new Map<string, StoredEntry>();
42
+
43
+ async match<T = CacheValue>(key: string): Promise<CacheEntry<T> | undefined> {
44
+ const entry = this.cache.get(key);
45
+
46
+ if (!entry) {
47
+ return undefined;
48
+ }
49
+
50
+ // Check expiration
51
+ if (entry.metadata.expiresAt && Date.now() > entry.metadata.expiresAt) {
52
+ this.cache.delete(key);
53
+ return undefined;
54
+ }
55
+
56
+ // Reconstruct value based on original type
57
+ const value = this.reconstructValue(entry);
58
+
59
+ return {
60
+ value: value as T,
61
+ metadata: entry.metadata,
62
+ };
63
+ }
64
+
65
+ async put<T extends CacheValue>(
66
+ key: string,
67
+ value: T,
68
+ options?: CachePutOptions
69
+ ): Promise<void> {
70
+ const ttl = options?.ttl ?? DEFAULT_TTL_SECONDS;
71
+ const expiresAt = Date.now() + ttl * 1000;
72
+
73
+ // Detect value type and convert for storage
74
+ const { storedValue, valueType, responseHeaders, responseStatus } =
75
+ await this.prepareForStorage(value);
76
+
77
+ const metadata: CacheMetadata = {
78
+ ...options?.metadata,
79
+ expiresAt,
80
+ valueType,
81
+ responseHeaders,
82
+ responseStatus,
83
+ };
84
+
85
+ this.cache.set(key, {
86
+ value: storedValue,
87
+ metadata,
88
+ });
89
+ }
90
+
91
+ async delete(key: string): Promise<boolean> {
92
+ return this.cache.delete(key);
93
+ }
94
+
95
+ /**
96
+ * Clear all entries (useful for testing)
97
+ */
98
+ clear(): void {
99
+ this.cache.clear();
100
+ }
101
+
102
+ /**
103
+ * Get current cache size (useful for testing/debugging)
104
+ */
105
+ get size(): number {
106
+ return this.cache.size;
107
+ }
108
+
109
+ /**
110
+ * Manually purge expired entries
111
+ */
112
+ purgeExpired(): number {
113
+ const now = Date.now();
114
+ let purged = 0;
115
+
116
+ for (const [key, entry] of this.cache) {
117
+ if (entry.metadata.expiresAt && now > entry.metadata.expiresAt) {
118
+ this.cache.delete(key);
119
+ purged++;
120
+ }
121
+ }
122
+
123
+ return purged;
124
+ }
125
+
126
+ /**
127
+ * Prepare a value for storage
128
+ * Converts streams and responses to ArrayBuffer, detects type
129
+ */
130
+ private async prepareForStorage(value: CacheValue): Promise<{
131
+ storedValue: ArrayBuffer | string | object;
132
+ valueType: CacheValueType;
133
+ responseHeaders?: Record<string, string>;
134
+ responseStatus?: number;
135
+ }> {
136
+ // ReadableStream -> ArrayBuffer
137
+ if (value instanceof ReadableStream) {
138
+ return {
139
+ storedValue: await streamToArrayBuffer(value),
140
+ valueType: "stream",
141
+ };
142
+ }
143
+
144
+ // Response -> ArrayBuffer + headers/status
145
+ if (value instanceof Response) {
146
+ const headers: Record<string, string> = {};
147
+ value.headers.forEach((v, k) => {
148
+ headers[k] = v;
149
+ });
150
+
151
+ return {
152
+ storedValue: await value.clone().arrayBuffer(),
153
+ valueType: "response",
154
+ responseHeaders: headers,
155
+ responseStatus: value.status,
156
+ };
157
+ }
158
+
159
+ // ArrayBuffer -> store as-is
160
+ if (value instanceof ArrayBuffer) {
161
+ return {
162
+ storedValue: value,
163
+ valueType: "arraybuffer",
164
+ };
165
+ }
166
+
167
+ // String -> store as-is
168
+ if (typeof value === "string") {
169
+ return {
170
+ storedValue: value,
171
+ valueType: "string",
172
+ };
173
+ }
174
+
175
+ // Object -> store as-is (JSON-serializable)
176
+ return {
177
+ storedValue: value,
178
+ valueType: "object",
179
+ };
180
+ }
181
+
182
+ /**
183
+ * Reconstruct original value type from stored entry
184
+ */
185
+ private reconstructValue(entry: StoredEntry): CacheValue {
186
+ const { value, metadata } = entry;
187
+
188
+ switch (metadata.valueType) {
189
+ case "stream":
190
+ return arrayBufferToStream(value as ArrayBuffer);
191
+
192
+ case "response": {
193
+ const status = metadata.responseStatus ?? 200;
194
+ // Status codes 204 (No Content) and 304 (Not Modified) cannot have a body
195
+ const isNullBodyStatus = status === 204 || status === 304;
196
+ return new Response(isNullBodyStatus ? null : (value as ArrayBuffer), {
197
+ status,
198
+ headers: metadata.responseHeaders,
199
+ });
200
+ }
201
+
202
+ case "arraybuffer":
203
+ case "string":
204
+ case "object":
205
+ default:
206
+ return value as CacheValue;
207
+ }
208
+ }
209
+ }
210
+
211
+ /**
212
+ * Convert a ReadableStream to ArrayBuffer.
213
+ * @internal
214
+ */
215
+ async function streamToArrayBuffer(
216
+ stream: ReadableStream<Uint8Array>
217
+ ): Promise<ArrayBuffer> {
218
+ const chunks: Uint8Array[] = [];
219
+ const reader = stream.getReader();
220
+
221
+ while (true) {
222
+ const { done, value } = await reader.read();
223
+ if (done) break;
224
+ chunks.push(value);
225
+ }
226
+
227
+ // Concatenate chunks
228
+ const totalLength = chunks.reduce((sum, chunk) => sum + chunk.length, 0);
229
+ const result = new Uint8Array(totalLength);
230
+ let offset = 0;
231
+
232
+ for (const chunk of chunks) {
233
+ result.set(chunk, offset);
234
+ offset += chunk.length;
235
+ }
236
+
237
+ return result.buffer;
238
+ }
239
+
240
+ /**
241
+ * Convert an ArrayBuffer to a ReadableStream.
242
+ * @internal
243
+ */
244
+ function arrayBufferToStream(buffer: ArrayBuffer): ReadableStream<Uint8Array> {
245
+ const uint8 = new Uint8Array(buffer);
246
+
247
+ return new ReadableStream({
248
+ start(controller) {
249
+ controller.enqueue(uint8);
250
+ controller.close();
251
+ },
252
+ });
253
+ }
@@ -0,0 +1,392 @@
1
+ /**
2
+ * Cache Store Types
3
+ *
4
+ * Generic caching interface supporting multiple value types.
5
+ * Designed to be implemented by different backends:
6
+ * - MemoryCacheStore (dev/testing)
7
+ * - Cloudflare Cache API adapter
8
+ * - Cloudflare KV adapter
9
+ * - Redis adapter
10
+ */
11
+
12
+ import type { ResolvedSegment } from "../types.js";
13
+ import type { RequestContext } from "../server/request-context.js";
14
+
15
+ // ============================================================================
16
+ // Segment Cache Store (low-level storage interface)
17
+ // ============================================================================
18
+
19
+ /**
20
+ * Result from cache get() including data and revalidation status
21
+ */
22
+ export interface CacheGetResult {
23
+ /** The cached entry data */
24
+ data: CachedEntryData;
25
+ /**
26
+ * Whether the caller should trigger background revalidation.
27
+ * True when entry is stale AND not already being revalidated.
28
+ * The store atomically marks the entry as REVALIDATING when returning true.
29
+ */
30
+ shouldRevalidate: boolean;
31
+ }
32
+
33
+ /**
34
+ * Low-level segment cache store interface.
35
+ *
36
+ * Implementations handle the actual storage (memory, KV, Redis, etc.).
37
+ * The store deals with serialized data - RSC serialization is handled
38
+ * by the cache provider layer.
39
+ *
40
+ * @typeParam TEnv - Platform bindings type (e.g., Cloudflare env)
41
+ */
42
+ export interface SegmentCacheStore<TEnv = unknown> {
43
+ /**
44
+ * Default cache options for this store.
45
+ * Used by cache() boundaries when ttl/swr are not explicitly specified.
46
+ */
47
+ readonly defaults?: CacheDefaults;
48
+
49
+ /**
50
+ * Custom key generator applied to all cache operations using this store.
51
+ * Receives the full RequestContext and the default-generated key.
52
+ * Return value becomes the final cache key (unless route overrides with `key` option).
53
+ *
54
+ * Resolution priority:
55
+ * 1. Route-level `key` function (full override)
56
+ * 2. Store-level `keyGenerator` (modifies default key)
57
+ * 3. Default key generation (prefix:pathname:params)
58
+ *
59
+ * @example Using headers for cache segmentation
60
+ * ```typescript
61
+ * keyGenerator: (ctx, defaultKey) => {
62
+ * const segment = ctx.request.headers.get('x-user-segment') || 'default';
63
+ * return `${segment}:${defaultKey}`;
64
+ * }
65
+ * ```
66
+ *
67
+ * @example Using env bindings (Cloudflare)
68
+ * ```typescript
69
+ * keyGenerator: (ctx, defaultKey) => {
70
+ * const region = ctx.env.REGION || 'us';
71
+ * return `${region}:${defaultKey}`;
72
+ * }
73
+ * ```
74
+ *
75
+ * @example Using cookies for locale
76
+ * ```typescript
77
+ * keyGenerator: (ctx, defaultKey) => {
78
+ * const locale = ctx.cookie('locale') || 'en';
79
+ * return `${locale}:${defaultKey}`;
80
+ * }
81
+ * ```
82
+ */
83
+ readonly keyGenerator?: (
84
+ ctx: RequestContext<TEnv>,
85
+ defaultKey: string
86
+ ) => string | Promise<string>;
87
+
88
+ /**
89
+ * Get cached entry data by key
90
+ * @returns Cache result with data and staleness, or null if not found/expired
91
+ */
92
+ get(key: string): Promise<CacheGetResult | null>;
93
+
94
+ /**
95
+ * Store entry data with TTL
96
+ * @param key - Cache key
97
+ * @param data - Serialized entry data
98
+ * @param ttl - Time-to-live in seconds
99
+ * @param swr - Optional stale-while-revalidate window in seconds
100
+ */
101
+ set(key: string, data: CachedEntryData, ttl: number, swr?: number): Promise<void>;
102
+
103
+ /**
104
+ * Delete a cached entry
105
+ * @returns true if deleted, false if not found
106
+ */
107
+ delete(key: string): Promise<boolean>;
108
+
109
+ /**
110
+ * Clear all cached entries (optional, for testing)
111
+ */
112
+ clear?(): Promise<void>;
113
+
114
+ // ============================================================================
115
+ // Document Cache Methods (optional)
116
+ // ============================================================================
117
+ // These methods are for caching full HTTP responses (document-level caching).
118
+ // Stores that support response caching should implement these methods.
119
+
120
+ /**
121
+ * Get a cached Response by key.
122
+ * Returns the response and whether it should be revalidated (SWR).
123
+ */
124
+ getResponse?(key: string): Promise<{ response: Response; shouldRevalidate: boolean } | null>;
125
+
126
+ /**
127
+ * Store a Response with TTL and optional SWR window.
128
+ * @param key - Cache key
129
+ * @param response - Response to cache (will be cloned)
130
+ * @param ttl - Time-to-live in seconds
131
+ * @param swr - Optional stale-while-revalidate window in seconds
132
+ */
133
+ putResponse?(key: string, response: Response, ttl: number, swr?: number): Promise<void>;
134
+ }
135
+
136
+ /**
137
+ * Serialized segment data stored in cache
138
+ * Note: loading is preserved to ensure consistent tree structure between cached and fresh renders
139
+ *
140
+ * @internal This type is an implementation detail and may change without notice.
141
+ */
142
+ export interface SerializedSegmentData {
143
+ /** RSC-encoded component string */
144
+ encoded: string;
145
+ /** RSC-encoded layout string (if present) */
146
+ encodedLayout?: string;
147
+ /** RSC-encoded loading skeleton string (if present), or "null" for explicit null */
148
+ encodedLoading?: string;
149
+ /** RSC-encoded loaderData (if present) */
150
+ encodedLoaderData?: string;
151
+ /** RSC-encoded loaderDataPromise (if present) */
152
+ encodedLoaderDataPromise?: string;
153
+ /** Segment metadata (everything except component, layout, loading, and loader data) */
154
+ metadata: Omit<ResolvedSegment, "component" | "layout" | "loading" | "loaderData" | "loaderDataPromise">;
155
+ }
156
+
157
+ /**
158
+ * Raw data stored in cache for an entry
159
+ *
160
+ * @internal This type is an implementation detail and may change without notice.
161
+ */
162
+ export interface CachedEntryData {
163
+ /** Serialized segments for this entry */
164
+ segments: SerializedSegmentData[];
165
+ /** Handle data keyed by segment ID */
166
+ handles: Record<string, SegmentHandleData>;
167
+ /** Expiration timestamp (ms since epoch) */
168
+ expiresAt: number;
169
+ }
170
+
171
+ // ============================================================================
172
+ // Cache Configuration
173
+ // ============================================================================
174
+
175
+ /**
176
+ * Default cache options applied to all cache() boundaries.
177
+ * Individual cache() calls can override any of these values.
178
+ *
179
+ * @example
180
+ * ```ts
181
+ * const store = new CFCacheStore({
182
+ * defaults: { ttl: 60, swr: 300 }
183
+ * });
184
+ * ```
185
+ */
186
+ export interface CacheDefaults {
187
+ /**
188
+ * Default time-to-live in seconds.
189
+ * After TTL expires, cached entry is considered stale.
190
+ */
191
+ ttl?: number;
192
+ /**
193
+ * Default stale-while-revalidate window in seconds.
194
+ * During SWR window, stale content is served while revalidating in background.
195
+ */
196
+ swr?: number;
197
+ }
198
+
199
+ /**
200
+ * Cache configuration for RSC handler
201
+ */
202
+ export interface CacheConfig {
203
+ /** Cache store implementation (includes defaults) */
204
+ store: SegmentCacheStore;
205
+ /** Enable/disable caching (default: true) */
206
+ enabled?: boolean;
207
+ }
208
+
209
+ /**
210
+ * Cache configuration - can be static or a function receiving env
211
+ */
212
+ export type CacheConfigOrFactory<TEnv> =
213
+ | CacheConfig
214
+ | ((env: TEnv) => CacheConfig);
215
+
216
+ // ============================================================================
217
+ // Segment Cache Provider (request-level interface)
218
+ // ============================================================================
219
+
220
+ /**
221
+ * Handle data for a single segment
222
+ * Structure: { handleName: [values...] }
223
+ */
224
+ export type SegmentHandleData = Record<string, unknown[]>;
225
+
226
+ /**
227
+ * Result from cache get() including segments and their handle data
228
+ * Each entry can produce multiple segments (main + parallels)
229
+ */
230
+ export interface CachedEntryResult {
231
+ /** All segments for this entry (main segment + parallels) */
232
+ segments: ResolvedSegment[];
233
+ /** Handle data keyed by segment ID */
234
+ handles: Record<string, SegmentHandleData>;
235
+ }
236
+
237
+
238
+ /**
239
+ * Segment cache provider interface
240
+ *
241
+ * Used by router to check/store segment cache during matching.
242
+ * Accessed via request context - if not present, caching is disabled.
243
+ *
244
+ * @internal Not currently implemented - CacheScope is used directly.
245
+ * Reserved for future extensibility.
246
+ */
247
+ export interface SegmentCacheProvider {
248
+ /** Whether caching is enabled for this request */
249
+ readonly enabled: boolean;
250
+
251
+ /**
252
+ * Get cached segments and restore handles/loaders.
253
+ *
254
+ * Combines cache get with handle replay and loader data restoration.
255
+ * Returns tuple of [segments, segmentIds] if cache hit, null if miss or disabled.
256
+ *
257
+ * @param cacheKey - Cache key to look up
258
+ * @param params - Route params for cache key generation
259
+ * @param loaderPromises - Map to restore loader data into
260
+ * @returns Tuple of [segments, segmentIds] or null if miss
261
+ */
262
+ restore(
263
+ cacheKey: string,
264
+ params: Record<string, string>,
265
+ loaderPromises: Map<string, Promise<any>>
266
+ ): Promise<[ResolvedSegment[], string[]] | null>;
267
+
268
+ /**
269
+ * Cache entry with automatic handle collection (non-blocking).
270
+ *
271
+ * Schedules caching via waitUntil - handles are collected after they settle.
272
+ * Validates segments have actual components before caching.
273
+ *
274
+ * @param cacheKey - The cache key to store under
275
+ * @param segments - All resolved segments for this entry
276
+ */
277
+ cacheEntry(cacheKey: string, segments: ResolvedSegment[]): void;
278
+ }
279
+
280
+ // ============================================================================
281
+ // Generic Cache Store (for future extensibility)
282
+ // ============================================================================
283
+ // These types support a general-purpose cache interface that can be used
284
+ // for caching arbitrary values (responses, streams, objects). Currently,
285
+ // the segment caching system uses SegmentCacheStore directly, but these
286
+ // types enable future use cases like response caching or data caching.
287
+
288
+ /**
289
+ * Supported cache value types for the generic CacheStore interface.
290
+ * @internal Reserved for future extensibility
291
+ */
292
+ export type CacheValue =
293
+ | ReadableStream<Uint8Array>
294
+ | Response
295
+ | ArrayBuffer
296
+ | string
297
+ | unknown[] // JSON-serializable array
298
+ | Record<string, unknown>; // JSON-serializable object
299
+
300
+ /**
301
+ * Cache entry returned by match().
302
+ * @internal Reserved for future extensibility
303
+ */
304
+ export interface CacheEntry<T = CacheValue> {
305
+ /** The cached value */
306
+ value: T;
307
+ /** Optional metadata stored with the entry */
308
+ metadata?: CacheMetadata;
309
+ }
310
+
311
+ /**
312
+ * Original value type for reconstruction.
313
+ * @internal Reserved for future extensibility
314
+ */
315
+ export type CacheValueType =
316
+ | "stream"
317
+ | "response"
318
+ | "arraybuffer"
319
+ | "string"
320
+ | "object";
321
+
322
+ /**
323
+ * Metadata associated with a cache entry.
324
+ * @internal Reserved for future extensibility
325
+ */
326
+ export interface CacheMetadata {
327
+ /** Timestamp when entry expires (ms since epoch) */
328
+ expiresAt?: number;
329
+ /** Tags for bulk invalidation */
330
+ tags?: string[];
331
+ /** Original value type for reconstruction on read */
332
+ valueType?: CacheValueType;
333
+ /** Response headers (preserved when caching Response) */
334
+ responseHeaders?: Record<string, string>;
335
+ /** Response status (preserved when caching Response) */
336
+ responseStatus?: number;
337
+ /** Custom metadata */
338
+ [key: string]: unknown;
339
+ }
340
+
341
+ /**
342
+ * Options for put().
343
+ * @internal Reserved for future extensibility
344
+ */
345
+ export interface CachePutOptions {
346
+ /** Time-to-live in seconds */
347
+ ttl?: number;
348
+ /** Metadata to store with entry */
349
+ metadata?: Omit<CacheMetadata, "expiresAt">;
350
+ }
351
+
352
+ /**
353
+ * Generic cache store interface for arbitrary value types.
354
+ *
355
+ * This interface is designed for future extensibility to support caching
356
+ * responses, streams, and other values. Currently, segment caching uses
357
+ * the SegmentCacheStore interface directly.
358
+ *
359
+ * Implementations must handle:
360
+ * - Stream values (clone before storing, streams can only be read once)
361
+ * - Promise values (await before storing)
362
+ * - Expiration/TTL
363
+ *
364
+ * @internal Reserved for future extensibility
365
+ */
366
+ export interface CacheStore {
367
+ /**
368
+ * Retrieve a cached entry by key.
369
+ * @param key - Cache key
370
+ * @returns The cached entry or undefined if not found/expired
371
+ */
372
+ match<T = CacheValue>(key: string): Promise<CacheEntry<T> | undefined>;
373
+
374
+ /**
375
+ * Store a value in the cache.
376
+ * @param key - Cache key
377
+ * @param value - Value to cache (stream, response, string, object, etc.)
378
+ * @param options - TTL, metadata, etc.
379
+ */
380
+ put<T extends CacheValue>(
381
+ key: string,
382
+ value: T,
383
+ options?: CachePutOptions
384
+ ): Promise<void>;
385
+
386
+ /**
387
+ * Delete a cached entry.
388
+ * @param key - Cache key
389
+ * @returns true if entry was deleted, false if not found
390
+ */
391
+ delete(key: string): Promise<boolean>;
392
+ }