@rangojs/router 0.0.0-experimental.8 → 0.0.0-experimental.8a4d0430
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/AGENTS.md +5 -0
- package/README.md +884 -4
- package/dist/bin/rango.js +1601 -0
- package/dist/vite/index.js +4474 -867
- package/package.json +60 -51
- package/skills/breadcrumbs/SKILL.md +250 -0
- package/skills/cache-guide/SKILL.md +262 -0
- package/skills/caching/SKILL.md +50 -21
- 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/hooks/SKILL.md +334 -72
- package/skills/host-router/SKILL.md +218 -0
- package/skills/intercept/SKILL.md +131 -8
- package/skills/layout/SKILL.md +100 -3
- package/skills/links/SKILL.md +89 -30
- package/skills/loader/SKILL.md +388 -38
- package/skills/middleware/SKILL.md +171 -34
- package/skills/mime-routes/SKILL.md +128 -0
- package/skills/parallel/SKILL.md +78 -1
- package/skills/prerender/SKILL.md +643 -0
- package/skills/rango/SKILL.md +85 -16
- package/skills/response-routes/SKILL.md +411 -0
- package/skills/route/SKILL.md +226 -14
- package/skills/router-setup/SKILL.md +123 -30
- package/skills/tailwind/SKILL.md +129 -0
- package/skills/theme/SKILL.md +9 -8
- package/skills/typesafety/SKILL.md +318 -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/event-controller.ts +87 -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 +285 -553
- package/src/browser/navigation-client.ts +124 -71
- package/src/browser/navigation-store.ts +33 -50
- package/src/browser/navigation-transaction.ts +295 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +258 -308
- package/src/browser/prefetch/cache.ts +146 -0
- package/src/browser/prefetch/fetch.ts +135 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +42 -0
- package/src/browser/prefetch/queue.ts +88 -0
- package/src/browser/rango-state.ts +112 -0
- package/src/browser/react/Link.tsx +185 -73
- package/src/browser/react/NavigationProvider.tsx +51 -11
- package/src/browser/react/context.ts +6 -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 +32 -79
- 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 +22 -63
- package/src/browser/react/use-params.ts +65 -0
- package/src/browser/react/use-pathname.ts +47 -0
- package/src/browser/react/use-router.ts +63 -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 +107 -26
- package/src/browser/scroll-restoration.ts +92 -16
- package/src/browser/segment-reconciler.ts +216 -0
- package/src/browser/segment-structure-assert.ts +16 -0
- package/src/browser/server-action-bridge.ts +504 -599
- package/src/browser/shallow.ts +6 -1
- package/src/browser/types.ts +109 -47
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +235 -24
- package/src/build/generate-route-types.ts +36 -0
- package/src/build/index.ts +13 -0
- package/src/build/route-trie.ts +265 -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 +411 -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 +469 -0
- package/src/build/route-types/scan-filter.ts +78 -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 +338 -0
- package/src/cache/cache-scope.ts +120 -303
- package/src/cache/cf/cf-cache-store.ts +119 -7
- package/src/cache/cf/index.ts +8 -2
- package/src/cache/document-cache.ts +101 -72
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/handle-snapshot.ts +41 -0
- package/src/cache/index.ts +0 -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 +98 -0
- package/src/cache/types.ts +72 -122
- package/src/client.rsc.tsx +3 -1
- package/src/client.tsx +106 -126
- package/src/component-utils.ts +4 -4
- package/src/components/DefaultDocument.tsx +5 -1
- package/src/context-var.ts +86 -0
- package/src/debug.ts +17 -7
- package/src/errors.ts +108 -2
- package/src/handle.ts +15 -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 +153 -19
- package/src/index.ts +211 -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 +185 -0
- package/src/prerender.ts +463 -0
- package/src/reverse.ts +330 -0
- package/src/root-error-boundary.tsx +41 -29
- package/src/route-content-wrapper.tsx +7 -4
- package/src/route-definition/dsl-helpers.ts +934 -0
- package/src/route-definition/helper-factories.ts +200 -0
- package/src/route-definition/helpers-types.ts +430 -0
- package/src/route-definition/index.ts +52 -0
- package/src/route-definition/redirect.ts +93 -0
- package/src/route-definition.ts +1 -1428
- package/src/route-map-builder.ts +211 -123
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +59 -8
- package/src/router/content-negotiation.ts +116 -0
- package/src/router/debug-manifest.ts +72 -0
- package/src/router/error-handling.ts +9 -9
- package/src/router/find-match.ts +158 -0
- package/src/router/handler-context.ts +374 -81
- package/src/router/intercept-resolution.ts +395 -0
- package/src/router/lazy-includes.ts +234 -0
- package/src/router/loader-resolution.ts +215 -122
- package/src/router/logging.ts +248 -0
- package/src/router/manifest.ts +148 -35
- package/src/router/match-api.ts +620 -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 +80 -93
- package/src/router/match-middleware/cache-lookup.ts +382 -9
- package/src/router/match-middleware/cache-store.ts +51 -22
- package/src/router/match-middleware/intercept-resolution.ts +55 -17
- package/src/router/match-middleware/segment-resolution.ts +24 -6
- package/src/router/match-pipelines.ts +10 -45
- package/src/router/match-result.ts +34 -28
- package/src/router/metrics.ts +235 -15
- package/src/router/middleware-cookies.ts +55 -0
- package/src/router/middleware-types.ts +222 -0
- package/src/router/middleware.ts +324 -367
- package/src/router/pattern-matching.ts +211 -43
- package/src/router/prerender-match.ts +402 -0
- package/src/router/preview-match.ts +170 -0
- package/src/router/revalidation.ts +137 -38
- package/src/router/router-context.ts +36 -21
- package/src/router/router-interfaces.ts +452 -0
- package/src/router/router-options.ts +592 -0
- package/src/router/router-registry.ts +24 -0
- package/src/router/segment-resolution/fresh.ts +570 -0
- package/src/router/segment-resolution/helpers.ts +263 -0
- package/src/router/segment-resolution/loader-cache.ts +198 -0
- package/src/router/segment-resolution/revalidation.ts +1241 -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 +289 -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 +77 -3
- package/src/router.ts +692 -4257
- package/src/rsc/handler-context.ts +45 -0
- package/src/rsc/handler.ts +764 -754
- package/src/rsc/helpers.ts +140 -6
- package/src/rsc/index.ts +0 -20
- package/src/rsc/loader-fetch.ts +209 -0
- package/src/rsc/manifest-init.ts +86 -0
- package/src/rsc/nonce.ts +14 -0
- package/src/rsc/origin-guard.ts +141 -0
- package/src/rsc/progressive-enhancement.ts +379 -0
- package/src/rsc/response-error.ts +37 -0
- package/src/rsc/response-route-handler.ts +347 -0
- package/src/rsc/rsc-rendering.ts +235 -0
- package/src/rsc/runtime-warnings.ts +42 -0
- package/src/rsc/server-action.ts +348 -0
- package/src/rsc/ssr-setup.ts +128 -0
- package/src/rsc/types.ts +38 -11
- package/src/search-params.ts +230 -0
- package/src/segment-system.tsx +25 -13
- package/src/server/context.ts +182 -51
- package/src/server/cookie-store.ts +190 -0
- package/src/server/fetchable-loader-store.ts +37 -0
- package/src/server/handle-store.ts +94 -15
- package/src/server/loader-registry.ts +15 -56
- package/src/server/request-context.ts +430 -70
- package/src/server.ts +35 -130
- package/src/ssr/index.tsx +100 -31
- package/src/static-handler.ts +114 -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 +687 -0
- package/src/types/index.ts +88 -0
- package/src/types/loader-types.ts +183 -0
- package/src/types/route-config.ts +170 -0
- package/src/types/route-entry.ts +102 -0
- package/src/types/segments.ts +148 -0
- package/src/types.ts +1 -1623
- package/src/urls/include-helper.ts +197 -0
- package/src/urls/index.ts +53 -0
- package/src/urls/path-helper-types.ts +339 -0
- package/src/urls/path-helper.ts +329 -0
- package/src/urls/pattern-types.ts +95 -0
- package/src/urls/response-types.ts +106 -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 +85 -77
- package/src/vite/discovery/bundle-postprocess.ts +184 -0
- package/src/vite/discovery/discover-routers.ts +344 -0
- package/src/vite/discovery/prerender-collection.ts +385 -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 +110 -0
- package/src/vite/discovery/virtual-module-codegen.ts +203 -0
- package/src/vite/index.ts +11 -1133
- package/src/vite/plugin-types.ts +131 -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/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -51
- package/src/vite/plugins/expose-id-utils.ts +287 -0
- package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +179 -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 +569 -0
- package/src/vite/plugins/refresh-cmd.ts +65 -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 +254 -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 +510 -0
- package/src/vite/router-discovery.ts +785 -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 +189 -0
- package/src/vite/utils/shared-utils.ts +169 -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,72 @@
|
|
|
1
|
+
import { type EntryData, getContext } from "../server/context";
|
|
2
|
+
import { serializeManifest, type SerializedManifest } from "../debug.js";
|
|
3
|
+
import { createRouteHelpers } from "../route-definition.js";
|
|
4
|
+
import MapRootLayout from "../server/root-layout.js";
|
|
5
|
+
import type { RouteEntry, TrailingSlashMode } from "../types";
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Build a serialized manifest from all route entries for debug inspection.
|
|
9
|
+
* Used by the `debugManifest()` method on the router instance.
|
|
10
|
+
*/
|
|
11
|
+
export async function buildDebugManifest<TEnv = any>(
|
|
12
|
+
routesEntries: RouteEntry<TEnv>[],
|
|
13
|
+
): Promise<SerializedManifest> {
|
|
14
|
+
const manifest = new Map<string, EntryData>();
|
|
15
|
+
|
|
16
|
+
for (const entry of routesEntries) {
|
|
17
|
+
const Store = {
|
|
18
|
+
manifest,
|
|
19
|
+
namespace: `debug.M${entry.mountIndex}`,
|
|
20
|
+
parent: null as EntryData | null,
|
|
21
|
+
counters: {} as Record<string, number>,
|
|
22
|
+
mountIndex: entry.mountIndex,
|
|
23
|
+
patterns: new Map<string, string>(),
|
|
24
|
+
trailingSlash: new Map<string, TrailingSlashMode>(),
|
|
25
|
+
};
|
|
26
|
+
|
|
27
|
+
await getContext().runWithStore(
|
|
28
|
+
Store,
|
|
29
|
+
`debug.M${entry.mountIndex}`,
|
|
30
|
+
null,
|
|
31
|
+
async () => {
|
|
32
|
+
const helpers = createRouteHelpers();
|
|
33
|
+
|
|
34
|
+
// Wrap handler execution in root layout (same as loadManifest)
|
|
35
|
+
let promiseResult: Promise<any> | null = null;
|
|
36
|
+
helpers.layout(MapRootLayout, () => {
|
|
37
|
+
const result = entry.handler();
|
|
38
|
+
if (result instanceof Promise) {
|
|
39
|
+
promiseResult = result;
|
|
40
|
+
return [];
|
|
41
|
+
}
|
|
42
|
+
return result;
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
if (promiseResult !== null) {
|
|
46
|
+
const load = await (promiseResult as Promise<any>);
|
|
47
|
+
if (load && typeof load === "object" && "default" in load) {
|
|
48
|
+
// Promise<{ default: fn }> — e.g. dynamic import
|
|
49
|
+
if (typeof load.default !== "function") {
|
|
50
|
+
throw new Error(
|
|
51
|
+
`[@rangojs/router] Unsupported async handler: { default } must be a function, ` +
|
|
52
|
+
`got ${typeof load.default}. Use () => import('./urls') for lazy loading.`,
|
|
53
|
+
);
|
|
54
|
+
}
|
|
55
|
+
load.default(helpers);
|
|
56
|
+
} else if (typeof load === "function") {
|
|
57
|
+
// Promise<fn>
|
|
58
|
+
load(helpers);
|
|
59
|
+
} else {
|
|
60
|
+
// Reject unsupported async handler results (same policy as manifest.ts)
|
|
61
|
+
throw new Error(
|
|
62
|
+
`[@rangojs/router] Unsupported async handler result (${typeof load}). ` +
|
|
63
|
+
`Lazy route handlers must resolve to a function or { default: fn }.`,
|
|
64
|
+
);
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
},
|
|
68
|
+
);
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
return serializeManifest(manifest);
|
|
72
|
+
}
|
|
@@ -60,7 +60,7 @@ export function invokeOnError<TEnv = any>(
|
|
|
60
60
|
error: unknown,
|
|
61
61
|
phase: ErrorPhase,
|
|
62
62
|
context: InvokeOnErrorContext<TEnv>,
|
|
63
|
-
logPrefix: string = "Router"
|
|
63
|
+
logPrefix: string = "Router",
|
|
64
64
|
): void {
|
|
65
65
|
if (!onError) return;
|
|
66
66
|
|
|
@@ -74,8 +74,8 @@ export function invokeOnError<TEnv = any>(
|
|
|
74
74
|
phase,
|
|
75
75
|
request: context.request,
|
|
76
76
|
url: context.url,
|
|
77
|
-
pathname: context.url
|
|
78
|
-
method: context.request
|
|
77
|
+
pathname: context.url?.pathname,
|
|
78
|
+
method: context.request?.method,
|
|
79
79
|
routeKey: context.routeKey,
|
|
80
80
|
params: context.params,
|
|
81
81
|
segmentId: context.segmentId,
|
|
@@ -112,7 +112,7 @@ export function invokeOnError<TEnv = any>(
|
|
|
112
112
|
*/
|
|
113
113
|
export function findNearestErrorBoundary(
|
|
114
114
|
entry: EntryData | null,
|
|
115
|
-
defaultErrorBoundary?: ReactNode | ErrorBoundaryHandler
|
|
115
|
+
defaultErrorBoundary?: ReactNode | ErrorBoundaryHandler,
|
|
116
116
|
): ReactNode | ErrorBoundaryHandler | null {
|
|
117
117
|
let current: EntryData | null = entry;
|
|
118
118
|
|
|
@@ -148,7 +148,7 @@ export function findNearestErrorBoundary(
|
|
|
148
148
|
*/
|
|
149
149
|
export function findNearestNotFoundBoundary(
|
|
150
150
|
entry: EntryData | null,
|
|
151
|
-
defaultNotFoundBoundary?: ReactNode | NotFoundBoundaryHandler
|
|
151
|
+
defaultNotFoundBoundary?: ReactNode | NotFoundBoundaryHandler,
|
|
152
152
|
): ReactNode | NotFoundBoundaryHandler | null {
|
|
153
153
|
let current: EntryData | null = entry;
|
|
154
154
|
|
|
@@ -172,7 +172,7 @@ export function findNearestNotFoundBoundary(
|
|
|
172
172
|
export function createErrorInfo(
|
|
173
173
|
error: unknown,
|
|
174
174
|
segmentId: string,
|
|
175
|
-
segmentType: ErrorInfo["segmentType"]
|
|
175
|
+
segmentType: ErrorInfo["segmentType"],
|
|
176
176
|
): ErrorInfo {
|
|
177
177
|
const isDev = process.env.NODE_ENV !== "production";
|
|
178
178
|
|
|
@@ -205,7 +205,7 @@ export function createErrorSegment(
|
|
|
205
205
|
errorInfo: ErrorInfo,
|
|
206
206
|
fallback: ReactNode | ErrorBoundaryHandler,
|
|
207
207
|
entry: EntryData,
|
|
208
|
-
params: Record<string, string
|
|
208
|
+
params: Record<string, string>,
|
|
209
209
|
): ResolvedSegment {
|
|
210
210
|
// Determine the component to render
|
|
211
211
|
let component: ReactNode;
|
|
@@ -241,7 +241,7 @@ export function createNotFoundInfo(
|
|
|
241
241
|
error: { message: string },
|
|
242
242
|
segmentId: string,
|
|
243
243
|
segmentType: NotFoundInfo["segmentType"],
|
|
244
|
-
pathname?: string
|
|
244
|
+
pathname?: string,
|
|
245
245
|
): NotFoundInfo {
|
|
246
246
|
return {
|
|
247
247
|
message: error.message,
|
|
@@ -259,7 +259,7 @@ export function createNotFoundSegment(
|
|
|
259
259
|
notFoundInfo: NotFoundInfo,
|
|
260
260
|
fallback: ReactNode | NotFoundBoundaryHandler,
|
|
261
261
|
entry: EntryData,
|
|
262
|
-
params: Record<string, string
|
|
262
|
+
params: Record<string, string>,
|
|
263
263
|
): ResolvedSegment {
|
|
264
264
|
// Determine the component to render
|
|
265
265
|
let component: ReactNode;
|
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
import { tryTrieMatch } from "./trie-matching.js";
|
|
2
|
+
import { getRouteTrie, getRouterTrie } from "../route-map-builder.js";
|
|
3
|
+
import {
|
|
4
|
+
findMatch as findRouteMatch,
|
|
5
|
+
isLazyEvaluationNeeded,
|
|
6
|
+
type RouteMatchResult,
|
|
7
|
+
} from "./pattern-matching.js";
|
|
8
|
+
import type { MetricsStore } from "../server/context";
|
|
9
|
+
import type { RouteEntry } from "../types";
|
|
10
|
+
|
|
11
|
+
export interface FindMatchDeps<TEnv = any> {
|
|
12
|
+
routesEntries: RouteEntry<TEnv>[];
|
|
13
|
+
evaluateLazyEntry: (entry: RouteEntry<TEnv>) => void;
|
|
14
|
+
routerId: string;
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Create a findMatch function bound to router state.
|
|
19
|
+
* Includes single-entry cache to avoid redundant matching within the same request.
|
|
20
|
+
*/
|
|
21
|
+
export function createFindMatch<TEnv = any>(
|
|
22
|
+
deps: FindMatchDeps<TEnv>,
|
|
23
|
+
): (pathname: string, ms?: MetricsStore) => RouteMatchResult<TEnv> | null {
|
|
24
|
+
// Single-entry cache for findMatch to avoid redundant matching within the same request.
|
|
25
|
+
// previewMatch and match both call findMatch with the same pathname — this ensures
|
|
26
|
+
// the route matching work (which may check thousands of routes) only happens once.
|
|
27
|
+
let lastFindMatchPathname: string | null = null;
|
|
28
|
+
let lastFindMatchResult: RouteMatchResult<TEnv> | null = null;
|
|
29
|
+
|
|
30
|
+
// Wrapper for findMatch that uses routesEntries
|
|
31
|
+
// Handles lazy evaluation by evaluating lazy entries on first match.
|
|
32
|
+
// Phase 1: try O(path_length) trie match.
|
|
33
|
+
// Phase 2: fall back to regex iteration.
|
|
34
|
+
return function findMatch(
|
|
35
|
+
pathname: string,
|
|
36
|
+
ms?: MetricsStore,
|
|
37
|
+
): RouteMatchResult<TEnv> | null {
|
|
38
|
+
// Return cached result if same pathname (avoids double-match per request)
|
|
39
|
+
if (lastFindMatchPathname === pathname) {
|
|
40
|
+
return lastFindMatchResult;
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
// Helper to push sub-metrics
|
|
44
|
+
const pushMetric = ms
|
|
45
|
+
? (label: string, start: number) => {
|
|
46
|
+
ms.metrics.push({
|
|
47
|
+
label,
|
|
48
|
+
duration: performance.now() - start,
|
|
49
|
+
startTime: start - ms.requestStart,
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
: undefined;
|
|
53
|
+
|
|
54
|
+
// Phase 1: Try trie match (O(path_length))
|
|
55
|
+
// Prefer per-router trie (isolated) over global trie (merged).
|
|
56
|
+
const routeTrie = getRouterTrie(deps.routerId) ?? getRouteTrie();
|
|
57
|
+
if (routeTrie) {
|
|
58
|
+
const trieStart = performance.now();
|
|
59
|
+
const trieResult = tryTrieMatch(routeTrie, pathname);
|
|
60
|
+
pushMetric?.("match:trie", trieStart);
|
|
61
|
+
|
|
62
|
+
if (trieResult) {
|
|
63
|
+
// Find the RouteEntry that contains this route.
|
|
64
|
+
// Multiple entries can share the same staticPrefix (e.g., several
|
|
65
|
+
// include("/", patterns) calls all produce staticPrefix=""). Evaluate
|
|
66
|
+
// each candidate and pick the one whose routes include the matched key.
|
|
67
|
+
const entryStart = performance.now();
|
|
68
|
+
let entry: RouteEntry<TEnv> | undefined;
|
|
69
|
+
let fallbackEntry: RouteEntry<TEnv> | undefined;
|
|
70
|
+
|
|
71
|
+
for (const e of deps.routesEntries) {
|
|
72
|
+
if (e.staticPrefix !== trieResult.sp) continue;
|
|
73
|
+
if (!fallbackEntry) fallbackEntry = e;
|
|
74
|
+
deps.evaluateLazyEntry(e);
|
|
75
|
+
if (
|
|
76
|
+
e.routes &&
|
|
77
|
+
trieResult.routeKey in (e.routes as Record<string, unknown>)
|
|
78
|
+
) {
|
|
79
|
+
entry = e;
|
|
80
|
+
break;
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
// If no entry had the route in its routes map, use the first matching
|
|
85
|
+
// entry as fallback (handles main entry with inline routes not yet
|
|
86
|
+
// reflected in its routes object).
|
|
87
|
+
if (!entry) entry = fallbackEntry;
|
|
88
|
+
|
|
89
|
+
// If entry not found (nested include not yet discovered), evaluate parent
|
|
90
|
+
if (!entry) {
|
|
91
|
+
const parent = deps.routesEntries.find(
|
|
92
|
+
(e) =>
|
|
93
|
+
trieResult.sp.startsWith(e.staticPrefix) &&
|
|
94
|
+
e.staticPrefix !== trieResult.sp,
|
|
95
|
+
);
|
|
96
|
+
if (parent) {
|
|
97
|
+
const lazyStart = performance.now();
|
|
98
|
+
deps.evaluateLazyEntry(parent);
|
|
99
|
+
pushMetric?.("match:lazy-eval", lazyStart);
|
|
100
|
+
}
|
|
101
|
+
entry = deps.routesEntries.find(
|
|
102
|
+
(e) => e.staticPrefix === trieResult.sp,
|
|
103
|
+
);
|
|
104
|
+
}
|
|
105
|
+
pushMetric?.("match:entry-resolve", entryStart);
|
|
106
|
+
|
|
107
|
+
if (entry) {
|
|
108
|
+
lastFindMatchPathname = pathname;
|
|
109
|
+
lastFindMatchResult = {
|
|
110
|
+
entry,
|
|
111
|
+
routeKey: trieResult.routeKey,
|
|
112
|
+
params: trieResult.params,
|
|
113
|
+
optionalParams: new Set(trieResult.optionalParams || []),
|
|
114
|
+
redirectTo: trieResult.redirectTo,
|
|
115
|
+
ancestry: trieResult.ancestry,
|
|
116
|
+
...(trieResult.pr ? { pr: true } : {}),
|
|
117
|
+
...(trieResult.pt ? { pt: true } : {}),
|
|
118
|
+
...(trieResult.responseType
|
|
119
|
+
? { responseType: trieResult.responseType }
|
|
120
|
+
: {}),
|
|
121
|
+
...(trieResult.negotiateVariants
|
|
122
|
+
? { negotiateVariants: trieResult.negotiateVariants }
|
|
123
|
+
: {}),
|
|
124
|
+
...(trieResult.rscFirst ? { rscFirst: true } : {}),
|
|
125
|
+
};
|
|
126
|
+
return lastFindMatchResult;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
// Phase 2: Fall back to existing matching (regex iteration)
|
|
132
|
+
const regexStart = performance.now();
|
|
133
|
+
let result = findRouteMatch(pathname, deps.routesEntries);
|
|
134
|
+
|
|
135
|
+
// If we hit a lazy entry that needs evaluation, evaluate and retry.
|
|
136
|
+
// Cap iterations to prevent infinite loops from pathological nesting.
|
|
137
|
+
const MAX_LAZY_ITERATIONS = 100;
|
|
138
|
+
let iterations = 0;
|
|
139
|
+
while (isLazyEvaluationNeeded(result)) {
|
|
140
|
+
if (++iterations > MAX_LAZY_ITERATIONS) {
|
|
141
|
+
console.error(
|
|
142
|
+
`[@rangojs/router] Exceeded ${MAX_LAZY_ITERATIONS} lazy evaluation iterations ` +
|
|
143
|
+
`for pathname "${pathname}". This likely indicates circular lazy includes.`,
|
|
144
|
+
);
|
|
145
|
+
lastFindMatchPathname = pathname;
|
|
146
|
+
lastFindMatchResult = null;
|
|
147
|
+
return null;
|
|
148
|
+
}
|
|
149
|
+
deps.evaluateLazyEntry(result.lazyEntry);
|
|
150
|
+
result = findRouteMatch(pathname, deps.routesEntries);
|
|
151
|
+
}
|
|
152
|
+
pushMetric?.("match:regex-fallback", regexStart);
|
|
153
|
+
|
|
154
|
+
lastFindMatchPathname = pathname;
|
|
155
|
+
lastFindMatchResult = result;
|
|
156
|
+
return result;
|
|
157
|
+
};
|
|
158
|
+
}
|