@rangojs/router 0.0.0-experimental.10 → 0.0.0-experimental.100
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 +1037 -4
- package/dist/bin/rango.js +1619 -157
- package/dist/vite/index.js +5762 -2301
- package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/package.json +71 -63
- package/skills/breadcrumbs/SKILL.md +252 -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 +6 -4
- package/skills/handler-use/SKILL.md +364 -0
- package/skills/hooks/SKILL.md +367 -71
- package/skills/host-router/SKILL.md +218 -0
- package/skills/i18n/SKILL.md +276 -0
- package/skills/intercept/SKILL.md +176 -8
- package/skills/layout/SKILL.md +124 -3
- package/skills/links/SKILL.md +304 -25
- package/skills/loader/SKILL.md +474 -47
- package/skills/middleware/SKILL.md +207 -37
- package/skills/migrate-nextjs/SKILL.md +562 -0
- package/skills/migrate-react-router/SKILL.md +769 -0
- package/skills/mime-routes/SKILL.md +15 -11
- package/skills/parallel/SKILL.md +272 -1
- package/skills/prerender/SKILL.md +467 -65
- package/skills/rango/SKILL.md +89 -21
- package/skills/response-routes/SKILL.md +152 -91
- package/skills/route/SKILL.md +305 -14
- package/skills/router-setup/SKILL.md +210 -32
- package/skills/server-actions/SKILL.md +739 -0
- package/skills/streams-and-websockets/SKILL.md +283 -0
- package/skills/theme/SKILL.md +9 -8
- package/skills/typesafety/SKILL.md +333 -86
- package/skills/use-cache/SKILL.md +324 -0
- package/skills/view-transitions/SKILL.md +212 -0
- package/src/__internal.ts +102 -4
- package/src/bin/rango.ts +312 -15
- package/src/browser/action-coordinator.ts +97 -0
- package/src/browser/action-response-classifier.ts +99 -0
- package/src/browser/app-shell.ts +52 -0
- package/src/browser/app-version.ts +14 -0
- package/src/browser/event-controller.ts +136 -68
- 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 +374 -561
- package/src/browser/navigation-client.ts +228 -70
- package/src/browser/navigation-store.ts +97 -55
- package/src/browser/navigation-transaction.ts +297 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +376 -315
- package/src/browser/prefetch/cache.ts +314 -0
- package/src/browser/prefetch/fetch.ts +282 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +48 -0
- package/src/browser/prefetch/queue.ts +191 -0
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/rango-state.ts +152 -0
- package/src/browser/react/Link.tsx +255 -71
- package/src/browser/react/NavigationProvider.tsx +152 -24
- package/src/browser/react/context.ts +11 -0
- package/src/browser/react/filter-segment-order.ts +55 -0
- package/src/browser/react/index.ts +15 -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 -120
- 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 +78 -0
- package/src/browser/react/use-pathname.ts +47 -0
- package/src/browser/react/use-reverse.ts +99 -0
- package/src/browser/react/use-router.ts +83 -0
- package/src/browser/react/use-search-params.ts +56 -0
- package/src/browser/react/use-segments.ts +85 -99
- package/src/browser/response-adapter.ts +73 -0
- package/src/browser/rsc-router.tsx +246 -64
- 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 +158 -48
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +84 -23
- package/src/build/generate-route-types.ts +39 -828
- package/src/build/index.ts +4 -5
- package/src/build/route-trie.ts +85 -32
- 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 -307
- package/src/cache/cf/cf-cache-store.ts +573 -21
- 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 +6 -1
- package/src/client.tsx +118 -302
- 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 +77 -7
- package/src/handle.ts +55 -10
- 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 +65 -45
- package/src/index.rsc.ts +138 -21
- package/src/index.ts +206 -51
- package/src/internal-debug.ts +11 -0
- package/src/loader.rsc.ts +25 -143
- package/src/loader.ts +27 -10
- package/src/network-error-thrower.tsx +3 -1
- package/src/outlet-context.ts +1 -1
- package/src/outlet-provider.tsx +45 -0
- package/src/prerender/param-hash.ts +4 -2
- package/src/prerender/store.ts +159 -13
- package/src/prerender.ts +397 -29
- package/src/response-utils.ts +28 -0
- package/src/reverse.ts +231 -121
- package/src/root-error-boundary.tsx +41 -29
- package/src/route-content-wrapper.tsx +7 -4
- package/src/route-definition/dsl-helpers.ts +1134 -0
- package/src/route-definition/helper-factories.ts +200 -0
- package/src/route-definition/helpers-types.ts +483 -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 +155 -0
- package/src/route-definition.ts +1 -1431
- package/src/route-map-builder.ts +162 -123
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +66 -9
- 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 +418 -86
- package/src/router/intercept-resolution.ts +35 -20
- package/src/router/lazy-includes.ts +237 -0
- package/src/router/loader-resolution.ts +359 -128
- package/src/router/logging.ts +251 -0
- package/src/router/manifest.ts +98 -32
- package/src/router/match-api.ts +196 -261
- package/src/router/match-context.ts +4 -2
- package/src/router/match-handlers.ts +441 -0
- package/src/router/match-middleware/background-revalidation.ts +108 -93
- package/src/router/match-middleware/cache-lookup.ts +415 -86
- package/src/router/match-middleware/cache-store.ts +91 -29
- package/src/router/match-middleware/intercept-resolution.ts +48 -21
- package/src/router/match-middleware/segment-resolution.ts +73 -9
- package/src/router/match-pipelines.ts +10 -45
- package/src/router/match-result.ts +154 -35
- package/src/router/metrics.ts +240 -15
- package/src/router/middleware-cookies.ts +55 -0
- package/src/router/middleware-types.ts +209 -0
- package/src/router/middleware.ts +373 -371
- package/src/router/navigation-snapshot.ts +182 -0
- package/src/router/pattern-matching.ts +292 -52
- 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 +152 -39
- 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 +756 -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 +1407 -0
- package/src/router/segment-resolution/static-store.ts +67 -0
- package/src/router/segment-resolution.ts +21 -1315
- package/src/router/segment-wrappers.ts +291 -0
- package/src/router/substitute-pattern-params.ts +56 -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 +111 -39
- package/src/router/types.ts +17 -9
- package/src/router/url-params.ts +49 -0
- package/src/router.ts +642 -2011
- package/src/rsc/handler-context.ts +45 -0
- package/src/rsc/handler.ts +864 -1114
- package/src/rsc/helpers.ts +181 -19
- 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 +395 -0
- package/src/rsc/response-error.ts +37 -0
- package/src/rsc/response-route-handler.ts +360 -0
- package/src/rsc/rsc-rendering.ts +256 -0
- package/src/rsc/runtime-warnings.ts +42 -0
- package/src/rsc/server-action.ts +360 -0
- package/src/rsc/ssr-setup.ts +128 -0
- package/src/rsc/types.ts +52 -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 +187 -38
- package/src/server/context.ts +333 -59
- 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 +603 -109
- package/src/server.ts +35 -155
- package/src/ssr/index.tsx +107 -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 +764 -0
- package/src/types/index.ts +88 -0
- package/src/types/loader-types.ts +209 -0
- package/src/types/request-scope.ts +126 -0
- package/src/types/route-config.ts +170 -0
- package/src/types/route-entry.ts +120 -0
- package/src/types/segments.ts +167 -0
- package/src/types.ts +1 -1757
- 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 +108 -0
- package/src/urls/type-extraction.ts +372 -0
- package/src/urls/urls-function.ts +98 -0
- package/src/urls.ts +1 -1282
- package/src/use-loader.tsx +161 -81
- package/src/vite/debug.ts +184 -0
- package/src/vite/discovery/bundle-postprocess.ts +181 -0
- package/src/vite/discovery/discover-routers.ts +376 -0
- package/src/vite/discovery/gate-state.ts +171 -0
- package/src/vite/discovery/prerender-collection.ts +486 -0
- package/src/vite/discovery/route-types-writer.ts +258 -0
- package/src/vite/discovery/self-gen-tracking.ts +73 -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 -2063
- package/src/vite/plugin-types.ts +103 -0
- package/src/vite/plugins/cjs-to-esm.ts +98 -0
- package/src/vite/plugins/client-ref-dedup.ts +131 -0
- package/src/vite/plugins/client-ref-hashing.ts +117 -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} +107 -64
- 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 +127 -0
- package/src/vite/plugins/expose-ids/types.ts +45 -0
- package/src/vite/plugins/expose-internal-ids.ts +816 -0
- package/src/vite/plugins/performance-tracks.ts +96 -0
- package/src/vite/plugins/refresh-cmd.ts +127 -0
- package/src/vite/plugins/use-cache-transform.ts +336 -0
- package/src/vite/plugins/version-injector.ts +109 -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 +497 -0
- package/src/vite/router-discovery.ts +1423 -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/utils/package-resolution.ts +161 -0
- package/src/vite/utils/prerender-utils.ts +222 -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/router.gen.ts +0 -6
- package/src/urls.gen.ts +0 -8
- 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/expose-prerender-handler-id.ts +0 -429
- package/src/vite/package-resolution.ts +0 -125
- /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
// Global namespace (must be imported for side effects: `declare global`)
|
|
2
|
+
export type {
|
|
3
|
+
GetRegisteredRoutes,
|
|
4
|
+
DefaultHandlerRouteMap,
|
|
5
|
+
DefaultReverseRouteMap,
|
|
6
|
+
DefaultEnv,
|
|
7
|
+
} from "./global-namespace.js";
|
|
8
|
+
// Ensure the global namespace declaration is evaluated
|
|
9
|
+
import "./global-namespace.js";
|
|
10
|
+
|
|
11
|
+
// Route configuration
|
|
12
|
+
export type {
|
|
13
|
+
DocumentProps,
|
|
14
|
+
ExtractParams,
|
|
15
|
+
TrailingSlashMode,
|
|
16
|
+
RouteConfig,
|
|
17
|
+
RouteDefinitionOptions,
|
|
18
|
+
RouteDefinition,
|
|
19
|
+
ResolvedRouteMap,
|
|
20
|
+
} from "./route-config.js";
|
|
21
|
+
|
|
22
|
+
// Boundaries (error/notFound)
|
|
23
|
+
export type {
|
|
24
|
+
ErrorInfo,
|
|
25
|
+
ErrorBoundaryFallbackProps,
|
|
26
|
+
ErrorBoundaryHandler,
|
|
27
|
+
ClientErrorBoundaryFallbackProps,
|
|
28
|
+
LoaderDataResult,
|
|
29
|
+
NotFoundInfo,
|
|
30
|
+
NotFoundBoundaryFallbackProps,
|
|
31
|
+
NotFoundBoundaryHandler,
|
|
32
|
+
} from "./boundaries.js";
|
|
33
|
+
export { isLoaderDataResult } from "./boundaries.js";
|
|
34
|
+
|
|
35
|
+
// Handler context and related types
|
|
36
|
+
export type {
|
|
37
|
+
MiddlewareFn,
|
|
38
|
+
ScopedRouteMap,
|
|
39
|
+
Handler,
|
|
40
|
+
HandlerContext,
|
|
41
|
+
InternalHandlerContext,
|
|
42
|
+
GenericParams,
|
|
43
|
+
RevalidateParams,
|
|
44
|
+
ShouldRevalidateFn,
|
|
45
|
+
RouteKeys,
|
|
46
|
+
ExtractRouteParams,
|
|
47
|
+
HandlersForRouteMap,
|
|
48
|
+
Revalidate,
|
|
49
|
+
Middleware,
|
|
50
|
+
} from "./handler-context.js";
|
|
51
|
+
|
|
52
|
+
// Segments
|
|
53
|
+
export type {
|
|
54
|
+
ViewTransitionClass,
|
|
55
|
+
TransitionConfig,
|
|
56
|
+
ResolvedSegment,
|
|
57
|
+
SegmentMetadata,
|
|
58
|
+
SlotState,
|
|
59
|
+
RootLayoutProps,
|
|
60
|
+
MatchResult,
|
|
61
|
+
} from "./segments.js";
|
|
62
|
+
|
|
63
|
+
// Route entries
|
|
64
|
+
export type { LazyIncludeContext, RouteEntry } from "./route-entry.js";
|
|
65
|
+
|
|
66
|
+
// Loader types
|
|
67
|
+
export type {
|
|
68
|
+
LoaderContext,
|
|
69
|
+
LoaderFn,
|
|
70
|
+
FetchableLoaderOptions,
|
|
71
|
+
LoadOptions,
|
|
72
|
+
LoaderDefinition,
|
|
73
|
+
} from "./loader-types.js";
|
|
74
|
+
|
|
75
|
+
// Cache types
|
|
76
|
+
export type {
|
|
77
|
+
CacheContext,
|
|
78
|
+
CacheOptions,
|
|
79
|
+
PartialCacheOptions,
|
|
80
|
+
EntryCacheConfig,
|
|
81
|
+
} from "./cache-types.js";
|
|
82
|
+
|
|
83
|
+
// Error handling types
|
|
84
|
+
export type {
|
|
85
|
+
ErrorPhase,
|
|
86
|
+
OnErrorContext,
|
|
87
|
+
OnErrorCallback,
|
|
88
|
+
} from "./error-types.js";
|
|
@@ -0,0 +1,209 @@
|
|
|
1
|
+
import type { ContextVar } from "../context-var.js";
|
|
2
|
+
import type { Handle } from "../handle.js";
|
|
3
|
+
import type { MiddlewareFn } from "../router/middleware.js";
|
|
4
|
+
import type { ScopedReverseFunction } from "../reverse.js";
|
|
5
|
+
import type { SearchSchema, ResolveSearchSchema } from "../search-params.js";
|
|
6
|
+
import type { UseItems, LoaderUseItem } from "../route-types.js";
|
|
7
|
+
import type {
|
|
8
|
+
DefaultEnv,
|
|
9
|
+
DefaultReverseRouteMap,
|
|
10
|
+
DefaultVars,
|
|
11
|
+
} from "./global-namespace.js";
|
|
12
|
+
import type { RequestScope } from "./request-scope.js";
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Context passed to loader functions during execution
|
|
16
|
+
*
|
|
17
|
+
* Loaders run after middleware but before handlers, so they have access
|
|
18
|
+
* to middleware-set variables via get().
|
|
19
|
+
*
|
|
20
|
+
* @template TParams - Route params type (e.g., { slug: string })
|
|
21
|
+
* @template TEnv - Environment type for bindings/variables
|
|
22
|
+
*
|
|
23
|
+
* @example
|
|
24
|
+
* ```typescript
|
|
25
|
+
* const CartLoader = createLoader(async (ctx) => {
|
|
26
|
+
* "use server";
|
|
27
|
+
* const user = ctx.get("user"); // From auth middleware
|
|
28
|
+
* return await db.cart.get(user.id);
|
|
29
|
+
* });
|
|
30
|
+
*
|
|
31
|
+
* // With typed params:
|
|
32
|
+
* const ProductLoader = createLoader<Product, { slug: string }>(async (ctx) => {
|
|
33
|
+
* "use server";
|
|
34
|
+
* const { slug } = ctx.params; // slug is typed as string
|
|
35
|
+
* return await db.products.findBySlug(slug);
|
|
36
|
+
* });
|
|
37
|
+
* ```
|
|
38
|
+
*/
|
|
39
|
+
export type LoaderContext<
|
|
40
|
+
TParams = Record<string, string | undefined>,
|
|
41
|
+
TEnv = DefaultEnv,
|
|
42
|
+
TBody = unknown,
|
|
43
|
+
TSearch extends SearchSchema = {},
|
|
44
|
+
> = RequestScope<TEnv> & {
|
|
45
|
+
params: TParams;
|
|
46
|
+
/**
|
|
47
|
+
* Route params extracted from the URL pattern match (server-side only).
|
|
48
|
+
* Unlike `params`, these cannot be overridden by client-provided loader params.
|
|
49
|
+
* Use this when you need trusted, server-matched route params for auth or
|
|
50
|
+
* resource scoping.
|
|
51
|
+
*/
|
|
52
|
+
routeParams: Record<string, string>;
|
|
53
|
+
search: {} extends TSearch ? {} : ResolveSearchSchema<TSearch>;
|
|
54
|
+
get: {
|
|
55
|
+
<T>(contextVar: ContextVar<T>): T | undefined;
|
|
56
|
+
} & (<K extends keyof DefaultVars>(key: K) => DefaultVars[K]);
|
|
57
|
+
/**
|
|
58
|
+
* Access another loader's data, or read handle data after rendered().
|
|
59
|
+
*
|
|
60
|
+
* For loaders: returns a promise (loaders run in parallel).
|
|
61
|
+
* For handles: returns collected data (only after `await ctx.rendered()`).
|
|
62
|
+
*/
|
|
63
|
+
use: {
|
|
64
|
+
<T, TLoaderParams = any>(
|
|
65
|
+
loader: LoaderDefinition<T, TLoaderParams>,
|
|
66
|
+
): Promise<T>;
|
|
67
|
+
<TData, TAccumulated = TData[]>(
|
|
68
|
+
handle: Handle<TData, TAccumulated>,
|
|
69
|
+
): TAccumulated;
|
|
70
|
+
};
|
|
71
|
+
/**
|
|
72
|
+
* **Experimental.** Wait for all non-loader segments to settle.
|
|
73
|
+
*
|
|
74
|
+
* After the returned promise resolves, handle data is available via
|
|
75
|
+
* `ctx.use(handle)`. Only supported in DSL loaders on non-streaming
|
|
76
|
+
* trees (no `loading()`). Throws if called from a handler-invoked
|
|
77
|
+
* loader or when the tree uses streaming.
|
|
78
|
+
*
|
|
79
|
+
* @example
|
|
80
|
+
* ```typescript
|
|
81
|
+
* const PricesLoader = createLoader(async (ctx) => {
|
|
82
|
+
* "use server";
|
|
83
|
+
* await ctx.rendered();
|
|
84
|
+
* const products = ctx.use(Products); // reads handle data
|
|
85
|
+
* return pricing.getLive(products.map(p => p.id));
|
|
86
|
+
* });
|
|
87
|
+
* ```
|
|
88
|
+
*/
|
|
89
|
+
rendered: () => Promise<void>;
|
|
90
|
+
/**
|
|
91
|
+
* HTTP method (GET, POST, PUT, PATCH, DELETE)
|
|
92
|
+
* Available when loader is called via load({ method: "POST", ... })
|
|
93
|
+
*/
|
|
94
|
+
method: string;
|
|
95
|
+
/**
|
|
96
|
+
* Request body for POST/PUT/PATCH/DELETE requests
|
|
97
|
+
* Available when loader is called via load({ method: "POST", body: {...} })
|
|
98
|
+
*/
|
|
99
|
+
body: TBody | undefined;
|
|
100
|
+
/**
|
|
101
|
+
* Form data when loader is invoked via action (fetchable loaders)
|
|
102
|
+
* Available when loader is called via form submission
|
|
103
|
+
*/
|
|
104
|
+
formData?: FormData;
|
|
105
|
+
/**
|
|
106
|
+
* Generate URLs from route names.
|
|
107
|
+
* Same scoped reverse as route handlers — `.name` resolves within the
|
|
108
|
+
* current include() scope, `name` resolves globally.
|
|
109
|
+
*/
|
|
110
|
+
reverse: ScopedReverseFunction<
|
|
111
|
+
Record<string, string>,
|
|
112
|
+
DefaultReverseRouteMap
|
|
113
|
+
>;
|
|
114
|
+
};
|
|
115
|
+
|
|
116
|
+
/**
|
|
117
|
+
* Loader function signature
|
|
118
|
+
*
|
|
119
|
+
* @template T - The return type of the loader
|
|
120
|
+
* @template TParams - Route params type (defaults to generic Record)
|
|
121
|
+
* @template TEnv - Environment type for bindings/variables
|
|
122
|
+
*
|
|
123
|
+
* @example
|
|
124
|
+
* ```typescript
|
|
125
|
+
* const myLoader: LoaderFn<{ items: Item[] }> = async (ctx) => {
|
|
126
|
+
* "use server";
|
|
127
|
+
* return { items: await db.items.list() };
|
|
128
|
+
* };
|
|
129
|
+
*
|
|
130
|
+
* // With typed params:
|
|
131
|
+
* const productLoader: LoaderFn<Product, { slug: string }> = async (ctx) => {
|
|
132
|
+
* "use server";
|
|
133
|
+
* const { slug } = ctx.params; // typed as string
|
|
134
|
+
* return await db.products.findBySlug(slug);
|
|
135
|
+
* };
|
|
136
|
+
* ```
|
|
137
|
+
*/
|
|
138
|
+
export type LoaderFn<
|
|
139
|
+
T,
|
|
140
|
+
TParams = Record<string, string | undefined>,
|
|
141
|
+
TEnv = DefaultEnv,
|
|
142
|
+
> = (ctx: LoaderContext<TParams, TEnv>) => Promise<T> | T;
|
|
143
|
+
|
|
144
|
+
/**
|
|
145
|
+
* Options for fetchable loaders
|
|
146
|
+
*
|
|
147
|
+
* Middleware uses the same MiddlewareFn signature as route/app middleware,
|
|
148
|
+
* enabling reuse of the same middleware functions everywhere.
|
|
149
|
+
*/
|
|
150
|
+
export type FetchableLoaderOptions = {
|
|
151
|
+
fetchable?: true;
|
|
152
|
+
middleware?: MiddlewareFn[];
|
|
153
|
+
};
|
|
154
|
+
|
|
155
|
+
/**
|
|
156
|
+
* Options for load() calls - type-safe union based on method
|
|
157
|
+
*/
|
|
158
|
+
export type LoadOptions =
|
|
159
|
+
| {
|
|
160
|
+
method?: "GET";
|
|
161
|
+
params?: Record<string, string>;
|
|
162
|
+
}
|
|
163
|
+
| {
|
|
164
|
+
method: "POST" | "PUT" | "PATCH" | "DELETE";
|
|
165
|
+
params?: Record<string, string>;
|
|
166
|
+
body?: FormData | Record<string, any>;
|
|
167
|
+
};
|
|
168
|
+
|
|
169
|
+
/**
|
|
170
|
+
* Loader definition object
|
|
171
|
+
*
|
|
172
|
+
* Created via createLoader(). Contains the loader name and function.
|
|
173
|
+
* On client builds, the fn is stripped by the bundler (via "use server" directive).
|
|
174
|
+
*
|
|
175
|
+
* @template T - The return type of the loader
|
|
176
|
+
* @template TParams - Route params type (for type-safe params access)
|
|
177
|
+
*
|
|
178
|
+
* @example
|
|
179
|
+
* ```typescript
|
|
180
|
+
* // Definition (same file works on server and client)
|
|
181
|
+
* export const CartLoader = createLoader(async (ctx) => {
|
|
182
|
+
* "use server";
|
|
183
|
+
* return await db.cart.get(ctx.get("user").id);
|
|
184
|
+
* });
|
|
185
|
+
*
|
|
186
|
+
* // With typed params:
|
|
187
|
+
* export const ProductLoader = createLoader<Product, { slug: string }>(async (ctx) => {
|
|
188
|
+
* "use server";
|
|
189
|
+
* const { slug } = ctx.params; // slug is typed as string
|
|
190
|
+
* return await db.products.findBySlug(slug);
|
|
191
|
+
* });
|
|
192
|
+
*
|
|
193
|
+
* // Client usage (preferred — cache-safe, always fresh)
|
|
194
|
+
* const { data } = useLoader(CartLoader);
|
|
195
|
+
*
|
|
196
|
+
* // Server escape hatch (handler needs data directly)
|
|
197
|
+
* const cart = await ctx.use(CartLoader);
|
|
198
|
+
* ```
|
|
199
|
+
*/
|
|
200
|
+
export type LoaderDefinition<
|
|
201
|
+
T = any,
|
|
202
|
+
TParams = Record<string, string | undefined>,
|
|
203
|
+
> = {
|
|
204
|
+
__brand: "loader";
|
|
205
|
+
$$id: string; // Injected by Vite plugin (exposeInternalIds) - unique identifier
|
|
206
|
+
fn?: LoaderFn<T, TParams, any>; // Optional - server-side only, stored in registry for RSC
|
|
207
|
+
/** Composable default DSL items merged when the loader is mounted. */
|
|
208
|
+
use?: () => UseItems<LoaderUseItem>;
|
|
209
|
+
};
|
|
@@ -0,0 +1,126 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* RequestScope: the fields every user-facing context shares.
|
|
3
|
+
*
|
|
4
|
+
* A handler, middleware, loader, response handler, and the ALS-bound
|
|
5
|
+
* RequestContext are all different phases of the same request, and they
|
|
6
|
+
* all carry the same set of request-scoped capabilities: the raw Request,
|
|
7
|
+
* the parsed URL pair (`url` is cleaned of internal `_rsc*` params,
|
|
8
|
+
* `originalUrl` retains them), pathname/searchParams, platform bindings
|
|
9
|
+
* (`env`), and two escape hatches for work that outlives the response
|
|
10
|
+
* (`waitUntil`) or needs the raw Cloudflare runtime object
|
|
11
|
+
* (`executionContext`).
|
|
12
|
+
*
|
|
13
|
+
* Each public context type intersects `RequestScope<TEnv>` with its own
|
|
14
|
+
* phase-specific fields (e.g. `params`/`reverse` on HandlerContext,
|
|
15
|
+
* `headers`/`header()` on MiddlewareContext). That keeps platform surface
|
|
16
|
+
* in one place and lets the next runtime escape hatch we need land in
|
|
17
|
+
* one file instead of four.
|
|
18
|
+
*/
|
|
19
|
+
|
|
20
|
+
import type { DefaultEnv } from "./global-namespace.js";
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* Minimal subset of Cloudflare Workers' ExecutionContext that the router
|
|
24
|
+
* uses. Defined locally so the package does not depend on
|
|
25
|
+
* `@cloudflare/workers-types`. Consumers that want the full type can cast.
|
|
26
|
+
*
|
|
27
|
+
* On non-Cloudflare runtimes (Node, dev server, tests), this is undefined
|
|
28
|
+
* — portable apps should prefer `ctx.waitUntil(...)`, which degrades
|
|
29
|
+
* gracefully. `ctx.executionContext` is the escape hatch for libraries
|
|
30
|
+
* (MCP, Durable Object routing, etc.) that type their arguments as the
|
|
31
|
+
* raw ExecutionContext.
|
|
32
|
+
*/
|
|
33
|
+
export interface ExecutionContext {
|
|
34
|
+
waitUntil(promise: Promise<any>): void;
|
|
35
|
+
passThroughOnException(): void;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
/**
|
|
39
|
+
* Fallback `waitUntil` body used when no Cloudflare `ExecutionContext`
|
|
40
|
+
* is available (Node, dev, tests). Runs the work fire-and-forget and
|
|
41
|
+
* logs errors so they don't silently swallow.
|
|
42
|
+
*
|
|
43
|
+
* Exported so every `waitUntil` call site degrades identically instead
|
|
44
|
+
* of inventing its own fallback policy.
|
|
45
|
+
*/
|
|
46
|
+
export function fireAndForgetWaitUntil(fn: () => Promise<void>): void {
|
|
47
|
+
fn().catch((err) =>
|
|
48
|
+
console.error("[waitUntil] Background task failed:", err),
|
|
49
|
+
);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Fields present on every user-facing request context.
|
|
54
|
+
*
|
|
55
|
+
* @template TEnv - Platform bindings type (Cloudflare env, etc.).
|
|
56
|
+
*/
|
|
57
|
+
export interface RequestScope<TEnv = DefaultEnv> {
|
|
58
|
+
/**
|
|
59
|
+
* The original incoming Request object (transport URL intact).
|
|
60
|
+
* Use `url` / `searchParams` for application logic — those have
|
|
61
|
+
* internal `_rsc*` params stripped. `request` preserves the raw URL
|
|
62
|
+
* when you need original headers, method, or body.
|
|
63
|
+
*/
|
|
64
|
+
request: Request;
|
|
65
|
+
|
|
66
|
+
/**
|
|
67
|
+
* The request URL with internal `_rsc*` transport params stripped.
|
|
68
|
+
* Use this for routing, link generation, and display.
|
|
69
|
+
*/
|
|
70
|
+
url: URL;
|
|
71
|
+
|
|
72
|
+
/**
|
|
73
|
+
* The original request URL with all parameters intact, including
|
|
74
|
+
* internal `_rsc*` transport params. Use `url` for application logic
|
|
75
|
+
* — this is only needed for advanced cases like debugging or custom
|
|
76
|
+
* cache keying.
|
|
77
|
+
*/
|
|
78
|
+
originalUrl: URL;
|
|
79
|
+
|
|
80
|
+
/** URL pathname (same as `url.pathname`). */
|
|
81
|
+
pathname: string;
|
|
82
|
+
|
|
83
|
+
/**
|
|
84
|
+
* Query parameters from the URL (system params like `_rsc*` are
|
|
85
|
+
* filtered). Always a standard `URLSearchParams` instance.
|
|
86
|
+
*/
|
|
87
|
+
searchParams: URLSearchParams;
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* Platform bindings (DB, KV, secrets, etc.). On Cloudflare Workers
|
|
91
|
+
* these are the `env` object passed to the Worker's `fetch()` handler.
|
|
92
|
+
*/
|
|
93
|
+
env: TEnv;
|
|
94
|
+
|
|
95
|
+
/**
|
|
96
|
+
* Schedule work to run after the response is sent.
|
|
97
|
+
* On Cloudflare Workers, delegates to `executionContext.waitUntil()`.
|
|
98
|
+
* On Node / dev / tests, runs as fire-and-forget with error logging.
|
|
99
|
+
*
|
|
100
|
+
* @example
|
|
101
|
+
* ```typescript
|
|
102
|
+
* ctx.waitUntil(async () => {
|
|
103
|
+
* await cacheStore.set(key, data, ttl);
|
|
104
|
+
* });
|
|
105
|
+
* ```
|
|
106
|
+
*/
|
|
107
|
+
waitUntil(fn: () => Promise<void>): void;
|
|
108
|
+
|
|
109
|
+
/**
|
|
110
|
+
* Raw Cloudflare Workers `ExecutionContext`, when running on a
|
|
111
|
+
* Cloudflare-compatible runtime. Undefined elsewhere.
|
|
112
|
+
*
|
|
113
|
+
* Escape hatch for libraries that type their arguments as
|
|
114
|
+
* `ExecutionContext` (MCP `fetch`, `routeAgentRequest`, etc.).
|
|
115
|
+
* For the common "do work after the response" case, prefer
|
|
116
|
+
* `ctx.waitUntil(...)` — it is platform-neutral.
|
|
117
|
+
*
|
|
118
|
+
* @example
|
|
119
|
+
* ```typescript
|
|
120
|
+
* path.any("/mcp", (ctx) =>
|
|
121
|
+
* emailMcp.fetch(ctx.request, ctx.env, ctx.executionContext!),
|
|
122
|
+
* );
|
|
123
|
+
* ```
|
|
124
|
+
*/
|
|
125
|
+
executionContext?: ExecutionContext;
|
|
126
|
+
}
|
|
@@ -0,0 +1,170 @@
|
|
|
1
|
+
import type { ReactNode } from "react";
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Props for the Document component that wraps the entire application.
|
|
5
|
+
*/
|
|
6
|
+
export type DocumentProps = {
|
|
7
|
+
children: ReactNode;
|
|
8
|
+
};
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* Parse constraint values into a union type
|
|
12
|
+
* "a|b|c" -> "a" | "b" | "c"
|
|
13
|
+
*/
|
|
14
|
+
type ParseConstraint<T extends string> =
|
|
15
|
+
T extends `${infer First}|${infer Rest}` ? First | ParseConstraint<Rest> : T;
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Extract param info from a param segment
|
|
19
|
+
*
|
|
20
|
+
* Handles:
|
|
21
|
+
* - :param -> { name: "param", optional: false, type: string }
|
|
22
|
+
* - :param? -> { name: "param", optional: true, type: string }
|
|
23
|
+
* - :param(a|b) -> { name: "param", optional: false, type: "a" | "b" }
|
|
24
|
+
* - :param(a|b)? -> { name: "param", optional: true, type: "a" | "b" }
|
|
25
|
+
*/
|
|
26
|
+
type ExtractParamInfo<T extends string> =
|
|
27
|
+
// Optional + constrained (with optional suffix): :param(a|b)?suffix
|
|
28
|
+
T extends `${infer Name}(${infer Constraint})?${string}`
|
|
29
|
+
? { name: Name; optional: true; type: ParseConstraint<Constraint> }
|
|
30
|
+
: // Constrained (with optional suffix): :param(a|b)suffix
|
|
31
|
+
T extends `${infer Name}(${infer Constraint})${string}`
|
|
32
|
+
? { name: Name; optional: false; type: ParseConstraint<Constraint> }
|
|
33
|
+
: // Optional (with optional suffix): :param?suffix
|
|
34
|
+
T extends `${infer Name}?${string}`
|
|
35
|
+
? { name: Name; optional: true; type: string }
|
|
36
|
+
: // Param with dot-suffix: :param.html
|
|
37
|
+
T extends `${infer Name}.${string}`
|
|
38
|
+
? { name: Name; optional: false; type: string }
|
|
39
|
+
: // Param with dash-suffix: :param-slug
|
|
40
|
+
T extends `${infer Name}-${string}`
|
|
41
|
+
? { name: Name; optional: false; type: string }
|
|
42
|
+
: // Param with tilde-suffix: :param~v2
|
|
43
|
+
T extends `${infer Name}~${string}`
|
|
44
|
+
? { name: Name; optional: false; type: string }
|
|
45
|
+
: // Required: :param (no suffix)
|
|
46
|
+
{ name: T; optional: false; type: string };
|
|
47
|
+
|
|
48
|
+
/**
|
|
49
|
+
* Build param object from info
|
|
50
|
+
*/
|
|
51
|
+
type ParamFromInfo<Info> = Info extends {
|
|
52
|
+
name: infer N extends string;
|
|
53
|
+
optional: true;
|
|
54
|
+
type: infer V;
|
|
55
|
+
}
|
|
56
|
+
? { [K in N]?: V }
|
|
57
|
+
: Info extends {
|
|
58
|
+
name: infer N extends string;
|
|
59
|
+
optional: false;
|
|
60
|
+
type: infer V;
|
|
61
|
+
}
|
|
62
|
+
? { [K in N]: V }
|
|
63
|
+
: never;
|
|
64
|
+
|
|
65
|
+
/**
|
|
66
|
+
* Merge two param objects preserving optionality
|
|
67
|
+
* Uses Pick to preserve the modifiers from source types
|
|
68
|
+
*/
|
|
69
|
+
type MergeParams<A, B> = Pick<A, keyof A> & Pick<B, keyof B> extends infer O
|
|
70
|
+
? { [K in keyof O]: O[K] }
|
|
71
|
+
: never;
|
|
72
|
+
|
|
73
|
+
/**
|
|
74
|
+
* Extract route params from a pattern with depth limit to prevent infinite recursion
|
|
75
|
+
*
|
|
76
|
+
* Supports:
|
|
77
|
+
* - Required params: /:slug -> { slug: string }
|
|
78
|
+
* - Optional params: /:locale? -> { locale?: string }
|
|
79
|
+
* - Constrained params: /:locale(en|gb) -> { locale: "en" | "gb" }
|
|
80
|
+
* - Optional + constrained: /:locale(en|gb)? -> { locale?: "en" | "gb" }
|
|
81
|
+
*
|
|
82
|
+
* @example
|
|
83
|
+
* ExtractParams<"/products/:id"> // { id: string }
|
|
84
|
+
* ExtractParams<"/:locale?/blog/:slug"> // { locale?: string; slug: string }
|
|
85
|
+
* ExtractParams<"/:locale(en|gb)/blog"> // { locale: "en" | "gb" }
|
|
86
|
+
* ExtractParams<"/:locale(en|gb)?/blog/:slug"> // { locale?: "en" | "gb"; slug: string }
|
|
87
|
+
*/
|
|
88
|
+
export type ExtractParams<
|
|
89
|
+
T extends string,
|
|
90
|
+
Depth extends readonly unknown[] = [],
|
|
91
|
+
> = Depth["length"] extends 10
|
|
92
|
+
? { [key: string]: string | undefined } // Fallback to generic params if too deep
|
|
93
|
+
: // Match param with remaining path: :param.../rest
|
|
94
|
+
T extends `${infer _Start}:${infer Param}/${infer Rest}`
|
|
95
|
+
? MergeParams<
|
|
96
|
+
ParamFromInfo<ExtractParamInfo<Param>>,
|
|
97
|
+
ExtractParams<`/${Rest}`, readonly [...Depth, unknown]>
|
|
98
|
+
>
|
|
99
|
+
: // Match param at end: :param...
|
|
100
|
+
T extends `${infer _Start}:${infer Param}`
|
|
101
|
+
? ParamFromInfo<ExtractParamInfo<Param>>
|
|
102
|
+
: {};
|
|
103
|
+
|
|
104
|
+
/**
|
|
105
|
+
* Trailing slash handling mode
|
|
106
|
+
* - "never": Redirect URLs with trailing slash to without
|
|
107
|
+
* - "always": Redirect URLs without trailing slash to with
|
|
108
|
+
* - "ignore": Match both with and without trailing slash
|
|
109
|
+
*/
|
|
110
|
+
export type TrailingSlashMode = "never" | "always" | "ignore";
|
|
111
|
+
|
|
112
|
+
/**
|
|
113
|
+
* Route configuration object (alternative to string path)
|
|
114
|
+
*/
|
|
115
|
+
export type RouteConfig = {
|
|
116
|
+
path: string;
|
|
117
|
+
trailingSlash?: TrailingSlashMode;
|
|
118
|
+
};
|
|
119
|
+
|
|
120
|
+
/**
|
|
121
|
+
* Route definition options (global defaults)
|
|
122
|
+
*/
|
|
123
|
+
export type RouteDefinitionOptions = {
|
|
124
|
+
trailingSlash?: TrailingSlashMode;
|
|
125
|
+
};
|
|
126
|
+
|
|
127
|
+
export type RouteDefinition = {
|
|
128
|
+
[key: string]: string | RouteConfig | RouteDefinition;
|
|
129
|
+
};
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Recursively flatten nested routes with depth limit to prevent infinite recursion
|
|
133
|
+
* Transforms: { products: { detail: "/product/:slug" } } => { "products.detail": "/product/:slug" }
|
|
134
|
+
* Also handles RouteConfig objects: { api: { path: "/api" } } => { "api": "/api" }
|
|
135
|
+
*/
|
|
136
|
+
type FlattenRoutes<
|
|
137
|
+
T extends RouteDefinition,
|
|
138
|
+
Prefix extends string = "",
|
|
139
|
+
Depth extends readonly unknown[] = [],
|
|
140
|
+
> = Depth["length"] extends 5
|
|
141
|
+
? never
|
|
142
|
+
: {
|
|
143
|
+
[K in keyof T]: T[K] extends string
|
|
144
|
+
? Record<`${Prefix}${K & string}`, T[K]>
|
|
145
|
+
: T[K] extends RouteConfig
|
|
146
|
+
? Record<`${Prefix}${K & string}`, T[K]["path"]>
|
|
147
|
+
: T[K] extends RouteDefinition
|
|
148
|
+
? FlattenRoutes<
|
|
149
|
+
T[K],
|
|
150
|
+
`${Prefix}${K & string}.`,
|
|
151
|
+
readonly [...Depth, unknown]
|
|
152
|
+
>
|
|
153
|
+
: never;
|
|
154
|
+
}[keyof T];
|
|
155
|
+
|
|
156
|
+
/**
|
|
157
|
+
* Union to intersection helper
|
|
158
|
+
*/
|
|
159
|
+
type UnionToIntersection<U> = (
|
|
160
|
+
U extends unknown ? (k: U) => void : never
|
|
161
|
+
) extends (k: infer I) => void
|
|
162
|
+
? I
|
|
163
|
+
: never;
|
|
164
|
+
|
|
165
|
+
/**
|
|
166
|
+
* Resolved route map - flattened route definitions with full paths
|
|
167
|
+
*/
|
|
168
|
+
export type ResolvedRouteMap<T extends RouteDefinition> = UnionToIntersection<
|
|
169
|
+
FlattenRoutes<T>
|
|
170
|
+
>;
|