@rangojs/router 0.0.0-experimental.0f44aca1

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.
Files changed (305) hide show
  1. package/AGENTS.md +5 -0
  2. package/README.md +899 -0
  3. package/dist/bin/rango.js +1601 -0
  4. package/dist/vite/index.js +5214 -0
  5. package/package.json +176 -0
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +262 -0
  8. package/skills/caching/SKILL.md +220 -0
  9. package/skills/composability/SKILL.md +172 -0
  10. package/skills/debug-manifest/SKILL.md +112 -0
  11. package/skills/document-cache/SKILL.md +182 -0
  12. package/skills/fonts/SKILL.md +167 -0
  13. package/skills/hooks/SKILL.md +704 -0
  14. package/skills/host-router/SKILL.md +218 -0
  15. package/skills/intercept/SKILL.md +313 -0
  16. package/skills/layout/SKILL.md +310 -0
  17. package/skills/links/SKILL.md +239 -0
  18. package/skills/loader/SKILL.md +596 -0
  19. package/skills/middleware/SKILL.md +339 -0
  20. package/skills/mime-routes/SKILL.md +128 -0
  21. package/skills/parallel/SKILL.md +305 -0
  22. package/skills/prerender/SKILL.md +643 -0
  23. package/skills/rango/SKILL.md +118 -0
  24. package/skills/response-routes/SKILL.md +411 -0
  25. package/skills/route/SKILL.md +385 -0
  26. package/skills/router-setup/SKILL.md +439 -0
  27. package/skills/tailwind/SKILL.md +129 -0
  28. package/skills/theme/SKILL.md +79 -0
  29. package/skills/typesafety/SKILL.md +623 -0
  30. package/skills/use-cache/SKILL.md +324 -0
  31. package/src/__internal.ts +273 -0
  32. package/src/bin/rango.ts +321 -0
  33. package/src/browser/action-coordinator.ts +97 -0
  34. package/src/browser/action-response-classifier.ts +99 -0
  35. package/src/browser/event-controller.ts +899 -0
  36. package/src/browser/history-state.ts +80 -0
  37. package/src/browser/index.ts +18 -0
  38. package/src/browser/intercept-utils.ts +52 -0
  39. package/src/browser/link-interceptor.ts +141 -0
  40. package/src/browser/logging.ts +55 -0
  41. package/src/browser/merge-segment-loaders.ts +134 -0
  42. package/src/browser/navigation-bridge.ts +645 -0
  43. package/src/browser/navigation-client.ts +215 -0
  44. package/src/browser/navigation-store.ts +806 -0
  45. package/src/browser/navigation-transaction.ts +295 -0
  46. package/src/browser/network-error-handler.ts +61 -0
  47. package/src/browser/partial-update.ts +550 -0
  48. package/src/browser/prefetch/cache.ts +146 -0
  49. package/src/browser/prefetch/fetch.ts +135 -0
  50. package/src/browser/prefetch/observer.ts +65 -0
  51. package/src/browser/prefetch/policy.ts +42 -0
  52. package/src/browser/prefetch/queue.ts +88 -0
  53. package/src/browser/rango-state.ts +112 -0
  54. package/src/browser/react/Link.tsx +360 -0
  55. package/src/browser/react/NavigationProvider.tsx +386 -0
  56. package/src/browser/react/ScrollRestoration.tsx +94 -0
  57. package/src/browser/react/context.ts +59 -0
  58. package/src/browser/react/filter-segment-order.ts +11 -0
  59. package/src/browser/react/index.ts +52 -0
  60. package/src/browser/react/location-state-shared.ts +162 -0
  61. package/src/browser/react/location-state.ts +107 -0
  62. package/src/browser/react/mount-context.ts +37 -0
  63. package/src/browser/react/nonce-context.ts +23 -0
  64. package/src/browser/react/shallow-equal.ts +27 -0
  65. package/src/browser/react/use-action.ts +218 -0
  66. package/src/browser/react/use-client-cache.ts +58 -0
  67. package/src/browser/react/use-handle.ts +162 -0
  68. package/src/browser/react/use-href.tsx +40 -0
  69. package/src/browser/react/use-link-status.ts +135 -0
  70. package/src/browser/react/use-mount.ts +31 -0
  71. package/src/browser/react/use-navigation.ts +99 -0
  72. package/src/browser/react/use-params.ts +65 -0
  73. package/src/browser/react/use-pathname.ts +47 -0
  74. package/src/browser/react/use-router.ts +63 -0
  75. package/src/browser/react/use-search-params.ts +56 -0
  76. package/src/browser/react/use-segments.ts +171 -0
  77. package/src/browser/response-adapter.ts +73 -0
  78. package/src/browser/rsc-router.tsx +431 -0
  79. package/src/browser/scroll-restoration.ts +400 -0
  80. package/src/browser/segment-reconciler.ts +216 -0
  81. package/src/browser/segment-structure-assert.ts +83 -0
  82. package/src/browser/server-action-bridge.ts +667 -0
  83. package/src/browser/shallow.ts +40 -0
  84. package/src/browser/types.ts +538 -0
  85. package/src/browser/validate-redirect-origin.ts +29 -0
  86. package/src/build/generate-manifest.ts +438 -0
  87. package/src/build/generate-route-types.ts +36 -0
  88. package/src/build/index.ts +35 -0
  89. package/src/build/route-trie.ts +265 -0
  90. package/src/build/route-types/ast-helpers.ts +25 -0
  91. package/src/build/route-types/ast-route-extraction.ts +98 -0
  92. package/src/build/route-types/codegen.ts +102 -0
  93. package/src/build/route-types/include-resolution.ts +411 -0
  94. package/src/build/route-types/param-extraction.ts +48 -0
  95. package/src/build/route-types/per-module-writer.ts +128 -0
  96. package/src/build/route-types/router-processing.ts +469 -0
  97. package/src/build/route-types/scan-filter.ts +78 -0
  98. package/src/build/runtime-discovery.ts +231 -0
  99. package/src/cache/background-task.ts +34 -0
  100. package/src/cache/cache-key-utils.ts +44 -0
  101. package/src/cache/cache-policy.ts +125 -0
  102. package/src/cache/cache-runtime.ts +338 -0
  103. package/src/cache/cache-scope.ts +382 -0
  104. package/src/cache/cf/cf-cache-store.ts +540 -0
  105. package/src/cache/cf/index.ts +25 -0
  106. package/src/cache/document-cache.ts +369 -0
  107. package/src/cache/handle-capture.ts +81 -0
  108. package/src/cache/handle-snapshot.ts +41 -0
  109. package/src/cache/index.ts +43 -0
  110. package/src/cache/memory-segment-store.ts +328 -0
  111. package/src/cache/profile-registry.ts +73 -0
  112. package/src/cache/read-through-swr.ts +134 -0
  113. package/src/cache/segment-codec.ts +256 -0
  114. package/src/cache/taint.ts +98 -0
  115. package/src/cache/types.ts +342 -0
  116. package/src/client.rsc.tsx +85 -0
  117. package/src/client.tsx +601 -0
  118. package/src/component-utils.ts +76 -0
  119. package/src/components/DefaultDocument.tsx +27 -0
  120. package/src/context-var.ts +86 -0
  121. package/src/debug.ts +243 -0
  122. package/src/default-error-boundary.tsx +88 -0
  123. package/src/deps/browser.ts +8 -0
  124. package/src/deps/html-stream-client.ts +2 -0
  125. package/src/deps/html-stream-server.ts +2 -0
  126. package/src/deps/rsc.ts +10 -0
  127. package/src/deps/ssr.ts +2 -0
  128. package/src/errors.ts +365 -0
  129. package/src/handle.ts +135 -0
  130. package/src/handles/MetaTags.tsx +246 -0
  131. package/src/handles/breadcrumbs.ts +66 -0
  132. package/src/handles/index.ts +7 -0
  133. package/src/handles/meta.ts +264 -0
  134. package/src/host/cookie-handler.ts +165 -0
  135. package/src/host/errors.ts +97 -0
  136. package/src/host/index.ts +53 -0
  137. package/src/host/pattern-matcher.ts +214 -0
  138. package/src/host/router.ts +352 -0
  139. package/src/host/testing.ts +79 -0
  140. package/src/host/types.ts +146 -0
  141. package/src/host/utils.ts +25 -0
  142. package/src/href-client.ts +222 -0
  143. package/src/index.rsc.ts +233 -0
  144. package/src/index.ts +277 -0
  145. package/src/internal-debug.ts +11 -0
  146. package/src/loader.rsc.ts +89 -0
  147. package/src/loader.ts +64 -0
  148. package/src/network-error-thrower.tsx +23 -0
  149. package/src/outlet-context.ts +15 -0
  150. package/src/outlet-provider.tsx +45 -0
  151. package/src/prerender/param-hash.ts +37 -0
  152. package/src/prerender/store.ts +185 -0
  153. package/src/prerender.ts +463 -0
  154. package/src/reverse.ts +330 -0
  155. package/src/root-error-boundary.tsx +289 -0
  156. package/src/route-content-wrapper.tsx +196 -0
  157. package/src/route-definition/dsl-helpers.ts +934 -0
  158. package/src/route-definition/helper-factories.ts +200 -0
  159. package/src/route-definition/helpers-types.ts +430 -0
  160. package/src/route-definition/index.ts +52 -0
  161. package/src/route-definition/redirect.ts +93 -0
  162. package/src/route-definition.ts +1 -0
  163. package/src/route-map-builder.ts +275 -0
  164. package/src/route-name.ts +53 -0
  165. package/src/route-types.ts +259 -0
  166. package/src/router/content-negotiation.ts +116 -0
  167. package/src/router/debug-manifest.ts +72 -0
  168. package/src/router/error-handling.ts +287 -0
  169. package/src/router/find-match.ts +158 -0
  170. package/src/router/handler-context.ts +451 -0
  171. package/src/router/intercept-resolution.ts +395 -0
  172. package/src/router/lazy-includes.ts +234 -0
  173. package/src/router/loader-resolution.ts +420 -0
  174. package/src/router/logging.ts +248 -0
  175. package/src/router/manifest.ts +267 -0
  176. package/src/router/match-api.ts +620 -0
  177. package/src/router/match-context.ts +266 -0
  178. package/src/router/match-handlers.ts +440 -0
  179. package/src/router/match-middleware/background-revalidation.ts +223 -0
  180. package/src/router/match-middleware/cache-lookup.ts +634 -0
  181. package/src/router/match-middleware/cache-store.ts +295 -0
  182. package/src/router/match-middleware/index.ts +81 -0
  183. package/src/router/match-middleware/intercept-resolution.ts +306 -0
  184. package/src/router/match-middleware/segment-resolution.ts +192 -0
  185. package/src/router/match-pipelines.ts +179 -0
  186. package/src/router/match-result.ts +219 -0
  187. package/src/router/metrics.ts +282 -0
  188. package/src/router/middleware-cookies.ts +55 -0
  189. package/src/router/middleware-types.ts +222 -0
  190. package/src/router/middleware.ts +748 -0
  191. package/src/router/pattern-matching.ts +563 -0
  192. package/src/router/prerender-match.ts +402 -0
  193. package/src/router/preview-match.ts +170 -0
  194. package/src/router/revalidation.ts +289 -0
  195. package/src/router/router-context.ts +316 -0
  196. package/src/router/router-interfaces.ts +452 -0
  197. package/src/router/router-options.ts +592 -0
  198. package/src/router/router-registry.ts +24 -0
  199. package/src/router/segment-resolution/fresh.ts +570 -0
  200. package/src/router/segment-resolution/helpers.ts +263 -0
  201. package/src/router/segment-resolution/loader-cache.ts +198 -0
  202. package/src/router/segment-resolution/revalidation.ts +1239 -0
  203. package/src/router/segment-resolution/static-store.ts +67 -0
  204. package/src/router/segment-resolution.ts +21 -0
  205. package/src/router/segment-wrappers.ts +289 -0
  206. package/src/router/telemetry-otel.ts +299 -0
  207. package/src/router/telemetry.ts +300 -0
  208. package/src/router/timeout.ts +148 -0
  209. package/src/router/trie-matching.ts +239 -0
  210. package/src/router/types.ts +170 -0
  211. package/src/router.ts +1002 -0
  212. package/src/rsc/handler-context.ts +45 -0
  213. package/src/rsc/handler.ts +1089 -0
  214. package/src/rsc/helpers.ts +198 -0
  215. package/src/rsc/index.ts +36 -0
  216. package/src/rsc/loader-fetch.ts +209 -0
  217. package/src/rsc/manifest-init.ts +86 -0
  218. package/src/rsc/nonce.ts +32 -0
  219. package/src/rsc/origin-guard.ts +141 -0
  220. package/src/rsc/progressive-enhancement.ts +379 -0
  221. package/src/rsc/response-error.ts +37 -0
  222. package/src/rsc/response-route-handler.ts +347 -0
  223. package/src/rsc/rsc-rendering.ts +235 -0
  224. package/src/rsc/runtime-warnings.ts +42 -0
  225. package/src/rsc/server-action.ts +348 -0
  226. package/src/rsc/ssr-setup.ts +128 -0
  227. package/src/rsc/types.ts +263 -0
  228. package/src/search-params.ts +230 -0
  229. package/src/segment-system.tsx +454 -0
  230. package/src/server/context.ts +591 -0
  231. package/src/server/cookie-store.ts +190 -0
  232. package/src/server/fetchable-loader-store.ts +37 -0
  233. package/src/server/handle-store.ts +308 -0
  234. package/src/server/loader-registry.ts +133 -0
  235. package/src/server/request-context.ts +914 -0
  236. package/src/server/root-layout.tsx +10 -0
  237. package/src/server/tsconfig.json +14 -0
  238. package/src/server.ts +51 -0
  239. package/src/ssr/index.tsx +365 -0
  240. package/src/static-handler.ts +114 -0
  241. package/src/theme/ThemeProvider.tsx +297 -0
  242. package/src/theme/ThemeScript.tsx +61 -0
  243. package/src/theme/constants.ts +62 -0
  244. package/src/theme/index.ts +48 -0
  245. package/src/theme/theme-context.ts +44 -0
  246. package/src/theme/theme-script.ts +155 -0
  247. package/src/theme/types.ts +182 -0
  248. package/src/theme/use-theme.ts +44 -0
  249. package/src/types/boundaries.ts +158 -0
  250. package/src/types/cache-types.ts +198 -0
  251. package/src/types/error-types.ts +192 -0
  252. package/src/types/global-namespace.ts +100 -0
  253. package/src/types/handler-context.ts +687 -0
  254. package/src/types/index.ts +88 -0
  255. package/src/types/loader-types.ts +183 -0
  256. package/src/types/route-config.ts +170 -0
  257. package/src/types/route-entry.ts +102 -0
  258. package/src/types/segments.ts +148 -0
  259. package/src/types.ts +1 -0
  260. package/src/urls/include-helper.ts +197 -0
  261. package/src/urls/index.ts +53 -0
  262. package/src/urls/path-helper-types.ts +339 -0
  263. package/src/urls/path-helper.ts +329 -0
  264. package/src/urls/pattern-types.ts +95 -0
  265. package/src/urls/response-types.ts +106 -0
  266. package/src/urls/type-extraction.ts +372 -0
  267. package/src/urls/urls-function.ts +98 -0
  268. package/src/urls.ts +1 -0
  269. package/src/use-loader.tsx +354 -0
  270. package/src/vite/discovery/bundle-postprocess.ts +184 -0
  271. package/src/vite/discovery/discover-routers.ts +344 -0
  272. package/src/vite/discovery/prerender-collection.ts +385 -0
  273. package/src/vite/discovery/route-types-writer.ts +258 -0
  274. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  275. package/src/vite/discovery/state.ts +110 -0
  276. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  277. package/src/vite/index.ts +16 -0
  278. package/src/vite/plugin-types.ts +131 -0
  279. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  280. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  281. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  282. package/src/vite/plugins/expose-action-id.ts +365 -0
  283. package/src/vite/plugins/expose-id-utils.ts +287 -0
  284. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  285. package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
  286. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  287. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  288. package/src/vite/plugins/expose-ids/types.ts +45 -0
  289. package/src/vite/plugins/expose-internal-ids.ts +569 -0
  290. package/src/vite/plugins/refresh-cmd.ts +65 -0
  291. package/src/vite/plugins/use-cache-transform.ts +323 -0
  292. package/src/vite/plugins/version-injector.ts +83 -0
  293. package/src/vite/plugins/version-plugin.ts +254 -0
  294. package/src/vite/plugins/version.d.ts +12 -0
  295. package/src/vite/plugins/virtual-entries.ts +123 -0
  296. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  297. package/src/vite/rango.ts +510 -0
  298. package/src/vite/router-discovery.ts +785 -0
  299. package/src/vite/utils/ast-handler-extract.ts +517 -0
  300. package/src/vite/utils/banner.ts +36 -0
  301. package/src/vite/utils/bundle-analysis.ts +137 -0
  302. package/src/vite/utils/manifest-utils.ts +70 -0
  303. package/src/vite/utils/package-resolution.ts +121 -0
  304. package/src/vite/utils/prerender-utils.ts +189 -0
  305. package/src/vite/utils/shared-utils.ts +169 -0
