@rangojs/router 0.0.0-experimental.31 → 0.0.0-experimental.3232cd17
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 +4 -0
- package/README.md +198 -44
- package/dist/bin/rango.js +287 -105
- package/dist/testing/vitest.js +82 -0
- package/dist/vite/index.js +3248 -1117
- package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/package.json +73 -21
- package/skills/api-client/SKILL.md +211 -0
- package/skills/breadcrumbs/SKILL.md +107 -1
- package/skills/bundle-analysis/SKILL.md +159 -0
- package/skills/cache-guide/SKILL.md +245 -21
- package/skills/caching/SKILL.md +302 -6
- package/skills/composability/SKILL.md +27 -2
- package/skills/css/SKILL.md +76 -0
- package/skills/document-cache/SKILL.md +78 -55
- package/skills/handler-use/SKILL.md +364 -0
- package/skills/hooks/SKILL.md +270 -30
- package/skills/host-router/SKILL.md +82 -22
- package/skills/i18n/SKILL.md +276 -0
- package/skills/intercept/SKILL.md +49 -5
- package/skills/layout/SKILL.md +35 -9
- package/skills/links/SKILL.md +249 -17
- package/skills/loader/SKILL.md +294 -30
- package/skills/middleware/SKILL.md +52 -13
- package/skills/migrate-nextjs/SKILL.md +584 -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 +203 -7
- package/skills/prerender/SKILL.md +123 -100
- package/skills/rango/SKILL.md +250 -22
- package/skills/react-compiler/SKILL.md +168 -0
- package/skills/response-routes/SKILL.md +122 -47
- package/skills/route/SKILL.md +97 -5
- package/skills/router-setup/SKILL.md +90 -5
- package/skills/server-actions/SKILL.md +775 -0
- package/skills/streams-and-websockets/SKILL.md +283 -0
- package/skills/tailwind/SKILL.md +27 -3
- package/skills/testing/SKILL.md +129 -0
- package/skills/testing/bindings.md +89 -0
- package/skills/testing/cache-prerender.md +124 -0
- package/skills/testing/client-components.md +122 -0
- package/skills/testing/e2e-parity.md +125 -0
- package/skills/testing/flight.md +92 -0
- package/skills/testing/handles.md +129 -0
- package/skills/testing/loader.md +128 -0
- package/skills/testing/middleware.md +99 -0
- package/skills/testing/render-handler.md +121 -0
- package/skills/testing/response-routes.md +95 -0
- package/skills/testing/reverse-and-types.md +84 -0
- package/skills/testing/server-actions.md +107 -0
- package/skills/testing/server-tree.md +128 -0
- package/skills/testing/setup.md +120 -0
- package/skills/typesafety/SKILL.md +329 -27
- package/skills/use-cache/SKILL.md +36 -5
- package/skills/view-transitions/SKILL.md +294 -0
- package/src/__augment-tests__/augment.ts +81 -0
- package/src/__augment-tests__/augmented.check.ts +116 -0
- package/src/__internal.ts +67 -40
- package/src/browser/action-coordinator.ts +53 -36
- package/src/browser/action-fence.ts +47 -0
- package/src/browser/app-shell.ts +39 -0
- package/src/browser/app-version.ts +14 -0
- package/src/browser/cookie-name.ts +140 -0
- package/src/browser/event-controller.ts +86 -147
- package/src/browser/history-state.ts +21 -0
- package/src/browser/index.ts +3 -3
- package/src/browser/invalidate-client-cache.ts +52 -0
- package/src/browser/link-interceptor.ts +4 -0
- package/src/browser/navigation-bridge.ts +148 -19
- package/src/browser/navigation-client.ts +187 -67
- package/src/browser/navigation-store-handle.ts +38 -0
- package/src/browser/navigation-store.ts +76 -67
- package/src/browser/navigation-transaction.ts +18 -66
- package/src/browser/partial-update.ts +123 -94
- package/src/browser/prefetch/cache.ts +214 -36
- package/src/browser/prefetch/fetch.ts +260 -38
- package/src/browser/prefetch/policy.ts +6 -0
- package/src/browser/prefetch/queue.ts +126 -20
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/rango-state.ts +158 -76
- package/src/browser/react/Link.tsx +93 -11
- package/src/browser/react/NavigationProvider.tsx +115 -34
- package/src/browser/react/ScrollRestoration.tsx +10 -6
- package/src/browser/react/context.ts +7 -2
- package/src/browser/react/filter-segment-order.ts +49 -7
- package/src/browser/react/index.ts +0 -48
- package/src/browser/react/location-state-shared.ts +166 -8
- package/src/browser/react/location-state.ts +39 -14
- package/src/browser/react/use-action.ts +6 -15
- package/src/browser/react/use-handle.ts +23 -69
- package/src/browser/react/use-link-status.ts +0 -4
- package/src/browser/react/use-navigation.ts +22 -5
- package/src/browser/react/use-params.ts +20 -10
- package/src/browser/react/use-reverse.ts +106 -0
- package/src/browser/react/use-router.ts +46 -11
- package/src/browser/react/use-search-params.ts +0 -5
- package/src/browser/react/use-segments.ts +11 -21
- package/src/browser/response-adapter.ts +52 -1
- package/src/browser/rsc-router.tsx +215 -76
- package/src/browser/scroll-restoration.ts +46 -39
- package/src/browser/segment-reconciler.ts +36 -9
- package/src/browser/segment-structure-assert.ts +2 -2
- package/src/browser/server-action-bridge.ts +176 -50
- package/src/browser/types.ts +95 -11
- package/src/browser/validate-redirect-origin.ts +43 -16
- 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 +8 -2
- package/src/build/prefix-tree-utils.ts +123 -0
- package/src/build/route-trie.ts +137 -32
- package/src/build/route-types/codegen.ts +4 -4
- package/src/build/route-types/include-resolution.ts +9 -2
- package/src/build/route-types/param-extraction.ts +6 -3
- package/src/build/route-types/per-module-writer.ts +7 -4
- package/src/build/route-types/router-processing.ts +278 -96
- 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-error.ts +104 -0
- package/src/cache/cache-policy.ts +68 -28
- package/src/cache/cache-runtime.ts +149 -43
- package/src/cache/cache-scope.ts +148 -81
- package/src/cache/cache-tag.ts +98 -0
- package/src/cache/cf/cf-cache-store.ts +2550 -93
- package/src/cache/cf/index.ts +11 -17
- package/src/cache/document-cache.ts +78 -27
- package/src/cache/handle-snapshot.ts +63 -0
- package/src/cache/index.ts +23 -20
- package/src/cache/memory-segment-store.ts +136 -37
- package/src/cache/profile-registry.ts +6 -30
- package/src/cache/read-through-swr.ts +41 -11
- package/src/cache/segment-codec.ts +0 -16
- package/src/cache/tag-invalidation.ts +230 -0
- package/src/cache/taint.ts +55 -0
- package/src/cache/types.ts +33 -100
- package/src/cache/vercel/index.ts +11 -0
- package/src/cache/vercel/vercel-cache-store.ts +799 -0
- package/src/client.rsc.tsx +6 -21
- package/src/client.tsx +108 -290
- package/src/component-utils.ts +19 -0
- package/src/context-var.ts +84 -2
- package/src/debug.ts +2 -2
- package/src/decode-loader-results.ts +36 -0
- package/src/defer.ts +196 -0
- package/src/deps/ssr.ts +0 -1
- package/src/errors.ts +30 -4
- package/src/handle.ts +70 -22
- package/src/handles/MetaTags.tsx +0 -14
- package/src/handles/breadcrumbs.ts +16 -5
- package/src/handles/meta.ts +0 -39
- package/src/host/cookie-handler.ts +0 -36
- package/src/host/errors.ts +0 -24
- package/src/host/index.ts +8 -2
- package/src/host/pattern-matcher.ts +7 -50
- package/src/host/router.ts +107 -99
- package/src/host/testing.ts +40 -27
- package/src/host/types.ts +37 -4
- package/src/host/utils.ts +1 -1
- package/src/href-client.ts +137 -22
- package/src/index.rsc.ts +52 -26
- package/src/index.ts +100 -38
- package/src/internal-debug.ts +2 -4
- package/src/loader-store.ts +500 -0
- package/src/loader.rsc.ts +20 -13
- package/src/loader.ts +12 -11
- package/src/missing-id-error.ts +68 -0
- package/src/network-error-thrower.tsx +1 -6
- package/src/outlet-context.ts +1 -1
- package/src/outlet-provider.tsx +1 -5
- package/src/prerender/param-hash.ts +10 -11
- package/src/prerender/store.ts +37 -41
- package/src/prerender.ts +198 -82
- package/src/redirect-origin.ts +100 -0
- package/src/response-utils.ts +37 -0
- package/src/reverse.ts +65 -15
- package/src/root-error-boundary.tsx +1 -19
- package/src/route-content-wrapper.tsx +7 -72
- package/src/route-definition/dsl-helpers.ts +437 -274
- package/src/route-definition/helper-factories.ts +29 -139
- package/src/route-definition/helpers-types.ts +113 -37
- package/src/route-definition/index.ts +3 -0
- package/src/route-definition/redirect.ts +52 -10
- package/src/route-definition/resolve-handler-use.ts +161 -0
- package/src/route-definition/use-item-types.ts +32 -0
- package/src/route-map-builder.ts +7 -17
- package/src/route-types.ts +37 -41
- package/src/router/basename.ts +14 -0
- package/src/router/content-negotiation.ts +108 -9
- package/src/router/error-handling.ts +13 -17
- package/src/router/find-match.ts +45 -22
- package/src/router/handler-context.ts +83 -41
- package/src/router/intercept-resolution.ts +25 -23
- package/src/router/lazy-includes.ts +19 -53
- package/src/router/loader-resolution.ts +213 -30
- package/src/router/logging.ts +5 -8
- package/src/router/manifest.ts +49 -45
- package/src/router/match-api.ts +121 -205
- package/src/router/match-context.ts +0 -22
- package/src/router/match-handlers.ts +58 -58
- package/src/router/match-middleware/background-revalidation.ts +27 -6
- package/src/router/match-middleware/cache-lookup.ts +205 -249
- package/src/router/match-middleware/cache-store.ts +45 -32
- package/src/router/match-middleware/intercept-resolution.ts +8 -28
- package/src/router/match-middleware/segment-resolution.ts +52 -18
- package/src/router/match-pipelines.ts +1 -42
- package/src/router/match-result.ts +104 -40
- package/src/router/metrics.ts +5 -34
- package/src/router/middleware-types.ts +13 -142
- package/src/router/middleware.ts +173 -143
- package/src/router/navigation-snapshot.ts +131 -0
- package/src/router/params-util.ts +23 -0
- package/src/router/pattern-matching.ts +109 -63
- package/src/router/prerender-match.ts +192 -54
- package/src/router/preview-match.ts +32 -102
- package/src/router/request-classification.ts +276 -0
- package/src/router/revalidation.ts +63 -55
- package/src/router/route-snapshot.ts +244 -0
- package/src/router/router-context.ts +6 -28
- package/src/router/router-interfaces.ts +100 -35
- package/src/router/router-options.ts +91 -11
- package/src/router/router-registry.ts +2 -5
- package/src/router/segment-resolution/fresh.ts +242 -75
- package/src/router/segment-resolution/helpers.ts +64 -25
- package/src/router/segment-resolution/loader-cache.ts +41 -37
- package/src/router/segment-resolution/revalidation.ts +456 -372
- package/src/router/segment-resolution/static-store.ts +19 -5
- package/src/router/segment-resolution/streamed-handler-telemetry.ts +52 -0
- package/src/router/segment-resolution/view-transition-default.ts +36 -0
- package/src/router/segment-resolution.ts +4 -1
- package/src/router/segment-wrappers.ts +2 -3
- package/src/router/state-cookie-name.ts +33 -0
- package/src/router/substitute-pattern-params.ts +56 -0
- package/src/router/telemetry-otel.ts +0 -20
- package/src/router/telemetry.ts +96 -19
- package/src/router/timeout.ts +0 -20
- package/src/router/trie-matching.ts +91 -46
- package/src/router/types.ts +10 -63
- package/src/router/url-params.ts +44 -0
- package/src/router.ts +134 -43
- package/src/rsc/handler-context.ts +3 -2
- package/src/rsc/handler.ts +492 -383
- package/src/rsc/helpers.ts +162 -46
- package/src/rsc/index.ts +1 -1
- package/src/rsc/json-route-result.ts +38 -0
- package/src/rsc/loader-fetch.ts +23 -3
- package/src/rsc/manifest-init.ts +33 -42
- package/src/rsc/origin-guard.ts +39 -25
- package/src/rsc/progressive-enhancement.ts +30 -3
- package/src/rsc/redirect-guard.ts +99 -0
- package/src/rsc/response-error.ts +79 -12
- package/src/rsc/response-route-handler.ts +90 -63
- package/src/rsc/rsc-rendering.ts +56 -54
- package/src/rsc/runtime-warnings.ts +23 -10
- package/src/rsc/server-action.ts +74 -67
- package/src/rsc/ssr-setup.ts +18 -2
- package/src/rsc/types.ts +25 -6
- package/src/runtime-env.ts +18 -0
- package/src/search-params.ts +4 -20
- package/src/segment-content-promise.ts +67 -0
- package/src/segment-loader-promise.ts +134 -0
- package/src/segment-system.tsx +272 -129
- package/src/serialize.ts +243 -0
- package/src/server/context.ts +309 -61
- package/src/server/cookie-store.ts +80 -5
- package/src/server/handle-store.ts +26 -24
- package/src/server/loader-registry.ts +10 -28
- package/src/server/request-context.ts +348 -128
- package/src/ssr/index.tsx +23 -15
- package/src/static-handler.ts +27 -18
- package/src/testing/cache-status.ts +162 -0
- package/src/testing/collect-handle.ts +40 -0
- package/src/testing/dispatch.ts +618 -0
- package/src/testing/dom.entry.ts +22 -0
- package/src/testing/e2e/fixture.ts +188 -0
- package/src/testing/e2e/index.ts +128 -0
- package/src/testing/e2e/matchers.ts +35 -0
- package/src/testing/e2e/page-helpers.ts +272 -0
- package/src/testing/e2e/parity.ts +387 -0
- package/src/testing/e2e/server.ts +195 -0
- package/src/testing/flight-matchers.ts +97 -0
- package/src/testing/flight-normalize.ts +11 -0
- package/src/testing/flight-runtime.d.ts +57 -0
- package/src/testing/flight-tree.ts +682 -0
- package/src/testing/flight.entry.ts +52 -0
- package/src/testing/flight.ts +232 -0
- package/src/testing/generated-routes.ts +183 -0
- package/src/testing/index.ts +99 -0
- package/src/testing/internal/context.ts +348 -0
- package/src/testing/internal/flight-client-globals.ts +30 -0
- package/src/testing/internal/seed-vars.ts +54 -0
- package/src/testing/render-handler.ts +330 -0
- package/src/testing/render-route.tsx +566 -0
- package/src/testing/run-loader.ts +378 -0
- package/src/testing/run-middleware.ts +205 -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 +305 -0
- package/src/theme/ThemeProvider.tsx +0 -52
- package/src/theme/ThemeScript.tsx +0 -6
- package/src/theme/constants.ts +0 -12
- package/src/theme/index.ts +0 -7
- package/src/theme/theme-context.ts +1 -5
- package/src/theme/theme-script.ts +0 -14
- package/src/theme/use-theme.ts +0 -3
- package/src/types/boundaries.ts +0 -35
- package/src/types/cache-types.ts +17 -8
- package/src/types/error-types.ts +30 -90
- package/src/types/global-namespace.ts +54 -41
- package/src/types/handler-context.ts +233 -81
- package/src/types/index.ts +1 -10
- package/src/types/loader-types.ts +44 -15
- package/src/types/request-scope.ts +107 -0
- package/src/types/route-config.ts +6 -50
- package/src/types/route-entry.ts +19 -7
- package/src/types/segments.ts +37 -14
- package/src/urls/include-helper.ts +33 -70
- package/src/urls/index.ts +1 -11
- package/src/urls/path-helper-types.ts +58 -11
- package/src/urls/path-helper.ts +57 -111
- package/src/urls/pattern-types.ts +48 -19
- package/src/urls/response-types.ts +25 -22
- package/src/urls/type-extraction.ts +58 -139
- package/src/urls/urls-function.ts +1 -18
- package/src/use-loader.tsx +346 -89
- package/src/vite/debug.ts +185 -0
- package/src/vite/discovery/bundle-postprocess.ts +36 -38
- package/src/vite/discovery/discover-routers.ts +130 -85
- 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 +192 -99
- 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 +51 -6
- package/src/vite/discovery/virtual-module-codegen.ts +14 -34
- package/src/vite/index.ts +8 -0
- package/src/vite/plugin-types.ts +187 -69
- package/src/vite/plugins/cjs-to-esm.ts +8 -18
- package/src/vite/plugins/client-ref-dedup.ts +16 -11
- package/src/vite/plugins/client-ref-hashing.ts +28 -15
- 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 +194 -0
- package/src/vite/plugins/expose-action-id.ts +49 -98
- package/src/vite/plugins/expose-id-utils.ts +11 -50
- package/src/vite/plugins/expose-ids/export-analysis.ts +76 -34
- package/src/vite/plugins/expose-ids/handler-transform.ts +10 -48
- package/src/vite/plugins/expose-ids/loader-transform.ts +3 -20
- package/src/vite/plugins/expose-ids/router-transform.ts +20 -16
- package/src/vite/plugins/expose-internal-ids.ts +554 -317
- package/src/vite/plugins/performance-tracks.ts +89 -0
- package/src/vite/plugins/refresh-cmd.ts +89 -27
- package/src/vite/plugins/use-cache-transform.ts +73 -83
- package/src/vite/plugins/vercel-output.ts +258 -0
- package/src/vite/plugins/version-injector.ts +21 -25
- package/src/vite/plugins/version-plugin.ts +41 -20
- package/src/vite/plugins/virtual-entries.ts +2 -17
- package/src/vite/rango.ts +257 -289
- package/src/vite/router-discovery.ts +930 -140
- package/src/vite/utils/ast-handler-extract.ts +15 -31
- package/src/vite/utils/banner.ts +4 -4
- package/src/vite/utils/bundle-analysis.ts +10 -15
- package/src/vite/utils/client-chunks.ts +184 -0
- package/src/vite/utils/forward-user-plugins.ts +171 -0
- package/src/vite/utils/manifest-utils.ts +4 -59
- package/src/vite/utils/package-resolution.ts +20 -52
- package/src/vite/utils/prerender-utils.ts +27 -29
- package/src/vite/utils/shared-utils.ts +92 -42
- package/src/browser/action-response-classifier.ts +0 -99
- package/src/browser/react/use-client-cache.ts +0 -58
- package/src/browser/shallow.ts +0 -40
- package/src/handles/index.ts +0 -7
- package/src/router/middleware-cookies.ts +0 -55
package/src/ssr/index.tsx
CHANGED
|
@@ -71,7 +71,7 @@ export interface SSRRenderOptions {
|
|
|
71
71
|
*/
|
|
72
72
|
export interface SSRDependencies<TEnv = unknown> {
|
|
73
73
|
/**
|
|
74
|
-
* createFromReadableStream from @
|
|
74
|
+
* createFromReadableStream from @rangojs/router/internal/deps/ssr
|
|
75
75
|
*/
|
|
76
76
|
createFromReadableStream: <T>(
|
|
77
77
|
stream: ReadableStream<Uint8Array>,
|
|
@@ -86,7 +86,7 @@ export interface SSRDependencies<TEnv = unknown> {
|
|
|
86
86
|
) => Promise<ReactDOMReadableStream>;
|
|
87
87
|
|
|
88
88
|
/**
|
|
89
|
-
* injectRSCPayload from
|
|
89
|
+
* injectRSCPayload from @rangojs/router/internal/deps/html-stream-server
|
|
90
90
|
*/
|
|
91
91
|
injectRSCPayload: (
|
|
92
92
|
rscStream: ReadableStream<Uint8Array>,
|
|
@@ -129,6 +129,7 @@ interface RscPayload {
|
|
|
129
129
|
matched?: string[];
|
|
130
130
|
pathname?: string;
|
|
131
131
|
params?: Record<string, string>;
|
|
132
|
+
basename?: string;
|
|
132
133
|
themeConfig?: ResolvedThemeConfig | null;
|
|
133
134
|
initialTheme?: Theme;
|
|
134
135
|
version?: string;
|
|
@@ -161,13 +162,18 @@ function createSsrEventController(opts: {
|
|
|
161
162
|
}): EventController {
|
|
162
163
|
const location = new URL(opts.pathname, "http://localhost");
|
|
163
164
|
let params = opts.params ?? {};
|
|
165
|
+
const rawMatched = opts.matched ?? [];
|
|
164
166
|
const handleState = {
|
|
165
167
|
data: opts.handleData ?? {},
|
|
166
|
-
segmentOrder: filterSegmentOrder(
|
|
168
|
+
segmentOrder: filterSegmentOrder(rawMatched),
|
|
169
|
+
routeSegmentIds: rawMatched.filter(
|
|
170
|
+
(id) => !id.includes(".@") && !/D\d+\./.test(id),
|
|
171
|
+
),
|
|
167
172
|
};
|
|
168
173
|
const state: DerivedNavigationState = {
|
|
169
174
|
state: "idle",
|
|
170
175
|
isStreaming: false,
|
|
176
|
+
isNavigating: false,
|
|
171
177
|
location,
|
|
172
178
|
pendingUrl: null,
|
|
173
179
|
inflightActions: [],
|
|
@@ -212,10 +218,10 @@ function createSsrEventController(opts: {
|
|
|
212
218
|
*
|
|
213
219
|
* @example
|
|
214
220
|
* ```tsx
|
|
215
|
-
* import { createSSRHandler } from "
|
|
216
|
-
* import { createFromReadableStream } from "@
|
|
221
|
+
* import { createSSRHandler } from "@rangojs/router/ssr";
|
|
222
|
+
* import { createFromReadableStream } from "@rangojs/router/internal/deps/ssr";
|
|
217
223
|
* import { renderToReadableStream } from "react-dom/server.edge";
|
|
218
|
-
* import { injectRSCPayload } from "
|
|
224
|
+
* import { injectRSCPayload } from "@rangojs/router/internal/deps/html-stream-server";
|
|
219
225
|
*
|
|
220
226
|
* export const renderHTML = createSSRHandler({
|
|
221
227
|
* createFromReadableStream,
|
|
@@ -257,9 +263,11 @@ export function createSSRHandler<TEnv = unknown>(deps: SSRDependencies<TEnv>) {
|
|
|
257
263
|
let payload: Promise<RscPayload> | undefined;
|
|
258
264
|
let handlesPromise: Promise<HandleData> | undefined;
|
|
259
265
|
let ssrContextValue: NavigationStoreContextValue | undefined;
|
|
266
|
+
let rootPromise: Promise<React.ReactNode> | undefined;
|
|
260
267
|
function SsrRoot() {
|
|
261
268
|
payload ??= createFromReadableStream<RscPayload>(rscStream1);
|
|
262
269
|
const resolved = React.use(payload);
|
|
270
|
+
|
|
263
271
|
const themeConfig = resolved.metadata?.themeConfig ?? null;
|
|
264
272
|
const pathname = resolved.metadata?.pathname ?? "/";
|
|
265
273
|
|
|
@@ -285,20 +293,20 @@ export function createSSRHandler<TEnv = unknown>(deps: SSRDependencies<TEnv>) {
|
|
|
285
293
|
navigate: async () => {},
|
|
286
294
|
refresh: async () => {},
|
|
287
295
|
version: resolved.metadata?.version,
|
|
296
|
+
basename: resolved.metadata?.basename,
|
|
288
297
|
};
|
|
289
298
|
|
|
290
299
|
// Build content tree from segments.
|
|
291
|
-
// Order must match NavigationProvider: NavigationStoreContext > ThemeProvider > content
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
300
|
+
// Order must match NavigationProvider: NavigationStoreContext > NonceContext > ThemeProvider > content
|
|
301
|
+
// Memoize like payload/handles above: renderSegments is async, so
|
|
302
|
+
// React.use() on a fresh promise suspends and replays SsrRoot, which
|
|
303
|
+
// would re-run the entire segment-tree build on every initial render.
|
|
304
|
+
rootPromise ??= Promise.resolve(
|
|
305
|
+
renderSegments(resolved.metadata?.segments ?? [], {
|
|
295
306
|
rootLayout: resolved.metadata?.rootLayout,
|
|
296
|
-
},
|
|
307
|
+
}),
|
|
297
308
|
);
|
|
298
|
-
let content: React.ReactNode =
|
|
299
|
-
reconstructedRoot instanceof Promise
|
|
300
|
-
? React.use(reconstructedRoot)
|
|
301
|
-
: reconstructedRoot;
|
|
309
|
+
let content: React.ReactNode = React.use(rootPromise);
|
|
302
310
|
|
|
303
311
|
// Wrap content with ThemeProvider if theme is enabled
|
|
304
312
|
if (themeConfig) {
|
package/src/static-handler.ts
CHANGED
|
@@ -32,10 +32,19 @@
|
|
|
32
32
|
*/
|
|
33
33
|
import type { ReactNode } from "react";
|
|
34
34
|
import type { Handler } from "./types.js";
|
|
35
|
-
import type {
|
|
35
|
+
import type { StaticBuildContext } from "./prerender.js";
|
|
36
|
+
import type { UseItems, HandlerUseItem } from "./route-types.js";
|
|
36
37
|
import { isCachedFunction } from "./cache/taint.js";
|
|
38
|
+
import { isUnderTestRunner } from "./runtime-env.js";
|
|
37
39
|
|
|
38
|
-
|
|
40
|
+
export interface StaticHandlerOptions {
|
|
41
|
+
/**
|
|
42
|
+
* Keep handler in server bundle for live fallback (default: false).
|
|
43
|
+
* false: handler replaced with stub, source-only APIs excluded from bundle.
|
|
44
|
+
* true: handler stays in bundle, renders live at request time.
|
|
45
|
+
*/
|
|
46
|
+
passthrough?: boolean;
|
|
47
|
+
}
|
|
39
48
|
|
|
40
49
|
export interface StaticHandlerDefinition<
|
|
41
50
|
TParams extends Record<string, any> = any,
|
|
@@ -46,22 +55,24 @@ export interface StaticHandlerDefinition<
|
|
|
46
55
|
/** In dev mode, the actual handler function that layout/path/parallel can call. */
|
|
47
56
|
handler: Handler<TParams>;
|
|
48
57
|
/** Static handler options (passthrough support). */
|
|
49
|
-
options?:
|
|
58
|
+
options?: StaticHandlerOptions;
|
|
59
|
+
/** Composable default DSL items merged when the handler is mounted. */
|
|
60
|
+
use?: () => UseItems<HandlerUseItem>;
|
|
50
61
|
}
|
|
51
62
|
|
|
52
|
-
//
|
|
63
|
+
// Process-stable fallback id counter (mirrors createHandle/createLoader/Prerender).
|
|
64
|
+
// Only assigned in bare unit tests where the Vite plugin did not inject an id.
|
|
65
|
+
let runtimeStaticIdCounter = 0;
|
|
53
66
|
|
|
54
67
|
export function Static<TParams extends Record<string, any> = {}>(
|
|
55
68
|
handler: (ctx: StaticBuildContext) => ReactNode | Promise<ReactNode>,
|
|
56
|
-
options?:
|
|
69
|
+
options?: StaticHandlerOptions,
|
|
57
70
|
__injectedId?: string,
|
|
58
71
|
): StaticHandlerDefinition<TParams>;
|
|
59
72
|
|
|
60
|
-
// -- Implementation ---------------------------------------------------------
|
|
61
|
-
|
|
62
73
|
export function Static<TParams extends Record<string, any>>(
|
|
63
74
|
handler: Function,
|
|
64
|
-
optionsOrId?:
|
|
75
|
+
optionsOrId?: StaticHandlerOptions | string,
|
|
65
76
|
maybeId?: string,
|
|
66
77
|
): StaticHandlerDefinition<TParams> {
|
|
67
78
|
if (isCachedFunction(handler)) {
|
|
@@ -72,22 +83,25 @@ export function Static<TParams extends Record<string, any>>(
|
|
|
72
83
|
);
|
|
73
84
|
}
|
|
74
85
|
|
|
75
|
-
let options:
|
|
86
|
+
let options: StaticHandlerOptions | undefined;
|
|
76
87
|
let id: string;
|
|
77
88
|
|
|
78
89
|
if (typeof optionsOrId === "string") {
|
|
79
90
|
id = optionsOrId;
|
|
80
91
|
} else {
|
|
81
|
-
options = optionsOrId as
|
|
92
|
+
options = optionsOrId as StaticHandlerOptions | undefined;
|
|
82
93
|
id = maybeId ?? "";
|
|
83
94
|
}
|
|
84
95
|
|
|
85
|
-
if (!id) {
|
|
96
|
+
if (!id && !isUnderTestRunner()) {
|
|
86
97
|
throw new Error(
|
|
87
|
-
"[
|
|
88
|
-
"
|
|
98
|
+
"[rango] Static: missing $$id. Use `export const X = Static(...)` and " +
|
|
99
|
+
"ensure the exposeInternalIds Vite plugin is configured.",
|
|
89
100
|
);
|
|
90
101
|
}
|
|
102
|
+
if (!id) {
|
|
103
|
+
id = `__rango_runtime_static_${runtimeStaticIdCounter++}`;
|
|
104
|
+
}
|
|
91
105
|
|
|
92
106
|
return {
|
|
93
107
|
__brand: "staticHandler" as const,
|
|
@@ -97,11 +111,6 @@ export function Static<TParams extends Record<string, any>>(
|
|
|
97
111
|
};
|
|
98
112
|
}
|
|
99
113
|
|
|
100
|
-
// -- Type guard -------------------------------------------------------------
|
|
101
|
-
|
|
102
|
-
/**
|
|
103
|
-
* Type guard to check if a value is a StaticHandlerDefinition.
|
|
104
|
-
*/
|
|
105
114
|
export function isStaticHandler(
|
|
106
115
|
value: unknown,
|
|
107
116
|
): value is StaticHandlerDefinition {
|
|
@@ -0,0 +1,162 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Cache-status testing primitives for @rangojs/router consumers.
|
|
3
|
+
*
|
|
4
|
+
* Two complementary paths, both DEVELOPMENT/TEST ONLY:
|
|
5
|
+
*
|
|
6
|
+
* 1. Header path — `parseCacheHeader` / `assertCacheStatus` read the
|
|
7
|
+
* `X-Rango-Cache` response header. The header is emitted only when the
|
|
8
|
+
* router's debug cache signal gate is on (the `debugCacheSignal` option or
|
|
9
|
+
* `RANGO_TEST_SIGNALS=1`). With the gate off there is no header and these
|
|
10
|
+
* helpers throw a clear "header missing" error.
|
|
11
|
+
*
|
|
12
|
+
* 2. Telemetry path — `createCacheSink` returns a `{ sink, events }` pair the
|
|
13
|
+
* consumer wires via `createRouter({ telemetry: sink })`. This has ZERO
|
|
14
|
+
* production surface: no header, just structured `cache.decision` events
|
|
15
|
+
* (which carry the same coarse `segments` cache signal). Assert with
|
|
16
|
+
* `assertCacheDecision(events, routeKey, expected)` (the one-call counterpart
|
|
17
|
+
* of `assertCacheStatus`) or filter raw via `filterCacheDecisions`.
|
|
18
|
+
*
|
|
19
|
+
* Both paths report the SAME coarse route-level signal — pick by TRANSPORT, not
|
|
20
|
+
* by meaning: the header is the only signal a black-box Playwright `Response`
|
|
21
|
+
* carries (needs the debug gate ON); the sink is the only zero-production-surface
|
|
22
|
+
* option and the only one exposing per-segment `shouldRevalidate`.
|
|
23
|
+
*
|
|
24
|
+
* v1 cache status is COARSE (route-level): the router reports a single entry
|
|
25
|
+
* keyed by the route key (the route NAME), not per individual segment.
|
|
26
|
+
*
|
|
27
|
+
* Import path: from a Vitest unit/integration test use `@rangojs/router/testing`;
|
|
28
|
+
* from a Playwright e2e use `@rangojs/router/testing/e2e` (the barrel pulls a
|
|
29
|
+
* build-only virtual that does not resolve in a plain Playwright runner).
|
|
30
|
+
*/
|
|
31
|
+
|
|
32
|
+
import type {
|
|
33
|
+
CacheDecisionEvent,
|
|
34
|
+
CacheSegmentStatus,
|
|
35
|
+
TelemetryEvent,
|
|
36
|
+
TelemetrySink,
|
|
37
|
+
} from "../router/telemetry.js";
|
|
38
|
+
|
|
39
|
+
const CACHE_HEADER = "X-Rango-Cache";
|
|
40
|
+
|
|
41
|
+
/** Expected cache status passed to assertCacheStatus. */
|
|
42
|
+
export type ExpectedCacheStatus = CacheSegmentStatus;
|
|
43
|
+
|
|
44
|
+
/** A target carrying response headers (a Response or a `{ headers }` object). */
|
|
45
|
+
export type CacheStatusTarget = Response | { headers: Headers };
|
|
46
|
+
|
|
47
|
+
export function parseCacheHeader(
|
|
48
|
+
headerValue: string | null | undefined,
|
|
49
|
+
): Record<string, string> {
|
|
50
|
+
const result: Record<string, string> = {};
|
|
51
|
+
if (!headerValue) return result;
|
|
52
|
+
for (const rawEntry of headerValue.split(",")) {
|
|
53
|
+
const entry = rawEntry.trim();
|
|
54
|
+
if (entry.length === 0) continue;
|
|
55
|
+
const eq = entry.indexOf("=");
|
|
56
|
+
if (eq === -1) continue;
|
|
57
|
+
const id = entry.slice(0, eq).trim();
|
|
58
|
+
const status = entry.slice(eq + 1).trim();
|
|
59
|
+
if (id.length === 0 || status.length === 0) continue;
|
|
60
|
+
result[id] = status;
|
|
61
|
+
}
|
|
62
|
+
return result;
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
function getHeaders(target: CacheStatusTarget): Headers {
|
|
66
|
+
return target.headers;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
export function assertCacheStatus(
|
|
70
|
+
target: CacheStatusTarget,
|
|
71
|
+
segment: string,
|
|
72
|
+
expected: ExpectedCacheStatus,
|
|
73
|
+
): void {
|
|
74
|
+
const headerValue = getHeaders(target).get(CACHE_HEADER);
|
|
75
|
+
if (headerValue === null) {
|
|
76
|
+
throw new Error(
|
|
77
|
+
`assertCacheStatus: response has no ${CACHE_HEADER} header. ` +
|
|
78
|
+
`Enable the debug cache signal via createRouter({ debugCacheSignal: true }) ` +
|
|
79
|
+
`or RANGO_TEST_SIGNALS=1.`,
|
|
80
|
+
);
|
|
81
|
+
}
|
|
82
|
+
const map = parseCacheHeader(headerValue);
|
|
83
|
+
const actual = map[segment];
|
|
84
|
+
if (actual === undefined) {
|
|
85
|
+
const known = Object.keys(map);
|
|
86
|
+
throw new Error(
|
|
87
|
+
`assertCacheStatus: segment "${segment}" not found in ${CACHE_HEADER} ` +
|
|
88
|
+
`("${headerValue}"). Known segments: ${
|
|
89
|
+
known.length > 0 ? known.join(", ") : "(none)"
|
|
90
|
+
}.`,
|
|
91
|
+
);
|
|
92
|
+
}
|
|
93
|
+
if (actual !== expected) {
|
|
94
|
+
throw new Error(
|
|
95
|
+
`assertCacheStatus: segment "${segment}" expected "${expected}" but got "${actual}".`,
|
|
96
|
+
);
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
/**
|
|
101
|
+
* A telemetry sink paired with the array it records events into.
|
|
102
|
+
*/
|
|
103
|
+
export interface CacheSink {
|
|
104
|
+
/** Wire into `createRouter({ telemetry: sink })`. */
|
|
105
|
+
sink: TelemetrySink;
|
|
106
|
+
/** All telemetry events captured so far, in emit order. */
|
|
107
|
+
events: TelemetryEvent[];
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
export function createCacheSink(): CacheSink {
|
|
111
|
+
const events: TelemetryEvent[] = [];
|
|
112
|
+
const sink: TelemetrySink = {
|
|
113
|
+
emit(event: TelemetryEvent): void {
|
|
114
|
+
events.push(event);
|
|
115
|
+
},
|
|
116
|
+
};
|
|
117
|
+
return { sink, events };
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
export function filterCacheDecisions(
|
|
121
|
+
events: readonly TelemetryEvent[],
|
|
122
|
+
): CacheDecisionEvent[] {
|
|
123
|
+
return events.filter(
|
|
124
|
+
(e): e is CacheDecisionEvent => e.type === "cache.decision",
|
|
125
|
+
);
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
/**
|
|
129
|
+
* Telemetry-path counterpart of {@link assertCacheStatus}: assert a captured
|
|
130
|
+
* `cache.decision` event reported `expected` for the segment keyed by `routeKey`
|
|
131
|
+
* (the route NAME, the same coarse key the header path uses). Throws an
|
|
132
|
+
* actionable error when no matching segment was captured, or on a mismatch.
|
|
133
|
+
*
|
|
134
|
+
* Pairs with {@link createCacheSink}: wire `createRouter({ telemetry: sink })`,
|
|
135
|
+
* drive an RSC request, then assert against the recorded `events`. This is the
|
|
136
|
+
* zero-production-surface path (no header to enable). NOTE: `events` accumulates
|
|
137
|
+
* across requests, so the FIRST matching segment wins — slice or recreate the
|
|
138
|
+
* sink between requests for the same `routeKey`.
|
|
139
|
+
*/
|
|
140
|
+
export function assertCacheDecision(
|
|
141
|
+
events: readonly TelemetryEvent[],
|
|
142
|
+
routeKey: string,
|
|
143
|
+
expected: ExpectedCacheStatus,
|
|
144
|
+
): void {
|
|
145
|
+
const segments = filterCacheDecisions(events).flatMap(
|
|
146
|
+
(d) => d.segments ?? [],
|
|
147
|
+
);
|
|
148
|
+
const seg = segments.find((s) => s.id === routeKey);
|
|
149
|
+
if (seg === undefined) {
|
|
150
|
+
const known = segments.map((s) => s.id);
|
|
151
|
+
throw new Error(
|
|
152
|
+
`assertCacheDecision: no cache.decision segment for routeKey "${routeKey}". ` +
|
|
153
|
+
`Seen: ${known.length > 0 ? known.join(", ") : "(none)"}. Wire ` +
|
|
154
|
+
`createRouter({ telemetry: createCacheSink().sink }) and drive an RSC request.`,
|
|
155
|
+
);
|
|
156
|
+
}
|
|
157
|
+
if (seg.cacheStatus !== expected) {
|
|
158
|
+
throw new Error(
|
|
159
|
+
`assertCacheDecision: routeKey "${routeKey}" expected "${expected}" but got "${seg.cacheStatus}".`,
|
|
160
|
+
);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* collectHandle — unit-test a handle's `collect`/accumulator function directly.
|
|
3
|
+
*
|
|
4
|
+
* A handle's collect function (the `createHandle(collect)` argument that maps the
|
|
5
|
+
* per-segment pushed values into the accumulated result) is otherwise not
|
|
6
|
+
* directly reachable: createHandle keeps it in a private registry keyed by the
|
|
7
|
+
* handle's `$$id` and returns only `{ __brand, $$id }`. This primitive runs that
|
|
8
|
+
* REAL registered collect on per-segment values you provide and returns the
|
|
9
|
+
* accumulated result — so the mapper/accumulator is unit-testable without a full
|
|
10
|
+
* route match.
|
|
11
|
+
*
|
|
12
|
+
* It relies on createHandle registering the collect even in a bare test (it
|
|
13
|
+
* assigns a runtime fallback id when the Vite plugin did not inject one). If a
|
|
14
|
+
* handle's module was never imported (so createHandle never ran), the collect is
|
|
15
|
+
* unregistered and this falls back to a flat array with a warning.
|
|
16
|
+
*/
|
|
17
|
+
|
|
18
|
+
import { getCollectFn, type Handle } from "../handle.js";
|
|
19
|
+
|
|
20
|
+
export function collectHandle<TData, TAccumulated>(
|
|
21
|
+
handle: Handle<TData, TAccumulated>,
|
|
22
|
+
segments: ReadonlyArray<ReadonlyArray<TData>>,
|
|
23
|
+
): TAccumulated {
|
|
24
|
+
const collectFn = getCollectFn(handle.$$id) as
|
|
25
|
+
| ((segments: TData[][]) => TAccumulated)
|
|
26
|
+
| undefined;
|
|
27
|
+
|
|
28
|
+
if (!collectFn) {
|
|
29
|
+
console.warn(
|
|
30
|
+
`[rango] collectHandle: handle "${handle.$$id}" has no registered collect ` +
|
|
31
|
+
`function. Import the handle's module so createHandle() runs. Falling ` +
|
|
32
|
+
`back to a flat array.`,
|
|
33
|
+
);
|
|
34
|
+
return segments.flat() as unknown as TAccumulated;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
// Drop empty arrays matching production behavior (segment count/indices).
|
|
38
|
+
const nonEmpty = segments.filter((seg) => seg.length > 0) as TData[][];
|
|
39
|
+
return collectFn(nonEmpty);
|
|
40
|
+
}
|