@rangojs/router 0.0.0-experimental.7 → 0.0.0-experimental.71
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 +4951 -930
- package/package.json +70 -60
- 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/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 +92 -31
- package/skills/loader/SKILL.md +404 -44
- package/skills/middleware/SKILL.md +173 -34
- package/skills/mime-routes/SKILL.md +128 -0
- package/skills/parallel/SKILL.md +204 -1
- package/skills/prerender/SKILL.md +685 -0
- package/skills/rango/SKILL.md +85 -16
- package/skills/response-routes/SKILL.md +411 -0
- package/skills/route/SKILL.md +257 -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 +296 -558
- package/src/browser/navigation-client.ts +179 -69
- 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 +328 -313
- package/src/browser/prefetch/cache.ts +206 -0
- package/src/browser/prefetch/fetch.ts +150 -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 +230 -74
- package/src/browser/react/NavigationProvider.tsx +87 -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 +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 +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 +221 -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 +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 +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 +105 -179
- 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 +223 -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 +351 -0
- package/src/root-error-boundary.tsx +41 -29
- package/src/route-content-wrapper.tsx +7 -4
- package/src/route-definition/dsl-helpers.ts +982 -0
- package/src/route-definition/helper-factories.ts +200 -0
- package/src/route-definition/helpers-types.ts +434 -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 +70 -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 +435 -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 +154 -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 +459 -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 +391 -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 +356 -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-system.tsx +165 -17
- package/src/server/context.ts +315 -58
- 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 +109 -0
- package/src/types/segments.ts +151 -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 +346 -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 -1129
- 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/{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 +918 -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 +207 -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
package/src/route-map-builder.ts
CHANGED
|
@@ -1,115 +1,12 @@
|
|
|
1
1
|
/**
|
|
2
|
-
*
|
|
2
|
+
* Route manifest storage and retrieval.
|
|
3
3
|
*
|
|
4
|
-
*
|
|
5
|
-
*
|
|
4
|
+
* The route manifest maps route names to URL patterns. It is populated
|
|
5
|
+
* by the virtual module (which imports from .named-routes.gen.ts files)
|
|
6
|
+
* and consumed by reverse() and href() at runtime.
|
|
6
7
|
*
|
|
7
|
-
*
|
|
8
|
-
* ```typescript
|
|
9
|
-
* import { createRouteMap, registerRouteMap } from "rsc-router/browser";
|
|
10
|
-
*
|
|
11
|
-
* const routeMap = createRouteMap()
|
|
12
|
-
* .add(homeRoutes)
|
|
13
|
-
* .add(blogRoutes, "blog")
|
|
14
|
-
* .add(shopRoutes, "shop");
|
|
15
|
-
*
|
|
16
|
-
* registerRouteMap(routeMap.routes);
|
|
17
|
-
*
|
|
18
|
-
* declare global {
|
|
19
|
-
* namespace RSCRouter {
|
|
20
|
-
* interface RegisteredRoutes extends typeof routeMap.routes {}
|
|
21
|
-
* }
|
|
22
|
-
* }
|
|
23
|
-
* ```
|
|
24
|
-
*/
|
|
25
|
-
|
|
26
|
-
import type { PrefixRoutePatterns } from "./href.js";
|
|
27
|
-
|
|
28
|
-
/**
|
|
29
|
-
* Route map builder interface
|
|
30
|
-
*
|
|
31
|
-
* Accumulates route types through the builder chain for type-safe href.
|
|
32
|
-
*/
|
|
33
|
-
export interface RouteMapBuilder<TRoutes extends Record<string, string> = {}> {
|
|
34
|
-
/**
|
|
35
|
-
* Add routes without prefix
|
|
36
|
-
*/
|
|
37
|
-
add<T extends Record<string, string>>(routes: T): RouteMapBuilder<TRoutes & T>;
|
|
38
|
-
|
|
39
|
-
/**
|
|
40
|
-
* Add routes with prefix (only URL patterns are prefixed, keys stay unchanged)
|
|
41
|
-
* @param routes - Route definitions to add
|
|
42
|
-
* @param prefix - URL prefix WITHOUT leading slash (e.g., "blog" not "/blog")
|
|
43
|
-
*/
|
|
44
|
-
add<T extends Record<string, string>, P extends string>(
|
|
45
|
-
routes: T,
|
|
46
|
-
prefix: P
|
|
47
|
-
): RouteMapBuilder<TRoutes & PrefixRoutePatterns<T, `/${P}`>>;
|
|
48
|
-
|
|
49
|
-
/**
|
|
50
|
-
* The accumulated route map (for typeof extraction in module augmentation)
|
|
51
|
-
*/
|
|
52
|
-
readonly routes: TRoutes;
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
/**
|
|
56
|
-
* Add routes to a map with optional prefix
|
|
57
|
-
* Keys stay unchanged for composability - only URL patterns get prefixed.
|
|
58
|
-
*
|
|
59
|
-
* @param routeMap - The map to add routes to
|
|
60
|
-
* @param routes - Routes to add
|
|
61
|
-
* @param prefix - Optional prefix for URL paths WITHOUT leading slash (keys stay unchanged)
|
|
8
|
+
* See docs/manifests.md for the full data flow.
|
|
62
9
|
*/
|
|
63
|
-
function addRoutes(
|
|
64
|
-
routeMap: Record<string, string>,
|
|
65
|
-
routes: Record<string, string>,
|
|
66
|
-
prefix: string = ""
|
|
67
|
-
): void {
|
|
68
|
-
// Normalize prefix: remove leading slash if accidentally provided
|
|
69
|
-
const normalizedPrefix = prefix.startsWith("/") ? prefix.slice(1) : prefix;
|
|
70
|
-
|
|
71
|
-
for (const [key, pattern] of Object.entries(routes)) {
|
|
72
|
-
const prefixedPattern =
|
|
73
|
-
normalizedPrefix && pattern !== "/"
|
|
74
|
-
? `/${normalizedPrefix}${pattern}`
|
|
75
|
-
: normalizedPrefix && pattern === "/"
|
|
76
|
-
? `/${normalizedPrefix}`
|
|
77
|
-
: pattern;
|
|
78
|
-
// Use original key - enables reusable route modules
|
|
79
|
-
routeMap[key] = prefixedPattern;
|
|
80
|
-
}
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
/**
|
|
84
|
-
* Create a new route map builder
|
|
85
|
-
*
|
|
86
|
-
* @returns A builder for accumulating routes with type-safe prefixes
|
|
87
|
-
*
|
|
88
|
-
* @example
|
|
89
|
-
* ```typescript
|
|
90
|
-
* const routeMap = createRouteMap()
|
|
91
|
-
* .add(homeRoutes)
|
|
92
|
-
* .add(blogRoutes, "blog");
|
|
93
|
-
*
|
|
94
|
-
* // Types are accumulated through the chain
|
|
95
|
-
* type AppRoutes = typeof routeMap.routes;
|
|
96
|
-
* ```
|
|
97
|
-
*/
|
|
98
|
-
export function createRouteMap(): RouteMapBuilder<{}> {
|
|
99
|
-
const routeMap: Record<string, string> = {};
|
|
100
|
-
|
|
101
|
-
const builder: RouteMapBuilder<any> = {
|
|
102
|
-
add(routes: Record<string, string>, prefix?: string) {
|
|
103
|
-
addRoutes(routeMap, routes, prefix);
|
|
104
|
-
return builder;
|
|
105
|
-
},
|
|
106
|
-
get routes() {
|
|
107
|
-
return routeMap;
|
|
108
|
-
},
|
|
109
|
-
};
|
|
110
|
-
|
|
111
|
-
return builder;
|
|
112
|
-
}
|
|
113
10
|
|
|
114
11
|
// Singleton route map instance - populated incrementally as routes are encountered
|
|
115
12
|
let globalRouteMap: Record<string, string> = {};
|
|
@@ -118,22 +15,17 @@ let globalRouteMap: Record<string, string> = {};
|
|
|
118
15
|
// Set from runtime cache or build-time import
|
|
119
16
|
let cachedManifest: Record<string, string> | null = null;
|
|
120
17
|
|
|
18
|
+
// Pre-computed route entries from build-time prefix tree leaf nodes.
|
|
19
|
+
// Used by evaluateLazyEntry() to skip running the handler for route matching.
|
|
20
|
+
let cachedPrecomputedEntries: Array<{
|
|
21
|
+
staticPrefix: string;
|
|
22
|
+
routes: Record<string, string>;
|
|
23
|
+
}> | null = null;
|
|
24
|
+
|
|
121
25
|
/**
|
|
122
|
-
* Register the route map
|
|
123
|
-
*
|
|
124
|
-
* Call this after building your route map to make it available to href.
|
|
26
|
+
* Register routes into the global route map.
|
|
125
27
|
* Routes are merged with any existing registered routes.
|
|
126
|
-
*
|
|
127
|
-
* @param map - The route map to register
|
|
128
|
-
*
|
|
129
|
-
* @example
|
|
130
|
-
* ```typescript
|
|
131
|
-
* const routeMap = createRouteMap()
|
|
132
|
-
* .add(homeRoutes)
|
|
133
|
-
* .add(blogRoutes, "blog");
|
|
134
|
-
*
|
|
135
|
-
* registerRouteMap(routeMap.routes);
|
|
136
|
-
* ```
|
|
28
|
+
* Called by createRouter() during module evaluation.
|
|
137
29
|
*/
|
|
138
30
|
export function registerRouteMap(map: Record<string, string>): void {
|
|
139
31
|
// Always merge with existing map (don't replace)
|
|
@@ -143,11 +35,12 @@ export function registerRouteMap(map: Record<string, string>): void {
|
|
|
143
35
|
/**
|
|
144
36
|
* Get the globally registered route map
|
|
145
37
|
*
|
|
146
|
-
* Used internally by
|
|
38
|
+
* Used internally by reverse to resolve route names to URLs at runtime.
|
|
147
39
|
* Returns the cached manifest if available (complete with lazy includes),
|
|
148
40
|
* otherwise returns the runtime-accumulated route map.
|
|
149
41
|
*
|
|
150
42
|
* @returns The registered route map
|
|
43
|
+
* @internal
|
|
151
44
|
*/
|
|
152
45
|
export function getGlobalRouteMap(): Record<string, string> {
|
|
153
46
|
// Cached manifest is complete (includes lazy routes), so prefer it
|
|
@@ -185,3 +78,204 @@ export function hasCachedManifest(): boolean {
|
|
|
185
78
|
export function clearCachedManifest(): void {
|
|
186
79
|
cachedManifest = null;
|
|
187
80
|
}
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Set pre-computed route entries from build-time data.
|
|
84
|
+
*
|
|
85
|
+
* Each entry corresponds to a leaf node in the prefix tree (no nested includes).
|
|
86
|
+
* evaluateLazyEntry() checks these before running the handler, avoiding the
|
|
87
|
+
* 5-50ms cost of handler evaluation for route matching on the first request.
|
|
88
|
+
*
|
|
89
|
+
* @param entries - Array of { staticPrefix, routes } from build-time prefix tree leaves
|
|
90
|
+
*/
|
|
91
|
+
export function setPrecomputedEntries(
|
|
92
|
+
entries: Array<{
|
|
93
|
+
staticPrefix: string;
|
|
94
|
+
routes: Record<string, string>;
|
|
95
|
+
}> | null,
|
|
96
|
+
): void {
|
|
97
|
+
cachedPrecomputedEntries = entries;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
/**
|
|
101
|
+
* Get pre-computed route entries (if available)
|
|
102
|
+
*/
|
|
103
|
+
export function getPrecomputedEntries(): typeof cachedPrecomputedEntries {
|
|
104
|
+
return cachedPrecomputedEntries;
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
// Route trie for O(path_length) matching at runtime.
|
|
108
|
+
// Built at build time from the route manifest and serialized into the virtual module.
|
|
109
|
+
let cachedRouteTrie: import("./build/route-trie.js").TrieNode | null = null;
|
|
110
|
+
|
|
111
|
+
export function setRouteTrie(trie: typeof cachedRouteTrie): void {
|
|
112
|
+
cachedRouteTrie = trie;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
export function getRouteTrie(): typeof cachedRouteTrie {
|
|
116
|
+
return cachedRouteTrie;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
// Per-router isolated data: each router gets its own manifest, trie, and
|
|
120
|
+
// precomputed entries so multi-router setups (e.g. site + admin via
|
|
121
|
+
// createHostRouter()) don't see each other's routes.
|
|
122
|
+
const perRouterManifestMap: Map<string, Record<string, string>> = new Map();
|
|
123
|
+
const perRouterTrieMap: Map<string, import("./build/route-trie.js").TrieNode> =
|
|
124
|
+
new Map();
|
|
125
|
+
const perRouterPrecomputedEntriesMap: Map<
|
|
126
|
+
string,
|
|
127
|
+
Array<{ staticPrefix: string; routes: Record<string, string> }>
|
|
128
|
+
> = new Map();
|
|
129
|
+
|
|
130
|
+
/**
|
|
131
|
+
* Clear all cached route data (global and per-router).
|
|
132
|
+
* Called during HMR when route definitions change so the handler rebuilds
|
|
133
|
+
* the trie from the updated router.urlpatterns on the next request.
|
|
134
|
+
*
|
|
135
|
+
* The virtual module calls this before repopulating with fresh data,
|
|
136
|
+
* preventing stale entries from removed routes from accumulating.
|
|
137
|
+
*/
|
|
138
|
+
export function clearAllRouterData(): void {
|
|
139
|
+
globalRouteMap = {};
|
|
140
|
+
cachedManifest = null;
|
|
141
|
+
cachedPrecomputedEntries = null;
|
|
142
|
+
cachedRouteTrie = null;
|
|
143
|
+
rootScopeRoutes.clear();
|
|
144
|
+
globalSearchSchemas.clear();
|
|
145
|
+
perRouterManifestMap.clear();
|
|
146
|
+
perRouterTrieMap.clear();
|
|
147
|
+
perRouterPrecomputedEntriesMap.clear();
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
export function setRouterManifest(
|
|
151
|
+
routerId: string,
|
|
152
|
+
manifest: Record<string, string>,
|
|
153
|
+
): void {
|
|
154
|
+
perRouterManifestMap.set(routerId, manifest);
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
/** @internal */
|
|
158
|
+
export function getRouterManifest(
|
|
159
|
+
routerId: string,
|
|
160
|
+
): Record<string, string> | undefined {
|
|
161
|
+
return perRouterManifestMap.get(routerId);
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
export function setRouterTrie(
|
|
165
|
+
routerId: string,
|
|
166
|
+
trie: import("./build/route-trie.js").TrieNode,
|
|
167
|
+
): void {
|
|
168
|
+
perRouterTrieMap.set(routerId, trie);
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
export function getRouterTrie(
|
|
172
|
+
routerId: string,
|
|
173
|
+
): import("./build/route-trie.js").TrieNode | undefined {
|
|
174
|
+
return perRouterTrieMap.get(routerId);
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
export function setRouterPrecomputedEntries(
|
|
178
|
+
routerId: string,
|
|
179
|
+
entries: Array<{ staticPrefix: string; routes: Record<string, string> }>,
|
|
180
|
+
): void {
|
|
181
|
+
perRouterPrecomputedEntriesMap.set(routerId, entries);
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
export function getRouterPrecomputedEntries(
|
|
185
|
+
routerId: string,
|
|
186
|
+
): Array<{ staticPrefix: string; routes: Record<string, string> }> | undefined {
|
|
187
|
+
return perRouterPrecomputedEntriesMap.get(routerId);
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
// Lazy loader registry: per-router manifest modules are loaded on first request
|
|
191
|
+
// via import() to keep startup fast and allow Rollup to code-split per router.
|
|
192
|
+
const routerManifestLoaders: Map<string, () => Promise<any>> = new Map();
|
|
193
|
+
|
|
194
|
+
export function registerRouterManifestLoader(
|
|
195
|
+
routerId: string,
|
|
196
|
+
loader: () => Promise<any>,
|
|
197
|
+
): void {
|
|
198
|
+
routerManifestLoaders.set(routerId, loader);
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
export async function ensureRouterManifest(routerId: string): Promise<void> {
|
|
202
|
+
// Check both manifest AND trie. The virtual module's setRouterManifest()
|
|
203
|
+
// pre-sets the manifest at startup, but the per-router trie is only
|
|
204
|
+
// available from the lazy loader. Without this, the lazy loader never
|
|
205
|
+
// runs and findMatch falls back to the global merged trie — which
|
|
206
|
+
// contains routes from ALL routers and breaks multi-router setups.
|
|
207
|
+
if (perRouterManifestMap.has(routerId) && perRouterTrieMap.has(routerId))
|
|
208
|
+
return;
|
|
209
|
+
const loader = routerManifestLoaders.get(routerId);
|
|
210
|
+
if (loader) {
|
|
211
|
+
const mod = await loader();
|
|
212
|
+
if (mod.manifest) perRouterManifestMap.set(routerId, mod.manifest);
|
|
213
|
+
if (mod.trie) perRouterTrieMap.set(routerId, mod.trie);
|
|
214
|
+
if (mod.precomputedEntries)
|
|
215
|
+
perRouterPrecomputedEntriesMap.set(routerId, mod.precomputedEntries);
|
|
216
|
+
routerManifestLoaders.delete(routerId);
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
// Dev-mode manifest readiness gate.
|
|
221
|
+
// The Vite discovery plugin calls setManifestReadyPromise() before starting
|
|
222
|
+
// discovery, and resolves it when discovery completes. The handler awaits
|
|
223
|
+
// waitForManifestReady() on first request if the manifest isn't yet available.
|
|
224
|
+
let manifestReadyPromise: Promise<void> | null = null;
|
|
225
|
+
|
|
226
|
+
export function setManifestReadyPromise(promise: Promise<void>): void {
|
|
227
|
+
manifestReadyPromise = promise;
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
export function waitForManifestReady(): Promise<void> | null {
|
|
231
|
+
return manifestReadyPromise;
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
// ============================================================================
|
|
235
|
+
// Route Scope Registry
|
|
236
|
+
// ============================================================================
|
|
237
|
+
|
|
238
|
+
// Tracks whether each route is at root scope (no named include boundary above).
|
|
239
|
+
// Used by dot-local reverse resolution to decide whether bare-name fallback
|
|
240
|
+
// is allowed after scoped lookups are exhausted.
|
|
241
|
+
const rootScopeRoutes: Map<string, boolean> = new Map();
|
|
242
|
+
|
|
243
|
+
/**
|
|
244
|
+
* Register whether a route is at root scope.
|
|
245
|
+
* Called by path() during route evaluation.
|
|
246
|
+
*/
|
|
247
|
+
export function registerRouteRootScope(
|
|
248
|
+
routeName: string,
|
|
249
|
+
rootScoped: boolean,
|
|
250
|
+
): void {
|
|
251
|
+
rootScopeRoutes.set(routeName, rootScoped);
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
/**
|
|
255
|
+
* Check if a route is at root scope.
|
|
256
|
+
* Returns undefined if the route has not been registered (e.g. in unit tests).
|
|
257
|
+
*/
|
|
258
|
+
export function isRouteRootScoped(routeName: string): boolean | undefined {
|
|
259
|
+
return rootScopeRoutes.get(routeName);
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
// ============================================================================
|
|
263
|
+
// Search Schema Registry
|
|
264
|
+
// ============================================================================
|
|
265
|
+
|
|
266
|
+
import type { SearchSchema } from "./search-params.js";
|
|
267
|
+
|
|
268
|
+
// Global search schema map: route name -> search schema descriptor.
|
|
269
|
+
// Populated by path() when a search option is provided.
|
|
270
|
+
const globalSearchSchemas: Map<string, SearchSchema> = new Map();
|
|
271
|
+
|
|
272
|
+
export function registerSearchSchema(
|
|
273
|
+
routeName: string,
|
|
274
|
+
schema: SearchSchema,
|
|
275
|
+
): void {
|
|
276
|
+
globalSearchSchemas.set(routeName, schema);
|
|
277
|
+
}
|
|
278
|
+
|
|
279
|
+
export function getSearchSchema(routeName: string): SearchSchema | undefined {
|
|
280
|
+
return globalSearchSchemas.get(routeName);
|
|
281
|
+
}
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Route name utilities for filtering internal route names.
|
|
3
|
+
*
|
|
4
|
+
* Internal names stay active in the runtime manifest for matching and local
|
|
5
|
+
* reverse() resolution, but they must not leak into public APIs or generated
|
|
6
|
+
* route maps.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
export const AUTO_GENERATED_ROUTE_PREFIX = "$path_";
|
|
10
|
+
export const INTERNAL_INCLUDE_SCOPE_PREFIX = "$prefix_";
|
|
11
|
+
|
|
12
|
+
const RESERVED_PREFIXES = [
|
|
13
|
+
AUTO_GENERATED_ROUTE_PREFIX,
|
|
14
|
+
INTERNAL_INCLUDE_SCOPE_PREFIX,
|
|
15
|
+
] as const;
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Check if a route name is internal.
|
|
19
|
+
* Internal names include:
|
|
20
|
+
* - unnamed path() routes like "$path__health" or "docs.$path__health"
|
|
21
|
+
* - hidden include scopes like "$prefix_0.index" or "blog.$prefix_1.post"
|
|
22
|
+
*
|
|
23
|
+
* User-defined names containing "$" (e.g. "docs.$admin") are valid and must
|
|
24
|
+
* be preserved.
|
|
25
|
+
*/
|
|
26
|
+
export function isAutoGeneratedRouteName(name: string): boolean {
|
|
27
|
+
return name.split(".").some((segment) => {
|
|
28
|
+
return (
|
|
29
|
+
segment.startsWith(AUTO_GENERATED_ROUTE_PREFIX) ||
|
|
30
|
+
segment.startsWith(INTERNAL_INCLUDE_SCOPE_PREFIX)
|
|
31
|
+
);
|
|
32
|
+
});
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
/**
|
|
36
|
+
* Validate that a user-provided route name does not collide with
|
|
37
|
+
* reserved internal prefixes. Checks every dot-separated segment,
|
|
38
|
+
* mirroring the same rule used by isAutoGeneratedRouteName().
|
|
39
|
+
*
|
|
40
|
+
* Throws with a clear message when a reserved prefix is detected.
|
|
41
|
+
*/
|
|
42
|
+
export function validateUserRouteName(name: string): void {
|
|
43
|
+
for (const segment of name.split(".")) {
|
|
44
|
+
for (const prefix of RESERVED_PREFIXES) {
|
|
45
|
+
if (segment.startsWith(prefix)) {
|
|
46
|
+
throw new Error(
|
|
47
|
+
`Route name "${name}" contains segment "${segment}" which uses reserved internal prefix "${prefix}". ` +
|
|
48
|
+
`Choose a different name to avoid collision with auto-generated route names.`,
|
|
49
|
+
);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
}
|
package/src/route-types.ts
CHANGED
|
@@ -19,6 +19,7 @@ export declare const ErrorBoundaryBrand: unique symbol;
|
|
|
19
19
|
export declare const NotFoundBoundaryBrand: unique symbol;
|
|
20
20
|
export declare const WhenBrand: unique symbol;
|
|
21
21
|
export declare const CacheBrand: unique symbol;
|
|
22
|
+
export declare const TransitionBrand: unique symbol;
|
|
22
23
|
export declare const IncludeBrand: unique symbol;
|
|
23
24
|
export declare const UrlPatternsBrand: unique symbol;
|
|
24
25
|
|
|
@@ -34,9 +35,11 @@ export type LayoutItem = {
|
|
|
34
35
|
* Used for type inference in urls() API
|
|
35
36
|
*/
|
|
36
37
|
export type TypedLayoutItem<
|
|
37
|
-
TChildRoutes extends Record<string,
|
|
38
|
+
TChildRoutes extends Record<string, any> = Record<string, string>,
|
|
39
|
+
TChildResponses extends Record<string, unknown> = Record<string, unknown>,
|
|
38
40
|
> = LayoutItem & {
|
|
39
41
|
readonly __childRoutes?: TChildRoutes;
|
|
42
|
+
readonly __childResponses?: TChildResponses;
|
|
40
43
|
};
|
|
41
44
|
export type RouteItem = {
|
|
42
45
|
name: string;
|
|
@@ -51,10 +54,14 @@ export type RouteItem = {
|
|
|
51
54
|
*/
|
|
52
55
|
export type TypedRouteItem<
|
|
53
56
|
TName extends string = string,
|
|
54
|
-
TPattern extends string = string
|
|
57
|
+
TPattern extends string = string,
|
|
58
|
+
TData = unknown,
|
|
59
|
+
TSearch = {},
|
|
55
60
|
> = RouteItem & {
|
|
56
61
|
readonly __name?: TName;
|
|
57
62
|
readonly __pattern?: TPattern;
|
|
63
|
+
readonly __data?: TData;
|
|
64
|
+
readonly __search?: TSearch;
|
|
58
65
|
};
|
|
59
66
|
export type ParallelItem = {
|
|
60
67
|
name: string;
|
|
@@ -114,15 +121,34 @@ export type CacheItem = {
|
|
|
114
121
|
uses?: AllUseItems[];
|
|
115
122
|
[CacheBrand]: void;
|
|
116
123
|
};
|
|
124
|
+
export type TransitionItem = {
|
|
125
|
+
name: string;
|
|
126
|
+
type: "transition";
|
|
127
|
+
[TransitionBrand]: void;
|
|
128
|
+
};
|
|
129
|
+
|
|
130
|
+
/**
|
|
131
|
+
* Typed transition item that carries child routes as phantom type
|
|
132
|
+
* Used for type inference when transition() wraps child routes
|
|
133
|
+
*/
|
|
134
|
+
export type TypedTransitionItem<
|
|
135
|
+
TChildRoutes extends Record<string, any> = Record<string, string>,
|
|
136
|
+
TChildResponses extends Record<string, unknown> = Record<string, unknown>,
|
|
137
|
+
> = TransitionItem & {
|
|
138
|
+
readonly __childRoutes?: TChildRoutes;
|
|
139
|
+
readonly __childResponses?: TChildResponses;
|
|
140
|
+
};
|
|
117
141
|
|
|
118
142
|
/**
|
|
119
143
|
* Typed cache item that carries child routes as phantom type
|
|
120
144
|
* Used for type inference in urls() API
|
|
121
145
|
*/
|
|
122
146
|
export type TypedCacheItem<
|
|
123
|
-
TChildRoutes extends Record<string,
|
|
147
|
+
TChildRoutes extends Record<string, any> = Record<string, string>,
|
|
148
|
+
TChildResponses extends Record<string, unknown> = Record<string, unknown>,
|
|
124
149
|
> = CacheItem & {
|
|
125
150
|
readonly __childRoutes?: TChildRoutes;
|
|
151
|
+
readonly __childResponses?: TChildResponses;
|
|
126
152
|
};
|
|
127
153
|
|
|
128
154
|
/**
|
|
@@ -141,6 +167,15 @@ export type IncludeItem = {
|
|
|
141
167
|
urlPrefix: string;
|
|
142
168
|
namePrefix: string | undefined;
|
|
143
169
|
parent: unknown; // EntryData - avoid circular import
|
|
170
|
+
/** Counter snapshot from pattern extraction for consistent shortCode indices */
|
|
171
|
+
counters?: Record<string, number>;
|
|
172
|
+
/** Cache profiles for DSL-time cache("profileName") resolution */
|
|
173
|
+
cacheProfiles?: Record<
|
|
174
|
+
string,
|
|
175
|
+
import("./cache/profile-registry.js").CacheProfile
|
|
176
|
+
>;
|
|
177
|
+
/** Root scope flag for dot-local reverse resolution */
|
|
178
|
+
rootScoped?: boolean;
|
|
144
179
|
};
|
|
145
180
|
[IncludeBrand]: void;
|
|
146
181
|
};
|
|
@@ -150,13 +185,15 @@ export type IncludeItem = {
|
|
|
150
185
|
* Used for type inference in urls() API
|
|
151
186
|
*/
|
|
152
187
|
export type TypedIncludeItem<
|
|
153
|
-
TRoutes extends Record<string,
|
|
188
|
+
TRoutes extends Record<string, any> = Record<string, string>,
|
|
154
189
|
TNamePrefix extends string = string,
|
|
155
|
-
TUrlPrefix extends string = string
|
|
190
|
+
TUrlPrefix extends string = string,
|
|
191
|
+
TResponses extends Record<string, unknown> = Record<string, unknown>,
|
|
156
192
|
> = IncludeItem & {
|
|
157
193
|
readonly __routes?: TRoutes;
|
|
158
194
|
readonly __namePrefix?: TNamePrefix;
|
|
159
195
|
readonly __urlPrefix?: TUrlPrefix;
|
|
196
|
+
readonly __responses?: TResponses;
|
|
160
197
|
};
|
|
161
198
|
|
|
162
199
|
/**
|
|
@@ -174,6 +211,7 @@ export type AllUseItems =
|
|
|
174
211
|
| ErrorBoundaryItem
|
|
175
212
|
| NotFoundBoundaryItem
|
|
176
213
|
| CacheItem
|
|
214
|
+
| TransitionItem
|
|
177
215
|
| IncludeItem;
|
|
178
216
|
|
|
179
217
|
/** Items that can be used inside a layout callback */
|
|
@@ -188,13 +226,17 @@ export type RouteUseItem =
|
|
|
188
226
|
| LoadingItem
|
|
189
227
|
| ErrorBoundaryItem
|
|
190
228
|
| NotFoundBoundaryItem
|
|
191
|
-
| CacheItem
|
|
229
|
+
| CacheItem
|
|
230
|
+
| TransitionItem;
|
|
231
|
+
/** Items that can be used inside a response route (path.json(), etc.) */
|
|
232
|
+
export type ResponseRouteUseItem = MiddlewareItem | CacheItem;
|
|
192
233
|
export type ParallelUseItem =
|
|
193
234
|
| RevalidateItem
|
|
194
235
|
| LoaderItem
|
|
195
236
|
| LoadingItem
|
|
196
237
|
| ErrorBoundaryItem
|
|
197
|
-
| NotFoundBoundaryItem
|
|
238
|
+
| NotFoundBoundaryItem
|
|
239
|
+
| TransitionItem;
|
|
198
240
|
export type InterceptUseItem =
|
|
199
241
|
| MiddlewareItem
|
|
200
242
|
| RevalidateItem
|
|
@@ -204,5 +246,25 @@ export type InterceptUseItem =
|
|
|
204
246
|
| NotFoundBoundaryItem
|
|
205
247
|
| LayoutItem
|
|
206
248
|
| RouteItem
|
|
207
|
-
| WhenItem
|
|
249
|
+
| WhenItem
|
|
250
|
+
| TransitionItem;
|
|
208
251
|
export type LoaderUseItem = RevalidateItem | CacheItem;
|
|
252
|
+
|
|
253
|
+
/**
|
|
254
|
+
* Allow composition factories in use() callbacks.
|
|
255
|
+
* Factories return T[], which placed inside a use() callback array
|
|
256
|
+
* creates nested arrays like (T | T[])[]. These are flattened at
|
|
257
|
+
* runtime via .flat(3).
|
|
258
|
+
*/
|
|
259
|
+
export type UseItems<T> = (T | readonly T[])[];
|
|
260
|
+
|
|
261
|
+
/**
|
|
262
|
+
* Union of all items that handler.use() may return.
|
|
263
|
+
* A handler doesn't know its mount site at definition time, so the type
|
|
264
|
+
* is intentionally broad — validation happens per-mount-site at runtime.
|
|
265
|
+
*/
|
|
266
|
+
export type HandlerUseItem =
|
|
267
|
+
| RouteUseItem
|
|
268
|
+
| LayoutUseItem
|
|
269
|
+
| ParallelUseItem
|
|
270
|
+
| InterceptUseItem;
|