@rangojs/router 0.0.0-experimental.8 → 0.0.0-experimental.80

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 (312) hide show
  1. package/AGENTS.md +9 -0
  2. package/README.md +942 -4
  3. package/dist/bin/rango.js +1689 -0
  4. package/dist/vite/index.js +4960 -935
  5. package/package.json +70 -60
  6. package/skills/breadcrumbs/SKILL.md +250 -0
  7. package/skills/cache-guide/SKILL.md +294 -0
  8. package/skills/caching/SKILL.md +93 -23
  9. package/skills/composability/SKILL.md +172 -0
  10. package/skills/debug-manifest/SKILL.md +12 -8
  11. package/skills/document-cache/SKILL.md +18 -16
  12. package/skills/fonts/SKILL.md +167 -0
  13. package/skills/handler-use/SKILL.md +362 -0
  14. package/skills/hooks/SKILL.md +334 -72
  15. package/skills/host-router/SKILL.md +218 -0
  16. package/skills/intercept/SKILL.md +151 -8
  17. package/skills/layout/SKILL.md +122 -3
  18. package/skills/links/SKILL.md +92 -31
  19. package/skills/loader/SKILL.md +404 -44
  20. package/skills/middleware/SKILL.md +205 -37
  21. package/skills/migrate-nextjs/SKILL.md +560 -0
  22. package/skills/migrate-react-router/SKILL.md +764 -0
  23. package/skills/mime-routes/SKILL.md +128 -0
  24. package/skills/parallel/SKILL.md +263 -1
  25. package/skills/prerender/SKILL.md +685 -0
  26. package/skills/rango/SKILL.md +87 -16
  27. package/skills/response-routes/SKILL.md +411 -0
  28. package/skills/route/SKILL.md +281 -14
  29. package/skills/router-setup/SKILL.md +210 -32
  30. package/skills/tailwind/SKILL.md +129 -0
  31. package/skills/theme/SKILL.md +9 -8
  32. package/skills/typesafety/SKILL.md +328 -89
  33. package/skills/use-cache/SKILL.md +324 -0
  34. package/src/__internal.ts +102 -4
  35. package/src/bin/rango.ts +321 -0
  36. package/src/browser/action-coordinator.ts +97 -0
  37. package/src/browser/action-response-classifier.ts +99 -0
  38. package/src/browser/app-version.ts +14 -0
  39. package/src/browser/event-controller.ts +92 -64
  40. package/src/browser/history-state.ts +80 -0
  41. package/src/browser/intercept-utils.ts +52 -0
  42. package/src/browser/link-interceptor.ts +24 -4
  43. package/src/browser/logging.ts +55 -0
  44. package/src/browser/merge-segment-loaders.ts +20 -12
  45. package/src/browser/navigation-bridge.ts +317 -560
  46. package/src/browser/navigation-client.ts +206 -68
  47. package/src/browser/navigation-store.ts +73 -55
  48. package/src/browser/navigation-transaction.ts +297 -0
  49. package/src/browser/network-error-handler.ts +61 -0
  50. package/src/browser/partial-update.ts +343 -316
  51. package/src/browser/prefetch/cache.ts +216 -0
  52. package/src/browser/prefetch/fetch.ts +206 -0
  53. package/src/browser/prefetch/observer.ts +65 -0
  54. package/src/browser/prefetch/policy.ts +48 -0
  55. package/src/browser/prefetch/queue.ts +160 -0
  56. package/src/browser/prefetch/resource-ready.ts +77 -0
  57. package/src/browser/rango-state.ts +112 -0
  58. package/src/browser/react/Link.tsx +253 -74
  59. package/src/browser/react/NavigationProvider.tsx +87 -11
  60. package/src/browser/react/context.ts +11 -0
  61. package/src/browser/react/filter-segment-order.ts +11 -0
  62. package/src/browser/react/index.ts +12 -12
  63. package/src/browser/react/location-state-shared.ts +95 -53
  64. package/src/browser/react/location-state.ts +60 -15
  65. package/src/browser/react/mount-context.ts +6 -1
  66. package/src/browser/react/nonce-context.ts +23 -0
  67. package/src/browser/react/shallow-equal.ts +27 -0
  68. package/src/browser/react/use-action.ts +29 -51
  69. package/src/browser/react/use-client-cache.ts +5 -3
  70. package/src/browser/react/use-handle.ts +30 -126
  71. package/src/browser/react/use-href.tsx +2 -2
  72. package/src/browser/react/use-link-status.ts +6 -5
  73. package/src/browser/react/use-navigation.ts +44 -65
  74. package/src/browser/react/use-params.ts +65 -0
  75. package/src/browser/react/use-pathname.ts +47 -0
  76. package/src/browser/react/use-router.ts +76 -0
  77. package/src/browser/react/use-search-params.ts +56 -0
  78. package/src/browser/react/use-segments.ts +80 -97
  79. package/src/browser/response-adapter.ts +73 -0
  80. package/src/browser/rsc-router.tsx +214 -58
  81. package/src/browser/scroll-restoration.ts +127 -52
  82. package/src/browser/segment-reconciler.ts +243 -0
  83. package/src/browser/segment-structure-assert.ts +16 -0
  84. package/src/browser/server-action-bridge.ts +510 -603
  85. package/src/browser/shallow.ts +6 -1
  86. package/src/browser/types.ts +141 -48
  87. package/src/browser/validate-redirect-origin.ts +29 -0
  88. package/src/build/generate-manifest.ts +235 -24
  89. package/src/build/generate-route-types.ts +39 -0
  90. package/src/build/index.ts +13 -0
  91. package/src/build/route-trie.ts +291 -0
  92. package/src/build/route-types/ast-helpers.ts +25 -0
  93. package/src/build/route-types/ast-route-extraction.ts +98 -0
  94. package/src/build/route-types/codegen.ts +102 -0
  95. package/src/build/route-types/include-resolution.ts +418 -0
  96. package/src/build/route-types/param-extraction.ts +48 -0
  97. package/src/build/route-types/per-module-writer.ts +128 -0
  98. package/src/build/route-types/router-processing.ts +618 -0
  99. package/src/build/route-types/scan-filter.ts +85 -0
  100. package/src/build/runtime-discovery.ts +231 -0
  101. package/src/cache/background-task.ts +34 -0
  102. package/src/cache/cache-key-utils.ts +44 -0
  103. package/src/cache/cache-policy.ts +125 -0
  104. package/src/cache/cache-runtime.ts +342 -0
  105. package/src/cache/cache-scope.ts +167 -309
  106. package/src/cache/cf/cf-cache-store.ts +571 -17
  107. package/src/cache/cf/index.ts +13 -3
  108. package/src/cache/document-cache.ts +116 -77
  109. package/src/cache/handle-capture.ts +81 -0
  110. package/src/cache/handle-snapshot.ts +41 -0
  111. package/src/cache/index.ts +1 -15
  112. package/src/cache/memory-segment-store.ts +191 -13
  113. package/src/cache/profile-registry.ts +73 -0
  114. package/src/cache/read-through-swr.ts +134 -0
  115. package/src/cache/segment-codec.ts +256 -0
  116. package/src/cache/taint.ts +153 -0
  117. package/src/cache/types.ts +72 -122
  118. package/src/client.rsc.tsx +3 -1
  119. package/src/client.tsx +135 -301
  120. package/src/component-utils.ts +4 -4
  121. package/src/components/DefaultDocument.tsx +5 -1
  122. package/src/context-var.ts +156 -0
  123. package/src/debug.ts +19 -9
  124. package/src/errors.ts +108 -2
  125. package/src/handle.ts +55 -29
  126. package/src/handles/MetaTags.tsx +73 -20
  127. package/src/handles/breadcrumbs.ts +66 -0
  128. package/src/handles/index.ts +1 -0
  129. package/src/handles/meta.ts +30 -13
  130. package/src/host/cookie-handler.ts +21 -15
  131. package/src/host/errors.ts +8 -8
  132. package/src/host/index.ts +4 -7
  133. package/src/host/pattern-matcher.ts +27 -27
  134. package/src/host/router.ts +61 -39
  135. package/src/host/testing.ts +8 -8
  136. package/src/host/types.ts +15 -7
  137. package/src/host/utils.ts +1 -1
  138. package/src/href-client.ts +119 -29
  139. package/src/index.rsc.ts +155 -19
  140. package/src/index.ts +251 -30
  141. package/src/internal-debug.ts +11 -0
  142. package/src/loader.rsc.ts +26 -157
  143. package/src/loader.ts +27 -10
  144. package/src/network-error-thrower.tsx +3 -1
  145. package/src/outlet-provider.tsx +45 -0
  146. package/src/prerender/param-hash.ts +37 -0
  147. package/src/prerender/store.ts +186 -0
  148. package/src/prerender.ts +524 -0
  149. package/src/reverse.ts +354 -0
  150. package/src/root-error-boundary.tsx +41 -29
  151. package/src/route-content-wrapper.tsx +7 -4
  152. package/src/route-definition/dsl-helpers.ts +1121 -0
  153. package/src/route-definition/helper-factories.ts +200 -0
  154. package/src/route-definition/helpers-types.ts +478 -0
  155. package/src/route-definition/index.ts +55 -0
  156. package/src/route-definition/redirect.ts +101 -0
  157. package/src/route-definition/resolve-handler-use.ts +149 -0
  158. package/src/route-definition.ts +1 -1428
  159. package/src/route-map-builder.ts +217 -123
  160. package/src/route-name.ts +53 -0
  161. package/src/route-types.ts +77 -8
  162. package/src/router/content-negotiation.ts +215 -0
  163. package/src/router/debug-manifest.ts +72 -0
  164. package/src/router/error-handling.ts +9 -9
  165. package/src/router/find-match.ts +160 -0
  166. package/src/router/handler-context.ts +438 -86
  167. package/src/router/intercept-resolution.ts +402 -0
  168. package/src/router/lazy-includes.ts +237 -0
  169. package/src/router/loader-resolution.ts +356 -128
  170. package/src/router/logging.ts +251 -0
  171. package/src/router/manifest.ts +163 -35
  172. package/src/router/match-api.ts +555 -0
  173. package/src/router/match-context.ts +5 -3
  174. package/src/router/match-handlers.ts +440 -0
  175. package/src/router/match-middleware/background-revalidation.ts +108 -93
  176. package/src/router/match-middleware/cache-lookup.ts +460 -10
  177. package/src/router/match-middleware/cache-store.ts +98 -26
  178. package/src/router/match-middleware/intercept-resolution.ts +57 -17
  179. package/src/router/match-middleware/segment-resolution.ts +80 -6
  180. package/src/router/match-pipelines.ts +10 -45
  181. package/src/router/match-result.ts +135 -35
  182. package/src/router/metrics.ts +240 -15
  183. package/src/router/middleware-cookies.ts +55 -0
  184. package/src/router/middleware-types.ts +220 -0
  185. package/src/router/middleware.ts +324 -369
  186. package/src/router/navigation-snapshot.ts +182 -0
  187. package/src/router/pattern-matching.ts +211 -43
  188. package/src/router/prerender-match.ts +502 -0
  189. package/src/router/preview-match.ts +98 -0
  190. package/src/router/request-classification.ts +310 -0
  191. package/src/router/revalidation.ts +137 -38
  192. package/src/router/route-snapshot.ts +245 -0
  193. package/src/router/router-context.ts +41 -21
  194. package/src/router/router-interfaces.ts +484 -0
  195. package/src/router/router-options.ts +618 -0
  196. package/src/router/router-registry.ts +24 -0
  197. package/src/router/segment-resolution/fresh.ts +748 -0
  198. package/src/router/segment-resolution/helpers.ts +268 -0
  199. package/src/router/segment-resolution/loader-cache.ts +199 -0
  200. package/src/router/segment-resolution/revalidation.ts +1379 -0
  201. package/src/router/segment-resolution/static-store.ts +67 -0
  202. package/src/router/segment-resolution.ts +21 -0
  203. package/src/router/segment-wrappers.ts +291 -0
  204. package/src/router/telemetry-otel.ts +299 -0
  205. package/src/router/telemetry.ts +300 -0
  206. package/src/router/timeout.ts +148 -0
  207. package/src/router/trie-matching.ts +239 -0
  208. package/src/router/types.ts +78 -3
  209. package/src/router.ts +740 -4252
  210. package/src/rsc/handler-context.ts +45 -0
  211. package/src/rsc/handler.ts +907 -797
  212. package/src/rsc/helpers.ts +140 -6
  213. package/src/rsc/index.ts +0 -20
  214. package/src/rsc/loader-fetch.ts +229 -0
  215. package/src/rsc/manifest-init.ts +90 -0
  216. package/src/rsc/nonce.ts +14 -0
  217. package/src/rsc/origin-guard.ts +141 -0
  218. package/src/rsc/progressive-enhancement.ts +391 -0
  219. package/src/rsc/response-error.ts +37 -0
  220. package/src/rsc/response-route-handler.ts +347 -0
  221. package/src/rsc/rsc-rendering.ts +246 -0
  222. package/src/rsc/runtime-warnings.ts +42 -0
  223. package/src/rsc/server-action.ts +356 -0
  224. package/src/rsc/ssr-setup.ts +128 -0
  225. package/src/rsc/types.ts +46 -11
  226. package/src/search-params.ts +230 -0
  227. package/src/segment-content-promise.ts +67 -0
  228. package/src/segment-loader-promise.ts +122 -0
  229. package/src/segment-system.tsx +134 -36
  230. package/src/server/context.ts +341 -61
  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 +113 -15
  234. package/src/server/loader-registry.ts +24 -64
  235. package/src/server/request-context.ts +607 -81
  236. package/src/server.ts +35 -130
  237. package/src/ssr/index.tsx +103 -30
  238. package/src/static-handler.ts +126 -0
  239. package/src/theme/ThemeProvider.tsx +21 -15
  240. package/src/theme/ThemeScript.tsx +5 -5
  241. package/src/theme/constants.ts +5 -2
  242. package/src/theme/index.ts +4 -14
  243. package/src/theme/theme-context.ts +4 -30
  244. package/src/theme/theme-script.ts +21 -18
  245. package/src/types/boundaries.ts +158 -0
  246. package/src/types/cache-types.ts +198 -0
  247. package/src/types/error-types.ts +192 -0
  248. package/src/types/global-namespace.ts +100 -0
  249. package/src/types/handler-context.ts +791 -0
  250. package/src/types/index.ts +88 -0
  251. package/src/types/loader-types.ts +210 -0
  252. package/src/types/route-config.ts +170 -0
  253. package/src/types/route-entry.ts +120 -0
  254. package/src/types/segments.ts +150 -0
  255. package/src/types.ts +1 -1623
  256. package/src/urls/include-helper.ts +207 -0
  257. package/src/urls/index.ts +53 -0
  258. package/src/urls/path-helper-types.ts +372 -0
  259. package/src/urls/path-helper.ts +364 -0
  260. package/src/urls/pattern-types.ts +107 -0
  261. package/src/urls/response-types.ts +116 -0
  262. package/src/urls/type-extraction.ts +372 -0
  263. package/src/urls/urls-function.ts +98 -0
  264. package/src/urls.ts +1 -802
  265. package/src/use-loader.tsx +161 -81
  266. package/src/vite/discovery/bundle-postprocess.ts +181 -0
  267. package/src/vite/discovery/discover-routers.ts +348 -0
  268. package/src/vite/discovery/prerender-collection.ts +439 -0
  269. package/src/vite/discovery/route-types-writer.ts +258 -0
  270. package/src/vite/discovery/self-gen-tracking.ts +47 -0
  271. package/src/vite/discovery/state.ts +117 -0
  272. package/src/vite/discovery/virtual-module-codegen.ts +203 -0
  273. package/src/vite/index.ts +15 -1133
  274. package/src/vite/plugin-types.ts +103 -0
  275. package/src/vite/plugins/cjs-to-esm.ts +93 -0
  276. package/src/vite/plugins/client-ref-dedup.ts +115 -0
  277. package/src/vite/plugins/client-ref-hashing.ts +105 -0
  278. package/src/vite/{expose-action-id.ts → plugins/expose-action-id.ts} +72 -53
  279. package/src/vite/plugins/expose-id-utils.ts +299 -0
  280. package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
  281. package/src/vite/plugins/expose-ids/handler-transform.ts +209 -0
  282. package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
  283. package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
  284. package/src/vite/plugins/expose-ids/types.ts +45 -0
  285. package/src/vite/plugins/expose-internal-ids.ts +786 -0
  286. package/src/vite/plugins/performance-tracks.ts +88 -0
  287. package/src/vite/plugins/refresh-cmd.ts +127 -0
  288. package/src/vite/plugins/use-cache-transform.ts +323 -0
  289. package/src/vite/plugins/version-injector.ts +83 -0
  290. package/src/vite/plugins/version-plugin.ts +266 -0
  291. package/src/vite/{virtual-entries.ts → plugins/virtual-entries.ts} +23 -14
  292. package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
  293. package/src/vite/rango.ts +462 -0
  294. package/src/vite/router-discovery.ts +918 -0
  295. package/src/vite/utils/ast-handler-extract.ts +517 -0
  296. package/src/vite/utils/banner.ts +36 -0
  297. package/src/vite/utils/bundle-analysis.ts +137 -0
  298. package/src/vite/utils/manifest-utils.ts +70 -0
  299. package/src/vite/{package-resolution.ts → utils/package-resolution.ts} +25 -29
  300. package/src/vite/utils/prerender-utils.ts +221 -0
  301. package/src/vite/utils/shared-utils.ts +170 -0
  302. package/CLAUDE.md +0 -43
  303. package/src/browser/lru-cache.ts +0 -69
  304. package/src/browser/request-controller.ts +0 -164
  305. package/src/cache/memory-store.ts +0 -253
  306. package/src/href-context.ts +0 -33
  307. package/src/href.ts +0 -255
  308. package/src/server/route-manifest-cache.ts +0 -173
  309. package/src/vite/expose-handle-id.ts +0 -209
  310. package/src/vite/expose-loader-id.ts +0 -426
  311. package/src/vite/expose-location-state-id.ts +0 -177
  312. /package/src/vite/{version.d.ts → plugins/version.d.ts} +0 -0
