@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
|
@@ -0,0 +1,330 @@
|
|
|
1
|
+
/// <reference path="./flight-runtime.d.ts" />
|
|
2
|
+
/**
|
|
3
|
+
* renderHandler — run a REAL route handler and assert what it renders.
|
|
4
|
+
*
|
|
5
|
+
* A Rango route handler is a pure function `(ctx) => RSC` (what you pass to
|
|
6
|
+
* `path("/p/:slug", ProductPage)`), NOT a component. To test one faithfully you
|
|
7
|
+
* must give it the HandlerContext the router builds at runtime, so `ctx.params`,
|
|
8
|
+
* `ctx.use(Loader)`, `ctx.use(Meta)` / `ctx.use(Breadcrumbs)` (handles),
|
|
9
|
+
* `ctx.reverse`, `ctx.get`/`ctx.header`/`cookies()` all work. renderHandler does
|
|
10
|
+
* exactly that, then serializes the handler's returned RSC and deserializes it
|
|
11
|
+
* to an inspectable tree (same serialize/deserialize core as renderServerTree).
|
|
12
|
+
*
|
|
13
|
+
* Loaders are SEEDED (no real loader execution) the same way `runLoader` seeds
|
|
14
|
+
* them — pass `loaders: [[ProductLoader, data]]`. Handle pushes
|
|
15
|
+
* (`ctx.use(Meta)({...})`) are captured on `result.handles`. The handler's
|
|
16
|
+
* cookie/header/flash effects and a thrown/returned redirect are surfaced too
|
|
17
|
+
* (like `runInRequestContext`). If the handler returns/throws a `Response`
|
|
18
|
+
* (a response route / `throw redirect()`), there is no RSC `tree`.
|
|
19
|
+
*
|
|
20
|
+
* Must run under the `react-server` export condition (the rsc Vitest project).
|
|
21
|
+
* Wire `rangoUseClientTransform()` so `"use client"` islands in the handler's RSC
|
|
22
|
+
* auto-register (or pass `clientComponents`).
|
|
23
|
+
*/
|
|
24
|
+
import type { ReactNode } from "react";
|
|
25
|
+
import {
|
|
26
|
+
createRequestContext,
|
|
27
|
+
runWithRequestContext,
|
|
28
|
+
setRequestContextParams,
|
|
29
|
+
type RequestContext,
|
|
30
|
+
} from "../server/request-context.js";
|
|
31
|
+
import { createHandlerContext } from "../router/handler-context.js";
|
|
32
|
+
import { resolveLocationStateEntries } from "../browser/react/location-state-shared.js";
|
|
33
|
+
import { isHandle, type Handle } from "../handle.js";
|
|
34
|
+
import { withDefer } from "../defer.js";
|
|
35
|
+
import type { HandlerContext } from "../types/handler-context.js";
|
|
36
|
+
import type { LoaderDefinition } from "../types.js";
|
|
37
|
+
import {
|
|
38
|
+
seedVariables,
|
|
39
|
+
resolveSeededStateCookieName,
|
|
40
|
+
type VarsInit,
|
|
41
|
+
type StateCookieSeed,
|
|
42
|
+
} from "./internal/seed-vars.js";
|
|
43
|
+
|
|
44
|
+
export type { StateCookieSeed } from "./internal/seed-vars.js";
|
|
45
|
+
import {
|
|
46
|
+
assertNoLegacyUrlOption,
|
|
47
|
+
serializeNodeToFlight,
|
|
48
|
+
isServerOnlyStubError,
|
|
49
|
+
} from "./flight.js";
|
|
50
|
+
import type { SegmentCacheStore } from "../cache/types.js";
|
|
51
|
+
import type { CacheProfile } from "../cache/profile-registry.js";
|
|
52
|
+
import {
|
|
53
|
+
deserializeFlight,
|
|
54
|
+
makeClientManifest,
|
|
55
|
+
registerClientComponents,
|
|
56
|
+
} from "./flight-tree.js";
|
|
57
|
+
|
|
58
|
+
const DEFAULT_URL = "http://localhost/";
|
|
59
|
+
|
|
60
|
+
/** A route handler under test: the `(ctx) => RSC | Response` function you pass to `path(...)`. */
|
|
61
|
+
export type TestableHandler<TEnv = any> = (
|
|
62
|
+
ctx: HandlerContext<any, TEnv>,
|
|
63
|
+
) => ReactNode | Promise<ReactNode> | Response | Promise<Response>;
|
|
64
|
+
|
|
65
|
+
/** Options for {@link renderHandler}. */
|
|
66
|
+
export interface RenderHandlerOptions<TEnv = any> {
|
|
67
|
+
/** Route params surfaced as `ctx.params`. */
|
|
68
|
+
params?: Record<string, string>;
|
|
69
|
+
/** Environment bindings surfaced as `ctx.env`. */
|
|
70
|
+
env?: TEnv;
|
|
71
|
+
/** Backing Request (string or Request); defaults to a localhost GET. */
|
|
72
|
+
request?: Request | string;
|
|
73
|
+
/** Request headers (e.g. Cookie) the handler reads via `cookies()`. */
|
|
74
|
+
headers?: HeadersInit;
|
|
75
|
+
/** Variables a prior middleware set, read via `ctx.get(...)`. Object or tuples. */
|
|
76
|
+
vars?: VarsInit;
|
|
77
|
+
/** Matched route name (drives `ctx.routeName` and scoped reverse). */
|
|
78
|
+
routeName?: string;
|
|
79
|
+
/** Route name -> pattern map enabling `ctx.reverse()`. */
|
|
80
|
+
routeMap?: Record<string, string>;
|
|
81
|
+
/**
|
|
82
|
+
* Seed the data `ctx.use(SomeLoader)` returns — NO real loader runs (same model
|
|
83
|
+
* as `runLoader`'s `loaders`). Matched by loader reference, so a real
|
|
84
|
+
* `createLoader()` handle resolves regardless of its build-injected `$$id`.
|
|
85
|
+
*/
|
|
86
|
+
loaders?: ReadonlyArray<readonly [LoaderDefinition<any, any>, unknown]>;
|
|
87
|
+
/**
|
|
88
|
+
* `"use client"` components in the handler's RSC, so they serialize as real
|
|
89
|
+
* boundaries when `rangoUseClientTransform()` is not wired. Keyed by name; see
|
|
90
|
+
* renderServerTree's `clientComponents`.
|
|
91
|
+
*/
|
|
92
|
+
clientComponents?: Record<string, unknown>;
|
|
93
|
+
/**
|
|
94
|
+
* Customize the rango state cookie a handler that calls
|
|
95
|
+
* `invalidateClientCache()` rotates. The name is ALWAYS seeded (default
|
|
96
|
+
* `rango-state_router_0`) so the rotation `Set-Cookie` is emitted like
|
|
97
|
+
* production rather than no-opping; override `prefix`/`routerId` to match your
|
|
98
|
+
* `createRouter({ stateCookiePrefix, id })`, or `version` (the build
|
|
99
|
+
* identifier prefixed to the rotated `{version}:{timestamp}` value, default
|
|
100
|
+
* `"0"`). Assert via `result.response.headers.getSetCookie()` against
|
|
101
|
+
* `result.stateCookieName`.
|
|
102
|
+
*/
|
|
103
|
+
stateCookie?: StateCookieSeed;
|
|
104
|
+
/**
|
|
105
|
+
* Segment cache store backing a `"use cache"` function the handler invokes
|
|
106
|
+
* (e.g. `new MemorySegmentCacheStore()`). Without it, `registerCachedFunction`
|
|
107
|
+
* takes the uncached bypass and the cached path is NOT exercised (the runtime
|
|
108
|
+
* emits a one-time warning under the test runner). Pair with `cacheProfiles`
|
|
109
|
+
* so the profile the directive names resolves.
|
|
110
|
+
*/
|
|
111
|
+
cacheStore?: SegmentCacheStore;
|
|
112
|
+
/**
|
|
113
|
+
* Cache profiles in the `createRouter({ cacheProfiles })` shape, required for
|
|
114
|
+
* `"use cache: profileName"` resolution once a `cacheStore` is wired.
|
|
115
|
+
*/
|
|
116
|
+
cacheProfiles?: Record<string, CacheProfile>;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
/** Result of {@link renderHandler}. */
|
|
120
|
+
export interface RenderHandlerResult {
|
|
121
|
+
/**
|
|
122
|
+
* The deserialized RSC the handler returned, as an inspectable React element
|
|
123
|
+
* tree — `undefined` when the handler returned or threw a `Response`. Use
|
|
124
|
+
* `findClientBoundaries` (from testing/flight) to locate client islands.
|
|
125
|
+
*/
|
|
126
|
+
tree: unknown;
|
|
127
|
+
/** The raw Flight wire string; `undefined` when the handler produced a `Response`. */
|
|
128
|
+
flight: string | undefined;
|
|
129
|
+
/** The value the handler THREW (a `redirect()`/`notFound()` Response), captured not re-thrown. */
|
|
130
|
+
thrown: unknown;
|
|
131
|
+
/** The merged Response (status + headers + Set-Cookie); a thrown/returned redirect merged with accumulated effects. */
|
|
132
|
+
response: Response;
|
|
133
|
+
/** Effective cookie view after the handler ran, as `{ name: value }`. */
|
|
134
|
+
cookies: Record<string, string>;
|
|
135
|
+
/** Response headers as `{ name: value }` (excludes set-cookie; includes a redirect Location). */
|
|
136
|
+
headers: Record<string, string>;
|
|
137
|
+
/**
|
|
138
|
+
* The resolved rango state cookie name this run seeded (default
|
|
139
|
+
* `rango-state_router_0`, or composed from `opts.stateCookie`). Assert the
|
|
140
|
+
* `invalidateClientCache()` rotation against it without recomputing:
|
|
141
|
+
* `response.headers.getSetCookie().some((c) => c.startsWith(stateCookieName + "="))`.
|
|
142
|
+
*/
|
|
143
|
+
stateCookieName: string;
|
|
144
|
+
/** Location state the handler set (`ctx.setLocationState`/`redirect({ state })`), as `{ key: value }`. */
|
|
145
|
+
locationState: Record<string, unknown>;
|
|
146
|
+
/** What the handler pushed via `ctx.use(Handle)(...)` (e.g. Meta, Breadcrumbs), keyed by handle. */
|
|
147
|
+
handles: Map<Handle<any, any>, unknown[]>;
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
/**
|
|
151
|
+
* A renderHandler MISCONFIGURATION (e.g. an unseeded loader) — distinct from a
|
|
152
|
+
* value the handler intentionally threw (a redirect). Setup errors REJECT;
|
|
153
|
+
* handler throws are captured on `result.thrown`.
|
|
154
|
+
*/
|
|
155
|
+
class RenderHandlerSetupError extends Error {}
|
|
156
|
+
|
|
157
|
+
function headersToObject(headers: Headers): Record<string, string> {
|
|
158
|
+
const out: Record<string, string> = {};
|
|
159
|
+
headers.forEach((value, name) => {
|
|
160
|
+
if (name.toLowerCase() !== "set-cookie") out[name] = value;
|
|
161
|
+
});
|
|
162
|
+
return out;
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
function toRequest(
|
|
166
|
+
request: Request | string | undefined,
|
|
167
|
+
headers?: HeadersInit,
|
|
168
|
+
): Request {
|
|
169
|
+
if (request instanceof Request) return request;
|
|
170
|
+
if (typeof request === "string") {
|
|
171
|
+
return new Request(new URL(request, DEFAULT_URL), { headers });
|
|
172
|
+
}
|
|
173
|
+
return new Request(DEFAULT_URL, { headers });
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
function buildResponse(reqCtx: RequestContext<any>, source: unknown): Response {
|
|
177
|
+
const stub = reqCtx.res;
|
|
178
|
+
if (source instanceof Response) {
|
|
179
|
+
const merged = new Headers(source.headers);
|
|
180
|
+
for (const cookie of stub.headers.getSetCookie()) {
|
|
181
|
+
merged.append("set-cookie", cookie);
|
|
182
|
+
}
|
|
183
|
+
stub.headers.forEach((value, name) => {
|
|
184
|
+
if (name.toLowerCase() === "set-cookie") return;
|
|
185
|
+
if (!merged.has(name)) merged.set(name, value);
|
|
186
|
+
});
|
|
187
|
+
return new Response(source.body, {
|
|
188
|
+
status: source.status,
|
|
189
|
+
headers: merged,
|
|
190
|
+
});
|
|
191
|
+
}
|
|
192
|
+
return new Response(null, { status: stub.status, headers: stub.headers });
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
/**
|
|
196
|
+
* Run a route handler with a seeded HandlerContext and return its rendered RSC
|
|
197
|
+
* (deserialized tree) plus the effects it produced. See the module header.
|
|
198
|
+
*
|
|
199
|
+
* @example
|
|
200
|
+
* ```ts
|
|
201
|
+
* // ProductPage is the real handler: (ctx) => <main>{ctx.params.slug}...</main>
|
|
202
|
+
* const { tree, handles } = await renderHandler(ProductPage, {
|
|
203
|
+
* params: { slug: "wine" },
|
|
204
|
+
* loaders: [[ProductLoader, { name: "Wine", price: 9 }]],
|
|
205
|
+
* vars: [[Tenant, tenant]],
|
|
206
|
+
* routeMap: { product: "/p/:slug" },
|
|
207
|
+
* });
|
|
208
|
+
* ```
|
|
209
|
+
*/
|
|
210
|
+
export async function renderHandler<TEnv = any>(
|
|
211
|
+
handler: TestableHandler<TEnv>,
|
|
212
|
+
opts: RenderHandlerOptions<TEnv> = {},
|
|
213
|
+
): Promise<RenderHandlerResult> {
|
|
214
|
+
assertNoLegacyUrlOption(opts, "renderHandler");
|
|
215
|
+
if (opts.clientComponents) registerClientComponents(opts.clientComponents);
|
|
216
|
+
const request = toRequest(opts.request, opts.headers);
|
|
217
|
+
const url = new URL(request.url);
|
|
218
|
+
const stateCookieName = resolveSeededStateCookieName(opts.stateCookie);
|
|
219
|
+
const reqCtx = createRequestContext<TEnv>({
|
|
220
|
+
env: (opts.env ?? {}) as TEnv,
|
|
221
|
+
request,
|
|
222
|
+
url,
|
|
223
|
+
variables: seedVariables({}, opts.vars),
|
|
224
|
+
stateCookieName,
|
|
225
|
+
version: opts.stateCookie?.version,
|
|
226
|
+
cacheStore: opts.cacheStore,
|
|
227
|
+
cacheProfiles: opts.cacheProfiles,
|
|
228
|
+
});
|
|
229
|
+
|
|
230
|
+
const loaderSeeds = new Map<unknown, unknown>(opts.loaders ?? []);
|
|
231
|
+
const handlePushes = new Map<Handle<any, any>, unknown[]>();
|
|
232
|
+
|
|
233
|
+
let out: ReactNode | Response | undefined;
|
|
234
|
+
let flight: string | undefined;
|
|
235
|
+
let thrown: unknown;
|
|
236
|
+
let didThrow = false;
|
|
237
|
+
|
|
238
|
+
await runWithRequestContext(reqCtx as RequestContext<TEnv>, async () => {
|
|
239
|
+
// Scope the request-context reverse to opts.routeMap too (not just the
|
|
240
|
+
// handler context built below), so a nested server component reading
|
|
241
|
+
// getRequestContext().reverse() resolves against the same map as the
|
|
242
|
+
// handler's ctx.reverse -- matching renderToFlightString/renderServerTree.
|
|
243
|
+
setRequestContextParams(opts.params ?? {}, opts.routeName, opts.routeMap);
|
|
244
|
+
const hctx = createHandlerContext<TEnv>(
|
|
245
|
+
opts.params ?? {},
|
|
246
|
+
reqCtx.request,
|
|
247
|
+
reqCtx.searchParams,
|
|
248
|
+
reqCtx.pathname,
|
|
249
|
+
reqCtx.url,
|
|
250
|
+
reqCtx.env,
|
|
251
|
+
opts.routeMap ?? {},
|
|
252
|
+
opts.routeName,
|
|
253
|
+
);
|
|
254
|
+
(hctx as { use: unknown }).use = (item: unknown) => {
|
|
255
|
+
if (isHandle(item)) {
|
|
256
|
+
const handle = item as Handle<any, any>;
|
|
257
|
+
// withDefer attaches .defer() so the harness mirrors production's push.
|
|
258
|
+
return withDefer((dataOrFn: unknown) => {
|
|
259
|
+
const value =
|
|
260
|
+
typeof dataOrFn === "function"
|
|
261
|
+
? (dataOrFn as () => unknown)()
|
|
262
|
+
: dataOrFn;
|
|
263
|
+
const pushed = handlePushes.get(handle) ?? [];
|
|
264
|
+
pushed.push(value);
|
|
265
|
+
handlePushes.set(handle, pushed);
|
|
266
|
+
});
|
|
267
|
+
}
|
|
268
|
+
if (loaderSeeds.has(item)) return loaderSeeds.get(item);
|
|
269
|
+
throw new RenderHandlerSetupError(
|
|
270
|
+
`renderHandler: ctx.use(loader) was not seeded. Pass ` +
|
|
271
|
+
`{ loaders: [[YourLoader, data]] } for each loader the handler reads.`,
|
|
272
|
+
);
|
|
273
|
+
};
|
|
274
|
+
(hctx as { _currentSegmentId?: string })._currentSegmentId = "test.segment";
|
|
275
|
+
|
|
276
|
+
try {
|
|
277
|
+
out = await handler(hctx as HandlerContext<any, TEnv>);
|
|
278
|
+
if (out !== undefined && !(out instanceof Response)) {
|
|
279
|
+
flight = await serializeNodeToFlight(
|
|
280
|
+
out as ReactNode,
|
|
281
|
+
makeClientManifest(),
|
|
282
|
+
url.pathname,
|
|
283
|
+
);
|
|
284
|
+
}
|
|
285
|
+
} catch (error) {
|
|
286
|
+
if (error instanceof RenderHandlerSetupError) throw error;
|
|
287
|
+
if (isServerOnlyStubError(error)) {
|
|
288
|
+
throw new RenderHandlerSetupError(
|
|
289
|
+
`renderHandler: the handler called a server-only API (getRequestContext/cookies/...) ` +
|
|
290
|
+
`but "@rangojs/router" resolved to the out-of-react-server stub. Add ` +
|
|
291
|
+
`rangoTestAliases({ preset }) to your vitest.rsc.config.ts \`resolve.alias\` so the ` +
|
|
292
|
+
`bare specifier maps to index.rsc.ts (the real react-server implementations). ` +
|
|
293
|
+
`Original: ${(error as Error).message}`,
|
|
294
|
+
);
|
|
295
|
+
}
|
|
296
|
+
didThrow = true;
|
|
297
|
+
thrown = error;
|
|
298
|
+
}
|
|
299
|
+
});
|
|
300
|
+
|
|
301
|
+
const cookies = { ...reqCtx.cookies() };
|
|
302
|
+
const responseSource = didThrow
|
|
303
|
+
? thrown
|
|
304
|
+
: out instanceof Response
|
|
305
|
+
? out
|
|
306
|
+
: undefined;
|
|
307
|
+
const response = buildResponse(reqCtx as RequestContext<any>, responseSource);
|
|
308
|
+
const headers = headersToObject(response.headers);
|
|
309
|
+
const locationState = resolveLocationStateEntries(
|
|
310
|
+
(
|
|
311
|
+
reqCtx as {
|
|
312
|
+
_locationState?: Parameters<typeof resolveLocationStateEntries>[0];
|
|
313
|
+
}
|
|
314
|
+
)._locationState ?? [],
|
|
315
|
+
);
|
|
316
|
+
const tree =
|
|
317
|
+
flight !== undefined ? await deserializeFlight(flight) : undefined;
|
|
318
|
+
|
|
319
|
+
return {
|
|
320
|
+
tree,
|
|
321
|
+
flight,
|
|
322
|
+
thrown,
|
|
323
|
+
response,
|
|
324
|
+
cookies,
|
|
325
|
+
headers,
|
|
326
|
+
stateCookieName,
|
|
327
|
+
locationState,
|
|
328
|
+
handles: handlePushes,
|
|
329
|
+
};
|
|
330
|
+
}
|