@@ -0,0 +1,88 @@
1
+ // Global namespace (must be imported for side effects: `declare global`)
2
+ export type {
3
+ GetRegisteredRoutes,
4
+ DefaultHandlerRouteMap,
5
+ DefaultReverseRouteMap,
6
+ DefaultEnv,
7
+ } from "./global-namespace.js";
8
+ // Ensure the global namespace declaration is evaluated
9
+ import "./global-namespace.js";
10
+
11
+ // Route configuration
12
+ export type {
13
+ DocumentProps,
14
+ ExtractParams,
15
+ TrailingSlashMode,
16
+ RouteConfig,
17
+ RouteDefinitionOptions,
18
+ RouteDefinition,
19
+ ResolvedRouteMap,
20
+ } from "./route-config.js";
21
+
22
+ // Boundaries (error/notFound)
23
+ export type {
24
+ ErrorInfo,
25
+ ErrorBoundaryFallbackProps,
26
+ ErrorBoundaryHandler,
27
+ ClientErrorBoundaryFallbackProps,
28
+ LoaderDataResult,
29
+ NotFoundInfo,
30
+ NotFoundBoundaryFallbackProps,
31
+ NotFoundBoundaryHandler,
32
+ } from "./boundaries.js";
33
+ export { isLoaderDataResult } from "./boundaries.js";
34
+
35
+ // Handler context and related types
36
+ export type {
37
+ MiddlewareFn,
38
+ ScopedRouteMap,
39
+ Handler,
40
+ HandlerContext,
41
+ InternalHandlerContext,
42
+ GenericParams,
43
+ RevalidateParams,
44
+ ShouldRevalidateFn,
45
+ RouteKeys,
46
+ ExtractRouteParams,
47
+ HandlersForRouteMap,
48
+ Revalidate,
49
+ Middleware,
50
+ } from "./handler-context.js";
51
+
52
+ // Segments
53
+ export type {
54
+ ViewTransitionClass,
55
+ TransitionConfig,
56
+ ResolvedSegment,
57
+ SegmentMetadata,
58
+ SlotState,
59
+ RootLayoutProps,
60
+ MatchResult,
61
+ } from "./segments.js";
62
+
63
+ // Route entries
64
+ export type { LazyIncludeContext, RouteEntry } from "./route-entry.js";
65
+
66
+ // Loader types
67
+ export type {
68
+ LoaderContext,
69
+ LoaderFn,
70
+ FetchableLoaderOptions,
71
+ LoadOptions,
72
+ LoaderDefinition,
73
+ } from "./loader-types.js";
74
+
75
+ // Cache types
76
+ export type {
77
+ CacheContext,
78
+ CacheOptions,
79
+ PartialCacheOptions,
80
+ EntryCacheConfig,
81
+ } from "./cache-types.js";
82
+
83
+ // Error handling types
84
+ export type {
85
+ ErrorPhase,
86
+ OnErrorContext,
87
+ OnErrorCallback,
88
+ } from "./error-types.js";
@@ -0,0 +1,183 @@
1
+ import type { ContextVar } from "../context-var.js";
2
+ import type { MiddlewareFn } from "../router/middleware.js";
3
+ import type { ScopedReverseFunction } from "../reverse.js";
4
+ import type { SearchSchema, ResolveSearchSchema } from "../search-params.js";
5
+ import type {
6
+ DefaultEnv,
7
+ DefaultReverseRouteMap,
8
+ DefaultVars,
9
+ } from "./global-namespace.js";
10
+
11
+ /**
12
+ * Context passed to loader functions during execution
13
+ *
14
+ * Loaders run after middleware but before handlers, so they have access
15
+ * to middleware-set variables via get().
16
+ *
17
+ * @template TParams - Route params type (e.g., { slug: string })
18
+ * @template TEnv - Environment type for bindings/variables
19
+ *
20
+ * @example
21
+ * ```typescript
22
+ * const CartLoader = createLoader(async (ctx) => {
23
+ * "use server";
24
+ * const user = ctx.get("user"); // From auth middleware
25
+ * return await db.cart.get(user.id);
26
+ * });
27
+ *
28
+ * // With typed params:
29
+ * const ProductLoader = createLoader<Product, { slug: string }>(async (ctx) => {
30
+ * "use server";
31
+ * const { slug } = ctx.params; // slug is typed as string
32
+ * return await db.products.findBySlug(slug);
33
+ * });
34
+ * ```
35
+ */
36
+ export type LoaderContext<
37
+ TParams = Record<string, string | undefined>,
38
+ TEnv = DefaultEnv,
39
+ TBody = unknown,
40
+ TSearch extends SearchSchema = {},
41
+ > = {
42
+ params: TParams;
43
+ /**
44
+ * Route params extracted from the URL pattern match (server-side only).
45
+ * Unlike `params`, these cannot be overridden by client-provided loader params.
46
+ * Use this when you need trusted, server-matched route params for auth or
47
+ * resource scoping.
48
+ */
49
+ routeParams: Record<string, string>;
50
+ request: Request;
51
+ searchParams: URLSearchParams;
52
+ search: {} extends TSearch ? {} : ResolveSearchSchema<TSearch>;
53
+ pathname: string;
54
+ url: URL;
55
+ env: TEnv;
56
+ var: DefaultVars;
57
+ get: {
58
+ <T>(contextVar: ContextVar<T>): T | undefined;
59
+ } & (<K extends keyof DefaultVars>(key: K) => DefaultVars[K]);
60
+ /**
61
+ * Access another loader's data (returns promise since loaders run in parallel)
62
+ */
63
+ use: <T, TLoaderParams = any>(
64
+ loader: LoaderDefinition<T, TLoaderParams>,
65
+ ) => Promise<T>;
66
+ /**
67
+ * HTTP method (GET, POST, PUT, PATCH, DELETE)
68
+ * Available when loader is called via load({ method: "POST", ... })
69
+ */
70
+ method: string;
71
+ /**
72
+ * Request body for POST/PUT/PATCH/DELETE requests
73
+ * Available when loader is called via load({ method: "POST", body: {...} })
74
+ */
75
+ body: TBody | undefined;
76
+ /**
77
+ * Form data when loader is invoked via action (fetchable loaders)
78
+ * Available when loader is called via form submission
79
+ */
80
+ formData?: FormData;
81
+ /**
82
+ * Generate URLs from route names.
83
+ * Same scoped reverse as route handlers — `.name` resolves within the
84
+ * current include() scope, `name` resolves globally.
85
+ */
86
+ reverse: ScopedReverseFunction<
87
+ Record<string, string>,
88
+ DefaultReverseRouteMap
89
+ >;
90
+ };
91
+
92
+ /**
93
+ * Loader function signature
94
+ *
95
+ * @template T - The return type of the loader
96
+ * @template TParams - Route params type (defaults to generic Record)
97
+ * @template TEnv - Environment type for bindings/variables
98
+ *
99
+ * @example
100
+ * ```typescript
101
+ * const myLoader: LoaderFn<{ items: Item[] }> = async (ctx) => {
102
+ * "use server";
103
+ * return { items: await db.items.list() };
104
+ * };
105
+ *
106
+ * // With typed params:
107
+ * const productLoader: LoaderFn<Product, { slug: string }> = async (ctx) => {
108
+ * "use server";
109
+ * const { slug } = ctx.params; // typed as string
110
+ * return await db.products.findBySlug(slug);
111
+ * };
112
+ * ```
113
+ */
114
+ export type LoaderFn<
115
+ T,
116
+ TParams = Record<string, string | undefined>,
117
+ TEnv = DefaultEnv,
118
+ > = (ctx: LoaderContext<TParams, TEnv>) => Promise<T> | T;
119
+
120
+ /**
121
+ * Options for fetchable loaders
122
+ *
123
+ * Middleware uses the same MiddlewareFn signature as route/app middleware,
124
+ * enabling reuse of the same middleware functions everywhere.
125
+ */
126
+ export type FetchableLoaderOptions = {
127
+ fetchable?: true;
128
+ middleware?: MiddlewareFn[];
129
+ };
130
+
131
+ /**
132
+ * Options for load() calls - type-safe union based on method
133
+ */
134
+ export type LoadOptions =
135
+ | {
136
+ method?: "GET";
137
+ params?: Record<string, string>;
138
+ }
139
+ | {
140
+ method: "POST" | "PUT" | "PATCH" | "DELETE";
141
+ params?: Record<string, string>;
142
+ body?: FormData | Record<string, any>;
143
+ };
144
+
145
+ /**
146
+ * Loader definition object
147
+ *
148
+ * Created via createLoader(). Contains the loader name and function.
149
+ * On client builds, the fn is stripped by the bundler (via "use server" directive).
150
+ *
151
+ * @template T - The return type of the loader
152
+ * @template TParams - Route params type (for type-safe params access)
153
+ *
154
+ * @example
155
+ * ```typescript
156
+ * // Definition (same file works on server and client)
157
+ * export const CartLoader = createLoader(async (ctx) => {
158
+ * "use server";
159
+ * return await db.cart.get(ctx.get("user").id);
160
+ * });
161
+ *
162
+ * // With typed params:
163
+ * export const ProductLoader = createLoader<Product, { slug: string }>(async (ctx) => {
164
+ * "use server";
165
+ * const { slug } = ctx.params; // slug is typed as string
166
+ * return await db.products.findBySlug(slug);
167
+ * });
168
+ *
169
+ * // Server usage
170
+ * const cart = ctx.use(CartLoader);
171
+ *
172
+ * // Client usage (fn is stripped, only name remains)
173
+ * const cart = useLoader(CartLoader);
174
+ * ```
175
+ */
176
+ export type LoaderDefinition<
177
+ T = any,
178
+ TParams = Record<string, string | undefined>,
179
+ > = {
180
+ __brand: "loader";
181
+ $$id: string; // Injected by Vite plugin (exposeInternalIds) - unique identifier
182
+ fn?: LoaderFn<T, TParams, any>; // Optional - server-side only, stored in registry for RSC
183
+ };
@@ -0,0 +1,170 @@
1
+ import type { ReactNode } from "react";
2
+
3
+ /**
4
+ * Props for the Document component that wraps the entire application.
5
+ */
6
+ export type DocumentProps = {
7
+ children: ReactNode;
8
+ };
9
+
10
+ /**
11
+ * Parse constraint values into a union type
12
+ * "a|b|c" -> "a" | "b" | "c"
13
+ */
14
+ type ParseConstraint<T extends string> =
15
+ T extends `${infer First}|${infer Rest}` ? First | ParseConstraint<Rest> : T;
16
+
17
+ /**
18
+ * Extract param info from a param segment
19
+ *
20
+ * Handles:
21
+ * - :param -> { name: "param", optional: false, type: string }
22
+ * - :param? -> { name: "param", optional: true, type: string }
23
+ * - :param(a|b) -> { name: "param", optional: false, type: "a" | "b" }
24
+ * - :param(a|b)? -> { name: "param", optional: true, type: "a" | "b" }
25
+ */
26
+ type ExtractParamInfo<T extends string> =
27
+ // Optional + constrained (with optional suffix): :param(a|b)?suffix
28
+ T extends `${infer Name}(${infer Constraint})?${string}`
29
+ ? { name: Name; optional: true; type: ParseConstraint<Constraint> }
30
+ : // Constrained (with optional suffix): :param(a|b)suffix
31
+ T extends `${infer Name}(${infer Constraint})${string}`
32
+ ? { name: Name; optional: false; type: ParseConstraint<Constraint> }
33
+ : // Optional (with optional suffix): :param?suffix
34
+ T extends `${infer Name}?${string}`
35
+ ? { name: Name; optional: true; type: string }
36
+ : // Param with dot-suffix: :param.html
37
+ T extends `${infer Name}.${string}`
38
+ ? { name: Name; optional: false; type: string }
39
+ : // Param with dash-suffix: :param-slug
40
+ T extends `${infer Name}-${string}`
41
+ ? { name: Name; optional: false; type: string }
42
+ : // Param with tilde-suffix: :param~v2
43
+ T extends `${infer Name}~${string}`
44
+ ? { name: Name; optional: false; type: string }
45
+ : // Required: :param (no suffix)
46
+ { name: T; optional: false; type: string };
47
+
48
+ /**
49
+ * Build param object from info
50
+ */
51
+ type ParamFromInfo<Info> = Info extends {
52
+ name: infer N extends string;
53
+ optional: true;
54
+ type: infer V;
55
+ }
56
+ ? { [K in N]?: V }
57
+ : Info extends {
58
+ name: infer N extends string;
59
+ optional: false;
60
+ type: infer V;
61
+ }
62
+ ? { [K in N]: V }
63
+ : never;
64
+
65
+ /**
66
+ * Merge two param objects preserving optionality
67
+ * Uses Pick to preserve the modifiers from source types
68
+ */
69
+ type MergeParams<A, B> = Pick<A, keyof A> & Pick<B, keyof B> extends infer O
70
+ ? { [K in keyof O]: O[K] }
71
+ : never;
72
+
73
+ /**
74
+ * Extract route params from a pattern with depth limit to prevent infinite recursion
75
+ *
76
+ * Supports:
77
+ * - Required params: /:slug -> { slug: string }
78
+ * - Optional params: /:locale? -> { locale?: string }
79
+ * - Constrained params: /:locale(en|gb) -> { locale: "en" | "gb" }
80
+ * - Optional + constrained: /:locale(en|gb)? -> { locale?: "en" | "gb" }
81
+ *
82
+ * @example
83
+ * ExtractParams<"/products/:id"> // { id: string }
84
+ * ExtractParams<"/:locale?/blog/:slug"> // { locale?: string; slug: string }
85
+ * ExtractParams<"/:locale(en|gb)/blog"> // { locale: "en" | "gb" }
86
+ * ExtractParams<"/:locale(en|gb)?/blog/:slug"> // { locale?: "en" | "gb"; slug: string }
87
+ */
88
+ export type ExtractParams<
89
+ T extends string,
90
+ Depth extends readonly unknown[] = [],
91
+ > = Depth["length"] extends 10
92
+ ? { [key: string]: string | undefined } // Fallback to generic params if too deep
93
+ : // Match param with remaining path: :param.../rest
94
+ T extends `${infer _Start}:${infer Param}/${infer Rest}`
95
+ ? MergeParams<
96
+ ParamFromInfo<ExtractParamInfo<Param>>,
97
+ ExtractParams<`/${Rest}`, readonly [...Depth, unknown]>
98
+ >
99
+ : // Match param at end: :param...
100
+ T extends `${infer _Start}:${infer Param}`
101
+ ? ParamFromInfo<ExtractParamInfo<Param>>
102
+ : {};
103
+
104
+ /**
105
+ * Trailing slash handling mode
106
+ * - "never": Redirect URLs with trailing slash to without
107
+ * - "always": Redirect URLs without trailing slash to with
108
+ * - "ignore": Match both with and without trailing slash
109
+ */
110
+ export type TrailingSlashMode = "never" | "always" | "ignore";
111
+
112
+ /**
113
+ * Route configuration object (alternative to string path)
114
+ */
115
+ export type RouteConfig = {
116
+ path: string;
117
+ trailingSlash?: TrailingSlashMode;
118
+ };
119
+
120
+ /**
121
+ * Route definition options (global defaults)
122
+ */
123
+ export type RouteDefinitionOptions = {
124
+ trailingSlash?: TrailingSlashMode;
125
+ };
126
+
127
+ export type RouteDefinition = {
128
+ [key: string]: string | RouteConfig | RouteDefinition;
129
+ };
130
+
131
+ /**
132
+ * Recursively flatten nested routes with depth limit to prevent infinite recursion
133
+ * Transforms: { products: { detail: "/product/:slug" } } => { "products.detail": "/product/:slug" }
134
+ * Also handles RouteConfig objects: { api: { path: "/api" } } => { "api": "/api" }
135
+ */
136
+ type FlattenRoutes<
137
+ T extends RouteDefinition,
138
+ Prefix extends string = "",
139
+ Depth extends readonly unknown[] = [],
140
+ > = Depth["length"] extends 5
141
+ ? never
142
+ : {
143
+ [K in keyof T]: T[K] extends string
144
+ ? Record<`${Prefix}${K & string}`, T[K]>
145
+ : T[K] extends RouteConfig
146
+ ? Record<`${Prefix}${K & string}`, T[K]["path"]>
147
+ : T[K] extends RouteDefinition
148
+ ? FlattenRoutes<
149
+ T[K],
150
+ `${Prefix}${K & string}.`,
151
+ readonly [...Depth, unknown]
152
+ >
153
+ : never;
154
+ }[keyof T];
155
+
156
+ /**
157
+ * Union to intersection helper
158
+ */
159
+ type UnionToIntersection<U> = (
160
+ U extends unknown ? (k: U) => void : never
161
+ ) extends (k: infer I) => void
162
+ ? I
163
+ : never;
164
+
165
+ /**
166
+ * Resolved route map - flattened route definitions with full paths
167
+ */
168
+ export type ResolvedRouteMap<T extends RouteDefinition> = UnionToIntersection<
169
+ FlattenRoutes<T>
170
+ >;
@@ -0,0 +1,102 @@
1
+ import type { AllUseItems } from "../route-types.js";
2
+ import type { TrailingSlashMode, ResolvedRouteMap } from "./route-config.js";
3
+
4
+ /**
5
+ * Context captured for lazy include evaluation
6
+ */
7
+ export interface LazyIncludeContext {
8
+ urlPrefix: string;
9
+ namePrefix: string | undefined;
10
+ parent: unknown; // EntryData - avoid circular import
11
+ cacheProfiles?: Record<
12
+ string,
13
+ import("../cache/profile-registry.js").CacheProfile
14
+ >;
15
+ }
16
+
17
+ /**
18
+ * Internal route entry stored in router
19
+ */
20
+ export interface RouteEntry<TEnv = any> {
21
+ prefix: string;
22
+ /**
23
+ * Pre-computed static prefix for fast short-circuit matching.
24
+ * Extracted from prefix at registration time (everything before first param).
25
+ *
26
+ * Examples:
27
+ * - "/api" -> staticPrefix = "/api"
28
+ * - "/site/:locale" -> staticPrefix = "/site"
29
+ * - "/:locale" -> staticPrefix = "" (empty, can't optimize)
30
+ *
31
+ * At runtime: if staticPrefix && !pathname.startsWith(staticPrefix), skip entry.
32
+ */
33
+ staticPrefix: string;
34
+ /**
35
+ * Route patterns map. For lazy entries, this starts as empty and is
36
+ * populated on first request.
37
+ */
38
+ routes: ResolvedRouteMap<any>;
39
+ /**
40
+ * Trailing slash config per route key
41
+ * If not specified for a route, defaults to pattern-based detection
42
+ */
43
+ trailingSlash?: Record<string, TrailingSlashMode>;
44
+ /**
45
+ * Supported handler shapes:
46
+ * - sync: () => Array<AllUseItems>
47
+ * - lazy import: () => Promise<{ default: () => Array<AllUseItems> }>
48
+ * - lazy function: () => Promise<() => Array<AllUseItems>>
49
+ *
50
+ * Direct Promise<Array> is NOT supported and rejected at runtime.
51
+ */
52
+ handler: () =>
53
+ | Array<AllUseItems>
54
+ | Promise<{ default: () => Array<AllUseItems> }>
55
+ | Promise<() => Array<AllUseItems>>;
56
+ mountIndex: number;
57
+
58
+ /**
59
+ * Route keys in this entry that have pre-render handlers.
60
+ * Used by the non-trie match path to set the `pr` flag.
61
+ */
62
+ prerenderRouteKeys?: Set<string>;
63
+
64
+ /**
65
+ * Route keys in this entry that use `{ passthrough: true }`.
66
+ * Used by the non-trie match path to set the `pt` flag.
67
+ */
68
+ passthroughRouteKeys?: Set<string>;
69
+
70
+ // === Lazy evaluation fields ===
71
+
72
+ /**
73
+ * Whether this entry is lazily evaluated.
74
+ * When true, routes are populated on first matching request.
75
+ */
76
+ lazy?: boolean;
77
+
78
+ /**
79
+ * For lazy entries: the UrlPatterns to evaluate
80
+ */
81
+ lazyPatterns?: unknown;
82
+
83
+ /**
84
+ * For lazy entries: captured context at definition time
85
+ */
86
+ lazyContext?: LazyIncludeContext;
87
+
88
+ /**
89
+ * For lazy entries: whether patterns have been evaluated
90
+ */
91
+ lazyEvaluated?: boolean;
92
+
93
+ /**
94
+ * Cache profiles for DSL-time cache("profileName") resolution.
95
+ * Set on all entries (lazy and non-lazy) so loadManifest() can
96
+ * propagate them into the HelperContext Store.
97
+ */
98
+ cacheProfiles?: Record<
99
+ string,
100
+ import("../cache/profile-registry.js").CacheProfile
101
+ >;
102
+ }
@@ -0,0 +1,148 @@
1
+ import type { ReactNode } from "react";
2
+ import type { ErrorInfo, NotFoundInfo } from "./boundaries.js";
3
+
4
+ /**
5
+ * CSS class(es) for a ViewTransition phase.
6
+ * Can be a simple string or an object mapping transition types to class names
7
+ * for direction-aware transitions (e.g., { "navigation": "slide-right", "navigation-back": "slide-left" }).
8
+ */
9
+ export type ViewTransitionClass = Record<string, string> | string;
10
+
11
+ /**
12
+ * Configuration for React's <ViewTransition> component.
13
+ * Maps directly to ViewTransitionProps (minus children/ref/callbacks).
14
+ */
15
+ export interface TransitionConfig {
16
+ enter?: ViewTransitionClass;
17
+ exit?: ViewTransitionClass;
18
+ update?: ViewTransitionClass;
19
+ share?: ViewTransitionClass;
20
+ default?: ViewTransitionClass;
21
+ name?: string;
22
+ }
23
+
24
+ /**
25
+ * Resolved segment with component
26
+ *
27
+ * Segment types:
28
+ * - layout: Wraps child content via <Outlet />
29
+ * - route: The leaf content for a URL
30
+ * - parallel: Named slots rendered via <ParallelOutlet name="@slot" />
31
+ * - loader: Data segment (no visual rendering, carries loaderData)
32
+ * - error: Error fallback segment (replaces failed segment with error UI)
33
+ * - notFound: Not found fallback segment (replaces segment when data not found)
34
+ *
35
+ * @internal This type is an implementation detail and may change without notice.
36
+ */
37
+ export interface ResolvedSegment {
38
+ id: string;
39
+ namespace: string; // Optional namespace for segment (used for parallel groups)
40
+ type: "layout" | "route" | "parallel" | "loader" | "error" | "notFound";
41
+ index: number;
42
+ component: ReactNode; // Component, handler promise, or resolved element
43
+ loading?: ReactNode; // Loading component for this segment (shown during navigation)
44
+ transition?: TransitionConfig; // ViewTransition config for this segment
45
+ layout?: ReactNode; // Layout element to wrap content (used by intercept segments)
46
+ params?: Record<string, string>;
47
+ slot?: string; // For parallel segments: '@sidebar', '@modal', etc.
48
+ belongsToRoute?: boolean; // True if segment belongs to the matched route (route itself + its children)
49
+ layoutName?: string; // For layouts: the layout name identifier
50
+ parallelName?: string; // For parallels: the parallel group name (used to match with revalidations)
51
+ // Loader-specific fields
52
+ loaderId?: string; // For loaders: the loader $$id identifier
53
+ loaderData?: any; // For loaders: the resolved data from loader execution
54
+ // Intercept loader fields (for streaming loader data in parallel segments)
55
+ loaderDataPromise?: Promise<any[]> | any[]; // Loader data promise or resolved array
56
+ loaderIds?: string[]; // IDs ($$id) of loaders for this segment
57
+ // Error-specific fields
58
+ error?: ErrorInfo; // For error segments: the error information
59
+ // NotFound-specific fields
60
+ notFoundInfo?: NotFoundInfo; // For notFound segments: the not found information
61
+ // Mount path from include() scope, used for MountContext.Provider wrapping
62
+ mountPath?: string;
63
+ }
64
+
65
+ /**
66
+ * Segment metadata (without component)
67
+ *
68
+ * @internal This type is an implementation detail and may change without notice.
69
+ */
70
+ export interface SegmentMetadata {
71
+ id: string;
72
+ type: "layout" | "route" | "parallel" | "loader" | "error" | "notFound";
73
+ index: number;
74
+ params?: Record<string, string>;
75
+ slot?: string;
76
+ loaderId?: string;
77
+ error?: ErrorInfo;
78
+ notFoundInfo?: NotFoundInfo;
79
+ }
80
+
81
+ // Note: route symbols are now defined in route-definition.ts
82
+ // as properties on the route() function
83
+
84
+ /**
85
+ * State of a named slot (e.g., @modal, @sidebar)
86
+ * Used for intercepting routes where slots render alternative content
87
+ *
88
+ * @internal This type is an implementation detail and may change without notice.
89
+ */
90
+ export interface SlotState {
91
+ /**
92
+ * Whether the slot is currently active (has content to render)
93
+ */
94
+ active: boolean;
95
+ /**
96
+ * Segments for this slot when active
97
+ */
98
+ segments?: ResolvedSegment[];
99
+ }
100
+
101
+ /**
102
+ * Props passed to the root layout component
103
+ */
104
+ export interface RootLayoutProps {
105
+ children: ReactNode;
106
+ }
107
+
108
+ /**
109
+ * Router match result
110
+ *
111
+ * @internal This type is an implementation detail and may change without notice.
112
+ */
113
+ export interface MatchResult {
114
+ segments: ResolvedSegment[];
115
+ matched: string[];
116
+ diff: string[];
117
+ /**
118
+ * Merged route params from all matched segments
119
+ * Available for use by the handler after route matching
120
+ */
121
+ params: Record<string, string>;
122
+ /**
123
+ * The matched route name (includes name prefix from include()).
124
+ * Used by ctx.reverse() for local name resolution.
125
+ */
126
+ routeName?: string;
127
+ /**
128
+ * State of named slots for this route match
129
+ * Key is slot name (e.g., "@modal"), value is slot state
130
+ * Slots are used for intercepting routes during soft navigation
131
+ */
132
+ slots?: Record<string, SlotState>;
133
+ /**
134
+ * Redirect URL for trailing slash normalization.
135
+ * When set, the RSC handler should return a 308 redirect to this URL
136
+ * instead of rendering the page.
137
+ */
138
+ redirect?: string;
139
+ /**
140
+ * Route-level middleware collected from the matched entry tree.
141
+ * These run with the same onion-style execution as app-level middleware,
142
+ * wrapping the entire RSC response creation.
143
+ */
144
+ routeMiddleware?: Array<{
145
+ handler: import("../router/middleware.js").MiddlewareFn;
146
+ params: Record<string, string>;
147
+ }>;
148
+ }
package/src/types.ts ADDED
@@ -0,0 +1 @@
1
+ export * from "./types/index.js";