@rangojs/router 0.0.0-experimental.b9cb8739 → 0.0.0-experimental.bd6e11bc
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/README.md +196 -43
- package/dist/bin/rango.js +277 -99
- package/dist/testing/vitest.js +48 -0
- package/dist/vite/index.js +2779 -1064
- package/dist/vite/index.js.bak +5448 -0
- package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/package.json +57 -11
- package/skills/breadcrumbs/SKILL.md +3 -1
- package/skills/bundle-analysis/SKILL.md +159 -0
- package/skills/cache-guide/SKILL.md +243 -21
- package/skills/caching/SKILL.md +155 -6
- package/skills/composability/SKILL.md +27 -2
- package/skills/document-cache/SKILL.md +78 -55
- package/skills/handler-use/SKILL.md +364 -0
- package/skills/hooks/SKILL.md +229 -20
- package/skills/host-router/SKILL.md +45 -20
- package/skills/i18n/SKILL.md +276 -0
- package/skills/intercept/SKILL.md +46 -4
- package/skills/layout/SKILL.md +28 -7
- package/skills/links/SKILL.md +249 -17
- package/skills/loader/SKILL.md +273 -53
- package/skills/middleware/SKILL.md +49 -12
- package/skills/migrate-nextjs/SKILL.md +562 -0
- package/skills/migrate-react-router/SKILL.md +769 -0
- package/skills/mime-routes/SKILL.md +27 -0
- package/skills/observability/SKILL.md +137 -0
- package/skills/parallel/SKILL.md +197 -6
- package/skills/prerender/SKILL.md +123 -100
- package/skills/rango/SKILL.md +242 -22
- package/skills/react-compiler/SKILL.md +168 -0
- package/skills/response-routes/SKILL.md +66 -9
- package/skills/route/SKILL.md +88 -4
- package/skills/router-setup/SKILL.md +90 -5
- package/skills/server-actions/SKILL.md +751 -0
- package/skills/streams-and-websockets/SKILL.md +283 -0
- package/skills/testing/SKILL.md +716 -0
- package/skills/typesafety/SKILL.md +329 -27
- package/skills/use-cache/SKILL.md +34 -5
- package/skills/view-transitions/SKILL.md +294 -0
- package/src/__augment-tests__/augment.ts +81 -0
- package/src/__augment-tests__/augmented.check.ts +117 -0
- package/src/__internal.ts +1 -1
- package/src/browser/action-coordinator.ts +53 -36
- package/src/browser/app-shell.ts +52 -0
- package/src/browser/app-version.ts +14 -0
- package/src/browser/event-controller.ts +91 -70
- package/src/browser/history-state.ts +21 -0
- package/src/browser/index.ts +3 -3
- package/src/browser/navigation-bridge.ts +102 -16
- package/src/browser/navigation-client.ts +164 -59
- package/src/browser/navigation-store.ts +75 -17
- package/src/browser/navigation-transaction.ts +21 -37
- package/src/browser/partial-update.ts +139 -38
- package/src/browser/prefetch/cache.ts +175 -15
- package/src/browser/prefetch/fetch.ts +180 -33
- package/src/browser/prefetch/queue.ts +123 -20
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/rango-state.ts +53 -13
- package/src/browser/react/Link.tsx +81 -9
- package/src/browser/react/NavigationProvider.tsx +110 -33
- package/src/browser/react/context.ts +7 -2
- package/src/browser/react/filter-segment-order.ts +51 -7
- package/src/browser/react/index.ts +3 -0
- package/src/browser/react/location-state-shared.ts +175 -4
- package/src/browser/react/location-state.ts +39 -13
- package/src/browser/react/use-handle.ts +23 -64
- package/src/browser/react/use-navigation.ts +22 -2
- package/src/browser/react/use-params.ts +20 -8
- package/src/browser/react/use-reverse.ts +106 -0
- package/src/browser/react/use-router.ts +43 -10
- package/src/browser/react/use-segments.ts +11 -8
- package/src/browser/response-adapter.ts +25 -0
- package/src/browser/rsc-router.tsx +191 -74
- package/src/browser/scroll-restoration.ts +41 -14
- package/src/browser/segment-reconciler.ts +36 -9
- package/src/browser/segment-structure-assert.ts +2 -2
- package/src/browser/server-action-bridge.ts +31 -36
- package/src/browser/types.ts +57 -5
- package/src/build/collect-fallback-refs.ts +107 -0
- package/src/build/generate-manifest.ts +65 -40
- package/src/build/generate-route-types.ts +5 -0
- package/src/build/index.ts +2 -0
- package/src/build/route-trie.ts +52 -25
- package/src/build/route-types/codegen.ts +4 -4
- package/src/build/route-types/include-resolution.ts +9 -2
- package/src/build/route-types/per-module-writer.ts +7 -4
- package/src/build/route-types/router-processing.ts +278 -88
- package/src/build/route-types/scan-filter.ts +9 -2
- package/src/build/route-types/source-scan.ts +118 -0
- package/src/build/runtime-discovery.ts +9 -20
- package/src/cache/cache-runtime.ts +15 -11
- package/src/cache/cache-scope.ts +76 -49
- package/src/cache/cf/cf-cache-store.ts +501 -18
- package/src/cache/cf/index.ts +5 -1
- package/src/cache/document-cache.ts +17 -7
- package/src/cache/index.ts +1 -0
- package/src/cache/taint.ts +55 -0
- package/src/client.rsc.tsx +3 -0
- package/src/client.tsx +94 -238
- package/src/context-var.ts +72 -2
- package/src/debug.ts +2 -2
- package/src/decode-loader-results.ts +36 -0
- package/src/errors.ts +30 -1
- package/src/handle.ts +65 -12
- package/src/host/index.ts +2 -2
- package/src/host/router.ts +129 -57
- package/src/host/types.ts +31 -2
- package/src/host/utils.ts +1 -1
- package/src/href-client.ts +140 -20
- package/src/index.rsc.ts +12 -5
- package/src/index.ts +61 -11
- package/src/loader-store.ts +500 -0
- package/src/loader.rsc.ts +2 -5
- package/src/loader.ts +3 -10
- package/src/missing-id-error.ts +68 -0
- package/src/outlet-context.ts +1 -1
- package/src/prerender/store.ts +5 -4
- package/src/prerender.ts +141 -80
- package/src/response-utils.ts +37 -0
- package/src/reverse.ts +65 -15
- package/src/route-content-wrapper.tsx +6 -28
- package/src/route-definition/dsl-helpers.ts +435 -260
- package/src/route-definition/helper-factories.ts +29 -139
- package/src/route-definition/helpers-types.ts +110 -34
- package/src/route-definition/index.ts +3 -0
- package/src/route-definition/redirect.ts +11 -3
- package/src/route-definition/resolve-handler-use.ts +155 -0
- package/src/route-definition/use-item-types.ts +32 -0
- package/src/route-map-builder.ts +7 -1
- package/src/route-types.ts +37 -41
- package/src/router/basename.ts +14 -0
- package/src/router/content-negotiation.ts +113 -1
- package/src/router/error-handling.ts +1 -1
- package/src/router/find-match.ts +4 -2
- package/src/router/handler-context.ts +77 -38
- package/src/router/intercept-resolution.ts +15 -22
- package/src/router/lazy-includes.ts +12 -9
- package/src/router/loader-resolution.ts +174 -22
- package/src/router/logging.ts +5 -2
- package/src/router/manifest.ts +31 -16
- package/src/router/match-api.ts +128 -192
- package/src/router/match-handlers.ts +63 -20
- package/src/router/match-middleware/background-revalidation.ts +30 -2
- package/src/router/match-middleware/cache-lookup.ts +136 -106
- package/src/router/match-middleware/cache-store.ts +54 -10
- package/src/router/match-middleware/intercept-resolution.ts +9 -7
- package/src/router/match-middleware/segment-resolution.ts +61 -5
- package/src/router/match-result.ts +125 -10
- package/src/router/metrics.ts +7 -2
- package/src/router/middleware-types.ts +21 -34
- package/src/router/middleware.ts +103 -90
- package/src/router/navigation-snapshot.ts +182 -0
- package/src/router/pattern-matching.ts +101 -17
- package/src/router/prerender-match.ts +110 -10
- package/src/router/preview-match.ts +32 -102
- package/src/router/request-classification.ts +286 -0
- package/src/router/revalidation.ts +58 -2
- package/src/router/route-snapshot.ts +245 -0
- package/src/router/router-context.ts +6 -1
- package/src/router/router-interfaces.ts +77 -28
- package/src/router/router-options.ts +76 -11
- package/src/router/router-registry.ts +2 -5
- package/src/router/segment-resolution/fresh.ts +223 -24
- package/src/router/segment-resolution/helpers.ts +29 -24
- package/src/router/segment-resolution/loader-cache.ts +1 -0
- package/src/router/segment-resolution/revalidation.ts +466 -285
- package/src/router/segment-resolution/view-transition-default.ts +36 -0
- package/src/router/segment-wrappers.ts +2 -0
- package/src/router/substitute-pattern-params.ts +56 -0
- package/src/router/telemetry.ts +99 -0
- package/src/router/trie-matching.ts +18 -13
- package/src/router/types.ts +9 -0
- package/src/router/url-params.ts +49 -0
- package/src/router.ts +91 -23
- package/src/rsc/handler-context.ts +2 -2
- package/src/rsc/handler.ts +440 -381
- package/src/rsc/helpers.ts +91 -43
- package/src/rsc/index.ts +1 -1
- package/src/rsc/loader-fetch.ts +23 -3
- package/src/rsc/manifest-init.ts +5 -1
- package/src/rsc/origin-guard.ts +28 -10
- package/src/rsc/progressive-enhancement.ts +18 -2
- package/src/rsc/response-route-handler.ts +46 -53
- package/src/rsc/rsc-rendering.ts +41 -48
- package/src/rsc/runtime-warnings.ts +9 -10
- package/src/rsc/server-action.ts +25 -37
- package/src/rsc/ssr-setup.ts +18 -2
- package/src/rsc/types.ts +17 -3
- package/src/search-params.ts +4 -4
- package/src/segment-content-promise.ts +67 -0
- package/src/segment-loader-promise.ts +122 -0
- package/src/segment-system.tsx +219 -67
- package/src/serialize.ts +243 -0
- package/src/server/context.ts +277 -61
- package/src/server/cookie-store.ts +28 -4
- package/src/server/handle-store.ts +19 -0
- package/src/server/loader-registry.ts +9 -8
- package/src/server/request-context.ts +204 -60
- package/src/ssr/index.tsx +9 -1
- package/src/static-handler.ts +19 -7
- package/src/testing/cache-status.ts +166 -0
- package/src/testing/collect-handle.ts +63 -0
- package/src/testing/dispatch.ts +440 -0
- package/src/testing/dom.entry.ts +22 -0
- package/src/testing/e2e/fixture.ts +154 -0
- package/src/testing/e2e/index.ts +149 -0
- package/src/testing/e2e/matchers.ts +51 -0
- package/src/testing/e2e/page-helpers.ts +272 -0
- package/src/testing/e2e/parity.ts +306 -0
- package/src/testing/e2e/server.ts +183 -0
- package/src/testing/flight-matchers.ts +104 -0
- package/src/testing/flight-runtime.d.ts +21 -0
- package/src/testing/flight.entry.ts +22 -0
- package/src/testing/flight.ts +182 -0
- package/src/testing/generated-routes.ts +223 -0
- package/src/testing/index.ts +106 -0
- package/src/testing/internal/context.ts +255 -0
- package/src/testing/render-route.tsx +565 -0
- package/src/testing/run-loader.ts +296 -0
- package/src/testing/run-middleware.ts +179 -0
- package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
- package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
- package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
- package/src/testing/vitest-stubs/version.ts +5 -0
- package/src/testing/vitest.ts +183 -0
- package/src/types/cache-types.ts +4 -4
- package/src/types/global-namespace.ts +39 -26
- package/src/types/handler-context.ts +194 -72
- package/src/types/index.ts +1 -0
- package/src/types/loader-types.ts +41 -15
- package/src/types/request-scope.ts +126 -0
- package/src/types/route-entry.ts +19 -1
- package/src/types/segments.ts +37 -1
- package/src/urls/include-helper.ts +34 -67
- package/src/urls/index.ts +0 -3
- package/src/urls/path-helper-types.ts +50 -9
- package/src/urls/path-helper.ts +63 -63
- package/src/urls/pattern-types.ts +48 -19
- package/src/urls/response-types.ts +25 -22
- package/src/urls/type-extraction.ts +26 -116
- package/src/urls/urls-function.ts +1 -5
- package/src/use-loader.tsx +487 -44
- package/src/vite/debug.ts +185 -0
- package/src/vite/discovery/bundle-postprocess.ts +34 -37
- package/src/vite/discovery/discover-routers.ts +105 -51
- package/src/vite/discovery/discovery-errors.ts +194 -0
- package/src/vite/discovery/gate-state.ts +171 -0
- package/src/vite/discovery/prerender-collection.ts +188 -93
- package/src/vite/discovery/route-types-writer.ts +40 -84
- package/src/vite/discovery/self-gen-tracking.ts +27 -1
- package/src/vite/discovery/state.ts +46 -6
- package/src/vite/discovery/virtual-module-codegen.ts +13 -23
- package/src/vite/index.ts +6 -0
- package/src/vite/plugin-types.ts +111 -72
- package/src/vite/plugins/cjs-to-esm.ts +8 -7
- package/src/vite/plugins/client-ref-dedup.ts +16 -0
- package/src/vite/plugins/client-ref-hashing.ts +28 -5
- 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/plugins/expose-action-id.ts +55 -33
- package/src/vite/plugins/expose-id-utils.ts +24 -8
- package/src/vite/plugins/expose-ids/export-analysis.ts +100 -20
- package/src/vite/plugins/expose-ids/handler-transform.ts +12 -35
- package/src/vite/plugins/expose-ids/loader-transform.ts +3 -5
- package/src/vite/plugins/expose-ids/router-transform.ts +20 -3
- package/src/vite/plugins/expose-internal-ids.ts +544 -317
- package/src/vite/plugins/performance-tracks.ts +92 -0
- package/src/vite/plugins/refresh-cmd.ts +88 -26
- package/src/vite/plugins/use-cache-transform.ts +65 -50
- package/src/vite/plugins/version-injector.ts +39 -23
- package/src/vite/plugins/version-plugin.ts +72 -3
- package/src/vite/plugins/virtual-entries.ts +2 -2
- package/src/vite/rango.ts +265 -226
- package/src/vite/router-discovery.ts +920 -137
- package/src/vite/utils/ast-handler-extract.ts +15 -15
- package/src/vite/utils/banner.ts +4 -4
- package/src/vite/utils/bundle-analysis.ts +4 -2
- package/src/vite/utils/client-chunks.ts +190 -0
- package/src/vite/utils/forward-user-plugins.ts +193 -0
- package/src/vite/utils/manifest-utils.ts +21 -5
- package/src/vite/utils/package-resolution.ts +41 -1
- package/src/vite/utils/prerender-utils.ts +38 -5
- package/src/vite/utils/shared-utils.ts +109 -27
- package/src/browser/action-response-classifier.ts +0 -99
package/src/debug.ts
CHANGED
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
* Debug utilities for manifest inspection and comparison
|
|
3
3
|
*/
|
|
4
4
|
|
|
5
|
-
import type
|
|
5
|
+
import { getParallelSlotCount, type EntryData } from "./server/context";
|
|
6
6
|
|
|
7
7
|
/**
|
|
8
8
|
* Serialized entry for debug output
|
|
@@ -64,7 +64,7 @@ export function serializeManifest(
|
|
|
64
64
|
hasLoader: entry.loader?.length > 0,
|
|
65
65
|
hasMiddleware: entry.middleware?.length > 0,
|
|
66
66
|
hasErrorBoundary: entry.errorBoundary?.length > 0,
|
|
67
|
-
parallelCount: entry.parallel
|
|
67
|
+
parallelCount: getParallelSlotCount(entry.parallel),
|
|
68
68
|
interceptCount: entry.intercept?.length ?? 0,
|
|
69
69
|
};
|
|
70
70
|
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import type { ReactNode } from "react";
|
|
2
|
+
import { isLoaderDataResult } from "./types.js";
|
|
3
|
+
|
|
4
|
+
// Shared by segment-system (server) and LoaderResolver (client) so the
|
|
5
|
+
// legacy/ok/error-fallback/throw decode of resolved loader values lives once.
|
|
6
|
+
// Last failing loader wins errorFallback; an error without a fallback throws.
|
|
7
|
+
export function decodeLoaderResults(
|
|
8
|
+
resolvedData: any[],
|
|
9
|
+
loaderIds: string[],
|
|
10
|
+
): { loaderData: Record<string, any>; errorFallback: ReactNode } {
|
|
11
|
+
const loaderData: Record<string, any> = {};
|
|
12
|
+
let errorFallback: ReactNode = null;
|
|
13
|
+
|
|
14
|
+
for (let i = 0; i < loaderIds.length; i++) {
|
|
15
|
+
const id = loaderIds[i];
|
|
16
|
+
const result = resolvedData[i];
|
|
17
|
+
|
|
18
|
+
if (!isLoaderDataResult(result)) {
|
|
19
|
+
loaderData[id] = result;
|
|
20
|
+
continue;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
if (result.ok) {
|
|
24
|
+
loaderData[id] = result.data;
|
|
25
|
+
continue;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
if (result.fallback) {
|
|
29
|
+
errorFallback = result.fallback;
|
|
30
|
+
} else {
|
|
31
|
+
throw new Error(result.error.message);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
return { loaderData, errorFallback };
|
|
36
|
+
}
|
package/src/errors.ts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Custom error classes for
|
|
2
|
+
* Custom error classes for Rango
|
|
3
3
|
*
|
|
4
4
|
* All errors include:
|
|
5
5
|
* - Descriptive names for easy identification
|
|
@@ -27,6 +27,17 @@ export class RouteNotFoundError extends Error {
|
|
|
27
27
|
}
|
|
28
28
|
}
|
|
29
29
|
|
|
30
|
+
// name fallback covers cross-realm errors (Vite dev dupes, RSC serialization)
|
|
31
|
+
// where instanceof fails.
|
|
32
|
+
export function isRouteNotFoundError(
|
|
33
|
+
error: unknown,
|
|
34
|
+
): error is RouteNotFoundError {
|
|
35
|
+
return (
|
|
36
|
+
error instanceof RouteNotFoundError ||
|
|
37
|
+
(error instanceof Error && error.name === "RouteNotFoundError")
|
|
38
|
+
);
|
|
39
|
+
}
|
|
40
|
+
|
|
30
41
|
/**
|
|
31
42
|
* Thrown when data is not found (e.g., product with ID doesn't exist)
|
|
32
43
|
* Use this in handlers/loaders to trigger the nearest notFoundBoundary
|
|
@@ -109,6 +120,24 @@ export class BuildError extends Error {
|
|
|
109
120
|
}
|
|
110
121
|
}
|
|
111
122
|
|
|
123
|
+
/**
|
|
124
|
+
* Thrown when a route-definition DSL helper (route/layout/loader/cache/…) is
|
|
125
|
+
* called outside an active urls()/map() builder, so there is no
|
|
126
|
+
* AsyncLocalStorage build context to attach to. The message names the specific
|
|
127
|
+
* helper and how to fix it; the `cause` records the mechanical reason so the
|
|
128
|
+
* failure mode is identifiable (not conflated with an unrelated throw).
|
|
129
|
+
*/
|
|
130
|
+
export class DslContextError extends Error {
|
|
131
|
+
name = "DslContextError" as const;
|
|
132
|
+
cause?: unknown;
|
|
133
|
+
|
|
134
|
+
constructor(message: string, options?: ErrorOptions) {
|
|
135
|
+
super(message);
|
|
136
|
+
Object.setPrototypeOf(this, DslContextError.prototype);
|
|
137
|
+
this.cause = options?.cause;
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
|
|
112
141
|
/**
|
|
113
142
|
* Thrown when a network request fails (server unreachable, no internet, etc.)
|
|
114
143
|
* This error triggers the root error boundary with retry capability.
|
package/src/handle.ts
CHANGED
|
@@ -1,3 +1,5 @@
|
|
|
1
|
+
import { missingInjectedIdError } from "./missing-id-error.js";
|
|
2
|
+
|
|
1
3
|
/**
|
|
2
4
|
* Handle definition for accumulating data across route segments.
|
|
3
5
|
*
|
|
@@ -43,6 +45,11 @@ function defaultCollect<T>(segments: T[][]): T[] {
|
|
|
43
45
|
// Used by useHandle() to recover collect when handle is deserialized from RSC prop.
|
|
44
46
|
const collectRegistry = new Map<string, (segments: unknown[][]) => unknown>();
|
|
45
47
|
|
|
48
|
+
// Monotonic counter for runtime fallback ids (see createHandle). Module-scoped
|
|
49
|
+
// and deterministic, so each createHandle() call gets a stable, unique id within
|
|
50
|
+
// the process. Only used when no build id was injected (a bare unit test).
|
|
51
|
+
let runtimeHandleIdCounter = 0;
|
|
52
|
+
|
|
46
53
|
/**
|
|
47
54
|
* Look up a collect function from the registry by handle $$id.
|
|
48
55
|
* Returns undefined if not registered (falls back to defaultCollect in useHandle).
|
|
@@ -93,14 +100,22 @@ export function createHandle<TData, TAccumulated = TData[]>(
|
|
|
93
100
|
collect?: (segments: TData[][]) => TAccumulated,
|
|
94
101
|
__injectedId?: string,
|
|
95
102
|
): Handle<TData, TAccumulated> {
|
|
96
|
-
|
|
103
|
+
let handleId = __injectedId ?? "";
|
|
97
104
|
|
|
98
105
|
if (!handleId && process.env.NODE_ENV === "development") {
|
|
99
|
-
throw
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
106
|
+
throw missingInjectedIdError("Handle", "createHandle");
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
// No build-injected id. This only happens in a bare unit test — every real
|
|
110
|
+
// build runs the rango Vite plugin, which always injects a stable id (and the
|
|
111
|
+
// line above throws for a genuinely non-exported handle in dev). Assign a
|
|
112
|
+
// process-stable runtime id so the collect registers below and the handle is
|
|
113
|
+
// fully exercisable in tests (useHandle, collectHandle, renderRoute's `handles`
|
|
114
|
+
// seeding run the REAL collect). Provably inert in production: the fallback
|
|
115
|
+
// never triggers when the plugin injects the id, so server/client id
|
|
116
|
+
// consistency (required for RSC recovery) is unaffected.
|
|
117
|
+
if (!handleId) {
|
|
118
|
+
handleId = `__rango_runtime_handle_${runtimeHandleIdCounter++}`;
|
|
104
119
|
}
|
|
105
120
|
|
|
106
121
|
const collectFn =
|
|
@@ -109,12 +124,10 @@ export function createHandle<TData, TAccumulated = TData[]>(
|
|
|
109
124
|
|
|
110
125
|
// Register collect in module-level registry so useHandle() can recover it
|
|
111
126
|
// when the handle is deserialized from RSC props (toJSON strips collect).
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
);
|
|
117
|
-
}
|
|
127
|
+
collectRegistry.set(
|
|
128
|
+
handleId,
|
|
129
|
+
collectFn as (segments: unknown[][]) => unknown,
|
|
130
|
+
);
|
|
118
131
|
|
|
119
132
|
return {
|
|
120
133
|
__brand: "handle" as const,
|
|
@@ -133,3 +146,43 @@ export function isHandle(value: unknown): value is Handle<unknown, unknown> {
|
|
|
133
146
|
(value as { __brand: unknown }).__brand === "handle"
|
|
134
147
|
);
|
|
135
148
|
}
|
|
149
|
+
|
|
150
|
+
/**
|
|
151
|
+
* Collect handle data from a HandleData map, applying the handle's collect
|
|
152
|
+
* function over segments in order. Shared between server-side rendered()
|
|
153
|
+
* reads and client-side useHandle().
|
|
154
|
+
*
|
|
155
|
+
* @param handle - The handle to collect data for
|
|
156
|
+
* @param data - Full handle data map (handleName -> segmentId -> entries[])
|
|
157
|
+
* @param segmentOrder - Segment IDs in parent -> child resolution order
|
|
158
|
+
*/
|
|
159
|
+
export function collectHandleData<TData, TAccumulated>(
|
|
160
|
+
handle: Handle<TData, TAccumulated>,
|
|
161
|
+
data: Record<string, Record<string, unknown[]>>,
|
|
162
|
+
segmentOrder: string[],
|
|
163
|
+
): TAccumulated {
|
|
164
|
+
const collectFn = getCollectFn(handle.$$id);
|
|
165
|
+
if (!collectFn && process.env.NODE_ENV !== "production") {
|
|
166
|
+
console.warn(
|
|
167
|
+
`[rango] Handle "${handle.$$id}" has no registered collect function. ` +
|
|
168
|
+
`Falling back to flat array. Ensure the handle module is imported so ` +
|
|
169
|
+
`createHandle() runs and registers the collect function.`,
|
|
170
|
+
);
|
|
171
|
+
}
|
|
172
|
+
const collect = (collectFn ??
|
|
173
|
+
(defaultCollect as unknown as (segments: unknown[][]) => unknown)) as (
|
|
174
|
+
segments: TData[][],
|
|
175
|
+
) => TAccumulated;
|
|
176
|
+
|
|
177
|
+
const segmentData = data[handle.$$id];
|
|
178
|
+
if (!segmentData) return collect([]);
|
|
179
|
+
|
|
180
|
+
const segmentArrays: TData[][] = [];
|
|
181
|
+
for (const segmentId of segmentOrder) {
|
|
182
|
+
const entries = segmentData[segmentId];
|
|
183
|
+
if (entries && entries.length > 0) {
|
|
184
|
+
segmentArrays.push(entries as TData[]);
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
return collect(segmentArrays);
|
|
188
|
+
}
|
package/src/host/index.ts
CHANGED
|
@@ -11,8 +11,8 @@
|
|
|
11
11
|
*
|
|
12
12
|
* const router = createHostRouter();
|
|
13
13
|
*
|
|
14
|
-
* router.host(['.']).
|
|
15
|
-
* router.host(['admin.*']).
|
|
14
|
+
* router.host(['.']).lazy(() => import('./apps/main'));
|
|
15
|
+
* router.host(['admin.*']).lazy(() => import('./apps/admin'));
|
|
16
16
|
*
|
|
17
17
|
* export default {
|
|
18
18
|
* fetch(request) {
|
package/src/host/router.ts
CHANGED
|
@@ -52,6 +52,34 @@ export const HostRouterRegistry: Map<string, HostRouterRegistryEntry> =
|
|
|
52
52
|
|
|
53
53
|
let hostRouterAutoId = 0;
|
|
54
54
|
|
|
55
|
+
/** Whether a value is thenable (a Promise or Promise-like). */
|
|
56
|
+
function isThenable(value: unknown): value is PromiseLike<unknown> {
|
|
57
|
+
return (
|
|
58
|
+
value !== null &&
|
|
59
|
+
(typeof value === "object" || typeof value === "function") &&
|
|
60
|
+
typeof (value as { then?: unknown }).then === "function"
|
|
61
|
+
);
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Whether a resolved value looks like a module namespace from a lazy import -
|
|
66
|
+
* an object with a `default` export that is a function (a Handler) or a host
|
|
67
|
+
* router (an object with `match`). Used to detect a `.map(() => import(...))`
|
|
68
|
+
* misuse: an inline handler should return a Response, not a module.
|
|
69
|
+
*/
|
|
70
|
+
function looksLikeLazyModule(value: unknown): boolean {
|
|
71
|
+
if (value === null || typeof value !== "object" || !("default" in value)) {
|
|
72
|
+
return false;
|
|
73
|
+
}
|
|
74
|
+
const defaultExport = (value as { default: unknown }).default;
|
|
75
|
+
return (
|
|
76
|
+
typeof defaultExport === "function" ||
|
|
77
|
+
(typeof defaultExport === "object" &&
|
|
78
|
+
defaultExport !== null &&
|
|
79
|
+
"match" in defaultExport)
|
|
80
|
+
);
|
|
81
|
+
}
|
|
82
|
+
|
|
55
83
|
/**
|
|
56
84
|
* Create a host router
|
|
57
85
|
*/
|
|
@@ -77,32 +105,44 @@ export function createHostRouter(options: HostRouterOptions = {}): HostRouter {
|
|
|
77
105
|
): HostRouteBuilder {
|
|
78
106
|
const middleware: Middleware[] = [];
|
|
79
107
|
|
|
108
|
+
function register(
|
|
109
|
+
handler: Handler | LazyHandler,
|
|
110
|
+
kind: RouteEntry["kind"],
|
|
111
|
+
): HostRouter {
|
|
112
|
+
const entry: RouteEntry = {
|
|
113
|
+
patterns,
|
|
114
|
+
middleware,
|
|
115
|
+
handler,
|
|
116
|
+
kind,
|
|
117
|
+
isFallback,
|
|
118
|
+
};
|
|
119
|
+
|
|
120
|
+
if (isFallback) {
|
|
121
|
+
fallbackRoute = entry;
|
|
122
|
+
} else {
|
|
123
|
+
routes.push(entry);
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
log(
|
|
127
|
+
`Registered ${isFallback ? "fallback" : "route"} (${kind}):`,
|
|
128
|
+
patterns.join(", "),
|
|
129
|
+
);
|
|
130
|
+
|
|
131
|
+
return router;
|
|
132
|
+
}
|
|
133
|
+
|
|
80
134
|
return {
|
|
81
135
|
use(...mw: Middleware[]): HostRouteBuilder {
|
|
82
136
|
middleware.push(...mw);
|
|
83
137
|
return this;
|
|
84
138
|
},
|
|
85
139
|
|
|
86
|
-
map(handler: Handler
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
middleware,
|
|
90
|
-
handler,
|
|
91
|
-
isFallback,
|
|
92
|
-
};
|
|
93
|
-
|
|
94
|
-
if (isFallback) {
|
|
95
|
-
fallbackRoute = entry;
|
|
96
|
-
} else {
|
|
97
|
-
routes.push(entry);
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
log(
|
|
101
|
-
`Registered ${isFallback ? "fallback" : "route"}:`,
|
|
102
|
-
patterns.join(", "),
|
|
103
|
-
);
|
|
140
|
+
map(handler: Handler): HostRouter {
|
|
141
|
+
return register(handler, "handler");
|
|
142
|
+
},
|
|
104
143
|
|
|
105
|
-
|
|
144
|
+
lazy(handler: LazyHandler): HostRouter {
|
|
145
|
+
return register(handler, "lazy");
|
|
106
146
|
},
|
|
107
147
|
};
|
|
108
148
|
}
|
|
@@ -169,50 +209,82 @@ export function createHostRouter(options: HostRouterOptions = {}): HostRouter {
|
|
|
169
209
|
}
|
|
170
210
|
|
|
171
211
|
/**
|
|
172
|
-
* Execute
|
|
212
|
+
* Execute a route entry, branching on its declared kind:
|
|
213
|
+
* - "lazy": await the loader, then delegate to the default export
|
|
214
|
+
* (a nested HostRouter via `.match`, or a request Handler directly).
|
|
215
|
+
* - "handler": call the inline handler with the request. A `.map()` handler
|
|
216
|
+
* that resolves to a module namespace (`{ default }`) is almost certainly
|
|
217
|
+
* a misused lazy import, so it is rejected with a clear message rather
|
|
218
|
+
* than silently returning a module object as the response.
|
|
173
219
|
*/
|
|
174
220
|
async function executeHandler(
|
|
175
|
-
|
|
221
|
+
entry: RouteEntry,
|
|
176
222
|
request: Request,
|
|
177
223
|
input: RouterRequestInput<any>,
|
|
178
224
|
): Promise<Response> {
|
|
179
|
-
|
|
180
|
-
if (typeof handler === "function") {
|
|
181
|
-
const result = handler(request, input);
|
|
182
|
-
|
|
183
|
-
// If it returns a promise with default export
|
|
184
|
-
if (result && typeof result === "object" && "then" in result) {
|
|
185
|
-
const module = await result;
|
|
186
|
-
if (
|
|
187
|
-
typeof module === "object" &&
|
|
188
|
-
module !== null &&
|
|
189
|
-
"default" in module
|
|
190
|
-
) {
|
|
191
|
-
const defaultExport = (module as { default: Handler | HostRouter })
|
|
192
|
-
.default;
|
|
193
|
-
|
|
194
|
-
// If default export is a router with match method
|
|
195
|
-
if (
|
|
196
|
-
typeof defaultExport === "object" &&
|
|
197
|
-
defaultExport !== null &&
|
|
198
|
-
"match" in defaultExport
|
|
199
|
-
) {
|
|
200
|
-
return (defaultExport as HostRouter).match(request, input);
|
|
201
|
-
}
|
|
225
|
+
const { handler, kind } = entry;
|
|
202
226
|
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
}
|
|
206
|
-
|
|
207
|
-
|
|
227
|
+
if (typeof handler !== "function") {
|
|
228
|
+
throw new InvalidHandlerError(handler, {
|
|
229
|
+
cause: { handlerType: typeof handler },
|
|
230
|
+
});
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
if (kind === "lazy") {
|
|
234
|
+
return executeLazyMount(handler as LazyHandler, request, input);
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
const result = (handler as Handler)(request, input);
|
|
238
|
+
|
|
239
|
+
// Inline handlers may be async; await to obtain the Response and to run the
|
|
240
|
+
// misuse guard below.
|
|
241
|
+
if (isThenable(result)) {
|
|
242
|
+
const awaited = await result;
|
|
243
|
+
if (looksLikeLazyModule(awaited)) {
|
|
244
|
+
throw new HostRouterError(
|
|
245
|
+
".map() is for inline request handlers; use .lazy(() => import(...)) for lazy host mounts.",
|
|
246
|
+
);
|
|
247
|
+
}
|
|
248
|
+
return awaited as Response;
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
return result;
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
/**
|
|
255
|
+
* Resolve a `.lazy()` mount: invoke the zero-arg loader, then dispatch to the
|
|
256
|
+
* module's default export.
|
|
257
|
+
*/
|
|
258
|
+
async function executeLazyMount(
|
|
259
|
+
loader: LazyHandler,
|
|
260
|
+
request: Request,
|
|
261
|
+
input: RouterRequestInput<any>,
|
|
262
|
+
): Promise<Response> {
|
|
263
|
+
const module = await loader();
|
|
264
|
+
|
|
265
|
+
if (typeof module === "object" && module !== null && "default" in module) {
|
|
266
|
+
const defaultExport = (module as { default: Handler | HostRouter })
|
|
267
|
+
.default;
|
|
268
|
+
|
|
269
|
+
// Default export is a nested host router
|
|
270
|
+
if (
|
|
271
|
+
typeof defaultExport === "object" &&
|
|
272
|
+
defaultExport !== null &&
|
|
273
|
+
"match" in defaultExport
|
|
274
|
+
) {
|
|
275
|
+
return (defaultExport as HostRouter).match(request, input);
|
|
208
276
|
}
|
|
209
277
|
|
|
210
|
-
//
|
|
211
|
-
return
|
|
278
|
+
// Otherwise treat the default export as a request handler
|
|
279
|
+
return (defaultExport as Handler)(request, input);
|
|
212
280
|
}
|
|
213
281
|
|
|
214
|
-
throw new InvalidHandlerError(
|
|
215
|
-
cause: {
|
|
282
|
+
throw new InvalidHandlerError(loader, {
|
|
283
|
+
cause: {
|
|
284
|
+
reason:
|
|
285
|
+
"lazy mount did not resolve to a module with a default export; " +
|
|
286
|
+
"use .lazy(() => import('./sub-app')) where the module default-exports a handler or host router",
|
|
287
|
+
},
|
|
216
288
|
});
|
|
217
289
|
}
|
|
218
290
|
|
|
@@ -252,6 +324,7 @@ export function createHostRouter(options: HostRouterOptions = {}): HostRouter {
|
|
|
252
324
|
return {
|
|
253
325
|
pattern,
|
|
254
326
|
handler: route.handler,
|
|
327
|
+
kind: route.kind,
|
|
255
328
|
};
|
|
256
329
|
}
|
|
257
330
|
}
|
|
@@ -288,8 +361,7 @@ export function createHostRouter(options: HostRouterOptions = {}): HostRouter {
|
|
|
288
361
|
allMiddleware,
|
|
289
362
|
request,
|
|
290
363
|
fallbackInput,
|
|
291
|
-
() =>
|
|
292
|
-
executeHandler(fallbackRoute!.handler, request, fallbackInput),
|
|
364
|
+
() => executeHandler(fallbackRoute!, request, fallbackInput),
|
|
293
365
|
);
|
|
294
366
|
}
|
|
295
367
|
|
|
@@ -330,14 +402,14 @@ export function createHostRouter(options: HostRouterOptions = {}): HostRouter {
|
|
|
330
402
|
|
|
331
403
|
// Execute middleware chain and handler
|
|
332
404
|
return executeMiddleware(allMiddleware, request, input, () =>
|
|
333
|
-
executeHandler(matchedRoute
|
|
405
|
+
executeHandler(matchedRoute, request, input),
|
|
334
406
|
);
|
|
335
407
|
},
|
|
336
408
|
};
|
|
337
409
|
|
|
338
410
|
// Register in the global HostRouterRegistry for build-time discovery.
|
|
339
411
|
// The routes array and fallbackRoute ref are live - they reflect routes
|
|
340
|
-
// added via .host().map() after this point.
|
|
412
|
+
// added via .host().map()/.lazy() after this point.
|
|
341
413
|
const registryId = `host-router-${hostRouterAutoId++}`;
|
|
342
414
|
HostRouterRegistry.set(registryId, {
|
|
343
415
|
get routes() {
|
package/src/host/types.ts
CHANGED
|
@@ -35,12 +35,24 @@ export type Middleware = (
|
|
|
35
35
|
*/
|
|
36
36
|
export type HostPattern = string | string[];
|
|
37
37
|
|
|
38
|
+
/**
|
|
39
|
+
* Whether a route entry is an inline request handler or a lazy module mount.
|
|
40
|
+
*
|
|
41
|
+
* Stored on the entry so discovery and runtime act on the consumer's declared
|
|
42
|
+
* intent instead of inferring it from the function's shape (arity/return value),
|
|
43
|
+
* which is ambiguous: a lazy loader may declare an ignored param, and an inline
|
|
44
|
+
* handler may be async. `.map()` registers `"handler"`, `.lazy()` registers
|
|
45
|
+
* `"lazy"`.
|
|
46
|
+
*/
|
|
47
|
+
export type RouteEntryKind = "handler" | "lazy";
|
|
48
|
+
|
|
38
49
|
/**
|
|
39
50
|
* Result from testing a hostname against patterns
|
|
40
51
|
*/
|
|
41
52
|
export interface HostMatchResult {
|
|
42
53
|
pattern: string;
|
|
43
54
|
handler: Handler | LazyHandler;
|
|
55
|
+
kind: RouteEntryKind;
|
|
44
56
|
}
|
|
45
57
|
|
|
46
58
|
/**
|
|
@@ -53,9 +65,24 @@ export interface HostRouteBuilder {
|
|
|
53
65
|
use(...middleware: Middleware[]): HostRouteBuilder;
|
|
54
66
|
|
|
55
67
|
/**
|
|
56
|
-
* Map to
|
|
68
|
+
* Map to an inline request handler `(request, input) => Response`.
|
|
69
|
+
*
|
|
70
|
+
* For a lazily-imported sub-app or handler module, use {@link lazy} instead -
|
|
71
|
+
* `.map(() => import(...))` is rejected (the return type is not a `Response`)
|
|
72
|
+
* and would not be discovered at build time.
|
|
73
|
+
*/
|
|
74
|
+
map(handler: Handler): HostRouter;
|
|
75
|
+
|
|
76
|
+
/**
|
|
77
|
+
* Mount a lazily-imported handler or host router:
|
|
78
|
+
* `.lazy(() => import("./sub-app"))`.
|
|
79
|
+
*
|
|
80
|
+
* The loader takes no arguments and resolves to a module whose `default`
|
|
81
|
+
* export is a request `Handler` or a nested `HostRouter`. Only `.lazy()`
|
|
82
|
+
* entries are invoked during build-time discovery to trigger the sub-app's
|
|
83
|
+
* `createRouter()` registration.
|
|
57
84
|
*/
|
|
58
|
-
|
|
85
|
+
lazy(handler: LazyHandler): HostRouter;
|
|
59
86
|
}
|
|
60
87
|
|
|
61
88
|
/**
|
|
@@ -134,6 +161,8 @@ export interface RouteEntry {
|
|
|
134
161
|
patterns: string[];
|
|
135
162
|
middleware: Middleware[];
|
|
136
163
|
handler: Handler | LazyHandler;
|
|
164
|
+
/** Whether `handler` is an inline request handler or a lazy module mount. */
|
|
165
|
+
kind: RouteEntryKind;
|
|
137
166
|
isFallback?: boolean;
|
|
138
167
|
}
|
|
139
168
|
|
package/src/host/utils.ts
CHANGED
|
@@ -15,7 +15,7 @@
|
|
|
15
15
|
* app: ['*', 'www.*']
|
|
16
16
|
* });
|
|
17
17
|
*
|
|
18
|
-
* router.host(hosts.admin).
|
|
18
|
+
* router.host(hosts.admin).lazy(() => import("./apps/admin")); // Type-safe!
|
|
19
19
|
* ```
|
|
20
20
|
*/
|
|
21
21
|
export function defineHosts<T extends Record<string, string | string[]>>(
|