@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,265 @@
1
+ /**
2
+ * Build-time Route Trie Construction
3
+ *
4
+ * Builds a serializable trie from the route manifest for O(path_length)
5
+ * route matching at runtime. Each trie leaf embeds the route's ancestry
6
+ * shortCodes for layout pruning.
7
+ */
8
+
9
+ import {
10
+ parsePattern,
11
+ type ParsedSegment,
12
+ } from "../router/pattern-matching.js";
13
+
14
+ // -- Trie data structures (compact keys for JSON serialization) --
15
+
16
+ export interface TrieLeaf {
17
+ /** Route name (e.g., "site.l1_500") */
18
+ n: string;
19
+ /** Static prefix of the entry (e.g., "/site") */
20
+ sp: string;
21
+ /** Ancestry shortCodes from root to route [M0L0, M0L0L0, M0L0L0R499] */
22
+ a: string[];
23
+ /** Optional param names (absent params get empty string value) */
24
+ op?: string[];
25
+ /** Constraint validation: paramName -> allowed values */
26
+ cv?: Record<string, string[]>;
27
+ /** Ordered param names for this route (positional) */
28
+ pa?: string[];
29
+ /** Trailing slash mode */
30
+ ts?: string;
31
+ /** Route has pre-rendered data available */
32
+ pr?: true;
33
+ /** Passthrough: handler kept in bundle for live fallback on unknown params */
34
+ pt?: true;
35
+ /** Response type for non-RSC routes (json, text, image, any) */
36
+ rt?: string;
37
+ /** Negotiate variants: response-type routes sharing this path */
38
+ nv?: Array<{ routeKey: string; responseType: string }>;
39
+ /** RSC-first: RSC route was defined before response-type variants */
40
+ rf?: true;
41
+ }
42
+
43
+ export interface TrieNode {
44
+ /** Route terminal at this node */
45
+ r?: TrieLeaf;
46
+ /** Static segment children */
47
+ s?: Record<string, TrieNode>;
48
+ /** Param child: { n: paramName, c: child node } */
49
+ p?: { n: string; c: TrieNode };
50
+ /** Suffix-param children keyed by suffix (e.g., ".html" → { n: "productId", c: ... }) */
51
+ xp?: Record<string, { n: string; c: TrieNode }>;
52
+ /** Wildcard terminal: leaf + paramName */
53
+ w?: TrieLeaf & { pn: string };
54
+ }
55
+
56
+ /**
57
+ * Build a route trie from build-time manifest data.
58
+ *
59
+ * @param routeManifest - Map of route name to full URL pattern
60
+ * @param routeAncestry - Map of route name to ancestry shortCodes
61
+ * @param routeToStaticPrefix - Map of route name to its entry's staticPrefix
62
+ * @param routeTrailingSlash - Optional map of route name to trailing slash mode
63
+ */
64
+ export function buildRouteTrie(
65
+ routeManifest: Record<string, string>,
66
+ routeAncestry: Record<string, string[]>,
67
+ routeToStaticPrefix: Record<string, string>,
68
+ routeTrailingSlash?: Record<string, string>,
69
+ prerenderRouteNames?: Set<string>,
70
+ passthroughRouteNames?: Set<string>,
71
+ responseTypeRoutes?: Record<string, string>,
72
+ ): TrieNode {
73
+ const root: TrieNode = {};
74
+
75
+ for (const [routeName, pattern] of Object.entries(routeManifest)) {
76
+ const ancestry = routeAncestry[routeName] || [];
77
+ const staticPrefix = routeToStaticPrefix[routeName] || "";
78
+ const trailingSlash = routeTrailingSlash?.[routeName];
79
+ const responseType = responseTypeRoutes?.[routeName];
80
+
81
+ // Detect and strip trailing slash from pattern for parsing
82
+ const hasTrailingSlash = pattern.length > 1 && pattern.endsWith("/");
83
+ const normalizedPattern = hasTrailingSlash ? pattern.slice(0, -1) : pattern;
84
+
85
+ const segments = parsePattern(normalizedPattern);
86
+ insertRoute(root, segments, 0, {
87
+ n: routeName,
88
+ sp: staticPrefix,
89
+ a: ancestry,
90
+ ...(trailingSlash ? { ts: trailingSlash } : {}),
91
+ ...(prerenderRouteNames?.has(routeName) ? { pr: true } : {}),
92
+ ...(passthroughRouteNames?.has(routeName) ? { pt: true } : {}),
93
+ ...(responseType ? { rt: responseType } : {}),
94
+ });
95
+ }
96
+
97
+ return root;
98
+ }
99
+
100
+ /**
101
+ * Insert a route into the trie, handling optional params by forking
102
+ * the insertion path (one terminal without the param, one with).
103
+ */
104
+ function insertRoute(
105
+ node: TrieNode,
106
+ segments: ParsedSegment[],
107
+ index: number,
108
+ leaf: Omit<TrieLeaf, "op" | "cv" | "pa">,
109
+ ): void {
110
+ // Collect param names, optional param names, and constraints across all segments
111
+ const paramNames: string[] = [];
112
+ const optionalParams: string[] = [];
113
+ const constraints: Record<string, string[]> = {};
114
+
115
+ for (const seg of segments) {
116
+ if (seg.type === "param") {
117
+ paramNames.push(seg.value);
118
+ if (seg.optional) {
119
+ optionalParams.push(seg.value);
120
+ }
121
+ if (seg.constraint) {
122
+ constraints[seg.value] = seg.constraint;
123
+ }
124
+ }
125
+ }
126
+
127
+ const fullLeaf: TrieLeaf = {
128
+ ...leaf,
129
+ ...(paramNames.length > 0 ? { pa: paramNames } : {}),
130
+ ...(optionalParams.length > 0 ? { op: optionalParams } : {}),
131
+ ...(Object.keys(constraints).length > 0 ? { cv: constraints } : {}),
132
+ };
133
+
134
+ insertSegments(node, segments, index, fullLeaf);
135
+ }
136
+
137
+ /**
138
+ * Recursively insert segments into the trie.
139
+ * For optional params, we add a terminal at the current node (param absent)
140
+ * AND continue inserting into the param child (param present).
141
+ */
142
+ /**
143
+ * Extract ancestry map from a built trie by visiting all leaf nodes.
144
+ * Returns { routeName: ancestryShortCodes[] } for every route in the trie.
145
+ */
146
+ export function extractAncestryFromTrie(
147
+ root: TrieNode,
148
+ ): Record<string, string[]> {
149
+ const result: Record<string, string[]> = {};
150
+
151
+ function visit(node: TrieNode): void {
152
+ if (node.r) {
153
+ result[node.r.n] = node.r.a;
154
+ }
155
+ if (node.w) {
156
+ result[node.w.n] = node.w.a;
157
+ }
158
+ if (node.s) {
159
+ for (const child of Object.values(node.s)) {
160
+ visit(child);
161
+ }
162
+ }
163
+ if (node.xp) {
164
+ for (const child of Object.values(node.xp)) {
165
+ visit(child.c);
166
+ }
167
+ }
168
+ if (node.p) {
169
+ visit(node.p.c);
170
+ }
171
+ }
172
+
173
+ visit(root);
174
+ return result;
175
+ }
176
+
177
+ /**
178
+ * Merge a new leaf with an existing leaf, handling content negotiation.
179
+ * When an RSC route and response-type routes share the same URL pattern,
180
+ * the RSC route becomes the primary leaf and response-type routes are
181
+ * appended to the nv (negotiate variants) array.
182
+ * Multiple response types on the same path are supported (json + text + xml).
183
+ */
184
+ function mergeLeaves(existing: TrieLeaf | undefined, leaf: TrieLeaf): TrieLeaf {
185
+ if (!existing) return leaf;
186
+
187
+ if (existing.rt && leaf.rt) {
188
+ // Both are response-type: preserve old as variant
189
+ const merged = leaf;
190
+ merged.nv = existing.nv || [];
191
+ merged.nv.push({ routeKey: existing.n, responseType: existing.rt });
192
+ return merged;
193
+ }
194
+ if (leaf.rt && !existing.rt) {
195
+ // RSC primary exists, new leaf is response-type: append variant
196
+ // RSC was defined first (it was already the existing leaf)
197
+ if (!existing.nv) {
198
+ existing.nv = [];
199
+ existing.rf = true;
200
+ }
201
+ existing.nv.push({ routeKey: leaf.n, responseType: leaf.rt });
202
+ return existing;
203
+ }
204
+ if (!leaf.rt && existing.rt) {
205
+ // Response-type was primary, new leaf is RSC: swap and move old to variants
206
+ // RSC was defined second (response-type was already the existing leaf)
207
+ if (!leaf.nv) leaf.nv = [];
208
+ if (existing.nv) leaf.nv.push(...existing.nv);
209
+ leaf.nv.push({ routeKey: existing.n, responseType: existing.rt });
210
+ // rf intentionally not set — RSC came after response-type variants
211
+ return leaf;
212
+ }
213
+ // Both RSC (last wins): overwrite
214
+ return leaf;
215
+ }
216
+
217
+ function mergeLeaf(node: TrieNode, leaf: TrieLeaf): void {
218
+ node.r = mergeLeaves(node.r, leaf);
219
+ }
220
+
221
+ function insertSegments(
222
+ node: TrieNode,
223
+ segments: ParsedSegment[],
224
+ index: number,
225
+ leaf: TrieLeaf,
226
+ ): void {
227
+ // Base case: all segments consumed, add terminal
228
+ if (index >= segments.length) {
229
+ mergeLeaf(node, leaf);
230
+ return;
231
+ }
232
+
233
+ const segment = segments[index];
234
+
235
+ if (segment.type === "static") {
236
+ if (!node.s) node.s = {};
237
+ if (!node.s[segment.value]) node.s[segment.value] = {};
238
+ insertSegments(node.s[segment.value], segments, index + 1, leaf);
239
+ } else if (segment.type === "param") {
240
+ if (segment.optional) {
241
+ // Optional param: add terminal at current node (param absent)
242
+ mergeLeaf(node, leaf);
243
+ // AND continue with param child (param present)
244
+ }
245
+ if (segment.suffix) {
246
+ // Suffix param: keyed by suffix string (e.g., ".html")
247
+ if (!node.xp) node.xp = {};
248
+ if (!node.xp[segment.suffix]) {
249
+ node.xp[segment.suffix] = { n: segment.value, c: {} };
250
+ }
251
+ insertSegments(node.xp[segment.suffix].c, segments, index + 1, leaf);
252
+ } else {
253
+ if (!node.p) {
254
+ node.p = { n: segment.value, c: {} };
255
+ }
256
+ insertSegments(node.p.c, segments, index + 1, leaf);
257
+ }
258
+ } else if (segment.type === "wildcard") {
259
+ // Wildcard consumes all remaining segments
260
+ const wildLeaf = { ...leaf, pn: "*" };
261
+ const existing = node.w ? ({ ...node.w } as TrieLeaf) : undefined;
262
+ const merged = mergeLeaves(existing, wildLeaf);
263
+ node.w = merged as TrieLeaf & { pn: string };
264
+ }
265
+ }
@@ -0,0 +1,25 @@
1
+ import ts from "typescript";
2
+
3
+ export function getStringValue(node: ts.Node): string | null {
4
+ if (ts.isStringLiteral(node)) return node.text;
5
+ if (ts.isNoSubstitutionTemplateLiteral(node)) return node.text;
6
+ return null;
7
+ }
8
+
9
+ export function extractObjectStringProperties(
10
+ node: ts.ObjectLiteralExpression,
11
+ ): Record<string, string> {
12
+ const result: Record<string, string> = {};
13
+ for (const prop of node.properties) {
14
+ if (!ts.isPropertyAssignment(prop)) continue;
15
+ const key = ts.isIdentifier(prop.name)
16
+ ? prop.name.text
17
+ : ts.isStringLiteral(prop.name)
18
+ ? prop.name.text
19
+ : null;
20
+ if (!key) continue;
21
+ const val = getStringValue(prop.initializer);
22
+ if (val !== null) result[key] = val;
23
+ }
24
+ return result;
25
+ }
@@ -0,0 +1,98 @@
1
+ import ts from "typescript";
2
+ import {
3
+ getStringValue,
4
+ extractObjectStringProperties,
5
+ } from "./ast-helpers.js";
6
+ import { extractParamsFromPattern } from "./param-extraction.js";
7
+
8
+ // ---------------------------------------------------------------------------
9
+ // AST-based route extraction
10
+ // ---------------------------------------------------------------------------
11
+
12
+ /**
13
+ * Extract route definitions from source code by walking the TypeScript AST.
14
+ * Finds path() and path.json(), path.md(), etc. call expressions and extracts
15
+ * the pattern, name, params, and optional search schema from each.
16
+ * Skips unnamed paths (no { name: "..." }).
17
+ */
18
+ export function extractRoutesFromSource(code: string): Array<{
19
+ name: string;
20
+ pattern: string;
21
+ params?: Record<string, string>;
22
+ search?: Record<string, string>;
23
+ }> {
24
+ const sourceFile = ts.createSourceFile(
25
+ "input.tsx",
26
+ code,
27
+ ts.ScriptTarget.Latest,
28
+ true,
29
+ ts.ScriptKind.TSX,
30
+ );
31
+ const routes: Array<{
32
+ name: string;
33
+ pattern: string;
34
+ params?: Record<string, string>;
35
+ search?: Record<string, string>;
36
+ }> = [];
37
+
38
+ function visit(node: ts.Node) {
39
+ if (ts.isCallExpression(node)) {
40
+ const callee = node.expression;
41
+ const isPath =
42
+ (ts.isIdentifier(callee) && callee.text === "path") ||
43
+ (ts.isPropertyAccessExpression(callee) &&
44
+ ts.isIdentifier(callee.expression) &&
45
+ callee.expression.text === "path");
46
+
47
+ if (isPath && node.arguments.length >= 1) {
48
+ const route = extractRouteFromCallExpression(node);
49
+ if (route) routes.push(route);
50
+ }
51
+ }
52
+ ts.forEachChild(node, visit);
53
+ }
54
+
55
+ visit(sourceFile);
56
+ return routes;
57
+ }
58
+
59
+ function extractRouteFromCallExpression(node: ts.CallExpression): {
60
+ name: string;
61
+ pattern: string;
62
+ params?: Record<string, string>;
63
+ search?: Record<string, string>;
64
+ } | null {
65
+ const patternNode = node.arguments[0];
66
+ const pattern = getStringValue(patternNode);
67
+ if (pattern === null) return null;
68
+
69
+ let name: string | null = null;
70
+ let search: Record<string, string> | undefined;
71
+
72
+ for (let i = 1; i < node.arguments.length; i++) {
73
+ const arg = node.arguments[i];
74
+ if (ts.isObjectLiteralExpression(arg)) {
75
+ for (const prop of arg.properties) {
76
+ if (!ts.isPropertyAssignment(prop)) continue;
77
+ const propName = ts.isIdentifier(prop.name) ? prop.name.text : null;
78
+ if (propName === "name") {
79
+ name = getStringValue(prop.initializer);
80
+ } else if (
81
+ propName === "search" &&
82
+ ts.isObjectLiteralExpression(prop.initializer)
83
+ ) {
84
+ search = extractObjectStringProperties(prop.initializer);
85
+ }
86
+ }
87
+ }
88
+ }
89
+
90
+ if (!name) return null;
91
+ const params = extractParamsFromPattern(pattern);
92
+ return {
93
+ name,
94
+ pattern,
95
+ ...(params ? { params } : {}),
96
+ ...(search && Object.keys(search).length > 0 ? { search } : {}),
97
+ };
98
+ }
@@ -0,0 +1,102 @@
1
+ import {
2
+ extractParamsFromPattern,
3
+ formatRouteEntry,
4
+ } from "./param-extraction.js";
5
+ import { isAutoGeneratedRouteName } from "../../route-name.js";
6
+
7
+ // ---------------------------------------------------------------------------
8
+ // Code generation
9
+ // ---------------------------------------------------------------------------
10
+
11
+ /**
12
+ * Generate a per-module types file from extracted routes.
13
+ * Output has zero imports, preventing circular references.
14
+ */
15
+ export function generatePerModuleTypesSource(
16
+ routes: Array<{
17
+ name: string;
18
+ pattern: string;
19
+ params?: Record<string, string>;
20
+ search?: Record<string, string>;
21
+ }>,
22
+ ): string {
23
+ const valid = routes.filter(({ name }) => {
24
+ if (!name || /["'\\`\n\r]/.test(name)) {
25
+ console.warn(
26
+ `[rsc-router] Skipping route with invalid name: ${JSON.stringify(name)}`,
27
+ );
28
+ return false;
29
+ }
30
+ return true;
31
+ });
32
+
33
+ // Deduplicate by name (first definition wins -- primary route before variants)
34
+ const deduped = new Map<
35
+ string,
36
+ {
37
+ pattern: string;
38
+ params?: Record<string, string>;
39
+ search?: Record<string, string>;
40
+ }
41
+ >();
42
+ for (const { name, pattern, params, search } of valid) {
43
+ if (deduped.has(name)) {
44
+ console.warn(
45
+ `[rsc-router] Duplicate route name "${name}" — keeping first definition`,
46
+ );
47
+ continue;
48
+ }
49
+ deduped.set(name, { pattern, params, search });
50
+ }
51
+ const sorted = [...deduped.entries()].sort(([a], [b]) => a.localeCompare(b));
52
+ const body = sorted
53
+ .map(([name, { pattern, params, search }]) => {
54
+ const key = /^[a-zA-Z_$][a-zA-Z0-9_$]*$/.test(name) ? name : `"${name}"`;
55
+ return formatRouteEntry(key, pattern, params, search);
56
+ })
57
+ .join("\n");
58
+ return `// Auto-generated by @rangojs/router - do not edit\nexport const routes = {\n${body}\n} as const;\nexport type routes = typeof routes;\n`;
59
+ }
60
+
61
+ /**
62
+ * Generates a .ts file that augments RSCRouter.GeneratedRouteMap
63
+ * with route name -> pattern mappings. This enables Handler<"routeName">
64
+ * without circular references since the file has no imports from the app.
65
+ */
66
+ export function generateRouteTypesSource(
67
+ routeManifest: Record<string, string>,
68
+ searchSchemas?: Record<string, Record<string, string>>,
69
+ ): string {
70
+ const entries = Object.entries(routeManifest)
71
+ .filter(([name]) => !isAutoGeneratedRouteName(name))
72
+ .sort(([a], [b]) => a.localeCompare(b));
73
+
74
+ const filteredSearchSchemas = searchSchemas
75
+ ? Object.fromEntries(
76
+ Object.entries(searchSchemas).filter(
77
+ ([name]) => !isAutoGeneratedRouteName(name),
78
+ ),
79
+ )
80
+ : undefined;
81
+
82
+ const objectBody = entries
83
+ .map(([name, pattern]) => {
84
+ const key = /^[a-zA-Z_$][a-zA-Z0-9_$]*$/.test(name) ? name : `"${name}"`;
85
+ const params = extractParamsFromPattern(pattern);
86
+ const search = filteredSearchSchemas?.[name];
87
+ return formatRouteEntry(key, pattern, params, search);
88
+ })
89
+ .join("\n");
90
+
91
+ return `// Auto-generated by @rangojs/router - do not edit
92
+ export const NamedRoutes = {
93
+ ${objectBody}
94
+ } as const;
95
+
96
+ declare global {
97
+ namespace RSCRouter {
98
+ interface GeneratedRouteMap extends Readonly<typeof NamedRoutes> {}
99
+ }
100
+ }
101
+ `;
102
+ }