@rangojs/router 0.0.0-experimental.fb4fdc18 → 0.0.0-experimental.fce7fbd1
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 +9 -9
- package/dist/bin/rango.js +147 -57
- package/dist/testing/vitest.js +48 -0
- package/dist/vite/index.js +914 -485
- package/package.json +55 -11
- package/skills/bundle-analysis/SKILL.md +159 -0
- package/skills/cache-guide/SKILL.md +220 -30
- package/skills/caching/SKILL.md +116 -8
- package/skills/composability/SKILL.md +27 -2
- package/skills/document-cache/SKILL.md +78 -55
- package/skills/handler-use/SKILL.md +3 -1
- package/skills/hooks/SKILL.md +214 -18
- package/skills/host-router/SKILL.md +45 -20
- package/skills/intercept/SKILL.md +26 -4
- package/skills/layout/SKILL.md +6 -7
- package/skills/links/SKILL.md +173 -17
- package/skills/loader/SKILL.md +149 -6
- package/skills/middleware/SKILL.md +13 -9
- package/skills/migrate-nextjs/SKILL.md +1 -1
- package/skills/mime-routes/SKILL.md +27 -0
- package/skills/observability/SKILL.md +137 -0
- package/skills/parallel/SKILL.md +5 -6
- package/skills/prerender/SKILL.md +14 -33
- package/skills/rango/SKILL.md +242 -26
- package/skills/react-compiler/SKILL.md +168 -0
- package/skills/response-routes/SKILL.md +58 -9
- package/skills/route/SKILL.md +13 -4
- package/skills/router-setup/SKILL.md +3 -3
- package/skills/server-actions/SKILL.md +53 -41
- package/skills/testing/SKILL.md +599 -0
- package/skills/typesafety/SKILL.md +310 -26
- package/skills/use-cache/SKILL.md +34 -5
- package/skills/view-transitions/SKILL.md +294 -0
- package/src/__augment-tests__/augment.ts +81 -0
- package/src/__augment-tests__/augmented.check.ts +117 -0
- package/src/browser/action-coordinator.ts +53 -36
- package/src/browser/event-controller.ts +42 -66
- package/src/browser/history-state.ts +21 -0
- package/src/browser/index.ts +3 -3
- package/src/browser/navigation-bridge.ts +6 -6
- package/src/browser/navigation-client.ts +12 -15
- package/src/browser/navigation-store.ts +7 -8
- package/src/browser/navigation-transaction.ts +10 -28
- package/src/browser/partial-update.ts +9 -19
- package/src/browser/react/NavigationProvider.tsx +29 -40
- package/src/browser/react/index.ts +3 -0
- package/src/browser/react/location-state-shared.ts +175 -4
- package/src/browser/react/location-state.ts +39 -13
- package/src/browser/react/use-handle.ts +17 -9
- package/src/browser/react/use-params.ts +3 -4
- package/src/browser/react/use-reverse.ts +106 -0
- package/src/browser/react/use-router.ts +14 -1
- package/src/browser/response-adapter.ts +25 -0
- package/src/browser/rsc-router.tsx +30 -16
- package/src/browser/scroll-restoration.ts +22 -14
- package/src/browser/segment-structure-assert.ts +2 -2
- package/src/browser/server-action-bridge.ts +23 -30
- package/src/browser/types.ts +2 -0
- package/src/build/collect-fallback-refs.ts +107 -0
- package/src/build/generate-manifest.ts +60 -35
- package/src/build/generate-route-types.ts +2 -0
- package/src/build/index.ts +2 -0
- package/src/build/route-types/codegen.ts +4 -4
- package/src/build/route-types/include-resolution.ts +1 -1
- package/src/build/route-types/per-module-writer.ts +7 -4
- package/src/build/route-types/router-processing.ts +55 -14
- package/src/build/route-types/scan-filter.ts +1 -1
- package/src/build/route-types/source-scan.ts +118 -0
- package/src/build/runtime-discovery.ts +9 -20
- package/src/cache/cache-scope.ts +28 -42
- package/src/cache/cf/cf-cache-store.ts +49 -6
- package/src/client.rsc.tsx +3 -0
- package/src/client.tsx +10 -8
- package/src/context-var.ts +5 -5
- package/src/decode-loader-results.ts +36 -0
- package/src/errors.ts +30 -1
- package/src/handle.ts +26 -13
- package/src/host/index.ts +2 -2
- package/src/host/router.ts +129 -57
- package/src/host/types.ts +31 -2
- package/src/host/utils.ts +1 -1
- package/src/href-client.ts +140 -20
- package/src/index.rsc.ts +6 -4
- package/src/index.ts +13 -6
- package/src/loader-store.ts +500 -0
- package/src/loader.rsc.ts +2 -5
- package/src/loader.ts +3 -10
- package/src/missing-id-error.ts +68 -0
- package/src/prerender.ts +4 -4
- package/src/response-utils.ts +9 -0
- package/src/reverse.ts +65 -41
- package/src/route-content-wrapper.tsx +6 -28
- package/src/route-definition/dsl-helpers.ts +238 -263
- package/src/route-definition/helper-factories.ts +29 -139
- package/src/route-definition/helpers-types.ts +37 -14
- package/src/route-definition/use-item-types.ts +32 -0
- package/src/route-types.ts +19 -41
- package/src/router/basename.ts +14 -0
- package/src/router/content-negotiation.ts +15 -2
- package/src/router/error-handling.ts +1 -1
- package/src/router/handler-context.ts +4 -42
- package/src/router/intercept-resolution.ts +4 -18
- package/src/router/lazy-includes.ts +2 -2
- package/src/router/loader-resolution.ts +16 -2
- package/src/router/match-handlers.ts +62 -20
- package/src/router/match-middleware/cache-lookup.ts +44 -91
- package/src/router/match-middleware/cache-store.ts +3 -2
- package/src/router/match-result.ts +32 -30
- package/src/router/metrics.ts +1 -1
- package/src/router/middleware-types.ts +1 -1
- package/src/router/middleware.ts +46 -78
- package/src/router/prerender-match.ts +1 -1
- package/src/router/preview-match.ts +3 -1
- package/src/router/request-classification.ts +4 -28
- package/src/router/revalidation.ts +43 -1
- package/src/router/router-interfaces.ts +45 -28
- package/src/router/router-options.ts +40 -1
- package/src/router/router-registry.ts +2 -5
- package/src/router/segment-resolution/fresh.ts +19 -6
- package/src/router/segment-resolution/revalidation.ts +19 -6
- package/src/router/segment-resolution/view-transition-default.ts +36 -0
- package/src/router/substitute-pattern-params.ts +56 -0
- package/src/router/telemetry.ts +99 -0
- package/src/router/types.ts +8 -0
- package/src/router.ts +37 -21
- package/src/rsc/handler-context.ts +2 -2
- package/src/rsc/handler.ts +20 -65
- package/src/rsc/helpers.ts +22 -2
- package/src/rsc/index.ts +1 -1
- package/src/rsc/origin-guard.ts +28 -10
- package/src/rsc/response-route-handler.ts +32 -52
- package/src/rsc/rsc-rendering.ts +27 -53
- package/src/rsc/runtime-warnings.ts +9 -10
- package/src/rsc/server-action.ts +13 -37
- package/src/rsc/ssr-setup.ts +16 -0
- package/src/rsc/types.ts +2 -2
- package/src/search-params.ts +4 -4
- package/src/segment-system.tsx +121 -65
- package/src/serialize.ts +243 -0
- package/src/server/context.ts +118 -51
- package/src/server/cookie-store.ts +28 -4
- package/src/server/request-context.ts +10 -0
- package/src/static-handler.ts +1 -1
- package/src/testing/cache-status.ts +166 -0
- package/src/testing/collect-handle.ts +63 -0
- package/src/testing/dispatch.ts +440 -0
- package/src/testing/dom.entry.ts +22 -0
- package/src/testing/e2e/fixture.ts +154 -0
- package/src/testing/e2e/index.ts +149 -0
- package/src/testing/e2e/matchers.ts +51 -0
- package/src/testing/e2e/page-helpers.ts +272 -0
- package/src/testing/e2e/parity.ts +306 -0
- package/src/testing/e2e/server.ts +183 -0
- package/src/testing/flight-matchers.ts +104 -0
- package/src/testing/flight-runtime.d.ts +21 -0
- package/src/testing/flight.entry.ts +22 -0
- package/src/testing/flight.ts +182 -0
- package/src/testing/generated-routes.ts +223 -0
- package/src/testing/index.ts +105 -0
- package/src/testing/internal/context.ts +193 -0
- package/src/testing/render-route.tsx +536 -0
- package/src/testing/run-loader.ts +296 -0
- package/src/testing/run-middleware.ts +170 -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 +183 -0
- package/src/types/global-namespace.ts +39 -26
- package/src/types/handler-context.ts +56 -11
- package/src/types/index.ts +1 -0
- package/src/types/segments.ts +18 -1
- package/src/urls/include-helper.ts +10 -53
- package/src/urls/index.ts +0 -3
- package/src/urls/path-helper-types.ts +11 -3
- package/src/urls/path-helper.ts +17 -52
- package/src/urls/pattern-types.ts +36 -19
- package/src/urls/response-types.ts +20 -19
- package/src/urls/type-extraction.ts +26 -116
- package/src/urls/urls-function.ts +1 -5
- package/src/use-loader.tsx +413 -42
- package/src/vite/debug.ts +1 -0
- package/src/vite/discovery/bundle-postprocess.ts +6 -6
- package/src/vite/discovery/discover-routers.ts +70 -48
- package/src/vite/discovery/discovery-errors.ts +194 -0
- package/src/vite/discovery/prerender-collection.ts +19 -25
- package/src/vite/discovery/route-types-writer.ts +40 -84
- package/src/vite/discovery/state.ts +33 -0
- package/src/vite/discovery/virtual-module-codegen.ts +13 -23
- package/src/vite/index.ts +2 -0
- package/src/vite/plugin-types.ts +67 -0
- package/src/vite/plugins/cjs-to-esm.ts +3 -7
- package/src/vite/plugins/client-ref-hashing.ts +12 -1
- package/src/vite/plugins/cloudflare-protocol-stub.ts +1 -1
- package/src/vite/plugins/expose-action-id.ts +2 -2
- package/src/vite/plugins/expose-id-utils.ts +12 -8
- package/src/vite/plugins/expose-ids/export-analysis.ts +100 -20
- package/src/vite/plugins/expose-ids/handler-transform.ts +8 -61
- package/src/vite/plugins/expose-ids/loader-transform.ts +3 -5
- package/src/vite/plugins/expose-internal-ids.ts +47 -67
- package/src/vite/plugins/performance-tracks.ts +12 -16
- package/src/vite/plugins/use-cache-transform.ts +13 -11
- package/src/vite/plugins/version-injector.ts +2 -12
- package/src/vite/plugins/version-plugin.ts +59 -2
- package/src/vite/plugins/virtual-entries.ts +2 -2
- package/src/vite/rango.ts +67 -15
- package/src/vite/router-discovery.ts +208 -63
- package/src/vite/utils/ast-handler-extract.ts +15 -15
- package/src/vite/utils/bundle-analysis.ts +4 -2
- package/src/vite/utils/client-chunks.ts +190 -0
- package/src/vite/utils/forward-user-plugins.ts +193 -0
- package/src/vite/utils/manifest-utils.ts +21 -5
- package/src/vite/utils/shared-utils.ts +107 -26
- package/src/browser/action-response-classifier.ts +0 -99
package/src/urls/path-helper.ts
CHANGED
|
@@ -1,16 +1,11 @@
|
|
|
1
1
|
import type { ReactNode } from "react";
|
|
2
2
|
import type { Handler } from "../types.js";
|
|
3
|
-
import type {
|
|
4
|
-
AllUseItems,
|
|
5
|
-
RouteItem,
|
|
6
|
-
RouteUseItem,
|
|
7
|
-
UseItems,
|
|
8
|
-
} from "../route-types.js";
|
|
3
|
+
import type { RouteItem, RouteUseItem, UseItems } from "../route-types.js";
|
|
9
4
|
import {
|
|
10
|
-
getContext,
|
|
11
5
|
getUrlPrefix,
|
|
12
6
|
getNamePrefix,
|
|
13
7
|
getRootScoped,
|
|
8
|
+
requireDslContext,
|
|
14
9
|
} from "../server/context";
|
|
15
10
|
import { invariant, DataNotFoundError } from "../errors";
|
|
16
11
|
import { validateUserRouteName } from "../route-name.js";
|
|
@@ -39,35 +34,10 @@ import {
|
|
|
39
34
|
resolveHandlerUse,
|
|
40
35
|
mergeHandlerUse,
|
|
41
36
|
} from "../route-definition/resolve-handler-use.js";
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
const isValidUseItem = (item: any): item is AllUseItems | undefined | null => {
|
|
47
|
-
return (
|
|
48
|
-
typeof item === "undefined" ||
|
|
49
|
-
item === null ||
|
|
50
|
-
(item &&
|
|
51
|
-
typeof item === "object" &&
|
|
52
|
-
"type" in item &&
|
|
53
|
-
[
|
|
54
|
-
"layout",
|
|
55
|
-
"route",
|
|
56
|
-
"middleware",
|
|
57
|
-
"revalidate",
|
|
58
|
-
"parallel",
|
|
59
|
-
"intercept",
|
|
60
|
-
"loader",
|
|
61
|
-
"loading",
|
|
62
|
-
"errorBoundary",
|
|
63
|
-
"notFoundBoundary",
|
|
64
|
-
"when",
|
|
65
|
-
"cache",
|
|
66
|
-
"transition",
|
|
67
|
-
"include",
|
|
68
|
-
].includes(item.type))
|
|
69
|
-
);
|
|
70
|
-
};
|
|
37
|
+
import {
|
|
38
|
+
emptySegmentBase,
|
|
39
|
+
runAndValidateUseItems,
|
|
40
|
+
} from "../route-definition/dsl-helpers.js";
|
|
71
41
|
|
|
72
42
|
/**
|
|
73
43
|
* Apply URL prefix to a pattern
|
|
@@ -112,9 +82,9 @@ export function createPathHelper<TEnv>(): PathFn<TEnv> {
|
|
|
112
82
|
optionsOrUse?: PathOptions | (() => UseItems<RouteUseItem>),
|
|
113
83
|
maybeUse?: () => UseItems<RouteUseItem>,
|
|
114
84
|
): RouteItem => {
|
|
115
|
-
const store =
|
|
116
|
-
|
|
117
|
-
|
|
85
|
+
const { store, ctx } = requireDslContext(
|
|
86
|
+
"path() must be called inside urls()",
|
|
87
|
+
);
|
|
118
88
|
|
|
119
89
|
invariant(
|
|
120
90
|
!ctx.parent || ctx.parent.type !== "parallel",
|
|
@@ -214,6 +184,7 @@ export function createPathHelper<TEnv>(): PathFn<TEnv> {
|
|
|
214
184
|
: () => handler;
|
|
215
185
|
|
|
216
186
|
const entry = {
|
|
187
|
+
...emptySegmentBase(),
|
|
217
188
|
id: namespace,
|
|
218
189
|
shortCode: store.getShortCode("route"),
|
|
219
190
|
type: "route" as const,
|
|
@@ -221,15 +192,6 @@ export function createPathHelper<TEnv>(): PathFn<TEnv> {
|
|
|
221
192
|
handler: wrappedHandler,
|
|
222
193
|
// Store the PREFIXED pattern for route matching
|
|
223
194
|
pattern: prefixedPattern,
|
|
224
|
-
loading: undefined,
|
|
225
|
-
middleware: [],
|
|
226
|
-
revalidate: [],
|
|
227
|
-
errorBoundary: [],
|
|
228
|
-
notFoundBoundary: [],
|
|
229
|
-
layout: [],
|
|
230
|
-
parallel: {},
|
|
231
|
-
intercept: [],
|
|
232
|
-
loader: [],
|
|
233
195
|
...(urlPrefix ? { mountPath: urlPrefix } : {}),
|
|
234
196
|
...(isPassthroughHandler(handler)
|
|
235
197
|
? {
|
|
@@ -301,10 +263,13 @@ export function createPathHelper<TEnv>(): PathFn<TEnv> {
|
|
|
301
263
|
|
|
302
264
|
// Run merged use callback (handler.use defaults + explicit use) if present
|
|
303
265
|
if (mergedUse) {
|
|
304
|
-
const result =
|
|
305
|
-
|
|
306
|
-
|
|
307
|
-
|
|
266
|
+
const result = runAndValidateUseItems(
|
|
267
|
+
store,
|
|
268
|
+
namespace,
|
|
269
|
+
entry,
|
|
270
|
+
mergedUse,
|
|
271
|
+
"path",
|
|
272
|
+
"use",
|
|
308
273
|
);
|
|
309
274
|
return { name: namespace, type: "route", uses: result } as RouteItem;
|
|
310
275
|
}
|
|
@@ -1,10 +1,5 @@
|
|
|
1
|
-
import type {
|
|
2
|
-
import type {
|
|
3
|
-
import type {
|
|
4
|
-
AllUseItems,
|
|
5
|
-
RouteUseItem,
|
|
6
|
-
UrlPatternsBrand,
|
|
7
|
-
} from "../route-types.js";
|
|
1
|
+
import type { TrailingSlashMode } from "../types.js";
|
|
2
|
+
import type { AllUseItems, UrlPatternsBrand } from "../route-types.js";
|
|
8
3
|
import type { SearchSchema } from "../search-params.js";
|
|
9
4
|
import { RESPONSE_TYPE } from "./response-types.js";
|
|
10
5
|
import type { DefaultEnv } from "../types.js";
|
|
@@ -54,16 +49,6 @@ export interface PathOptions<
|
|
|
54
49
|
[RESPONSE_TYPE]?: string;
|
|
55
50
|
}
|
|
56
51
|
|
|
57
|
-
/**
|
|
58
|
-
* Internal representation of a URL pattern definition
|
|
59
|
-
*/
|
|
60
|
-
export interface PathDefinition {
|
|
61
|
-
pattern: string;
|
|
62
|
-
name?: string;
|
|
63
|
-
handler: ReactNode | Handler<any, any, any>;
|
|
64
|
-
use?: RouteUseItem[];
|
|
65
|
-
}
|
|
66
|
-
|
|
67
52
|
/**
|
|
68
53
|
* Result of urls() - contains the route definitions
|
|
69
54
|
*/
|
|
@@ -72,8 +57,6 @@ export interface UrlPatterns<
|
|
|
72
57
|
TRoutes extends Record<string, any> = Record<string, string>,
|
|
73
58
|
TResponses extends Record<string, unknown> = Record<string, unknown>,
|
|
74
59
|
> {
|
|
75
|
-
/** Internal: route definitions */
|
|
76
|
-
readonly definitions: PathDefinition[];
|
|
77
60
|
/** Internal: compiled handler function */
|
|
78
61
|
readonly handler: () => AllUseItems[];
|
|
79
62
|
/** Internal: trailing slash config per route name */
|
|
@@ -88,6 +71,40 @@ export interface UrlPatterns<
|
|
|
88
71
|
readonly _responses?: TResponses;
|
|
89
72
|
}
|
|
90
73
|
|
|
74
|
+
/**
|
|
75
|
+
* Extract the phantom env type carried by a UrlPatterns value.
|
|
76
|
+
*/
|
|
77
|
+
export type UrlPatternsEnv<T> =
|
|
78
|
+
T extends UrlPatterns<infer TEnv, any, any> ? TEnv : never;
|
|
79
|
+
|
|
80
|
+
/**
|
|
81
|
+
* Guards `routes()` env compatibility without over-constraining.
|
|
82
|
+
*
|
|
83
|
+
* - An env-agnostic block (its env is `unknown` — e.g. a shared urls() module,
|
|
84
|
+
* or an app that does not augment `Rango.Env`) attaches to any router.
|
|
85
|
+
* - A block carrying a concrete env is accepted only when the router env
|
|
86
|
+
* (`TRouterEnv`) satisfies it; resolves to `never` otherwise, so a
|
|
87
|
+
* `urls<{ DB: D1Database }>()` cannot be mounted on a `createRouter<{}>()`.
|
|
88
|
+
*
|
|
89
|
+
* Use as `patterns: T & EnvCompatible<T, TEnv>` so `T` still infers from the
|
|
90
|
+
* argument — a bare `EnvCompatible<T, TEnv>` parameter sits in a non-inferrable
|
|
91
|
+
* conditional position and would collapse `T` to its constraint.
|
|
92
|
+
*
|
|
93
|
+
* Known limitation: `TRouterEnv extends ...` distributes over a union router env,
|
|
94
|
+
* so a `urls<A>()` block is accepted on `createRouter<A | B>()` even though the
|
|
95
|
+
* `B` arm cannot supply `A`'s env. Suppressing distribution with
|
|
96
|
+
* `[TRouterEnv] extends [...]` would close that edge but breaks the common
|
|
97
|
+
* generic-`TEnv` call sites (a deferred type parameter can't resolve the tuple
|
|
98
|
+
* conditional, so the intersection stops reducing to `T`). A router has one env,
|
|
99
|
+
* so a union env is not a supported pattern; the distributive form is kept.
|
|
100
|
+
*/
|
|
101
|
+
export type EnvCompatible<TPatterns, TRouterEnv> =
|
|
102
|
+
unknown extends UrlPatternsEnv<TPatterns>
|
|
103
|
+
? TPatterns
|
|
104
|
+
: TRouterEnv extends UrlPatternsEnv<TPatterns>
|
|
105
|
+
? TPatterns
|
|
106
|
+
: never;
|
|
107
|
+
|
|
91
108
|
/**
|
|
92
109
|
* Options for include()
|
|
93
110
|
*/
|
|
@@ -32,21 +32,32 @@ type ResponseReverseFunction = [DefaultReverseRouteMap] extends [
|
|
|
32
32
|
* Symbol marking a route as a response route (non-RSC).
|
|
33
33
|
* Stored on PathOptions and UrlPatterns to signal the trie to short-circuit.
|
|
34
34
|
*/
|
|
35
|
-
export const RESPONSE_TYPE: unique symbol = Symbol.for(
|
|
36
|
-
"rangojs.responseType",
|
|
37
|
-
) as any;
|
|
35
|
+
export const RESPONSE_TYPE: unique symbol = Symbol.for("rangojs.responseType");
|
|
38
36
|
|
|
39
37
|
/**
|
|
40
|
-
*
|
|
41
|
-
*
|
|
38
|
+
* Shared shape of a response-route handler: a function returning TReturn (or a
|
|
39
|
+
* promise of it), plus an optional composable `use` thunk merged at mount time.
|
|
42
40
|
*/
|
|
43
|
-
|
|
41
|
+
type ResponseHandlerOf<
|
|
42
|
+
TReturn,
|
|
43
|
+
TParams = Record<string, string>,
|
|
44
|
+
TEnv = any,
|
|
45
|
+
> = ((
|
|
44
46
|
ctx: ResponseHandlerContext<TParams, TEnv>,
|
|
45
|
-
) =>
|
|
47
|
+
) => TReturn | Promise<TReturn>) & {
|
|
46
48
|
/** Composable default DSL items merged when the handler is mounted. */
|
|
47
49
|
use?: () => UseItems<ResponseRouteUseItem>;
|
|
48
50
|
};
|
|
49
51
|
|
|
52
|
+
/**
|
|
53
|
+
* Handler that must return Response (not ReactNode).
|
|
54
|
+
* Used by path.image(), path.stream(), path.any() (binary/streaming data).
|
|
55
|
+
*/
|
|
56
|
+
export type ResponseHandler<
|
|
57
|
+
TParams = Record<string, string>,
|
|
58
|
+
TEnv = any,
|
|
59
|
+
> = ResponseHandlerOf<Response, TParams, TEnv>;
|
|
60
|
+
|
|
50
61
|
/**
|
|
51
62
|
* JSON-serializable value type for auto-wrap support.
|
|
52
63
|
*/
|
|
@@ -65,12 +76,7 @@ export type JsonValue =
|
|
|
65
76
|
export type JsonResponseHandler<
|
|
66
77
|
TParams = Record<string, string>,
|
|
67
78
|
TEnv = any,
|
|
68
|
-
> =
|
|
69
|
-
ctx: ResponseHandlerContext<TParams, TEnv>,
|
|
70
|
-
) => JsonValue | Response | Promise<JsonValue | Response>) & {
|
|
71
|
-
/** Composable default DSL items merged when the handler is mounted. */
|
|
72
|
-
use?: () => UseItems<ResponseRouteUseItem>;
|
|
73
|
-
};
|
|
79
|
+
> = ResponseHandlerOf<JsonValue | Response, TParams, TEnv>;
|
|
74
80
|
|
|
75
81
|
/**
|
|
76
82
|
* Handler for text-based response routes (text, html, xml).
|
|
@@ -79,12 +85,7 @@ export type JsonResponseHandler<
|
|
|
79
85
|
export type TextResponseHandler<
|
|
80
86
|
TParams = Record<string, string>,
|
|
81
87
|
TEnv = any,
|
|
82
|
-
> =
|
|
83
|
-
ctx: ResponseHandlerContext<TParams, TEnv>,
|
|
84
|
-
) => string | Response | Promise<string | Response>) & {
|
|
85
|
-
/** Composable default DSL items merged when the handler is mounted. */
|
|
86
|
-
use?: () => UseItems<ResponseRouteUseItem>;
|
|
87
|
-
};
|
|
88
|
+
> = ResponseHandlerOf<string | Response, TParams, TEnv>;
|
|
88
89
|
|
|
89
90
|
/**
|
|
90
91
|
* Lighter handler context for response routes.
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import type {
|
|
1
|
+
import type { JsonSerialize } from "../serialize.js";
|
|
2
2
|
import type {
|
|
3
3
|
TypedRouteItem,
|
|
4
4
|
TypedIncludeItem,
|
|
@@ -6,11 +6,7 @@ import type {
|
|
|
6
6
|
TypedCacheItem,
|
|
7
7
|
TypedTransitionItem,
|
|
8
8
|
} from "../route-types.js";
|
|
9
|
-
import type {
|
|
10
|
-
LocalOnlyInclude,
|
|
11
|
-
UnnamedRoute,
|
|
12
|
-
UrlPatterns,
|
|
13
|
-
} from "./pattern-types.js";
|
|
9
|
+
import type { LocalOnlyInclude, UnnamedRoute } from "./pattern-types.js";
|
|
14
10
|
|
|
15
11
|
// ============================================================================
|
|
16
12
|
// Route Type Extraction Utilities
|
|
@@ -67,62 +63,6 @@ type PrefixPatterns<
|
|
|
67
63
|
: TRoutes[K];
|
|
68
64
|
};
|
|
69
65
|
|
|
70
|
-
/**
|
|
71
|
-
* Depth counter for limiting recursion (max 40 levels)
|
|
72
|
-
* Supports up to 40 sibling items at any level of a urls() call
|
|
73
|
-
* Note: Higher values hit TypeScript's internal recursion limits
|
|
74
|
-
*/
|
|
75
|
-
type Depth = [
|
|
76
|
-
never,
|
|
77
|
-
0,
|
|
78
|
-
1,
|
|
79
|
-
2,
|
|
80
|
-
3,
|
|
81
|
-
4,
|
|
82
|
-
5,
|
|
83
|
-
6,
|
|
84
|
-
7,
|
|
85
|
-
8,
|
|
86
|
-
9,
|
|
87
|
-
10,
|
|
88
|
-
11,
|
|
89
|
-
12,
|
|
90
|
-
13,
|
|
91
|
-
14,
|
|
92
|
-
15,
|
|
93
|
-
16,
|
|
94
|
-
17,
|
|
95
|
-
18,
|
|
96
|
-
19,
|
|
97
|
-
20,
|
|
98
|
-
21,
|
|
99
|
-
22,
|
|
100
|
-
23,
|
|
101
|
-
24,
|
|
102
|
-
25,
|
|
103
|
-
26,
|
|
104
|
-
27,
|
|
105
|
-
28,
|
|
106
|
-
29,
|
|
107
|
-
30,
|
|
108
|
-
31,
|
|
109
|
-
32,
|
|
110
|
-
33,
|
|
111
|
-
34,
|
|
112
|
-
35,
|
|
113
|
-
36,
|
|
114
|
-
37,
|
|
115
|
-
38,
|
|
116
|
-
39,
|
|
117
|
-
];
|
|
118
|
-
|
|
119
|
-
/**
|
|
120
|
-
* Force TypeScript to eagerly evaluate a type.
|
|
121
|
-
* This helps with interface extension by creating a "concrete" object type.
|
|
122
|
-
*/
|
|
123
|
-
type Simplify<T> =
|
|
124
|
-
T extends Record<string, string> ? { [K in keyof T]: T[K] } : T;
|
|
125
|
-
|
|
126
66
|
/**
|
|
127
67
|
* Convert a union type to an intersection type.
|
|
128
68
|
* Used to combine route maps from multiple siblings without recursive tuple processing.
|
|
@@ -135,13 +75,11 @@ type UnionToIntersection<U> = (U extends any ? (k: U) => void : never) extends (
|
|
|
135
75
|
|
|
136
76
|
/**
|
|
137
77
|
* Extract routes from a single item (path, include, layout, cache with children)
|
|
138
|
-
* D is the current depth level for nested layouts/caches
|
|
139
78
|
*/
|
|
140
|
-
type ExtractRoutesFromItem<T
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
T extends TypedRouteItem<infer TName, infer TPattern, any, infer TSearch>
|
|
79
|
+
type ExtractRoutesFromItem<T> =
|
|
80
|
+
// TypedRouteItem: extract name -> pattern (exclude unnamed routes)
|
|
81
|
+
// When search schema is non-empty, value becomes { path, search } object
|
|
82
|
+
T extends TypedRouteItem<infer TName, infer TPattern, any, infer TSearch>
|
|
145
83
|
? TName extends string
|
|
146
84
|
? TName extends UnnamedRoute
|
|
147
85
|
? {} // Exclude unnamed routes from type map
|
|
@@ -185,14 +123,10 @@ type ExtractRoutesFromItem<T, D extends number = 40> = [D] extends [never]
|
|
|
185
123
|
* Extract routes from an array of items using mapped types.
|
|
186
124
|
* Uses UnionToIntersection to combine routes without recursive tuple processing,
|
|
187
125
|
* removing the sibling limit that was caused by TypeScript recursion limits.
|
|
188
|
-
* D is passed to ExtractRoutesFromItem for nested depth tracking.
|
|
189
126
|
*/
|
|
190
|
-
type ExtractRoutesFromItems<
|
|
191
|
-
T extends readonly any[],
|
|
192
|
-
D extends number = 40,
|
|
193
|
-
> = T extends readonly any[]
|
|
127
|
+
type ExtractRoutesFromItems<T extends readonly any[]> = T extends readonly any[]
|
|
194
128
|
? UnionToIntersection<
|
|
195
|
-
{ [K in keyof T]: ExtractRoutesFromItem<T[K]
|
|
129
|
+
{ [K in keyof T]: ExtractRoutesFromItem<T[K]> }[number]
|
|
196
130
|
> extends infer R
|
|
197
131
|
? R extends Record<string, any>
|
|
198
132
|
? R
|
|
@@ -203,12 +137,8 @@ type ExtractRoutesFromItems<
|
|
|
203
137
|
/**
|
|
204
138
|
* Main utility: extract route map from urls() callback return type
|
|
205
139
|
* Uses mapped types for sibling processing (no sibling limit).
|
|
206
|
-
* Uses Simplify to force eager evaluation for interface extension compatibility.
|
|
207
140
|
*/
|
|
208
|
-
export type ExtractRoutes<T extends readonly any[]> = ExtractRoutesFromItems<
|
|
209
|
-
T,
|
|
210
|
-
40
|
|
211
|
-
>;
|
|
141
|
+
export type ExtractRoutes<T extends readonly any[]> = ExtractRoutesFromItems<T>;
|
|
212
142
|
|
|
213
143
|
// ============================================================================
|
|
214
144
|
// Response Type Extraction Utilities
|
|
@@ -236,9 +166,8 @@ type PrefixKeys<
|
|
|
236
166
|
* Extract response data types from a single item.
|
|
237
167
|
* Parallel to ExtractRoutesFromItem but extracts name -> TData mapping.
|
|
238
168
|
*/
|
|
239
|
-
type ExtractResponsesFromItem<T
|
|
240
|
-
|
|
241
|
-
: T extends TypedRouteItem<infer TName, any, infer TData>
|
|
169
|
+
type ExtractResponsesFromItem<T> =
|
|
170
|
+
T extends TypedRouteItem<infer TName, any, infer TData>
|
|
242
171
|
? TName extends string
|
|
243
172
|
? TName extends UnnamedRoute
|
|
244
173
|
? {}
|
|
@@ -272,46 +201,23 @@ type ExtractResponsesFromItem<T, D extends number = 40> = [D] extends [never]
|
|
|
272
201
|
* Extract responses from an array of items using mapped types.
|
|
273
202
|
* Parallel to ExtractRoutesFromItems.
|
|
274
203
|
*/
|
|
275
|
-
type ExtractResponsesFromItems<
|
|
276
|
-
T extends readonly any[]
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
? R
|
|
204
|
+
type ExtractResponsesFromItems<T extends readonly any[]> =
|
|
205
|
+
T extends readonly any[]
|
|
206
|
+
? UnionToIntersection<
|
|
207
|
+
{ [K in keyof T]: ExtractResponsesFromItem<T[K]> }[number]
|
|
208
|
+
> extends infer R
|
|
209
|
+
? R extends Record<string, unknown>
|
|
210
|
+
? R
|
|
211
|
+
: {}
|
|
284
212
|
: {}
|
|
285
|
-
: {}
|
|
286
|
-
: {};
|
|
213
|
+
: {};
|
|
287
214
|
|
|
288
215
|
/**
|
|
289
216
|
* Main utility: extract response data type map from urls() callback return type.
|
|
290
217
|
* Parallel to ExtractRoutes.
|
|
291
218
|
*/
|
|
292
219
|
export type ExtractResponses<T extends readonly any[]> =
|
|
293
|
-
ExtractResponsesFromItems<T
|
|
294
|
-
|
|
295
|
-
// ============================================================================
|
|
296
|
-
// Type Utilities for path()
|
|
297
|
-
// ============================================================================
|
|
298
|
-
|
|
299
|
-
/**
|
|
300
|
-
* Extract route names from a UrlPatterns result
|
|
301
|
-
* Used for type-safe href() generation
|
|
302
|
-
*/
|
|
303
|
-
export type ExtractRouteNames<T extends UrlPatterns<any>> =
|
|
304
|
-
T extends UrlPatterns<infer _TEnv>
|
|
305
|
-
? string // For now, will be refined with full implementation
|
|
306
|
-
: never;
|
|
307
|
-
|
|
308
|
-
/**
|
|
309
|
-
* Extract params for a specific route name
|
|
310
|
-
*/
|
|
311
|
-
export type ExtractPathParams<
|
|
312
|
-
T extends UrlPatterns<any>,
|
|
313
|
-
K extends string,
|
|
314
|
-
> = ExtractParams<string>; // Will be refined with pattern tracking
|
|
220
|
+
ExtractResponsesFromItems<T>;
|
|
315
221
|
|
|
316
222
|
// ============================================================================
|
|
317
223
|
// Response Envelope Types
|
|
@@ -362,11 +268,15 @@ export type ResponseEnvelope<T> =
|
|
|
362
268
|
* type HealthData = RouteResponse<typeof apiPatterns, "health">;
|
|
363
269
|
* // ResponseEnvelope<{ status: string; timestamp: number }>
|
|
364
270
|
* ```
|
|
271
|
+
*
|
|
272
|
+
* The payload is the JSON wire shape (via `Rango.JsonSerialize`), matching
|
|
273
|
+
* `Rango.PathResponse` and what `fetch().then(r => r.json())` actually yields —
|
|
274
|
+
* e.g. a `Date` field resolves as `string`.
|
|
365
275
|
*/
|
|
366
276
|
export type RouteResponse<TPatterns, TName extends string> = TPatterns extends {
|
|
367
277
|
readonly _responses?: infer R;
|
|
368
278
|
}
|
|
369
279
|
? TName extends keyof R
|
|
370
|
-
? ResponseEnvelope<Exclude<R[TName], Response
|
|
280
|
+
? ResponseEnvelope<JsonSerialize<Exclude<R[TName], Response>>>
|
|
371
281
|
: never
|
|
372
282
|
: never;
|
|
@@ -3,7 +3,7 @@ import type { AllUseItems } from "../route-types.js";
|
|
|
3
3
|
import { getContext } from "../server/context";
|
|
4
4
|
import { invariant } from "../errors";
|
|
5
5
|
import { createRouteHelpers } from "../route-definition.js";
|
|
6
|
-
import type {
|
|
6
|
+
import type { UrlPatterns } from "./pattern-types.js";
|
|
7
7
|
import type { PathHelpers } from "./path-helper-types.js";
|
|
8
8
|
import type { ExtractRoutes, ExtractResponses } from "./type-extraction.js";
|
|
9
9
|
import { createPathHelper, attachPathResponseTags } from "./path-helper.js";
|
|
@@ -34,9 +34,6 @@ export function urls<
|
|
|
34
34
|
>(
|
|
35
35
|
builder: (helpers: PathHelpers<TEnv>) => TItems,
|
|
36
36
|
): UrlPatterns<TEnv, ExtractRoutes<TItems>, ExtractResponses<TItems>> {
|
|
37
|
-
// Collect path definitions during build
|
|
38
|
-
const definitions: PathDefinition[] = [];
|
|
39
|
-
|
|
40
37
|
// Create the handler function that will be called by the router
|
|
41
38
|
const handler = () => {
|
|
42
39
|
invariant(
|
|
@@ -82,7 +79,6 @@ export function urls<
|
|
|
82
79
|
// trailingSlash config is populated when handler() runs
|
|
83
80
|
// We expose it via a getter that reads from the context after handler execution
|
|
84
81
|
return {
|
|
85
|
-
definitions,
|
|
86
82
|
handler,
|
|
87
83
|
get trailingSlash() {
|
|
88
84
|
// Get the trailingSlash map from the current context
|