@rangojs/router 0.0.0-experimental.83 → 0.0.0-experimental.8332dbe4
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.
- package/README.md +112 -17
- package/dist/vite/index.js +1197 -454
- package/package.json +4 -2
- package/skills/breadcrumbs/SKILL.md +3 -1
- package/skills/handler-use/SKILL.md +2 -0
- package/skills/hooks/SKILL.md +30 -2
- package/skills/i18n/SKILL.md +276 -0
- package/skills/intercept/SKILL.md +25 -0
- package/skills/layout/SKILL.md +2 -0
- package/skills/links/SKILL.md +234 -16
- package/skills/loader/SKILL.md +70 -3
- package/skills/middleware/SKILL.md +2 -0
- package/skills/migrate-nextjs/SKILL.md +3 -1
- package/skills/migrate-react-router/SKILL.md +4 -0
- package/skills/parallel/SKILL.md +9 -0
- package/skills/rango/SKILL.md +2 -0
- package/skills/response-routes/SKILL.md +8 -0
- package/skills/route/SKILL.md +24 -0
- package/skills/server-actions/SKILL.md +739 -0
- package/skills/streams-and-websockets/SKILL.md +283 -0
- package/skills/typesafety/SKILL.md +9 -1
- package/skills/view-transitions/SKILL.md +212 -0
- package/src/browser/app-shell.ts +52 -0
- package/src/browser/event-controller.ts +44 -4
- package/src/browser/navigation-bridge.ts +113 -6
- package/src/browser/navigation-store.ts +25 -1
- package/src/browser/partial-update.ts +44 -10
- package/src/browser/prefetch/cache.ts +16 -0
- package/src/browser/rango-state.ts +53 -13
- package/src/browser/react/NavigationProvider.tsx +64 -16
- package/src/browser/react/filter-segment-order.ts +51 -7
- package/src/browser/react/index.ts +3 -0
- package/src/browser/react/use-params.ts +8 -5
- package/src/browser/react/use-reverse.ts +99 -0
- package/src/browser/react/use-router.ts +8 -1
- package/src/browser/react/use-segments.ts +11 -8
- package/src/browser/rsc-router.tsx +34 -6
- package/src/browser/types.ts +19 -0
- package/src/build/route-trie.ts +2 -1
- package/src/cache/cf/cf-cache-store.ts +5 -7
- package/src/client.rsc.tsx +3 -0
- package/src/client.tsx +5 -1
- package/src/href-client.ts +4 -1
- package/src/index.rsc.ts +3 -0
- package/src/index.ts +3 -0
- package/src/outlet-context.ts +1 -1
- package/src/response-utils.ts +28 -0
- package/src/reverse.ts +62 -39
- package/src/route-definition/dsl-helpers.ts +16 -3
- package/src/route-definition/helpers-types.ts +6 -1
- package/src/route-definition/resolve-handler-use.ts +6 -0
- package/src/router/handler-context.ts +21 -41
- package/src/router/lazy-includes.ts +1 -1
- package/src/router/loader-resolution.ts +3 -0
- package/src/router/match-api.ts +4 -3
- package/src/router/match-handlers.ts +1 -0
- package/src/router/match-result.ts +21 -2
- package/src/router/middleware-types.ts +14 -25
- package/src/router/middleware.ts +54 -7
- package/src/router/pattern-matching.ts +101 -17
- package/src/router/revalidation.ts +15 -1
- package/src/router/segment-resolution/fresh.ts +8 -0
- package/src/router/segment-resolution/revalidation.ts +128 -100
- package/src/router/substitute-pattern-params.ts +56 -0
- package/src/router/trie-matching.ts +18 -13
- package/src/router/url-params.ts +49 -0
- package/src/router.ts +1 -2
- package/src/rsc/handler.ts +8 -4
- package/src/rsc/progressive-enhancement.ts +2 -0
- package/src/rsc/response-route-handler.ts +11 -10
- package/src/rsc/rsc-rendering.ts +3 -0
- package/src/rsc/server-action.ts +2 -0
- package/src/rsc/types.ts +6 -0
- package/src/segment-system.tsx +60 -9
- package/src/server/request-context.ts +10 -42
- package/src/ssr/index.tsx +5 -1
- package/src/types/handler-context.ts +12 -39
- package/src/types/loader-types.ts +5 -6
- package/src/types/request-scope.ts +126 -0
- package/src/types/segments.ts +17 -0
- package/src/urls/response-types.ts +2 -10
- package/src/vite/debug.ts +184 -0
- package/src/vite/discovery/discover-routers.ts +31 -3
- package/src/vite/discovery/gate-state.ts +171 -0
- package/src/vite/discovery/prerender-collection.ts +48 -1
- package/src/vite/discovery/self-gen-tracking.ts +27 -1
- package/src/vite/plugins/cjs-to-esm.ts +5 -0
- package/src/vite/plugins/client-ref-dedup.ts +16 -0
- package/src/vite/plugins/client-ref-hashing.ts +16 -4
- package/src/vite/plugins/expose-action-id.ts +52 -28
- package/src/vite/plugins/expose-ids/router-transform.ts +20 -3
- package/src/vite/plugins/expose-internal-ids.ts +516 -486
- package/src/vite/plugins/performance-tracks.ts +17 -9
- package/src/vite/plugins/use-cache-transform.ts +56 -43
- package/src/vite/plugins/version-injector.ts +37 -11
- package/src/vite/rango.ts +49 -14
- package/src/vite/router-discovery.ts +498 -52
- package/src/vite/utils/banner.ts +1 -1
- package/src/vite/utils/package-resolution.ts +41 -1
- package/src/vite/utils/prerender-utils.ts +5 -4
|
@@ -35,7 +35,10 @@ import {
|
|
|
35
35
|
type DiscoveryState,
|
|
36
36
|
type PluginOptions,
|
|
37
37
|
} from "./discovery/state.js";
|
|
38
|
-
import {
|
|
38
|
+
import {
|
|
39
|
+
consumeSelfGenWrite,
|
|
40
|
+
peekSelfGenWrite,
|
|
41
|
+
} from "./discovery/self-gen-tracking.js";
|
|
39
42
|
import { discoverRouters } from "./discovery/discover-routers.js";
|
|
40
43
|
import {
|
|
41
44
|
writeCombinedRouteTypesWithTracking,
|
|
@@ -47,7 +50,14 @@ import {
|
|
|
47
50
|
generatePerRouterModule,
|
|
48
51
|
} from "./discovery/virtual-module-codegen.js";
|
|
49
52
|
import { postprocessBundle } from "./discovery/bundle-postprocess.js";
|
|
53
|
+
import { createDiscoveryGate } from "./discovery/gate-state.js";
|
|
50
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);
|
|
51
61
|
|
|
52
62
|
export { VIRTUAL_ROUTES_MANIFEST_ID };
|
|
53
63
|
|
|
@@ -354,6 +364,17 @@ export function createRouterDiscoveryPlugin(
|
|
|
354
364
|
resolveDiscovery = resolve;
|
|
355
365
|
});
|
|
356
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
|
+
|
|
357
378
|
// Compute dev server origin from resolved URLs (preferred) or config port (fallback).
|
|
358
379
|
// Called after discovery (or in the load hook) when the server may be listening.
|
|
359
380
|
const getDevServerOrigin = () =>
|
|
@@ -376,10 +397,103 @@ export function createRouterDiscoveryPlugin(
|
|
|
376
397
|
releaseBuildEnv(s).catch(() => {});
|
|
377
398
|
});
|
|
378
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
|
+
|
|
379
439
|
async function getOrCreateTempServer(): Promise<any | null> {
|
|
380
|
-
if
|
|
381
|
-
|
|
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
|
+
}
|
|
382
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
|
+
);
|
|
383
497
|
try {
|
|
384
498
|
prerenderTempServer = await createTempRscServer(s, {
|
|
385
499
|
cacheDir: "node_modules/.vite_prerender",
|
|
@@ -387,14 +501,17 @@ export function createRouterDiscoveryPlugin(
|
|
|
387
501
|
|
|
388
502
|
const tempRscEnv = (prerenderTempServer.environments as any)?.rsc;
|
|
389
503
|
if (tempRscEnv?.runner) {
|
|
390
|
-
await tempRscEnv
|
|
391
|
-
const serverMod = await tempRscEnv.runner.import(
|
|
392
|
-
"@rangojs/router/server",
|
|
393
|
-
);
|
|
394
|
-
prerenderNodeRegistry = serverMod.RouterRegistry;
|
|
504
|
+
await importEntryAndRegistry(tempRscEnv);
|
|
395
505
|
return tempRscEnv;
|
|
396
506
|
}
|
|
507
|
+
debugDiscovery?.(
|
|
508
|
+
"getOrCreateTempServer: tempRscEnv.runner unavailable",
|
|
509
|
+
);
|
|
397
510
|
} catch (err: any) {
|
|
511
|
+
debugDiscovery?.(
|
|
512
|
+
"getOrCreateTempServer: FAILED message=%s",
|
|
513
|
+
err.message,
|
|
514
|
+
);
|
|
398
515
|
console.warn(
|
|
399
516
|
`[rsc-router] Failed to create temp runner: ${err.message}`,
|
|
400
517
|
);
|
|
@@ -402,24 +519,137 @@ export function createRouterDiscoveryPlugin(
|
|
|
402
519
|
return null;
|
|
403
520
|
}
|
|
404
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
|
+
|
|
405
620
|
const discover = async () => {
|
|
621
|
+
const discoverStart = performance.now();
|
|
406
622
|
const rscEnv = (server.environments as any)?.rsc;
|
|
407
623
|
if (!rscEnv?.runner) {
|
|
408
624
|
// Cloudflare dev: no module runner available (workerd-based RSC env).
|
|
409
625
|
// Set devServerOrigin so the virtual module can inject __PRERENDER_DEV_URL
|
|
410
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
|
+
);
|
|
411
631
|
s.devServerOrigin = getDevServerOrigin();
|
|
412
632
|
|
|
413
633
|
// Create a temp Node.js server to run runtime discovery and generate
|
|
414
634
|
// named route types (static parser can't resolve factory calls).
|
|
415
635
|
try {
|
|
416
636
|
// Acquire build-time env bindings for dev prerender
|
|
417
|
-
await
|
|
637
|
+
await timed(debugDiscovery, "acquireBuildEnv", () =>
|
|
638
|
+
acquireBuildEnv(s, viteCommand, viteMode),
|
|
639
|
+
);
|
|
418
640
|
|
|
419
|
-
const tempRscEnv = await
|
|
641
|
+
const tempRscEnv = await timed(
|
|
642
|
+
debugDiscovery,
|
|
643
|
+
"getOrCreateTempServer",
|
|
644
|
+
() => getOrCreateTempServer(),
|
|
645
|
+
);
|
|
420
646
|
if (tempRscEnv) {
|
|
421
|
-
await discoverRouters(
|
|
422
|
-
|
|
647
|
+
await timed(debugDiscovery, "discoverRouters (cloudflare)", () =>
|
|
648
|
+
discoverRouters(s, tempRscEnv),
|
|
649
|
+
);
|
|
650
|
+
timedSync(debugDiscovery, "writeRouteTypesFiles", () =>
|
|
651
|
+
writeRouteTypesFiles(s),
|
|
652
|
+
);
|
|
423
653
|
}
|
|
424
654
|
} catch (err: any) {
|
|
425
655
|
console.warn(
|
|
@@ -427,24 +657,35 @@ export function createRouterDiscoveryPlugin(
|
|
|
427
657
|
);
|
|
428
658
|
}
|
|
429
659
|
|
|
660
|
+
debugDiscovery?.(
|
|
661
|
+
"dev discovery done (%sms)",
|
|
662
|
+
(performance.now() - discoverStart).toFixed(1),
|
|
663
|
+
);
|
|
430
664
|
resolveDiscovery!();
|
|
431
665
|
return;
|
|
432
666
|
}
|
|
433
667
|
|
|
434
668
|
try {
|
|
435
669
|
// Acquire build-time env bindings for dev prerender (Node.js path)
|
|
436
|
-
|
|
670
|
+
debugDiscovery?.("dev: node path start");
|
|
671
|
+
await timed(debugDiscovery, "acquireBuildEnv", () =>
|
|
672
|
+
acquireBuildEnv(s, viteCommand, viteMode),
|
|
673
|
+
);
|
|
437
674
|
|
|
438
675
|
// Set the readiness gate BEFORE discovery so early requests
|
|
439
676
|
// block until manifest is populated
|
|
440
|
-
const serverMod = await
|
|
441
|
-
|
|
677
|
+
const serverMod = await timed(
|
|
678
|
+
debugDiscovery,
|
|
679
|
+
"import @rangojs/router/server",
|
|
680
|
+
() => rscEnv.runner.import("@rangojs/router/server"),
|
|
442
681
|
);
|
|
443
682
|
if (serverMod?.setManifestReadyPromise) {
|
|
444
683
|
serverMod.setManifestReadyPromise(discoveryPromise);
|
|
445
684
|
}
|
|
446
685
|
|
|
447
|
-
await
|
|
686
|
+
await timed(debugDiscovery, "discoverRouters", () =>
|
|
687
|
+
discoverRouters(s, rscEnv),
|
|
688
|
+
);
|
|
448
689
|
|
|
449
690
|
// Store server origin for dev prerender endpoint (virtual module injection)
|
|
450
691
|
s.devServerOrigin = getDevServerOrigin();
|
|
@@ -454,24 +695,36 @@ export function createRouterDiscoveryPlugin(
|
|
|
454
695
|
// routes (e.g. Array.from loops) that the static parser cannot see.
|
|
455
696
|
// writeRouteTypesFiles() only writes when content changes, so this
|
|
456
697
|
// won't cause unnecessary HMR triggers.
|
|
457
|
-
writeRouteTypesFiles(
|
|
698
|
+
timedSync(debugDiscovery, "writeRouteTypesFiles", () =>
|
|
699
|
+
writeRouteTypesFiles(s),
|
|
700
|
+
);
|
|
458
701
|
|
|
459
702
|
// Populate the route map and per-router data in the RSC env
|
|
460
|
-
await propagateDiscoveryState(
|
|
703
|
+
await timed(debugDiscovery, "propagateDiscoveryState", () =>
|
|
704
|
+
propagateDiscoveryState(rscEnv),
|
|
705
|
+
);
|
|
461
706
|
} catch (err: any) {
|
|
462
707
|
console.warn(
|
|
463
708
|
`[rsc-router] Router discovery failed: ${err.message}\n${err.stack}`,
|
|
464
709
|
);
|
|
465
710
|
} finally {
|
|
711
|
+
debugDiscovery?.(
|
|
712
|
+
"dev discovery done (%sms)",
|
|
713
|
+
(performance.now() - discoverStart).toFixed(1),
|
|
714
|
+
);
|
|
466
715
|
resolveDiscovery!();
|
|
467
716
|
}
|
|
468
717
|
};
|
|
469
718
|
|
|
470
719
|
// Schedule after all plugins have finished configureServer.
|
|
471
|
-
//
|
|
472
|
-
|
|
473
|
-
|
|
474
|
-
|
|
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
|
+
);
|
|
475
728
|
|
|
476
729
|
// Dev-mode on-demand prerender endpoint.
|
|
477
730
|
// When workerd hits a prerender route, it fetches this endpoint instead of
|
|
@@ -529,6 +782,17 @@ export function createRouterDiscoveryPlugin(
|
|
|
529
782
|
};
|
|
530
783
|
|
|
531
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
|
+
|
|
532
796
|
if (s.discoveryDone) await s.discoveryDone;
|
|
533
797
|
|
|
534
798
|
const url = new URL(req.url || "/", "http://localhost");
|
|
@@ -536,6 +800,7 @@ export function createRouterDiscoveryPlugin(
|
|
|
536
800
|
if (!pathname) {
|
|
537
801
|
res.statusCode = 400;
|
|
538
802
|
res.end("Missing pathname");
|
|
803
|
+
logResult(400, "missing pathname");
|
|
539
804
|
return;
|
|
540
805
|
}
|
|
541
806
|
|
|
@@ -559,6 +824,7 @@ export function createRouterDiscoveryPlugin(
|
|
|
559
824
|
);
|
|
560
825
|
res.statusCode = 500;
|
|
561
826
|
res.end(`Prerender handler error: ${err.message}`);
|
|
827
|
+
logResult(500, "module refresh failed");
|
|
562
828
|
return;
|
|
563
829
|
}
|
|
564
830
|
} else {
|
|
@@ -577,6 +843,7 @@ export function createRouterDiscoveryPlugin(
|
|
|
577
843
|
if (!registry || registry.size === 0) {
|
|
578
844
|
res.statusCode = 503;
|
|
579
845
|
res.end("Prerender runner not available");
|
|
846
|
+
logResult(503, "no registry");
|
|
580
847
|
return;
|
|
581
848
|
}
|
|
582
849
|
|
|
@@ -615,6 +882,7 @@ export function createRouterDiscoveryPlugin(
|
|
|
615
882
|
payload = { segments: result.segments, handles: result.handles };
|
|
616
883
|
}
|
|
617
884
|
res.end(JSON.stringify(payload));
|
|
885
|
+
logResult(200, `match ${result.routeName}`);
|
|
618
886
|
return;
|
|
619
887
|
} catch (err: any) {
|
|
620
888
|
console.warn(
|
|
@@ -625,6 +893,7 @@ export function createRouterDiscoveryPlugin(
|
|
|
625
893
|
|
|
626
894
|
res.statusCode = 404;
|
|
627
895
|
res.end("No prerender match");
|
|
896
|
+
logResult(404, "no match");
|
|
628
897
|
});
|
|
629
898
|
|
|
630
899
|
// Watch url module and router files for changes and regenerate named-routes.gen.ts.
|
|
@@ -667,45 +936,117 @@ export function createRouterDiscoveryPlugin(
|
|
|
667
936
|
|
|
668
937
|
// Re-run runtime discovery so factory-generated routes that the
|
|
669
938
|
// static parser cannot see are refreshed after source changes.
|
|
670
|
-
|
|
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.
|
|
671
942
|
const refreshRuntimeDiscovery = async () => {
|
|
672
943
|
const rscEnv = (server.environments as any)?.rsc;
|
|
673
|
-
|
|
674
|
-
|
|
675
|
-
|
|
676
|
-
|
|
677
|
-
|
|
678
|
-
|
|
679
|
-
|
|
680
|
-
|
|
681
|
-
|
|
682
|
-
);
|
|
683
|
-
|
|
684
|
-
|
|
685
|
-
|
|
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
|
+
});
|
|
686
1001
|
};
|
|
687
1002
|
|
|
688
1003
|
const scheduleRouteRegeneration = () => {
|
|
689
1004
|
clearTimeout(routeChangeTimer);
|
|
690
1005
|
routeChangeTimer = setTimeout(() => {
|
|
691
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;
|
|
692
1011
|
try {
|
|
693
|
-
|
|
694
|
-
|
|
695
|
-
|
|
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
|
+
}
|
|
696
1029
|
}
|
|
697
1030
|
} catch (err: any) {
|
|
698
1031
|
console.error(
|
|
699
1032
|
`[rsc-router] Route regeneration error: ${err.message}`,
|
|
700
1033
|
);
|
|
701
1034
|
}
|
|
1035
|
+
debugDiscovery?.(
|
|
1036
|
+
"watcher: regenerated gen files (%sms)",
|
|
1037
|
+
(performance.now() - regenStart).toFixed(1),
|
|
1038
|
+
);
|
|
702
1039
|
// Async: re-run runtime discovery to refresh factory-generated
|
|
703
|
-
// routes that the static parser cannot resolve.
|
|
1040
|
+
// routes that the static parser cannot resolve. Resolves the
|
|
1041
|
+
// discovery gate when complete.
|
|
704
1042
|
if (s.perRouterManifests.length > 0) {
|
|
705
1043
|
refreshRuntimeDiscovery().catch((err: any) => {
|
|
706
1044
|
console.warn(
|
|
707
1045
|
`[rsc-router] Runtime re-discovery error: ${err.message}`,
|
|
708
1046
|
);
|
|
1047
|
+
// Even on error, unblock the gate so workerd's reload
|
|
1048
|
+
// doesn't hang indefinitely against the previous manifest.
|
|
1049
|
+
resolveDiscoveryGate();
|
|
709
1050
|
});
|
|
710
1051
|
}
|
|
711
1052
|
}, 100);
|
|
@@ -733,6 +1074,12 @@ export function createRouterDiscoveryPlugin(
|
|
|
733
1074
|
const hasUrls = source.includes("urls(");
|
|
734
1075
|
const hasCreateRouter = /\bcreateRouter\s*[<(]/.test(source);
|
|
735
1076
|
if (!hasUrls && !hasCreateRouter) return;
|
|
1077
|
+
debugDiscovery?.(
|
|
1078
|
+
"watcher: %s matches (urls=%s, router=%s)",
|
|
1079
|
+
filePath,
|
|
1080
|
+
hasUrls,
|
|
1081
|
+
hasCreateRouter,
|
|
1082
|
+
);
|
|
736
1083
|
// Invalidate cache when a router file changes (new router added/removed)
|
|
737
1084
|
if (hasCreateRouter) {
|
|
738
1085
|
const nestedRouterConflict = findNestedRouterConflict([
|
|
@@ -747,6 +1094,17 @@ export function createRouterDiscoveryPlugin(
|
|
|
747
1094
|
}
|
|
748
1095
|
s.cachedRouterFiles = undefined;
|
|
749
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
|
+
}
|
|
750
1108
|
scheduleRouteRegeneration();
|
|
751
1109
|
} catch {
|
|
752
1110
|
// Ignore read errors for deleted/moved files
|
|
@@ -777,13 +1135,23 @@ export function createRouterDiscoveryPlugin(
|
|
|
777
1135
|
async buildStart() {
|
|
778
1136
|
if (!s.isBuildMode) return;
|
|
779
1137
|
// Only run once across environment builds
|
|
780
|
-
if (s.mergedRouteManifest !== null)
|
|
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 ?? "?");
|
|
781
1147
|
resetStagedBuildAssets(s.projectRoot);
|
|
782
1148
|
s.prerenderManifestEntries = null;
|
|
783
1149
|
s.staticManifestEntries = null;
|
|
784
1150
|
|
|
785
1151
|
// Acquire build-time env bindings if configured
|
|
786
|
-
await
|
|
1152
|
+
await timed(debugDiscovery, "build acquireBuildEnv", () =>
|
|
1153
|
+
acquireBuildEnv(s, viteCommand, viteMode),
|
|
1154
|
+
);
|
|
787
1155
|
|
|
788
1156
|
let tempServer: any = null;
|
|
789
1157
|
// Signal to user-space code (e.g. reverse.ts) that build-time discovery
|
|
@@ -792,7 +1160,11 @@ export function createRouterDiscoveryPlugin(
|
|
|
792
1160
|
// between the vite plugin and user code loaded via runner.import().
|
|
793
1161
|
(globalThis as any).__rscRouterDiscoveryActive = true;
|
|
794
1162
|
try {
|
|
795
|
-
tempServer = await
|
|
1163
|
+
tempServer = await timed(
|
|
1164
|
+
debugDiscovery,
|
|
1165
|
+
"build createTempRscServer",
|
|
1166
|
+
() => createTempRscServer(s, { forceBuild: true }),
|
|
1167
|
+
);
|
|
796
1168
|
|
|
797
1169
|
const rscEnv = (tempServer.environments as any)?.rsc;
|
|
798
1170
|
if (!rscEnv?.runner) {
|
|
@@ -812,11 +1184,15 @@ export function createRouterDiscoveryPlugin(
|
|
|
812
1184
|
s.resolvedStaticModules = tempIdsPlugin.api.staticHandlerModules;
|
|
813
1185
|
}
|
|
814
1186
|
|
|
815
|
-
await
|
|
1187
|
+
await timed(debugDiscovery, "build discoverRouters", () =>
|
|
1188
|
+
discoverRouters(s, rscEnv),
|
|
1189
|
+
);
|
|
816
1190
|
// Update named-routes.gen.ts from runtime discovery.
|
|
817
1191
|
// The runtime manifest includes dynamically generated routes
|
|
818
1192
|
// that the static parser cannot extract from source code.
|
|
819
|
-
writeRouteTypesFiles(
|
|
1193
|
+
timedSync(debugDiscovery, "build writeRouteTypesFiles", () =>
|
|
1194
|
+
writeRouteTypesFiles(s),
|
|
1195
|
+
);
|
|
820
1196
|
} catch (err: any) {
|
|
821
1197
|
// Extract the user source file from the stack trace (skip internal frames)
|
|
822
1198
|
const sourceFile = err.stack
|
|
@@ -841,9 +1217,44 @@ export function createRouterDiscoveryPlugin(
|
|
|
841
1217
|
} finally {
|
|
842
1218
|
delete (globalThis as any).__rscRouterDiscoveryActive;
|
|
843
1219
|
if (tempServer) {
|
|
844
|
-
await tempServer.close()
|
|
1220
|
+
await timed(debugDiscovery, "build tempServer.close", () =>
|
|
1221
|
+
tempServer.close(),
|
|
1222
|
+
);
|
|
845
1223
|
}
|
|
846
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 [];
|
|
847
1258
|
}
|
|
848
1259
|
},
|
|
849
1260
|
|
|
@@ -867,19 +1278,38 @@ export function createRouterDiscoveryPlugin(
|
|
|
867
1278
|
// This is critical for Cloudflare dev where the worker runs in a separate
|
|
868
1279
|
// Miniflare process and can only receive manifest data via the virtual module.
|
|
869
1280
|
if (s.discoveryDone) {
|
|
870
|
-
await
|
|
1281
|
+
await timed(
|
|
1282
|
+
debugRoutes,
|
|
1283
|
+
"await discoveryDone (manifest)",
|
|
1284
|
+
() => s.discoveryDone,
|
|
1285
|
+
);
|
|
871
1286
|
}
|
|
872
|
-
|
|
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;
|
|
873
1294
|
}
|
|
874
1295
|
// Per-router virtual modules: pure data exports (no side effects).
|
|
875
1296
|
// ensureRouterManifest() imports the module and stores the data.
|
|
876
1297
|
const perRouterPrefix = "\0" + VIRTUAL_ROUTES_MANIFEST_ID + "/";
|
|
877
1298
|
if (id.startsWith(perRouterPrefix)) {
|
|
878
1299
|
if (s.discoveryDone) {
|
|
879
|
-
await
|
|
1300
|
+
await timed(
|
|
1301
|
+
debugRoutes,
|
|
1302
|
+
"await discoveryDone (per-router)",
|
|
1303
|
+
() => s.discoveryDone,
|
|
1304
|
+
);
|
|
880
1305
|
}
|
|
881
1306
|
const routerId = id.slice(perRouterPrefix.length);
|
|
882
|
-
|
|
1307
|
+
const code = await timed(
|
|
1308
|
+
debugRoutes,
|
|
1309
|
+
`generatePerRouterModule ${routerId}`,
|
|
1310
|
+
() => generatePerRouterModule(s, routerId),
|
|
1311
|
+
);
|
|
1312
|
+
return code;
|
|
883
1313
|
}
|
|
884
1314
|
// virtual:rsc-router/prerender-paths load handler removed
|
|
885
1315
|
return null;
|
|
@@ -889,6 +1319,7 @@ export function createRouterDiscoveryPlugin(
|
|
|
889
1319
|
// Used by closeBundle for handler code eviction and prerender data injection.
|
|
890
1320
|
generateBundle(_options: any, bundle: any) {
|
|
891
1321
|
if (this.environment?.name !== "rsc") return;
|
|
1322
|
+
const genStart = debugBuild ? performance.now() : 0;
|
|
892
1323
|
|
|
893
1324
|
// Record RSC entry chunk filename for closeBundle injection
|
|
894
1325
|
for (const [fileName, chunk] of Object.entries(bundle) as [
|
|
@@ -901,8 +1332,13 @@ export function createRouterDiscoveryPlugin(
|
|
|
901
1332
|
}
|
|
902
1333
|
}
|
|
903
1334
|
|
|
904
|
-
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
|
+
);
|
|
905
1340
|
return;
|
|
1341
|
+
}
|
|
906
1342
|
|
|
907
1343
|
// Clear maps at the start of each RSC generateBundle pass.
|
|
908
1344
|
// Vite 6 multi-environment builds run RSC twice (analysis + production);
|
|
@@ -957,6 +1393,14 @@ export function createRouterDiscoveryPlugin(
|
|
|
957
1393
|
}
|
|
958
1394
|
}
|
|
959
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
|
+
);
|
|
960
1404
|
},
|
|
961
1405
|
|
|
962
1406
|
// Build-time pre-rendering: evict handler code and inject collected prerender data.
|
|
@@ -970,7 +1414,9 @@ export function createRouterDiscoveryPlugin(
|
|
|
970
1414
|
// Only run for the RSC environment — other environments (client, ssr) have
|
|
971
1415
|
// no prerender/static data to process and would just do redundant file I/O.
|
|
972
1416
|
if (this.environment && this.environment.name !== "rsc") return;
|
|
973
|
-
postprocessBundle(
|
|
1417
|
+
timedSync(debugBuild, "closeBundle postprocessBundle", () =>
|
|
1418
|
+
postprocessBundle(s),
|
|
1419
|
+
);
|
|
974
1420
|
},
|
|
975
1421
|
},
|
|
976
1422
|
};
|