@rangojs/router 0.0.0-experimental.8874d8d2 → 0.0.0-experimental.89

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 (178) hide show
  1. package/AGENTS.md +4 -0
  2. package/README.md +126 -38
  3. package/dist/bin/rango.js +138 -50
  4. package/dist/vite/index.js +1705 -701
  5. package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  6. package/package.json +19 -16
  7. package/skills/breadcrumbs/SKILL.md +3 -1
  8. package/skills/cache-guide/SKILL.md +32 -0
  9. package/skills/caching/SKILL.md +45 -4
  10. package/skills/handler-use/SKILL.md +362 -0
  11. package/skills/hooks/SKILL.md +28 -20
  12. package/skills/intercept/SKILL.md +20 -0
  13. package/skills/layout/SKILL.md +22 -0
  14. package/skills/links/SKILL.md +91 -17
  15. package/skills/loader/SKILL.md +88 -45
  16. package/skills/middleware/SKILL.md +34 -3
  17. package/skills/migrate-nextjs/SKILL.md +560 -0
  18. package/skills/migrate-react-router/SKILL.md +765 -0
  19. package/skills/parallel/SKILL.md +185 -0
  20. package/skills/prerender/SKILL.md +110 -68
  21. package/skills/rango/SKILL.md +24 -22
  22. package/skills/response-routes/SKILL.md +8 -0
  23. package/skills/route/SKILL.md +55 -0
  24. package/skills/router-setup/SKILL.md +87 -2
  25. package/skills/streams-and-websockets/SKILL.md +283 -0
  26. package/skills/typesafety/SKILL.md +13 -1
  27. package/src/__internal.ts +1 -1
  28. package/src/browser/app-shell.ts +52 -0
  29. package/src/browser/app-version.ts +14 -0
  30. package/src/browser/event-controller.ts +5 -0
  31. package/src/browser/navigation-bridge.ts +90 -16
  32. package/src/browser/navigation-client.ts +167 -59
  33. package/src/browser/navigation-store.ts +68 -9
  34. package/src/browser/navigation-transaction.ts +11 -9
  35. package/src/browser/partial-update.ts +113 -17
  36. package/src/browser/prefetch/cache.ts +175 -15
  37. package/src/browser/prefetch/fetch.ts +180 -33
  38. package/src/browser/prefetch/queue.ts +123 -20
  39. package/src/browser/prefetch/resource-ready.ts +77 -0
  40. package/src/browser/rango-state.ts +53 -13
  41. package/src/browser/react/Link.tsx +81 -9
  42. package/src/browser/react/NavigationProvider.tsx +89 -14
  43. package/src/browser/react/context.ts +7 -2
  44. package/src/browser/react/use-handle.ts +9 -58
  45. package/src/browser/react/use-navigation.ts +22 -2
  46. package/src/browser/react/use-params.ts +11 -1
  47. package/src/browser/react/use-router.ts +29 -9
  48. package/src/browser/rsc-router.tsx +168 -65
  49. package/src/browser/scroll-restoration.ts +41 -42
  50. package/src/browser/segment-reconciler.ts +36 -9
  51. package/src/browser/server-action-bridge.ts +8 -6
  52. package/src/browser/types.ts +49 -5
  53. package/src/build/generate-manifest.ts +6 -6
  54. package/src/build/generate-route-types.ts +3 -0
  55. package/src/build/route-trie.ts +50 -24
  56. package/src/build/route-types/include-resolution.ts +8 -1
  57. package/src/build/route-types/router-processing.ts +223 -74
  58. package/src/build/route-types/scan-filter.ts +8 -1
  59. package/src/cache/cache-runtime.ts +15 -11
  60. package/src/cache/cache-scope.ts +48 -7
  61. package/src/cache/cf/cf-cache-store.ts +455 -15
  62. package/src/cache/cf/index.ts +5 -1
  63. package/src/cache/document-cache.ts +17 -7
  64. package/src/cache/index.ts +1 -0
  65. package/src/cache/taint.ts +55 -0
  66. package/src/client.tsx +84 -230
  67. package/src/context-var.ts +72 -2
  68. package/src/debug.ts +2 -2
  69. package/src/handle.ts +40 -0
  70. package/src/index.rsc.ts +6 -1
  71. package/src/index.ts +49 -6
  72. package/src/outlet-context.ts +1 -1
  73. package/src/prerender/store.ts +5 -4
  74. package/src/prerender.ts +138 -77
  75. package/src/response-utils.ts +28 -0
  76. package/src/reverse.ts +27 -2
  77. package/src/route-definition/dsl-helpers.ts +240 -40
  78. package/src/route-definition/helpers-types.ts +67 -19
  79. package/src/route-definition/index.ts +3 -0
  80. package/src/route-definition/redirect.ts +11 -3
  81. package/src/route-definition/resolve-handler-use.ts +155 -0
  82. package/src/route-map-builder.ts +7 -1
  83. package/src/route-types.ts +18 -0
  84. package/src/router/content-negotiation.ts +100 -1
  85. package/src/router/find-match.ts +4 -2
  86. package/src/router/handler-context.ts +101 -25
  87. package/src/router/intercept-resolution.ts +11 -4
  88. package/src/router/lazy-includes.ts +10 -7
  89. package/src/router/loader-resolution.ts +159 -21
  90. package/src/router/logging.ts +5 -2
  91. package/src/router/manifest.ts +31 -16
  92. package/src/router/match-api.ts +127 -192
  93. package/src/router/match-middleware/background-revalidation.ts +30 -2
  94. package/src/router/match-middleware/cache-lookup.ts +94 -17
  95. package/src/router/match-middleware/cache-store.ts +53 -10
  96. package/src/router/match-middleware/intercept-resolution.ts +9 -7
  97. package/src/router/match-middleware/segment-resolution.ts +61 -5
  98. package/src/router/match-result.ts +104 -10
  99. package/src/router/metrics.ts +6 -1
  100. package/src/router/middleware-types.ts +8 -30
  101. package/src/router/middleware.ts +36 -10
  102. package/src/router/navigation-snapshot.ts +182 -0
  103. package/src/router/pattern-matching.ts +60 -9
  104. package/src/router/prerender-match.ts +110 -10
  105. package/src/router/preview-match.ts +30 -102
  106. package/src/router/request-classification.ts +310 -0
  107. package/src/router/route-snapshot.ts +245 -0
  108. package/src/router/router-context.ts +6 -1
  109. package/src/router/router-interfaces.ts +36 -4
  110. package/src/router/router-options.ts +37 -11
  111. package/src/router/segment-resolution/fresh.ts +198 -20
  112. package/src/router/segment-resolution/helpers.ts +29 -24
  113. package/src/router/segment-resolution/loader-cache.ts +1 -0
  114. package/src/router/segment-resolution/revalidation.ts +437 -297
  115. package/src/router/segment-wrappers.ts +2 -0
  116. package/src/router/trie-matching.ts +10 -4
  117. package/src/router/types.ts +1 -0
  118. package/src/router/url-params.ts +49 -0
  119. package/src/router.ts +60 -8
  120. package/src/rsc/handler.ts +478 -374
  121. package/src/rsc/helpers.ts +69 -41
  122. package/src/rsc/loader-fetch.ts +23 -3
  123. package/src/rsc/manifest-init.ts +5 -1
  124. package/src/rsc/progressive-enhancement.ts +16 -2
  125. package/src/rsc/response-route-handler.ts +14 -1
  126. package/src/rsc/rsc-rendering.ts +17 -1
  127. package/src/rsc/server-action.ts +10 -0
  128. package/src/rsc/ssr-setup.ts +2 -2
  129. package/src/rsc/types.ts +9 -1
  130. package/src/segment-content-promise.ts +67 -0
  131. package/src/segment-loader-promise.ts +122 -0
  132. package/src/segment-system.tsx +109 -23
  133. package/src/server/context.ts +166 -17
  134. package/src/server/handle-store.ts +19 -0
  135. package/src/server/loader-registry.ts +9 -8
  136. package/src/server/request-context.ts +194 -60
  137. package/src/ssr/index.tsx +4 -0
  138. package/src/static-handler.ts +18 -6
  139. package/src/types/cache-types.ts +4 -4
  140. package/src/types/handler-context.ts +137 -65
  141. package/src/types/loader-types.ts +41 -15
  142. package/src/types/request-scope.ts +126 -0
  143. package/src/types/route-entry.ts +19 -1
  144. package/src/types/segments.ts +2 -0
  145. package/src/urls/include-helper.ts +24 -14
  146. package/src/urls/path-helper-types.ts +39 -6
  147. package/src/urls/path-helper.ts +48 -13
  148. package/src/urls/pattern-types.ts +12 -0
  149. package/src/urls/response-types.ts +18 -16
  150. package/src/use-loader.tsx +77 -5
  151. package/src/vite/debug.ts +184 -0
  152. package/src/vite/discovery/bundle-postprocess.ts +30 -33
  153. package/src/vite/discovery/discover-routers.ts +36 -4
  154. package/src/vite/discovery/prerender-collection.ts +175 -74
  155. package/src/vite/discovery/state.ts +13 -6
  156. package/src/vite/index.ts +4 -0
  157. package/src/vite/plugin-types.ts +51 -79
  158. package/src/vite/plugins/cjs-to-esm.ts +5 -0
  159. package/src/vite/plugins/client-ref-dedup.ts +16 -0
  160. package/src/vite/plugins/client-ref-hashing.ts +16 -4
  161. package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
  162. package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  163. package/src/vite/plugins/cloudflare-protocol-stub.ts +214 -0
  164. package/src/vite/plugins/expose-action-id.ts +53 -31
  165. package/src/vite/plugins/expose-id-utils.ts +12 -0
  166. package/src/vite/plugins/expose-ids/handler-transform.ts +30 -0
  167. package/src/vite/plugins/expose-ids/router-transform.ts +20 -3
  168. package/src/vite/plugins/expose-internal-ids.ts +563 -316
  169. package/src/vite/plugins/performance-tracks.ts +96 -0
  170. package/src/vite/plugins/refresh-cmd.ts +88 -26
  171. package/src/vite/plugins/use-cache-transform.ts +56 -43
  172. package/src/vite/plugins/version-plugin.ts +13 -1
  173. package/src/vite/rango.ts +204 -217
  174. package/src/vite/router-discovery.ts +393 -67
  175. package/src/vite/utils/banner.ts +4 -4
  176. package/src/vite/utils/package-resolution.ts +41 -1
  177. package/src/vite/utils/prerender-utils.ts +37 -5
  178. package/src/vite/utils/shared-utils.ts +3 -2
