@rangojs/router 0.0.0-experimental.8 → 0.0.0-experimental.80
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 +9 -0
- package/README.md +942 -4
- package/dist/bin/rango.js +1689 -0
- package/dist/vite/index.js +4960 -935
- package/package.json +70 -60
- package/skills/breadcrumbs/SKILL.md +250 -0
- package/skills/cache-guide/SKILL.md +294 -0
- package/skills/caching/SKILL.md +93 -23
- package/skills/composability/SKILL.md +172 -0
- package/skills/debug-manifest/SKILL.md +12 -8
- package/skills/document-cache/SKILL.md +18 -16
- package/skills/fonts/SKILL.md +167 -0
- package/skills/handler-use/SKILL.md +362 -0
- package/skills/hooks/SKILL.md +334 -72
- package/skills/host-router/SKILL.md +218 -0
- package/skills/intercept/SKILL.md +151 -8
- package/skills/layout/SKILL.md +122 -3
- package/skills/links/SKILL.md +92 -31
- package/skills/loader/SKILL.md +404 -44
- package/skills/middleware/SKILL.md +205 -37
- package/skills/migrate-nextjs/SKILL.md +560 -0
- package/skills/migrate-react-router/SKILL.md +764 -0
- package/skills/mime-routes/SKILL.md +128 -0
- package/skills/parallel/SKILL.md +263 -1
- package/skills/prerender/SKILL.md +685 -0
- package/skills/rango/SKILL.md +87 -16
- package/skills/response-routes/SKILL.md +411 -0
- package/skills/route/SKILL.md +281 -14
- package/skills/router-setup/SKILL.md +210 -32
- package/skills/tailwind/SKILL.md +129 -0
- package/skills/theme/SKILL.md +9 -8
- package/skills/typesafety/SKILL.md +328 -89
- package/skills/use-cache/SKILL.md +324 -0
- package/src/__internal.ts +102 -4
- package/src/bin/rango.ts +321 -0
- package/src/browser/action-coordinator.ts +97 -0
- package/src/browser/action-response-classifier.ts +99 -0
- package/src/browser/app-version.ts +14 -0
- package/src/browser/event-controller.ts +92 -64
- package/src/browser/history-state.ts +80 -0
- package/src/browser/intercept-utils.ts +52 -0
- package/src/browser/link-interceptor.ts +24 -4
- package/src/browser/logging.ts +55 -0
- package/src/browser/merge-segment-loaders.ts +20 -12
- package/src/browser/navigation-bridge.ts +317 -560
- package/src/browser/navigation-client.ts +206 -68
- package/src/browser/navigation-store.ts +73 -55
- package/src/browser/navigation-transaction.ts +297 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +343 -316
- package/src/browser/prefetch/cache.ts +216 -0
- package/src/browser/prefetch/fetch.ts +206 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +48 -0
- package/src/browser/prefetch/queue.ts +160 -0
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/rango-state.ts +112 -0
- package/src/browser/react/Link.tsx +253 -74
- package/src/browser/react/NavigationProvider.tsx +87 -11
- package/src/browser/react/context.ts +11 -0
- package/src/browser/react/filter-segment-order.ts +11 -0
- package/src/browser/react/index.ts +12 -12
- package/src/browser/react/location-state-shared.ts +95 -53
- package/src/browser/react/location-state.ts +60 -15
- package/src/browser/react/mount-context.ts +6 -1
- package/src/browser/react/nonce-context.ts +23 -0
- package/src/browser/react/shallow-equal.ts +27 -0
- package/src/browser/react/use-action.ts +29 -51
- package/src/browser/react/use-client-cache.ts +5 -3
- package/src/browser/react/use-handle.ts +30 -126
- package/src/browser/react/use-href.tsx +2 -2
- package/src/browser/react/use-link-status.ts +6 -5
- package/src/browser/react/use-navigation.ts +44 -65
- package/src/browser/react/use-params.ts +65 -0
- package/src/browser/react/use-pathname.ts +47 -0
- package/src/browser/react/use-router.ts +76 -0
- package/src/browser/react/use-search-params.ts +56 -0
- package/src/browser/react/use-segments.ts +80 -97
- package/src/browser/response-adapter.ts +73 -0
- package/src/browser/rsc-router.tsx +214 -58
- package/src/browser/scroll-restoration.ts +127 -52
- package/src/browser/segment-reconciler.ts +243 -0
- package/src/browser/segment-structure-assert.ts +16 -0
- package/src/browser/server-action-bridge.ts +510 -603
- package/src/browser/shallow.ts +6 -1
- package/src/browser/types.ts +141 -48
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +235 -24
- package/src/build/generate-route-types.ts +39 -0
- package/src/build/index.ts +13 -0
- package/src/build/route-trie.ts +291 -0
- package/src/build/route-types/ast-helpers.ts +25 -0
- package/src/build/route-types/ast-route-extraction.ts +98 -0
- package/src/build/route-types/codegen.ts +102 -0
- package/src/build/route-types/include-resolution.ts +418 -0
- package/src/build/route-types/param-extraction.ts +48 -0
- package/src/build/route-types/per-module-writer.ts +128 -0
- package/src/build/route-types/router-processing.ts +618 -0
- package/src/build/route-types/scan-filter.ts +85 -0
- package/src/build/runtime-discovery.ts +231 -0
- package/src/cache/background-task.ts +34 -0
- package/src/cache/cache-key-utils.ts +44 -0
- package/src/cache/cache-policy.ts +125 -0
- package/src/cache/cache-runtime.ts +342 -0
- package/src/cache/cache-scope.ts +167 -309
- package/src/cache/cf/cf-cache-store.ts +571 -17
- package/src/cache/cf/index.ts +13 -3
- package/src/cache/document-cache.ts +116 -77
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/handle-snapshot.ts +41 -0
- package/src/cache/index.ts +1 -15
- package/src/cache/memory-segment-store.ts +191 -13
- package/src/cache/profile-registry.ts +73 -0
- package/src/cache/read-through-swr.ts +134 -0
- package/src/cache/segment-codec.ts +256 -0
- package/src/cache/taint.ts +153 -0
- package/src/cache/types.ts +72 -122
- package/src/client.rsc.tsx +3 -1
- package/src/client.tsx +135 -301
- package/src/component-utils.ts +4 -4
- package/src/components/DefaultDocument.tsx +5 -1
- package/src/context-var.ts +156 -0
- package/src/debug.ts +19 -9
- package/src/errors.ts +108 -2
- package/src/handle.ts +55 -29
- package/src/handles/MetaTags.tsx +73 -20
- package/src/handles/breadcrumbs.ts +66 -0
- package/src/handles/index.ts +1 -0
- package/src/handles/meta.ts +30 -13
- package/src/host/cookie-handler.ts +21 -15
- package/src/host/errors.ts +8 -8
- package/src/host/index.ts +4 -7
- package/src/host/pattern-matcher.ts +27 -27
- package/src/host/router.ts +61 -39
- package/src/host/testing.ts +8 -8
- package/src/host/types.ts +15 -7
- package/src/host/utils.ts +1 -1
- package/src/href-client.ts +119 -29
- package/src/index.rsc.ts +155 -19
- package/src/index.ts +251 -30
- package/src/internal-debug.ts +11 -0
- package/src/loader.rsc.ts +26 -157
- package/src/loader.ts +27 -10
- package/src/network-error-thrower.tsx +3 -1
- package/src/outlet-provider.tsx +45 -0
- package/src/prerender/param-hash.ts +37 -0
- package/src/prerender/store.ts +186 -0
- package/src/prerender.ts +524 -0
- package/src/reverse.ts +354 -0
- package/src/root-error-boundary.tsx +41 -29
- package/src/route-content-wrapper.tsx +7 -4
- package/src/route-definition/dsl-helpers.ts +1121 -0
- package/src/route-definition/helper-factories.ts +200 -0
- package/src/route-definition/helpers-types.ts +478 -0
- package/src/route-definition/index.ts +55 -0
- package/src/route-definition/redirect.ts +101 -0
- package/src/route-definition/resolve-handler-use.ts +149 -0
- package/src/route-definition.ts +1 -1428
- package/src/route-map-builder.ts +217 -123
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +77 -8
- package/src/router/content-negotiation.ts +215 -0
- package/src/router/debug-manifest.ts +72 -0
- package/src/router/error-handling.ts +9 -9
- package/src/router/find-match.ts +160 -0
- package/src/router/handler-context.ts +438 -86
- package/src/router/intercept-resolution.ts +402 -0
- package/src/router/lazy-includes.ts +237 -0
- package/src/router/loader-resolution.ts +356 -128
- package/src/router/logging.ts +251 -0
- package/src/router/manifest.ts +163 -35
- package/src/router/match-api.ts +555 -0
- package/src/router/match-context.ts +5 -3
- package/src/router/match-handlers.ts +440 -0
- package/src/router/match-middleware/background-revalidation.ts +108 -93
- package/src/router/match-middleware/cache-lookup.ts +460 -10
- package/src/router/match-middleware/cache-store.ts +98 -26
- package/src/router/match-middleware/intercept-resolution.ts +57 -17
- package/src/router/match-middleware/segment-resolution.ts +80 -6
- package/src/router/match-pipelines.ts +10 -45
- package/src/router/match-result.ts +135 -35
- package/src/router/metrics.ts +240 -15
- package/src/router/middleware-cookies.ts +55 -0
- package/src/router/middleware-types.ts +220 -0
- package/src/router/middleware.ts +324 -369
- package/src/router/navigation-snapshot.ts +182 -0
- package/src/router/pattern-matching.ts +211 -43
- package/src/router/prerender-match.ts +502 -0
- package/src/router/preview-match.ts +98 -0
- package/src/router/request-classification.ts +310 -0
- package/src/router/revalidation.ts +137 -38
- package/src/router/route-snapshot.ts +245 -0
- package/src/router/router-context.ts +41 -21
- package/src/router/router-interfaces.ts +484 -0
- package/src/router/router-options.ts +618 -0
- package/src/router/router-registry.ts +24 -0
- package/src/router/segment-resolution/fresh.ts +748 -0
- package/src/router/segment-resolution/helpers.ts +268 -0
- package/src/router/segment-resolution/loader-cache.ts +199 -0
- package/src/router/segment-resolution/revalidation.ts +1379 -0
- package/src/router/segment-resolution/static-store.ts +67 -0
- package/src/router/segment-resolution.ts +21 -0
- package/src/router/segment-wrappers.ts +291 -0
- package/src/router/telemetry-otel.ts +299 -0
- package/src/router/telemetry.ts +300 -0
- package/src/router/timeout.ts +148 -0
- package/src/router/trie-matching.ts +239 -0
- package/src/router/types.ts +78 -3
- package/src/router.ts +740 -4252
- package/src/rsc/handler-context.ts +45 -0
- package/src/rsc/handler.ts +907 -797
- package/src/rsc/helpers.ts +140 -6
- package/src/rsc/index.ts +0 -20
- package/src/rsc/loader-fetch.ts +229 -0
- package/src/rsc/manifest-init.ts +90 -0
- package/src/rsc/nonce.ts +14 -0
- package/src/rsc/origin-guard.ts +141 -0
- package/src/rsc/progressive-enhancement.ts +391 -0
- package/src/rsc/response-error.ts +37 -0
- package/src/rsc/response-route-handler.ts +347 -0
- package/src/rsc/rsc-rendering.ts +246 -0
- package/src/rsc/runtime-warnings.ts +42 -0
- package/src/rsc/server-action.ts +356 -0
- package/src/rsc/ssr-setup.ts +128 -0
- package/src/rsc/types.ts +46 -11
- package/src/search-params.ts +230 -0
- package/src/segment-content-promise.ts +67 -0
- package/src/segment-loader-promise.ts +122 -0
- package/src/segment-system.tsx +134 -36
- package/src/server/context.ts +341 -61
- package/src/server/cookie-store.ts +190 -0
- package/src/server/fetchable-loader-store.ts +37 -0
- package/src/server/handle-store.ts +113 -15
- package/src/server/loader-registry.ts +24 -64
- package/src/server/request-context.ts +607 -81
- package/src/server.ts +35 -130
- package/src/ssr/index.tsx +103 -30
- package/src/static-handler.ts +126 -0
- package/src/theme/ThemeProvider.tsx +21 -15
- package/src/theme/ThemeScript.tsx +5 -5
- package/src/theme/constants.ts +5 -2
- package/src/theme/index.ts +4 -14
- package/src/theme/theme-context.ts +4 -30
- package/src/theme/theme-script.ts +21 -18
- package/src/types/boundaries.ts +158 -0
- package/src/types/cache-types.ts +198 -0
- package/src/types/error-types.ts +192 -0
- package/src/types/global-namespace.ts +100 -0
- package/src/types/handler-context.ts +791 -0
- package/src/types/index.ts +88 -0
- package/src/types/loader-types.ts +210 -0
- package/src/types/route-config.ts +170 -0
- package/src/types/route-entry.ts +120 -0
- package/src/types/segments.ts +150 -0
- package/src/types.ts +1 -1623
- package/src/urls/include-helper.ts +207 -0
- package/src/urls/index.ts +53 -0
- package/src/urls/path-helper-types.ts +372 -0
- package/src/urls/path-helper.ts +364 -0
- package/src/urls/pattern-types.ts +107 -0
- package/src/urls/response-types.ts +116 -0
- package/src/urls/type-extraction.ts +372 -0
- package/src/urls/urls-function.ts +98 -0
- package/src/urls.ts +1 -802
- package/src/use-loader.tsx +161 -81
- package/src/vite/discovery/bundle-postprocess.ts +181 -0
- package/src/vite/discovery/discover-routers.ts +348 -0
- package/src/vite/discovery/prerender-collection.ts +439 -0
- package/src/vite/discovery/route-types-writer.ts +258 -0
- package/src/vite/discovery/self-gen-tracking.ts +47 -0
- package/src/vite/discovery/state.ts +117 -0
- package/src/vite/discovery/virtual-module-codegen.ts +203 -0
- package/src/vite/index.ts +15 -1133
- package/src/vite/plugin-types.ts +103 -0
- package/src/vite/plugins/cjs-to-esm.ts +93 -0
- package/src/vite/plugins/client-ref-dedup.ts +115 -0
- package/src/vite/plugins/client-ref-hashing.ts +105 -0
- package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -53
- package/src/vite/plugins/expose-id-utils.ts +299 -0
- package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +209 -0
- package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
- package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
- package/src/vite/plugins/expose-ids/types.ts +45 -0
- package/src/vite/plugins/expose-internal-ids.ts +786 -0
- package/src/vite/plugins/performance-tracks.ts +88 -0
- package/src/vite/plugins/refresh-cmd.ts +127 -0
- package/src/vite/plugins/use-cache-transform.ts +323 -0
- package/src/vite/plugins/version-injector.ts +83 -0
- package/src/vite/plugins/version-plugin.ts +266 -0
- package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
- package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
- package/src/vite/rango.ts +462 -0
- package/src/vite/router-discovery.ts +918 -0
- package/src/vite/utils/ast-handler-extract.ts +517 -0
- package/src/vite/utils/banner.ts +36 -0
- package/src/vite/utils/bundle-analysis.ts +137 -0
- package/src/vite/utils/manifest-utils.ts +70 -0
- package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
- package/src/vite/utils/prerender-utils.ts +221 -0
- package/src/vite/utils/shared-utils.ts +170 -0
- package/CLAUDE.md +0 -43
- package/src/browser/lru-cache.ts +0 -69
- package/src/browser/request-controller.ts +0 -164
- package/src/cache/memory-store.ts +0 -253
- package/src/href-context.ts +0 -33
- package/src/href.ts +0 -255
- package/src/server/route-manifest-cache.ts +0 -173
- package/src/vite/expose-handle-id.ts +0 -209
- package/src/vite/expose-loader-id.ts +0 -426
- package/src/vite/expose-location-state-id.ts +0 -177
- /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
|
@@ -0,0 +1,128 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: mime-routes
|
|
3
|
+
description: Content negotiation — serve different response types (RSC, JSON, text, XML) from the same URL based on Accept header
|
|
4
|
+
argument-hint: [negotiate|vary|accept]
|
|
5
|
+
---
|
|
6
|
+
|
|
7
|
+
# Content Negotiation (MIME Routes)
|
|
8
|
+
|
|
9
|
+
Content negotiation lets you register multiple response types on the same URL pattern.
|
|
10
|
+
The router inspects the `Accept` header and dispatches to the matching handler.
|
|
11
|
+
All negotiated responses include `Vary: Accept` for correct CDN/cache behavior.
|
|
12
|
+
|
|
13
|
+
See also: `/response-routes` for the base response route API (path.json, path.text, etc.).
|
|
14
|
+
|
|
15
|
+
## Defining Negotiated Routes
|
|
16
|
+
|
|
17
|
+
Declare the same URL pattern with both an RSC route and one or more response-type routes.
|
|
18
|
+
Order within the `urls()` array does not matter — the trie merges them at build time.
|
|
19
|
+
|
|
20
|
+
```typescript
|
|
21
|
+
import { urls } from "@rangojs/router";
|
|
22
|
+
|
|
23
|
+
export const urlpatterns = urls(({ path, layout, include }) => [
|
|
24
|
+
// RSC page + JSON API on the same URL
|
|
25
|
+
path("/products/:id", ProductPage, { name: "product" }),
|
|
26
|
+
path.json(
|
|
27
|
+
"/products/:id",
|
|
28
|
+
(ctx) => {
|
|
29
|
+
return db.getProduct(ctx.params.id);
|
|
30
|
+
},
|
|
31
|
+
{ name: "productJson" },
|
|
32
|
+
),
|
|
33
|
+
]);
|
|
34
|
+
```
|
|
35
|
+
|
|
36
|
+
When a browser requests `/products/42` (`Accept: text/html`), the RSC page renders.
|
|
37
|
+
When an API client requests the same URL (`Accept: application/json`), the JSON handler runs.
|
|
38
|
+
|
|
39
|
+
## Negotiation Rules
|
|
40
|
+
|
|
41
|
+
1. **Q-value priority** — higher `q` wins (`Accept: application/json;q=0.9, text/html;q=1.0` serves RSC)
|
|
42
|
+
2. **Client order tiebreaker** — when q-values are equal, the type listed first in Accept wins (matches Express/Hono behavior)
|
|
43
|
+
3. **Specific MIME match** — the variant whose MIME type appears in Accept wins
|
|
44
|
+
4. **Wildcard / empty Accept** — `*/*` and missing Accept fall back to route definition order (the first-defined variant wins)
|
|
45
|
+
5. **All responses** on a negotiated URL get `Vary: Accept` header, including the RSC side
|
|
46
|
+
|
|
47
|
+
RSC participates as a `text/html` candidate alongside response-type variants.
|
|
48
|
+
There is no special short-circuit — RSC follows the same negotiation rules as other types.
|
|
49
|
+
|
|
50
|
+
The MIME mapping used for matching:
|
|
51
|
+
|
|
52
|
+
| Tag | MIME type |
|
|
53
|
+
| -------------------- | ------------------------------------------------------------ |
|
|
54
|
+
| RSC (plain `path()`) | `text/html` (negotiation) / `text/x-component` (wire format) |
|
|
55
|
+
| `json` | `application/json` |
|
|
56
|
+
| `text` | `text/plain` |
|
|
57
|
+
| `xml` | `application/xml` |
|
|
58
|
+
| `html` | `text/html` |
|
|
59
|
+
| `md` | `text/markdown` |
|
|
60
|
+
|
|
61
|
+
RSC routes negotiate as `text/html` but respond with `text/x-component` (the RSC wire format).
|
|
62
|
+
The browser's RSC runtime decodes this transparently — clients requesting `text/html` get
|
|
63
|
+
the RSC page rendered normally.
|
|
64
|
+
|
|
65
|
+
Tags `image`, `stream`, and `any` are pass-through and do not participate in Accept matching.
|
|
66
|
+
|
|
67
|
+
## Multiple Response Types
|
|
68
|
+
|
|
69
|
+
A single URL can have an RSC route plus multiple response-type variants:
|
|
70
|
+
|
|
71
|
+
```typescript
|
|
72
|
+
export const urlpatterns = urls(({ path }) => [
|
|
73
|
+
path("/data", DataPage, { name: "data" }),
|
|
74
|
+
path.json("/data", () => ({ format: "json" }), { name: "dataJson" }),
|
|
75
|
+
path.text("/data", () => "plain text", { name: "dataText" }),
|
|
76
|
+
path.xml("/data", () => "<root>xml</root>", { name: "dataXml" }),
|
|
77
|
+
]);
|
|
78
|
+
```
|
|
79
|
+
|
|
80
|
+
- `Accept: text/html` — RSC page
|
|
81
|
+
- `Accept: application/json` — JSON handler
|
|
82
|
+
- `Accept: text/plain` — text handler
|
|
83
|
+
- `Accept: application/xml` — XML handler
|
|
84
|
+
- `Accept: */*` — first variant (JSON, since it was registered first)
|
|
85
|
+
|
|
86
|
+
## Wildcard Routes
|
|
87
|
+
|
|
88
|
+
Content negotiation works with wildcard `/*` patterns:
|
|
89
|
+
|
|
90
|
+
```typescript
|
|
91
|
+
path("/files/*", FileBrowserPage, { name: "files" }),
|
|
92
|
+
path.json("/files/*", (ctx) => {
|
|
93
|
+
const filePath = ctx.params["*"];
|
|
94
|
+
return { entries: listDir(filePath) };
|
|
95
|
+
}, { name: "filesJson" }),
|
|
96
|
+
```
|
|
97
|
+
|
|
98
|
+
## Response-Only Negotiation (No RSC Primary)
|
|
99
|
+
|
|
100
|
+
Two or more response-type routes can share a URL without an RSC route.
|
|
101
|
+
The last registered route becomes the primary; earlier ones become variants:
|
|
102
|
+
|
|
103
|
+
```typescript
|
|
104
|
+
path.json("/api/data", () => ({ format: "json" }), { name: "dataJson" }),
|
|
105
|
+
path.text("/api/data", () => "plain text version", { name: "dataText" }),
|
|
106
|
+
```
|
|
107
|
+
|
|
108
|
+
Without an RSC primary, there is no `text/html` candidate — the Accept header
|
|
109
|
+
picks among the response-type candidates directly.
|
|
110
|
+
|
|
111
|
+
## How It Works
|
|
112
|
+
|
|
113
|
+
1. **Build time**: `buildRouteTrie()` calls `mergeLeaves()` when multiple routes share a pattern.
|
|
114
|
+
RSC routes become the primary trie leaf; response-type routes are stored in the `nv`
|
|
115
|
+
(negotiate variants) array on the leaf. The `rf` (rsc-first) flag tracks definition order.
|
|
116
|
+
2. **Runtime**: `previewRoute()` reads `negotiateVariants` from the trie match result.
|
|
117
|
+
It parses the `Accept` header (extracting q-values and order), builds a candidate list
|
|
118
|
+
(RSC as `text/html` + response-type variants), and calls `pickNegotiateVariant()`.
|
|
119
|
+
3. **Candidate matching**: walks the client's sorted Accept list (by q desc, then order asc),
|
|
120
|
+
matching each entry against candidates. Wildcards (`*/*`, `text/*`) fall back to definition order.
|
|
121
|
+
4. **Vary header**: both the response-route handler wrapper and the RSC handler wrapper
|
|
122
|
+
append `Vary: Accept` when the `negotiated` flag is set on the preview result.
|
|
123
|
+
|
|
124
|
+
## Caching Considerations
|
|
125
|
+
|
|
126
|
+
`Vary: Accept` is set automatically on all negotiated responses. This tells CDNs and
|
|
127
|
+
HTTP caches to store separate entries per Accept header value. No additional cache
|
|
128
|
+
configuration is needed for negotiated routes — the framework handles it.
|
package/skills/parallel/SKILL.md
CHANGED
|
@@ -8,6 +8,9 @@ argument-hint: [@slot-name]
|
|
|
8
8
|
|
|
9
9
|
Parallel routes render multiple components simultaneously in named slots.
|
|
10
10
|
|
|
11
|
+
Canonical semantics reference:
|
|
12
|
+
[docs/execution-model.md](../../docs/internal/execution-model.md)
|
|
13
|
+
|
|
11
14
|
## Basic Parallel Routes
|
|
12
15
|
|
|
13
16
|
```typescript
|
|
@@ -54,6 +57,108 @@ parallel({
|
|
|
54
57
|
})
|
|
55
58
|
```
|
|
56
59
|
|
|
60
|
+
## Reading Handler Data
|
|
61
|
+
|
|
62
|
+
Parallels can read `ctx.set()` values from their parent handler or layout
|
|
63
|
+
via `ctx.get()`. The handler always executes before its parallels
|
|
64
|
+
(handler-first).
|
|
65
|
+
|
|
66
|
+
Visibility follows tree structure:
|
|
67
|
+
|
|
68
|
+
- Layout-level parallels see layout data, but not path handler data
|
|
69
|
+
(the path is a separate entry).
|
|
70
|
+
- Parallels inside a path (or its orphan layouts) see both layout and
|
|
71
|
+
path handler data.
|
|
72
|
+
|
|
73
|
+
This applies to full render passes. During partial action revalidation,
|
|
74
|
+
only revalidated segments are recomputed. If a parallel depends on data
|
|
75
|
+
set by an outer handler or layout, revalidate that outer segment too, or
|
|
76
|
+
have the parallel reload/guard the data itself.
|
|
77
|
+
|
|
78
|
+
```typescript
|
|
79
|
+
path("/dashboard/:id", (ctx) => {
|
|
80
|
+
const user = await getUser(ctx.params.id);
|
|
81
|
+
ctx.set("user", user);
|
|
82
|
+
return <DashboardPage user={user} />;
|
|
83
|
+
}, { name: "dashboard" }, () => [
|
|
84
|
+
layout(DashboardLayout, () => [
|
|
85
|
+
parallel({
|
|
86
|
+
"@sidebar": (ctx) => {
|
|
87
|
+
const user = ctx.get("user");
|
|
88
|
+
return <Sidebar role={user?.role} />;
|
|
89
|
+
},
|
|
90
|
+
}),
|
|
91
|
+
]),
|
|
92
|
+
])
|
|
93
|
+
```
|
|
94
|
+
|
|
95
|
+
## Setting Handles (Meta, Breadcrumbs)
|
|
96
|
+
|
|
97
|
+
Parallel slot handlers can call `ctx.use(Meta)` or `ctx.use(Breadcrumbs)` to
|
|
98
|
+
push handle data. The data is associated with the **parent** layout or route
|
|
99
|
+
segment, not the parallel segment itself. This is because parallels execute
|
|
100
|
+
after their parent handler and inherit its segment scope.
|
|
101
|
+
|
|
102
|
+
This works well for document-level metadata — the handle data follows the
|
|
103
|
+
parent's lifecycle (appears when the parent is mounted, removed when it
|
|
104
|
+
unmounts).
|
|
105
|
+
|
|
106
|
+
```typescript
|
|
107
|
+
parallel({
|
|
108
|
+
"@meta": (ctx) => {
|
|
109
|
+
const meta = ctx.use(Meta);
|
|
110
|
+
meta({ title: "Product Detail" });
|
|
111
|
+
meta({ name: "description", content: "..." });
|
|
112
|
+
return null; // UI-less slot, only sets metadata
|
|
113
|
+
},
|
|
114
|
+
"@sidebar": (ctx) => <Sidebar />,
|
|
115
|
+
})
|
|
116
|
+
```
|
|
117
|
+
|
|
118
|
+
Multiple parallels on the same parent can each push handle data — they all
|
|
119
|
+
accumulate under the parent segment ID.
|
|
120
|
+
|
|
121
|
+
### Pattern: `@meta` slot for per-route metadata overrides
|
|
122
|
+
|
|
123
|
+
A dedicated `@meta` parallel slot lets routes define metadata separately from
|
|
124
|
+
their handler logic. The layout sets defaults via a title template, and each
|
|
125
|
+
route overrides via its own `@meta` slot. Since child segments push after
|
|
126
|
+
parents and `collectMeta` uses last-wins deduplication, overrides work
|
|
127
|
+
naturally.
|
|
128
|
+
|
|
129
|
+
```typescript
|
|
130
|
+
// Layout sets defaults
|
|
131
|
+
layout((ctx) => {
|
|
132
|
+
ctx.use(Meta)({ title: { template: "%s | Store", default: "Store" } });
|
|
133
|
+
return <StoreLayout />;
|
|
134
|
+
}, () => [
|
|
135
|
+
// Route with @meta override — decoupled from handler rendering
|
|
136
|
+
path("/:slug", ProductPage, { name: "product" }, () => [
|
|
137
|
+
parallel({
|
|
138
|
+
"@meta": async (ctx) => {
|
|
139
|
+
const product = await ctx.use(ProductLoader);
|
|
140
|
+
const meta = ctx.use(Meta);
|
|
141
|
+
meta({ title: product.name });
|
|
142
|
+
meta({ name: "description", content: product.description });
|
|
143
|
+
meta({
|
|
144
|
+
"script:ld+json": {
|
|
145
|
+
"@context": "https://schema.org",
|
|
146
|
+
"@type": "Product",
|
|
147
|
+
name: product.name,
|
|
148
|
+
description: product.description,
|
|
149
|
+
},
|
|
150
|
+
});
|
|
151
|
+
return null; // UI-less slot
|
|
152
|
+
},
|
|
153
|
+
}),
|
|
154
|
+
]),
|
|
155
|
+
])
|
|
156
|
+
```
|
|
157
|
+
|
|
158
|
+
This keeps the route handler focused on rendering UI while metadata
|
|
159
|
+
(title, description, Open Graph, JSON-LD) lives in a composable slot that
|
|
160
|
+
can be added, removed, or swapped per route without touching the handler.
|
|
161
|
+
|
|
57
162
|
## Parallel Routes with Loaders
|
|
58
163
|
|
|
59
164
|
Add loaders and loading states to parallel routes:
|
|
@@ -71,6 +176,124 @@ parallel(
|
|
|
71
176
|
)
|
|
72
177
|
```
|
|
73
178
|
|
|
179
|
+
### Streaming Behavior
|
|
180
|
+
|
|
181
|
+
Parallels with `loading()` are **independent streaming units**. They don't
|
|
182
|
+
block the parent layout or sibling routes during SSR:
|
|
183
|
+
|
|
184
|
+
- **With `loading()`**: The skeleton renders immediately. The loader runs
|
|
185
|
+
in the background and streams data to the client when ready. The rest
|
|
186
|
+
of the page (layout, route content, other parallels) renders without
|
|
187
|
+
waiting.
|
|
188
|
+
- **Without `loading()`**: The parallel's loaders block the parent layout's
|
|
189
|
+
rendering. Use this when the data must be available before the page
|
|
190
|
+
paints (e.g., critical above-the-fold content).
|
|
191
|
+
- **SPA navigation**: Parallel loaders resolve in the background. The
|
|
192
|
+
existing parallel UI stays visible — no skeleton flash on route changes
|
|
193
|
+
within the same layout.
|
|
194
|
+
|
|
195
|
+
```typescript
|
|
196
|
+
// Sidebar streams independently — page renders immediately
|
|
197
|
+
parallel(
|
|
198
|
+
{ "@sidebar": () => <Sidebar /> },
|
|
199
|
+
() => [loader(SlowSidebarLoader), loading(<SidebarSkeleton />)]
|
|
200
|
+
)
|
|
201
|
+
|
|
202
|
+
// Cart data blocks layout — must be ready before paint
|
|
203
|
+
parallel(
|
|
204
|
+
{ "@cartBadge": () => <CartBadge /> },
|
|
205
|
+
() => [loader(CartCountLoader)] // No loading() = awaited
|
|
206
|
+
)
|
|
207
|
+
```
|
|
208
|
+
|
|
209
|
+
## Composable Slots via `handler.use`
|
|
210
|
+
|
|
211
|
+
Slot handlers can carry their own loader, loading, error/notFound boundaries, revalidation, and transition defaults via `.use`. The mount site then declares **just the slot names** — no per-call data wiring.
|
|
212
|
+
|
|
213
|
+
```typescript
|
|
214
|
+
const CartSummary: Handler = async (ctx) => {
|
|
215
|
+
const cart = await ctx.use(CartLoader);
|
|
216
|
+
return <CartSummaryView cart={cart} />;
|
|
217
|
+
};
|
|
218
|
+
CartSummary.use = () => [
|
|
219
|
+
loader(CartLoader),
|
|
220
|
+
loading(<CartSkeleton />),
|
|
221
|
+
revalidate(revalidateCartData),
|
|
222
|
+
];
|
|
223
|
+
|
|
224
|
+
// Same slot, no copy-pasted plumbing across layouts.
|
|
225
|
+
layout(<DashboardLayout />, () => [
|
|
226
|
+
parallel({ "@cart": CartSummary }),
|
|
227
|
+
path("/dashboard", DashboardIndex, { name: "dashboard.index" }),
|
|
228
|
+
]);
|
|
229
|
+
|
|
230
|
+
layout(<AccountLayout />, () => [
|
|
231
|
+
parallel({ "@cart": CartSummary }),
|
|
232
|
+
path("/account", AccountIndex, { name: "account.index" }),
|
|
233
|
+
]);
|
|
234
|
+
```
|
|
235
|
+
|
|
236
|
+
A slot's `loading()` (whether from `handler.use` or explicit) makes that slot an independent streaming unit, exactly as in the **Streaming Behavior** section above.
|
|
237
|
+
|
|
238
|
+
The `parallel` mount site has the narrowest allow-list for `handler.use` items — slots cannot bring their own middleware or layout, only `revalidate`, `loader`, `loading`, `errorBoundary`, `notFoundBoundary`, and `transition`. See [skills/handler-use](../handler-use/SKILL.md) for the full table and merge rules.
|
|
239
|
+
|
|
240
|
+
### Two scopes for explicit `use`: shared (broadcast) and slot-local
|
|
241
|
+
|
|
242
|
+
`parallel({...slots}, () => [...use])` runs the shared `use()` callback **once per slot** ([dsl-helpers.ts](../../src/route-definition/dsl-helpers.ts)) — items in that callback land on every slot's entry. That's the right behavior for the items the parallel allow-list permits and that accumulate (`loader`, `revalidate`, `errorBoundary`, `notFoundBoundary`, `transition`). (Slots cannot bring `middleware` or `layout` — see the allowed-types note above.)
|
|
243
|
+
|
|
244
|
+
For single-assignment items like `loading()`, broadcasting overwrites every slot's `handler.use` default. Pass a **slot descriptor** `{ handler, use }` instead — items in the descriptor's `use` apply only to that slot:
|
|
245
|
+
|
|
246
|
+
```typescript
|
|
247
|
+
// @cart gets a custom skeleton; @notifs keeps its handler.use default.
|
|
248
|
+
parallel({
|
|
249
|
+
"@cart": {
|
|
250
|
+
handler: Cart,
|
|
251
|
+
use: () => [loading(<CustomCartSkeleton />)],
|
|
252
|
+
},
|
|
253
|
+
"@notifs": Notifs,
|
|
254
|
+
});
|
|
255
|
+
|
|
256
|
+
// Opt one slot out of streaming while siblings still stream the broadcast.
|
|
257
|
+
parallel(
|
|
258
|
+
{
|
|
259
|
+
"@cart": { handler: Cart, use: () => [loading(false)] },
|
|
260
|
+
"@notifs": Notifs,
|
|
261
|
+
},
|
|
262
|
+
() => [loading(<BroadcastSkeleton />)],
|
|
263
|
+
);
|
|
264
|
+
```
|
|
265
|
+
|
|
266
|
+
Per-slot merge order is **handler.use → shared use → slot-local use**. Slot-local is the narrowest scope, so it wins for last-write-wins items. See [skills/handler-use § `loading()` is a single-assignment item — scope it correctly](../handler-use/SKILL.md#loading-is-a-single-assignment-item--scope-it-correctly) for the full reasoning.
|
|
267
|
+
|
|
268
|
+
## Slot Override Semantics
|
|
269
|
+
|
|
270
|
+
When multiple `parallel()` calls define the same slot name, **the last
|
|
271
|
+
definition wins**. Earlier definitions of that slot are removed. Other
|
|
272
|
+
slots from the earlier call are preserved.
|
|
273
|
+
|
|
274
|
+
This enables composition patterns where included routes override
|
|
275
|
+
parent-defined slots:
|
|
276
|
+
|
|
277
|
+
```typescript
|
|
278
|
+
layout(DashboardLayout, () => [
|
|
279
|
+
// Base slots
|
|
280
|
+
parallel({
|
|
281
|
+
"@sidebar": () => <DefaultSidebar />,
|
|
282
|
+
"@footer": () => <Footer />,
|
|
283
|
+
}),
|
|
284
|
+
|
|
285
|
+
// Override just @sidebar — @footer is preserved
|
|
286
|
+
parallel({ "@sidebar": () => <CustomSidebar /> }),
|
|
287
|
+
|
|
288
|
+
path("/", DashboardIndex, { name: "index" }),
|
|
289
|
+
])
|
|
290
|
+
```
|
|
291
|
+
|
|
292
|
+
After resolution, the layout has two parallel entries:
|
|
293
|
+
|
|
294
|
+
- `{ "@footer": () => <Footer /> }` (first call, `@sidebar` removed)
|
|
295
|
+
- `{ "@sidebar": () => <CustomSidebar /> }` (second call, wins)
|
|
296
|
+
|
|
74
297
|
## Multiple Parallel Slots
|
|
75
298
|
|
|
76
299
|
```typescript
|
|
@@ -97,7 +320,7 @@ Render different content based on context:
|
|
|
97
320
|
```typescript
|
|
98
321
|
parallel({
|
|
99
322
|
"@sidebar": (ctx) => {
|
|
100
|
-
const user = ctx.
|
|
323
|
+
const user = ctx.get("user");
|
|
101
324
|
return user ? <UserSidebar user={user} /> : <GuestSidebar />;
|
|
102
325
|
},
|
|
103
326
|
})
|
|
@@ -120,6 +343,45 @@ parallel(
|
|
|
120
343
|
)
|
|
121
344
|
```
|
|
122
345
|
|
|
346
|
+
Revalidating only the parallel does not re-run outer handlers/layouts.
|
|
347
|
+
If the slot reads `ctx.get()` data established above it, opt the outer
|
|
348
|
+
segment into revalidation as well.
|
|
349
|
+
|
|
350
|
+
### Revalidation Contracts for Parallel Dependencies
|
|
351
|
+
|
|
352
|
+
Prefer named revalidation contracts shared by both the upstream producer and
|
|
353
|
+
the parallel consumer:
|
|
354
|
+
|
|
355
|
+
```typescript
|
|
356
|
+
// revalidation-contracts.ts
|
|
357
|
+
export const revalidateCartData = ({ actionId }) =>
|
|
358
|
+
actionId?.includes("src/actions/cart.ts#") ?? false;
|
|
359
|
+
|
|
360
|
+
layout(CartLayout, () => [
|
|
361
|
+
revalidate(revalidateCartData), // producer reruns
|
|
362
|
+
parallel(
|
|
363
|
+
{ "@cart": CartSummary },
|
|
364
|
+
() => [revalidate(revalidateCartData)], // consumer reruns
|
|
365
|
+
),
|
|
366
|
+
]);
|
|
367
|
+
```
|
|
368
|
+
|
|
369
|
+
If the slot consumes multiple upstream domains, compose the contracts on both
|
|
370
|
+
segments.
|
|
371
|
+
|
|
372
|
+
Handoff helper style also works:
|
|
373
|
+
|
|
374
|
+
```typescript
|
|
375
|
+
import { revalidate } from "@rangojs/router";
|
|
376
|
+
|
|
377
|
+
export const revalidateCart = () => [revalidate(revalidateCartData)];
|
|
378
|
+
|
|
379
|
+
layout(CartLayout, () => [
|
|
380
|
+
revalidateCart(),
|
|
381
|
+
parallel({ "@cart": CartSummary }, () => [revalidateCart()]),
|
|
382
|
+
]);
|
|
383
|
+
```
|
|
384
|
+
|
|
123
385
|
## Named Outlets
|
|
124
386
|
|
|
125
387
|
Use `ParallelOutlet` to render slots in layouts:
|