@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,348 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared internals for the consumer testing primitives.
|
|
3
|
+
*
|
|
4
|
+
* Builds a real RequestContext via the same createRequestContext the RSC
|
|
5
|
+
* handler uses, with test-friendly defaults, so loaders and middleware run
|
|
6
|
+
* with production-fidelity context (cookies, headers, get/set, use, reverse)
|
|
7
|
+
* instead of a hand-rolled mock.
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
import {
|
|
11
|
+
createRequestContext,
|
|
12
|
+
runWithRequestContext,
|
|
13
|
+
type RequestContext,
|
|
14
|
+
} from "../../server/request-context.js";
|
|
15
|
+
import { resolveLocationStateEntries } from "../../browser/react/location-state-shared.js";
|
|
16
|
+
import { createReverseFunction } from "../../router/handler-context.js";
|
|
17
|
+
import { normalizeBasename } from "../../router/basename.js";
|
|
18
|
+
import {
|
|
19
|
+
seedVariables,
|
|
20
|
+
resolveSeededStateCookieName,
|
|
21
|
+
type VarsInit,
|
|
22
|
+
type StateCookieSeed,
|
|
23
|
+
} from "./seed-vars.js";
|
|
24
|
+
import type { ThemeConfig } from "../../theme/types.js";
|
|
25
|
+
import { resolveThemeConfig } from "../../theme/constants.js";
|
|
26
|
+
import type { SegmentCacheStore } from "../../cache/types.js";
|
|
27
|
+
import type { CacheProfile } from "../../cache/profile-registry.js";
|
|
28
|
+
|
|
29
|
+
const DEFAULT_ORIGIN = "http://localhost/";
|
|
30
|
+
|
|
31
|
+
export type { VarsInit, StateCookieSeed };
|
|
32
|
+
export { seedVariables };
|
|
33
|
+
|
|
34
|
+
/** Normalize a Request | string | undefined into a concrete Request. */
|
|
35
|
+
export function toRequest(
|
|
36
|
+
request: Request | string | undefined,
|
|
37
|
+
init?: RequestInit,
|
|
38
|
+
): Request {
|
|
39
|
+
if (request instanceof Request) return request;
|
|
40
|
+
return typeof request === "string"
|
|
41
|
+
? new Request(new URL(request, DEFAULT_ORIGIN), init)
|
|
42
|
+
: new Request(DEFAULT_ORIGIN, init);
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
export interface CreateTestContextOptions<TEnv> {
|
|
46
|
+
env?: TEnv;
|
|
47
|
+
request?: Request | string;
|
|
48
|
+
requestInit?: RequestInit;
|
|
49
|
+
/** Backing store for ctx.get()/ctx.set(); pre-seeded from `vars`. */
|
|
50
|
+
variables?: Record<string, unknown>;
|
|
51
|
+
/** Variables a prior middleware would have set (object or [key, value] list). */
|
|
52
|
+
vars?: VarsInit;
|
|
53
|
+
/** Route name -> pattern map enabling ctx.reverse() without global state. */
|
|
54
|
+
routeMap?: Record<string, string>;
|
|
55
|
+
routeName?: string;
|
|
56
|
+
params?: Record<string, string>;
|
|
57
|
+
/**
|
|
58
|
+
* Router basename for this request (what the RSC handler stores on the
|
|
59
|
+
* context). Drives redirect() prefixing. Normalized exactly like
|
|
60
|
+
* createRouter({ basename }) (leading slash forced, trailing stripped, bare
|
|
61
|
+
* "/" -> undefined) so passing the same value your router takes yields the
|
|
62
|
+
* same redirect Location. Defaults to undefined (no basename).
|
|
63
|
+
*/
|
|
64
|
+
basename?: string;
|
|
65
|
+
/**
|
|
66
|
+
* Cache store backing `use cache` functions invoked during the test, the
|
|
67
|
+
* same shape `createRouter({ cache })` resolves. Without it,
|
|
68
|
+
* registerCachedFunction bypasses (it checks for a store FIRST), so a cached
|
|
69
|
+
* function runs uncached and its taint/profile guards never fire. Wire one
|
|
70
|
+
* (e.g. `new MemorySegmentCacheStore()`) to exercise real cache behavior.
|
|
71
|
+
*/
|
|
72
|
+
cacheStore?: SegmentCacheStore;
|
|
73
|
+
/**
|
|
74
|
+
* Cache profiles in the `createRouter({ cacheProfiles })` shape. Required for
|
|
75
|
+
* a `use cache: "profileName"` function to resolve its profile (an unknown
|
|
76
|
+
* profile throws), once a `cacheStore` is wired.
|
|
77
|
+
*/
|
|
78
|
+
cacheProfiles?: Record<string, CacheProfile>;
|
|
79
|
+
/**
|
|
80
|
+
* Theme config in the same shape `createRouter({ theme })` takes (resolved
|
|
81
|
+
* internally). Without it `ctx.theme`/`ctx.setTheme` are inert (undefined),
|
|
82
|
+
* mirroring an app with no theme configured. Pass one (e.g. `true`, or
|
|
83
|
+
* `{ themes: [...] }`) to exercise a handler that reads them.
|
|
84
|
+
*/
|
|
85
|
+
theme?: ThemeConfig | true;
|
|
86
|
+
/**
|
|
87
|
+
* Customize the rango state cookie that `invalidateClientCache()` rotates.
|
|
88
|
+
* The name is ALWAYS seeded (default `rango-state_router_0`) so a call to
|
|
89
|
+
* `invalidateClientCache()` rotates and emits the `Set-Cookie` exactly as in
|
|
90
|
+
* production, rather than silently no-opping. Override `prefix`/`routerId` to
|
|
91
|
+
* match your `createRouter({ stateCookiePrefix, id })` so the test asserts the
|
|
92
|
+
* same name, or `version` (the build identifier prefixed to the rotated
|
|
93
|
+
* `{version}:{timestamp}` value, default `"0"`). Assert
|
|
94
|
+
* `response.headers.getSetCookie()` against the resolved `stateCookieName`
|
|
95
|
+
* (returned by `runInRequestContext`).
|
|
96
|
+
*/
|
|
97
|
+
stateCookie?: StateCookieSeed;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
/**
|
|
101
|
+
* The seeded RequestContext with its `reverse` RELAXED to accept any route NAME
|
|
102
|
+
* from the `routeMap` you passed, rather than the global `Rango.GeneratedRouteMap`
|
|
103
|
+
* union — so reversing a test-only route name is not a type error (it works at
|
|
104
|
+
* runtime; the names come from your `routeMap`). Mirrors runLoader's
|
|
105
|
+
* `TestLoaderContext.reverse`. Everything else is the real `RequestContext`.
|
|
106
|
+
*/
|
|
107
|
+
export type TestRequestContextObject<TEnv> = Omit<
|
|
108
|
+
RequestContext<TEnv>,
|
|
109
|
+
"reverse"
|
|
110
|
+
> & {
|
|
111
|
+
reverse: (
|
|
112
|
+
name: string,
|
|
113
|
+
params?: Record<string, string>,
|
|
114
|
+
search?: Record<string, unknown>,
|
|
115
|
+
) => string;
|
|
116
|
+
};
|
|
117
|
+
|
|
118
|
+
export interface TestRequestContext<TEnv> {
|
|
119
|
+
ctx: TestRequestContextObject<TEnv>;
|
|
120
|
+
request: Request;
|
|
121
|
+
url: URL;
|
|
122
|
+
variables: Record<string, unknown>;
|
|
123
|
+
/**
|
|
124
|
+
* The resolved rango state cookie name seeded into the context (default
|
|
125
|
+
* `rango-state_router_0`, or composed from `opts.stateCookie`). The name a
|
|
126
|
+
* call to `invalidateClientCache()` rotates.
|
|
127
|
+
*/
|
|
128
|
+
stateCookieName: string;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
/**
|
|
132
|
+
* Create a real RequestContext for unit-testing loaders/middleware.
|
|
133
|
+
*
|
|
134
|
+
* The returned `ctx` must be ENTERED before use — wrap your call in
|
|
135
|
+
* `runWithRequestContext(ctx, fn)` (re-exported from `@rangojs/router/testing`)
|
|
136
|
+
* so that cookie/header mutations and `getRequestContext()` resolve. For the
|
|
137
|
+
* common case prefer {@link runInRequestContext}, which builds AND enters the
|
|
138
|
+
* context in a single call.
|
|
139
|
+
*/
|
|
140
|
+
export function createTestRequestContext<TEnv>(
|
|
141
|
+
opts: CreateTestContextOptions<TEnv> = {},
|
|
142
|
+
): TestRequestContext<TEnv> {
|
|
143
|
+
const request = toRequest(opts.request, opts.requestInit);
|
|
144
|
+
const url = new URL(request.url);
|
|
145
|
+
const variables = seedVariables(opts.variables ?? {}, opts.vars);
|
|
146
|
+
const stateCookieName = resolveSeededStateCookieName(opts.stateCookie);
|
|
147
|
+
const ctx = createRequestContext<TEnv>({
|
|
148
|
+
env: (opts.env ?? {}) as TEnv,
|
|
149
|
+
request,
|
|
150
|
+
url,
|
|
151
|
+
variables,
|
|
152
|
+
themeConfig:
|
|
153
|
+
opts.theme === undefined ? undefined : resolveThemeConfig(opts.theme),
|
|
154
|
+
cacheStore: opts.cacheStore,
|
|
155
|
+
cacheProfiles: opts.cacheProfiles,
|
|
156
|
+
stateCookieName,
|
|
157
|
+
version: opts.stateCookie?.version,
|
|
158
|
+
});
|
|
159
|
+
if (opts.basename !== undefined)
|
|
160
|
+
ctx._basename = normalizeBasename(opts.basename);
|
|
161
|
+
if (opts.params) ctx.params = opts.params;
|
|
162
|
+
if (opts.routeMap) {
|
|
163
|
+
ctx._routeName = opts.routeName;
|
|
164
|
+
ctx.reverse = createReverseFunction(
|
|
165
|
+
opts.routeMap,
|
|
166
|
+
opts.routeName,
|
|
167
|
+
opts.params ?? {},
|
|
168
|
+
) as RequestContext<TEnv>["reverse"];
|
|
169
|
+
}
|
|
170
|
+
return {
|
|
171
|
+
ctx: ctx as unknown as TestRequestContextObject<TEnv>,
|
|
172
|
+
request,
|
|
173
|
+
url,
|
|
174
|
+
variables,
|
|
175
|
+
stateCookieName,
|
|
176
|
+
};
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
/**
|
|
180
|
+
* What a run accumulated on the request context, surfaced as PUBLIC values so a
|
|
181
|
+
* test never has to cast through the `@internal` `ctx.res` / `ctx.cookies()` to
|
|
182
|
+
* assert what an action produced.
|
|
183
|
+
*/
|
|
184
|
+
export interface RunInRequestContextResult<T> {
|
|
185
|
+
/**
|
|
186
|
+
* The value `fn` returned (awaited), or `undefined` if `fn` threw — in which
|
|
187
|
+
* case the thrown value is on {@link thrown}. The snapshot below is captured
|
|
188
|
+
* either way.
|
|
189
|
+
*/
|
|
190
|
+
result: T | undefined;
|
|
191
|
+
/**
|
|
192
|
+
* The value `fn` threw, or `undefined` if it returned normally. Commonly a
|
|
193
|
+
* `Response` from `throw redirect(...)` / `throw notFound()` — the dominant
|
|
194
|
+
* cookie+flash case is an action that sets them then throws a redirect — so
|
|
195
|
+
* this (and the snapshot below) is observable WITHOUT wrapping the action in
|
|
196
|
+
* your own try/catch. NOTE: the value is captured, NOT re-thrown; assert on it
|
|
197
|
+
* for a throwing action.
|
|
198
|
+
*/
|
|
199
|
+
thrown: unknown;
|
|
200
|
+
/**
|
|
201
|
+
* A Response carrying the status, headers, and Set-Cookie the run set (via
|
|
202
|
+
* `cookies().set()`, `ctx.header()`, etc.). Assert Set-Cookie with
|
|
203
|
+
* `response.headers.getSetCookie()`. When `fn` threw a `Response` (a redirect),
|
|
204
|
+
* THIS is that Response with the accumulated Set-Cookie/headers merged in
|
|
205
|
+
* (mirroring how the framework merges them in production), so a redirect's
|
|
206
|
+
* Location AND the cookies it set are both observable here.
|
|
207
|
+
*/
|
|
208
|
+
response: Response;
|
|
209
|
+
/**
|
|
210
|
+
* The effective cookie view after the run: request cookies merged with
|
|
211
|
+
* anything the run set or deleted (last-write-wins), as `{ name: value }`.
|
|
212
|
+
*/
|
|
213
|
+
cookies: Record<string, string>;
|
|
214
|
+
/**
|
|
215
|
+
* The response headers the run set (via `ctx.header(...)`, plus a thrown
|
|
216
|
+
* redirect's `Location`), as a plain `{ name: value }` object — the same view
|
|
217
|
+
* as `response.headers`, but assertable like `cookies`/`locationState`.
|
|
218
|
+
* EXCLUDES `set-cookie` (use `cookies`, or `response.headers.getSetCookie()`).
|
|
219
|
+
* Header names are lowercased (HTTP headers are case-insensitive).
|
|
220
|
+
*/
|
|
221
|
+
headers: Record<string, string>;
|
|
222
|
+
/**
|
|
223
|
+
* Location state the run set via `ctx.setLocationState()` / `redirect({ state })`,
|
|
224
|
+
* resolved to the flat `{ key: value }` shape the client reads off
|
|
225
|
+
* `history.state` (empty object when none) — so a post-action flash ("Saved!")
|
|
226
|
+
* is assertable at the unit layer.
|
|
227
|
+
*/
|
|
228
|
+
locationState: Record<string, unknown>;
|
|
229
|
+
/**
|
|
230
|
+
* The resolved rango state cookie name seeded into the run (default
|
|
231
|
+
* `rango-state_router_0`, or composed from `opts.stateCookie`). Assert an
|
|
232
|
+
* action's `invalidateClientCache()` rotation against it without recomputing:
|
|
233
|
+
* `response.headers.getSetCookie().some((c) => c.startsWith(stateCookieName + "="))`.
|
|
234
|
+
*/
|
|
235
|
+
stateCookieName: string;
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
export function snapshotRunEffects<TEnv>(ctx: RequestContext<TEnv>): {
|
|
239
|
+
cookies: Record<string, string>;
|
|
240
|
+
locationState: Record<string, unknown>;
|
|
241
|
+
} {
|
|
242
|
+
return {
|
|
243
|
+
cookies: { ...ctx.cookies() },
|
|
244
|
+
locationState: resolveLocationStateEntries(ctx._locationState ?? []),
|
|
245
|
+
};
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
export function headersToObject(headers: Headers): Record<string, string> {
|
|
249
|
+
const out: Record<string, string> = {};
|
|
250
|
+
headers.forEach((value, name) => {
|
|
251
|
+
if (name.toLowerCase() === "set-cookie") return;
|
|
252
|
+
out[name] = value;
|
|
253
|
+
});
|
|
254
|
+
return out;
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
export function buildRunResponse<TEnv>(
|
|
258
|
+
ctx: RequestContext<TEnv>,
|
|
259
|
+
thrown: unknown,
|
|
260
|
+
): Response {
|
|
261
|
+
const stub = ctx.res;
|
|
262
|
+
if (thrown instanceof Response) {
|
|
263
|
+
const headers = new Headers(thrown.headers);
|
|
264
|
+
for (const cookie of stub.headers.getSetCookie()) {
|
|
265
|
+
headers.append("set-cookie", cookie);
|
|
266
|
+
}
|
|
267
|
+
stub.headers.forEach((value, name) => {
|
|
268
|
+
if (name.toLowerCase() === "set-cookie") return;
|
|
269
|
+
if (!headers.has(name)) headers.set(name, value);
|
|
270
|
+
});
|
|
271
|
+
return new Response(null, { status: thrown.status, headers });
|
|
272
|
+
}
|
|
273
|
+
return new Response(null, { status: stub.status, headers: stub.headers });
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
export function buildRunSnapshot<TEnv>(
|
|
277
|
+
ctx: RequestContext<TEnv>,
|
|
278
|
+
thrown: unknown,
|
|
279
|
+
stateCookieName: string,
|
|
280
|
+
): {
|
|
281
|
+
thrown: unknown;
|
|
282
|
+
response: Response;
|
|
283
|
+
cookies: Record<string, string>;
|
|
284
|
+
headers: Record<string, string>;
|
|
285
|
+
locationState: Record<string, unknown>;
|
|
286
|
+
stateCookieName: string;
|
|
287
|
+
} {
|
|
288
|
+
const { cookies, locationState } = snapshotRunEffects(ctx);
|
|
289
|
+
const response = buildRunResponse(ctx, thrown);
|
|
290
|
+
const headers = headersToObject(response.headers);
|
|
291
|
+
return { thrown, response, cookies, headers, locationState, stateCookieName };
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
/**
|
|
295
|
+
* Build a seeded RequestContext (via {@link createTestRequestContext}) and run
|
|
296
|
+
* `fn` inside it, so code under test that calls `getRequestContext()`,
|
|
297
|
+
* `cookies()`, or reads/mutates request headers resolves exactly as in
|
|
298
|
+
* production.
|
|
299
|
+
*
|
|
300
|
+
* This is the entry point for the advanced cases the unit wrappers
|
|
301
|
+
* (`runLoader` / `runMiddleware`) do not model — most notably a server ACTION
|
|
302
|
+
* that authenticates off the request cookie or sets a session cookie / flash:
|
|
303
|
+
* an action has no loader context, so `runLoader` is the wrong shape, yet it
|
|
304
|
+
* still needs a real request context to read the cookie and resolve
|
|
305
|
+
* `getRequestContext()`.
|
|
306
|
+
*
|
|
307
|
+
* Returns `{ result, thrown, response, cookies, headers, locationState }` so the
|
|
308
|
+
* action's OUTPUT (Set-Cookie, response headers, flash) is assertable without
|
|
309
|
+
* casting through the `@internal` `ctx.res` / `ctx.cookies()`. `fn` may be async — the context stays
|
|
310
|
+
* active across its awaits (AsyncLocalStorage), and the snapshot is captured
|
|
311
|
+
* whether `fn` returns OR throws. The throw path matters: the most common
|
|
312
|
+
* cookie+flash case is an auth action that sets a cookie + flash then
|
|
313
|
+
* `throw redirect(...)` on success — the thrown redirect is on `thrown` (NOT
|
|
314
|
+
* re-thrown) and its Location plus the cookies are on `response`/`cookies`.
|
|
315
|
+
*
|
|
316
|
+
* @example
|
|
317
|
+
* ```ts
|
|
318
|
+
* const { result, cookies, response, thrown } = await runInRequestContext(
|
|
319
|
+
* () => loginAction(input), // sets a session cookie, then `throw redirect("/app")`
|
|
320
|
+
* {
|
|
321
|
+
* env,
|
|
322
|
+
* request: new Request("https://app.test/", {
|
|
323
|
+
* headers: { Cookie: "sid=abc" },
|
|
324
|
+
* }),
|
|
325
|
+
* },
|
|
326
|
+
* );
|
|
327
|
+
* expect(cookies.session).toBe("new-token");
|
|
328
|
+
* expect(headers.location).toBe("/app"); // response headers as a plain object
|
|
329
|
+
* expect((thrown as Response).headers.get("Location")).toBe("/app");
|
|
330
|
+
* expect(response.headers.getSetCookie()).toContainEqual(
|
|
331
|
+
* expect.stringContaining("session="),
|
|
332
|
+
* );
|
|
333
|
+
* ```
|
|
334
|
+
*/
|
|
335
|
+
export async function runInRequestContext<T, TEnv = unknown>(
|
|
336
|
+
fn: (ctx: RequestContext<TEnv>) => T | Promise<T>,
|
|
337
|
+
opts: CreateTestContextOptions<TEnv> = {},
|
|
338
|
+
): Promise<RunInRequestContextResult<T>> {
|
|
339
|
+
const { ctx, stateCookieName } = createTestRequestContext<TEnv>(opts);
|
|
340
|
+
let result: T | undefined;
|
|
341
|
+
let thrown: unknown;
|
|
342
|
+
try {
|
|
343
|
+
result = (await runWithRequestContext(ctx, () => fn(ctx))) as T;
|
|
344
|
+
} catch (error) {
|
|
345
|
+
thrown = error;
|
|
346
|
+
}
|
|
347
|
+
return { result, ...buildRunSnapshot(ctx, thrown, stateCookieName) };
|
|
348
|
+
}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Side-effect module: define the webpack-style globals the vendored
|
|
3
|
+
* react-server-dom CLIENT deserializer reads at module-eval time.
|
|
4
|
+
*
|
|
5
|
+
* In a real app the plugin-rsc Vite plugin rewrites `__webpack_require__` ->
|
|
6
|
+
* `__vite_rsc_require__` and `__webpack_require__.u` -> `({}).u`
|
|
7
|
+
* (@vitejs/plugin-rsc `core/plugin.js`). That transform does NOT run in a bare
|
|
8
|
+
* Vitest process, so the vendored client's free `__webpack_require__` /
|
|
9
|
+
* `__webpack_chunk_load__` references would be undefined. We provide minimal
|
|
10
|
+
* shims: `__webpack_require__` routes to the loader installed via
|
|
11
|
+
* `setRequireModule`, and `__webpack_chunk_load__` is a no-op (renderServerTree
|
|
12
|
+
* serializes with empty `chunks`, so no chunk fetch ever happens).
|
|
13
|
+
*
|
|
14
|
+
* MUST be imported (for side effect) BEFORE `@vitejs/plugin-rsc/react/browser`,
|
|
15
|
+
* which is why flight-tree.ts lists it first.
|
|
16
|
+
*/
|
|
17
|
+
const g = globalThis as unknown as {
|
|
18
|
+
__webpack_require__?: ((id: string) => unknown) & { u?: unknown };
|
|
19
|
+
__webpack_chunk_load__?: (chunkId: string) => Promise<unknown>;
|
|
20
|
+
__vite_rsc_client_require__?: (id: string) => unknown;
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
if (!g.__webpack_require__) {
|
|
24
|
+
g.__webpack_require__ = (id: string) => g.__vite_rsc_client_require__!(id);
|
|
25
|
+
}
|
|
26
|
+
if (!g.__webpack_chunk_load__) {
|
|
27
|
+
g.__webpack_chunk_load__ = async () => {};
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
export {};
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Variable seeding shared by the node/DOM testing tier (internal/context.ts)
|
|
3
|
+
* AND the react-server Flight tier (flight.ts). Depends only on the
|
|
4
|
+
* dependency-free `context-var` module and the env-agnostic state-cookie-name
|
|
5
|
+
* composition (no window/document), so it is safe to import under the
|
|
6
|
+
* `react-server` condition (unlike internal/context.ts, which pulls
|
|
7
|
+
* client/browser modules).
|
|
8
|
+
*/
|
|
9
|
+
import { contextSet, type ContextVar } from "../../context-var.js";
|
|
10
|
+
import { resolveStateCookieName } from "../../router/state-cookie-name.js";
|
|
11
|
+
|
|
12
|
+
export interface StateCookieSeed {
|
|
13
|
+
/**
|
|
14
|
+
* Cookie-name prefix, sanitized then composed with `routerId` exactly like
|
|
15
|
+
* `createRouter({ stateCookiePrefix })`. Defaults to `"rango-state"`.
|
|
16
|
+
*/
|
|
17
|
+
prefix?: string;
|
|
18
|
+
/**
|
|
19
|
+
* Router id; the resolved name is `{sanitizedPrefix}_{sanitizedRouterId}`.
|
|
20
|
+
* Defaults to `"router_0"` (the name a single default router resolves to), so
|
|
21
|
+
* the default name is `rango-state_router_0`.
|
|
22
|
+
*/
|
|
23
|
+
routerId?: string;
|
|
24
|
+
/**
|
|
25
|
+
* Build version used as the rotated value's prefix (`{version}:{timestamp}`).
|
|
26
|
+
* Defaults to `"0"` (resolved inside createRequestContext).
|
|
27
|
+
*/
|
|
28
|
+
version?: string;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export function resolveSeededStateCookieName(seed?: StateCookieSeed): string {
|
|
32
|
+
return resolveStateCookieName(seed?.prefix, seed?.routerId ?? "router_0");
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export type VarsInit =
|
|
36
|
+
| Record<string, unknown>
|
|
37
|
+
| ReadonlyArray<readonly [ContextVar<unknown> | string, unknown]>;
|
|
38
|
+
|
|
39
|
+
export function seedVariables(
|
|
40
|
+
variables: Record<string, unknown>,
|
|
41
|
+
vars?: VarsInit,
|
|
42
|
+
): Record<string, unknown> {
|
|
43
|
+
if (!vars) return variables;
|
|
44
|
+
const entries: Iterable<readonly [ContextVar<unknown> | string, unknown]> =
|
|
45
|
+
Symbol.iterator in (vars as object)
|
|
46
|
+
? (vars as ReadonlyArray<
|
|
47
|
+
readonly [ContextVar<unknown> | string, unknown]
|
|
48
|
+
>)
|
|
49
|
+
: Object.entries(vars as Record<string, unknown>);
|
|
50
|
+
for (const [key, value] of entries) {
|
|
51
|
+
contextSet(variables, key as ContextVar<unknown>, value);
|
|
52
|
+
}
|
|
53
|
+
return variables;
|
|
54
|
+
}
|