@rangojs/router 0.0.0-experimental.8 → 0.0.0-experimental.8a4d0430
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 +5 -0
- package/README.md +884 -4
- package/dist/bin/rango.js +1601 -0
- package/dist/vite/index.js +4474 -867
- package/package.json +60 -51
- package/skills/breadcrumbs/SKILL.md +250 -0
- package/skills/cache-guide/SKILL.md +262 -0
- package/skills/caching/SKILL.md +50 -21
- 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/hooks/SKILL.md +334 -72
- package/skills/host-router/SKILL.md +218 -0
- package/skills/intercept/SKILL.md +131 -8
- package/skills/layout/SKILL.md +100 -3
- package/skills/links/SKILL.md +89 -30
- package/skills/loader/SKILL.md +388 -38
- package/skills/middleware/SKILL.md +171 -34
- package/skills/mime-routes/SKILL.md +128 -0
- package/skills/parallel/SKILL.md +78 -1
- package/skills/prerender/SKILL.md +643 -0
- package/skills/rango/SKILL.md +85 -16
- package/skills/response-routes/SKILL.md +411 -0
- package/skills/route/SKILL.md +226 -14
- package/skills/router-setup/SKILL.md +123 -30
- package/skills/tailwind/SKILL.md +129 -0
- package/skills/theme/SKILL.md +9 -8
- package/skills/typesafety/SKILL.md +318 -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/event-controller.ts +87 -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 +285 -553
- package/src/browser/navigation-client.ts +124 -71
- package/src/browser/navigation-store.ts +33 -50
- package/src/browser/navigation-transaction.ts +295 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +258 -308
- package/src/browser/prefetch/cache.ts +146 -0
- package/src/browser/prefetch/fetch.ts +135 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +42 -0
- package/src/browser/prefetch/queue.ts +88 -0
- package/src/browser/rango-state.ts +112 -0
- package/src/browser/react/Link.tsx +185 -73
- package/src/browser/react/NavigationProvider.tsx +51 -11
- package/src/browser/react/context.ts +6 -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 +32 -79
- 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 +22 -63
- 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 +63 -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 +107 -26
- package/src/browser/scroll-restoration.ts +92 -16
- package/src/browser/segment-reconciler.ts +216 -0
- package/src/browser/segment-structure-assert.ts +16 -0
- package/src/browser/server-action-bridge.ts +504 -599
- package/src/browser/shallow.ts +6 -1
- package/src/browser/types.ts +109 -47
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +235 -24
- package/src/build/generate-route-types.ts +36 -0
- package/src/build/index.ts +13 -0
- package/src/build/route-trie.ts +265 -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 +411 -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 +469 -0
- package/src/build/route-types/scan-filter.ts +78 -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 +338 -0
- package/src/cache/cache-scope.ts +120 -303
- package/src/cache/cf/cf-cache-store.ts +119 -7
- package/src/cache/cf/index.ts +8 -2
- package/src/cache/document-cache.ts +101 -72
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/handle-snapshot.ts +41 -0
- package/src/cache/index.ts +0 -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 +98 -0
- package/src/cache/types.ts +72 -122
- package/src/client.rsc.tsx +3 -1
- package/src/client.tsx +106 -126
- package/src/component-utils.ts +4 -4
- package/src/components/DefaultDocument.tsx +5 -1
- package/src/context-var.ts +86 -0
- package/src/debug.ts +17 -7
- package/src/errors.ts +108 -2
- package/src/handle.ts +15 -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 +153 -19
- package/src/index.ts +211 -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 +185 -0
- package/src/prerender.ts +463 -0
- package/src/reverse.ts +330 -0
- package/src/root-error-boundary.tsx +41 -29
- package/src/route-content-wrapper.tsx +7 -4
- package/src/route-definition/dsl-helpers.ts +934 -0
- package/src/route-definition/helper-factories.ts +200 -0
- package/src/route-definition/helpers-types.ts +430 -0
- package/src/route-definition/index.ts +52 -0
- package/src/route-definition/redirect.ts +93 -0
- package/src/route-definition.ts +1 -1428
- package/src/route-map-builder.ts +211 -123
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +59 -8
- package/src/router/content-negotiation.ts +116 -0
- package/src/router/debug-manifest.ts +72 -0
- package/src/router/error-handling.ts +9 -9
- package/src/router/find-match.ts +158 -0
- package/src/router/handler-context.ts +374 -81
- package/src/router/intercept-resolution.ts +395 -0
- package/src/router/lazy-includes.ts +234 -0
- package/src/router/loader-resolution.ts +215 -122
- package/src/router/logging.ts +248 -0
- package/src/router/manifest.ts +148 -35
- package/src/router/match-api.ts +620 -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 +80 -93
- package/src/router/match-middleware/cache-lookup.ts +382 -9
- package/src/router/match-middleware/cache-store.ts +51 -22
- package/src/router/match-middleware/intercept-resolution.ts +55 -17
- package/src/router/match-middleware/segment-resolution.ts +24 -6
- package/src/router/match-pipelines.ts +10 -45
- package/src/router/match-result.ts +34 -28
- package/src/router/metrics.ts +235 -15
- package/src/router/middleware-cookies.ts +55 -0
- package/src/router/middleware-types.ts +222 -0
- package/src/router/middleware.ts +324 -367
- package/src/router/pattern-matching.ts +211 -43
- package/src/router/prerender-match.ts +402 -0
- package/src/router/preview-match.ts +170 -0
- package/src/router/revalidation.ts +137 -38
- package/src/router/router-context.ts +36 -21
- package/src/router/router-interfaces.ts +452 -0
- package/src/router/router-options.ts +592 -0
- package/src/router/router-registry.ts +24 -0
- package/src/router/segment-resolution/fresh.ts +570 -0
- package/src/router/segment-resolution/helpers.ts +263 -0
- package/src/router/segment-resolution/loader-cache.ts +198 -0
- package/src/router/segment-resolution/revalidation.ts +1241 -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 +289 -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 +77 -3
- package/src/router.ts +692 -4257
- package/src/rsc/handler-context.ts +45 -0
- package/src/rsc/handler.ts +764 -754
- package/src/rsc/helpers.ts +140 -6
- package/src/rsc/index.ts +0 -20
- package/src/rsc/loader-fetch.ts +209 -0
- package/src/rsc/manifest-init.ts +86 -0
- package/src/rsc/nonce.ts +14 -0
- package/src/rsc/origin-guard.ts +141 -0
- package/src/rsc/progressive-enhancement.ts +379 -0
- package/src/rsc/response-error.ts +37 -0
- package/src/rsc/response-route-handler.ts +347 -0
- package/src/rsc/rsc-rendering.ts +235 -0
- package/src/rsc/runtime-warnings.ts +42 -0
- package/src/rsc/server-action.ts +348 -0
- package/src/rsc/ssr-setup.ts +128 -0
- package/src/rsc/types.ts +38 -11
- package/src/search-params.ts +230 -0
- package/src/segment-system.tsx +25 -13
- package/src/server/context.ts +182 -51
- package/src/server/cookie-store.ts +190 -0
- package/src/server/fetchable-loader-store.ts +37 -0
- package/src/server/handle-store.ts +94 -15
- package/src/server/loader-registry.ts +15 -56
- package/src/server/request-context.ts +430 -70
- package/src/server.ts +35 -130
- package/src/ssr/index.tsx +100 -31
- package/src/static-handler.ts +114 -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 +687 -0
- package/src/types/index.ts +88 -0
- package/src/types/loader-types.ts +183 -0
- package/src/types/route-config.ts +170 -0
- package/src/types/route-entry.ts +102 -0
- package/src/types/segments.ts +148 -0
- package/src/types.ts +1 -1623
- package/src/urls/include-helper.ts +197 -0
- package/src/urls/index.ts +53 -0
- package/src/urls/path-helper-types.ts +339 -0
- package/src/urls/path-helper.ts +329 -0
- package/src/urls/pattern-types.ts +95 -0
- package/src/urls/response-types.ts +106 -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 +85 -77
- package/src/vite/discovery/bundle-postprocess.ts +184 -0
- package/src/vite/discovery/discover-routers.ts +344 -0
- package/src/vite/discovery/prerender-collection.ts +385 -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 +110 -0
- package/src/vite/discovery/virtual-module-codegen.ts +203 -0
- package/src/vite/index.ts +11 -1133
- package/src/vite/plugin-types.ts +131 -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 -51
- package/src/vite/plugins/expose-id-utils.ts +287 -0
- package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +179 -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 +569 -0
- package/src/vite/plugins/refresh-cmd.ts +65 -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 +254 -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 +510 -0
- package/src/vite/router-discovery.ts +785 -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 +189 -0
- package/src/vite/utils/shared-utils.ts +169 -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
|
@@ -8,12 +8,93 @@ argument-hint: [middleware-name]
|
|
|
8
8
|
|
|
9
9
|
Middleware runs before/after route handlers using the onion model.
|
|
10
10
|
|
|
11
|
+
## Execution Model
|
|
12
|
+
|
|
13
|
+
Canonical semantics reference:
|
|
14
|
+
[docs/execution-model.md](../../docs/internal/execution-model.md)
|
|
15
|
+
|
|
16
|
+
There are two levels of middleware with different execution scopes:
|
|
17
|
+
|
|
18
|
+
### Global middleware (`router.use()`)
|
|
19
|
+
|
|
20
|
+
Registered on the router instance. Wraps the **entire request**, including server actions, rendering, and progressive enhancement (PE) re-renders.
|
|
21
|
+
|
|
22
|
+
```typescript
|
|
23
|
+
const router = createRouter<AppEnv>({})
|
|
24
|
+
.use(loggerMiddleware) // all routes
|
|
25
|
+
.use("/admin/*", authMiddleware) // pattern-scoped
|
|
26
|
+
.routes(urlpatterns);
|
|
27
|
+
```
|
|
28
|
+
|
|
29
|
+
### Route middleware (`middleware()` in `urls()`)
|
|
30
|
+
|
|
31
|
+
Registered inside `urls()` callback. Wraps **rendering only** -- it does NOT wrap server action execution. Actions run before route middleware, so when route middleware executes during post-action revalidation, it can observe state that the action set (cookies, context variables, headers).
|
|
32
|
+
|
|
33
|
+
```
|
|
34
|
+
Request flow (with action):
|
|
35
|
+
global mw -> action executes -> route mw -> layout -> handler -> loaders
|
|
36
|
+
|
|
37
|
+
Request flow (no action):
|
|
38
|
+
global mw -> route mw -> layout -> handler -> loaders
|
|
39
|
+
|
|
40
|
+
Progressive enhancement (no-JS form POST):
|
|
41
|
+
global mw -> action executes -> route mw -> full page re-render
|
|
42
|
+
```
|
|
43
|
+
|
|
44
|
+
The contract is: **route middleware wraps rendering regardless of transport** (JS-enabled RSC stream or no-JS HTML). During PE re-render, route middleware observes action-set state (cookies, context variables) the same way it does during JS-enabled post-action revalidation.
|
|
45
|
+
|
|
46
|
+
Revalidation is still partial. Route middleware wraps the render pass that
|
|
47
|
+
does happen, but it does not force unrelated outer segments to recompute.
|
|
48
|
+
If a child segment depends on data established by an outer handler/layout,
|
|
49
|
+
revalidate that outer segment too, or have the child guard/reload the
|
|
50
|
+
data itself.
|
|
51
|
+
|
|
52
|
+
### Revalidation Contracts with Middleware-Backed Trees
|
|
53
|
+
|
|
54
|
+
Middleware can establish request-level context (`ctx.set`) for segments that
|
|
55
|
+
execute in the current render pass. It does not change partial revalidation
|
|
56
|
+
boundaries between handler/layout/parallel segments.
|
|
57
|
+
|
|
58
|
+
For shared segment data, use named revalidation contracts on both the producer
|
|
59
|
+
and consumer segments, even when middleware is present in the chain.
|
|
60
|
+
|
|
61
|
+
```typescript
|
|
62
|
+
export const revalidateCartData = ({ actionId }) =>
|
|
63
|
+
actionId?.includes("src/actions/cart.ts#") ?? false;
|
|
64
|
+
|
|
65
|
+
layout(CartLayout, () => [
|
|
66
|
+
middleware(cartRenderMiddleware),
|
|
67
|
+
revalidate(revalidateCartData), // producer reruns
|
|
68
|
+
parallel(
|
|
69
|
+
{ "@cart": CartSummary },
|
|
70
|
+
() => [revalidate(revalidateCartData)], // consumer reruns
|
|
71
|
+
),
|
|
72
|
+
]);
|
|
73
|
+
```
|
|
74
|
+
|
|
75
|
+
You can package those contracts as importable helpers to avoid repeating
|
|
76
|
+
`revalidate(...)` at each segment:
|
|
77
|
+
|
|
78
|
+
```typescript
|
|
79
|
+
import { revalidate } from "@rangojs/router";
|
|
80
|
+
|
|
81
|
+
export const revalidateCart = () => [revalidate(revalidateCartData)];
|
|
82
|
+
|
|
83
|
+
layout(CartLayout, () => [
|
|
84
|
+
middleware(cartRenderMiddleware),
|
|
85
|
+
revalidateCart(),
|
|
86
|
+
parallel({ "@cart": CartSummary }, () => [revalidateCart()]),
|
|
87
|
+
]);
|
|
88
|
+
```
|
|
89
|
+
|
|
90
|
+
Route middleware is the right place for per-route concerns that affect rendering (setting context variables for handlers, adding response headers, reading cookies set by actions). It is NOT the right place for action guards -- use global middleware for that.
|
|
91
|
+
|
|
11
92
|
## Basic Middleware
|
|
12
93
|
|
|
13
94
|
```typescript
|
|
14
|
-
import {
|
|
95
|
+
import type { Middleware } from "@rangojs/router";
|
|
15
96
|
|
|
16
|
-
export const authMiddleware =
|
|
97
|
+
export const authMiddleware: Middleware = async (ctx, next) => {
|
|
17
98
|
const token = ctx.request.headers.get("Authorization");
|
|
18
99
|
|
|
19
100
|
if (!token) {
|
|
@@ -21,10 +102,10 @@ export const authMiddleware = createMiddleware(async (ctx, next) => {
|
|
|
21
102
|
}
|
|
22
103
|
|
|
23
104
|
const user = await verifyToken(token);
|
|
24
|
-
ctx.
|
|
105
|
+
ctx.set("user", user);
|
|
25
106
|
|
|
26
107
|
await next();
|
|
27
|
-
}
|
|
108
|
+
};
|
|
28
109
|
```
|
|
29
110
|
|
|
30
111
|
## Using Middleware in Routes
|
|
@@ -68,42 +149,98 @@ layout(<ShopLayout />, () => [
|
|
|
68
149
|
## Middleware Context
|
|
69
150
|
|
|
70
151
|
```typescript
|
|
71
|
-
export const myMiddleware =
|
|
152
|
+
export const myMiddleware: Middleware = async (ctx, next) => {
|
|
72
153
|
// Access request
|
|
73
|
-
ctx.request;
|
|
74
|
-
ctx.url;
|
|
75
|
-
ctx.params;
|
|
154
|
+
ctx.request; // Request object
|
|
155
|
+
ctx.url; // Parsed URL
|
|
156
|
+
ctx.params; // Route parameters
|
|
76
157
|
|
|
77
|
-
// Access
|
|
78
|
-
ctx.env.
|
|
79
|
-
ctx.env.
|
|
158
|
+
// Access platform bindings (plain bindings from createRouter<TEnv>())
|
|
159
|
+
ctx.env.DB; // D1Database
|
|
160
|
+
ctx.env.KV; // KVNamespace
|
|
80
161
|
|
|
81
|
-
// Set variables for downstream handlers
|
|
82
|
-
ctx.
|
|
162
|
+
// Set variables for downstream handlers (typed via RSCRouter.Vars)
|
|
163
|
+
ctx.set("user", { id: "123", name: "John" });
|
|
83
164
|
|
|
84
165
|
// Continue to next middleware/handler
|
|
85
166
|
await next();
|
|
86
167
|
|
|
87
168
|
// After handler (response intercepting)
|
|
88
169
|
console.log("Handler completed");
|
|
170
|
+
};
|
|
171
|
+
```
|
|
172
|
+
|
|
173
|
+
### Typed context variables in middleware
|
|
174
|
+
|
|
175
|
+
Use `createVar<T>()` for type-safe data sharing between middleware and handlers:
|
|
176
|
+
|
|
177
|
+
```typescript
|
|
178
|
+
import { createVar } from "@rangojs/router";
|
|
179
|
+
import type { Middleware } from "@rangojs/router";
|
|
180
|
+
|
|
181
|
+
interface AuthUser { id: string; email: string; role: string }
|
|
182
|
+
export const CurrentUser = createVar<AuthUser>();
|
|
183
|
+
|
|
184
|
+
export const authMiddleware: Middleware = async (ctx, next) => {
|
|
185
|
+
const token = ctx.request.headers.get("Authorization");
|
|
186
|
+
if (!token) throw new Response("Unauthorized", { status: 401 });
|
|
187
|
+
|
|
188
|
+
const user = await verifyToken(token);
|
|
189
|
+
ctx.set(CurrentUser, user); // type-checked
|
|
190
|
+
await next();
|
|
191
|
+
};
|
|
192
|
+
|
|
193
|
+
// In a handler -- typed read
|
|
194
|
+
import { CurrentUser } from "./middleware";
|
|
195
|
+
|
|
196
|
+
const Dashboard: Handler<"dashboard"> = (ctx) => {
|
|
197
|
+
const user = ctx.get(CurrentUser); // typed as AuthUser | undefined
|
|
198
|
+
return <DashboardPage user={user!} />;
|
|
199
|
+
};
|
|
200
|
+
```
|
|
201
|
+
|
|
202
|
+
This works alongside `ctx.get("key")` / `ctx.set("key", value)` (global typing
|
|
203
|
+
via RSCRouter.Vars augmentation). Use `createVar` for route-local or feature-scoped
|
|
204
|
+
data; use RSCRouter.Vars for app-wide middleware state.
|
|
205
|
+
|
|
206
|
+
## Redirect with State in Middleware
|
|
207
|
+
|
|
208
|
+
```typescript
|
|
209
|
+
import { redirect, createLocationState } from "@rangojs/router";
|
|
210
|
+
import type { Middleware } from "@rangojs/router";
|
|
211
|
+
|
|
212
|
+
export const FlashMessage = createLocationState<{ text: string }>({
|
|
213
|
+
flash: true,
|
|
89
214
|
});
|
|
215
|
+
|
|
216
|
+
export const requireAuthMiddleware: Middleware = async (ctx, next) => {
|
|
217
|
+
const token = ctx.request.headers.get("Authorization");
|
|
218
|
+
if (!token) {
|
|
219
|
+
return redirect("/login", {
|
|
220
|
+
state: [FlashMessage({ text: "Please log in to continue" })],
|
|
221
|
+
});
|
|
222
|
+
}
|
|
223
|
+
await next();
|
|
224
|
+
};
|
|
90
225
|
```
|
|
91
226
|
|
|
227
|
+
Read the flash on the target page with `useLocationState(FlashMessage)`. The `{ flash: true }` option makes it auto-clear after first render. See `/hooks`.
|
|
228
|
+
|
|
92
229
|
## Authentication Middleware
|
|
93
230
|
|
|
94
231
|
```typescript
|
|
95
|
-
export const requireAuthMiddleware =
|
|
96
|
-
const user = ctx.
|
|
232
|
+
export const requireAuthMiddleware: Middleware = async (ctx, next) => {
|
|
233
|
+
const user = ctx.get("user");
|
|
97
234
|
|
|
98
235
|
if (!user) {
|
|
99
236
|
throw new Response("Unauthorized", { status: 401 });
|
|
100
237
|
}
|
|
101
238
|
|
|
102
239
|
await next();
|
|
103
|
-
}
|
|
240
|
+
};
|
|
104
241
|
|
|
105
|
-
export const permissionsMiddleware =
|
|
106
|
-
const user = ctx.
|
|
242
|
+
export const permissionsMiddleware: Middleware = async (ctx, next) => {
|
|
243
|
+
const user = ctx.get("user");
|
|
107
244
|
const requiredPermission = "admin";
|
|
108
245
|
|
|
109
246
|
if (!user?.permissions?.includes(requiredPermission)) {
|
|
@@ -111,13 +248,13 @@ export const permissionsMiddleware = createMiddleware(async (ctx, next) => {
|
|
|
111
248
|
}
|
|
112
249
|
|
|
113
250
|
await next();
|
|
114
|
-
}
|
|
251
|
+
};
|
|
115
252
|
```
|
|
116
253
|
|
|
117
254
|
## Logger Middleware
|
|
118
255
|
|
|
119
256
|
```typescript
|
|
120
|
-
export const loggerMiddleware =
|
|
257
|
+
export const loggerMiddleware: Middleware = async (ctx, next) => {
|
|
121
258
|
const start = Date.now();
|
|
122
259
|
|
|
123
260
|
console.log(`[${ctx.request.method}] ${ctx.url.pathname}`);
|
|
@@ -126,54 +263,54 @@ export const loggerMiddleware = createMiddleware(async (ctx, next) => {
|
|
|
126
263
|
|
|
127
264
|
const duration = Date.now() - start;
|
|
128
265
|
console.log(`[${ctx.request.method}] ${ctx.url.pathname} - ${duration}ms`);
|
|
129
|
-
}
|
|
266
|
+
};
|
|
130
267
|
```
|
|
131
268
|
|
|
132
269
|
## Rate Limiting Middleware
|
|
133
270
|
|
|
134
271
|
```typescript
|
|
135
|
-
export const rateLimitMiddleware =
|
|
272
|
+
export const rateLimitMiddleware: Middleware = async (ctx, next) => {
|
|
136
273
|
const ip = ctx.request.headers.get("CF-Connecting-IP") ?? "unknown";
|
|
137
274
|
const key = `rate-limit:${ip}`;
|
|
138
275
|
|
|
139
|
-
const count = await ctx.env.
|
|
276
|
+
const count = await ctx.env.KV.get(key);
|
|
140
277
|
const requests = count ? parseInt(count) : 0;
|
|
141
278
|
|
|
142
279
|
if (requests > 100) {
|
|
143
280
|
throw new Response("Too Many Requests", { status: 429 });
|
|
144
281
|
}
|
|
145
282
|
|
|
146
|
-
await ctx.env.
|
|
283
|
+
await ctx.env.KV.put(key, String(requests + 1), {
|
|
147
284
|
expirationTtl: 60,
|
|
148
285
|
});
|
|
149
286
|
|
|
150
287
|
await next();
|
|
151
|
-
}
|
|
288
|
+
};
|
|
152
289
|
```
|
|
153
290
|
|
|
154
291
|
## Complete Example
|
|
155
292
|
|
|
156
293
|
```typescript
|
|
157
294
|
// middleware/index.ts
|
|
158
|
-
import {
|
|
295
|
+
import type { Middleware } from "@rangojs/router";
|
|
159
296
|
|
|
160
|
-
export const loggerMiddleware =
|
|
297
|
+
export const loggerMiddleware: Middleware = async (ctx, next) => {
|
|
161
298
|
console.log(`[${ctx.request.method}] ${ctx.url.pathname}`);
|
|
162
299
|
await next();
|
|
163
|
-
}
|
|
300
|
+
};
|
|
164
301
|
|
|
165
|
-
export const mockAuthMiddleware =
|
|
302
|
+
export const mockAuthMiddleware: Middleware = async (ctx, next) => {
|
|
166
303
|
// Mock user for development
|
|
167
|
-
ctx.
|
|
304
|
+
ctx.set("user", { id: "1", name: "Demo User" });
|
|
168
305
|
await next();
|
|
169
|
-
}
|
|
306
|
+
};
|
|
170
307
|
|
|
171
|
-
export const requireAuthMiddleware =
|
|
172
|
-
if (!ctx.
|
|
308
|
+
export const requireAuthMiddleware: Middleware = async (ctx, next) => {
|
|
309
|
+
if (!ctx.get("user")) {
|
|
173
310
|
throw new Response("Unauthorized", { status: 401 });
|
|
174
311
|
}
|
|
175
312
|
await next();
|
|
176
|
-
}
|
|
313
|
+
};
|
|
177
314
|
|
|
178
315
|
// urls.tsx
|
|
179
316
|
import { urls } from "@rangojs/router";
|
|
@@ -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,41 @@ 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
|
+
|
|
57
95
|
## Parallel Routes with Loaders
|
|
58
96
|
|
|
59
97
|
Add loaders and loading states to parallel routes:
|
|
@@ -97,7 +135,7 @@ Render different content based on context:
|
|
|
97
135
|
```typescript
|
|
98
136
|
parallel({
|
|
99
137
|
"@sidebar": (ctx) => {
|
|
100
|
-
const user = ctx.
|
|
138
|
+
const user = ctx.get("user");
|
|
101
139
|
return user ? <UserSidebar user={user} /> : <GuestSidebar />;
|
|
102
140
|
},
|
|
103
141
|
})
|
|
@@ -120,6 +158,45 @@ parallel(
|
|
|
120
158
|
)
|
|
121
159
|
```
|
|
122
160
|
|
|
161
|
+
Revalidating only the parallel does not re-run outer handlers/layouts.
|
|
162
|
+
If the slot reads `ctx.get()` data established above it, opt the outer
|
|
163
|
+
segment into revalidation as well.
|
|
164
|
+
|
|
165
|
+
### Revalidation Contracts for Parallel Dependencies
|
|
166
|
+
|
|
167
|
+
Prefer named revalidation contracts shared by both the upstream producer and
|
|
168
|
+
the parallel consumer:
|
|
169
|
+
|
|
170
|
+
```typescript
|
|
171
|
+
// revalidation-contracts.ts
|
|
172
|
+
export const revalidateCartData = ({ actionId }) =>
|
|
173
|
+
actionId?.includes("src/actions/cart.ts#") ?? false;
|
|
174
|
+
|
|
175
|
+
layout(CartLayout, () => [
|
|
176
|
+
revalidate(revalidateCartData), // producer reruns
|
|
177
|
+
parallel(
|
|
178
|
+
{ "@cart": CartSummary },
|
|
179
|
+
() => [revalidate(revalidateCartData)], // consumer reruns
|
|
180
|
+
),
|
|
181
|
+
]);
|
|
182
|
+
```
|
|
183
|
+
|
|
184
|
+
If the slot consumes multiple upstream domains, compose the contracts on both
|
|
185
|
+
segments.
|
|
186
|
+
|
|
187
|
+
Handoff helper style also works:
|
|
188
|
+
|
|
189
|
+
```typescript
|
|
190
|
+
import { revalidate } from "@rangojs/router";
|
|
191
|
+
|
|
192
|
+
export const revalidateCart = () => [revalidate(revalidateCartData)];
|
|
193
|
+
|
|
194
|
+
layout(CartLayout, () => [
|
|
195
|
+
revalidateCart(),
|
|
196
|
+
parallel({ "@cart": CartSummary }, () => [revalidateCart()]),
|
|
197
|
+
]);
|
|
198
|
+
```
|
|
199
|
+
|
|
123
200
|
## Named Outlets
|
|
124
201
|
|
|
125
202
|
Use `ParallelOutlet` to render slots in layouts:
|