@rangojs/router 0.0.0-experimental.b02a2fec → 0.0.0-experimental.b30bbf02

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 (112) hide show
  1. package/README.md +112 -17
  2. package/dist/vite/index.js +1338 -462
  3. package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  4. package/package.json +7 -5
  5. package/skills/breadcrumbs/SKILL.md +3 -1
  6. package/skills/handler-use/SKILL.md +362 -0
  7. package/skills/hooks/SKILL.md +33 -20
  8. package/skills/intercept/SKILL.md +20 -0
  9. package/skills/layout/SKILL.md +22 -0
  10. package/skills/links/SKILL.md +90 -16
  11. package/skills/loader/SKILL.md +70 -3
  12. package/skills/middleware/SKILL.md +34 -3
  13. package/skills/migrate-nextjs/SKILL.md +562 -0
  14. package/skills/migrate-react-router/SKILL.md +769 -0
  15. package/skills/parallel/SKILL.md +66 -0
  16. package/skills/rango/SKILL.md +25 -22
  17. package/skills/response-routes/SKILL.md +8 -0
  18. package/skills/route/SKILL.md +24 -0
  19. package/skills/server-actions/SKILL.md +739 -0
  20. package/skills/streams-and-websockets/SKILL.md +283 -0
  21. package/skills/typesafety/SKILL.md +3 -1
  22. package/src/browser/app-shell.ts +52 -0
  23. package/src/browser/event-controller.ts +44 -4
  24. package/src/browser/navigation-bridge.ts +71 -5
  25. package/src/browser/navigation-client.ts +64 -13
  26. package/src/browser/navigation-store.ts +25 -1
  27. package/src/browser/partial-update.ts +34 -3
  28. package/src/browser/prefetch/cache.ts +129 -21
  29. package/src/browser/prefetch/fetch.ts +148 -16
  30. package/src/browser/prefetch/queue.ts +36 -5
  31. package/src/browser/rango-state.ts +53 -13
  32. package/src/browser/react/Link.tsx +30 -2
  33. package/src/browser/react/NavigationProvider.tsx +70 -18
  34. package/src/browser/react/filter-segment-order.ts +51 -7
  35. package/src/browser/react/use-navigation.ts +22 -2
  36. package/src/browser/react/use-params.ts +11 -1
  37. package/src/browser/react/use-router.ts +8 -1
  38. package/src/browser/react/use-segments.ts +11 -8
  39. package/src/browser/rsc-router.tsx +34 -6
  40. package/src/browser/segment-reconciler.ts +36 -14
  41. package/src/browser/types.ts +19 -0
  42. package/src/build/route-trie.ts +50 -24
  43. package/src/cache/cf/cf-cache-store.ts +5 -7
  44. package/src/client.tsx +82 -174
  45. package/src/index.rsc.ts +3 -0
  46. package/src/index.ts +40 -9
  47. package/src/outlet-context.ts +1 -1
  48. package/src/response-utils.ts +28 -0
  49. package/src/reverse.ts +7 -3
  50. package/src/route-definition/dsl-helpers.ts +175 -23
  51. package/src/route-definition/helpers-types.ts +63 -14
  52. package/src/route-definition/resolve-handler-use.ts +6 -0
  53. package/src/route-types.ts +7 -0
  54. package/src/router/handler-context.ts +24 -4
  55. package/src/router/lazy-includes.ts +6 -6
  56. package/src/router/loader-resolution.ts +3 -0
  57. package/src/router/manifest.ts +22 -13
  58. package/src/router/match-api.ts +4 -3
  59. package/src/router/match-handlers.ts +1 -0
  60. package/src/router/match-result.ts +21 -2
  61. package/src/router/middleware-types.ts +2 -22
  62. package/src/router/middleware.ts +54 -7
  63. package/src/router/pattern-matching.ts +87 -17
  64. package/src/router/revalidation.ts +15 -1
  65. package/src/router/segment-resolution/fresh.ts +8 -0
  66. package/src/router/segment-resolution/revalidation.ts +128 -100
  67. package/src/router/trie-matching.ts +18 -13
  68. package/src/router/url-params.ts +49 -0
  69. package/src/router.ts +1 -2
  70. package/src/rsc/handler.ts +8 -4
  71. package/src/rsc/helpers.ts +69 -41
  72. package/src/rsc/progressive-enhancement.ts +4 -0
  73. package/src/rsc/response-route-handler.ts +14 -1
  74. package/src/rsc/rsc-rendering.ts +10 -0
  75. package/src/rsc/server-action.ts +4 -0
  76. package/src/rsc/types.ts +6 -0
  77. package/src/segment-content-promise.ts +67 -0
  78. package/src/segment-loader-promise.ts +122 -0
  79. package/src/segment-system.tsx +11 -61
  80. package/src/server/context.ts +26 -3
  81. package/src/server/request-context.ts +10 -42
  82. package/src/ssr/index.tsx +5 -1
  83. package/src/types/handler-context.ts +12 -39
  84. package/src/types/loader-types.ts +5 -6
  85. package/src/types/request-scope.ts +126 -0
  86. package/src/types/route-entry.ts +11 -0
  87. package/src/types/segments.ts +17 -1
  88. package/src/urls/include-helper.ts +24 -14
  89. package/src/urls/path-helper-types.ts +30 -4
  90. package/src/urls/response-types.ts +2 -10
  91. package/src/vite/debug.ts +184 -0
  92. package/src/vite/discovery/discover-routers.ts +31 -3
  93. package/src/vite/discovery/gate-state.ts +171 -0
  94. package/src/vite/discovery/prerender-collection.ts +48 -1
  95. package/src/vite/discovery/self-gen-tracking.ts +27 -1
  96. package/src/vite/plugins/cjs-to-esm.ts +5 -0
  97. package/src/vite/plugins/client-ref-dedup.ts +16 -0
  98. package/src/vite/plugins/client-ref-hashing.ts +16 -4
  99. package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
  100. package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
  101. package/src/vite/plugins/cloudflare-protocol-stub.ts +214 -0
  102. package/src/vite/plugins/expose-action-id.ts +52 -28
  103. package/src/vite/plugins/expose-ids/router-transform.ts +20 -3
  104. package/src/vite/plugins/expose-internal-ids.ts +516 -486
  105. package/src/vite/plugins/performance-tracks.ts +17 -9
  106. package/src/vite/plugins/use-cache-transform.ts +56 -43
  107. package/src/vite/plugins/version-injector.ts +37 -11
  108. package/src/vite/rango.ts +49 -14
  109. package/src/vite/router-discovery.ts +558 -53
  110. package/src/vite/utils/banner.ts +1 -1
  111. package/src/vite/utils/package-resolution.ts +41 -1
  112. package/src/vite/utils/prerender-utils.ts +20 -6
