@rangojs/router 0.0.0-experimental.8 → 0.0.0-experimental.81
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 +9 -0
- package/README.md +942 -4
- package/dist/bin/rango.js +1689 -0
- package/dist/vite/index.js +5091 -941
- package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/package.json +61 -52
- package/skills/breadcrumbs/SKILL.md +250 -0
- package/skills/cache-guide/SKILL.md +294 -0
- package/skills/caching/SKILL.md +93 -23
- package/skills/composability/SKILL.md +172 -0
- package/skills/debug-manifest/SKILL.md +12 -8
- package/skills/document-cache/SKILL.md +18 -16
- package/skills/fonts/SKILL.md +167 -0
- package/skills/handler-use/SKILL.md +362 -0
- package/skills/hooks/SKILL.md +340 -72
- package/skills/host-router/SKILL.md +218 -0
- package/skills/intercept/SKILL.md +151 -8
- package/skills/layout/SKILL.md +122 -3
- package/skills/links/SKILL.md +92 -31
- package/skills/loader/SKILL.md +404 -44
- package/skills/middleware/SKILL.md +205 -37
- package/skills/migrate-nextjs/SKILL.md +560 -0
- package/skills/migrate-react-router/SKILL.md +765 -0
- package/skills/mime-routes/SKILL.md +128 -0
- package/skills/parallel/SKILL.md +263 -1
- package/skills/prerender/SKILL.md +685 -0
- package/skills/rango/SKILL.md +87 -16
- package/skills/response-routes/SKILL.md +411 -0
- package/skills/route/SKILL.md +281 -14
- package/skills/router-setup/SKILL.md +210 -32
- package/skills/tailwind/SKILL.md +129 -0
- package/skills/theme/SKILL.md +9 -8
- package/skills/typesafety/SKILL.md +328 -89
- package/skills/use-cache/SKILL.md +324 -0
- package/src/__internal.ts +102 -4
- package/src/bin/rango.ts +321 -0
- package/src/browser/action-coordinator.ts +97 -0
- package/src/browser/action-response-classifier.ts +99 -0
- package/src/browser/app-version.ts +14 -0
- package/src/browser/event-controller.ts +92 -64
- package/src/browser/history-state.ts +80 -0
- package/src/browser/intercept-utils.ts +52 -0
- package/src/browser/link-interceptor.ts +24 -4
- package/src/browser/logging.ts +55 -0
- package/src/browser/merge-segment-loaders.ts +20 -12
- package/src/browser/navigation-bridge.ts +317 -560
- package/src/browser/navigation-client.ts +206 -68
- package/src/browser/navigation-store.ts +73 -55
- package/src/browser/navigation-transaction.ts +297 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +343 -316
- package/src/browser/prefetch/cache.ts +216 -0
- package/src/browser/prefetch/fetch.ts +206 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +48 -0
- package/src/browser/prefetch/queue.ts +160 -0
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/rango-state.ts +112 -0
- package/src/browser/react/Link.tsx +253 -74
- package/src/browser/react/NavigationProvider.tsx +91 -11
- package/src/browser/react/context.ts +11 -0
- package/src/browser/react/filter-segment-order.ts +11 -0
- package/src/browser/react/index.ts +12 -12
- package/src/browser/react/location-state-shared.ts +95 -53
- package/src/browser/react/location-state.ts +60 -15
- package/src/browser/react/mount-context.ts +6 -1
- package/src/browser/react/nonce-context.ts +23 -0
- package/src/browser/react/shallow-equal.ts +27 -0
- package/src/browser/react/use-action.ts +29 -51
- package/src/browser/react/use-client-cache.ts +5 -3
- package/src/browser/react/use-handle.ts +30 -126
- package/src/browser/react/use-href.tsx +2 -2
- package/src/browser/react/use-link-status.ts +6 -5
- package/src/browser/react/use-navigation.ts +44 -65
- package/src/browser/react/use-params.ts +75 -0
- package/src/browser/react/use-pathname.ts +47 -0
- package/src/browser/react/use-router.ts +76 -0
- package/src/browser/react/use-search-params.ts +56 -0
- package/src/browser/react/use-segments.ts +80 -97
- package/src/browser/response-adapter.ts +73 -0
- package/src/browser/rsc-router.tsx +214 -58
- package/src/browser/scroll-restoration.ts +127 -52
- package/src/browser/segment-reconciler.ts +243 -0
- package/src/browser/segment-structure-assert.ts +16 -0
- package/src/browser/server-action-bridge.ts +510 -603
- package/src/browser/shallow.ts +6 -1
- package/src/browser/types.ts +141 -48
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +235 -24
- package/src/build/generate-route-types.ts +39 -0
- package/src/build/index.ts +13 -0
- package/src/build/route-trie.ts +291 -0
- package/src/build/route-types/ast-helpers.ts +25 -0
- package/src/build/route-types/ast-route-extraction.ts +98 -0
- package/src/build/route-types/codegen.ts +102 -0
- package/src/build/route-types/include-resolution.ts +418 -0
- package/src/build/route-types/param-extraction.ts +48 -0
- package/src/build/route-types/per-module-writer.ts +128 -0
- package/src/build/route-types/router-processing.ts +618 -0
- package/src/build/route-types/scan-filter.ts +85 -0
- package/src/build/runtime-discovery.ts +231 -0
- package/src/cache/background-task.ts +34 -0
- package/src/cache/cache-key-utils.ts +44 -0
- package/src/cache/cache-policy.ts +125 -0
- package/src/cache/cache-runtime.ts +342 -0
- package/src/cache/cache-scope.ts +167 -309
- package/src/cache/cf/cf-cache-store.ts +571 -17
- package/src/cache/cf/index.ts +13 -3
- package/src/cache/document-cache.ts +116 -77
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/handle-snapshot.ts +41 -0
- package/src/cache/index.ts +1 -15
- package/src/cache/memory-segment-store.ts +191 -13
- package/src/cache/profile-registry.ts +73 -0
- package/src/cache/read-through-swr.ts +134 -0
- package/src/cache/segment-codec.ts +256 -0
- package/src/cache/taint.ts +153 -0
- package/src/cache/types.ts +72 -122
- package/src/client.rsc.tsx +3 -1
- package/src/client.tsx +135 -301
- package/src/component-utils.ts +4 -4
- package/src/components/DefaultDocument.tsx +5 -1
- package/src/context-var.ts +156 -0
- package/src/debug.ts +19 -9
- package/src/errors.ts +108 -2
- package/src/handle.ts +55 -29
- package/src/handles/MetaTags.tsx +73 -20
- package/src/handles/breadcrumbs.ts +66 -0
- package/src/handles/index.ts +1 -0
- package/src/handles/meta.ts +30 -13
- package/src/host/cookie-handler.ts +21 -15
- package/src/host/errors.ts +8 -8
- package/src/host/index.ts +4 -7
- package/src/host/pattern-matcher.ts +27 -27
- package/src/host/router.ts +61 -39
- package/src/host/testing.ts +8 -8
- package/src/host/types.ts +15 -7
- package/src/host/utils.ts +1 -1
- package/src/href-client.ts +119 -29
- package/src/index.rsc.ts +155 -19
- package/src/index.ts +251 -30
- package/src/internal-debug.ts +11 -0
- package/src/loader.rsc.ts +26 -157
- package/src/loader.ts +27 -10
- package/src/network-error-thrower.tsx +3 -1
- package/src/outlet-provider.tsx +45 -0
- package/src/prerender/param-hash.ts +37 -0
- package/src/prerender/store.ts +186 -0
- package/src/prerender.ts +524 -0
- package/src/reverse.ts +354 -0
- package/src/root-error-boundary.tsx +41 -29
- package/src/route-content-wrapper.tsx +7 -4
- package/src/route-definition/dsl-helpers.ts +1121 -0
- package/src/route-definition/helper-factories.ts +200 -0
- package/src/route-definition/helpers-types.ts +478 -0
- package/src/route-definition/index.ts +55 -0
- package/src/route-definition/redirect.ts +101 -0
- package/src/route-definition/resolve-handler-use.ts +149 -0
- package/src/route-definition.ts +1 -1428
- package/src/route-map-builder.ts +217 -123
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +77 -8
- package/src/router/content-negotiation.ts +215 -0
- package/src/router/debug-manifest.ts +72 -0
- package/src/router/error-handling.ts +9 -9
- package/src/router/find-match.ts +160 -0
- package/src/router/handler-context.ts +438 -86
- package/src/router/intercept-resolution.ts +402 -0
- package/src/router/lazy-includes.ts +237 -0
- package/src/router/loader-resolution.ts +356 -128
- package/src/router/logging.ts +251 -0
- package/src/router/manifest.ts +163 -35
- package/src/router/match-api.ts +555 -0
- package/src/router/match-context.ts +5 -3
- package/src/router/match-handlers.ts +440 -0
- package/src/router/match-middleware/background-revalidation.ts +108 -93
- package/src/router/match-middleware/cache-lookup.ts +460 -10
- package/src/router/match-middleware/cache-store.ts +98 -26
- package/src/router/match-middleware/intercept-resolution.ts +57 -17
- package/src/router/match-middleware/segment-resolution.ts +80 -6
- package/src/router/match-pipelines.ts +10 -45
- package/src/router/match-result.ts +135 -35
- package/src/router/metrics.ts +240 -15
- package/src/router/middleware-cookies.ts +55 -0
- package/src/router/middleware-types.ts +220 -0
- package/src/router/middleware.ts +324 -369
- package/src/router/navigation-snapshot.ts +182 -0
- package/src/router/pattern-matching.ts +211 -43
- package/src/router/prerender-match.ts +502 -0
- package/src/router/preview-match.ts +98 -0
- package/src/router/request-classification.ts +310 -0
- package/src/router/revalidation.ts +137 -38
- package/src/router/route-snapshot.ts +245 -0
- package/src/router/router-context.ts +41 -21
- package/src/router/router-interfaces.ts +484 -0
- package/src/router/router-options.ts +618 -0
- package/src/router/router-registry.ts +24 -0
- package/src/router/segment-resolution/fresh.ts +748 -0
- package/src/router/segment-resolution/helpers.ts +268 -0
- package/src/router/segment-resolution/loader-cache.ts +199 -0
- package/src/router/segment-resolution/revalidation.ts +1379 -0
- package/src/router/segment-resolution/static-store.ts +67 -0
- package/src/router/segment-resolution.ts +21 -0
- package/src/router/segment-wrappers.ts +291 -0
- package/src/router/telemetry-otel.ts +299 -0
- package/src/router/telemetry.ts +300 -0
- package/src/router/timeout.ts +148 -0
- package/src/router/trie-matching.ts +239 -0
- package/src/router/types.ts +78 -3
- package/src/router.ts +740 -4252
- package/src/rsc/handler-context.ts +45 -0
- package/src/rsc/handler.ts +907 -797
- package/src/rsc/helpers.ts +140 -6
- package/src/rsc/index.ts +0 -20
- package/src/rsc/loader-fetch.ts +229 -0
- package/src/rsc/manifest-init.ts +90 -0
- package/src/rsc/nonce.ts +14 -0
- package/src/rsc/origin-guard.ts +141 -0
- package/src/rsc/progressive-enhancement.ts +393 -0
- package/src/rsc/response-error.ts +37 -0
- package/src/rsc/response-route-handler.ts +347 -0
- package/src/rsc/rsc-rendering.ts +246 -0
- package/src/rsc/runtime-warnings.ts +42 -0
- package/src/rsc/server-action.ts +358 -0
- package/src/rsc/ssr-setup.ts +128 -0
- package/src/rsc/types.ts +46 -11
- package/src/search-params.ts +230 -0
- package/src/segment-content-promise.ts +67 -0
- package/src/segment-loader-promise.ts +122 -0
- package/src/segment-system.tsx +134 -36
- package/src/server/context.ts +341 -61
- package/src/server/cookie-store.ts +190 -0
- package/src/server/fetchable-loader-store.ts +37 -0
- package/src/server/handle-store.ts +113 -15
- package/src/server/loader-registry.ts +24 -64
- package/src/server/request-context.ts +607 -81
- package/src/server.ts +35 -130
- package/src/ssr/index.tsx +103 -30
- package/src/static-handler.ts +126 -0
- package/src/theme/ThemeProvider.tsx +21 -15
- package/src/theme/ThemeScript.tsx +5 -5
- package/src/theme/constants.ts +5 -2
- package/src/theme/index.ts +4 -14
- package/src/theme/theme-context.ts +4 -30
- package/src/theme/theme-script.ts +21 -18
- package/src/types/boundaries.ts +158 -0
- package/src/types/cache-types.ts +198 -0
- package/src/types/error-types.ts +192 -0
- package/src/types/global-namespace.ts +100 -0
- package/src/types/handler-context.ts +791 -0
- package/src/types/index.ts +88 -0
- package/src/types/loader-types.ts +210 -0
- package/src/types/route-config.ts +170 -0
- package/src/types/route-entry.ts +120 -0
- package/src/types/segments.ts +150 -0
- package/src/types.ts +1 -1623
- package/src/urls/include-helper.ts +207 -0
- package/src/urls/index.ts +53 -0
- package/src/urls/path-helper-types.ts +372 -0
- package/src/urls/path-helper.ts +364 -0
- package/src/urls/pattern-types.ts +107 -0
- package/src/urls/response-types.ts +116 -0
- package/src/urls/type-extraction.ts +372 -0
- package/src/urls/urls-function.ts +98 -0
- package/src/urls.ts +1 -802
- package/src/use-loader.tsx +161 -81
- package/src/vite/discovery/bundle-postprocess.ts +181 -0
- package/src/vite/discovery/discover-routers.ts +348 -0
- package/src/vite/discovery/prerender-collection.ts +439 -0
- package/src/vite/discovery/route-types-writer.ts +258 -0
- package/src/vite/discovery/self-gen-tracking.ts +47 -0
- package/src/vite/discovery/state.ts +117 -0
- package/src/vite/discovery/virtual-module-codegen.ts +203 -0
- package/src/vite/index.ts +15 -1133
- package/src/vite/plugin-types.ts +103 -0
- package/src/vite/plugins/cjs-to-esm.ts +93 -0
- package/src/vite/plugins/client-ref-dedup.ts +115 -0
- package/src/vite/plugins/client-ref-hashing.ts +105 -0
- 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 +214 -0
- package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -53
- package/src/vite/plugins/expose-id-utils.ts +299 -0
- package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +209 -0
- package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
- package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
- package/src/vite/plugins/expose-ids/types.ts +45 -0
- package/src/vite/plugins/expose-internal-ids.ts +786 -0
- package/src/vite/plugins/performance-tracks.ts +88 -0
- package/src/vite/plugins/refresh-cmd.ts +127 -0
- package/src/vite/plugins/use-cache-transform.ts +323 -0
- package/src/vite/plugins/version-injector.ts +83 -0
- package/src/vite/plugins/version-plugin.ts +266 -0
- package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
- package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
- package/src/vite/rango.ts +462 -0
- package/src/vite/router-discovery.ts +977 -0
- package/src/vite/utils/ast-handler-extract.ts +517 -0
- package/src/vite/utils/banner.ts +36 -0
- package/src/vite/utils/bundle-analysis.ts +137 -0
- package/src/vite/utils/manifest-utils.ts +70 -0
- package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
- package/src/vite/utils/prerender-utils.ts +221 -0
- package/src/vite/utils/shared-utils.ts +170 -0
- package/CLAUDE.md +0 -43
- package/src/browser/lru-cache.ts +0 -69
- package/src/browser/request-controller.ts +0 -164
- package/src/cache/memory-store.ts +0 -253
- package/src/href-context.ts +0 -33
- package/src/href.ts +0 -255
- package/src/server/route-manifest-cache.ts +0 -173
- package/src/vite/expose-handle-id.ts +0 -209
- package/src/vite/expose-loader-id.ts +0 -426
- package/src/vite/expose-location-state-id.ts +0 -177
- /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
|
@@ -0,0 +1,310 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Request Classification
|
|
3
|
+
*
|
|
4
|
+
* Replaces the implicit "preview then match again" model with a clean
|
|
5
|
+
* two-stage architecture:
|
|
6
|
+
*
|
|
7
|
+
* 1. Classification — classifyRequest() produces a RequestPlan that answers
|
|
8
|
+
* all routing questions once: target route, request mode, route middleware,
|
|
9
|
+
* response-route info, negotiation state.
|
|
10
|
+
*
|
|
11
|
+
* 2. Execution — executeRequest() dispatches on the plan to the appropriate
|
|
12
|
+
* handler (response route, loader fetch, full render, partial render,
|
|
13
|
+
* action revalidation, PE render).
|
|
14
|
+
*
|
|
15
|
+
* Builds on RouteSnapshot from route-snapshot.ts.
|
|
16
|
+
*/
|
|
17
|
+
|
|
18
|
+
import { RouteNotFoundError } from "../errors.js";
|
|
19
|
+
import type { EntryData } from "../server/context.js";
|
|
20
|
+
import type { CollectedMiddleware } from "./middleware-types.js";
|
|
21
|
+
import type { RouteMatchResult } from "./pattern-matching.js";
|
|
22
|
+
import { negotiateRoute } from "./content-negotiation.js";
|
|
23
|
+
import { stripInternalParams } from "./handler-context.js";
|
|
24
|
+
import { resolveRoute, type RouteSnapshot } from "./route-snapshot.js";
|
|
25
|
+
|
|
26
|
+
// ---------------------------------------------------------------------------
|
|
27
|
+
// RequestPlan — discriminated union
|
|
28
|
+
// ---------------------------------------------------------------------------
|
|
29
|
+
|
|
30
|
+
interface RedirectPlan<TEnv = any> {
|
|
31
|
+
mode: "redirect";
|
|
32
|
+
route: RouteSnapshot<TEnv>;
|
|
33
|
+
redirectUrl: string;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
interface VersionMismatchPlan<TEnv = any> {
|
|
37
|
+
mode: "version-mismatch";
|
|
38
|
+
/** May be undefined when version mismatch is detected before route resolution */
|
|
39
|
+
route?: RouteSnapshot<TEnv>;
|
|
40
|
+
reloadUrl: string;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
interface ResponseRoutePlan<TEnv = any> {
|
|
44
|
+
mode: "response";
|
|
45
|
+
route: RouteSnapshot<TEnv>;
|
|
46
|
+
handler: Function;
|
|
47
|
+
responseType: string;
|
|
48
|
+
negotiated: boolean;
|
|
49
|
+
manifestEntry: EntryData;
|
|
50
|
+
routeMiddleware: CollectedMiddleware[];
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
interface LoaderFetchPlan<TEnv = any> {
|
|
54
|
+
mode: "loader";
|
|
55
|
+
route: RouteSnapshot<TEnv>;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
interface PeRenderPlan<TEnv = any> {
|
|
59
|
+
mode: "pe-render";
|
|
60
|
+
route: RouteSnapshot<TEnv>;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
interface ActionPlan<TEnv = any> {
|
|
64
|
+
mode: "action";
|
|
65
|
+
route: RouteSnapshot<TEnv>;
|
|
66
|
+
actionId: string;
|
|
67
|
+
negotiated: boolean;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
interface FullRenderPlan<TEnv = any> {
|
|
71
|
+
mode: "full-render";
|
|
72
|
+
route: RouteSnapshot<TEnv>;
|
|
73
|
+
negotiated: boolean;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
interface PartialRenderPlan<TEnv = any> {
|
|
77
|
+
mode: "partial-render";
|
|
78
|
+
route: RouteSnapshot<TEnv>;
|
|
79
|
+
negotiated: boolean;
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* The output of request classification. A discriminated union where each
|
|
84
|
+
* variant carries exactly the fields needed for its execution path.
|
|
85
|
+
*/
|
|
86
|
+
export type RequestPlan<TEnv = any> =
|
|
87
|
+
| RedirectPlan<TEnv>
|
|
88
|
+
| VersionMismatchPlan<TEnv>
|
|
89
|
+
| ResponseRoutePlan<TEnv>
|
|
90
|
+
| LoaderFetchPlan<TEnv>
|
|
91
|
+
| PeRenderPlan<TEnv>
|
|
92
|
+
| ActionPlan<TEnv>
|
|
93
|
+
| FullRenderPlan<TEnv>
|
|
94
|
+
| PartialRenderPlan<TEnv>;
|
|
95
|
+
|
|
96
|
+
/**
|
|
97
|
+
* Plans that have passed the terminal-check gate (version-mismatch handled)
|
|
98
|
+
* and are ready for execution. Always have a `route` field.
|
|
99
|
+
*/
|
|
100
|
+
export type ExecutableRequestPlan<TEnv = any> = Exclude<
|
|
101
|
+
RequestPlan<TEnv>,
|
|
102
|
+
VersionMismatchPlan<TEnv>
|
|
103
|
+
>;
|
|
104
|
+
|
|
105
|
+
/**
|
|
106
|
+
* Re-export individual plan types for consumers that need to narrow.
|
|
107
|
+
*/
|
|
108
|
+
export type {
|
|
109
|
+
RedirectPlan,
|
|
110
|
+
VersionMismatchPlan,
|
|
111
|
+
ResponseRoutePlan,
|
|
112
|
+
LoaderFetchPlan,
|
|
113
|
+
PeRenderPlan,
|
|
114
|
+
ActionPlan,
|
|
115
|
+
FullRenderPlan,
|
|
116
|
+
PartialRenderPlan,
|
|
117
|
+
};
|
|
118
|
+
|
|
119
|
+
// ---------------------------------------------------------------------------
|
|
120
|
+
// classifyRequest — the single authoritative classification step
|
|
121
|
+
// ---------------------------------------------------------------------------
|
|
122
|
+
|
|
123
|
+
export interface ClassifyRequestDeps<TEnv = any> {
|
|
124
|
+
findMatch: (pathname: string) => RouteMatchResult<TEnv> | null;
|
|
125
|
+
routerVersion: string;
|
|
126
|
+
routerId: string;
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
/**
|
|
130
|
+
* Classify an incoming request into a RequestPlan.
|
|
131
|
+
*
|
|
132
|
+
* This is the single source of truth for request mode detection. It replaces
|
|
133
|
+
* the scattered previewMatch + isAction/isLoaderFetch/isPartial checks in
|
|
134
|
+
* handler.ts.
|
|
135
|
+
*
|
|
136
|
+
* Classification order:
|
|
137
|
+
* 1. Route resolution (findMatch + loadManifest via resolveRoute lite)
|
|
138
|
+
* 2. Redirect detection
|
|
139
|
+
* 3. Version mismatch
|
|
140
|
+
* 4. Response route + content negotiation
|
|
141
|
+
* 5. Mode detection from headers/params
|
|
142
|
+
*/
|
|
143
|
+
export async function classifyRequest<TEnv = any>(
|
|
144
|
+
request: Request,
|
|
145
|
+
url: URL,
|
|
146
|
+
deps: ClassifyRequestDeps<TEnv>,
|
|
147
|
+
): Promise<RequestPlan<TEnv>> {
|
|
148
|
+
const pathname = url.pathname;
|
|
149
|
+
const isAction =
|
|
150
|
+
request.headers.has("rsc-action") || url.searchParams.has("_rsc_action");
|
|
151
|
+
|
|
152
|
+
// Version mismatch — runs BEFORE route resolution so stale clients
|
|
153
|
+
// requesting removed routes get a reload, not a 404.
|
|
154
|
+
const clientVersion = url.searchParams.get("_rsc_v");
|
|
155
|
+
if (
|
|
156
|
+
deps.routerVersion &&
|
|
157
|
+
clientVersion &&
|
|
158
|
+
clientVersion !== deps.routerVersion
|
|
159
|
+
) {
|
|
160
|
+
// Strip internal _rsc_* params so the browser reloads to a clean URL
|
|
161
|
+
let reloadUrl = stripInternalParams(url).toString();
|
|
162
|
+
if (isAction) {
|
|
163
|
+
const referer = request.headers.get("referer");
|
|
164
|
+
if (referer) {
|
|
165
|
+
try {
|
|
166
|
+
const refererUrl = new URL(referer);
|
|
167
|
+
if (refererUrl.origin === url.origin) {
|
|
168
|
+
reloadUrl = referer;
|
|
169
|
+
}
|
|
170
|
+
} catch {
|
|
171
|
+
// Malformed referer, fall back to stripped url
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
return {
|
|
177
|
+
mode: "version-mismatch",
|
|
178
|
+
reloadUrl,
|
|
179
|
+
};
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
// No metricsStore — classification is a lightweight gating step.
|
|
183
|
+
// Route-matching and manifest-loading metrics belong in the match path
|
|
184
|
+
// (createMatchContextForFull/Partial) which runs the authoritative resolution.
|
|
185
|
+
const result = await resolveRoute<TEnv>(pathname, {
|
|
186
|
+
findMatch: deps.findMatch,
|
|
187
|
+
lite: true,
|
|
188
|
+
});
|
|
189
|
+
|
|
190
|
+
if (!result) {
|
|
191
|
+
throw new RouteNotFoundError(`No route matched for ${pathname}`, {
|
|
192
|
+
cause: { pathname, method: request.method },
|
|
193
|
+
});
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
// Redirect
|
|
197
|
+
if (result.type === "redirect") {
|
|
198
|
+
const snapshot: RouteSnapshot<TEnv> = {
|
|
199
|
+
matched: result as any,
|
|
200
|
+
manifestEntry: null as any,
|
|
201
|
+
entries: [],
|
|
202
|
+
routeKey: "",
|
|
203
|
+
localRouteName: "",
|
|
204
|
+
params: {},
|
|
205
|
+
routeMiddleware: [],
|
|
206
|
+
cacheScope: null,
|
|
207
|
+
isPassthrough: false,
|
|
208
|
+
};
|
|
209
|
+
return {
|
|
210
|
+
mode: "redirect",
|
|
211
|
+
route: snapshot,
|
|
212
|
+
redirectUrl: result.redirectTo + url.search,
|
|
213
|
+
};
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
const snapshot = result.snapshot;
|
|
217
|
+
|
|
218
|
+
// Response route — non-RSC short-circuit (JSON, streaming, etc.)
|
|
219
|
+
const responseResult = await classifyResponseRoute(
|
|
220
|
+
request,
|
|
221
|
+
pathname,
|
|
222
|
+
snapshot,
|
|
223
|
+
);
|
|
224
|
+
if (responseResult) {
|
|
225
|
+
return responseResult;
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
// Mode detection from request signals
|
|
229
|
+
const actionId =
|
|
230
|
+
request.headers.get("rsc-action") || url.searchParams.get("_rsc_action");
|
|
231
|
+
const isLoaderFetch = url.searchParams.has("_rsc_loader");
|
|
232
|
+
|
|
233
|
+
const hasVariants =
|
|
234
|
+
snapshot.matched.negotiateVariants &&
|
|
235
|
+
snapshot.matched.negotiateVariants.length > 0;
|
|
236
|
+
const negotiated = !!hasVariants;
|
|
237
|
+
|
|
238
|
+
if (isAction && actionId) {
|
|
239
|
+
return { mode: "action", route: snapshot, actionId, negotiated };
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
if (isLoaderFetch) {
|
|
243
|
+
return { mode: "loader", route: snapshot };
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
// PE detection: POST with form content-type, but not a server action
|
|
247
|
+
const contentType = request.headers.get("content-type") || "";
|
|
248
|
+
const isFormSubmission =
|
|
249
|
+
contentType.includes("multipart/form-data") ||
|
|
250
|
+
contentType.includes("application/x-www-form-urlencoded");
|
|
251
|
+
if (request.method === "POST" && !isAction && isFormSubmission) {
|
|
252
|
+
return { mode: "pe-render", route: snapshot };
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
// App switch: client's routerId doesn't match this router
|
|
256
|
+
const clientRouterId = url.searchParams.get("_rsc_rid");
|
|
257
|
+
const isAppSwitch = !!(clientRouterId && clientRouterId !== deps.routerId);
|
|
258
|
+
const isPartial = url.searchParams.has("_rsc_partial") && !isAppSwitch;
|
|
259
|
+
|
|
260
|
+
if (isPartial) {
|
|
261
|
+
return { mode: "partial-render", route: snapshot, negotiated };
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
return { mode: "full-render", route: snapshot, negotiated };
|
|
265
|
+
}
|
|
266
|
+
|
|
267
|
+
// ---------------------------------------------------------------------------
|
|
268
|
+
// Content negotiation for response routes
|
|
269
|
+
// ---------------------------------------------------------------------------
|
|
270
|
+
|
|
271
|
+
/**
|
|
272
|
+
* Check if the route is a response route and perform content negotiation
|
|
273
|
+
* if negotiate variants exist. Returns a ResponseRoutePlan if the route
|
|
274
|
+
* is a response route, null otherwise (RSC route).
|
|
275
|
+
*/
|
|
276
|
+
async function classifyResponseRoute<TEnv>(
|
|
277
|
+
request: Request,
|
|
278
|
+
pathname: string,
|
|
279
|
+
snapshot: RouteSnapshot<TEnv>,
|
|
280
|
+
): Promise<ResponseRoutePlan<TEnv> | null> {
|
|
281
|
+
const { manifestEntry, responseType } = snapshot;
|
|
282
|
+
|
|
283
|
+
const negotiation = await negotiateRoute(request, pathname, snapshot);
|
|
284
|
+
if (negotiation) {
|
|
285
|
+
return {
|
|
286
|
+
mode: "response",
|
|
287
|
+
route: snapshot,
|
|
288
|
+
...negotiation,
|
|
289
|
+
};
|
|
290
|
+
}
|
|
291
|
+
|
|
292
|
+
// Non-negotiated response route (no variants, or RSC won negotiation)
|
|
293
|
+
if (responseType) {
|
|
294
|
+
const handler =
|
|
295
|
+
manifestEntry.type === "route" ? manifestEntry.handler : undefined;
|
|
296
|
+
if (handler) {
|
|
297
|
+
return {
|
|
298
|
+
mode: "response",
|
|
299
|
+
route: snapshot,
|
|
300
|
+
handler,
|
|
301
|
+
responseType,
|
|
302
|
+
negotiated: false,
|
|
303
|
+
manifestEntry,
|
|
304
|
+
routeMiddleware: snapshot.routeMiddleware,
|
|
305
|
+
};
|
|
306
|
+
}
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
return null;
|
|
310
|
+
}
|
|
@@ -6,6 +6,30 @@
|
|
|
6
6
|
|
|
7
7
|
import type { ResolvedSegment, HandlerContext } from "../types";
|
|
8
8
|
import type { ActionContext } from "./types";
|
|
9
|
+
import {
|
|
10
|
+
debugLog,
|
|
11
|
+
pushRevalidationTraceEntry,
|
|
12
|
+
isTraceActive,
|
|
13
|
+
} from "./logging.js";
|
|
14
|
+
import type { RevalidationTraceEntry } from "./logging.js";
|
|
15
|
+
import { _getRequestContext } from "../server/request-context.js";
|
|
16
|
+
import { isAutoGeneratedRouteName } from "../route-name.js";
|
|
17
|
+
|
|
18
|
+
function paramsEqual(
|
|
19
|
+
a: Record<string, string>,
|
|
20
|
+
b: Record<string, string>,
|
|
21
|
+
): boolean {
|
|
22
|
+
if (a === b) return true;
|
|
23
|
+
|
|
24
|
+
const keysA = Object.keys(a);
|
|
25
|
+
if (keysA.length !== Object.keys(b).length) return false;
|
|
26
|
+
|
|
27
|
+
for (const key of keysA) {
|
|
28
|
+
if (a[key] !== b[key]) return false;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
return true;
|
|
32
|
+
}
|
|
9
33
|
|
|
10
34
|
/**
|
|
11
35
|
* Options for revalidation evaluation
|
|
@@ -33,6 +57,8 @@ interface EvaluateRevalidationOptions<TEnv> {
|
|
|
33
57
|
actionContext?: ActionContext;
|
|
34
58
|
/** If true, this is a stale cache revalidation request */
|
|
35
59
|
stale?: boolean;
|
|
60
|
+
/** Trace source hint for the revalidation trace */
|
|
61
|
+
traceSource?: RevalidationTraceEntry["source"];
|
|
36
62
|
}
|
|
37
63
|
|
|
38
64
|
/**
|
|
@@ -40,7 +66,7 @@ interface EvaluateRevalidationOptions<TEnv> {
|
|
|
40
66
|
* Optimized to use prevParams directly and avoid building previous segments
|
|
41
67
|
*/
|
|
42
68
|
export async function evaluateRevalidation<TEnv>(
|
|
43
|
-
options: EvaluateRevalidationOptions<TEnv
|
|
69
|
+
options: EvaluateRevalidationOptions<TEnv>,
|
|
44
70
|
): Promise<boolean> {
|
|
45
71
|
const {
|
|
46
72
|
segment,
|
|
@@ -54,71 +80,118 @@ export async function evaluateRevalidation<TEnv>(
|
|
|
54
80
|
context,
|
|
55
81
|
actionContext,
|
|
56
82
|
stale,
|
|
83
|
+
traceSource,
|
|
57
84
|
} = options;
|
|
58
85
|
const nextParams = segment.params || {};
|
|
59
|
-
const paramsChanged =
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
86
|
+
const paramsChanged = !paramsEqual(nextParams, prevParams);
|
|
87
|
+
const searchChanged = prevUrl.search !== nextUrl.search;
|
|
88
|
+
|
|
89
|
+
// Trace helper: push a structured entry to the request-scoped trace buffer.
|
|
90
|
+
// Guarded by isTraceActive() so object construction is skipped in production.
|
|
91
|
+
function pushTrace(
|
|
92
|
+
defaultVal: boolean,
|
|
93
|
+
finalVal: boolean,
|
|
94
|
+
reason: string,
|
|
95
|
+
): void {
|
|
96
|
+
if (!isTraceActive()) return;
|
|
97
|
+
pushRevalidationTraceEntry({
|
|
98
|
+
segmentId: segment.id,
|
|
99
|
+
segmentType: segment.type,
|
|
100
|
+
belongsToRoute: segment.belongsToRoute ?? false,
|
|
101
|
+
source: traceSource ?? "segment-resolution",
|
|
102
|
+
defaultShouldRevalidate: defaultVal,
|
|
103
|
+
finalShouldRevalidate: finalVal,
|
|
104
|
+
reason,
|
|
105
|
+
customRevalidators: revalidations.length || undefined,
|
|
106
|
+
});
|
|
107
|
+
}
|
|
64
108
|
|
|
65
109
|
// Calculate default revalidation based on segment type and request method
|
|
66
110
|
let defaultShouldRevalidate: boolean;
|
|
111
|
+
let defaultReason: string;
|
|
67
112
|
|
|
68
113
|
if (request.method === "POST") {
|
|
69
114
|
// Actions: revalidate segments that belong to the route, skip parent chain
|
|
70
115
|
if (segment.type === "route") {
|
|
71
116
|
// Route segment always revalidates on actions
|
|
72
117
|
defaultShouldRevalidate = true;
|
|
118
|
+
defaultReason = "action:route-segment";
|
|
73
119
|
} else if (segment.type === "loader") {
|
|
74
120
|
// Loaders always revalidate on actions - they often contain action-sensitive data
|
|
75
121
|
// (e.g., cart count after add-to-cart action)
|
|
76
122
|
defaultShouldRevalidate = true;
|
|
123
|
+
defaultReason = "action:loader-segment";
|
|
77
124
|
} else if (segment.belongsToRoute) {
|
|
78
125
|
// Segment belongs to route (orphan layouts/parallels) - revalidate
|
|
79
126
|
defaultShouldRevalidate = true;
|
|
127
|
+
defaultReason = "action:belongs-to-route";
|
|
80
128
|
} else {
|
|
81
129
|
// Parent chain segment (shared layouts/parallels) - don't revalidate
|
|
82
130
|
defaultShouldRevalidate = false;
|
|
131
|
+
defaultReason = "action:parent-chain-skip";
|
|
83
132
|
}
|
|
84
133
|
} else {
|
|
85
134
|
// Navigation (GET): Conservative defaults to minimize unnecessary revalidations
|
|
86
135
|
// Only the route segment revalidates by default - all others require explicit opt-in
|
|
87
136
|
|
|
88
137
|
if (segment.type === "route") {
|
|
89
|
-
// Route segments revalidate when params change
|
|
90
|
-
//
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
138
|
+
// Route segments revalidate when path params OR search params change.
|
|
139
|
+
// Search params (e.g., ?page=2&sort=price) are server-parsed via ctx.search,
|
|
140
|
+
// so the handler must re-execute to produce updated content.
|
|
141
|
+
const routeChanged = paramsChanged || searchChanged;
|
|
142
|
+
defaultShouldRevalidate = routeChanged;
|
|
143
|
+
defaultReason = paramsChanged
|
|
144
|
+
? "nav:params-changed"
|
|
145
|
+
: searchChanged
|
|
146
|
+
? "nav:search-changed"
|
|
147
|
+
: "nav:params-unchanged";
|
|
148
|
+
if (routeChanged) {
|
|
149
|
+
debugLog("revalidation", "route revalidating", {
|
|
150
|
+
segmentId: segment.id,
|
|
151
|
+
paramsChanged,
|
|
152
|
+
searchChanged,
|
|
153
|
+
});
|
|
96
154
|
}
|
|
155
|
+
} else if (segment.belongsToRoute && (paramsChanged || searchChanged)) {
|
|
156
|
+
// Children of the route path (loaders, orphan layouts/parallels)
|
|
157
|
+
// revalidate when path params or search params change
|
|
158
|
+
defaultShouldRevalidate = true;
|
|
159
|
+
defaultReason = paramsChanged
|
|
160
|
+
? "nav:route-child-params-changed"
|
|
161
|
+
: "nav:route-child-search-changed";
|
|
162
|
+
debugLog("revalidation", "route child revalidating", {
|
|
163
|
+
segmentId: segment.id,
|
|
164
|
+
segmentType: segment.type,
|
|
165
|
+
paramsChanged,
|
|
166
|
+
searchChanged,
|
|
167
|
+
});
|
|
97
168
|
} else {
|
|
98
|
-
//
|
|
169
|
+
// Parent layouts and parallels default to no revalidation
|
|
99
170
|
// Cannot assume these segments depend on params without explicit declaration
|
|
100
171
|
// Use custom revalidation functions to opt-in when needed
|
|
101
172
|
defaultShouldRevalidate = false;
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
);
|
|
173
|
+
defaultReason = "nav:non-route-skip";
|
|
174
|
+
debugLog("revalidation", "non-route segment skipped by default", {
|
|
175
|
+
segmentId: segment.id,
|
|
176
|
+
segmentType: segment.type,
|
|
177
|
+
});
|
|
107
178
|
}
|
|
108
179
|
}
|
|
109
180
|
|
|
110
181
|
// No custom revalidations defined - return default behavior without prev segment
|
|
111
182
|
if (revalidations.length === 0) {
|
|
112
183
|
if (defaultShouldRevalidate) {
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
184
|
+
debugLog("revalidation", "default revalidate=true", {
|
|
185
|
+
segmentId: segment.id,
|
|
186
|
+
prevParams,
|
|
187
|
+
nextParams,
|
|
188
|
+
});
|
|
117
189
|
} else {
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
);
|
|
190
|
+
debugLog("revalidation", "default revalidate=false", {
|
|
191
|
+
segmentId: segment.id,
|
|
192
|
+
});
|
|
121
193
|
}
|
|
194
|
+
pushTrace(defaultShouldRevalidate, defaultShouldRevalidate, defaultReason);
|
|
122
195
|
return defaultShouldRevalidate;
|
|
123
196
|
}
|
|
124
197
|
|
|
@@ -129,6 +202,16 @@ export async function evaluateRevalidation<TEnv>(
|
|
|
129
202
|
// Execute revalidation functions with soft/hard decision pattern
|
|
130
203
|
let currentSuggestion = defaultShouldRevalidate;
|
|
131
204
|
|
|
205
|
+
// Compute public route names (filtered: undefined for auto-generated routes)
|
|
206
|
+
const toRouteName =
|
|
207
|
+
routeKey && !isAutoGeneratedRouteName(routeKey) ? routeKey : undefined;
|
|
208
|
+
const reqCtx = _getRequestContext();
|
|
209
|
+
const prevRouteKey = reqCtx?._prevRouteKey;
|
|
210
|
+
const fromRouteName =
|
|
211
|
+
prevRouteKey && !isAutoGeneratedRouteName(prevRouteKey)
|
|
212
|
+
? prevRouteKey
|
|
213
|
+
: undefined;
|
|
214
|
+
|
|
132
215
|
for (const { name, fn } of revalidations) {
|
|
133
216
|
const result = fn({
|
|
134
217
|
currentParams: prevSegment?.params || prevParams, // Use segment params if available, else route params
|
|
@@ -147,7 +230,9 @@ export async function evaluateRevalidation<TEnv>(
|
|
|
147
230
|
actionResult: actionContext?.actionResult,
|
|
148
231
|
formData: actionContext?.formData,
|
|
149
232
|
method: request.method, // GET for navigation, POST for actions
|
|
150
|
-
routeName:
|
|
233
|
+
routeName: toRouteName, // Navigation target route name (filtered)
|
|
234
|
+
fromRouteName, // Navigation source route name (filtered)
|
|
235
|
+
toRouteName, // Navigation target route name (filtered)
|
|
151
236
|
// Stale cache context (only true for background revalidation after stale cache render)
|
|
152
237
|
stale,
|
|
153
238
|
});
|
|
@@ -158,9 +243,12 @@ export async function evaluateRevalidation<TEnv>(
|
|
|
158
243
|
// - null/undefined: use default behavior (equivalent to returning { defaultShouldRevalidate })
|
|
159
244
|
if (typeof result === "boolean") {
|
|
160
245
|
// Hard decision - short-circuit
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
246
|
+
debugLog("revalidation", "hard decision", {
|
|
247
|
+
segmentId: segment.id,
|
|
248
|
+
revalidator: name,
|
|
249
|
+
revalidate: result,
|
|
250
|
+
});
|
|
251
|
+
pushTrace(defaultShouldRevalidate, result, `hard:${name}`);
|
|
164
252
|
return result;
|
|
165
253
|
} else if (
|
|
166
254
|
result &&
|
|
@@ -169,22 +257,33 @@ export async function evaluateRevalidation<TEnv>(
|
|
|
169
257
|
) {
|
|
170
258
|
// Soft decision - update suggestion and continue
|
|
171
259
|
currentSuggestion = result.defaultShouldRevalidate;
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
260
|
+
debugLog("revalidation", "soft decision", {
|
|
261
|
+
segmentId: segment.id,
|
|
262
|
+
revalidator: name,
|
|
263
|
+
revalidate: currentSuggestion,
|
|
264
|
+
});
|
|
175
265
|
} else if (result === null || result === undefined) {
|
|
176
266
|
// Defer to default - equivalent to { defaultShouldRevalidate: currentSuggestion }
|
|
177
267
|
// This means "I don't care, use whatever the default is"
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
268
|
+
debugLog("revalidation", "deferred to current default", {
|
|
269
|
+
segmentId: segment.id,
|
|
270
|
+
revalidator: name,
|
|
271
|
+
revalidate: currentSuggestion,
|
|
272
|
+
});
|
|
181
273
|
// currentSuggestion stays the same, continue to next function
|
|
182
274
|
}
|
|
183
275
|
}
|
|
184
276
|
|
|
185
277
|
// All revalidators completed - use final suggestion
|
|
186
|
-
|
|
187
|
-
|
|
278
|
+
debugLog("revalidation", "final decision", {
|
|
279
|
+
segmentId: segment.id,
|
|
280
|
+
revalidate: currentSuggestion,
|
|
281
|
+
});
|
|
282
|
+
const softNames = revalidations.map((r) => r.name).join(",");
|
|
283
|
+
pushTrace(
|
|
284
|
+
defaultShouldRevalidate,
|
|
285
|
+
currentSuggestion,
|
|
286
|
+
`soft-chain:${softNames}`,
|
|
188
287
|
);
|
|
189
288
|
return currentSuggestion;
|
|
190
289
|
}
|