@rangojs/router 0.0.0-experimental.32 → 0.0.0-experimental.3232cd17
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.
- package/AGENTS.md +4 -0
- package/README.md +198 -44
- package/dist/bin/rango.js +287 -105
- package/dist/testing/vitest.js +82 -0
- package/dist/vite/index.js +3248 -1117
- package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/package.json +73 -21
- package/skills/api-client/SKILL.md +211 -0
- package/skills/breadcrumbs/SKILL.md +107 -1
- package/skills/bundle-analysis/SKILL.md +159 -0
- package/skills/cache-guide/SKILL.md +245 -21
- package/skills/caching/SKILL.md +302 -6
- package/skills/composability/SKILL.md +27 -2
- package/skills/css/SKILL.md +76 -0
- package/skills/document-cache/SKILL.md +78 -55
- package/skills/handler-use/SKILL.md +364 -0
- package/skills/hooks/SKILL.md +270 -30
- package/skills/host-router/SKILL.md +82 -22
- package/skills/i18n/SKILL.md +276 -0
- package/skills/intercept/SKILL.md +49 -5
- package/skills/layout/SKILL.md +35 -9
- package/skills/links/SKILL.md +249 -17
- package/skills/loader/SKILL.md +294 -30
- package/skills/middleware/SKILL.md +52 -13
- package/skills/migrate-nextjs/SKILL.md +584 -0
- package/skills/migrate-react-router/SKILL.md +769 -0
- package/skills/mime-routes/SKILL.md +27 -0
- package/skills/observability/SKILL.md +137 -0
- package/skills/parallel/SKILL.md +203 -7
- package/skills/prerender/SKILL.md +123 -100
- package/skills/rango/SKILL.md +250 -22
- package/skills/react-compiler/SKILL.md +168 -0
- package/skills/response-routes/SKILL.md +122 -47
- package/skills/route/SKILL.md +97 -5
- package/skills/router-setup/SKILL.md +90 -5
- package/skills/server-actions/SKILL.md +775 -0
- package/skills/streams-and-websockets/SKILL.md +283 -0
- package/skills/tailwind/SKILL.md +27 -3
- package/skills/testing/SKILL.md +129 -0
- package/skills/testing/bindings.md +89 -0
- package/skills/testing/cache-prerender.md +124 -0
- package/skills/testing/client-components.md +122 -0
- package/skills/testing/e2e-parity.md +125 -0
- package/skills/testing/flight.md +92 -0
- package/skills/testing/handles.md +129 -0
- package/skills/testing/loader.md +128 -0
- package/skills/testing/middleware.md +99 -0
- package/skills/testing/render-handler.md +121 -0
- package/skills/testing/response-routes.md +95 -0
- package/skills/testing/reverse-and-types.md +84 -0
- package/skills/testing/server-actions.md +107 -0
- package/skills/testing/server-tree.md +128 -0
- package/skills/testing/setup.md +120 -0
- package/skills/typesafety/SKILL.md +329 -27
- package/skills/use-cache/SKILL.md +36 -5
- package/skills/view-transitions/SKILL.md +294 -0
- package/src/__augment-tests__/augment.ts +81 -0
- package/src/__augment-tests__/augmented.check.ts +116 -0
- package/src/__internal.ts +67 -40
- package/src/browser/action-coordinator.ts +53 -36
- package/src/browser/action-fence.ts +47 -0
- package/src/browser/app-shell.ts +39 -0
- package/src/browser/app-version.ts +14 -0
- package/src/browser/cookie-name.ts +140 -0
- package/src/browser/event-controller.ts +86 -147
- package/src/browser/history-state.ts +21 -0
- package/src/browser/index.ts +3 -3
- package/src/browser/invalidate-client-cache.ts +52 -0
- package/src/browser/link-interceptor.ts +4 -0
- package/src/browser/navigation-bridge.ts +148 -19
- package/src/browser/navigation-client.ts +187 -67
- package/src/browser/navigation-store-handle.ts +38 -0
- package/src/browser/navigation-store.ts +76 -67
- package/src/browser/navigation-transaction.ts +18 -66
- package/src/browser/partial-update.ts +123 -94
- package/src/browser/prefetch/cache.ts +214 -36
- package/src/browser/prefetch/fetch.ts +260 -38
- package/src/browser/prefetch/policy.ts +6 -0
- package/src/browser/prefetch/queue.ts +126 -20
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/rango-state.ts +158 -76
- package/src/browser/react/Link.tsx +93 -11
- package/src/browser/react/NavigationProvider.tsx +115 -34
- package/src/browser/react/ScrollRestoration.tsx +10 -6
- package/src/browser/react/context.ts +7 -2
- package/src/browser/react/filter-segment-order.ts +49 -7
- package/src/browser/react/index.ts +0 -48
- package/src/browser/react/location-state-shared.ts +166 -8
- package/src/browser/react/location-state.ts +39 -14
- package/src/browser/react/use-action.ts +6 -15
- package/src/browser/react/use-handle.ts +23 -69
- package/src/browser/react/use-link-status.ts +0 -4
- package/src/browser/react/use-navigation.ts +22 -5
- package/src/browser/react/use-params.ts +20 -10
- package/src/browser/react/use-reverse.ts +106 -0
- package/src/browser/react/use-router.ts +46 -11
- package/src/browser/react/use-search-params.ts +0 -5
- package/src/browser/react/use-segments.ts +11 -21
- package/src/browser/response-adapter.ts +52 -1
- package/src/browser/rsc-router.tsx +215 -76
- package/src/browser/scroll-restoration.ts +46 -39
- package/src/browser/segment-reconciler.ts +36 -9
- package/src/browser/segment-structure-assert.ts +2 -2
- package/src/browser/server-action-bridge.ts +176 -50
- package/src/browser/types.ts +95 -11
- package/src/browser/validate-redirect-origin.ts +43 -16
- package/src/build/collect-fallback-refs.ts +107 -0
- package/src/build/generate-manifest.ts +65 -40
- package/src/build/generate-route-types.ts +5 -0
- package/src/build/index.ts +8 -2
- package/src/build/prefix-tree-utils.ts +123 -0
- package/src/build/route-trie.ts +137 -32
- package/src/build/route-types/codegen.ts +4 -4
- package/src/build/route-types/include-resolution.ts +9 -2
- package/src/build/route-types/param-extraction.ts +6 -3
- package/src/build/route-types/per-module-writer.ts +7 -4
- package/src/build/route-types/router-processing.ts +278 -96
- package/src/build/route-types/scan-filter.ts +9 -2
- package/src/build/route-types/source-scan.ts +118 -0
- package/src/build/runtime-discovery.ts +9 -20
- package/src/cache/cache-error.ts +104 -0
- package/src/cache/cache-policy.ts +68 -28
- package/src/cache/cache-runtime.ts +149 -43
- package/src/cache/cache-scope.ts +148 -81
- package/src/cache/cache-tag.ts +98 -0
- package/src/cache/cf/cf-cache-store.ts +2550 -93
- package/src/cache/cf/index.ts +11 -17
- package/src/cache/document-cache.ts +78 -27
- package/src/cache/handle-snapshot.ts +63 -0
- package/src/cache/index.ts +23 -20
- package/src/cache/memory-segment-store.ts +136 -37
- package/src/cache/profile-registry.ts +6 -30
- package/src/cache/read-through-swr.ts +41 -11
- package/src/cache/segment-codec.ts +0 -16
- package/src/cache/tag-invalidation.ts +230 -0
- package/src/cache/taint.ts +55 -0
- package/src/cache/types.ts +33 -100
- package/src/cache/vercel/index.ts +11 -0
- package/src/cache/vercel/vercel-cache-store.ts +799 -0
- package/src/client.rsc.tsx +6 -21
- package/src/client.tsx +108 -290
- package/src/component-utils.ts +19 -0
- package/src/context-var.ts +84 -2
- package/src/debug.ts +2 -2
- package/src/decode-loader-results.ts +36 -0
- package/src/defer.ts +196 -0
- package/src/deps/ssr.ts +0 -1
- package/src/errors.ts +30 -4
- package/src/handle.ts +70 -22
- package/src/handles/MetaTags.tsx +0 -14
- package/src/handles/breadcrumbs.ts +16 -5
- package/src/handles/meta.ts +0 -39
- package/src/host/cookie-handler.ts +0 -36
- package/src/host/errors.ts +0 -24
- package/src/host/index.ts +8 -2
- package/src/host/pattern-matcher.ts +7 -50
- package/src/host/router.ts +107 -99
- package/src/host/testing.ts +40 -27
- package/src/host/types.ts +37 -4
- package/src/host/utils.ts +1 -1
- package/src/href-client.ts +137 -22
- package/src/index.rsc.ts +52 -26
- package/src/index.ts +100 -38
- package/src/internal-debug.ts +2 -4
- package/src/loader-store.ts +500 -0
- package/src/loader.rsc.ts +20 -13
- package/src/loader.ts +12 -11
- package/src/missing-id-error.ts +68 -0
- package/src/network-error-thrower.tsx +1 -6
- package/src/outlet-context.ts +1 -1
- package/src/outlet-provider.tsx +1 -5
- package/src/prerender/param-hash.ts +10 -11
- package/src/prerender/store.ts +37 -41
- package/src/prerender.ts +198 -82
- package/src/redirect-origin.ts +100 -0
- package/src/response-utils.ts +37 -0
- package/src/reverse.ts +65 -15
- package/src/root-error-boundary.tsx +1 -19
- package/src/route-content-wrapper.tsx +7 -72
- package/src/route-definition/dsl-helpers.ts +437 -274
- package/src/route-definition/helper-factories.ts +29 -139
- package/src/route-definition/helpers-types.ts +113 -37
- package/src/route-definition/index.ts +3 -0
- package/src/route-definition/redirect.ts +52 -10
- package/src/route-definition/resolve-handler-use.ts +161 -0
- package/src/route-definition/use-item-types.ts +32 -0
- package/src/route-map-builder.ts +7 -17
- package/src/route-types.ts +37 -41
- package/src/router/basename.ts +14 -0
- package/src/router/content-negotiation.ts +108 -9
- package/src/router/error-handling.ts +13 -17
- package/src/router/find-match.ts +45 -22
- package/src/router/handler-context.ts +83 -41
- package/src/router/intercept-resolution.ts +25 -23
- package/src/router/lazy-includes.ts +19 -53
- package/src/router/loader-resolution.ts +213 -30
- package/src/router/logging.ts +5 -8
- package/src/router/manifest.ts +49 -45
- package/src/router/match-api.ts +120 -204
- package/src/router/match-context.ts +0 -22
- package/src/router/match-handlers.ts +58 -58
- package/src/router/match-middleware/background-revalidation.ts +27 -6
- package/src/router/match-middleware/cache-lookup.ts +205 -249
- package/src/router/match-middleware/cache-store.ts +45 -32
- package/src/router/match-middleware/intercept-resolution.ts +8 -28
- package/src/router/match-middleware/segment-resolution.ts +52 -18
- package/src/router/match-pipelines.ts +1 -42
- package/src/router/match-result.ts +104 -40
- package/src/router/metrics.ts +5 -34
- package/src/router/middleware-types.ts +13 -142
- package/src/router/middleware.ts +173 -143
- package/src/router/navigation-snapshot.ts +131 -0
- package/src/router/params-util.ts +23 -0
- package/src/router/pattern-matching.ts +109 -63
- package/src/router/prerender-match.ts +190 -54
- package/src/router/preview-match.ts +32 -102
- package/src/router/request-classification.ts +276 -0
- package/src/router/revalidation.ts +63 -55
- package/src/router/route-snapshot.ts +244 -0
- package/src/router/router-context.ts +6 -28
- package/src/router/router-interfaces.ts +100 -35
- package/src/router/router-options.ts +91 -11
- package/src/router/router-registry.ts +2 -5
- package/src/router/segment-resolution/fresh.ts +242 -75
- package/src/router/segment-resolution/helpers.ts +63 -24
- package/src/router/segment-resolution/loader-cache.ts +41 -37
- package/src/router/segment-resolution/revalidation.ts +456 -372
- package/src/router/segment-resolution/static-store.ts +19 -5
- package/src/router/segment-resolution/streamed-handler-telemetry.ts +52 -0
- package/src/router/segment-resolution/view-transition-default.ts +36 -0
- package/src/router/segment-resolution.ts +4 -1
- package/src/router/segment-wrappers.ts +2 -3
- package/src/router/state-cookie-name.ts +33 -0
- package/src/router/substitute-pattern-params.ts +56 -0
- package/src/router/telemetry-otel.ts +0 -20
- package/src/router/telemetry.ts +96 -19
- package/src/router/timeout.ts +0 -20
- package/src/router/trie-matching.ts +91 -46
- package/src/router/types.ts +10 -63
- package/src/router/url-params.ts +44 -0
- package/src/router.ts +134 -43
- package/src/rsc/handler-context.ts +3 -2
- package/src/rsc/handler.ts +492 -383
- package/src/rsc/helpers.ts +162 -46
- package/src/rsc/index.ts +1 -1
- package/src/rsc/json-route-result.ts +38 -0
- package/src/rsc/loader-fetch.ts +23 -3
- package/src/rsc/manifest-init.ts +33 -42
- package/src/rsc/origin-guard.ts +39 -25
- package/src/rsc/progressive-enhancement.ts +30 -3
- package/src/rsc/redirect-guard.ts +99 -0
- package/src/rsc/response-error.ts +79 -12
- package/src/rsc/response-route-handler.ts +90 -63
- package/src/rsc/rsc-rendering.ts +56 -54
- package/src/rsc/runtime-warnings.ts +23 -10
- package/src/rsc/server-action.ts +74 -67
- package/src/rsc/ssr-setup.ts +18 -2
- package/src/rsc/types.ts +25 -6
- package/src/runtime-env.ts +18 -0
- package/src/search-params.ts +4 -20
- package/src/segment-content-promise.ts +67 -0
- package/src/segment-loader-promise.ts +134 -0
- package/src/segment-system.tsx +272 -129
- package/src/serialize.ts +243 -0
- package/src/server/context.ts +309 -61
- package/src/server/cookie-store.ts +80 -5
- package/src/server/handle-store.ts +26 -24
- package/src/server/loader-registry.ts +10 -28
- package/src/server/request-context.ts +338 -126
- package/src/ssr/index.tsx +23 -15
- package/src/static-handler.ts +27 -18
- package/src/testing/cache-status.ts +162 -0
- package/src/testing/collect-handle.ts +40 -0
- package/src/testing/dispatch.ts +618 -0
- package/src/testing/dom.entry.ts +22 -0
- package/src/testing/e2e/fixture.ts +188 -0
- package/src/testing/e2e/index.ts +128 -0
- package/src/testing/e2e/matchers.ts +35 -0
- package/src/testing/e2e/page-helpers.ts +272 -0
- package/src/testing/e2e/parity.ts +387 -0
- package/src/testing/e2e/server.ts +195 -0
- package/src/testing/flight-matchers.ts +97 -0
- package/src/testing/flight-normalize.ts +11 -0
- package/src/testing/flight-runtime.d.ts +57 -0
- package/src/testing/flight-tree.ts +682 -0
- package/src/testing/flight.entry.ts +52 -0
- package/src/testing/flight.ts +232 -0
- package/src/testing/generated-routes.ts +183 -0
- package/src/testing/index.ts +99 -0
- package/src/testing/internal/context.ts +348 -0
- package/src/testing/internal/flight-client-globals.ts +30 -0
- package/src/testing/internal/seed-vars.ts +54 -0
- package/src/testing/render-handler.ts +330 -0
- package/src/testing/render-route.tsx +566 -0
- package/src/testing/run-loader.ts +378 -0
- package/src/testing/run-middleware.ts +205 -0
- package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
- package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
- package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
- package/src/testing/vitest-stubs/version.ts +5 -0
- package/src/testing/vitest.ts +305 -0
- package/src/theme/ThemeProvider.tsx +0 -52
- package/src/theme/ThemeScript.tsx +0 -6
- package/src/theme/constants.ts +0 -12
- package/src/theme/index.ts +0 -7
- package/src/theme/theme-context.ts +1 -5
- package/src/theme/theme-script.ts +0 -14
- package/src/theme/use-theme.ts +0 -3
- package/src/types/boundaries.ts +0 -35
- package/src/types/cache-types.ts +17 -8
- package/src/types/error-types.ts +30 -90
- package/src/types/global-namespace.ts +54 -41
- package/src/types/handler-context.ts +233 -81
- package/src/types/index.ts +1 -10
- package/src/types/loader-types.ts +44 -15
- package/src/types/request-scope.ts +107 -0
- package/src/types/route-config.ts +6 -50
- package/src/types/route-entry.ts +19 -7
- package/src/types/segments.ts +37 -14
- package/src/urls/include-helper.ts +33 -70
- package/src/urls/index.ts +1 -11
- package/src/urls/path-helper-types.ts +58 -11
- package/src/urls/path-helper.ts +57 -111
- package/src/urls/pattern-types.ts +48 -19
- package/src/urls/response-types.ts +25 -22
- package/src/urls/type-extraction.ts +58 -139
- package/src/urls/urls-function.ts +1 -18
- package/src/use-loader.tsx +346 -89
- package/src/vite/debug.ts +185 -0
- package/src/vite/discovery/bundle-postprocess.ts +36 -38
- package/src/vite/discovery/discover-routers.ts +130 -85
- package/src/vite/discovery/discovery-errors.ts +194 -0
- package/src/vite/discovery/gate-state.ts +171 -0
- package/src/vite/discovery/prerender-collection.ts +192 -99
- package/src/vite/discovery/route-types-writer.ts +40 -84
- package/src/vite/discovery/self-gen-tracking.ts +27 -1
- package/src/vite/discovery/state.ts +51 -6
- package/src/vite/discovery/virtual-module-codegen.ts +14 -34
- package/src/vite/index.ts +8 -0
- package/src/vite/plugin-types.ts +187 -69
- package/src/vite/plugins/cjs-to-esm.ts +8 -18
- package/src/vite/plugins/client-ref-dedup.ts +16 -11
- package/src/vite/plugins/client-ref-hashing.ts +28 -15
- package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
- package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/src/vite/plugins/cloudflare-protocol-stub.ts +194 -0
- package/src/vite/plugins/expose-action-id.ts +49 -98
- package/src/vite/plugins/expose-id-utils.ts +11 -50
- package/src/vite/plugins/expose-ids/export-analysis.ts +76 -34
- package/src/vite/plugins/expose-ids/handler-transform.ts +10 -48
- package/src/vite/plugins/expose-ids/loader-transform.ts +3 -20
- package/src/vite/plugins/expose-ids/router-transform.ts +20 -16
- package/src/vite/plugins/expose-internal-ids.ts +554 -317
- package/src/vite/plugins/performance-tracks.ts +89 -0
- package/src/vite/plugins/refresh-cmd.ts +89 -27
- package/src/vite/plugins/use-cache-transform.ts +73 -83
- package/src/vite/plugins/vercel-output.ts +258 -0
- package/src/vite/plugins/version-injector.ts +21 -25
- package/src/vite/plugins/version-plugin.ts +41 -20
- package/src/vite/plugins/virtual-entries.ts +2 -17
- package/src/vite/rango.ts +257 -289
- package/src/vite/router-discovery.ts +930 -140
- package/src/vite/utils/ast-handler-extract.ts +15 -31
- package/src/vite/utils/banner.ts +4 -4
- package/src/vite/utils/bundle-analysis.ts +10 -15
- package/src/vite/utils/client-chunks.ts +184 -0
- package/src/vite/utils/forward-user-plugins.ts +171 -0
- package/src/vite/utils/manifest-utils.ts +4 -59
- package/src/vite/utils/package-resolution.ts +20 -52
- package/src/vite/utils/prerender-utils.ts +27 -29
- package/src/vite/utils/shared-utils.ts +92 -42
- package/src/browser/action-response-classifier.ts +0 -99
- package/src/browser/react/use-client-cache.ts +0 -58
- package/src/browser/shallow.ts +0 -40
- package/src/handles/index.ts +0 -7
- package/src/router/middleware-cookies.ts +0 -55
|
@@ -1,54 +1,160 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Prefetch Cache
|
|
3
3
|
*
|
|
4
|
-
* In-memory cache storing prefetch
|
|
5
|
-
* on subsequent navigation.
|
|
6
|
-
*
|
|
7
|
-
*
|
|
4
|
+
* In-memory cache storing eagerly-decoded prefetch payloads for instant,
|
|
5
|
+
* already-warm cache hits on subsequent navigation. A prefetch fetches the
|
|
6
|
+
* RSC partial AND decodes it (createFromFetch) up front — decoding the Flight
|
|
7
|
+
* stream resolves the route's client references, so the route's JS chunks are
|
|
8
|
+
* imported during prefetch rather than on click. The decoded payload is reused
|
|
9
|
+
* verbatim by navigation, so a prefetched click loads no new code. Two key
|
|
10
|
+
* scopes are in play:
|
|
11
|
+
* - Wildcard (default): built by `buildPrefetchKey(rangoState, target)` —
|
|
12
|
+
* shape `rangoState\0/target?...`. Shared across all source pages and
|
|
13
|
+
* invalidated automatically when Rango state bumps (deploy or
|
|
14
|
+
* server-action invalidation).
|
|
15
|
+
* - Source-scoped: built by `buildSourceKey(rangoState, sourceHref, target)`
|
|
16
|
+
* — shape `rangoState\0sourceHref\0/target?...`. Embeds the Rango state
|
|
17
|
+
* (so rotation invalidates source-scoped entries too) plus the source
|
|
18
|
+
* href (so each originating page gets its own slot). Populated when the
|
|
19
|
+
* server tags a response with `X-RSC-Prefetch-Scope: source` (intercept
|
|
20
|
+
* modals etc.), OR when a Link opts in with `prefetchKey=":source"` — in
|
|
21
|
+
* both cases so source-sensitive responses cannot bleed into navigations
|
|
22
|
+
* from other pages.
|
|
23
|
+
*
|
|
24
|
+
* Also tracks in-flight prefetch promises. Each promise resolves to the
|
|
25
|
+
* decoded prefetch entry (or null), letting navigation adopt a
|
|
26
|
+
* still-downloading prefetch without issuing a duplicate request. A
|
|
27
|
+
* single promise can be registered under multiple alias keys (see
|
|
28
|
+
* `setInflightPromiseWithAliases`) so same-source navigations adopt via
|
|
29
|
+
* their source key while cross-source ones fall through to the wildcard
|
|
30
|
+
* alias — with consume/clear atomically removing every alias.
|
|
8
31
|
*
|
|
9
32
|
* Replaces the previous browser HTTP cache approach which was unreliable
|
|
10
33
|
* due to response draining race conditions and browser inconsistencies.
|
|
34
|
+
*
|
|
35
|
+
* State here lives in module-level singletons (cache, inflight, generation,
|
|
36
|
+
* cacheTTL, etc.) rather than a per-instance factory. This is correct because
|
|
37
|
+
* exactly one router is live per document — an SPA navigation crossing a
|
|
38
|
+
* host-router boundary forces a full document reload — so the singletons are
|
|
39
|
+
* effectively per-document. Unit tests reset them via clearPrefetchCache().
|
|
11
40
|
*/
|
|
12
41
|
|
|
13
|
-
import {
|
|
42
|
+
import { abortAllPrefetches } from "./queue.js";
|
|
14
43
|
import { invalidateRangoState } from "../rango-state.js";
|
|
44
|
+
import type { RscPayload } from "../types.js";
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* A prefetch that has been fetched AND eagerly decoded. Storing the decoded
|
|
48
|
+
* payload (not the raw Response) is what makes a prefetched navigation "warm":
|
|
49
|
+
* decoding the Flight stream during prefetch pulls the route's client chunks,
|
|
50
|
+
* so the click reuses ready elements and loads no new JS.
|
|
51
|
+
*/
|
|
52
|
+
export interface DecodedPrefetch {
|
|
53
|
+
/** The eagerly-decoded RSC payload. Reused verbatim by navigation. */
|
|
54
|
+
payload: Promise<RscPayload>;
|
|
55
|
+
/**
|
|
56
|
+
* Resolves when the underlying RSC stream finishes draining. Navigation
|
|
57
|
+
* forwards this as its streamComplete so scroll/revalidation gating is
|
|
58
|
+
* unchanged from the fresh-fetch path.
|
|
59
|
+
*/
|
|
60
|
+
streamComplete: Promise<void>;
|
|
61
|
+
/**
|
|
62
|
+
* Prefetch scope as tagged by the server via `X-RSC-Prefetch-Scope`.
|
|
63
|
+
* `"source"` means the response is source-page-sensitive and must not be
|
|
64
|
+
* reused by a navigation from a different page — navigation enforces this
|
|
65
|
+
* when it adopted an inflight entry through the wildcard key.
|
|
66
|
+
*/
|
|
67
|
+
scope: "source" | "wildcard";
|
|
68
|
+
}
|
|
15
69
|
|
|
16
|
-
// Default TTL: 5 minutes. Overridden by initPrefetchCache() with
|
|
17
|
-
// the server-configured prefetchCacheTTL from router options.
|
|
18
|
-
// 0 disables the in-memory cache entirely.
|
|
19
70
|
let cacheTTL = 300_000;
|
|
20
71
|
|
|
21
72
|
/**
|
|
22
73
|
* Initialize the prefetch cache with the configured TTL.
|
|
23
74
|
* Called once at app startup with the value from server metadata.
|
|
24
|
-
* A TTL of 0 disables the in-memory cache.
|
|
75
|
+
* A TTL of 0 disables the in-memory cache and all prefetching.
|
|
25
76
|
*/
|
|
26
77
|
export function initPrefetchCache(ttlMs: number): void {
|
|
27
78
|
cacheTTL = ttlMs;
|
|
28
79
|
}
|
|
80
|
+
|
|
81
|
+
/**
|
|
82
|
+
* Check if the prefetch cache is disabled (TTL <= 0).
|
|
83
|
+
* When disabled, no prefetch requests should be issued.
|
|
84
|
+
*/
|
|
85
|
+
export function isPrefetchCacheDisabled(): boolean {
|
|
86
|
+
return cacheTTL <= 0;
|
|
87
|
+
}
|
|
29
88
|
const MAX_PREFETCH_CACHE_SIZE = 50;
|
|
30
89
|
|
|
31
90
|
interface PrefetchCacheEntry {
|
|
32
|
-
|
|
91
|
+
entry: DecodedPrefetch;
|
|
33
92
|
timestamp: number;
|
|
34
93
|
}
|
|
35
94
|
|
|
36
95
|
const cache = new Map<string, PrefetchCacheEntry>();
|
|
37
96
|
const inflight = new Set<string>();
|
|
38
97
|
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
98
|
+
const inflightPromises = new Map<string, Promise<DecodedPrefetch | null>>();
|
|
99
|
+
|
|
100
|
+
const inflightAliases = new Map<string, string[]>();
|
|
101
|
+
|
|
102
|
+
const adoptedKeys = new Set<string>();
|
|
103
|
+
|
|
42
104
|
let generation = 0;
|
|
43
105
|
|
|
44
106
|
/**
|
|
45
|
-
* Build a
|
|
46
|
-
*
|
|
47
|
-
*
|
|
48
|
-
*
|
|
107
|
+
* Build a cache key by combining a scope prefix with the target URL.
|
|
108
|
+
*
|
|
109
|
+
* Low-level primitive — callers that want a specific scope should use
|
|
110
|
+
* one of:
|
|
111
|
+
* - Wildcard (source-agnostic): prefix is the Rango state value from
|
|
112
|
+
* `getRangoState()`. Shared across all source pages. Invalidated
|
|
113
|
+
* automatically when Rango state bumps (deploy or server-action).
|
|
114
|
+
* Key shape: `rangoState\0/target?...`.
|
|
115
|
+
* - Source-scoped: use `buildSourceKey()`. Key shape:
|
|
116
|
+
* `rangoState\0sourceHref\0/target?...` — embeds the Rango state so
|
|
117
|
+
* rotation invalidates source-scoped entries alongside wildcard ones,
|
|
118
|
+
* plus the source page href so the key is unique per originating page.
|
|
119
|
+
* Populated either when the server tags a response with
|
|
120
|
+
* `X-RSC-Prefetch-Scope: source` (intercept modals, etc.) or when a
|
|
121
|
+
* Link opts in via `prefetchKey=":source"`.
|
|
122
|
+
*
|
|
123
|
+
* The `_rsc_segments` query param that travels in the target URL means
|
|
124
|
+
* clients with different mounted segment trees naturally get different
|
|
125
|
+
* keys — so segment-level diffs remain consistent across both scopes.
|
|
126
|
+
*/
|
|
127
|
+
export function buildPrefetchKey(prefix: string, targetUrl: URL): string {
|
|
128
|
+
return prefix + "\0" + targetUrl.pathname + targetUrl.search;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Build a source-scoped cache key. Key shape:
|
|
133
|
+
* `rangoState\0sourceHref\0/target?...`.
|
|
134
|
+
*
|
|
135
|
+
* - `rangoState` is included so state rotation invalidates source-scoped
|
|
136
|
+
* entries alongside wildcard ones.
|
|
137
|
+
* - `sourceHref` makes the key unique per originating page.
|
|
138
|
+
*/
|
|
139
|
+
export function buildSourceKey(
|
|
140
|
+
rangoState: string,
|
|
141
|
+
sourceHref: string,
|
|
142
|
+
targetUrl: URL,
|
|
143
|
+
): string {
|
|
144
|
+
return buildPrefetchKey(rangoState + "\0" + sourceHref, targetUrl);
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
/**
|
|
148
|
+
* Walk an inflight key plus any sibling aliases registered via
|
|
149
|
+
* `setInflightPromiseWithAliases`, invoking `fn` for each.
|
|
49
150
|
*/
|
|
50
|
-
|
|
51
|
-
|
|
151
|
+
function forEachAlias(key: string, fn: (k: string) => void): void {
|
|
152
|
+
const aliases = inflightAliases.get(key);
|
|
153
|
+
if (aliases) {
|
|
154
|
+
for (const k of aliases) fn(k);
|
|
155
|
+
} else {
|
|
156
|
+
fn(key);
|
|
157
|
+
}
|
|
52
158
|
}
|
|
53
159
|
|
|
54
160
|
/**
|
|
@@ -67,11 +173,14 @@ export function hasPrefetch(key: string): boolean {
|
|
|
67
173
|
}
|
|
68
174
|
|
|
69
175
|
/**
|
|
70
|
-
* Consume a cached prefetch
|
|
71
|
-
* One-time consumption: the entry is deleted after retrieval.
|
|
176
|
+
* Consume a cached, eagerly-decoded prefetch. Returns null if not found or
|
|
177
|
+
* expired. One-time consumption: the entry is deleted after retrieval.
|
|
72
178
|
* Returns null when caching is disabled (TTL <= 0).
|
|
179
|
+
*
|
|
180
|
+
* Does NOT check in-flight prefetches — use consumeInflightPrefetch()
|
|
181
|
+
* for that (returns a Promise instead of a resolved entry).
|
|
73
182
|
*/
|
|
74
|
-
export function consumePrefetch(key: string):
|
|
183
|
+
export function consumePrefetch(key: string): DecodedPrefetch | null {
|
|
75
184
|
if (cacheTTL <= 0) return null;
|
|
76
185
|
const entry = cache.get(key);
|
|
77
186
|
if (!entry) return null;
|
|
@@ -80,29 +189,72 @@ export function consumePrefetch(key: string): Response | null {
|
|
|
80
189
|
return null;
|
|
81
190
|
}
|
|
82
191
|
cache.delete(key);
|
|
83
|
-
return entry.
|
|
192
|
+
return entry.entry;
|
|
84
193
|
}
|
|
85
194
|
|
|
86
195
|
/**
|
|
87
|
-
*
|
|
88
|
-
*
|
|
89
|
-
*
|
|
196
|
+
* Consume an in-flight prefetch promise. Returns null if no prefetch is
|
|
197
|
+
* in-flight for this key. The returned Promise resolves to the decoded
|
|
198
|
+
* prefetch entry (or null if the fetch failed/was aborted, or carried a
|
|
199
|
+
* control header the navigation must re-fetch to honor).
|
|
200
|
+
*
|
|
201
|
+
* One-time consumption: the promise entry is removed (along with any
|
|
202
|
+
* sibling aliases registered via `setInflightPromiseWithAliases`) so a
|
|
203
|
+
* second call on any alias returns null — only one caller can adopt the
|
|
204
|
+
* shared Response stream. The `inflight` set entry is intentionally
|
|
205
|
+
* kept so that `hasPrefetch()` continues to return true while the
|
|
206
|
+
* underlying fetch is still downloading — this prevents
|
|
207
|
+
* `prefetchDirect()` or other callers from starting a duplicate request
|
|
208
|
+
* during the handoff window. The inflight flag is cleaned up naturally
|
|
209
|
+
* by `clearPrefetchInflight()` in the fetch's `.finally()`.
|
|
210
|
+
*/
|
|
211
|
+
export function consumeInflightPrefetch(
|
|
212
|
+
key: string,
|
|
213
|
+
): Promise<DecodedPrefetch | null> | null {
|
|
214
|
+
const promise = inflightPromises.get(key);
|
|
215
|
+
if (!promise) return null;
|
|
216
|
+
// Remove the promise under every alias so a second consumer cannot
|
|
217
|
+
// adopt the same stream and race on the body, and mark every alias as
|
|
218
|
+
// adopted so the pending `storePrefetch` (which resolves later, after this
|
|
219
|
+
// adoption) does not leave the now-owned, single-use entry in the cache map.
|
|
220
|
+
// `inflightAliases` is intentionally preserved — `clearPrefetchInflight()` in
|
|
221
|
+
// the fetch's `.finally()` still needs it to clear every inflight flag and
|
|
222
|
+
// adopted marker; deleting here would strand the sibling's flag forever.
|
|
223
|
+
forEachAlias(key, (k) => {
|
|
224
|
+
inflightPromises.delete(k);
|
|
225
|
+
adoptedKeys.add(k);
|
|
226
|
+
});
|
|
227
|
+
return promise;
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
/**
|
|
231
|
+
* Store an eagerly-decoded prefetch in the in-memory cache.
|
|
90
232
|
*
|
|
91
233
|
* Skips storage if the generation has changed since the fetch started
|
|
92
234
|
* (a server action invalidated the cache mid-flight).
|
|
93
235
|
*/
|
|
94
236
|
export function storePrefetch(
|
|
95
237
|
key: string,
|
|
96
|
-
|
|
238
|
+
entry: DecodedPrefetch,
|
|
97
239
|
fetchGeneration: number,
|
|
98
240
|
): void {
|
|
99
241
|
if (cacheTTL <= 0) return;
|
|
100
242
|
if (fetchGeneration !== generation) return;
|
|
101
243
|
|
|
244
|
+
// If a navigation already adopted this prefetch's in-flight promise, it owns
|
|
245
|
+
// the single-use entry (and has drained its handle generator). Do NOT also
|
|
246
|
+
// publish it to the cache map, or a later navigation would be served the
|
|
247
|
+
// exhausted entry and lose that route's handles. Clear the marker (under all
|
|
248
|
+
// aliases) now that the decision is made.
|
|
249
|
+
if (adoptedKeys.has(key)) {
|
|
250
|
+
forEachAlias(key, (k) => adoptedKeys.delete(k));
|
|
251
|
+
return;
|
|
252
|
+
}
|
|
253
|
+
|
|
102
254
|
// Evict expired entries
|
|
103
255
|
const now = Date.now();
|
|
104
|
-
for (const [k,
|
|
105
|
-
if (now -
|
|
256
|
+
for (const [k, cached] of cache) {
|
|
257
|
+
if (now - cached.timestamp > cacheTTL) {
|
|
106
258
|
cache.delete(k);
|
|
107
259
|
}
|
|
108
260
|
}
|
|
@@ -113,7 +265,7 @@ export function storePrefetch(
|
|
|
113
265
|
if (oldest) cache.delete(oldest);
|
|
114
266
|
}
|
|
115
267
|
|
|
116
|
-
cache.set(key, {
|
|
268
|
+
cache.set(key, { entry, timestamp: now });
|
|
117
269
|
}
|
|
118
270
|
|
|
119
271
|
/**
|
|
@@ -128,19 +280,45 @@ export function markPrefetchInflight(key: string): void {
|
|
|
128
280
|
inflight.add(key);
|
|
129
281
|
}
|
|
130
282
|
|
|
131
|
-
export function
|
|
132
|
-
|
|
283
|
+
export function setInflightPromise(
|
|
284
|
+
key: string,
|
|
285
|
+
promise: Promise<DecodedPrefetch | null>,
|
|
286
|
+
): void {
|
|
287
|
+
inflightPromises.set(key, promise);
|
|
133
288
|
}
|
|
134
289
|
|
|
135
290
|
/**
|
|
136
|
-
*
|
|
137
|
-
*
|
|
138
|
-
*
|
|
291
|
+
* Store the same in-flight Promise under multiple keys, recording them
|
|
292
|
+
* as sibling aliases. Consuming or clearing any one alias atomically
|
|
293
|
+
* removes every entry, guaranteeing the shared Response stream has a
|
|
294
|
+
* single consumer even when navigation looks up either key.
|
|
139
295
|
*/
|
|
296
|
+
export function setInflightPromiseWithAliases(
|
|
297
|
+
keys: string[],
|
|
298
|
+
promise: Promise<DecodedPrefetch | null>,
|
|
299
|
+
): void {
|
|
300
|
+
for (const k of keys) {
|
|
301
|
+
inflightPromises.set(k, promise);
|
|
302
|
+
inflightAliases.set(k, keys);
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
export function clearPrefetchInflight(key: string): void {
|
|
307
|
+
forEachAlias(key, (k) => {
|
|
308
|
+
inflight.delete(k);
|
|
309
|
+
inflightPromises.delete(k);
|
|
310
|
+
inflightAliases.delete(k);
|
|
311
|
+
adoptedKeys.delete(k);
|
|
312
|
+
});
|
|
313
|
+
}
|
|
314
|
+
|
|
140
315
|
export function clearPrefetchCache(): void {
|
|
141
316
|
generation++;
|
|
142
317
|
inflight.clear();
|
|
318
|
+
inflightPromises.clear();
|
|
319
|
+
inflightAliases.clear();
|
|
320
|
+
adoptedKeys.clear();
|
|
143
321
|
cache.clear();
|
|
144
|
-
|
|
322
|
+
abortAllPrefetches();
|
|
145
323
|
invalidateRangoState();
|
|
146
324
|
}
|