@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,76 @@
|
|
|
1
|
+
// Node ESM loader hook that resolves `cloudflare:*` imports to the same
|
|
2
|
+
// stub ESM the Vite transform produces for rewritten specifiers.
|
|
3
|
+
//
|
|
4
|
+
// Why both? The Vite transform (cloudflare-protocol-stub.ts) catches
|
|
5
|
+
// imports in modules that flow through Vite's plugin pipeline — covers
|
|
6
|
+
// user source and any node_modules package Vite fetches and transforms.
|
|
7
|
+
// But Vite/Rollup externalize certain packages (e.g. `partyserver`,
|
|
8
|
+
// which has `import { DurableObject, env } from "cloudflare:workers"`
|
|
9
|
+
// at its top level, and similar "workerd-native" libraries). Externalized
|
|
10
|
+
// modules bypass the transform: Rollup hands their resolution to Node's
|
|
11
|
+
// native ESM loader, which rejects URL-scheme specifiers. This loader
|
|
12
|
+
// hook registers via `module.register()` from `createTempRscServer` and
|
|
13
|
+
// intercepts `cloudflare:*` at Node's resolve layer — before the default
|
|
14
|
+
// loader throws ERR_UNSUPPORTED_ESM_URL_SCHEME.
|
|
15
|
+
//
|
|
16
|
+
// Lifecycle: the hook runs in a dedicated worker thread (Node ESM loader
|
|
17
|
+
// architecture) with its own globalThis. It cannot see the main thread's
|
|
18
|
+
// `__rango_build_env__` bridge, so the `env` export here is always `{}`.
|
|
19
|
+
// That's fine in practice — externalized libraries don't typically touch
|
|
20
|
+
// `env` at module top level; they read it at request time in workerd
|
|
21
|
+
// where the real module exists. Build-time prerender handlers in user
|
|
22
|
+
// source DO read `env`, but they flow through the Vite transform (which
|
|
23
|
+
// does bridge `env` from `getPlatformProxy()`), not through this loader.
|
|
24
|
+
//
|
|
25
|
+
// Keep STUBS in sync with cloudflare-protocol-stub.ts — both paths need
|
|
26
|
+
// to hand out the same base classes.
|
|
27
|
+
|
|
28
|
+
const CF_PREFIX = "cloudflare:";
|
|
29
|
+
|
|
30
|
+
const STUBS = {
|
|
31
|
+
"cloudflare:workers": `
|
|
32
|
+
export class DurableObject { constructor(_ctx, _env) {} }
|
|
33
|
+
export class WorkerEntrypoint { constructor(_ctx, _env) {} }
|
|
34
|
+
export class WorkflowEntrypoint { constructor(_ctx, _env) {} }
|
|
35
|
+
export class RpcTarget {}
|
|
36
|
+
export const env = {};
|
|
37
|
+
export default {};
|
|
38
|
+
`,
|
|
39
|
+
"cloudflare:email": `
|
|
40
|
+
export class EmailMessage { constructor(_from, _to, _raw) {} }
|
|
41
|
+
export default {};
|
|
42
|
+
`,
|
|
43
|
+
"cloudflare:sockets": `
|
|
44
|
+
export function connect() { return {}; }
|
|
45
|
+
export default {};
|
|
46
|
+
`,
|
|
47
|
+
"cloudflare:workflows": `
|
|
48
|
+
export class NonRetryableError extends Error {
|
|
49
|
+
constructor(message, name) { super(message); this.name = name ?? "NonRetryableError"; }
|
|
50
|
+
}
|
|
51
|
+
export default {};
|
|
52
|
+
`,
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
// Policy: unknown `cloudflare:*` specifiers resolve permissively to an
|
|
56
|
+
// empty default export rather than throwing. Same reasoning as
|
|
57
|
+
// cloudflare-protocol-stub.ts's FALLBACK_STUB — we prioritize
|
|
58
|
+
// dependency-graph resilience over strict validation, because third-party
|
|
59
|
+
// packages can pull `cloudflare:*` modules we haven't curated.
|
|
60
|
+
const FALLBACK_STUB = `export default {};\n`;
|
|
61
|
+
|
|
62
|
+
function dataUrlFor(specifier) {
|
|
63
|
+
const body = STUBS[specifier] ?? FALLBACK_STUB;
|
|
64
|
+
return "data:text/javascript;base64," + Buffer.from(body).toString("base64");
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
export async function resolve(specifier, context, nextResolve) {
|
|
68
|
+
if (specifier.startsWith(CF_PREFIX)) {
|
|
69
|
+
return {
|
|
70
|
+
shortCircuit: true,
|
|
71
|
+
url: dataUrlFor(specifier),
|
|
72
|
+
format: "module",
|
|
73
|
+
};
|
|
74
|
+
}
|
|
75
|
+
return nextResolve(specifier, context);
|
|
76
|
+
}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@rangojs/router",
|
|
3
|
-
"version": "0.0.0-experimental.
|
|
3
|
+
"version": "0.0.0-experimental.3232cd17",
|
|
4
4
|
"description": "Django-inspired RSC router with composable URL patterns",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"react",
|
|
@@ -126,51 +126,103 @@
|
|
|
126
126
|
"./host/testing": {
|
|
127
127
|
"types": "./src/host/testing.ts",
|
|
128
128
|
"default": "./src/host/testing.ts"
|
|
129
|
+
},
|
|
130
|
+
"./testing": {
|
|
131
|
+
"types": "./src/testing/index.ts",
|
|
132
|
+
"default": "./src/testing/index.ts"
|
|
133
|
+
},
|
|
134
|
+
"./testing/vitest": {
|
|
135
|
+
"types": "./src/testing/vitest.ts",
|
|
136
|
+
"default": "./dist/testing/vitest.js"
|
|
137
|
+
},
|
|
138
|
+
"./testing/dom": {
|
|
139
|
+
"types": "./src/testing/dom.entry.ts",
|
|
140
|
+
"default": "./src/testing/dom.entry.ts"
|
|
141
|
+
},
|
|
142
|
+
"./testing/e2e": {
|
|
143
|
+
"types": "./src/testing/e2e/index.ts",
|
|
144
|
+
"default": "./src/testing/e2e/index.ts"
|
|
145
|
+
},
|
|
146
|
+
"./testing/flight": {
|
|
147
|
+
"types": "./src/testing/flight.entry.ts",
|
|
148
|
+
"react-server": "./src/testing/flight.entry.ts",
|
|
149
|
+
"default": "./src/testing/flight.entry.ts"
|
|
150
|
+
},
|
|
151
|
+
"./testing/flight-matchers": {
|
|
152
|
+
"types": "./src/testing/flight-matchers.ts",
|
|
153
|
+
"default": "./src/testing/flight-matchers.ts"
|
|
129
154
|
}
|
|
130
155
|
},
|
|
131
156
|
"publishConfig": {
|
|
132
157
|
"access": "public",
|
|
133
158
|
"tag": "experimental"
|
|
134
159
|
},
|
|
160
|
+
"scripts": {
|
|
161
|
+
"build": "pnpm dlx esbuild src/vite/index.ts --bundle --format=esm --outfile=dist/vite/index.js --platform=node --packages=external && mkdir -p dist/vite/plugins && cp src/vite/plugins/cloudflare-protocol-loader-hook.mjs dist/vite/plugins/cloudflare-protocol-loader-hook.mjs && pnpm dlx esbuild src/testing/vitest.ts --bundle --format=esm --outfile=dist/testing/vitest.js --platform=node --packages=external && pnpm dlx esbuild src/bin/rango.ts --bundle --format=esm --outfile=dist/bin/rango.js --platform=node --packages=external --banner:js='#!/usr/bin/env node' && chmod +x dist/bin/rango.js",
|
|
162
|
+
"prepublishOnly": "pnpm build",
|
|
163
|
+
"typecheck": "tsc --noEmit && tsc -p tsconfig.strict-check.json --noEmit && tsc -p tsconfig.augment-check.json --noEmit",
|
|
164
|
+
"test": "playwright test",
|
|
165
|
+
"test:ui": "playwright test --ui",
|
|
166
|
+
"test:hmr-local": "playwright test --project=dev-warmup --project=hmr-routes --project=hmr-basename --project=hmr-prerender --no-deps --workers=1",
|
|
167
|
+
"test:unit": "vitest run",
|
|
168
|
+
"test:unit:watch": "vitest",
|
|
169
|
+
"test:unit:rsc": "vitest run --config vitest.rsc.config.ts"
|
|
170
|
+
},
|
|
135
171
|
"dependencies": {
|
|
136
|
-
"@
|
|
172
|
+
"@types/debug": "^4.1.12",
|
|
173
|
+
"@vitejs/plugin-rsc": "^0.5.26",
|
|
174
|
+
"debug": "^4.4.1",
|
|
137
175
|
"magic-string": "^0.30.17",
|
|
138
176
|
"picomatch": "^4.0.3",
|
|
139
|
-
"rsc-html-stream": "^0.0.7"
|
|
177
|
+
"rsc-html-stream": "^0.0.7",
|
|
178
|
+
"srvx": "^0.11.15",
|
|
179
|
+
"tinyexec": "^0.3.2"
|
|
140
180
|
},
|
|
141
181
|
"devDependencies": {
|
|
142
182
|
"@playwright/test": "^1.49.1",
|
|
183
|
+
"@shared/e2e": "workspace:*",
|
|
184
|
+
"@testing-library/dom": "^10.4.1",
|
|
185
|
+
"@testing-library/react": "^16.3.2",
|
|
143
186
|
"@types/node": "^24.10.1",
|
|
144
|
-
"@types/react": "
|
|
145
|
-
"@types/react-dom": "
|
|
187
|
+
"@types/react": "catalog:",
|
|
188
|
+
"@types/react-dom": "catalog:",
|
|
146
189
|
"esbuild": "^0.27.0",
|
|
190
|
+
"happy-dom": "^20.10.1",
|
|
147
191
|
"jiti": "^2.6.1",
|
|
148
|
-
"react": "
|
|
149
|
-
"react-dom": "
|
|
150
|
-
"tinyexec": "^0.3.2",
|
|
192
|
+
"react": "catalog:",
|
|
193
|
+
"react-dom": "catalog:",
|
|
151
194
|
"typescript": "^5.3.0",
|
|
152
195
|
"vitest": "^4.0.0"
|
|
153
196
|
},
|
|
154
197
|
"peerDependencies": {
|
|
155
|
-
"@cloudflare/vite-plugin": "^1.
|
|
156
|
-
"@
|
|
157
|
-
"react": "
|
|
158
|
-
"
|
|
198
|
+
"@cloudflare/vite-plugin": "^1.38.0",
|
|
199
|
+
"@playwright/test": "^1.49.1",
|
|
200
|
+
"@testing-library/react": ">=16",
|
|
201
|
+
"@vercel/functions": "^3.0.0",
|
|
202
|
+
"@vitejs/plugin-rsc": "^0.5.26",
|
|
203
|
+
"react": ">=19.2.6 <20",
|
|
204
|
+
"react-dom": ">=19.2.6 <20",
|
|
205
|
+
"vite": "^8.0.0",
|
|
206
|
+
"vitest": ">=3"
|
|
159
207
|
},
|
|
160
208
|
"peerDependenciesMeta": {
|
|
161
209
|
"@cloudflare/vite-plugin": {
|
|
162
210
|
"optional": true
|
|
163
211
|
},
|
|
212
|
+
"@playwright/test": {
|
|
213
|
+
"optional": true
|
|
214
|
+
},
|
|
215
|
+
"@testing-library/react": {
|
|
216
|
+
"optional": true
|
|
217
|
+
},
|
|
218
|
+
"@vercel/functions": {
|
|
219
|
+
"optional": true
|
|
220
|
+
},
|
|
164
221
|
"vite": {
|
|
165
222
|
"optional": true
|
|
223
|
+
},
|
|
224
|
+
"vitest": {
|
|
225
|
+
"optional": true
|
|
166
226
|
}
|
|
167
|
-
},
|
|
168
|
-
"scripts": {
|
|
169
|
-
"build": "pnpm dlx esbuild src/vite/index.ts --bundle --format=esm --outfile=dist/vite/index.js --platform=node --packages=external && pnpm dlx esbuild src/bin/rango.ts --bundle --format=esm --outfile=dist/bin/rango.js --platform=node --packages=external --banner:js='#!/usr/bin/env node' && chmod +x dist/bin/rango.js",
|
|
170
|
-
"typecheck": "tsc --noEmit",
|
|
171
|
-
"test": "playwright test",
|
|
172
|
-
"test:ui": "playwright test --ui",
|
|
173
|
-
"test:unit": "vitest run",
|
|
174
|
-
"test:unit:watch": "vitest"
|
|
175
227
|
}
|
|
176
|
-
}
|
|
228
|
+
}
|
|
@@ -0,0 +1,211 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: api-client
|
|
3
|
+
description: Build a typed client for consuming your own response-route JSON APIs (no codegen)
|
|
4
|
+
---
|
|
5
|
+
|
|
6
|
+
# Typed API Client
|
|
7
|
+
|
|
8
|
+
Response routes (`path.json()`) already ship typed responses — `RouteResponse<typeof patterns, "name">` resolves to the **bare payload**, inferred from your handler with no codegen. This skill wraps that inference in a small **typed client** so first-party TypeScript code calls your endpoints like functions instead of hand-writing `fetch` + URL building per call site.
|
|
9
|
+
|
|
10
|
+
This is a **recipe, not a framework feature** — copy the helper below into your app. It depends only on **type-only** imports from `@rangojs/router` (`RouteResponse`, `ExtractParams`, `ProblemDetails`), which are erased at build time, so it runs anywhere a `fetch` does — **browser, worker, or server**. Nothing new to install or version.
|
|
11
|
+
|
|
12
|
+
> **Scope:** the typed client is a **first-party TypeScript** convenience. External/third-party consumers use the plain wire directly — bare JSON on success, RFC 9457 `application/problem+json` on error — which needs no client. (Language-agnostic OpenAPI generation is a separate, future feature.)
|
|
13
|
+
|
|
14
|
+
## What you get
|
|
15
|
+
|
|
16
|
+
```ts
|
|
17
|
+
const api = createApiClient(apiShopPatterns, routes, { baseUrl });
|
|
18
|
+
|
|
19
|
+
await api.health.get(); // no params → callable bare
|
|
20
|
+
await api.product.get({ params: { productId } }); // params typed + required
|
|
21
|
+
await api.cart.post({ body: { productId, qty: 2 } }); // body sent as JSON
|
|
22
|
+
// ^ result is the bare payload type (RouteResponse), not `any`, no `.data`
|
|
23
|
+
```
|
|
24
|
+
|
|
25
|
+
- **Output typed** from the handler's return (`RouteResponse`), zero codegen.
|
|
26
|
+
- **Params required + typed** from the route pattern (`/catalog/:productId` → `{ productId: string }`); a missing or misspelled param is a **compile error**, not a runtime 404.
|
|
27
|
+
- **Autocomplete** over every route name; rename-safe.
|
|
28
|
+
- **Errors throw a typed `ApiError`** carrying the `ProblemDetails` body.
|
|
29
|
+
|
|
30
|
+
(`search` and `body` are _not_ route-typed — see Notes.)
|
|
31
|
+
|
|
32
|
+
## The two inputs
|
|
33
|
+
|
|
34
|
+
1. **The `urls()` patterns value** — the type source. `typeof apiShopPatterns` carries the per-route response payloads (`_responses`) and patterns (`_routes`).
|
|
35
|
+
2. **The generated route map** — the name → pattern source. `rango generate` emits a per-module `<name>.gen.ts` exporting `routes`:
|
|
36
|
+
|
|
37
|
+
```ts
|
|
38
|
+
// api-shop.gen.ts (generated — do not edit)
|
|
39
|
+
export const routes = {
|
|
40
|
+
catalog: "/catalog",
|
|
41
|
+
product: "/catalog/:productId",
|
|
42
|
+
cart: "/cart",
|
|
43
|
+
// ...
|
|
44
|
+
} as const;
|
|
45
|
+
```
|
|
46
|
+
|
|
47
|
+
Routes that declare a **search schema** are generated as objects instead — `index: { path: "/", search: { q: "string" } }`. The helper accepts both the string and `{ path }` forms. If a `urls()` block is mounted under a name prefix, build a local-keyed map from your global `NamedRoutes` so the keys match the block's route names (e.g. `{ catalog: NamedRoutes["apiShop.catalog"], ... } as const`).
|
|
48
|
+
|
|
49
|
+
## The helper (copy into your app)
|
|
50
|
+
|
|
51
|
+
```ts
|
|
52
|
+
// lib/api-client.ts
|
|
53
|
+
import type {
|
|
54
|
+
RouteResponse,
|
|
55
|
+
ExtractParams,
|
|
56
|
+
ProblemDetails,
|
|
57
|
+
} from "@rangojs/router";
|
|
58
|
+
|
|
59
|
+
type SearchParams = Record<string, string | number | boolean>;
|
|
60
|
+
|
|
61
|
+
// A generated route-map entry is a pattern string, or an object with `path`
|
|
62
|
+
// (routes that declare a search schema generate the object form).
|
|
63
|
+
type RouteMapEntry = string | { readonly path: string };
|
|
64
|
+
type PatternOf<E> = E extends string
|
|
65
|
+
? E
|
|
66
|
+
: E extends { readonly path: infer P extends string }
|
|
67
|
+
? P
|
|
68
|
+
: never;
|
|
69
|
+
|
|
70
|
+
// `params` is optional when the route has no *required* params (incl.
|
|
71
|
+
// optional-only routes like `/:locale?`), required otherwise. Typed as
|
|
72
|
+
// `ExtractParams` (not `undefined`) so optional params can still be passed.
|
|
73
|
+
type Args<TPattern extends string> =
|
|
74
|
+
{} extends ExtractParams<TPattern>
|
|
75
|
+
? {
|
|
76
|
+
params?: ExtractParams<TPattern>;
|
|
77
|
+
search?: SearchParams;
|
|
78
|
+
body?: unknown;
|
|
79
|
+
}
|
|
80
|
+
: {
|
|
81
|
+
params: ExtractParams<TPattern>;
|
|
82
|
+
search?: SearchParams;
|
|
83
|
+
body?: unknown;
|
|
84
|
+
};
|
|
85
|
+
|
|
86
|
+
type Method<TPatterns, K extends string, TEntry> =
|
|
87
|
+
{} extends ExtractParams<PatternOf<TEntry>>
|
|
88
|
+
? (args?: Args<PatternOf<TEntry>>) => Promise<RouteResponse<TPatterns, K>>
|
|
89
|
+
: (args: Args<PatternOf<TEntry>>) => Promise<RouteResponse<TPatterns, K>>;
|
|
90
|
+
|
|
91
|
+
type ApiClient<TPatterns, TRouteMap extends Record<string, RouteMapEntry>> = {
|
|
92
|
+
[K in keyof TRouteMap & string]: {
|
|
93
|
+
get: Method<TPatterns, K, TRouteMap[K]>;
|
|
94
|
+
post: Method<TPatterns, K, TRouteMap[K]>;
|
|
95
|
+
put: Method<TPatterns, K, TRouteMap[K]>;
|
|
96
|
+
patch: Method<TPatterns, K, TRouteMap[K]>;
|
|
97
|
+
delete: Method<TPatterns, K, TRouteMap[K]>;
|
|
98
|
+
};
|
|
99
|
+
};
|
|
100
|
+
|
|
101
|
+
/** Thrown on a non-2xx response; carries the RFC 9457 problem body. */
|
|
102
|
+
export class ApiError extends Error {
|
|
103
|
+
status: number;
|
|
104
|
+
problem: ProblemDetails;
|
|
105
|
+
constructor(status: number, problem: ProblemDetails) {
|
|
106
|
+
super(problem.detail || `HTTP ${status}`);
|
|
107
|
+
this.name = "ApiError";
|
|
108
|
+
this.status = status;
|
|
109
|
+
this.problem = problem;
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
// Client-safe path builder: substitutes :params (incl. optional/constrained
|
|
114
|
+
// forms) into the pattern. No dependency on the server-only createReverse.
|
|
115
|
+
function fillPath(pattern: string, params?: Record<string, string>): string {
|
|
116
|
+
return pattern
|
|
117
|
+
.replace(/:([A-Za-z0-9_]+)(?:\([^)]*\))?\??/g, (_m, name: string) => {
|
|
118
|
+
const v = params?.[name];
|
|
119
|
+
return v == null ? "" : encodeURIComponent(String(v));
|
|
120
|
+
})
|
|
121
|
+
.replace(/\/{2,}/g, "/");
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
export function createApiClient<
|
|
125
|
+
TPatterns,
|
|
126
|
+
const TRouteMap extends Record<string, RouteMapEntry>,
|
|
127
|
+
>(
|
|
128
|
+
_patterns: TPatterns,
|
|
129
|
+
routeMap: TRouteMap,
|
|
130
|
+
opts: { baseUrl?: string; fetch?: typeof fetch } = {},
|
|
131
|
+
): ApiClient<TPatterns, TRouteMap> {
|
|
132
|
+
const doFetch = opts.fetch ?? fetch;
|
|
133
|
+
const baseUrl = opts.baseUrl ?? "";
|
|
134
|
+
const call =
|
|
135
|
+
(name: string, method: string) =>
|
|
136
|
+
async (args?: {
|
|
137
|
+
params?: Record<string, string>;
|
|
138
|
+
search?: SearchParams;
|
|
139
|
+
body?: unknown;
|
|
140
|
+
}) => {
|
|
141
|
+
const entry = routeMap[name];
|
|
142
|
+
const pattern = typeof entry === "string" ? entry : entry.path;
|
|
143
|
+
let url = baseUrl + fillPath(pattern, args?.params);
|
|
144
|
+
if (args?.search) {
|
|
145
|
+
const qs = new URLSearchParams();
|
|
146
|
+
for (const [k, v] of Object.entries(args.search)) {
|
|
147
|
+
if (v != null) qs.append(k, String(v));
|
|
148
|
+
}
|
|
149
|
+
const s = qs.toString();
|
|
150
|
+
if (s) url += (url.includes("?") ? "&" : "?") + s;
|
|
151
|
+
}
|
|
152
|
+
const res = await doFetch(url, {
|
|
153
|
+
method,
|
|
154
|
+
...(args?.body !== undefined
|
|
155
|
+
? {
|
|
156
|
+
body: JSON.stringify(args.body),
|
|
157
|
+
headers: { "content-type": "application/json" },
|
|
158
|
+
}
|
|
159
|
+
: {}),
|
|
160
|
+
});
|
|
161
|
+
if (!res.ok) {
|
|
162
|
+
const problem = (await res.json().catch(() => ({}))) as ProblemDetails;
|
|
163
|
+
throw new ApiError(res.status, problem);
|
|
164
|
+
}
|
|
165
|
+
return res.json();
|
|
166
|
+
};
|
|
167
|
+
return new Proxy({} as any, {
|
|
168
|
+
get: (_t, name: string) => ({
|
|
169
|
+
get: call(name, "GET"),
|
|
170
|
+
post: call(name, "POST"),
|
|
171
|
+
put: call(name, "PUT"),
|
|
172
|
+
patch: call(name, "PATCH"),
|
|
173
|
+
delete: call(name, "DELETE"),
|
|
174
|
+
}),
|
|
175
|
+
}) as ApiClient<TPatterns, TRouteMap>;
|
|
176
|
+
}
|
|
177
|
+
```
|
|
178
|
+
|
|
179
|
+
## Using it
|
|
180
|
+
|
|
181
|
+
```ts
|
|
182
|
+
import { apiShopPatterns } from "./urls/api-shop";
|
|
183
|
+
import { routes } from "./urls/api-shop.gen";
|
|
184
|
+
import { createApiClient, ApiError } from "./lib/api-client";
|
|
185
|
+
|
|
186
|
+
const api = createApiClient(apiShopPatterns, routes, {
|
|
187
|
+
baseUrl: import.meta.env.VITE_API_URL ?? "",
|
|
188
|
+
});
|
|
189
|
+
|
|
190
|
+
try {
|
|
191
|
+
const product = await api.product.get({ params: { productId: "42" } });
|
|
192
|
+
// `product` is the handler's bare return type — e.g. `product.name` is typed.
|
|
193
|
+
} catch (err) {
|
|
194
|
+
if (err instanceof ApiError && err.status === 404) {
|
|
195
|
+
console.warn(err.problem.code, err.problem.detail); // typed ProblemDetails
|
|
196
|
+
} else {
|
|
197
|
+
throw err;
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
```
|
|
201
|
+
|
|
202
|
+
## Notes
|
|
203
|
+
|
|
204
|
+
- **Client-safe by construction.** The helper imports only **types** from `@rangojs/router` (erased at build) and builds URLs itself by substituting `:params` into the pattern — it does **not** use `createReverse`, which is a server/RSC-only export that throws in the browser. So `createApiClient` works in client components, workers, and on the server alike.
|
|
205
|
+
- **Params are route-typed; search and body are not.** Path params come from the route pattern (`ExtractParams`), so they are precise and required. `search` is generically typed (`Record<string, string | number | boolean>`), and `body` is `unknown` (serialized to JSON). Typed request **input** needs a declared schema layer, which is intentionally out of scope here — thread per-route schemas in yourself if you want typed search/body.
|
|
206
|
+
- **Verb-agnostic wire.** Rango response routes do not dispatch on HTTP method — `.get`/`.post`/etc. set the request method but hit the same handler. Use whichever verb reads best for the operation.
|
|
207
|
+
- **Path building.** `fillPath` handles standard `:param`, optional `:param?`, and constrained `:param(a|b)` forms. For exotic patterns or strict trailing-slash policies, swap in your own builder (or the router's `reverse` on the server).
|
|
208
|
+
- **Want a return-based style instead of throwing?** Branch on `res.ok` yourself: the wire is the bare value on 2xx and `ProblemDetails` on non-2xx (see `/response-routes`). Wrapping the calls in a `{ ok, data } | { ok: false, error }` result type is a small variation on the same helper.
|
|
209
|
+
- **Third parties.** The typed client is TypeScript-only and needs your route types. External consumers in any language use the plain wire as-is (bare JSON + problem+json); no client required.
|
|
210
|
+
|
|
211
|
+
See `/response-routes` for the endpoint side and `/typesafety` for how `RouteResponse` / `PathResponse` inference works.
|
|
@@ -81,6 +81,66 @@ path("/product/:id", async (ctx) => {
|
|
|
81
81
|
Async content is a `Promise<ReactNode>`. Resolve it in your component
|
|
82
82
|
with React's `use()` hook wrapped in `<Suspense>`.
|
|
83
83
|
|
|
84
|
+
### Deferred content (decide now, resolve from a deep component)
|
|
85
|
+
|
|
86
|
+
When the handler should DECIDE to push a crumb (it holds `ctx`, so the decision
|
|
87
|
+
must land before the handles stream seals) but the value is produced far away — by
|
|
88
|
+
a deep async component, not the handler — call `.defer()` on the push function.
|
|
89
|
+
`ctx.use(Handle)` returns the push function; `.defer(options)` reserves the crumb's
|
|
90
|
+
slot synchronously and returns a **resolver that is push-equal** — you call it
|
|
91
|
+
later, anywhere in the render, with the same argument you'd have passed to the
|
|
92
|
+
push (a value, a `Promise`, or a thunk). The only added behavior is a timeout, so a
|
|
93
|
+
forgotten resolve can't hold the Flight stream (and the HTTP response) open forever.
|
|
94
|
+
|
|
95
|
+
Reserve the slot in the handler, then resolve it from a nested async component
|
|
96
|
+
that closes over the resolver — no extra wiring (the resolver is a plain closure,
|
|
97
|
+
not outlet context):
|
|
98
|
+
|
|
99
|
+
```tsx
|
|
100
|
+
import { Breadcrumbs } from "@rangojs/router";
|
|
101
|
+
import { Outlet } from "@rangojs/router/client";
|
|
102
|
+
import { Suspense } from "react";
|
|
103
|
+
|
|
104
|
+
function DocsLayout(ctx) {
|
|
105
|
+
const breadcrumb = ctx.use(Breadcrumbs);
|
|
106
|
+
// Decide now (the slot is reserved before the stream seals); resolve later.
|
|
107
|
+
const resolveCrumb = breadcrumb.defer({ timeoutMs: 5000, else: null });
|
|
108
|
+
|
|
109
|
+
// Deep, async, far from the handler — closes over the resolver, never touches ctx.
|
|
110
|
+
// Same call shape as breadcrumb({ ... }), just deferred:
|
|
111
|
+
async function LiveCrumb() {
|
|
112
|
+
const n = await countOpenIssues();
|
|
113
|
+
resolveCrumb({ label: "Docs", href: "/docs", content: <span>{n}</span> });
|
|
114
|
+
return null;
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
return (
|
|
118
|
+
<>
|
|
119
|
+
<Suspense>
|
|
120
|
+
<LiveCrumb />
|
|
121
|
+
</Suspense>
|
|
122
|
+
<Outlet />
|
|
123
|
+
</>
|
|
124
|
+
);
|
|
125
|
+
}
|
|
126
|
+
```
|
|
127
|
+
|
|
128
|
+
If the resolver is never called, the slot auto-resolves to `else` after
|
|
129
|
+
`timeoutMs` (default 10s) and warns in dev — graceful degradation instead of a
|
|
130
|
+
hung request. `timeoutMs: 0` or `Infinity` disable the timeout intentionally; any
|
|
131
|
+
other non-finite or negative value falls back to the default rather than silently
|
|
132
|
+
disabling the safety net.
|
|
133
|
+
|
|
134
|
+
**Consumer note:** because `.defer()` reserves the slot for the WHOLE item, a
|
|
135
|
+
client reading the handle (`useHandle(Breadcrumbs)`) sees that entry as a
|
|
136
|
+
`Promise` until it resolves. Type such reads with the exported
|
|
137
|
+
`DeferredHandleEntry<BreadcrumbItem>` (from `@rangojs/router/client`); a
|
|
138
|
+
deferred-aware consumer should `use()` thenable entries inside `<Suspense>`, while
|
|
139
|
+
a simple one can skip them (`typeof entry.then === "function"`). Use `.defer()`
|
|
140
|
+
only when even `label`/`href` are unknown at handler time — if you know them and
|
|
141
|
+
only the `content` is async, push a concrete item with a `Promise` `content` field
|
|
142
|
+
instead (no `.defer()` needed).
|
|
143
|
+
|
|
84
144
|
## Consuming Breadcrumbs (Client)
|
|
85
145
|
|
|
86
146
|
Use `useHandle(Breadcrumbs)` in a client component to read the accumulated items:
|
|
@@ -141,9 +201,11 @@ path("/dashboard", (ctx) => {
|
|
|
141
201
|
breadcrumb({ label: "Dashboard", href: "/dashboard" });
|
|
142
202
|
return <DashboardNav handle={Breadcrumbs} />;
|
|
143
203
|
});
|
|
204
|
+
```
|
|
144
205
|
|
|
206
|
+
```tsx
|
|
145
207
|
// Client component
|
|
146
|
-
|
|
208
|
+
"use client";
|
|
147
209
|
import { useHandle, type Breadcrumbs } from "@rangojs/router/client";
|
|
148
210
|
|
|
149
211
|
function DashboardNav({ handle }: { handle: typeof Breadcrumbs }) {
|
|
@@ -204,3 +266,47 @@ export const urlpatterns = urls(({ path, layout }) => [
|
|
|
204
266
|
```
|
|
205
267
|
|
|
206
268
|
Navigating to `/shop/widget` produces: `Home / Shop / widget`
|
|
269
|
+
|
|
270
|
+
## Custom Handles
|
|
271
|
+
|
|
272
|
+
Create your own handle with `createHandle()`:
|
|
273
|
+
|
|
274
|
+
```typescript
|
|
275
|
+
import { createHandle } from "@rangojs/router";
|
|
276
|
+
|
|
277
|
+
// Default: flatten into array
|
|
278
|
+
export const PageTitle = createHandle<string, string>(
|
|
279
|
+
(segments) => segments.flat().at(-1) ?? "Default Title",
|
|
280
|
+
);
|
|
281
|
+
|
|
282
|
+
// No collect function: default flattens into T[]
|
|
283
|
+
export const Warnings = createHandle<string>();
|
|
284
|
+
```
|
|
285
|
+
|
|
286
|
+
The Vite `exposeInternalIds` plugin auto-injects a stable `$$id` based on
|
|
287
|
+
file path and export name. No manual naming required for project-local code.
|
|
288
|
+
|
|
289
|
+
### Handles in 3rd-party packages
|
|
290
|
+
|
|
291
|
+
The `exposeInternalIds` plugin skips `node_modules/`, so handles defined in
|
|
292
|
+
published packages won't get auto-injected IDs. Pass a manual tag as the
|
|
293
|
+
second argument to `createHandle()`:
|
|
294
|
+
|
|
295
|
+
```typescript
|
|
296
|
+
import { createHandle } from "@rangojs/router";
|
|
297
|
+
|
|
298
|
+
// With a collect function (reducer): collect is first arg, tag is second
|
|
299
|
+
export const Breadcrumbs = createHandle<BreadcrumbItem, BreadcrumbItem[]>(
|
|
300
|
+
collectBreadcrumbs,
|
|
301
|
+
"__my_package_breadcrumbs__",
|
|
302
|
+
);
|
|
303
|
+
|
|
304
|
+
// Without a collect function: pass undefined, then the tag
|
|
305
|
+
export const Warnings = createHandle<string>(
|
|
306
|
+
undefined,
|
|
307
|
+
"__my_package_warnings__",
|
|
308
|
+
);
|
|
309
|
+
```
|
|
310
|
+
|
|
311
|
+
The tag must be globally unique and stable across builds. Without it,
|
|
312
|
+
`createHandle` throws in development mode.
|