@@ -0,0 +1,70 @@
1
+ /**
2
+ * Flatten prefix tree leaf nodes into precomputed route entries.
3
+ * Leaf nodes have no children (no nested includes), so their routes can be
4
+ * used directly by evaluateLazyEntry() without running the handler.
5
+ * Non-leaf nodes are skipped because they have nested lazy includes that
6
+ * require the handler to run for discovery.
7
+ */
8
+ export function flattenLeafEntries(
9
+ prefixTree: Record<string, any>,
10
+ routeManifest: Record<string, string>,
11
+ result: Array<{ staticPrefix: string; routes: Record<string, string> }>,
12
+ ): void {
13
+ function visit(node: any): void {
14
+ const children = node.children || {};
15
+ if (
16
+ Object.keys(children).length === 0 &&
17
+ node.routes &&
18
+ node.routes.length > 0
19
+ ) {
20
+ // Leaf node: collect its routes from the manifest
21
+ const routes: Record<string, string> = {};
22
+ for (const name of node.routes) {
23
+ if (name in routeManifest) {
24
+ routes[name] = routeManifest[name];
25
+ }
26
+ }
27
+ result.push({ staticPrefix: node.staticPrefix, routes });
28
+ } else {
29
+ // Non-leaf: recurse into children
30
+ for (const child of Object.values(children)) {
31
+ visit(child);
32
+ }
33
+ }
34
+ }
35
+ for (const node of Object.values(prefixTree)) {
36
+ visit(node);
37
+ }
38
+ }
39
+
40
+ /**
41
+ * Walk prefix tree to map each route name to its scope's staticPrefix.
42
+ */
43
+ export function buildRouteToStaticPrefix(
44
+ prefixTree: Record<string, any>,
45
+ result: Record<string, string>,
46
+ ): void {
47
+ function visit(node: any): void {
48
+ const sp = node.staticPrefix || "";
49
+ for (const name of node.routes || []) {
50
+ result[name] = sp;
51
+ }
52
+ for (const child of Object.values(node.children || {})) {
53
+ visit(child);
54
+ }
55
+ }
56
+ for (const node of Object.values(prefixTree)) {
57
+ visit(node);
58
+ }
59
+ }
60
+
61
+ /**
62
+ * Wrap a value as `JSON.parse('...')` instead of a JS object literal.
63
+ * V8's JSON parser is significantly faster than its full JS parser for large
64
+ * objects, so this improves startup time for big route manifests.
65
+ */
66
+ export function jsonParseExpression(value: unknown): string {
67
+ const json = JSON.stringify(value);
68
+ const escaped = json.replace(/\\/g, "\\\\").replace(/'/g, "\\'");
69
+ return `JSON.parse('${escaped}')`;
70
+ }
@@ -7,7 +7,7 @@
7
7
 
8
8
  import { existsSync } from "node:fs";
9
9
  import { resolve } from "node:path";
10
- import packageJson from "../../package.json" with { type: "json" };
10
+ import packageJson from "../../../package.json" with { type: "json" };
11
11
 
12
12
  /**
13
13
  * The canonical name used in virtual entries (without scope)
@@ -46,21 +46,26 @@ export function isWorkspaceDevelopment(): boolean {
46
46
  }
47
47
 
48
48
  /**
49
- * Subpaths that need to be excluded from Vite's dependency optimization
50
- * and potentially aliased
49
+ * Subpaths derived from package.json exports that use TypeScript source.
50
+ * These must be excluded from Vite's dependency optimization (they ship
51
+ * as .ts/.tsx, not compiled JS) and aliased when installed from npm.
52
+ *
53
+ * Derived automatically from the exports field to prevent drift.
51
54
  */
52
- const PACKAGE_SUBPATHS = [
53
- "",
54
- "/browser",
55
- "/client",
56
- "/server",
57
- "/rsc",
58
- "/ssr",
59
- "/internal/deps/browser",
60
- "/internal/deps/html-stream-client",
61
- "/internal/deps/ssr",
62
- "/internal/deps/rsc",
63
- ] as const;
55
+ const SOURCE_EXPORT_SUBPATHS = Object.keys(packageJson.exports)
56
+ .filter((key) => {
57
+ const entry = (
58
+ packageJson.exports as Record<string, Record<string, string>>
59
+ )[key];
60
+ // Include if any non-types condition points to TypeScript source
61
+ return Object.entries(entry).some(
62
+ ([condition, path]) =>
63
+ condition !== "types" &&
64
+ typeof path === "string" &&
65
+ /\.tsx?$/.test(path),
66
+ );
67
+ })
68
+ .map((key) => key.replace(/^\./, ""));
64
69
 
65
70
  /**
66
71
  * Generate the list of modules to exclude from Vite's dependency optimization.
@@ -72,7 +77,7 @@ export function getExcludeDeps(): string[] {
72
77
  const packageName = getPublishedPackageName();
73
78
  const excludes: string[] = [];
74
79
 
75
- for (const subpath of PACKAGE_SUBPATHS) {
80
+ for (const subpath of SOURCE_EXPORT_SUBPATHS) {
76
81
  // Add scoped package paths
77
82
  excludes.push(`${packageName}${subpath}`);
78
83
  // Add virtual/aliased paths (before alias resolution)
@@ -85,20 +90,11 @@ export function getExcludeDeps(): string[] {
85
90
  }
86
91
 
87
92
  /**
88
- * Subpaths that need aliasing (subset of PACKAGE_SUBPATHS)
93
+ * Subpaths that need aliasing same as SOURCE_EXPORT_SUBPATHS.
94
+ * When installed from npm, virtual entries may use a different package name
95
+ * than the published one; aliases bridge them.
89
96
  */
90
- const ALIAS_SUBPATHS = [
91
- "/internal/deps/browser",
92
- "/internal/deps/ssr",
93
- "/internal/deps/rsc",
94
- "/internal/deps/html-stream-client",
95
- "/internal/deps/html-stream-server",
96
- "/browser",
97
- "/client",
98
- "/server",
99
- "/rsc",
100
- "/ssr",
101
- ] as const;
97
+ const ALIAS_SUBPATHS = SOURCE_EXPORT_SUBPATHS;
102
98
 
103
99
  /**
104
100
  * Generate aliases to map virtual package paths to the actual published package.
@@ -0,0 +1,221 @@
1
+ import { createHash } from "node:crypto";
2
+ import {
3
+ copyFileSync,
4
+ existsSync,
5
+ mkdirSync,
6
+ rmSync,
7
+ statSync,
8
+ writeFileSync,
9
+ } from "node:fs";
10
+ import { resolve } from "node:path";
11
+
12
+ /**
13
+ * Escape special RegExp characters in a string for safe interpolation
14
+ * into new RegExp() patterns.
15
+ */
16
+ export function escapeRegExp(str: string): string {
17
+ return str.replace(/[.*+?^${}()|[\]\\]/g, "\\$&");
18
+ }
19
+
20
+ /**
21
+ * Encode route param values for path interpolation while preserving path
22
+ * separators for wildcard params (splat-style values can include `/`).
23
+ */
24
+ export function encodePathParam(value: unknown): string {
25
+ return String(value)
26
+ .split("/")
27
+ .map((segment) => encodeURIComponent(segment))
28
+ .join("/");
29
+ }
30
+
31
+ /**
32
+ * Substitute route params into a pattern, stripping constraint and optional
33
+ * syntax (:param(a|b)? -> value). Also handles wildcard params (*key).
34
+ * Optional params not present in `params` are removed from the output.
35
+ */
36
+ export function substituteRouteParams(
37
+ pattern: string,
38
+ params: Record<string, string>,
39
+ encode: (value: string) => string = encodeURIComponent,
40
+ ): string {
41
+ let result = pattern;
42
+ let hadOmittedOptional = false;
43
+
44
+ // First pass: substitute provided params.
45
+ // Empty string on an optional placeholder is treated as omitted (the trie
46
+ // matcher fills unmatched optionals with "" — letting the second pass
47
+ // strip them keeps slash cleanup consistent). Empty string on required
48
+ // `:key` or wildcard `*key` still substitutes, matching prior behaviour.
49
+ for (const [key, value] of Object.entries(params)) {
50
+ const escaped = escapeRegExp(key);
51
+ if (value === "") {
52
+ // Only replace required placeholders (negative lookahead for `?`);
53
+ // leave `:key?` for the second pass.
54
+ result = result.replace(
55
+ new RegExp(`:${escaped}(\\([^)]*\\))?(?!\\?)`),
56
+ "",
57
+ );
58
+ result = result.replace(`*${key}`, "");
59
+ } else {
60
+ result = result.replace(
61
+ new RegExp(`:${escaped}(\\([^)]*\\))?\\??`),
62
+ encode(value),
63
+ );
64
+ result = result.replace(`*${key}`, encode(value));
65
+ }
66
+ }
67
+
68
+ // Second pass: strip remaining optional param placeholders not in params
69
+ result = result.replace(/:([a-zA-Z_][a-zA-Z0-9_]*)(\([^)]*\))?\?/g, () => {
70
+ hadOmittedOptional = true;
71
+ return "";
72
+ });
73
+
74
+ // Clean up slashes from omitted optional segments
75
+ if (hadOmittedOptional) {
76
+ const hadTrailingSlash = pattern.length > 1 && pattern.endsWith("/");
77
+ result = result.replace(/\/\/+/g, "/").replace(/\/+$/, "") || "/";
78
+ if (hadTrailingSlash && !result.endsWith("/")) result += "/";
79
+ }
80
+
81
+ return result;
82
+ }
83
+
84
+ /**
85
+ * Run an async function over items with bounded concurrency.
86
+ * Errors propagate immediately and abort remaining work.
87
+ */
88
+ export async function runWithConcurrency<T>(
89
+ items: T[],
90
+ concurrency: number,
91
+ fn: (item: T) => Promise<void>,
92
+ ): Promise<void> {
93
+ const limit = Math.max(1, Math.min(concurrency, items.length));
94
+ if (limit <= 1) {
95
+ for (const item of items) await fn(item);
96
+ return;
97
+ }
98
+ let nextIndex = 0;
99
+ async function worker() {
100
+ while (nextIndex < items.length) {
101
+ const idx = nextIndex++;
102
+ await fn(items[idx]);
103
+ }
104
+ }
105
+ await Promise.all(Array.from({ length: limit }, () => worker()));
106
+ }
107
+
108
+ /**
109
+ * Group prerender entries by their concurrency setting so each group
110
+ * can be rendered with the appropriate parallelism.
111
+ */
112
+ export function groupByConcurrency<T extends { concurrency: number }>(
113
+ entries: T[],
114
+ ): { concurrency: number; entries: T[] }[] {
115
+ const map = new Map<number, T[]>();
116
+ for (const entry of entries) {
117
+ const key = entry.concurrency;
118
+ let group = map.get(key);
119
+ if (!group) {
120
+ group = [];
121
+ map.set(key, group);
122
+ }
123
+ group.push(entry);
124
+ }
125
+ return Array.from(map.entries(), ([concurrency, items]) => ({
126
+ concurrency,
127
+ entries: items,
128
+ }));
129
+ }
130
+
131
+ /**
132
+ * Notify all routers' onError callbacks about a build-time error.
133
+ * Uses a synthetic request since there is no real request during build.
134
+ */
135
+ export function notifyOnError(
136
+ registry: Map<string, any>,
137
+ error: unknown,
138
+ phase: "prerender" | "static",
139
+ routeKey?: string,
140
+ pathname?: string,
141
+ skipped?: boolean,
142
+ ): void {
143
+ for (const [, routerInstance] of registry) {
144
+ const onError = routerInstance.onError;
145
+ if (!onError) continue;
146
+
147
+ const errorObj = error instanceof Error ? error : new Error(String(error));
148
+ const syntheticUrl = new URL("http://prerender" + (pathname || "/"));
149
+ const context = {
150
+ error: errorObj,
151
+ phase,
152
+ request: new Request(syntheticUrl),
153
+ url: syntheticUrl,
154
+ pathname: syntheticUrl.pathname,
155
+ method: "GET",
156
+ routeKey,
157
+ metadata: skipped ? { skipped: true } : undefined,
158
+ };
159
+
160
+ try {
161
+ const result = onError(context);
162
+ if (result instanceof Promise) {
163
+ result.catch((cbErr: unknown) => {
164
+ console.error(`[Build.onError] Callback error:`, cbErr);
165
+ });
166
+ }
167
+ } catch (cbErr) {
168
+ console.error(`[Build.onError] Callback error:`, cbErr);
169
+ }
170
+ break; // Only notify the first router with onError
171
+ }
172
+ }
173
+
174
+ function getStagedAssetDir(projectRoot: string): string {
175
+ return resolve(projectRoot, "node_modules/.rangojs-router-build/rsc-assets");
176
+ }
177
+
178
+ export function resetStagedBuildAssets(projectRoot: string): void {
179
+ rmSync(getStagedAssetDir(projectRoot), { recursive: true, force: true });
180
+ }
181
+
182
+ export function stageBuildAssetModule(
183
+ projectRoot: string,
184
+ prefix: "__pr" | "__st",
185
+ exportValue: string,
186
+ ): string {
187
+ const stagedDir = getStagedAssetDir(projectRoot);
188
+ mkdirSync(stagedDir, { recursive: true });
189
+
190
+ const contentHash = createHash("sha256")
191
+ .update(exportValue)
192
+ .digest("hex")
193
+ .slice(0, 8);
194
+ const fileName = `${prefix}-${contentHash}.js`;
195
+ const filePath = resolve(stagedDir, fileName);
196
+
197
+ if (!existsSync(filePath)) {
198
+ writeFileSync(filePath, `export default ${exportValue};\n`);
199
+ }
200
+
201
+ return fileName;
202
+ }
203
+
204
+ export function copyStagedBuildAssets(
205
+ projectRoot: string,
206
+ fileNames: Iterable<string>,
207
+ ): number {
208
+ const stagedDir = getStagedAssetDir(projectRoot);
209
+ const distAssetsDir = resolve(projectRoot, "dist/rsc/assets");
210
+ mkdirSync(distAssetsDir, { recursive: true });
211
+
212
+ let totalBytes = 0;
213
+ for (const fileName of new Set(fileNames)) {
214
+ const stagedPath = resolve(stagedDir, fileName);
215
+ const distPath = resolve(distAssetsDir, fileName);
216
+ copyFileSync(stagedPath, distPath);
217
+ totalBytes += statSync(stagedPath).size;
218
+ }
219
+
220
+ return totalBytes;
221
+ }
@@ -0,0 +1,170 @@
1
+ import type { Plugin } from "vite";
2
+ import * as Vite from "vite";
3
+ import { getPublishedPackageName } from "./package-resolution.js";
4
+ import { performanceTracksOptimizeDepsPlugin } from "../plugins/performance-tracks.js";
5
+ import {
6
+ VIRTUAL_ENTRY_BROWSER,
7
+ VIRTUAL_ENTRY_SSR,
8
+ getVirtualEntryRSC,
9
+ VIRTUAL_IDS,
10
+ } from "../plugins/virtual-entries.js";
11
+
12
+ /**
13
+ * esbuild plugin to provide rsc-router:version virtual module during optimization.
14
+ * This is needed because esbuild runs during Vite's dependency optimization phase,
15
+ * before Vite's plugin system can handle virtual modules.
16
+ */
17
+ const versionEsbuildPlugin = {
18
+ name: "@rangojs/router-version",
19
+ setup(build: any): void {
20
+ build.onResolve({ filter: /^rsc-router:version$/ }, (args: any) => ({
21
+ path: args.path,
22
+ namespace: "@rangojs/router-virtual",
23
+ }));
24
+ build.onLoad(
25
+ { filter: /.*/, namespace: "@rangojs/router-virtual" },
26
+ () => ({
27
+ contents: `export const VERSION = "dev";`,
28
+ loader: "js",
29
+ }),
30
+ );
31
+ },
32
+ };
33
+
34
+ /**
35
+ * Shared esbuild options for dependency optimization.
36
+ * Includes the version stub plugin for all environments.
37
+ */
38
+ export const sharedEsbuildOptions: {
39
+ plugins: any[];
40
+ } = {
41
+ plugins: [versionEsbuildPlugin, performanceTracksOptimizeDepsPlugin()],
42
+ };
43
+
44
+ /**
45
+ * Create a virtual modules plugin for default entry files.
46
+ * Provides virtual module content when entries use VIRTUAL_IDS (no custom entry configured).
47
+ */
48
+ export function createVirtualEntriesPlugin(
49
+ entries: { client: string; ssr: string; rsc?: string },
50
+ routerPathRef?: { path?: string },
51
+ ): Plugin {
52
+ // Build virtual modules map based on which entries use virtual IDs
53
+ const virtualModules: Record<string, string> = {};
54
+
55
+ if (entries.client === VIRTUAL_IDS.browser) {
56
+ virtualModules[VIRTUAL_IDS.browser] = VIRTUAL_ENTRY_BROWSER;
57
+ }
58
+ if (entries.ssr === VIRTUAL_IDS.ssr) {
59
+ virtualModules[VIRTUAL_IDS.ssr] = VIRTUAL_ENTRY_SSR;
60
+ }
61
+
62
+ // RSC entry is resolved lazily in load() because routerPath may be
63
+ // set after plugin creation (e.g. by the auto-discover config() hook).
64
+ // Track all known virtual IDs for resolveId (content is separate).
65
+ const knownIds = new Set(Object.keys(virtualModules));
66
+ if (entries.rsc === VIRTUAL_IDS.rsc) {
67
+ knownIds.add(VIRTUAL_IDS.rsc);
68
+ }
69
+
70
+ return {
71
+ name: "@rangojs/router:virtual-entries",
72
+ enforce: "pre",
73
+
74
+ resolveId(id) {
75
+ if (knownIds.has(id)) {
76
+ return "\0" + id;
77
+ }
78
+ // Handle if the id already has the null prefix (RSC plugin wrapper imports)
79
+ if (id.startsWith("\0") && knownIds.has(id.slice(1))) {
80
+ return id;
81
+ }
82
+ return null;
83
+ },
84
+
85
+ load(id) {
86
+ if (id.startsWith("\0virtual:rsc-router/")) {
87
+ const virtualId = id.slice(1);
88
+ if (virtualId in virtualModules) {
89
+ return virtualModules[virtualId];
90
+ }
91
+ // Lazy RSC entry: routerPath may have been set by a config() hook
92
+ if (virtualId === VIRTUAL_IDS.rsc && routerPathRef?.path) {
93
+ const raw = routerPathRef.path.startsWith(".")
94
+ ? "/" + routerPathRef.path.slice(2) // ./src/router.tsx -> /src/router.tsx
95
+ : routerPathRef.path;
96
+ // Normalize backslashes for Windows (path.join/slice preserve native separators)
97
+ const absoluteRouterPath = raw.replaceAll("\\", "/");
98
+ return getVirtualEntryRSC(absoluteRouterPath);
99
+ }
100
+ }
101
+ return null;
102
+ },
103
+ };
104
+ }
105
+
106
+ /**
107
+ * Rollup onwarn handler that suppresses known harmless warnings:
108
+ * - "use client" directives: handled by the RSC plugin, not relevant to Rollup
109
+ * - sourcemap errors: caused by "use client" directive at line 1:0 confusing sourcemap resolution
110
+ * - sourcemap incomplete: plugins that transform without generating sourcemaps (router + RSC plugin)
111
+ * - dynamic/static mixed imports: expected for router internals (e.g. request-context, cache-scope)
112
+ * - empty bundle: @vitejs/plugin-rsc scan build (step 1/5) produces an empty "index" chunk
113
+ * because the RSC entry is fully externalized during client-reference analysis
114
+ */
115
+ export function onwarn(
116
+ warning: Vite.Rollup.RollupLog,
117
+ defaultHandler: (warning: Vite.Rollup.RollupLog) => void,
118
+ ): void {
119
+ if (
120
+ warning.code === "MODULE_LEVEL_DIRECTIVE" ||
121
+ warning.code === "SOURCEMAP_ERROR" ||
122
+ warning.code === "EMPTY_BUNDLE"
123
+ ) {
124
+ return;
125
+ }
126
+ // @vitejs/plugin-rsc@0.5.14: rsc:virtual:vite-rsc/assets-manifest renderChunk
127
+ // returns { code } without map, causing Rollup to warn about incorrect sourcemaps.
128
+ // This is harmless (simple string replacement). Remove this suppression if a
129
+ // future version of @vitejs/plugin-rsc fixes the missing sourcemap.
130
+ if (warning.message?.includes("Sourcemap is likely to be incorrect")) {
131
+ return;
132
+ }
133
+ if (
134
+ warning.plugin === "vite:reporter" &&
135
+ warning.message?.includes(
136
+ "dynamic import will not move module into another chunk",
137
+ )
138
+ ) {
139
+ return;
140
+ }
141
+ defaultHandler(warning);
142
+ }
143
+
144
+ /**
145
+ * Manual chunks configuration for client build.
146
+ * Splits React and router packages into separate chunks for better caching.
147
+ */
148
+ export function getManualChunks(id: string): string | undefined {
149
+ const normalized = Vite.normalizePath(id);
150
+
151
+ if (
152
+ normalized.includes("node_modules/react/") ||
153
+ normalized.includes("node_modules/react-dom/") ||
154
+ normalized.includes("node_modules/react-server-dom-webpack/") ||
155
+ normalized.includes("node_modules/@vitejs/plugin-rsc/")
156
+ ) {
157
+ return "react";
158
+ }
159
+ // Use dynamic package name from package.json
160
+ // Check both npm install path and workspace symlink resolved path
161
+ const packageName = getPublishedPackageName();
162
+ if (
163
+ normalized.includes(`node_modules/${packageName}/`) ||
164
+ normalized.includes("packages/rsc-router/") ||
165
+ normalized.includes("packages/rangojs-router/")
166
+ ) {
167
+ return "router";
168
+ }
169
+ return undefined;
170
+ }
package/CLAUDE.md DELETED
@@ -1,43 +0,0 @@
1
- # @rangojs/router
2
-
3
- Run `/rango` first to understand the API. Skills are in `node_modules/@rangojs/router/skills/`.
4
-
5
- ## Tree-Structure-Critical Files (DO NOT MODIFY without understanding)
6
-
7
- The following files control the React tree structure. Changing the tree structure
8
- (element types, nesting depth, or keys at any position) between SSR, navigation,
9
- and action renders will cause React to remount components, destroying client state
10
- like `useActionState`, refs, and local state. This is extremely hard to debug.
11
-
12
- **Protected files:**
13
-
14
- - `src/segment-system.tsx` - `renderSegments()` builds the React tree from segments.
15
- The `loading` property determines tree structure:
16
- - `undefined` / `null` -> OutletProvider directly (no boundary)
17
- - `false` -> LoaderBoundary + OutletProvider (boundary, no RouteContentWrapper)
18
- - truthy (ReactNode) -> LoaderBoundary + OutletProvider + RouteContentWrapper
19
-
20
- - `src/route-content-wrapper.tsx` - `LoaderBoundary` and `RouteContentWrapper`.
21
- These add structural depth (Suspense boundaries) to the React tree.
22
-
23
- - `src/browser/server-action-bridge.ts` - Merges server action segments with
24
- cached segments. Must preserve cached `loading` values to prevent tree drift.
25
-
26
- - `src/browser/partial-update.ts` - Merges navigation segments with cached segments.
27
-
28
- **Rules:**
29
-
30
- 1. Never change the conditional logic in `renderSegments()` that decides between
31
- LoaderBoundary/RouteContentWrapper/OutletProvider without verifying all three
32
- render paths (SSR, navigation, action) produce identical tree structures.
33
-
34
- 2. Never add or remove wrapper elements (Suspense, div, Fragment) around segment
35
- content without checking that the same wrappers exist in ALL render paths.
36
-
37
- 3. When merging segments (action bridge, partial update), always preserve the
38
- cached `loading` value if it differs from the server value. The server may
39
- return different `loading` values based on `isSSR` context.
40
-
41
- 4. Run `pnpm --filter @rangojs/router exec playwright test loader-behavior` after
42
- any changes to these files. The skipSSR action tests specifically catch tree
43
- structure regressions.
@@ -1,69 +0,0 @@
1
- /**
2
- * Simple LRU (Least Recently Used) cache implementation
3
- * Used for caching navigation segments to enable instant back/forward
4
- */
5
- export class LRUCache<K, V> {
6
- private cache = new Map<K, V>();
7
- private maxSize: number;
8
-
9
- constructor(maxSize: number) {
10
- this.maxSize = maxSize;
11
- }
12
-
13
- get(key: K): V | undefined {
14
- if (!this.cache.has(key)) {
15
- return undefined;
16
- }
17
-
18
- // Move to end (most recently used)
19
- const value = this.cache.get(key)!;
20
- this.cache.delete(key);
21
- this.cache.set(key, value);
22
- return value;
23
- }
24
-
25
- set(key: K, value: V): void {
26
- // If key exists, delete it first to update position
27
- if (this.cache.has(key)) {
28
- this.cache.delete(key);
29
- }
30
-
31
- this.cache.set(key, value);
32
-
33
- // Evict oldest entries if over capacity
34
- while (this.cache.size > this.maxSize) {
35
- const oldestKey = this.cache.keys().next().value;
36
- if (oldestKey !== undefined) {
37
- this.cache.delete(oldestKey);
38
- }
39
- }
40
- }
41
-
42
- has(key: K): boolean {
43
- if (!this.cache.has(key)) {
44
- return false;
45
- }
46
-
47
- // Move to end (most recently used) - same as get()
48
- const value = this.cache.get(key)!;
49
- this.cache.delete(key);
50
- this.cache.set(key, value);
51
- return true;
52
- }
53
-
54
- delete(key: K): boolean {
55
- return this.cache.delete(key);
56
- }
57
-
58
- clear(): void {
59
- this.cache.clear();
60
- }
61
-
62
- keys(): IterableIterator<K> {
63
- return this.cache.keys();
64
- }
65
-
66
- get size(): number {
67
- return this.cache.size;
68
- }
69
- }