@rangojs/router 0.0.0-experimental.66 → 0.0.0-experimental.66cdebe3
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/README.md +112 -17
- package/dist/vite/index.js +1462 -422
- package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/package.json +7 -5
- package/skills/breadcrumbs/SKILL.md +3 -1
- package/skills/handler-use/SKILL.md +364 -0
- package/skills/hooks/SKILL.md +54 -20
- package/skills/i18n/SKILL.md +276 -0
- package/skills/intercept/SKILL.md +45 -0
- package/skills/layout/SKILL.md +24 -0
- package/skills/links/SKILL.md +234 -16
- package/skills/loader/SKILL.md +70 -3
- package/skills/middleware/SKILL.md +34 -3
- package/skills/migrate-nextjs/SKILL.md +562 -0
- package/skills/migrate-react-router/SKILL.md +769 -0
- package/skills/parallel/SKILL.md +68 -0
- package/skills/rango/SKILL.md +26 -22
- package/skills/response-routes/SKILL.md +8 -0
- package/skills/route/SKILL.md +48 -0
- package/skills/server-actions/SKILL.md +739 -0
- package/skills/streams-and-websockets/SKILL.md +283 -0
- package/skills/typesafety/SKILL.md +9 -1
- package/skills/view-transitions/SKILL.md +212 -0
- package/src/browser/app-shell.ts +52 -0
- package/src/browser/event-controller.ts +44 -4
- package/src/browser/navigation-bridge.ts +151 -9
- package/src/browser/navigation-client.ts +64 -13
- package/src/browser/navigation-store.ts +25 -1
- package/src/browser/partial-update.ts +58 -12
- package/src/browser/prefetch/cache.ts +129 -21
- package/src/browser/prefetch/fetch.ts +148 -16
- package/src/browser/prefetch/queue.ts +36 -5
- package/src/browser/rango-state.ts +53 -13
- package/src/browser/react/Link.tsx +30 -2
- package/src/browser/react/NavigationProvider.tsx +95 -44
- package/src/browser/react/filter-segment-order.ts +51 -7
- package/src/browser/react/index.ts +3 -0
- package/src/browser/react/use-navigation.ts +22 -2
- package/src/browser/react/use-params.ts +17 -4
- package/src/browser/react/use-reverse.ts +99 -0
- package/src/browser/react/use-router.ts +8 -1
- package/src/browser/react/use-segments.ts +11 -8
- package/src/browser/rsc-router.tsx +34 -6
- package/src/browser/scroll-restoration.ts +69 -28
- package/src/browser/segment-reconciler.ts +36 -14
- package/src/browser/types.ts +19 -0
- package/src/build/route-trie.ts +52 -25
- package/src/cache/cf/cf-cache-store.ts +5 -7
- package/src/client.rsc.tsx +3 -0
- package/src/client.tsx +87 -175
- package/src/href-client.ts +4 -1
- package/src/index.rsc.ts +3 -0
- package/src/index.ts +44 -9
- package/src/outlet-context.ts +1 -1
- package/src/response-utils.ts +28 -0
- package/src/reverse.ts +62 -36
- package/src/route-definition/dsl-helpers.ts +175 -23
- package/src/route-definition/helpers-types.ts +63 -14
- package/src/route-definition/resolve-handler-use.ts +6 -0
- package/src/route-types.ts +7 -0
- package/src/router/handler-context.ts +21 -38
- package/src/router/lazy-includes.ts +6 -6
- package/src/router/loader-resolution.ts +3 -0
- package/src/router/manifest.ts +22 -13
- package/src/router/match-api.ts +4 -3
- package/src/router/match-handlers.ts +1 -0
- package/src/router/match-middleware/cache-lookup.ts +2 -1
- package/src/router/match-result.ts +101 -4
- package/src/router/middleware-types.ts +14 -25
- package/src/router/middleware.ts +54 -7
- package/src/router/pattern-matching.ts +101 -17
- package/src/router/revalidation.ts +15 -1
- package/src/router/segment-resolution/fresh.ts +13 -0
- package/src/router/segment-resolution/revalidation.ts +135 -101
- package/src/router/substitute-pattern-params.ts +56 -0
- package/src/router/trie-matching.ts +18 -13
- package/src/router/url-params.ts +49 -0
- package/src/router.ts +1 -2
- package/src/rsc/handler.ts +16 -8
- package/src/rsc/helpers.ts +69 -41
- package/src/rsc/progressive-enhancement.ts +4 -0
- package/src/rsc/response-route-handler.ts +14 -1
- package/src/rsc/rsc-rendering.ts +10 -0
- package/src/rsc/server-action.ts +4 -0
- package/src/rsc/types.ts +6 -0
- package/src/segment-content-promise.ts +67 -0
- package/src/segment-loader-promise.ts +122 -0
- package/src/segment-system.tsx +71 -70
- package/src/server/context.ts +26 -3
- package/src/server/request-context.ts +10 -42
- package/src/ssr/index.tsx +5 -1
- package/src/types/handler-context.ts +12 -39
- package/src/types/loader-types.ts +5 -6
- package/src/types/request-scope.ts +126 -0
- package/src/types/route-entry.ts +11 -0
- package/src/types/segments.ts +18 -1
- package/src/urls/include-helper.ts +24 -14
- package/src/urls/path-helper-types.ts +30 -4
- package/src/urls/response-types.ts +2 -10
- package/src/use-loader.tsx +4 -1
- package/src/vite/debug.ts +184 -0
- package/src/vite/discovery/discover-routers.ts +31 -3
- package/src/vite/discovery/gate-state.ts +171 -0
- package/src/vite/discovery/prerender-collection.ts +172 -84
- package/src/vite/discovery/self-gen-tracking.ts +27 -1
- package/src/vite/plugins/cjs-to-esm.ts +5 -0
- package/src/vite/plugins/client-ref-dedup.ts +16 -0
- package/src/vite/plugins/client-ref-hashing.ts +16 -4
- 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 +214 -0
- package/src/vite/plugins/expose-action-id.ts +52 -28
- package/src/vite/plugins/expose-id-utils.ts +12 -0
- package/src/vite/plugins/expose-ids/router-transform.ts +20 -3
- package/src/vite/plugins/expose-internal-ids.ts +545 -304
- package/src/vite/plugins/performance-tracks.ts +17 -9
- package/src/vite/plugins/use-cache-transform.ts +56 -43
- package/src/vite/plugins/version-injector.ts +37 -11
- package/src/vite/rango.ts +49 -14
- package/src/vite/router-discovery.ts +558 -53
- package/src/vite/utils/banner.ts +1 -1
- package/src/vite/utils/package-resolution.ts +41 -1
- package/src/vite/utils/prerender-utils.ts +21 -6
package/skills/loader/SKILL.md
CHANGED
|
@@ -6,7 +6,10 @@ argument-hint: [loader]
|
|
|
6
6
|
|
|
7
7
|
# Data Loaders with loader()
|
|
8
8
|
|
|
9
|
-
Loaders fetch data on the server and stream it to the client.
|
|
9
|
+
Loaders fetch data on the server and stream it to the client. For mutations
|
|
10
|
+
(writes triggered by forms or buttons), use server actions instead — see
|
|
11
|
+
`/server-actions`. Loaders re-resolve after an action runs, so the typical
|
|
12
|
+
flow is _action mutates → loader re-reads → UI updates_.
|
|
10
13
|
|
|
11
14
|
## Creating a Loader
|
|
12
15
|
|
|
@@ -139,7 +142,29 @@ same memoized result — loaders never run twice per request.
|
|
|
139
142
|
|
|
140
143
|
## Loader Context
|
|
141
144
|
|
|
142
|
-
Loaders receive the same context as route handlers
|
|
145
|
+
Loaders receive the same context shape as route handlers.
|
|
146
|
+
|
|
147
|
+
### Full field surface
|
|
148
|
+
|
|
149
|
+
| Field | Type | Notes |
|
|
150
|
+
| -------------- | ------------------------------ | --------------------------------------------------------------------------------------------------- |
|
|
151
|
+
| `params` | `TParams` | Merged route + explicit loader params; overridable by fetchable `load({ params })`. |
|
|
152
|
+
| `routeParams` | `Record<string, string>` | Server-trusted route params from URL pattern matching; cannot be overridden. |
|
|
153
|
+
| `request` | `Request` | The incoming `Request` (headers, method, body, `signal` for abort). |
|
|
154
|
+
| `url` | `URL` | Parsed request URL. |
|
|
155
|
+
| `pathname` | `string` | URL pathname (shortcut for `ctx.url.pathname`). |
|
|
156
|
+
| `searchParams` | `URLSearchParams` | Shortcut for `ctx.url.searchParams`. |
|
|
157
|
+
| `search` | `ResolveSearchSchema<TSearch>` | Typed query params when a search schema is declared on the route; `{}` otherwise. |
|
|
158
|
+
| `env` | `TEnv` | Plain bindings from `createRouter<TEnv>()` (DB, KV, secrets, etc.). |
|
|
159
|
+
| `get` | `(key \| ContextVar) => value` | Reads variables/context-vars set by middleware. |
|
|
160
|
+
| `use` | `(loader \| handle) => T` | Access another loader's data (Promise) or a handle's collected data (after `await ctx.rendered()`). |
|
|
161
|
+
| `rendered` | `() => Promise<void>` | **Experimental.** DSL loaders only — waits for non-loader segments before reading handle data. |
|
|
162
|
+
| `method` | `string` | HTTP method. `"GET"` for SSR loader runs; reflects real method for fetchable loaders. |
|
|
163
|
+
| `body` | `TBody \| undefined` | Parsed request body for fetchable POST/PUT/PATCH/DELETE calls. |
|
|
164
|
+
| `formData` | `FormData \| undefined` | Present when a fetchable loader is invoked via form submission. |
|
|
165
|
+
| `reverse` | `ScopedReverseFunction` | Generate type-checked URLs from route names (same scoped semantics as route handlers). |
|
|
166
|
+
|
|
167
|
+
### Example
|
|
143
168
|
|
|
144
169
|
```typescript
|
|
145
170
|
export const ProductLoader = createLoader(async (ctx) => {
|
|
@@ -163,10 +188,21 @@ export const ProductLoader = createLoader(async (ctx) => {
|
|
|
163
188
|
// Variables set by middleware (from RSCRouter.Vars augmentation)
|
|
164
189
|
const user = ctx.get("user");
|
|
165
190
|
|
|
166
|
-
|
|
191
|
+
// Type-checked URLs for payloads. `.name` resolves within the current
|
|
192
|
+
// include() scope; a bare `name` resolves globally. See /route and
|
|
193
|
+
// /typesafety for scope rules and route-name autocomplete.
|
|
194
|
+
const detailUrl = ctx.reverse(".detail", { slug });
|
|
195
|
+
|
|
196
|
+
return {
|
|
197
|
+
product: await fetchProduct(slug),
|
|
198
|
+
links: { self: detailUrl },
|
|
199
|
+
};
|
|
167
200
|
});
|
|
168
201
|
```
|
|
169
202
|
|
|
203
|
+
See `/route` for the full handler-context contract (shared with loaders) and
|
|
204
|
+
`/typesafety` for route-name typing that powers `ctx.reverse` autocomplete.
|
|
205
|
+
|
|
170
206
|
### params vs routeParams
|
|
171
207
|
|
|
172
208
|
- `ctx.params` — merged route params + explicit loader params. For fetchable
|
|
@@ -215,6 +251,37 @@ path("/product/:slug", ProductPage, { name: "product" }, () => [
|
|
|
215
251
|
]);
|
|
216
252
|
```
|
|
217
253
|
|
|
254
|
+
### `revalidate()` return shapes
|
|
255
|
+
|
|
256
|
+
A `revalidate(fn)` callback can return one of four shapes. The chain
|
|
257
|
+
processes revalidators in order; each call's return controls how the
|
|
258
|
+
chain continues:
|
|
259
|
+
|
|
260
|
+
```typescript
|
|
261
|
+
// 1) Hard decision — short-circuits the chain, used as the final answer.
|
|
262
|
+
revalidate(() => true);
|
|
263
|
+
revalidate(({ actionId }) => actionId?.includes("Cart") ?? false);
|
|
264
|
+
|
|
265
|
+
// 2) Soft decision — updates the running suggestion for downstream
|
|
266
|
+
// revalidators on the same segment, chain continues.
|
|
267
|
+
revalidate(({ defaultShouldRevalidate }) => ({
|
|
268
|
+
defaultShouldRevalidate: !defaultShouldRevalidate,
|
|
269
|
+
}));
|
|
270
|
+
|
|
271
|
+
// 3) Defer (no opinion) — leaves the running suggestion unchanged and
|
|
272
|
+
// continues to the next revalidator. Implicit return / null /
|
|
273
|
+
// undefined are all equivalent and consumer-friendly.
|
|
274
|
+
revalidate(({ actionId }) => {
|
|
275
|
+
if (actionId?.includes("Cart")) return true; // hard for this branch only
|
|
276
|
+
// implicit return — let downstream revalidators or the segment default decide
|
|
277
|
+
});
|
|
278
|
+
revalidate(() => undefined); // explicit defer
|
|
279
|
+
revalidate(() => null); // explicit defer
|
|
280
|
+
```
|
|
281
|
+
|
|
282
|
+
If every revalidator on a segment defers, the segment-type default
|
|
283
|
+
(e.g. params-changed for routes, `false` for parallels) is used.
|
|
284
|
+
|
|
218
285
|
### Revalidation Contracts for Loader Dependencies
|
|
219
286
|
|
|
220
287
|
If a loader reads `ctx.get()` data produced by an outer handler/layout, share
|
|
@@ -32,6 +32,8 @@ When the router has a `basename`, pattern-scoped `.use()` patterns are automatic
|
|
|
32
32
|
|
|
33
33
|
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).
|
|
34
34
|
|
|
35
|
+
> **Implication for auth:** route middleware cannot guard server actions. Use `router.use("/admin/*", requireAuth)` (global, scoped) for action protection, or check inside the action body. See `/server-actions` for action-side auth patterns.
|
|
36
|
+
|
|
35
37
|
```
|
|
36
38
|
Request flow (with action):
|
|
37
39
|
global mw -> action executes -> route mw -> layout -> handler -> loaders
|
|
@@ -137,17 +139,46 @@ export const urlpatterns = urls(({ path, layout, middleware }) => [
|
|
|
137
139
|
## Middleware with Multiple Handlers
|
|
138
140
|
|
|
139
141
|
```typescript
|
|
140
|
-
//
|
|
142
|
+
// Group multiple middleware in an array
|
|
141
143
|
export const shopMiddleware = [loggerMiddleware, mockAuthMiddleware];
|
|
142
144
|
|
|
143
|
-
// In routes
|
|
145
|
+
// In routes — pass the array directly
|
|
144
146
|
layout(<ShopLayout />, () => [
|
|
145
|
-
middleware(
|
|
147
|
+
middleware(shopMiddleware),
|
|
146
148
|
|
|
147
149
|
path("/shop", ShopIndex, { name: "shop" }),
|
|
148
150
|
])
|
|
149
151
|
```
|
|
150
152
|
|
|
153
|
+
## Wrapping Middleware (Scoped to Children)
|
|
154
|
+
|
|
155
|
+
Use the wrapping form to scope middleware to a subset of routes without
|
|
156
|
+
introducing a visible layout:
|
|
157
|
+
|
|
158
|
+
```typescript
|
|
159
|
+
urls(({ path, middleware }) => [
|
|
160
|
+
// authMw only applies to /admin and /admin/settings
|
|
161
|
+
middleware(authMw, () => [
|
|
162
|
+
path("/admin", AdminPage, { name: "admin" }),
|
|
163
|
+
path("/admin/settings", SettingsPage, { name: "settings" }),
|
|
164
|
+
]),
|
|
165
|
+
|
|
166
|
+
// Public route — no authMw
|
|
167
|
+
path("/", HomePage, { name: "home" }),
|
|
168
|
+
]);
|
|
169
|
+
```
|
|
170
|
+
|
|
171
|
+
Multiple middleware with wrapping:
|
|
172
|
+
|
|
173
|
+
```typescript
|
|
174
|
+
middleware([authMw, loggingMw], () => [
|
|
175
|
+
path("/admin", AdminPage, { name: "admin" }),
|
|
176
|
+
]);
|
|
177
|
+
```
|
|
178
|
+
|
|
179
|
+
This creates a transparent layout (`<Outlet />`) that carries the middleware.
|
|
180
|
+
The middleware does not affect sibling routes outside the callback.
|
|
181
|
+
|
|
151
182
|
## Middleware Context
|
|
152
183
|
|
|
153
184
|
```typescript
|