@rangojs/router 0.0.0-experimental.8 → 0.0.0-experimental.8a4d0430
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/AGENTS.md +5 -0
- package/README.md +884 -4
- package/dist/bin/rango.js +1601 -0
- package/dist/vite/index.js +4474 -867
- package/package.json +60 -51
- package/skills/breadcrumbs/SKILL.md +250 -0
- package/skills/cache-guide/SKILL.md +262 -0
- package/skills/caching/SKILL.md +50 -21
- package/skills/composability/SKILL.md +172 -0
- package/skills/debug-manifest/SKILL.md +12 -8
- package/skills/document-cache/SKILL.md +18 -16
- package/skills/fonts/SKILL.md +167 -0
- package/skills/hooks/SKILL.md +334 -72
- package/skills/host-router/SKILL.md +218 -0
- package/skills/intercept/SKILL.md +131 -8
- package/skills/layout/SKILL.md +100 -3
- package/skills/links/SKILL.md +89 -30
- package/skills/loader/SKILL.md +388 -38
- package/skills/middleware/SKILL.md +171 -34
- package/skills/mime-routes/SKILL.md +128 -0
- package/skills/parallel/SKILL.md +78 -1
- package/skills/prerender/SKILL.md +643 -0
- package/skills/rango/SKILL.md +85 -16
- package/skills/response-routes/SKILL.md +411 -0
- package/skills/route/SKILL.md +226 -14
- package/skills/router-setup/SKILL.md +123 -30
- package/skills/tailwind/SKILL.md +129 -0
- package/skills/theme/SKILL.md +9 -8
- package/skills/typesafety/SKILL.md +318 -89
- package/skills/use-cache/SKILL.md +324 -0
- package/src/__internal.ts +102 -4
- package/src/bin/rango.ts +321 -0
- package/src/browser/action-coordinator.ts +97 -0
- package/src/browser/action-response-classifier.ts +99 -0
- package/src/browser/event-controller.ts +87 -64
- package/src/browser/history-state.ts +80 -0
- package/src/browser/intercept-utils.ts +52 -0
- package/src/browser/link-interceptor.ts +24 -4
- package/src/browser/logging.ts +55 -0
- package/src/browser/merge-segment-loaders.ts +20 -12
- package/src/browser/navigation-bridge.ts +285 -553
- package/src/browser/navigation-client.ts +124 -71
- package/src/browser/navigation-store.ts +33 -50
- package/src/browser/navigation-transaction.ts +295 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +258 -308
- package/src/browser/prefetch/cache.ts +146 -0
- package/src/browser/prefetch/fetch.ts +135 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +42 -0
- package/src/browser/prefetch/queue.ts +88 -0
- package/src/browser/rango-state.ts +112 -0
- package/src/browser/react/Link.tsx +185 -73
- package/src/browser/react/NavigationProvider.tsx +51 -11
- package/src/browser/react/context.ts +6 -0
- package/src/browser/react/filter-segment-order.ts +11 -0
- package/src/browser/react/index.ts +12 -12
- package/src/browser/react/location-state-shared.ts +95 -53
- package/src/browser/react/location-state.ts +60 -15
- package/src/browser/react/mount-context.ts +6 -1
- package/src/browser/react/nonce-context.ts +23 -0
- package/src/browser/react/shallow-equal.ts +27 -0
- package/src/browser/react/use-action.ts +29 -51
- package/src/browser/react/use-client-cache.ts +5 -3
- package/src/browser/react/use-handle.ts +32 -79
- package/src/browser/react/use-href.tsx +2 -2
- package/src/browser/react/use-link-status.ts +6 -5
- package/src/browser/react/use-navigation.ts +22 -63
- package/src/browser/react/use-params.ts +65 -0
- package/src/browser/react/use-pathname.ts +47 -0
- package/src/browser/react/use-router.ts +63 -0
- package/src/browser/react/use-search-params.ts +56 -0
- package/src/browser/react/use-segments.ts +80 -97
- package/src/browser/response-adapter.ts +73 -0
- package/src/browser/rsc-router.tsx +107 -26
- package/src/browser/scroll-restoration.ts +92 -16
- package/src/browser/segment-reconciler.ts +216 -0
- package/src/browser/segment-structure-assert.ts +16 -0
- package/src/browser/server-action-bridge.ts +504 -599
- package/src/browser/shallow.ts +6 -1
- package/src/browser/types.ts +109 -47
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +235 -24
- package/src/build/generate-route-types.ts +36 -0
- package/src/build/index.ts +13 -0
- package/src/build/route-trie.ts +265 -0
- package/src/build/route-types/ast-helpers.ts +25 -0
- package/src/build/route-types/ast-route-extraction.ts +98 -0
- package/src/build/route-types/codegen.ts +102 -0
- package/src/build/route-types/include-resolution.ts +411 -0
- package/src/build/route-types/param-extraction.ts +48 -0
- package/src/build/route-types/per-module-writer.ts +128 -0
- package/src/build/route-types/router-processing.ts +469 -0
- package/src/build/route-types/scan-filter.ts +78 -0
- package/src/build/runtime-discovery.ts +231 -0
- package/src/cache/background-task.ts +34 -0
- package/src/cache/cache-key-utils.ts +44 -0
- package/src/cache/cache-policy.ts +125 -0
- package/src/cache/cache-runtime.ts +338 -0
- package/src/cache/cache-scope.ts +120 -303
- package/src/cache/cf/cf-cache-store.ts +119 -7
- package/src/cache/cf/index.ts +8 -2
- package/src/cache/document-cache.ts +101 -72
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/handle-snapshot.ts +41 -0
- package/src/cache/index.ts +0 -15
- package/src/cache/memory-segment-store.ts +191 -13
- package/src/cache/profile-registry.ts +73 -0
- package/src/cache/read-through-swr.ts +134 -0
- package/src/cache/segment-codec.ts +256 -0
- package/src/cache/taint.ts +98 -0
- package/src/cache/types.ts +72 -122
- package/src/client.rsc.tsx +3 -1
- package/src/client.tsx +106 -126
- package/src/component-utils.ts +4 -4
- package/src/components/DefaultDocument.tsx +5 -1
- package/src/context-var.ts +86 -0
- package/src/debug.ts +17 -7
- package/src/errors.ts +108 -2
- package/src/handle.ts +15 -29
- package/src/handles/MetaTags.tsx +73 -20
- package/src/handles/breadcrumbs.ts +66 -0
- package/src/handles/index.ts +1 -0
- package/src/handles/meta.ts +30 -13
- package/src/host/cookie-handler.ts +21 -15
- package/src/host/errors.ts +8 -8
- package/src/host/index.ts +4 -7
- package/src/host/pattern-matcher.ts +27 -27
- package/src/host/router.ts +61 -39
- package/src/host/testing.ts +8 -8
- package/src/host/types.ts +15 -7
- package/src/host/utils.ts +1 -1
- package/src/href-client.ts +119 -29
- package/src/index.rsc.ts +153 -19
- package/src/index.ts +211 -30
- package/src/internal-debug.ts +11 -0
- package/src/loader.rsc.ts +26 -157
- package/src/loader.ts +27 -10
- package/src/network-error-thrower.tsx +3 -1
- package/src/outlet-provider.tsx +45 -0
- package/src/prerender/param-hash.ts +37 -0
- package/src/prerender/store.ts +185 -0
- package/src/prerender.ts +463 -0
- package/src/reverse.ts +330 -0
- package/src/root-error-boundary.tsx +41 -29
- package/src/route-content-wrapper.tsx +7 -4
- package/src/route-definition/dsl-helpers.ts +934 -0
- package/src/route-definition/helper-factories.ts +200 -0
- package/src/route-definition/helpers-types.ts +430 -0
- package/src/route-definition/index.ts +52 -0
- package/src/route-definition/redirect.ts +93 -0
- package/src/route-definition.ts +1 -1428
- package/src/route-map-builder.ts +211 -123
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +59 -8
- package/src/router/content-negotiation.ts +116 -0
- package/src/router/debug-manifest.ts +72 -0
- package/src/router/error-handling.ts +9 -9
- package/src/router/find-match.ts +158 -0
- package/src/router/handler-context.ts +374 -81
- package/src/router/intercept-resolution.ts +395 -0
- package/src/router/lazy-includes.ts +234 -0
- package/src/router/loader-resolution.ts +215 -122
- package/src/router/logging.ts +248 -0
- package/src/router/manifest.ts +148 -35
- package/src/router/match-api.ts +620 -0
- package/src/router/match-context.ts +5 -3
- package/src/router/match-handlers.ts +440 -0
- package/src/router/match-middleware/background-revalidation.ts +80 -93
- package/src/router/match-middleware/cache-lookup.ts +382 -9
- package/src/router/match-middleware/cache-store.ts +51 -22
- package/src/router/match-middleware/intercept-resolution.ts +55 -17
- package/src/router/match-middleware/segment-resolution.ts +24 -6
- package/src/router/match-pipelines.ts +10 -45
- package/src/router/match-result.ts +34 -28
- package/src/router/metrics.ts +235 -15
- package/src/router/middleware-cookies.ts +55 -0
- package/src/router/middleware-types.ts +222 -0
- package/src/router/middleware.ts +324 -367
- package/src/router/pattern-matching.ts +211 -43
- package/src/router/prerender-match.ts +402 -0
- package/src/router/preview-match.ts +170 -0
- package/src/router/revalidation.ts +137 -38
- package/src/router/router-context.ts +36 -21
- package/src/router/router-interfaces.ts +452 -0
- package/src/router/router-options.ts +592 -0
- package/src/router/router-registry.ts +24 -0
- package/src/router/segment-resolution/fresh.ts +570 -0
- package/src/router/segment-resolution/helpers.ts +263 -0
- package/src/router/segment-resolution/loader-cache.ts +198 -0
- package/src/router/segment-resolution/revalidation.ts +1241 -0
- package/src/router/segment-resolution/static-store.ts +67 -0
- package/src/router/segment-resolution.ts +21 -0
- package/src/router/segment-wrappers.ts +289 -0
- package/src/router/telemetry-otel.ts +299 -0
- package/src/router/telemetry.ts +300 -0
- package/src/router/timeout.ts +148 -0
- package/src/router/trie-matching.ts +239 -0
- package/src/router/types.ts +77 -3
- package/src/router.ts +692 -4257
- package/src/rsc/handler-context.ts +45 -0
- package/src/rsc/handler.ts +764 -754
- package/src/rsc/helpers.ts +140 -6
- package/src/rsc/index.ts +0 -20
- package/src/rsc/loader-fetch.ts +209 -0
- package/src/rsc/manifest-init.ts +86 -0
- package/src/rsc/nonce.ts +14 -0
- package/src/rsc/origin-guard.ts +141 -0
- package/src/rsc/progressive-enhancement.ts +379 -0
- package/src/rsc/response-error.ts +37 -0
- package/src/rsc/response-route-handler.ts +347 -0
- package/src/rsc/rsc-rendering.ts +235 -0
- package/src/rsc/runtime-warnings.ts +42 -0
- package/src/rsc/server-action.ts +348 -0
- package/src/rsc/ssr-setup.ts +128 -0
- package/src/rsc/types.ts +38 -11
- package/src/search-params.ts +230 -0
- package/src/segment-system.tsx +25 -13
- package/src/server/context.ts +182 -51
- package/src/server/cookie-store.ts +190 -0
- package/src/server/fetchable-loader-store.ts +37 -0
- package/src/server/handle-store.ts +94 -15
- package/src/server/loader-registry.ts +15 -56
- package/src/server/request-context.ts +430 -70
- package/src/server.ts +35 -130
- package/src/ssr/index.tsx +100 -31
- package/src/static-handler.ts +114 -0
- package/src/theme/ThemeProvider.tsx +21 -15
- package/src/theme/ThemeScript.tsx +5 -5
- package/src/theme/constants.ts +5 -2
- package/src/theme/index.ts +4 -14
- package/src/theme/theme-context.ts +4 -30
- package/src/theme/theme-script.ts +21 -18
- package/src/types/boundaries.ts +158 -0
- package/src/types/cache-types.ts +198 -0
- package/src/types/error-types.ts +192 -0
- package/src/types/global-namespace.ts +100 -0
- package/src/types/handler-context.ts +687 -0
- package/src/types/index.ts +88 -0
- package/src/types/loader-types.ts +183 -0
- package/src/types/route-config.ts +170 -0
- package/src/types/route-entry.ts +102 -0
- package/src/types/segments.ts +148 -0
- package/src/types.ts +1 -1623
- package/src/urls/include-helper.ts +197 -0
- package/src/urls/index.ts +53 -0
- package/src/urls/path-helper-types.ts +339 -0
- package/src/urls/path-helper.ts +329 -0
- package/src/urls/pattern-types.ts +95 -0
- package/src/urls/response-types.ts +106 -0
- package/src/urls/type-extraction.ts +372 -0
- package/src/urls/urls-function.ts +98 -0
- package/src/urls.ts +1 -802
- package/src/use-loader.tsx +85 -77
- package/src/vite/discovery/bundle-postprocess.ts +184 -0
- package/src/vite/discovery/discover-routers.ts +344 -0
- package/src/vite/discovery/prerender-collection.ts +385 -0
- package/src/vite/discovery/route-types-writer.ts +258 -0
- package/src/vite/discovery/self-gen-tracking.ts +47 -0
- package/src/vite/discovery/state.ts +110 -0
- package/src/vite/discovery/virtual-module-codegen.ts +203 -0
- package/src/vite/index.ts +11 -1133
- package/src/vite/plugin-types.ts +131 -0
- package/src/vite/plugins/cjs-to-esm.ts +93 -0
- package/src/vite/plugins/client-ref-dedup.ts +115 -0
- package/src/vite/plugins/client-ref-hashing.ts +105 -0
- package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -51
- package/src/vite/plugins/expose-id-utils.ts +287 -0
- package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
- package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
- package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
- package/src/vite/plugins/expose-ids/types.ts +45 -0
- package/src/vite/plugins/expose-internal-ids.ts +569 -0
- package/src/vite/plugins/refresh-cmd.ts +65 -0
- package/src/vite/plugins/use-cache-transform.ts +323 -0
- package/src/vite/plugins/version-injector.ts +83 -0
- package/src/vite/plugins/version-plugin.ts +254 -0
- package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
- package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
- package/src/vite/rango.ts +510 -0
- package/src/vite/router-discovery.ts +785 -0
- package/src/vite/utils/ast-handler-extract.ts +517 -0
- package/src/vite/utils/banner.ts +36 -0
- package/src/vite/utils/bundle-analysis.ts +137 -0
- package/src/vite/utils/manifest-utils.ts +70 -0
- package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
- package/src/vite/utils/prerender-utils.ts +189 -0
- package/src/vite/utils/shared-utils.ts +169 -0
- package/CLAUDE.md +0 -43
- package/src/browser/lru-cache.ts +0 -69
- package/src/browser/request-controller.ts +0 -164
- package/src/cache/memory-store.ts +0 -253
- package/src/href-context.ts +0 -33
- package/src/href.ts +0 -255
- package/src/server/route-manifest-cache.ts +0 -173
- package/src/vite/expose-handle-id.ts +0 -209
- package/src/vite/expose-loader-id.ts +0 -426
- package/src/vite/expose-location-state-id.ts +0 -177
- /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
|
@@ -2,6 +2,7 @@
|
|
|
2
2
|
|
|
3
3
|
import { useContext, useState, useEffect, useRef } from "react";
|
|
4
4
|
import { NavigationStoreContext } from "./context.js";
|
|
5
|
+
import { shallowEqual } from "./shallow-equal.js";
|
|
5
6
|
|
|
6
7
|
/**
|
|
7
8
|
* Segments state returned by useSegments hook
|
|
@@ -15,65 +16,6 @@ export interface SegmentsState {
|
|
|
15
16
|
location: URL;
|
|
16
17
|
}
|
|
17
18
|
|
|
18
|
-
/**
|
|
19
|
-
* SSR module-level state.
|
|
20
|
-
* Populated by initSegmentsSync before React renders.
|
|
21
|
-
* Used by useState initializer during SSR.
|
|
22
|
-
*/
|
|
23
|
-
let ssrSegmentOrder: string[] = [];
|
|
24
|
-
let ssrPathname: string = "/";
|
|
25
|
-
|
|
26
|
-
/**
|
|
27
|
-
* Filter segment IDs to only include routes and layouts.
|
|
28
|
-
* Excludes parallels (contain .@) and loaders (contain D followed by digit).
|
|
29
|
-
*/
|
|
30
|
-
function filterSegmentOrder(matched: string[]): string[] {
|
|
31
|
-
return matched.filter((id) => {
|
|
32
|
-
if (id.includes(".@")) return false;
|
|
33
|
-
if (/D\d+\./.test(id)) return false;
|
|
34
|
-
return true;
|
|
35
|
-
});
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
/**
|
|
39
|
-
* Initialize segments data synchronously for SSR.
|
|
40
|
-
* Called before rendering to populate state for useState initializer.
|
|
41
|
-
*
|
|
42
|
-
* @param matched - Segment order from RSC metadata
|
|
43
|
-
* @param pathname - Current pathname
|
|
44
|
-
*/
|
|
45
|
-
export function initSegmentsSync(matched?: string[], pathname?: string): void {
|
|
46
|
-
ssrSegmentOrder = filterSegmentOrder(matched ?? []);
|
|
47
|
-
ssrPathname = pathname ?? "/";
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
/**
|
|
51
|
-
* Shallow equality check for selector results
|
|
52
|
-
*/
|
|
53
|
-
function shallowEqual<T>(a: T, b: T): boolean {
|
|
54
|
-
if (Object.is(a, b)) return true;
|
|
55
|
-
if (
|
|
56
|
-
typeof a !== "object" ||
|
|
57
|
-
a === null ||
|
|
58
|
-
typeof b !== "object" ||
|
|
59
|
-
b === null
|
|
60
|
-
) {
|
|
61
|
-
return false;
|
|
62
|
-
}
|
|
63
|
-
const keysA = Object.keys(a);
|
|
64
|
-
const keysB = Object.keys(b);
|
|
65
|
-
if (keysA.length !== keysB.length) return false;
|
|
66
|
-
for (const key of keysA) {
|
|
67
|
-
if (
|
|
68
|
-
!Object.hasOwn(b, key) ||
|
|
69
|
-
!Object.is((a as any)[key], (b as any)[key])
|
|
70
|
-
) {
|
|
71
|
-
return false;
|
|
72
|
-
}
|
|
73
|
-
}
|
|
74
|
-
return true;
|
|
75
|
-
}
|
|
76
|
-
|
|
77
19
|
/**
|
|
78
20
|
* Parse pathname into path segments
|
|
79
21
|
* /shop/products/123 → ["shop", "products", "123"]
|
|
@@ -87,7 +29,7 @@ function parsePathname(pathname: string): string[] {
|
|
|
87
29
|
*/
|
|
88
30
|
function buildSegmentsState(
|
|
89
31
|
location: URL,
|
|
90
|
-
segmentOrder: string[]
|
|
32
|
+
segmentOrder: string[],
|
|
91
33
|
): SegmentsState {
|
|
92
34
|
return {
|
|
93
35
|
path: parsePathname(location.pathname),
|
|
@@ -96,18 +38,6 @@ function buildSegmentsState(
|
|
|
96
38
|
};
|
|
97
39
|
}
|
|
98
40
|
|
|
99
|
-
/**
|
|
100
|
-
* Build SSR state from module-level variables
|
|
101
|
-
*/
|
|
102
|
-
function buildSsrState(): SegmentsState {
|
|
103
|
-
const location = new URL(ssrPathname, "http://localhost");
|
|
104
|
-
return {
|
|
105
|
-
path: parsePathname(ssrPathname),
|
|
106
|
-
segmentIds: ssrSegmentOrder,
|
|
107
|
-
location,
|
|
108
|
-
};
|
|
109
|
-
}
|
|
110
|
-
|
|
111
41
|
/**
|
|
112
42
|
* Hook to access current route segments with optional selector for performance
|
|
113
43
|
*
|
|
@@ -127,62 +57,115 @@ function buildSsrState(): SegmentsState {
|
|
|
127
57
|
export function useSegments(): SegmentsState;
|
|
128
58
|
export function useSegments<T>(selector: (state: SegmentsState) => T): T;
|
|
129
59
|
export function useSegments<T>(
|
|
130
|
-
selector?: (state: SegmentsState) => T
|
|
60
|
+
selector?: (state: SegmentsState) => T,
|
|
131
61
|
): T | SegmentsState {
|
|
132
62
|
const ctx = useContext(NavigationStoreContext);
|
|
133
63
|
|
|
134
|
-
// Build initial state from
|
|
64
|
+
// Build initial state from event controller when context exists.
|
|
65
|
+
// Inlined rather than calling recompute() because the segmentsCache ref
|
|
66
|
+
// is not yet initialized during the useState initializer.
|
|
135
67
|
const [state, setState] = useState<T | SegmentsState>(() => {
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
const
|
|
139
|
-
return selector ? selector(
|
|
68
|
+
if (!ctx) {
|
|
69
|
+
const fallbackLocation = new URL("/", "http://localhost");
|
|
70
|
+
const fallbackState = buildSegmentsState(fallbackLocation, []);
|
|
71
|
+
return selector ? selector(fallbackState) : fallbackState;
|
|
140
72
|
}
|
|
141
|
-
|
|
142
|
-
const navState = ctx.eventController.getState();
|
|
73
|
+
const location = ctx.eventController.getLocation();
|
|
143
74
|
const handleState = ctx.eventController.getHandleState();
|
|
144
75
|
const segmentsState = buildSegmentsState(
|
|
145
|
-
|
|
146
|
-
handleState.segmentOrder
|
|
76
|
+
location as URL,
|
|
77
|
+
handleState.segmentOrder,
|
|
147
78
|
);
|
|
148
79
|
return selector ? selector(segmentsState) : segmentsState;
|
|
149
80
|
});
|
|
150
81
|
|
|
151
82
|
const prevState = useRef(state);
|
|
83
|
+
const selectorRef = useRef(selector);
|
|
84
|
+
selectorRef.current = selector;
|
|
85
|
+
|
|
86
|
+
// Track selector identity to detect when the selector function changes.
|
|
87
|
+
// Only then do we eagerly recompute during render to avoid staleness.
|
|
88
|
+
// Without this guard, no-selector mode causes infinite re-renders because
|
|
89
|
+
// buildSegmentsState creates fresh arrays that fail Object.is checks.
|
|
90
|
+
const prevSelectorIdentity = useRef(selector);
|
|
91
|
+
|
|
92
|
+
// Cache SegmentsState to stabilize nested references (path, segmentIds
|
|
93
|
+
// arrays) so selectors returning composite values don't cause spurious
|
|
94
|
+
// render-time setState calls.
|
|
95
|
+
const segmentsCache = useRef<{
|
|
96
|
+
location: URL;
|
|
97
|
+
segmentOrder: string[];
|
|
98
|
+
state: SegmentsState;
|
|
99
|
+
} | null>(null);
|
|
100
|
+
|
|
101
|
+
// Recompute selected value from current store state and apply selector.
|
|
102
|
+
// Shared by the render-time eager check and the subscription callback.
|
|
103
|
+
function recompute(
|
|
104
|
+
sel: ((state: SegmentsState) => T) | undefined,
|
|
105
|
+
): T | SegmentsState {
|
|
106
|
+
const location = ctx!.eventController.getLocation();
|
|
107
|
+
const handleState = ctx!.eventController.getHandleState();
|
|
108
|
+
|
|
109
|
+
// Reuse cached state when inputs haven't changed by reference,
|
|
110
|
+
// keeping array/object references stable for composite selectors.
|
|
111
|
+
const cache = segmentsCache.current;
|
|
112
|
+
let segmentsState: SegmentsState;
|
|
113
|
+
if (
|
|
114
|
+
cache &&
|
|
115
|
+
cache.location === location &&
|
|
116
|
+
cache.segmentOrder === handleState.segmentOrder
|
|
117
|
+
) {
|
|
118
|
+
segmentsState = cache.state;
|
|
119
|
+
} else {
|
|
120
|
+
segmentsState = buildSegmentsState(
|
|
121
|
+
location as URL,
|
|
122
|
+
handleState.segmentOrder,
|
|
123
|
+
);
|
|
124
|
+
segmentsCache.current = {
|
|
125
|
+
location: location as URL,
|
|
126
|
+
segmentOrder: handleState.segmentOrder,
|
|
127
|
+
state: segmentsState,
|
|
128
|
+
};
|
|
129
|
+
}
|
|
130
|
+
return sel ? sel(segmentsState) : segmentsState;
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
if (ctx && selector !== prevSelectorIdentity.current) {
|
|
134
|
+
prevSelectorIdentity.current = selector;
|
|
135
|
+
const nextSelected = recompute(selector);
|
|
136
|
+
if (!shallowEqual(nextSelected, prevState.current)) {
|
|
137
|
+
prevState.current = nextSelected;
|
|
138
|
+
setState(nextSelected);
|
|
139
|
+
}
|
|
140
|
+
}
|
|
152
141
|
|
|
153
|
-
// Subscribe to
|
|
142
|
+
// Subscribe to store changes. The eager block above handles selector
|
|
143
|
+
// changes and SSR drift, so no initial updateState() call is needed.
|
|
154
144
|
useEffect(() => {
|
|
155
145
|
if (!ctx) {
|
|
156
146
|
return;
|
|
157
147
|
}
|
|
158
148
|
|
|
159
149
|
const updateState = () => {
|
|
160
|
-
const
|
|
161
|
-
const handleState = ctx.eventController.getHandleState();
|
|
162
|
-
const segmentsState = buildSegmentsState(
|
|
163
|
-
navState.location as URL,
|
|
164
|
-
handleState.segmentOrder
|
|
165
|
-
);
|
|
166
|
-
const nextSelected = selector ? selector(segmentsState) : segmentsState;
|
|
167
|
-
|
|
150
|
+
const nextSelected = recompute(selectorRef.current);
|
|
168
151
|
if (!shallowEqual(nextSelected, prevState.current)) {
|
|
169
152
|
prevState.current = nextSelected;
|
|
170
153
|
setState(nextSelected);
|
|
171
154
|
}
|
|
172
155
|
};
|
|
173
156
|
|
|
174
|
-
// Initial update in case SSR state differs from client state
|
|
175
|
-
updateState();
|
|
176
|
-
|
|
177
|
-
// Subscribe to both state sources
|
|
178
157
|
const unsubscribeNav = ctx.eventController.subscribe(updateState);
|
|
179
|
-
const unsubscribeHandles =
|
|
158
|
+
const unsubscribeHandles =
|
|
159
|
+
ctx.eventController.subscribeToHandles(updateState);
|
|
180
160
|
|
|
181
161
|
return () => {
|
|
182
162
|
unsubscribeNav();
|
|
183
163
|
unsubscribeHandles();
|
|
184
164
|
};
|
|
185
|
-
|
|
165
|
+
// Stable subscription: selector changes are handled via selectorRef,
|
|
166
|
+
// state comparison uses prevState ref. No re-subscribe needed.
|
|
167
|
+
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
168
|
+
}, []);
|
|
186
169
|
|
|
187
170
|
return state as T | SegmentsState;
|
|
188
171
|
}
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
import { validateRedirectOrigin } from "./validate-redirect-origin.js";
|
|
2
|
+
|
|
3
|
+
type HeaderResult = { url: string } | "blocked" | null;
|
|
4
|
+
|
|
5
|
+
/**
|
|
6
|
+
* Extract and validate an RSC response header URL (X-RSC-Reload, X-RSC-Redirect).
|
|
7
|
+
* Returns { url } if valid, "blocked" if present but invalid origin, null if absent.
|
|
8
|
+
*/
|
|
9
|
+
export function extractRscHeaderUrl(
|
|
10
|
+
response: Response,
|
|
11
|
+
header: string,
|
|
12
|
+
): HeaderResult {
|
|
13
|
+
const raw = response.headers.get(header);
|
|
14
|
+
if (!raw) return null;
|
|
15
|
+
const url = validateRedirectOrigin(raw, window.location.origin);
|
|
16
|
+
return url ? { url } : "blocked";
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
/**
|
|
20
|
+
* Empty 200 response that won't choke Flight parsing.
|
|
21
|
+
* Used when a header URL is blocked by origin validation.
|
|
22
|
+
*/
|
|
23
|
+
export function emptyResponse(): Response {
|
|
24
|
+
return new Response(null, { status: 200 });
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/**
|
|
28
|
+
* Tee a response body for RSC parsing and stream completion tracking.
|
|
29
|
+
* Returns a new Response with one branch; the other is consumed to detect
|
|
30
|
+
* end-of-stream, calling onComplete when done.
|
|
31
|
+
*
|
|
32
|
+
* If the response has no body, onComplete fires synchronously.
|
|
33
|
+
* If signal is provided, an abort cancels the tracking reader.
|
|
34
|
+
*/
|
|
35
|
+
export function teeWithCompletion(
|
|
36
|
+
response: Response,
|
|
37
|
+
onComplete: () => void,
|
|
38
|
+
signal?: AbortSignal,
|
|
39
|
+
): Response {
|
|
40
|
+
if (!response.body) {
|
|
41
|
+
onComplete();
|
|
42
|
+
return response;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
const [rscStream, trackingStream] = response.body.tee();
|
|
46
|
+
|
|
47
|
+
(async () => {
|
|
48
|
+
const reader = trackingStream.getReader();
|
|
49
|
+
const onAbort = signal ? reader.cancel.bind(reader) : undefined;
|
|
50
|
+
if (onAbort) signal!.addEventListener("abort", onAbort, { once: true });
|
|
51
|
+
try {
|
|
52
|
+
while (true) {
|
|
53
|
+
const { done } = await reader.read();
|
|
54
|
+
if (done) break;
|
|
55
|
+
}
|
|
56
|
+
} finally {
|
|
57
|
+
if (onAbort) signal!.removeEventListener("abort", onAbort);
|
|
58
|
+
reader.releaseLock();
|
|
59
|
+
onComplete();
|
|
60
|
+
}
|
|
61
|
+
})().catch((error) => {
|
|
62
|
+
if (!signal?.aborted) {
|
|
63
|
+
console.error("[Browser] Error reading tracking stream:", error);
|
|
64
|
+
}
|
|
65
|
+
onComplete();
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
return new Response(rscStream, {
|
|
69
|
+
headers: response.headers,
|
|
70
|
+
status: response.status,
|
|
71
|
+
statusText: response.statusText,
|
|
72
|
+
});
|
|
73
|
+
}
|
|
@@ -11,8 +11,7 @@ import { createEventController } from "./event-controller.js";
|
|
|
11
11
|
import { createNavigationClient } from "./navigation-client.js";
|
|
12
12
|
import { createServerActionBridge } from "./server-action-bridge.js";
|
|
13
13
|
import { createNavigationBridge } from "./navigation-bridge.js";
|
|
14
|
-
import { NavigationProvider
|
|
15
|
-
import { initThemeConfigSync } from "../theme/theme-context.js";
|
|
14
|
+
import { NavigationProvider } from "./react/index.js";
|
|
16
15
|
import type {
|
|
17
16
|
RscPayload,
|
|
18
17
|
RscBrowserDependencies,
|
|
@@ -22,6 +21,12 @@ import type {
|
|
|
22
21
|
} from "./types.js";
|
|
23
22
|
import type { EventController } from "./event-controller.js";
|
|
24
23
|
import type { ResolvedThemeConfig, Theme } from "../theme/types.js";
|
|
24
|
+
import { initRangoState } from "./rango-state.js";
|
|
25
|
+
import { initPrefetchCache } from "./prefetch/cache.js";
|
|
26
|
+
import {
|
|
27
|
+
isInterceptSegment,
|
|
28
|
+
splitInterceptSegments,
|
|
29
|
+
} from "./intercept-utils.js";
|
|
25
30
|
|
|
26
31
|
// Vite HMR types are provided by vite/client
|
|
27
32
|
|
|
@@ -106,6 +111,8 @@ export interface BrowserAppContext {
|
|
|
106
111
|
initialTheme?: Theme;
|
|
107
112
|
/** Whether connection warmup is enabled */
|
|
108
113
|
warmupEnabled?: boolean;
|
|
114
|
+
/** App version for prefetch version mismatch detection */
|
|
115
|
+
version?: string;
|
|
109
116
|
}
|
|
110
117
|
|
|
111
118
|
// Module-level state for the initialized app
|
|
@@ -121,9 +128,16 @@ let browserAppContext: BrowserAppContext | null = null;
|
|
|
121
128
|
* - Configures HMR support
|
|
122
129
|
*/
|
|
123
130
|
export async function initBrowserApp(
|
|
124
|
-
options: InitBrowserAppOptions
|
|
131
|
+
options: InitBrowserAppOptions,
|
|
125
132
|
): Promise<BrowserAppContext> {
|
|
126
|
-
const {
|
|
133
|
+
const {
|
|
134
|
+
rscStream,
|
|
135
|
+
deps,
|
|
136
|
+
storeOptions,
|
|
137
|
+
linkInterception = true,
|
|
138
|
+
themeConfig,
|
|
139
|
+
initialTheme,
|
|
140
|
+
} = options;
|
|
127
141
|
|
|
128
142
|
// Load initial payload from SSR-injected __FLIGHT_DATA__
|
|
129
143
|
const initialPayload =
|
|
@@ -131,8 +145,10 @@ export async function initBrowserApp(
|
|
|
131
145
|
|
|
132
146
|
// Extract themeConfig and initialTheme from payload if not explicitly provided
|
|
133
147
|
// This allows virtual entries to work without importing the router
|
|
134
|
-
const effectiveThemeConfig =
|
|
135
|
-
|
|
148
|
+
const effectiveThemeConfig =
|
|
149
|
+
themeConfig ?? initialPayload.metadata?.themeConfig ?? null;
|
|
150
|
+
const effectiveInitialTheme =
|
|
151
|
+
initialTheme ?? initialPayload.metadata?.initialTheme;
|
|
136
152
|
|
|
137
153
|
// Get initial segments and compute history key from current URL
|
|
138
154
|
const initialSegments = (initialPayload.metadata?.segments ??
|
|
@@ -153,15 +169,12 @@ export async function initBrowserApp(
|
|
|
153
169
|
initialLocation: new URL(window.location.href),
|
|
154
170
|
});
|
|
155
171
|
|
|
156
|
-
// Initialize segments state BEFORE hydration to avoid mismatch
|
|
157
|
-
initSegmentsSync(initialPayload.metadata?.matched, initialPayload.metadata?.pathname);
|
|
158
|
-
|
|
159
|
-
// Initialize theme config for MetaTags (must match SSR state)
|
|
160
|
-
initThemeConfigSync(effectiveThemeConfig);
|
|
161
|
-
|
|
162
172
|
// Initialize event controller with segment order (even without handles)
|
|
163
173
|
eventController.setHandleData({}, initialPayload.metadata?.matched);
|
|
164
174
|
|
|
175
|
+
// Initialize route params
|
|
176
|
+
eventController.setParams(initialPayload.metadata?.params ?? {});
|
|
177
|
+
|
|
165
178
|
// Initialize handle data from initial payload BEFORE hydration
|
|
166
179
|
// This ensures useHandle returns correct data during hydration to avoid mismatch
|
|
167
180
|
// The handles property is an async generator that yields on each push
|
|
@@ -171,16 +184,17 @@ export async function initBrowserApp(
|
|
|
171
184
|
for await (const handleData of handlesGenerator) {
|
|
172
185
|
lastHandleData = handleData;
|
|
173
186
|
}
|
|
174
|
-
// Initialize
|
|
175
|
-
eventController.setHandleData(
|
|
176
|
-
|
|
187
|
+
// Initialize event controller with initial handle state before hydration.
|
|
188
|
+
eventController.setHandleData(
|
|
189
|
+
lastHandleData,
|
|
190
|
+
initialPayload.metadata?.matched,
|
|
191
|
+
);
|
|
177
192
|
|
|
178
193
|
// Update the initial cache entry with the processed handleData
|
|
179
194
|
// The cache entry was created by createNavigationStore but without handleData
|
|
180
195
|
store.updateCacheHandleData(initialHistoryKey, lastHandleData);
|
|
181
196
|
}
|
|
182
197
|
|
|
183
|
-
|
|
184
198
|
// Create composable utilities
|
|
185
199
|
const client = createNavigationClient(deps);
|
|
186
200
|
|
|
@@ -188,12 +202,27 @@ export async function initBrowserApp(
|
|
|
188
202
|
const rootLayout = initialPayload.metadata?.rootLayout;
|
|
189
203
|
const version = initialPayload.metadata?.version;
|
|
190
204
|
|
|
205
|
+
// Initialize the localStorage state key for cache invalidation.
|
|
206
|
+
// Uses the build version so a new deploy automatically busts all cached prefetches.
|
|
207
|
+
initRangoState(version ?? "0");
|
|
208
|
+
|
|
209
|
+
// Initialize the in-memory prefetch cache TTL from server config.
|
|
210
|
+
// A value of 0 disables the cache; undefined falls back to the module default.
|
|
211
|
+
const prefetchCacheTTL = initialPayload.metadata?.prefetchCacheTTL;
|
|
212
|
+
if (prefetchCacheTTL !== undefined) {
|
|
213
|
+
initPrefetchCache(prefetchCacheTTL);
|
|
214
|
+
}
|
|
215
|
+
|
|
191
216
|
// Create a bound renderSegments that includes rootLayout
|
|
192
217
|
const renderSegments = (
|
|
193
218
|
segments: ResolvedSegment[],
|
|
194
|
-
options?: RenderSegmentsOptions
|
|
219
|
+
options?: RenderSegmentsOptions,
|
|
195
220
|
) => baseRenderSegments(segments, { ...options, rootLayout });
|
|
196
221
|
|
|
222
|
+
// Lazy reference for navigation bridge — the action bridge is created first
|
|
223
|
+
// but may need to trigger SPA navigation for action redirects.
|
|
224
|
+
let navigateFn: ((url: string, options?: any) => Promise<void>) | null = null;
|
|
225
|
+
|
|
197
226
|
// Setup server action bridge
|
|
198
227
|
const actionBridge = createServerActionBridge({
|
|
199
228
|
store,
|
|
@@ -203,6 +232,13 @@ export async function initBrowserApp(
|
|
|
203
232
|
onUpdate: (update) => store.emitUpdate(update),
|
|
204
233
|
renderSegments,
|
|
205
234
|
version,
|
|
235
|
+
onNavigate: (url, options) => {
|
|
236
|
+
if (!navigateFn) {
|
|
237
|
+
window.location.href = url;
|
|
238
|
+
return Promise.resolve();
|
|
239
|
+
}
|
|
240
|
+
return navigateFn(url, options);
|
|
241
|
+
},
|
|
206
242
|
});
|
|
207
243
|
actionBridge.register();
|
|
208
244
|
|
|
@@ -216,6 +252,9 @@ export async function initBrowserApp(
|
|
|
216
252
|
version,
|
|
217
253
|
});
|
|
218
254
|
|
|
255
|
+
// Connect action redirect → navigation bridge (now that both are initialized)
|
|
256
|
+
navigateFn = (url, options) => navigationBridge.navigate(url, options);
|
|
257
|
+
|
|
219
258
|
// Optionally enable global link interception
|
|
220
259
|
if (linkInterception) {
|
|
221
260
|
navigationBridge.registerLinkInterception();
|
|
@@ -234,37 +273,61 @@ export async function initBrowserApp(
|
|
|
234
273
|
});
|
|
235
274
|
const streamingToken = handle.startStreaming();
|
|
236
275
|
|
|
276
|
+
const interceptSourceUrl = store.getInterceptSourceUrl();
|
|
277
|
+
|
|
237
278
|
try {
|
|
238
279
|
const { payload, streamComplete } = await client.fetchPartial({
|
|
239
280
|
targetUrl: window.location.href,
|
|
240
281
|
segmentIds: [],
|
|
241
282
|
previousUrl: store.getSegmentState().currentUrl,
|
|
283
|
+
interceptSourceUrl: interceptSourceUrl || undefined,
|
|
284
|
+
hmr: true,
|
|
242
285
|
});
|
|
243
286
|
|
|
244
287
|
if (payload.metadata?.isPartial) {
|
|
245
288
|
const segments = payload.metadata.segments || [];
|
|
246
289
|
const matched = payload.metadata.matched || [];
|
|
247
290
|
|
|
291
|
+
// Derive intercept state from the returned payload, not the
|
|
292
|
+
// pre-fetch store snapshot. If the HMR edit removed intercept
|
|
293
|
+
// behavior, the response won't contain intercept segments.
|
|
294
|
+
const responseIsIntercept = segments.some(isInterceptSegment);
|
|
295
|
+
|
|
296
|
+
// Sync store intercept state with what the server returned
|
|
297
|
+
if (!responseIsIntercept && interceptSourceUrl) {
|
|
298
|
+
store.setInterceptSourceUrl(null);
|
|
299
|
+
}
|
|
300
|
+
|
|
248
301
|
store.setSegmentIds(matched);
|
|
249
302
|
store.setCurrentUrl(window.location.href);
|
|
250
303
|
|
|
251
|
-
const historyKey = generateHistoryKey(window.location.href
|
|
304
|
+
const historyKey = generateHistoryKey(window.location.href, {
|
|
305
|
+
intercept: responseIsIntercept,
|
|
306
|
+
});
|
|
252
307
|
store.setHistoryKey(historyKey);
|
|
253
308
|
const currentHandleData = eventController.getHandleState().data;
|
|
254
|
-
store.cacheSegmentsForHistory(
|
|
309
|
+
store.cacheSegmentsForHistory(
|
|
310
|
+
historyKey,
|
|
311
|
+
segments,
|
|
312
|
+
currentHandleData,
|
|
313
|
+
);
|
|
255
314
|
|
|
315
|
+
const { main, intercept } = splitInterceptSegments(segments);
|
|
256
316
|
store.emitUpdate({
|
|
257
|
-
root: renderSegments(
|
|
317
|
+
root: renderSegments(main, {
|
|
318
|
+
interceptSegments: intercept.length > 0 ? intercept : undefined,
|
|
319
|
+
}),
|
|
258
320
|
metadata: payload.metadata,
|
|
259
321
|
});
|
|
260
322
|
}
|
|
261
323
|
|
|
262
324
|
await streamComplete;
|
|
325
|
+
handle.complete(new URL(window.location.href));
|
|
326
|
+
console.log("[RSCRouter] HMR: RSC stream complete");
|
|
263
327
|
} finally {
|
|
264
328
|
streamingToken.end();
|
|
329
|
+
handle[Symbol.dispose]();
|
|
265
330
|
}
|
|
266
|
-
handle.complete(new URL(window.location.href));
|
|
267
|
-
console.log("[RSCRouter] HMR: RSC stream complete");
|
|
268
331
|
});
|
|
269
332
|
}
|
|
270
333
|
|
|
@@ -278,6 +341,7 @@ export async function initBrowserApp(
|
|
|
278
341
|
themeConfig: effectiveThemeConfig,
|
|
279
342
|
initialTheme: effectiveInitialTheme,
|
|
280
343
|
warmupEnabled: initialPayload.metadata?.warmupEnabled ?? true,
|
|
344
|
+
version,
|
|
281
345
|
};
|
|
282
346
|
browserAppContext = context;
|
|
283
347
|
|
|
@@ -290,7 +354,7 @@ export async function initBrowserApp(
|
|
|
290
354
|
export function getBrowserAppContext(): BrowserAppContext {
|
|
291
355
|
if (!browserAppContext) {
|
|
292
356
|
throw new Error(
|
|
293
|
-
"RSCRouter: initBrowserApp() must be called before rendering RSCRouter"
|
|
357
|
+
"RSCRouter: initBrowserApp() must be called before rendering RSCRouter",
|
|
294
358
|
);
|
|
295
359
|
}
|
|
296
360
|
return browserAppContext;
|
|
@@ -333,18 +397,35 @@ export interface RSCRouterProps {}
|
|
|
333
397
|
* ```
|
|
334
398
|
*/
|
|
335
399
|
export function RSCRouter(_props: RSCRouterProps): React.ReactElement {
|
|
336
|
-
const {
|
|
337
|
-
|
|
400
|
+
const {
|
|
401
|
+
store,
|
|
402
|
+
eventController,
|
|
403
|
+
bridge,
|
|
404
|
+
initialPayload,
|
|
405
|
+
initialTree,
|
|
406
|
+
themeConfig,
|
|
407
|
+
initialTheme,
|
|
408
|
+
warmupEnabled,
|
|
409
|
+
version,
|
|
410
|
+
} = getBrowserAppContext();
|
|
411
|
+
|
|
412
|
+
// Signal that the React tree has hydrated. useEffect only fires after
|
|
413
|
+
// hydration completes, so this attribute is a stable readiness marker
|
|
414
|
+
// that does not depend on React internals like __reactFiber.
|
|
415
|
+
React.useEffect(() => {
|
|
416
|
+
document.documentElement.dataset.hydrated = "";
|
|
417
|
+
}, []);
|
|
338
418
|
|
|
339
419
|
return (
|
|
340
420
|
<NavigationProvider
|
|
341
421
|
store={store}
|
|
342
422
|
eventController={eventController}
|
|
343
|
-
initialPayload={{
|
|
423
|
+
initialPayload={{ root: initialTree, metadata: initialPayload.metadata! }}
|
|
344
424
|
bridge={bridge}
|
|
345
425
|
themeConfig={themeConfig}
|
|
346
426
|
initialTheme={initialTheme}
|
|
347
427
|
warmupEnabled={warmupEnabled}
|
|
428
|
+
version={version}
|
|
348
429
|
/>
|
|
349
430
|
);
|
|
350
431
|
}
|