@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,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
+ }
@@ -0,0 +1,121 @@
1
+ /**
2
+ * Package Resolution Utilities
3
+ *
4
+ * Handles detection of workspace vs npm install context and generates
5
+ * appropriate aliases and exclude lists for Vite configuration.
6
+ */
7
+
8
+ import { existsSync } from "node:fs";
9
+ import { resolve } from "node:path";
10
+ import packageJson from "../../../package.json" with { type: "json" };
11
+
12
+ /**
13
+ * The canonical name used in virtual entries (without scope)
14
+ */
15
+ const VIRTUAL_PACKAGE_NAME = "@rangojs/router";
16
+
17
+ /**
18
+ * Get the published package name (e.g., "@rangojs/router")
19
+ */
20
+ export function getPublishedPackageName(): string {
21
+ return packageJson.name;
22
+ }
23
+
24
+ /**
25
+ * Check if the package is installed from npm (scoped) vs workspace (unscoped)
26
+ *
27
+ * In workspace development:
28
+ * - Package is installed as "@rangojs/router" via pnpm workspace alias
29
+ * - The scoped name (@rangojs/router) doesn't exist in node_modules
30
+ *
31
+ * When installed from npm:
32
+ * - Package is installed as "@rangojs/router"
33
+ * - We need aliases to map "@rangojs/router/*" to "@rangojs/router/*"
34
+ */
35
+ export function isInstalledFromNpm(): boolean {
36
+ const packageName = getPublishedPackageName();
37
+ // Check if the scoped package exists in node_modules
38
+ return existsSync(resolve(process.cwd(), "node_modules", packageName));
39
+ }
40
+
41
+ /**
42
+ * Check if we're in a monorepo/workspace development context
43
+ */
44
+ export function isWorkspaceDevelopment(): boolean {
45
+ return !isInstalledFromNpm();
46
+ }
47
+
48
+ /**
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.
54
+ */
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(/^\./, ""));
69
+
70
+ /**
71
+ * Generate the list of modules to exclude from Vite's dependency optimization.
72
+ *
73
+ * We include both the published name and the virtual name because
74
+ * Vite's optimizer runs before alias resolution.
75
+ */
76
+ export function getExcludeDeps(): string[] {
77
+ const packageName = getPublishedPackageName();
78
+ const excludes: string[] = [];
79
+
80
+ for (const subpath of SOURCE_EXPORT_SUBPATHS) {
81
+ // Add scoped package paths
82
+ excludes.push(`${packageName}${subpath}`);
83
+ // Add virtual/aliased paths (before alias resolution)
84
+ if (packageName !== VIRTUAL_PACKAGE_NAME) {
85
+ excludes.push(`${VIRTUAL_PACKAGE_NAME}${subpath}`);
86
+ }
87
+ }
88
+
89
+ return excludes;
90
+ }
91
+
92
+ /**
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.
96
+ */
97
+ const ALIAS_SUBPATHS = SOURCE_EXPORT_SUBPATHS;
98
+
99
+ /**
100
+ * Generate aliases to map virtual package paths to the actual published package.
101
+ *
102
+ * Only needed when installed from npm, where the package is under @rangojs/router
103
+ * but virtual entries import from rsc-router/*.
104
+ *
105
+ * Returns empty object in workspace development where rsc-router resolves directly.
106
+ */
107
+ export function getPackageAliases(): Record<string, string> {
108
+ if (isWorkspaceDevelopment()) {
109
+ // No aliases needed - rsc-router resolves directly
110
+ return {};
111
+ }
112
+
113
+ const packageName = getPublishedPackageName();
114
+ const aliases: Record<string, string> = {};
115
+
116
+ for (const subpath of ALIAS_SUBPATHS) {
117
+ aliases[`${VIRTUAL_PACKAGE_NAME}${subpath}`] = `${packageName}${subpath}`;
118
+ }
119
+
120
+ return aliases;
121
+ }
@@ -0,0 +1,189 @@
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
+ */
35
+ export function substituteRouteParams(
36
+ pattern: string,
37
+ params: Record<string, string>,
38
+ encode: (value: string) => string = encodeURIComponent,
39
+ ): string {
40
+ let result = pattern;
41
+ for (const [key, value] of Object.entries(params)) {
42
+ const escaped = escapeRegExp(key);
43
+ result = result.replace(
44
+ new RegExp(`:${escaped}(\\([^)]*\\))?\\??`),
45
+ encode(value),
46
+ );
47
+ result = result.replace(`*${key}`, encode(value));
48
+ }
49
+ return result;
50
+ }
51
+
52
+ /**
53
+ * Run an async function over items with bounded concurrency.
54
+ * Errors propagate immediately and abort remaining work.
55
+ */
56
+ export async function runWithConcurrency<T>(
57
+ items: T[],
58
+ concurrency: number,
59
+ fn: (item: T) => Promise<void>,
60
+ ): Promise<void> {
61
+ const limit = Math.max(1, Math.min(concurrency, items.length));
62
+ if (limit <= 1) {
63
+ for (const item of items) await fn(item);
64
+ return;
65
+ }
66
+ let nextIndex = 0;
67
+ async function worker() {
68
+ while (nextIndex < items.length) {
69
+ const idx = nextIndex++;
70
+ await fn(items[idx]);
71
+ }
72
+ }
73
+ await Promise.all(Array.from({ length: limit }, () => worker()));
74
+ }
75
+
76
+ /**
77
+ * Group prerender entries by their concurrency setting so each group
78
+ * can be rendered with the appropriate parallelism.
79
+ */
80
+ export function groupByConcurrency<T extends { concurrency: number }>(
81
+ entries: T[],
82
+ ): { concurrency: number; entries: T[] }[] {
83
+ const map = new Map<number, T[]>();
84
+ for (const entry of entries) {
85
+ const key = entry.concurrency;
86
+ let group = map.get(key);
87
+ if (!group) {
88
+ group = [];
89
+ map.set(key, group);
90
+ }
91
+ group.push(entry);
92
+ }
93
+ return Array.from(map.entries(), ([concurrency, items]) => ({
94
+ concurrency,
95
+ entries: items,
96
+ }));
97
+ }
98
+
99
+ /**
100
+ * Notify all routers' onError callbacks about a build-time error.
101
+ * Uses a synthetic request since there is no real request during build.
102
+ */
103
+ export function notifyOnError(
104
+ registry: Map<string, any>,
105
+ error: unknown,
106
+ phase: "prerender" | "static",
107
+ routeKey?: string,
108
+ pathname?: string,
109
+ skipped?: boolean,
110
+ ): void {
111
+ for (const [, routerInstance] of registry) {
112
+ const onError = routerInstance.onError;
113
+ if (!onError) continue;
114
+
115
+ const errorObj = error instanceof Error ? error : new Error(String(error));
116
+ const syntheticUrl = new URL("http://prerender" + (pathname || "/"));
117
+ const context = {
118
+ error: errorObj,
119
+ phase,
120
+ request: new Request(syntheticUrl),
121
+ url: syntheticUrl,
122
+ pathname: syntheticUrl.pathname,
123
+ method: "GET",
124
+ routeKey,
125
+ metadata: skipped ? { skipped: true } : undefined,
126
+ };
127
+
128
+ try {
129
+ const result = onError(context);
130
+ if (result instanceof Promise) {
131
+ result.catch((cbErr: unknown) => {
132
+ console.error(`[Build.onError] Callback error:`, cbErr);
133
+ });
134
+ }
135
+ } catch (cbErr) {
136
+ console.error(`[Build.onError] Callback error:`, cbErr);
137
+ }
138
+ break; // Only notify the first router with onError
139
+ }
140
+ }
141
+
142
+ function getStagedAssetDir(projectRoot: string): string {
143
+ return resolve(projectRoot, "node_modules/.rangojs-router-build/rsc-assets");
144
+ }
145
+
146
+ export function resetStagedBuildAssets(projectRoot: string): void {
147
+ rmSync(getStagedAssetDir(projectRoot), { recursive: true, force: true });
148
+ }
149
+
150
+ export function stageBuildAssetModule(
151
+ projectRoot: string,
152
+ prefix: "__pr" | "__st",
153
+ exportValue: string,
154
+ ): string {
155
+ const stagedDir = getStagedAssetDir(projectRoot);
156
+ mkdirSync(stagedDir, { recursive: true });
157
+
158
+ const contentHash = createHash("sha256")
159
+ .update(exportValue)
160
+ .digest("hex")
161
+ .slice(0, 8);
162
+ const fileName = `${prefix}-${contentHash}.js`;
163
+ const filePath = resolve(stagedDir, fileName);
164
+
165
+ if (!existsSync(filePath)) {
166
+ writeFileSync(filePath, `export default ${exportValue};\n`);
167
+ }
168
+
169
+ return fileName;
170
+ }
171
+
172
+ export function copyStagedBuildAssets(
173
+ projectRoot: string,
174
+ fileNames: Iterable<string>,
175
+ ): number {
176
+ const stagedDir = getStagedAssetDir(projectRoot);
177
+ const distAssetsDir = resolve(projectRoot, "dist/rsc/assets");
178
+ mkdirSync(distAssetsDir, { recursive: true });
179
+
180
+ let totalBytes = 0;
181
+ for (const fileName of new Set(fileNames)) {
182
+ const stagedPath = resolve(stagedDir, fileName);
183
+ const distPath = resolve(distAssetsDir, fileName);
184
+ copyFileSync(stagedPath, distPath);
185
+ totalBytes += statSync(stagedPath).size;
186
+ }
187
+
188
+ return totalBytes;
189
+ }
@@ -0,0 +1,169 @@
1
+ import type { Plugin } from "vite";
2
+ import * as Vite from "vite";
3
+ import { getPublishedPackageName } from "./package-resolution.js";
4
+ import {
5
+ VIRTUAL_ENTRY_BROWSER,
6
+ VIRTUAL_ENTRY_SSR,
7
+ getVirtualEntryRSC,
8
+ VIRTUAL_IDS,
9
+ } from "../plugins/virtual-entries.js";
10
+
11
+ /**
12
+ * esbuild plugin to provide rsc-router:version virtual module during optimization.
13
+ * This is needed because esbuild runs during Vite's dependency optimization phase,
14
+ * before Vite's plugin system can handle virtual modules.
15
+ */
16
+ const versionEsbuildPlugin = {
17
+ name: "@rangojs/router-version",
18
+ setup(build: any): void {
19
+ build.onResolve({ filter: /^rsc-router:version$/ }, (args: any) => ({
20
+ path: args.path,
21
+ namespace: "@rangojs/router-virtual",
22
+ }));
23
+ build.onLoad(
24
+ { filter: /.*/, namespace: "@rangojs/router-virtual" },
25
+ () => ({
26
+ contents: `export const VERSION = "dev";`,
27
+ loader: "js",
28
+ }),
29
+ );
30
+ },
31
+ };
32
+
33
+ /**
34
+ * Shared esbuild options for dependency optimization.
35
+ * Includes the version stub plugin for all environments.
36
+ */
37
+ export const sharedEsbuildOptions: {
38
+ plugins: (typeof versionEsbuildPlugin)[];
39
+ } = {
40
+ plugins: [versionEsbuildPlugin],
41
+ };
42
+
43
+ /**
44
+ * Create a virtual modules plugin for default entry files.
45
+ * Provides virtual module content when entries use VIRTUAL_IDS (no custom entry configured).
46
+ */
47
+ export function createVirtualEntriesPlugin(
48
+ entries: { client: string; ssr: string; rsc?: string },
49
+ routerPathRef?: { path?: string },
50
+ ): Plugin {
51
+ // Build virtual modules map based on which entries use virtual IDs
52
+ const virtualModules: Record<string, string> = {};
53
+
54
+ if (entries.client === VIRTUAL_IDS.browser) {
55
+ virtualModules[VIRTUAL_IDS.browser] = VIRTUAL_ENTRY_BROWSER;
56
+ }
57
+ if (entries.ssr === VIRTUAL_IDS.ssr) {
58
+ virtualModules[VIRTUAL_IDS.ssr] = VIRTUAL_ENTRY_SSR;
59
+ }
60
+
61
+ // RSC entry is resolved lazily in load() because routerPath may be
62
+ // set after plugin creation (e.g. by the auto-discover config() hook).
63
+ // Track all known virtual IDs for resolveId (content is separate).
64
+ const knownIds = new Set(Object.keys(virtualModules));
65
+ if (entries.rsc === VIRTUAL_IDS.rsc) {
66
+ knownIds.add(VIRTUAL_IDS.rsc);
67
+ }
68
+
69
+ return {
70
+ name: "@rangojs/router:virtual-entries",
71
+ enforce: "pre",
72
+
73
+ resolveId(id) {
74
+ if (knownIds.has(id)) {
75
+ return "\0" + id;
76
+ }
77
+ // Handle if the id already has the null prefix (RSC plugin wrapper imports)
78
+ if (id.startsWith("\0") && knownIds.has(id.slice(1))) {
79
+ return id;
80
+ }
81
+ return null;
82
+ },
83
+
84
+ load(id) {
85
+ if (id.startsWith("\0virtual:rsc-router/")) {
86
+ const virtualId = id.slice(1);
87
+ if (virtualId in virtualModules) {
88
+ return virtualModules[virtualId];
89
+ }
90
+ // Lazy RSC entry: routerPath may have been set by a config() hook
91
+ if (virtualId === VIRTUAL_IDS.rsc && routerPathRef?.path) {
92
+ const raw = routerPathRef.path.startsWith(".")
93
+ ? "/" + routerPathRef.path.slice(2) // ./src/router.tsx -> /src/router.tsx
94
+ : routerPathRef.path;
95
+ // Normalize backslashes for Windows (path.join/slice preserve native separators)
96
+ const absoluteRouterPath = raw.replaceAll("\\", "/");
97
+ return getVirtualEntryRSC(absoluteRouterPath);
98
+ }
99
+ }
100
+ return null;
101
+ },
102
+ };
103
+ }
104
+
105
+ /**
106
+ * Rollup onwarn handler that suppresses known harmless warnings:
107
+ * - "use client" directives: handled by the RSC plugin, not relevant to Rollup
108
+ * - sourcemap errors: caused by "use client" directive at line 1:0 confusing sourcemap resolution
109
+ * - sourcemap incomplete: plugins that transform without generating sourcemaps (router + RSC plugin)
110
+ * - dynamic/static mixed imports: expected for router internals (e.g. request-context, cache-scope)
111
+ * - empty bundle: @vitejs/plugin-rsc scan build (step 1/5) produces an empty "index" chunk
112
+ * because the RSC entry is fully externalized during client-reference analysis
113
+ */
114
+ export function onwarn(
115
+ warning: Vite.Rollup.RollupLog,
116
+ defaultHandler: (warning: Vite.Rollup.RollupLog) => void,
117
+ ): void {
118
+ if (
119
+ warning.code === "MODULE_LEVEL_DIRECTIVE" ||
120
+ warning.code === "SOURCEMAP_ERROR" ||
121
+ warning.code === "EMPTY_BUNDLE"
122
+ ) {
123
+ return;
124
+ }
125
+ // @vitejs/plugin-rsc@0.5.14: rsc:virtual:vite-rsc/assets-manifest renderChunk
126
+ // returns { code } without map, causing Rollup to warn about incorrect sourcemaps.
127
+ // This is harmless (simple string replacement). Remove this suppression if a
128
+ // future version of @vitejs/plugin-rsc fixes the missing sourcemap.
129
+ if (warning.message?.includes("Sourcemap is likely to be incorrect")) {
130
+ return;
131
+ }
132
+ if (
133
+ warning.plugin === "vite:reporter" &&
134
+ warning.message?.includes(
135
+ "dynamic import will not move module into another chunk",
136
+ )
137
+ ) {
138
+ return;
139
+ }
140
+ defaultHandler(warning);
141
+ }
142
+
143
+ /**
144
+ * Manual chunks configuration for client build.
145
+ * Splits React and router packages into separate chunks for better caching.
146
+ */
147
+ export function getManualChunks(id: string): string | undefined {
148
+ const normalized = Vite.normalizePath(id);
149
+
150
+ if (
151
+ normalized.includes("node_modules/react/") ||
152
+ normalized.includes("node_modules/react-dom/") ||
153
+ normalized.includes("node_modules/react-server-dom-webpack/") ||
154
+ normalized.includes("node_modules/@vitejs/plugin-rsc/")
155
+ ) {
156
+ return "react";
157
+ }
158
+ // Use dynamic package name from package.json
159
+ // Check both npm install path and workspace symlink resolved path
160
+ const packageName = getPublishedPackageName();
161
+ if (
162
+ normalized.includes(`node_modules/${packageName}/`) ||
163
+ normalized.includes("packages/rsc-router/") ||
164
+ normalized.includes("packages/rangojs-router/")
165
+ ) {
166
+ return "router";
167
+ }
168
+ return undefined;
169
+ }