@@ -31,25 +31,25 @@ export function postprocessBundle(state: DiscoveryState): void {
31
31
  state.rscEntryFileName ?? "index.js",
32
32
  );
33
33
 
34
- // 1. Evict handler code from __prerender-handlers and __static-handlers chunks.
35
- // handlerChunkInfo/staticHandlerChunkInfo are populated by generateBundle
34
+ // 1. Evict handler code from whichever chunks contain handler exports.
35
+ // handlerChunkInfoMap/staticHandlerChunkInfoMap are populated by generateBundle
36
36
  // after the production RSC build. In Vite 6 multi-environment builds, the
37
- // RSC build runs twice (analysis + production). Chunk info is only available
38
- // after the production pass, so we run eviction whenever it becomes available.
37
+ // RSC build runs twice (analysis + production). The maps are cleared at the
38
+ // start of each generateBundle pass so only production data is used here.
39
39
  const evictionTargets: Array<{
40
- info: typeof state.handlerChunkInfo;
40
+ infos: Iterable<import("./state.js").ChunkInfo>;
41
41
  fnName: string;
42
42
  brand: string;
43
43
  label: string;
44
44
  }> = [
45
45
  {
46
- info: state.handlerChunkInfo,
46
+ infos: state.handlerChunkInfoMap.values(),
47
47
  fnName: "Prerender",
48
48
  brand: "prerenderHandler",
49
49
  label: "handler code from RSC bundle",
50
50
  },
51
51
  {
52
- info: state.staticHandlerChunkInfo,
52
+ infos: state.staticHandlerChunkInfoMap.values(),
53
53
  fnName: "Static",
54
54
  brand: "staticHandler",
55
55
  label: "static handler code",
@@ -57,35 +57,32 @@ export function postprocessBundle(state: DiscoveryState): void {
57
57
  ];
58
58
 
59
59
  for (const target of evictionTargets) {
60
- if (!target.info) continue;
61
- const chunkPath = resolve(
62
- state.projectRoot,
63
- "dist/rsc",
64
- target.info.fileName,
65
- );
66
- try {
67
- const code = readFileSync(chunkPath, "utf-8");
68
- const result = evictHandlerCode(
69
- code,
70
- target.info.exports,
71
- target.fnName,
72
- target.brand,
73
- );
74
- if (result) {
75
- writeFileSync(chunkPath, result.code);
76
- const savedKB = (result.savedBytes / 1024).toFixed(1);
77
- console.log(
78
- `[rsc-router] Evicted ${target.label} (${savedKB} KB saved): ${target.info.fileName}`,
60
+ for (const info of target.infos) {
61
+ const chunkPath = resolve(state.projectRoot, "dist/rsc", info.fileName);
62
+ try {
63
+ const code = readFileSync(chunkPath, "utf-8");
64
+ const result = evictHandlerCode(
65
+ code,
66
+ info.exports,
67
+ target.fnName,
68
+ target.brand,
69
+ );
70
+ if (result) {
71
+ writeFileSync(chunkPath, result.code);
72
+ const savedKB = (result.savedBytes / 1024).toFixed(1);
73
+ console.log(
74
+ `[rsc-router] Evicted ${target.label} (${savedKB} KB saved): ${info.fileName}`,
75
+ );
76
+ }
77
+ } catch (replaceErr: any) {
78
+ console.warn(
79
+ `[rsc-router] Failed to evict ${target.label}: ${replaceErr.message}`,
79
80
  );
80
81
  }
81
- } catch (replaceErr: any) {
82
- console.warn(
83
- `[rsc-router] Failed to evict ${target.label}: ${replaceErr.message}`,
84
- );
85
82
  }
86
83
  }
87
- state.handlerChunkInfo = null;
88
- state.staticHandlerChunkInfo = null;
84
+ state.handlerChunkInfoMap.clear();
85
+ state.staticHandlerChunkInfoMap.clear();
89
86
 
90
87
  // 2. Write prerender data as separate importable asset modules
91
88
  // and inject a lazy manifest loader into the RSC entry.
@@ -138,7 +135,7 @@ export function postprocessBundle(state: DiscoveryState): void {
138
135
  // and inject a __STATIC_MANIFEST import into the RSC entry.
139
136
  if (hasStaticData && existsSync(rscEntryPath)) {
140
137
  const rscCode = readFileSync(rscEntryPath, "utf-8");
141
- if (!rscCode.includes("__STATIC_MANIFEST")) {
138
+ if (!rscCode.includes("__static-manifest.js")) {
142
139
  try {
143
140
  const manifestEntries: string[] = [];
144
141
  let totalBytes = copyStagedBuildAssets(
@@ -20,6 +20,9 @@ import {
20
20
  expandPrerenderRoutes,
21
21
  renderStaticHandlers,
22
22
  } from "./prerender-collection.js";
23
+ import { createRangoDebugger, timed, NS } from "../debug.js";
24
+
25
+ const debug = createRangoDebugger(NS.discovery);
23
26
 
24
27
  /**
25
28
  * Import the user's entry via RSC runner, generate manifests for each
@@ -38,10 +41,16 @@ export async function discoverRouters(
38
41
  // Import the entry file via RSC environment.
39
42
  // For node preset: this is the router file (createRouter() registers in RouterRegistry).
40
43
  // For cloudflare preset: this is the worker entry (which imports the router).
41
- await rscEnv.runner.import(state.resolvedEntryPath);
44
+ await timed(debug, "inner: import entry", () =>
45
+ rscEnv.runner.import(state.resolvedEntryPath),
46
+ );
42
47
 
43
48
  // Import the router package to access the registry
44
- const serverMod = await rscEnv.runner.import("@rangojs/router/server");
49
+ const serverMod = await timed(
50
+ debug,
51
+ "inner: import @rangojs/router/server",
52
+ () => rscEnv.runner.import("@rangojs/router/server"),
53
+ );
45
54
  let registry: Map<string, any> = serverMod.RouterRegistry;
46
55
 
47
56
  if (!registry || registry.size === 0) {
@@ -100,9 +109,15 @@ export async function discoverRouters(
100
109
  }
101
110
 
102
111
  // Import build utilities for manifest generation
103
- const buildMod = await rscEnv.runner.import("@rangojs/router/build");
112
+ const buildMod = await timed(
113
+ debug,
114
+ "inner: import @rangojs/router/build",
115
+ () => rscEnv.runner.import("@rangojs/router/build"),
116
+ );
104
117
  const generateManifestFull = buildMod.generateManifestFull;
105
118
 
119
+ debug?.("inner: found %d router(s) in registry", registry.size);
120
+
106
121
  const nestedRouterConflict = findNestedRouterConflict(
107
122
  [...registry.values()]
108
123
  .map((router) => router.__sourceFile)
@@ -130,12 +145,17 @@ export async function discoverRouters(
130
145
  // Collect all manifests for trie building (avoid re-running generateManifest)
131
146
  const allManifests: Array<{ id: string; manifest: any }> = [];
132
147
 
148
+ const manifestGenStart = debug ? performance.now() : 0;
133
149
  for (const [id, router] of registry) {
134
150
  if (!router.urlpatterns || !generateManifestFull) {
135
151
  continue;
136
152
  }
137
153
 
138
- const manifest = generateManifestFull(router.urlpatterns, routerMountIndex);
154
+ const manifest = generateManifestFull(
155
+ router.urlpatterns,
156
+ routerMountIndex,
157
+ router.__basename ? { urlPrefix: router.__basename } : undefined,
158
+ );
139
159
  routerMountIndex++;
140
160
  allManifests.push({ id, manifest });
141
161
  const routeCount = Object.keys(manifest.routeManifest).length;
@@ -230,8 +250,15 @@ export async function discoverRouters(
230
250
  }
231
251
  }
232
252
 
253
+ debug?.(
254
+ "inner: generated manifests for %d router(s) (%sms)",
255
+ allManifests.length,
256
+ (performance.now() - manifestGenStart).toFixed(1),
257
+ );
258
+
233
259
  // Build route trie from merged manifest + ancestry
234
260
  let newMergedRouteTrie: any = null;
261
+ const trieStart = debug ? performance.now() : 0;
235
262
  if (Object.keys(newMergedRouteManifest).length > 0) {
236
263
  const buildRouteTrie = buildMod.buildRouteTrie;
237
264
  if (buildRouteTrie && mergedRouteAncestry) {
@@ -325,6 +352,11 @@ export async function discoverRouters(
325
352
  }
326
353
  }
327
354
 
355
+ debug?.(
356
+ "inner: trie build done (%sms)",
357
+ (performance.now() - trieStart).toFixed(1),
358
+ );
359
+
328
360
  // Commit all local state to the shared discovery state atomically.
329
361
  // This ensures a failed re-discovery (e.g. from a transient module
330
362
  // evaluation error) preserves the last known-good state.
@@ -16,6 +16,9 @@ import {
16
16
  stageBuildAssetModule,
17
17
  } from "../utils/prerender-utils.js";
18
18
  import type { DiscoveryState } from "./state.js";
19
+ import { createRangoDebugger, NS } from "../debug.js";
20
+
21
+ const debug = createRangoDebugger(NS.prerender);
19
22
 
20
23
  /**
21
24
  * Expand prerender routes into concrete URLs and render them via the
@@ -30,6 +33,12 @@ export async function expandPrerenderRoutes(
30
33
  ): Promise<void> {
31
34
  if (!state.opts?.enableBuildPrerender || !state.isBuildMode) return;
32
35
 
36
+ const overallStart = debug ? performance.now() : 0;
37
+ debug?.(
38
+ "expandPrerenderRoutes: start (%d router manifest(s))",
39
+ allManifests.length,
40
+ );
41
+
33
42
  type PrerenderEntry = {
34
43
  urlPath: string;
35
44
  routeName: string;
@@ -51,96 +60,160 @@ export async function expandPrerenderRoutes(
51
60
  return substituteRouteParams(pattern, params);
52
61
  };
53
62
 
63
+ let resolvedRoutes = 0;
64
+ let totalDynamic = 0;
65
+
66
+ // Count dynamic routes upfront for progress reporting
54
67
  for (const { manifest } of allManifests) {
55
68
  if (!manifest.prerenderRoutes) continue;
56
- const defs = manifest._prerenderDefs || {};
57
69
  for (const routeName of manifest.prerenderRoutes) {
58
70
  const pattern = manifest.routeManifest[routeName];
59
- if (!pattern) continue;
60
- const def = defs[routeName];
61
- const isPassthroughRoute = !!def?.options?.passthrough;
62
- const hasDynamic = pattern.includes(":") || pattern.includes("*");
63
- if (!hasDynamic) {
64
- // Static route: use pattern directly (strip trailing slash for URL)
65
- entries.push({
66
- urlPath: pattern.replace(/\/$/, "") || "/",
67
- routeName,
68
- concurrency: 1,
69
- isPassthroughRoute,
70
- });
71
- } else {
72
- // Dynamic route: call getParams() to enumerate param combinations
73
- if (def?.getParams) {
74
- try {
75
- const buildVars: Record<string, any> = {};
76
- const getParamsCtx = {
77
- build: true as const,
78
- set: ((keyOrVar: any, value: any) => {
79
- contextSet(buildVars, keyOrVar, value);
80
- }) as any,
81
- reverse: getParamsReverse,
82
- };
83
- const paramsList = await def.getParams(getParamsCtx);
84
- const concurrency = def.options?.concurrency ?? 1;
85
- const hasBuildVars =
86
- Object.keys(buildVars).length > 0 ||
87
- Object.getOwnPropertySymbols(buildVars).length > 0;
88
- for (const params of paramsList) {
89
- let url = substituteRouteParams(
90
- pattern,
91
- params as Record<string, string>,
92
- encodePathParam,
71
+ if (pattern && (pattern.includes(":") || pattern.includes("*"))) {
72
+ totalDynamic++;
73
+ }
74
+ }
75
+ }
76
+
77
+ // Periodic progress log so long getParams() calls don't look stalled
78
+ const paramsStart = performance.now();
79
+ const progressInterval =
80
+ totalDynamic > 0
81
+ ? setInterval(() => {
82
+ const elapsed = ((performance.now() - paramsStart) / 1000).toFixed(1);
83
+ console.log(
84
+ `[rsc-router] Resolving prerender params... ${resolvedRoutes}/${totalDynamic} routes (${elapsed}s)`,
85
+ );
86
+ }, 5000)
87
+ : undefined;
88
+
89
+ try {
90
+ for (const { manifest } of allManifests) {
91
+ if (!manifest.prerenderRoutes) continue;
92
+ const defs = manifest._prerenderDefs || {};
93
+ const passthroughSet = new Set(manifest.passthroughRoutes || []);
94
+ for (const routeName of manifest.prerenderRoutes) {
95
+ const pattern = manifest.routeManifest[routeName];
96
+ if (!pattern) continue;
97
+ const def = defs[routeName];
98
+ const isPassthroughRoute = passthroughSet.has(routeName);
99
+ const hasDynamic = pattern.includes(":") || pattern.includes("*");
100
+ if (!hasDynamic) {
101
+ // Static route: use pattern directly (strip trailing slash for URL)
102
+ entries.push({
103
+ urlPath: pattern.replace(/\/$/, "") || "/",
104
+ routeName,
105
+ concurrency: 1,
106
+ isPassthroughRoute,
107
+ });
108
+ } else {
109
+ // Dynamic route: call getParams() to enumerate param combinations
110
+ if (def?.getParams) {
111
+ const getParamsStart = debug ? performance.now() : 0;
112
+ try {
113
+ const buildVars: Record<string, any> = {};
114
+ const buildEnv = state.resolvedBuildEnv;
115
+ const getParamsCtx = {
116
+ build: true as const,
117
+ dev: !state.isBuildMode,
118
+ set: ((keyOrVar: any, value: any) => {
119
+ contextSet(buildVars, keyOrVar, value);
120
+ }) as any,
121
+ reverse: getParamsReverse,
122
+ get env() {
123
+ if (buildEnv !== undefined) return buildEnv;
124
+ throw new Error(
125
+ "[rsc-router] ctx.env is not available during build-time getParams(). " +
126
+ "Configure buildEnv in your rango() plugin options to enable build-time env access.",
127
+ );
128
+ },
129
+ };
130
+ const paramsList = await def.getParams(getParamsCtx);
131
+ debug?.(
132
+ "getParams %s -> %d params (%sms)",
133
+ routeName,
134
+ paramsList.length,
135
+ (performance.now() - getParamsStart).toFixed(1),
93
136
  );
94
- // Anonymous wildcard fallback: use conventional keys if provided
95
- if (url.includes("*")) {
96
- const wildcardValue =
97
- (params as Record<string, string>)["*"] ??
98
- (params as Record<string, string>).splat;
99
- if (wildcardValue !== undefined) {
100
- url = url.replace(/\*[^/]*$/, encodePathParam(wildcardValue));
137
+ const concurrency = def.options?.concurrency ?? 1;
138
+ const hasBuildVars =
139
+ Object.keys(buildVars).length > 0 ||
140
+ Object.getOwnPropertySymbols(buildVars).length > 0;
141
+ for (const params of paramsList) {
142
+ let url = substituteRouteParams(
143
+ pattern,
144
+ params as Record<string, string>,
145
+ encodePathParam,
146
+ );
147
+ // Anonymous wildcard fallback: use conventional keys if provided
148
+ if (url.includes("*")) {
149
+ const wildcardValue =
150
+ (params as Record<string, string>)["*"] ??
151
+ (params as Record<string, string>).splat;
152
+ if (wildcardValue !== undefined) {
153
+ url = url.replace(
154
+ /\*[^/]*$/,
155
+ encodePathParam(wildcardValue),
156
+ );
157
+ }
101
158
  }
159
+ entries.push({
160
+ urlPath: url.replace(/\/$/, "") || "/",
161
+ routeName,
162
+ concurrency,
163
+ ...(hasBuildVars ? { buildVars } : {}),
164
+ isPassthroughRoute,
165
+ });
102
166
  }
103
- entries.push({
104
- urlPath: url.replace(/\/$/, "") || "/",
105
- routeName,
106
- concurrency,
107
- ...(hasBuildVars ? { buildVars } : {}),
108
- isPassthroughRoute,
109
- });
110
- }
111
- } catch (err: any) {
112
- // Skip in getParams() skips the entire route
113
- if (err.name === "Skip") {
114
- console.log(
115
- `[rsc-router] SKIP route "${routeName}" - ${err.message}`,
116
- );
117
- notifyOnError(
118
- registry,
119
- err,
120
- "prerender",
121
- routeName,
122
- undefined,
123
- true,
167
+ resolvedRoutes++;
168
+ } catch (err: any) {
169
+ resolvedRoutes++;
170
+ // Skip in getParams() skips the entire route
171
+ if (err.name === "Skip") {
172
+ console.log(
173
+ `[rsc-router] SKIP route "${routeName}" - ${err.message}`,
174
+ );
175
+ notifyOnError(
176
+ registry,
177
+ err,
178
+ "prerender",
179
+ routeName,
180
+ undefined,
181
+ true,
182
+ );
183
+ continue;
184
+ }
185
+ // Regular error: fail the build
186
+ console.error(
187
+ `[rsc-router] Failed to get params for prerender route "${routeName}": ${err.message}`,
124
188
  );
125
- continue;
189
+ notifyOnError(registry, err, "prerender", routeName);
190
+ throw err;
126
191
  }
127
- // Regular error: fail the build
128
- console.error(
129
- `[rsc-router] Failed to get params for prerender route "${routeName}": ${err.message}`,
192
+ } else {
193
+ console.warn(
194
+ `[rsc-router] Dynamic prerender route "${routeName}" has no getParams(), skipping`,
130
195
  );
131
- notifyOnError(registry, err, "prerender", routeName);
132
- throw err;
133
196
  }
134
- } else {
135
- console.warn(
136
- `[rsc-router] Dynamic prerender route "${routeName}" has no getParams(), skipping`,
137
- );
138
197
  }
139
198
  }
140
199
  }
200
+ } finally {
201
+ if (progressInterval) {
202
+ clearInterval(progressInterval);
203
+ const elapsed = ((performance.now() - paramsStart) / 1000).toFixed(1);
204
+ console.log(
205
+ `[rsc-router] Resolved prerender params: ${resolvedRoutes}/${totalDynamic} routes (${elapsed}s)`,
206
+ );
207
+ }
141
208
  }
142
209
 
143
- if (entries.length === 0) return;
210
+ if (entries.length === 0) {
211
+ debug?.(
212
+ "no prerender entries (done in %sms)",
213
+ (performance.now() - overallStart).toFixed(1),
214
+ );
215
+ return;
216
+ }
144
217
 
145
218
  // Determine the max concurrency for the log header
146
219
  const maxConcurrency = Math.max(...entries.map((e) => e.concurrency));
@@ -149,6 +222,11 @@ export async function expandPrerenderRoutes(
149
222
  console.log(
150
223
  `[rsc-router] Pre-rendering ${entries.length} URL(s)${concurrencyNote}...`,
151
224
  );
225
+ debug?.(
226
+ "prerender loop: %d entries, max concurrency %d",
227
+ entries.length,
228
+ maxConcurrency,
229
+ );
152
230
 
153
231
  const { hashParams } = await rscEnv.runner.import("@rangojs/router/build");
154
232
 
@@ -175,6 +253,7 @@ export async function expandPrerenderRoutes(
175
253
  {},
176
254
  entry.buildVars,
177
255
  entry.isPassthroughRoute,
256
+ state.resolvedBuildEnv,
178
257
  );
179
258
  if (!result) continue;
180
259
 
@@ -265,6 +344,13 @@ export async function expandPrerenderRoutes(
265
344
  console.log(
266
345
  `[rsc-router] Pre-render complete: ${parts.join(", ")} (${totalElapsed}ms total)`,
267
346
  );
347
+ debug?.(
348
+ "expandPrerenderRoutes done: %d done, %d skipped, %sms (overall %sms)",
349
+ doneCount,
350
+ skipCount,
351
+ totalElapsed,
352
+ (performance.now() - overallStart).toFixed(1),
353
+ );
268
354
  }
269
355
 
270
356
  /**
@@ -285,6 +371,12 @@ export async function renderStaticHandlers(
285
371
  )
286
372
  return;
287
373
 
374
+ const overallStart = debug ? performance.now() : 0;
375
+ debug?.(
376
+ "renderStaticHandlers: start (%d static module(s))",
377
+ state.resolvedStaticModules.size,
378
+ );
379
+
288
380
  const manifestEntries: Record<string, string> = {};
289
381
  let staticDone = 0;
290
382
  let staticSkip = 0;
@@ -326,6 +418,8 @@ export async function renderStaticHandlers(
326
418
  def.handler,
327
419
  def.$$id,
328
420
  (def as any).$$routePrefix,
421
+ state.resolvedBuildEnv,
422
+ !state.isBuildMode,
329
423
  );
330
424
  if (result) {
331
425
  const hasHandles = Object.keys(result.handles).length > 0;
@@ -382,4 +476,11 @@ export async function renderStaticHandlers(
382
476
  console.log(
383
477
  `[rsc-router] Static render complete: ${staticParts.join(", ")} (${totalStaticElapsed}ms total)`,
384
478
  );
479
+ debug?.(
480
+ "renderStaticHandlers done: %d done, %d skipped, %sms (overall %sms)",
481
+ staticDone,
482
+ staticSkip,
483
+ totalStaticElapsed,
484
+ (performance.now() - overallStart).toFixed(1),
485
+ );
385
486
  }
@@ -13,11 +13,13 @@ export const VIRTUAL_ROUTES_MANIFEST_ID = "virtual:rsc-router/routes-manifest";
13
13
  export interface PluginOptions {
14
14
  enableBuildPrerender?: boolean;
15
15
  staticRouteTypesGeneration?: boolean;
16
- include?: string[];
17
- exclude?: string[];
18
16
  // Mutable ref for deferred auto-discovery (node preset).
19
17
  // The auto-discover config() hook populates this before configResolved.
20
18
  routerPathRef?: { path?: string };
19
+ /** Build-time env option from rango() config. */
20
+ buildEnv?: import("../plugin-types.js").BuildEnvOption;
21
+ /** Deployment preset (needed for buildEnv "auto" resolution). */
22
+ preset?: "node" | "cloudflare";
21
23
  }
22
24
 
23
25
  export interface PrecomputedEntry {
@@ -58,8 +60,8 @@ export interface DiscoveryState {
58
60
 
59
61
  prerenderManifestEntries: Record<string, string> | null;
60
62
  staticManifestEntries: Record<string, string> | null;
61
- handlerChunkInfo: ChunkInfo | null;
62
- staticHandlerChunkInfo: ChunkInfo | null;
63
+ handlerChunkInfoMap: Map<string, ChunkInfo>;
64
+ staticHandlerChunkInfoMap: Map<string, ChunkInfo>;
63
65
  rscEntryFileName: string | null;
64
66
  resolvedPrerenderModules: Map<string, string[]> | undefined;
65
67
  resolvedStaticModules: Map<string, string[]> | undefined;
@@ -69,6 +71,11 @@ export interface DiscoveryState {
69
71
  devServer: any;
70
72
  selfWrittenGenFiles: Map<string, { at: number; hash: string }>;
71
73
  SELF_WRITE_WINDOW_MS: number;
74
+
75
+ /** Resolved build-time env bindings (set during buildStart/configureServer). */
76
+ resolvedBuildEnv?: Record<string, unknown>;
77
+ /** Cleanup function for build-time env resources (e.g., miniflare). */
78
+ buildEnvDispose?: (() => Promise<void> | void) | null;
72
79
  }
73
80
 
74
81
  export function createDiscoveryState(
@@ -95,8 +102,8 @@ export function createDiscoveryState(
95
102
 
96
103
  prerenderManifestEntries: null,
97
104
  staticManifestEntries: null,
98
- handlerChunkInfo: null,
99
- staticHandlerChunkInfo: null,
105
+ handlerChunkInfoMap: new Map(),
106
+ staticHandlerChunkInfoMap: new Map(),
100
107
  rscEntryFileName: null,
101
108
  resolvedPrerenderModules: undefined,
102
109
  resolvedStaticModules: undefined,
package/src/vite/index.ts CHANGED
@@ -13,4 +13,8 @@ export type {
13
13
  RangoNodeOptions,
14
14
  RangoCloudflareOptions,
15
15
  RangoOptions,
16
+ BuildEnvOption,
17
+ BuildEnvFactory,
18
+ BuildEnvFactoryContext,
19
+ BuildEnvResult,
16
20
  } from "./plugin-types.js";