@@ -10,7 +10,7 @@ import type { Plugin } from "vite";
10
10
  import { createServer as createViteServer } from "vite";
11
11
  import { resolve } from "node:path";
12
12
  import { readFileSync } from "node:fs";
13
- import { createRequire } from "node:module";
13
+ import { createRequire, register } from "node:module";
14
14
  import { pathToFileURL } from "node:url";
15
15
  import {
16
16
  formatNestedRouterConflictError,
@@ -19,6 +19,10 @@ import {
19
19
  } from "../build/generate-route-types.js";
20
20
  import { createVersionPlugin } from "./plugins/version-plugin.js";
21
21
  import { createVirtualStubPlugin } from "./plugins/virtual-stub-plugin.js";
22
+ import {
23
+ BUILD_ENV_GLOBAL_KEY,
24
+ createCloudflareProtocolStubPlugin,
25
+ } from "./plugins/cloudflare-protocol-stub.js";
22
26
  import {
23
27
  exposeInternalIds,
24
28
  exposeRouterId,
@@ -31,7 +35,10 @@ import {
31
35
  type DiscoveryState,
32
36
  type PluginOptions,
33
37
  } from "./discovery/state.js";
34
- import { consumeSelfGenWrite } from "./discovery/self-gen-tracking.js";
38
+ import {
39
+ consumeSelfGenWrite,
40
+ peekSelfGenWrite,
41
+ } from "./discovery/self-gen-tracking.js";
35
42
  import { discoverRouters } from "./discovery/discover-routers.js";
36
43
  import {
37
44
  writeCombinedRouteTypesWithTracking,
@@ -43,10 +50,60 @@ import {
43
50
  generatePerRouterModule,
44
51
  } from "./discovery/virtual-module-codegen.js";
45
52
  import { postprocessBundle } from "./discovery/bundle-postprocess.js";
53
+ import { createDiscoveryGate } from "./discovery/gate-state.js";
46
54
  import { resetStagedBuildAssets } from "./utils/prerender-utils.js";
55
+ import { createRangoDebugger, timed, timedSync, NS } from "./debug.js";
56
+
57
+ const debugDiscovery = createRangoDebugger(NS.discovery);
58
+ const debugRoutes = createRangoDebugger(NS.routes);
59
+ const debugBuild = createRangoDebugger(NS.build);
60
+ const debugDev = createRangoDebugger(NS.dev);
47
61
 
48
62
  export { VIRTUAL_ROUTES_MANIFEST_ID };
49
63
 
64
+ // ============================================================================
65
+ // Node ESM Loader Hook Registration
66
+ // ============================================================================
67
+
68
+ /**
69
+ * Registers a Node ESM loader hook that resolves `cloudflare:*` specifiers
70
+ * to a data: URL stub. Defense-in-depth alongside the Vite transform in
71
+ * `cloudflare-protocol-stub.ts`:
72
+ *
73
+ * - The Vite transform catches `cloudflare:*` imports in modules that flow
74
+ * through Vite's plugin pipeline. That's the vast majority of cases.
75
+ * - The Node loader catches imports in modules that Vite/Rollup externalize
76
+ * (e.g. the `partyserver` package, which has a top-level
77
+ * `import { DurableObject, env } from "cloudflare:workers"` and ships
78
+ * shapes plugin-rsc marks as external). Externalized modules are loaded
79
+ * via Node's native ESM loader, which rejects URL schemes.
80
+ *
81
+ * Registration is process-global and one-shot. The hook only intercepts
82
+ * `cloudflare:*` specifiers; everything else passes through via
83
+ * `nextResolve()`. It runs in a separate worker thread (Node ESM loader
84
+ * architecture), so it can't read the `globalThis[BUILD_ENV_GLOBAL_KEY]`
85
+ * bridge that the Vite transform uses — the stubs served here always
86
+ * return `env = {}`. That's fine because externalized libraries don't
87
+ * typically access `env` at module top level; user source (where real
88
+ * `env` matters at build time) flows through the Vite transform.
89
+ */
90
+ let loaderHookRegistered = false;
91
+ function ensureCloudflareProtocolLoaderRegistered(): void {
92
+ if (loaderHookRegistered) return;
93
+ loaderHookRegistered = true;
94
+ try {
95
+ register(
96
+ new URL("./plugins/cloudflare-protocol-loader-hook.mjs", import.meta.url),
97
+ );
98
+ } catch (err: any) {
99
+ // register() requires Node 18.19+ / 20.6+. Older Node still has the
100
+ // Vite transform as primary defense.
101
+ console.warn(
102
+ `[rsc-router] Could not register Node ESM loader hook for cloudflare:* imports (${err?.message ?? err}). Falling back to Vite transform only.`,
103
+ );
104
+ }
105
+ }
106
+
50
107
  // ============================================================================
51
108
  // Temp Server Factory
52
109
  // ============================================================================
@@ -66,6 +123,11 @@ async function createTempRscServer(
66
123
  state: DiscoveryState,
67
124
  options: { forceBuild?: boolean; cacheDir?: string } = {},
68
125
  ) {
126
+ // Install the Node ESM loader hook before any module evaluation so
127
+ // `cloudflare:*` specifiers in externalized/loader-delegated modules
128
+ // (e.g. packages plugin-rsc marks as external) resolve to stubs
129
+ // instead of crashing Node's native loader.
130
+ ensureCloudflareProtocolLoaderRegistered();
69
131
  const { default: rsc } = await import("@vitejs/plugin-rsc");
70
132
  return createViteServer({
71
133
  root: state.projectRoot,
@@ -88,6 +150,7 @@ async function createTempRscServer(
88
150
  ...(options.forceBuild ? [hashClientRefs(state.projectRoot)] : []),
89
151
  createVersionPlugin(),
90
152
  createVirtualStubPlugin(),
153
+ createCloudflareProtocolStubPlugin(),
91
154
  // Dev prerender must use dev-mode IDs (path-based) to match the workerd
92
155
  // runtime. forceBuild produces hashed IDs for production bundle consistency.
93
156
  exposeInternalIds(options.forceBuild ? { forceBuild: true } : undefined),
@@ -177,6 +240,11 @@ async function acquireBuildEnv(
177
240
 
178
241
  s.resolvedBuildEnv = result.env;
179
242
  s.buildEnvDispose = result.dispose ?? null;
243
+ // Bridge the resolved env into `cloudflare:workers`'s stubbed `env`
244
+ // export so user code that does `import { env } from "cloudflare:workers"`
245
+ // sees the real bindings proxy during discovery + prerender instead of
246
+ // an empty object. The stub reads this global at module-evaluation time.
247
+ (globalThis as Record<string, unknown>)[BUILD_ENV_GLOBAL_KEY] = result.env;
180
248
  return true;
181
249
  }
182
250
 
@@ -193,6 +261,7 @@ async function releaseBuildEnv(s: DiscoveryState): Promise<void> {
193
261
  s.buildEnvDispose = null;
194
262
  }
195
263
  s.resolvedBuildEnv = undefined;
264
+ delete (globalThis as Record<string, unknown>)[BUILD_ENV_GLOBAL_KEY];
196
265
  }
197
266
 
198
267
  /**
@@ -295,6 +364,17 @@ export function createRouterDiscoveryPlugin(
295
364
  resolveDiscovery = resolve;
296
365
  });
297
366
 
367
+ // Manifest-readiness gate + rediscovery scheduler.
368
+ // The virtual:rsc-router/routes-manifest module's `load()` hook
369
+ // awaits `s.discoveryDone`; the gate is reset on each discovery
370
+ // cycle so workerd's HMR reloads block until the new gen file is
371
+ // written. State machine + transitions are extracted into
372
+ // ./discovery/gate-state.ts and unit-tested there — see the
373
+ // module's JSDoc for the four-flag contract.
374
+ const gate = createDiscoveryGate(s, debugDiscovery);
375
+ const beginDiscoveryGate = gate.beginGate;
376
+ const resolveDiscoveryGate = gate.resolveGate;
377
+
298
378
  // Compute dev server origin from resolved URLs (preferred) or config port (fallback).
299
379
  // Called after discovery (or in the load hook) when the server may be listening.
300
380
  const getDevServerOrigin = () =>
@@ -317,10 +397,103 @@ export function createRouterDiscoveryPlugin(
317
397
  releaseBuildEnv(s).catch(() => {});
318
398
  });
319
399
 
400
+ // Mirror the build-path contract (router-discovery.ts ~line 878):
401
+ // set __rscRouterDiscoveryActive before running user modules so any
402
+ // module-level router.reverse() calls return a placeholder instead
403
+ // of throwing. The temp Vite server's module runner has its own
404
+ // module context; the flag must be on globalThis to cross that
405
+ // boundary. Cleared in finally so the dev request handlers run with
406
+ // strict reverse() semantics afterwards.
407
+ async function importEntryAndRegistry(tempRscEnv: any): Promise<void> {
408
+ const flagAlreadySet = !!(globalThis as any).__rscRouterDiscoveryActive;
409
+ if (!flagAlreadySet) {
410
+ (globalThis as any).__rscRouterDiscoveryActive = true;
411
+ }
412
+ try {
413
+ debugDiscovery?.(
414
+ "importEntryAndRegistry: importing entry (flag=%s)",
415
+ (globalThis as any).__rscRouterDiscoveryActive ?? false,
416
+ );
417
+ await tempRscEnv.runner.import(s.resolvedEntryPath!);
418
+ debugDiscovery?.(
419
+ "importEntryAndRegistry: entry import OK, fetching RouterRegistry",
420
+ );
421
+ const serverMod = await tempRscEnv.runner.import(
422
+ "@rangojs/router/server",
423
+ );
424
+ prerenderNodeRegistry = serverMod.RouterRegistry;
425
+ debugDiscovery?.(
426
+ "importEntryAndRegistry: registry size=%d",
427
+ prerenderNodeRegistry?.size ?? 0,
428
+ );
429
+ } finally {
430
+ if (!flagAlreadySet) {
431
+ delete (globalThis as any).__rscRouterDiscoveryActive;
432
+ debugDiscovery?.(
433
+ "importEntryAndRegistry: cleared __rscRouterDiscoveryActive",
434
+ );
435
+ }
436
+ }
437
+ }
438
+
320
439
  async function getOrCreateTempServer(): Promise<any | null> {
321
- if (prerenderNodeRegistry) {
322
- return (prerenderTempServer.environments as any)?.rsc ?? null;
440
+ // Reuse path: if a temp server is already alive, prefer reusing
441
+ // it over orphaning the existing instance and spinning up a new
442
+ // one. This handles two cases:
443
+ //
444
+ // 1. Steady-state cache hit (cold-start completed, registry
445
+ // cached) — return the env immediately.
446
+ // 2. Recovery from a failed refresh: refreshTempRscEnv() may
447
+ // have invalidated and nulled the registry, then thrown
448
+ // during importEntryAndRegistry. Without reuse, the next
449
+ // call would `createTempRscServer` and overwrite the
450
+ // handle, leaking the previous server. Try to re-import on
451
+ // the existing runner first; only if THAT fails do we
452
+ // close the orphan and create new.
453
+ if (prerenderTempServer) {
454
+ const existingEnv = (prerenderTempServer.environments as any)?.rsc;
455
+ if (existingEnv?.runner) {
456
+ if (prerenderNodeRegistry) {
457
+ debugDiscovery?.(
458
+ "getOrCreateTempServer: cached temp runner reused",
459
+ );
460
+ return existingEnv;
461
+ }
462
+ // Server alive but registry missing — likely after a prior
463
+ // refresh's invalidate + import threw. Try to re-import.
464
+ debugDiscovery?.(
465
+ "getOrCreateTempServer: server alive but registry missing — re-importing",
466
+ );
467
+ try {
468
+ await importEntryAndRegistry(existingEnv);
469
+ return existingEnv;
470
+ } catch (err: any) {
471
+ debugDiscovery?.(
472
+ "getOrCreateTempServer: reuse import failed (%s) — closing orphan and creating fresh",
473
+ err?.message ?? String(err),
474
+ );
475
+ await prerenderTempServer.close().catch(() => {});
476
+ prerenderTempServer = null;
477
+ prerenderNodeRegistry = null;
478
+ // Fall through to create-new path below.
479
+ }
480
+ } else {
481
+ // Server reference exists but its rsc env is unhealthy
482
+ // (no runner). Close and recreate.
483
+ debugDiscovery?.(
484
+ "getOrCreateTempServer: existing server has no rsc.runner — closing and recreating",
485
+ );
486
+ await prerenderTempServer.close().catch(() => {});
487
+ prerenderTempServer = null;
488
+ prerenderNodeRegistry = null;
489
+ }
323
490
  }
491
+
492
+ // Create path: no existing temp server (or just nullified above).
493
+ debugDiscovery?.(
494
+ "getOrCreateTempServer: creating new temp server, entry=%s",
495
+ s.resolvedEntryPath ?? "(unset)",
496
+ );
324
497
  try {
325
498
  prerenderTempServer = await createTempRscServer(s, {
326
499
  cacheDir: "node_modules/.vite_prerender",
@@ -328,14 +501,17 @@ export function createRouterDiscoveryPlugin(
328
501
 
329
502
  const tempRscEnv = (prerenderTempServer.environments as any)?.rsc;
330
503
  if (tempRscEnv?.runner) {
331
- await tempRscEnv.runner.import(s.resolvedEntryPath!);
332
- const serverMod = await tempRscEnv.runner.import(
333
- "@rangojs/router/server",
334
- );
335
- prerenderNodeRegistry = serverMod.RouterRegistry;
504
+ await importEntryAndRegistry(tempRscEnv);
336
505
  return tempRscEnv;
337
506
  }
507
+ debugDiscovery?.(
508
+ "getOrCreateTempServer: tempRscEnv.runner unavailable",
509
+ );
338
510
  } catch (err: any) {
511
+ debugDiscovery?.(
512
+ "getOrCreateTempServer: FAILED message=%s",
513
+ err.message,
514
+ );
339
515
  console.warn(
340
516
  `[rsc-router] Failed to create temp runner: ${err.message}`,
341
517
  );
@@ -343,24 +519,137 @@ export function createRouterDiscoveryPlugin(
343
519
  return null;
344
520
  }
345
521
 
522
+ // Clear the package-level singleton registries that survive a Vite
523
+ // moduleGraph.invalidateAll(). createRouter() / createHostRouter()
524
+ // call .set(id, ...) on these Maps; for "router removed" or
525
+ // "router id changed" edits, the OLD entry would persist after
526
+ // re-import without an explicit .clear(), leaving ghost routes
527
+ // in discoverRouters' output.
528
+ //
529
+ // We import the same module the runner imports, so the .clear()
530
+ // here mutates the same Map the freshly re-imported entry will
531
+ // populate.
532
+ async function clearTempRegistries(tempRscEnv: any): Promise<void> {
533
+ try {
534
+ const serverMod = await tempRscEnv.runner.import(
535
+ "@rangojs/router/server",
536
+ );
537
+ if (typeof serverMod?.RouterRegistry?.clear === "function") {
538
+ serverMod.RouterRegistry.clear();
539
+ }
540
+ if (typeof serverMod?.HostRouterRegistry?.clear === "function") {
541
+ serverMod.HostRouterRegistry.clear();
542
+ }
543
+ debugDiscovery?.(
544
+ "clearTempRegistries: cleared RouterRegistry + HostRouterRegistry",
545
+ );
546
+ } catch (err: any) {
547
+ // Non-fatal: if the import fails here, importEntryAndRegistry
548
+ // below will fail loudly with the same root cause and the
549
+ // caller will surface it.
550
+ debugDiscovery?.(
551
+ "clearTempRegistries: import @rangojs/router/server failed (%s)",
552
+ err?.message ?? String(err),
553
+ );
554
+ }
555
+ }
556
+
557
+ // HMR refresh: keep the temp Vite server alive across HMR cycles and
558
+ // invalidate its module graph instead of close+recreate. Closing the
559
+ // temp server during workerd's first post-cold-start module-fetch
560
+ // window disrupted the main dev server's transport — the user-visible
561
+ // symptom was a `transport was disconnected, cannot call "fetchModule"`
562
+ // error on the first urls.tsx edit (workerd's cache was cold, so its
563
+ // eval was still in flight when our close() ran). Module-graph
564
+ // invalidation is the architecturally cleaner refresh: same Vite
565
+ // instance, same transport, fresh source.
566
+ //
567
+ // Falls back to close+recreate when neither the env-level nor
568
+ // server-level moduleGraph exposes invalidateAll() (defensive — Vite
569
+ // versions / preset configurations may differ in which graph carries
570
+ // the module-runner cache).
571
+ async function refreshTempRscEnv(): Promise<any | null> {
572
+ let tempRscEnv = await getOrCreateTempServer();
573
+ if (!tempRscEnv) return null;
574
+
575
+ // Module-runner cache is on the per-environment graph in Vite 6+;
576
+ // older / non-environments setups carry it on the server graph.
577
+ // Try env first, server second.
578
+ const envGraph = (tempRscEnv as any).moduleGraph;
579
+ const serverGraph = (prerenderTempServer as any)?.moduleGraph;
580
+ const target = envGraph?.invalidateAll
581
+ ? envGraph
582
+ : serverGraph?.invalidateAll
583
+ ? serverGraph
584
+ : null;
585
+
586
+ if (!target) {
587
+ // No invalidate method available — fall back to close+recreate.
588
+ // This preserves the previous behavior in case a Vite version
589
+ // doesn't expose invalidateAll on either graph.
590
+ debugDiscovery?.(
591
+ "refreshTempRscEnv: invalidateAll unavailable on env+server graphs, falling back to close+recreate",
592
+ );
593
+ if (prerenderTempServer) {
594
+ await prerenderTempServer.close().catch(() => {});
595
+ prerenderTempServer = null;
596
+ prerenderNodeRegistry = null;
597
+ }
598
+ return await getOrCreateTempServer();
599
+ }
600
+
601
+ debugDiscovery?.(
602
+ "refreshTempRscEnv: invalidating module graph (%s)",
603
+ envGraph?.invalidateAll ? "env" : "server",
604
+ );
605
+ target.invalidateAll();
606
+ // Drop the cached registry so importEntryAndRegistry re-reads it
607
+ // through the now-invalidated module runner.
608
+ prerenderNodeRegistry = null;
609
+ // Clear singleton Maps that Vite's moduleGraph invalidation can't
610
+ // reach (RouterRegistry / HostRouterRegistry). Without this, an
611
+ // edit that REMOVES a createRouter() call or CHANGES a router id
612
+ // would leave the old entry in the registry, and discoverRouters
613
+ // would still emit its routes alongside whatever the new source
614
+ // declares.
615
+ await clearTempRegistries(tempRscEnv);
616
+ await importEntryAndRegistry(tempRscEnv);
617
+ return tempRscEnv;
618
+ }
619
+
346
620
  const discover = async () => {
621
+ const discoverStart = performance.now();
347
622
  const rscEnv = (server.environments as any)?.rsc;
348
623
  if (!rscEnv?.runner) {
349
624
  // Cloudflare dev: no module runner available (workerd-based RSC env).
350
625
  // Set devServerOrigin so the virtual module can inject __PRERENDER_DEV_URL
351
626
  // for on-demand prerender via the /__rsc_prerender endpoint.
627
+ debugDiscovery?.(
628
+ "dev: cloudflare path start, __rscRouterDiscoveryActive=%s",
629
+ (globalThis as any).__rscRouterDiscoveryActive ?? false,
630
+ );
352
631
  s.devServerOrigin = getDevServerOrigin();
353
632
 
354
633
  // Create a temp Node.js server to run runtime discovery and generate
355
634
  // named route types (static parser can't resolve factory calls).
356
635
  try {
357
636
  // Acquire build-time env bindings for dev prerender
358
- await acquireBuildEnv(s, viteCommand, viteMode);
637
+ await timed(debugDiscovery, "acquireBuildEnv", () =>
638
+ acquireBuildEnv(s, viteCommand, viteMode),
639
+ );
359
640
 
360
- const tempRscEnv = await getOrCreateTempServer();
641
+ const tempRscEnv = await timed(
642
+ debugDiscovery,
643
+ "getOrCreateTempServer",
644
+ () => getOrCreateTempServer(),
645
+ );
361
646
  if (tempRscEnv) {
362
- await discoverRouters(s, tempRscEnv);
363
- writeRouteTypesFiles(s);
647
+ await timed(debugDiscovery, "discoverRouters (cloudflare)", () =>
648
+ discoverRouters(s, tempRscEnv),
649
+ );
650
+ timedSync(debugDiscovery, "writeRouteTypesFiles", () =>
651
+ writeRouteTypesFiles(s),
652
+ );
364
653
  }
365
654
  } catch (err: any) {
366
655
  console.warn(
@@ -368,24 +657,35 @@ export function createRouterDiscoveryPlugin(
368
657
  );
369
658
  }
370
659
 
660
+ debugDiscovery?.(
661
+ "dev discovery done (%sms)",
662
+ (performance.now() - discoverStart).toFixed(1),
663
+ );
371
664
  resolveDiscovery!();
372
665
  return;
373
666
  }
374
667
 
375
668
  try {
376
669
  // Acquire build-time env bindings for dev prerender (Node.js path)
377
- await acquireBuildEnv(s, viteCommand, viteMode);
670
+ debugDiscovery?.("dev: node path start");
671
+ await timed(debugDiscovery, "acquireBuildEnv", () =>
672
+ acquireBuildEnv(s, viteCommand, viteMode),
673
+ );
378
674
 
379
675
  // Set the readiness gate BEFORE discovery so early requests
380
676
  // block until manifest is populated
381
- const serverMod = await rscEnv.runner.import(
382
- "@rangojs/router/server",
677
+ const serverMod = await timed(
678
+ debugDiscovery,
679
+ "import @rangojs/router/server",
680
+ () => rscEnv.runner.import("@rangojs/router/server"),
383
681
  );
384
682
  if (serverMod?.setManifestReadyPromise) {
385
683
  serverMod.setManifestReadyPromise(discoveryPromise);
386
684
  }
387
685
 
388
- await discoverRouters(s, rscEnv);
686
+ await timed(debugDiscovery, "discoverRouters", () =>
687
+ discoverRouters(s, rscEnv),
688
+ );
389
689
 
390
690
  // Store server origin for dev prerender endpoint (virtual module injection)
391
691
  s.devServerOrigin = getDevServerOrigin();
@@ -395,24 +695,36 @@ export function createRouterDiscoveryPlugin(
395
695
  // routes (e.g. Array.from loops) that the static parser cannot see.
396
696
  // writeRouteTypesFiles() only writes when content changes, so this
397
697
  // won't cause unnecessary HMR triggers.
398
- writeRouteTypesFiles(s);
698
+ timedSync(debugDiscovery, "writeRouteTypesFiles", () =>
699
+ writeRouteTypesFiles(s),
700
+ );
399
701
 
400
702
  // Populate the route map and per-router data in the RSC env
401
- await propagateDiscoveryState(rscEnv);
703
+ await timed(debugDiscovery, "propagateDiscoveryState", () =>
704
+ propagateDiscoveryState(rscEnv),
705
+ );
402
706
  } catch (err: any) {
403
707
  console.warn(
404
708
  `[rsc-router] Router discovery failed: ${err.message}\n${err.stack}`,
405
709
  );
406
710
  } finally {
711
+ debugDiscovery?.(
712
+ "dev discovery done (%sms)",
713
+ (performance.now() - discoverStart).toFixed(1),
714
+ );
407
715
  resolveDiscovery!();
408
716
  }
409
717
  };
410
718
 
411
719
  // Schedule after all plugins have finished configureServer.
412
- // Store the promise so the virtual module's load hook can await it.
413
- s.discoveryDone = new Promise<void>((resolve) => {
414
- setTimeout(() => discover().then(resolve, resolve), 0);
415
- });
720
+ // The gate (s.discoveryDone) is reset via beginDiscoveryGate() and
721
+ // resolved when discover() finishes, so the virtual manifest module's
722
+ // load() awaits the populated state.
723
+ beginDiscoveryGate();
724
+ setTimeout(
725
+ () => discover().then(resolveDiscoveryGate, resolveDiscoveryGate),
726
+ 0,
727
+ );
416
728
 
417
729
  // Dev-mode on-demand prerender endpoint.
418
730
  // When workerd hits a prerender route, it fetches this endpoint instead of
@@ -470,6 +782,17 @@ export function createRouterDiscoveryPlugin(
470
782
  };
471
783
 
472
784
  server.middlewares.use("/__rsc_prerender", async (req: any, res: any) => {
785
+ const reqStart = debugDev ? performance.now() : 0;
786
+ const logResult = (status: number, note: string) => {
787
+ debugDev?.(
788
+ "/__rsc_prerender %s -> %d %s (%sms)",
789
+ req.url,
790
+ status,
791
+ note,
792
+ (performance.now() - reqStart).toFixed(1),
793
+ );
794
+ };
795
+
473
796
  if (s.discoveryDone) await s.discoveryDone;
474
797
 
475
798
  const url = new URL(req.url || "/", "http://localhost");
@@ -477,6 +800,7 @@ export function createRouterDiscoveryPlugin(
477
800
  if (!pathname) {
478
801
  res.statusCode = 400;
479
802
  res.end("Missing pathname");
803
+ logResult(400, "missing pathname");
480
804
  return;
481
805
  }
482
806
 
@@ -500,6 +824,7 @@ export function createRouterDiscoveryPlugin(
500
824
  );
501
825
  res.statusCode = 500;
502
826
  res.end(`Prerender handler error: ${err.message}`);
827
+ logResult(500, "module refresh failed");
503
828
  return;
504
829
  }
505
830
  } else {
@@ -518,6 +843,7 @@ export function createRouterDiscoveryPlugin(
518
843
  if (!registry || registry.size === 0) {
519
844
  res.statusCode = 503;
520
845
  res.end("Prerender runner not available");
846
+ logResult(503, "no registry");
521
847
  return;
522
848
  }
523
849
 
@@ -556,6 +882,7 @@ export function createRouterDiscoveryPlugin(
556
882
  payload = { segments: result.segments, handles: result.handles };
557
883
  }
558
884
  res.end(JSON.stringify(payload));
885
+ logResult(200, `match ${result.routeName}`);
559
886
  return;
560
887
  } catch (err: any) {
561
888
  console.warn(
@@ -566,6 +893,7 @@ export function createRouterDiscoveryPlugin(
566
893
 
567
894
  res.statusCode = 404;
568
895
  res.end("No prerender match");
896
+ logResult(404, "no match");
569
897
  });
570
898
 
571
899
  // Watch url module and router files for changes and regenerate named-routes.gen.ts.
@@ -608,45 +936,117 @@ export function createRouterDiscoveryPlugin(
608
936
 
609
937
  // Re-run runtime discovery so factory-generated routes that the
610
938
  // static parser cannot see are refreshed after source changes.
611
- let runtimeRediscoveryInProgress = false;
939
+ // The state-machine concerns (queued/pending/gatePending) are
940
+ // owned by the gate created above (./discovery/gate-state.ts).
941
+ // Here we provide just the env-specific work.
612
942
  const refreshRuntimeDiscovery = async () => {
613
943
  const rscEnv = (server.environments as any)?.rsc;
614
- if (!rscEnv?.runner || runtimeRediscoveryInProgress) return;
615
- runtimeRediscoveryInProgress = true;
616
- try {
617
- await discoverRouters(s, rscEnv);
618
- writeRouteTypesFiles(s);
619
- await propagateDiscoveryState(rscEnv);
620
- } catch (err: any) {
621
- console.warn(
622
- `[rsc-router] Runtime re-discovery failed: ${err.message}`,
623
- );
624
- } finally {
625
- runtimeRediscoveryInProgress = false;
626
- }
944
+ const hasMainRunner = !!rscEnv?.runner;
945
+ // Cloudflare HMR has no main RSC runner (workerd is a separate
946
+ // runtime). When we have a populated runtime manifest from cold
947
+ // start, we can re-discover via the temp Node runner — the same
948
+ // mechanism getOrCreateTempServer() uses at startup. Without a
949
+ // populated manifest there's nothing useful to do, so bail
950
+ // before involving the gate machine at all.
951
+ if (!hasMainRunner && s.perRouterManifests.length === 0) return;
952
+ await gate.runRefreshCycle(async () => {
953
+ const hmrStart = performance.now();
954
+ try {
955
+ if (hasMainRunner) {
956
+ await timed(debugDiscovery, "hmr discoverRouters", () =>
957
+ discoverRouters(s, rscEnv),
958
+ );
959
+ timedSync(debugDiscovery, "hmr writeRouteTypesFiles", () =>
960
+ writeRouteTypesFiles(s),
961
+ );
962
+ await timed(debugDiscovery, "hmr propagateDiscoveryState", () =>
963
+ propagateDiscoveryState(rscEnv),
964
+ );
965
+ } else {
966
+ // Cloudflare HMR: invalidate the temp server's RSC module
967
+ // graph (or close+recreate as a fallback) so the runner
968
+ // re-reads the freshly edited source. Keeping the same
969
+ // Vite instance alive avoids disrupting workerd's transport
970
+ // during the first post-cold-start module-fetch window.
971
+ const tempRscEnv = await timed(
972
+ debugDiscovery,
973
+ "hmr refreshTempRscEnv (cloudflare)",
974
+ () => refreshTempRscEnv(),
975
+ );
976
+ if (!tempRscEnv) {
977
+ throw new Error(
978
+ "temp runner unavailable for cloudflare HMR rediscovery",
979
+ );
980
+ }
981
+ await timed(
982
+ debugDiscovery,
983
+ "hmr discoverRouters (cloudflare)",
984
+ () => discoverRouters(s, tempRscEnv),
985
+ );
986
+ timedSync(debugDiscovery, "hmr writeRouteTypesFiles", () =>
987
+ writeRouteTypesFiles(s),
988
+ );
989
+ }
990
+ } catch (err: any) {
991
+ console.warn(
992
+ `[rsc-router] Runtime re-discovery failed: ${err.message}`,
993
+ );
994
+ } finally {
995
+ debugDiscovery?.(
996
+ "hmr re-discovery done (%sms)",
997
+ (performance.now() - hmrStart).toFixed(1),
998
+ );
999
+ }
1000
+ });
627
1001
  };
628
1002
 
629
1003
  const scheduleRouteRegeneration = () => {
630
1004
  clearTimeout(routeChangeTimer);
631
1005
  routeChangeTimer = setTimeout(() => {
632
1006
  routeChangeTimer = undefined;
1007
+ const regenStart = debugDiscovery ? performance.now() : 0;
1008
+ const rscEnv = (server.environments as any)?.rsc;
1009
+ const skipStaticWrite =
1010
+ !rscEnv?.runner && s.perRouterManifests.length > 0;
633
1011
  try {
634
- writeCombinedRouteTypesWithTracking(s);
635
- if (s.perRouterManifests.length > 0) {
636
- supplementGenFilesWithRuntimeRoutes(s);
1012
+ // In cloudflare dev with a populated runtime manifest, the
1013
+ // static parser produces a strictly smaller (and actively
1014
+ // wrong) gen file — supplementGenFilesWithRuntimeRoutes can
1015
+ // only restore factory-only prefixes, and apps with mixed
1016
+ // static+factory routes under shared prefixes (cf-stress)
1017
+ // collapse to the 19-route static view. Skip the static
1018
+ // write entirely; runtime rediscovery below will overwrite
1019
+ // the gen file with the authoritative manifest.
1020
+ if (skipStaticWrite) {
1021
+ debugDiscovery?.(
1022
+ "watcher: skipping static write (cloudflare HMR — runtime rediscovery owns gen file)",
1023
+ );
1024
+ } else {
1025
+ writeCombinedRouteTypesWithTracking(s);
1026
+ if (s.perRouterManifests.length > 0) {
1027
+ supplementGenFilesWithRuntimeRoutes(s);
1028
+ }
637
1029
  }
638
1030
  } catch (err: any) {
639
1031
  console.error(
640
1032
  `[rsc-router] Route regeneration error: ${err.message}`,
641
1033
  );
642
1034
  }
1035
+ debugDiscovery?.(
1036
+ "watcher: regenerated gen files (%sms)",
1037
+ (performance.now() - regenStart).toFixed(1),
1038
+ );
643
1039
  // Async: re-run runtime discovery to refresh factory-generated
644
- // routes that the static parser cannot resolve.
1040
+ // routes that the static parser cannot resolve. Resolves the
1041
+ // discovery gate when complete.
645
1042
  if (s.perRouterManifests.length > 0) {
646
1043
  refreshRuntimeDiscovery().catch((err: any) => {
647
1044
  console.warn(
648
1045
  `[rsc-router] Runtime re-discovery error: ${err.message}`,
649
1046
  );
1047
+ // Even on error, unblock the gate so workerd's reload
1048
+ // doesn't hang indefinitely against the previous manifest.
1049
+ resolveDiscoveryGate();
650
1050
  });
651
1051
  }
652
1052
  }, 100);
@@ -674,6 +1074,12 @@ export function createRouterDiscoveryPlugin(
674
1074
  const hasUrls = source.includes("urls(");
675
1075
  const hasCreateRouter = /\bcreateRouter\s*[<(]/.test(source);
676
1076
  if (!hasUrls && !hasCreateRouter) return;
1077
+ debugDiscovery?.(
1078
+ "watcher: %s matches (urls=%s, router=%s)",
1079
+ filePath,
1080
+ hasUrls,
1081
+ hasCreateRouter,
1082
+ );
677
1083
  // Invalidate cache when a router file changes (new router added/removed)
678
1084
  if (hasCreateRouter) {
679
1085
  const nestedRouterConflict = findNestedRouterConflict([
@@ -688,6 +1094,17 @@ export function createRouterDiscoveryPlugin(
688
1094
  }
689
1095
  s.cachedRouterFiles = undefined;
690
1096
  }
1097
+ // Note the event in the gate machine IMMEDIATELY (before the
1098
+ // 100ms debounce and any downstream HMR fanout). This sets
1099
+ // both `pendingEvents` (so refresh's finally holds the gate
1100
+ // through the tail window even if no rediscovery is queued)
1101
+ // and resets `discoveryDone` to a fresh pending promise (so
1102
+ // workerd reloads triggered by the same source change can't
1103
+ // observe a stale resolved gate from cold-start). Resolved
1104
+ // by the trailing refreshRuntimeDiscovery() cycle.
1105
+ if (s.perRouterManifests.length > 0) {
1106
+ gate.noteRouteEvent();
1107
+ }
691
1108
  scheduleRouteRegeneration();
692
1109
  } catch {
693
1110
  // Ignore read errors for deleted/moved files
@@ -718,13 +1135,23 @@ export function createRouterDiscoveryPlugin(
718
1135
  async buildStart() {
719
1136
  if (!s.isBuildMode) return;
720
1137
  // Only run once across environment builds
721
- if (s.mergedRouteManifest !== null) return;
1138
+ if (s.mergedRouteManifest !== null) {
1139
+ debugDiscovery?.(
1140
+ "build: skip (already discovered, env=%s)",
1141
+ this.environment?.name ?? "?",
1142
+ );
1143
+ return;
1144
+ }
1145
+ const buildStartTime = performance.now();
1146
+ debugDiscovery?.("build: start (env=%s)", this.environment?.name ?? "?");
722
1147
  resetStagedBuildAssets(s.projectRoot);
723
1148
  s.prerenderManifestEntries = null;
724
1149
  s.staticManifestEntries = null;
725
1150
 
726
1151
  // Acquire build-time env bindings if configured
727
- await acquireBuildEnv(s, viteCommand, viteMode);
1152
+ await timed(debugDiscovery, "build acquireBuildEnv", () =>
1153
+ acquireBuildEnv(s, viteCommand, viteMode),
1154
+ );
728
1155
 
729
1156
  let tempServer: any = null;
730
1157
  // Signal to user-space code (e.g. reverse.ts) that build-time discovery
@@ -733,7 +1160,11 @@ export function createRouterDiscoveryPlugin(
733
1160
  // between the vite plugin and user code loaded via runner.import().
734
1161
  (globalThis as any).__rscRouterDiscoveryActive = true;
735
1162
  try {
736
- tempServer = await createTempRscServer(s, { forceBuild: true });
1163
+ tempServer = await timed(
1164
+ debugDiscovery,
1165
+ "build createTempRscServer",
1166
+ () => createTempRscServer(s, { forceBuild: true }),
1167
+ );
737
1168
 
738
1169
  const rscEnv = (tempServer.environments as any)?.rsc;
739
1170
  if (!rscEnv?.runner) {
@@ -753,11 +1184,15 @@ export function createRouterDiscoveryPlugin(
753
1184
  s.resolvedStaticModules = tempIdsPlugin.api.staticHandlerModules;
754
1185
  }
755
1186
 
756
- await discoverRouters(s, rscEnv);
1187
+ await timed(debugDiscovery, "build discoverRouters", () =>
1188
+ discoverRouters(s, rscEnv),
1189
+ );
757
1190
  // Update named-routes.gen.ts from runtime discovery.
758
1191
  // The runtime manifest includes dynamically generated routes
759
1192
  // that the static parser cannot extract from source code.
760
- writeRouteTypesFiles(s);
1193
+ timedSync(debugDiscovery, "build writeRouteTypesFiles", () =>
1194
+ writeRouteTypesFiles(s),
1195
+ );
761
1196
  } catch (err: any) {
762
1197
  // Extract the user source file from the stack trace (skip internal frames)
763
1198
  const sourceFile = err.stack
@@ -782,9 +1217,44 @@ export function createRouterDiscoveryPlugin(
782
1217
  } finally {
783
1218
  delete (globalThis as any).__rscRouterDiscoveryActive;
784
1219
  if (tempServer) {
785
- await tempServer.close();
1220
+ await timed(debugDiscovery, "build tempServer.close", () =>
1221
+ tempServer.close(),
1222
+ );
786
1223
  }
787
1224
  await releaseBuildEnv(s);
1225
+ debugDiscovery?.(
1226
+ "build discovery done (%sms)",
1227
+ (performance.now() - buildStartTime).toFixed(1),
1228
+ );
1229
+ }
1230
+ },
1231
+
1232
+ // Suppress vite's HMR cascade for our own gen-file writes.
1233
+ //
1234
+ // After every cf HMR cycle, refreshTempRscEnv → writeRouteTypesFiles
1235
+ // writes the configured gen files (default `router.named-routes.gen.ts`,
1236
+ // but the source filenames and gen suffix are user-configurable). The
1237
+ // chokidar watcher then fires twice independently: our
1238
+ // `handleRouteFileChange` (already short-circuited by
1239
+ // `consumeSelfGenWrite` inside `maybeHandleGeneratedRouteFileMutation`),
1240
+ // AND vite's own HMR pipeline (which invalidates the gen file's
1241
+ // importers and triggers a second workerd full reload — visible to the
1242
+ // user as a duplicate "[RSCRouter] HMR: version changed" on the client).
1243
+ //
1244
+ // `peekSelfGenWrite` is the authoritative filter: its map only contains
1245
+ // paths that `markSelfGenWrite` has registered, so it natively works
1246
+ // for any configured gen-file name. It is non-consuming so the chokidar
1247
+ // handler that fires later can still consume the same entry. Returning
1248
+ // [] tells vite "no modules invalidated by this change" — safe because
1249
+ // `s.perRouterManifests` is already up-to-date (the write that just
1250
+ // happened is the consequence of our just-completed rediscovery).
1251
+ handleHotUpdate(ctx) {
1252
+ if (peekSelfGenWrite(s, ctx.file)) {
1253
+ debugDiscovery?.(
1254
+ "handleHotUpdate: suppressing self-write HMR cascade for %s",
1255
+ ctx.file,
1256
+ );
1257
+ return [];
788
1258
  }
789
1259
  },
790
1260
 
@@ -808,19 +1278,38 @@ export function createRouterDiscoveryPlugin(
808
1278
  // This is critical for Cloudflare dev where the worker runs in a separate
809
1279
  // Miniflare process and can only receive manifest data via the virtual module.
810
1280
  if (s.discoveryDone) {
811
- await s.discoveryDone;
1281
+ await timed(
1282
+ debugRoutes,
1283
+ "await discoveryDone (manifest)",
1284
+ () => s.discoveryDone,
1285
+ );
812
1286
  }
813
- return generateRoutesManifestModule(s);
1287
+ const code = await timed(
1288
+ debugRoutes,
1289
+ "generateRoutesManifestModule",
1290
+ () => generateRoutesManifestModule(s),
1291
+ );
1292
+ debugRoutes?.("manifest module emitted (%d bytes)", code?.length ?? 0);
1293
+ return code;
814
1294
  }
815
1295
  // Per-router virtual modules: pure data exports (no side effects).
816
1296
  // ensureRouterManifest() imports the module and stores the data.
817
1297
  const perRouterPrefix = "\0" + VIRTUAL_ROUTES_MANIFEST_ID + "/";
818
1298
  if (id.startsWith(perRouterPrefix)) {
819
1299
  if (s.discoveryDone) {
820
- await s.discoveryDone;
1300
+ await timed(
1301
+ debugRoutes,
1302
+ "await discoveryDone (per-router)",
1303
+ () => s.discoveryDone,
1304
+ );
821
1305
  }
822
1306
  const routerId = id.slice(perRouterPrefix.length);
823
- return generatePerRouterModule(s, routerId);
1307
+ const code = await timed(
1308
+ debugRoutes,
1309
+ `generatePerRouterModule ${routerId}`,
1310
+ () => generatePerRouterModule(s, routerId),
1311
+ );
1312
+ return code;
824
1313
  }
825
1314
  // virtual:rsc-router/prerender-paths load handler removed
826
1315
  return null;
@@ -830,6 +1319,7 @@ export function createRouterDiscoveryPlugin(
830
1319
  // Used by closeBundle for handler code eviction and prerender data injection.
831
1320
  generateBundle(_options: any, bundle: any) {
832
1321
  if (this.environment?.name !== "rsc") return;
1322
+ const genStart = debugBuild ? performance.now() : 0;
833
1323
 
834
1324
  // Record RSC entry chunk filename for closeBundle injection
835
1325
  for (const [fileName, chunk] of Object.entries(bundle) as [
@@ -842,8 +1332,13 @@ export function createRouterDiscoveryPlugin(
842
1332
  }
843
1333
  }
844
1334
 
845
- if (!s.resolvedPrerenderModules?.size && !s.resolvedStaticModules?.size)
1335
+ if (!s.resolvedPrerenderModules?.size && !s.resolvedStaticModules?.size) {
1336
+ debugBuild?.(
1337
+ "generateBundle (rsc): no handlers to scan (%sms)",
1338
+ (performance.now() - genStart).toFixed(1),
1339
+ );
846
1340
  return;
1341
+ }
847
1342
 
848
1343
  // Clear maps at the start of each RSC generateBundle pass.
849
1344
  // Vite 6 multi-environment builds run RSC twice (analysis + production);
@@ -898,6 +1393,14 @@ export function createRouterDiscoveryPlugin(
898
1393
  }
899
1394
  }
900
1395
  }
1396
+
1397
+ debugBuild?.(
1398
+ "generateBundle (rsc): scanned %d chunks, %d prerender chunk(s), %d static chunk(s) (%sms)",
1399
+ Object.keys(bundle).length,
1400
+ s.handlerChunkInfoMap.size,
1401
+ s.staticHandlerChunkInfoMap.size,
1402
+ (performance.now() - genStart).toFixed(1),
1403
+ );
901
1404
  },
902
1405
 
903
1406
  // Build-time pre-rendering: evict handler code and inject collected prerender data.
@@ -911,7 +1414,9 @@ export function createRouterDiscoveryPlugin(
911
1414
  // Only run for the RSC environment — other environments (client, ssr) have
912
1415
  // no prerender/static data to process and would just do redundant file I/O.
913
1416
  if (this.environment && this.environment.name !== "rsc") return;
914
- postprocessBundle(s);
1417
+ timedSync(debugBuild, "closeBundle postprocessBundle", () =>
1418
+ postprocessBundle(s),
1419
+ );
915
1420
  },
916
1421
  },
917
1422
  };