@rangojs/router 0.0.0-experimental.32 → 0.0.0-experimental.3232cd17
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/AGENTS.md +4 -0
- package/README.md +198 -44
- package/dist/bin/rango.js +287 -105
- package/dist/testing/vitest.js +82 -0
- package/dist/vite/index.js +3248 -1117
- package/dist/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/package.json +73 -21
- package/skills/api-client/SKILL.md +211 -0
- package/skills/breadcrumbs/SKILL.md +107 -1
- package/skills/bundle-analysis/SKILL.md +159 -0
- package/skills/cache-guide/SKILL.md +245 -21
- package/skills/caching/SKILL.md +302 -6
- package/skills/composability/SKILL.md +27 -2
- package/skills/css/SKILL.md +76 -0
- package/skills/document-cache/SKILL.md +78 -55
- package/skills/handler-use/SKILL.md +364 -0
- package/skills/hooks/SKILL.md +270 -30
- package/skills/host-router/SKILL.md +82 -22
- package/skills/i18n/SKILL.md +276 -0
- package/skills/intercept/SKILL.md +49 -5
- package/skills/layout/SKILL.md +35 -9
- package/skills/links/SKILL.md +249 -17
- package/skills/loader/SKILL.md +294 -30
- package/skills/middleware/SKILL.md +52 -13
- package/skills/migrate-nextjs/SKILL.md +584 -0
- package/skills/migrate-react-router/SKILL.md +769 -0
- package/skills/mime-routes/SKILL.md +27 -0
- package/skills/observability/SKILL.md +137 -0
- package/skills/parallel/SKILL.md +203 -7
- package/skills/prerender/SKILL.md +123 -100
- package/skills/rango/SKILL.md +250 -22
- package/skills/react-compiler/SKILL.md +168 -0
- package/skills/response-routes/SKILL.md +122 -47
- package/skills/route/SKILL.md +97 -5
- package/skills/router-setup/SKILL.md +90 -5
- package/skills/server-actions/SKILL.md +775 -0
- package/skills/streams-and-websockets/SKILL.md +283 -0
- package/skills/tailwind/SKILL.md +27 -3
- package/skills/testing/SKILL.md +129 -0
- package/skills/testing/bindings.md +89 -0
- package/skills/testing/cache-prerender.md +124 -0
- package/skills/testing/client-components.md +122 -0
- package/skills/testing/e2e-parity.md +125 -0
- package/skills/testing/flight.md +92 -0
- package/skills/testing/handles.md +129 -0
- package/skills/testing/loader.md +128 -0
- package/skills/testing/middleware.md +99 -0
- package/skills/testing/render-handler.md +121 -0
- package/skills/testing/response-routes.md +95 -0
- package/skills/testing/reverse-and-types.md +84 -0
- package/skills/testing/server-actions.md +107 -0
- package/skills/testing/server-tree.md +128 -0
- package/skills/testing/setup.md +120 -0
- package/skills/typesafety/SKILL.md +329 -27
- package/skills/use-cache/SKILL.md +36 -5
- package/skills/view-transitions/SKILL.md +294 -0
- package/src/__augment-tests__/augment.ts +81 -0
- package/src/__augment-tests__/augmented.check.ts +116 -0
- package/src/__internal.ts +67 -40
- package/src/browser/action-coordinator.ts +53 -36
- package/src/browser/action-fence.ts +47 -0
- package/src/browser/app-shell.ts +39 -0
- package/src/browser/app-version.ts +14 -0
- package/src/browser/cookie-name.ts +140 -0
- package/src/browser/event-controller.ts +86 -147
- package/src/browser/history-state.ts +21 -0
- package/src/browser/index.ts +3 -3
- package/src/browser/invalidate-client-cache.ts +52 -0
- package/src/browser/link-interceptor.ts +4 -0
- package/src/browser/navigation-bridge.ts +148 -19
- package/src/browser/navigation-client.ts +187 -67
- package/src/browser/navigation-store-handle.ts +38 -0
- package/src/browser/navigation-store.ts +76 -67
- package/src/browser/navigation-transaction.ts +18 -66
- package/src/browser/partial-update.ts +123 -94
- package/src/browser/prefetch/cache.ts +214 -36
- package/src/browser/prefetch/fetch.ts +260 -38
- package/src/browser/prefetch/policy.ts +6 -0
- package/src/browser/prefetch/queue.ts +126 -20
- package/src/browser/prefetch/resource-ready.ts +77 -0
- package/src/browser/rango-state.ts +158 -76
- package/src/browser/react/Link.tsx +93 -11
- package/src/browser/react/NavigationProvider.tsx +115 -34
- package/src/browser/react/ScrollRestoration.tsx +10 -6
- package/src/browser/react/context.ts +7 -2
- package/src/browser/react/filter-segment-order.ts +49 -7
- package/src/browser/react/index.ts +0 -48
- package/src/browser/react/location-state-shared.ts +166 -8
- package/src/browser/react/location-state.ts +39 -14
- package/src/browser/react/use-action.ts +6 -15
- package/src/browser/react/use-handle.ts +23 -69
- package/src/browser/react/use-link-status.ts +0 -4
- package/src/browser/react/use-navigation.ts +22 -5
- package/src/browser/react/use-params.ts +20 -10
- package/src/browser/react/use-reverse.ts +106 -0
- package/src/browser/react/use-router.ts +46 -11
- package/src/browser/react/use-search-params.ts +0 -5
- package/src/browser/react/use-segments.ts +11 -21
- package/src/browser/response-adapter.ts +52 -1
- package/src/browser/rsc-router.tsx +215 -76
- package/src/browser/scroll-restoration.ts +46 -39
- package/src/browser/segment-reconciler.ts +36 -9
- package/src/browser/segment-structure-assert.ts +2 -2
- package/src/browser/server-action-bridge.ts +176 -50
- package/src/browser/types.ts +95 -11
- package/src/browser/validate-redirect-origin.ts +43 -16
- package/src/build/collect-fallback-refs.ts +107 -0
- package/src/build/generate-manifest.ts +65 -40
- package/src/build/generate-route-types.ts +5 -0
- package/src/build/index.ts +8 -2
- package/src/build/prefix-tree-utils.ts +123 -0
- package/src/build/route-trie.ts +137 -32
- package/src/build/route-types/codegen.ts +4 -4
- package/src/build/route-types/include-resolution.ts +9 -2
- package/src/build/route-types/param-extraction.ts +6 -3
- package/src/build/route-types/per-module-writer.ts +7 -4
- package/src/build/route-types/router-processing.ts +278 -96
- package/src/build/route-types/scan-filter.ts +9 -2
- package/src/build/route-types/source-scan.ts +118 -0
- package/src/build/runtime-discovery.ts +9 -20
- package/src/cache/cache-error.ts +104 -0
- package/src/cache/cache-policy.ts +68 -28
- package/src/cache/cache-runtime.ts +149 -43
- package/src/cache/cache-scope.ts +148 -81
- package/src/cache/cache-tag.ts +98 -0
- package/src/cache/cf/cf-cache-store.ts +2550 -93
- package/src/cache/cf/index.ts +11 -17
- package/src/cache/document-cache.ts +78 -27
- package/src/cache/handle-snapshot.ts +63 -0
- package/src/cache/index.ts +23 -20
- package/src/cache/memory-segment-store.ts +136 -37
- package/src/cache/profile-registry.ts +6 -30
- package/src/cache/read-through-swr.ts +41 -11
- package/src/cache/segment-codec.ts +0 -16
- package/src/cache/tag-invalidation.ts +230 -0
- package/src/cache/taint.ts +55 -0
- package/src/cache/types.ts +33 -100
- package/src/cache/vercel/index.ts +11 -0
- package/src/cache/vercel/vercel-cache-store.ts +799 -0
- package/src/client.rsc.tsx +6 -21
- package/src/client.tsx +108 -290
- package/src/component-utils.ts +19 -0
- package/src/context-var.ts +84 -2
- package/src/debug.ts +2 -2
- package/src/decode-loader-results.ts +36 -0
- package/src/defer.ts +196 -0
- package/src/deps/ssr.ts +0 -1
- package/src/errors.ts +30 -4
- package/src/handle.ts +70 -22
- package/src/handles/MetaTags.tsx +0 -14
- package/src/handles/breadcrumbs.ts +16 -5
- package/src/handles/meta.ts +0 -39
- package/src/host/cookie-handler.ts +0 -36
- package/src/host/errors.ts +0 -24
- package/src/host/index.ts +8 -2
- package/src/host/pattern-matcher.ts +7 -50
- package/src/host/router.ts +107 -99
- package/src/host/testing.ts +40 -27
- package/src/host/types.ts +37 -4
- package/src/host/utils.ts +1 -1
- package/src/href-client.ts +137 -22
- package/src/index.rsc.ts +52 -26
- package/src/index.ts +100 -38
- package/src/internal-debug.ts +2 -4
- package/src/loader-store.ts +500 -0
- package/src/loader.rsc.ts +20 -13
- package/src/loader.ts +12 -11
- package/src/missing-id-error.ts +68 -0
- package/src/network-error-thrower.tsx +1 -6
- package/src/outlet-context.ts +1 -1
- package/src/outlet-provider.tsx +1 -5
- package/src/prerender/param-hash.ts +10 -11
- package/src/prerender/store.ts +37 -41
- package/src/prerender.ts +198 -82
- package/src/redirect-origin.ts +100 -0
- package/src/response-utils.ts +37 -0
- package/src/reverse.ts +65 -15
- package/src/root-error-boundary.tsx +1 -19
- package/src/route-content-wrapper.tsx +7 -72
- package/src/route-definition/dsl-helpers.ts +437 -274
- package/src/route-definition/helper-factories.ts +29 -139
- package/src/route-definition/helpers-types.ts +113 -37
- package/src/route-definition/index.ts +3 -0
- package/src/route-definition/redirect.ts +52 -10
- package/src/route-definition/resolve-handler-use.ts +161 -0
- package/src/route-definition/use-item-types.ts +32 -0
- package/src/route-map-builder.ts +7 -17
- package/src/route-types.ts +37 -41
- package/src/router/basename.ts +14 -0
- package/src/router/content-negotiation.ts +108 -9
- package/src/router/error-handling.ts +13 -17
- package/src/router/find-match.ts +45 -22
- package/src/router/handler-context.ts +83 -41
- package/src/router/intercept-resolution.ts +25 -23
- package/src/router/lazy-includes.ts +19 -53
- package/src/router/loader-resolution.ts +213 -30
- package/src/router/logging.ts +5 -8
- package/src/router/manifest.ts +49 -45
- package/src/router/match-api.ts +120 -204
- package/src/router/match-context.ts +0 -22
- package/src/router/match-handlers.ts +58 -58
- package/src/router/match-middleware/background-revalidation.ts +27 -6
- package/src/router/match-middleware/cache-lookup.ts +205 -249
- package/src/router/match-middleware/cache-store.ts +45 -32
- package/src/router/match-middleware/intercept-resolution.ts +8 -28
- package/src/router/match-middleware/segment-resolution.ts +52 -18
- package/src/router/match-pipelines.ts +1 -42
- package/src/router/match-result.ts +104 -40
- package/src/router/metrics.ts +5 -34
- package/src/router/middleware-types.ts +13 -142
- package/src/router/middleware.ts +173 -143
- package/src/router/navigation-snapshot.ts +131 -0
- package/src/router/params-util.ts +23 -0
- package/src/router/pattern-matching.ts +109 -63
- package/src/router/prerender-match.ts +190 -54
- package/src/router/preview-match.ts +32 -102
- package/src/router/request-classification.ts +276 -0
- package/src/router/revalidation.ts +63 -55
- package/src/router/route-snapshot.ts +244 -0
- package/src/router/router-context.ts +6 -28
- package/src/router/router-interfaces.ts +100 -35
- package/src/router/router-options.ts +91 -11
- package/src/router/router-registry.ts +2 -5
- package/src/router/segment-resolution/fresh.ts +242 -75
- package/src/router/segment-resolution/helpers.ts +63 -24
- package/src/router/segment-resolution/loader-cache.ts +41 -37
- package/src/router/segment-resolution/revalidation.ts +456 -372
- package/src/router/segment-resolution/static-store.ts +19 -5
- package/src/router/segment-resolution/streamed-handler-telemetry.ts +52 -0
- package/src/router/segment-resolution/view-transition-default.ts +36 -0
- package/src/router/segment-resolution.ts +4 -1
- package/src/router/segment-wrappers.ts +2 -3
- package/src/router/state-cookie-name.ts +33 -0
- package/src/router/substitute-pattern-params.ts +56 -0
- package/src/router/telemetry-otel.ts +0 -20
- package/src/router/telemetry.ts +96 -19
- package/src/router/timeout.ts +0 -20
- package/src/router/trie-matching.ts +91 -46
- package/src/router/types.ts +10 -63
- package/src/router/url-params.ts +44 -0
- package/src/router.ts +134 -43
- package/src/rsc/handler-context.ts +3 -2
- package/src/rsc/handler.ts +492 -383
- package/src/rsc/helpers.ts +162 -46
- package/src/rsc/index.ts +1 -1
- package/src/rsc/json-route-result.ts +38 -0
- package/src/rsc/loader-fetch.ts +23 -3
- package/src/rsc/manifest-init.ts +33 -42
- package/src/rsc/origin-guard.ts +39 -25
- package/src/rsc/progressive-enhancement.ts +30 -3
- package/src/rsc/redirect-guard.ts +99 -0
- package/src/rsc/response-error.ts +79 -12
- package/src/rsc/response-route-handler.ts +90 -63
- package/src/rsc/rsc-rendering.ts +56 -54
- package/src/rsc/runtime-warnings.ts +23 -10
- package/src/rsc/server-action.ts +74 -67
- package/src/rsc/ssr-setup.ts +18 -2
- package/src/rsc/types.ts +25 -6
- package/src/runtime-env.ts +18 -0
- package/src/search-params.ts +4 -20
- package/src/segment-content-promise.ts +67 -0
- package/src/segment-loader-promise.ts +134 -0
- package/src/segment-system.tsx +272 -129
- package/src/serialize.ts +243 -0
- package/src/server/context.ts +309 -61
- package/src/server/cookie-store.ts +80 -5
- package/src/server/handle-store.ts +26 -24
- package/src/server/loader-registry.ts +10 -28
- package/src/server/request-context.ts +338 -126
- package/src/ssr/index.tsx +23 -15
- package/src/static-handler.ts +27 -18
- package/src/testing/cache-status.ts +162 -0
- package/src/testing/collect-handle.ts +40 -0
- package/src/testing/dispatch.ts +618 -0
- package/src/testing/dom.entry.ts +22 -0
- package/src/testing/e2e/fixture.ts +188 -0
- package/src/testing/e2e/index.ts +128 -0
- package/src/testing/e2e/matchers.ts +35 -0
- package/src/testing/e2e/page-helpers.ts +272 -0
- package/src/testing/e2e/parity.ts +387 -0
- package/src/testing/e2e/server.ts +195 -0
- package/src/testing/flight-matchers.ts +97 -0
- package/src/testing/flight-normalize.ts +11 -0
- package/src/testing/flight-runtime.d.ts +57 -0
- package/src/testing/flight-tree.ts +682 -0
- package/src/testing/flight.entry.ts +52 -0
- package/src/testing/flight.ts +232 -0
- package/src/testing/generated-routes.ts +183 -0
- package/src/testing/index.ts +99 -0
- package/src/testing/internal/context.ts +348 -0
- package/src/testing/internal/flight-client-globals.ts +30 -0
- package/src/testing/internal/seed-vars.ts +54 -0
- package/src/testing/render-handler.ts +330 -0
- package/src/testing/render-route.tsx +566 -0
- package/src/testing/run-loader.ts +378 -0
- package/src/testing/run-middleware.ts +205 -0
- package/src/testing/vitest-stubs/cloudflare-email.ts +9 -0
- package/src/testing/vitest-stubs/cloudflare-workers.ts +21 -0
- package/src/testing/vitest-stubs/plugin-rsc.ts +16 -0
- package/src/testing/vitest-stubs/version.ts +5 -0
- package/src/testing/vitest.ts +305 -0
- package/src/theme/ThemeProvider.tsx +0 -52
- package/src/theme/ThemeScript.tsx +0 -6
- package/src/theme/constants.ts +0 -12
- package/src/theme/index.ts +0 -7
- package/src/theme/theme-context.ts +1 -5
- package/src/theme/theme-script.ts +0 -14
- package/src/theme/use-theme.ts +0 -3
- package/src/types/boundaries.ts +0 -35
- package/src/types/cache-types.ts +17 -8
- package/src/types/error-types.ts +30 -90
- package/src/types/global-namespace.ts +54 -41
- package/src/types/handler-context.ts +233 -81
- package/src/types/index.ts +1 -10
- package/src/types/loader-types.ts +44 -15
- package/src/types/request-scope.ts +107 -0
- package/src/types/route-config.ts +6 -50
- package/src/types/route-entry.ts +19 -7
- package/src/types/segments.ts +37 -14
- package/src/urls/include-helper.ts +33 -70
- package/src/urls/index.ts +1 -11
- package/src/urls/path-helper-types.ts +58 -11
- package/src/urls/path-helper.ts +57 -111
- package/src/urls/pattern-types.ts +48 -19
- package/src/urls/response-types.ts +25 -22
- package/src/urls/type-extraction.ts +58 -139
- package/src/urls/urls-function.ts +1 -18
- package/src/use-loader.tsx +346 -89
- package/src/vite/debug.ts +185 -0
- package/src/vite/discovery/bundle-postprocess.ts +36 -38
- package/src/vite/discovery/discover-routers.ts +130 -85
- package/src/vite/discovery/discovery-errors.ts +194 -0
- package/src/vite/discovery/gate-state.ts +171 -0
- package/src/vite/discovery/prerender-collection.ts +192 -99
- package/src/vite/discovery/route-types-writer.ts +40 -84
- package/src/vite/discovery/self-gen-tracking.ts +27 -1
- package/src/vite/discovery/state.ts +51 -6
- package/src/vite/discovery/virtual-module-codegen.ts +14 -34
- package/src/vite/index.ts +8 -0
- package/src/vite/plugin-types.ts +187 -69
- package/src/vite/plugins/cjs-to-esm.ts +8 -18
- package/src/vite/plugins/client-ref-dedup.ts +16 -11
- package/src/vite/plugins/client-ref-hashing.ts +28 -15
- package/src/vite/plugins/cloudflare-protocol-loader-hook.d.mts +23 -0
- package/src/vite/plugins/cloudflare-protocol-loader-hook.mjs +76 -0
- package/src/vite/plugins/cloudflare-protocol-stub.ts +194 -0
- package/src/vite/plugins/expose-action-id.ts +49 -98
- package/src/vite/plugins/expose-id-utils.ts +11 -50
- package/src/vite/plugins/expose-ids/export-analysis.ts +76 -34
- package/src/vite/plugins/expose-ids/handler-transform.ts +10 -48
- package/src/vite/plugins/expose-ids/loader-transform.ts +3 -20
- package/src/vite/plugins/expose-ids/router-transform.ts +20 -16
- package/src/vite/plugins/expose-internal-ids.ts +554 -317
- package/src/vite/plugins/performance-tracks.ts +89 -0
- package/src/vite/plugins/refresh-cmd.ts +89 -27
- package/src/vite/plugins/use-cache-transform.ts +73 -83
- package/src/vite/plugins/vercel-output.ts +258 -0
- package/src/vite/plugins/version-injector.ts +21 -25
- package/src/vite/plugins/version-plugin.ts +41 -20
- package/src/vite/plugins/virtual-entries.ts +2 -17
- package/src/vite/rango.ts +257 -289
- package/src/vite/router-discovery.ts +930 -140
- package/src/vite/utils/ast-handler-extract.ts +15 -31
- package/src/vite/utils/banner.ts +4 -4
- package/src/vite/utils/bundle-analysis.ts +10 -15
- package/src/vite/utils/client-chunks.ts +184 -0
- package/src/vite/utils/forward-user-plugins.ts +171 -0
- package/src/vite/utils/manifest-utils.ts +4 -59
- package/src/vite/utils/package-resolution.ts +20 -52
- package/src/vite/utils/prerender-utils.ts +27 -29
- package/src/vite/utils/shared-utils.ts +92 -42
- package/src/browser/action-response-classifier.ts +0 -99
- package/src/browser/react/use-client-cache.ts +0 -58
- package/src/browser/shallow.ts +0 -40
- package/src/handles/index.ts +0 -7
- package/src/router/middleware-cookies.ts +0 -55
|
@@ -10,14 +10,21 @@ 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, register } from "node:module";
|
|
14
|
+
import { pathToFileURL } from "node:url";
|
|
13
15
|
import {
|
|
14
16
|
formatNestedRouterConflictError,
|
|
15
17
|
findNestedRouterConflict,
|
|
16
18
|
findRouterFiles,
|
|
17
19
|
createScanFilter,
|
|
18
20
|
} from "../build/generate-route-types.js";
|
|
21
|
+
import { firstCodeMatchIndex } from "../build/route-types/source-scan.js";
|
|
19
22
|
import { createVersionPlugin } from "./plugins/version-plugin.js";
|
|
20
23
|
import { createVirtualStubPlugin } from "./plugins/virtual-stub-plugin.js";
|
|
24
|
+
import {
|
|
25
|
+
BUILD_ENV_GLOBAL_KEY,
|
|
26
|
+
createCloudflareProtocolStubPlugin,
|
|
27
|
+
} from "./plugins/cloudflare-protocol-stub.js";
|
|
21
28
|
import {
|
|
22
29
|
exposeInternalIds,
|
|
23
30
|
exposeRouterId,
|
|
@@ -30,7 +37,10 @@ import {
|
|
|
30
37
|
type DiscoveryState,
|
|
31
38
|
type PluginOptions,
|
|
32
39
|
} from "./discovery/state.js";
|
|
33
|
-
import {
|
|
40
|
+
import {
|
|
41
|
+
consumeSelfGenWrite,
|
|
42
|
+
peekSelfGenWrite,
|
|
43
|
+
} from "./discovery/self-gen-tracking.js";
|
|
34
44
|
import { discoverRouters } from "./discovery/discover-routers.js";
|
|
35
45
|
import {
|
|
36
46
|
writeCombinedRouteTypesWithTracking,
|
|
@@ -42,10 +52,65 @@ import {
|
|
|
42
52
|
generatePerRouterModule,
|
|
43
53
|
} from "./discovery/virtual-module-codegen.js";
|
|
44
54
|
import { postprocessBundle } from "./discovery/bundle-postprocess.js";
|
|
55
|
+
import { createDiscoveryGate } from "./discovery/gate-state.js";
|
|
45
56
|
import { resetStagedBuildAssets } from "./utils/prerender-utils.js";
|
|
57
|
+
import { resolveRscEntryFromConfig } from "./utils/shared-utils.js";
|
|
58
|
+
import {
|
|
59
|
+
pickForwardedRunnerConfig,
|
|
60
|
+
selectForwardableResolvePlugins,
|
|
61
|
+
} from "./utils/forward-user-plugins.js";
|
|
62
|
+
import { createRangoDebugger, timed, timedSync, NS } from "./debug.js";
|
|
63
|
+
|
|
64
|
+
const debugDiscovery = createRangoDebugger(NS.discovery);
|
|
65
|
+
const debugRoutes = createRangoDebugger(NS.routes);
|
|
66
|
+
const debugBuild = createRangoDebugger(NS.build);
|
|
67
|
+
const debugDev = createRangoDebugger(NS.dev);
|
|
46
68
|
|
|
47
69
|
export { VIRTUAL_ROUTES_MANIFEST_ID };
|
|
48
70
|
|
|
71
|
+
// ============================================================================
|
|
72
|
+
// Node ESM Loader Hook Registration
|
|
73
|
+
// ============================================================================
|
|
74
|
+
|
|
75
|
+
/**
|
|
76
|
+
* Registers a Node ESM loader hook that resolves `cloudflare:*` specifiers
|
|
77
|
+
* to a data: URL stub. Defense-in-depth alongside the Vite transform in
|
|
78
|
+
* `cloudflare-protocol-stub.ts`:
|
|
79
|
+
*
|
|
80
|
+
* - The Vite transform catches `cloudflare:*` imports in modules that flow
|
|
81
|
+
* through Vite's plugin pipeline. That's the vast majority of cases.
|
|
82
|
+
* - The Node loader catches imports in modules that Vite/Rollup externalize
|
|
83
|
+
* (e.g. the `partyserver` package, which has a top-level
|
|
84
|
+
* `import { DurableObject, env } from "cloudflare:workers"` and ships
|
|
85
|
+
* shapes plugin-rsc marks as external). Externalized modules are loaded
|
|
86
|
+
* via Node's native ESM loader, which rejects URL schemes.
|
|
87
|
+
*
|
|
88
|
+
* Registration is process-global and one-shot. The hook only intercepts
|
|
89
|
+
* `cloudflare:*` specifiers; everything else passes through via
|
|
90
|
+
* `nextResolve()`. It runs in a separate worker thread (Node ESM loader
|
|
91
|
+
* architecture), so it can't read the `globalThis[BUILD_ENV_GLOBAL_KEY]`
|
|
92
|
+
* bridge that the Vite transform uses — the stubs served here always
|
|
93
|
+
* return `env = {}`. That's fine because externalized libraries don't
|
|
94
|
+
* typically access `env` at module top level; user source (where real
|
|
95
|
+
* `env` matters at build time) flows through the Vite transform.
|
|
96
|
+
*/
|
|
97
|
+
let loaderHookRegistered = false;
|
|
98
|
+
function ensureCloudflareProtocolLoaderRegistered(): void {
|
|
99
|
+
if (loaderHookRegistered) return;
|
|
100
|
+
loaderHookRegistered = true;
|
|
101
|
+
try {
|
|
102
|
+
register(
|
|
103
|
+
new URL("./plugins/cloudflare-protocol-loader-hook.mjs", import.meta.url),
|
|
104
|
+
);
|
|
105
|
+
} catch (err: any) {
|
|
106
|
+
// register() requires Node 18.19+ / 20.6+. Older Node still has the
|
|
107
|
+
// Vite transform as primary defense.
|
|
108
|
+
console.warn(
|
|
109
|
+
`[rango] Could not register Node ESM loader hook for cloudflare:* imports (${err?.message ?? err}). Falling back to Vite transform only.`,
|
|
110
|
+
);
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
|
|
49
114
|
// ============================================================================
|
|
50
115
|
// Temp Server Factory
|
|
51
116
|
// ============================================================================
|
|
@@ -65,15 +130,32 @@ async function createTempRscServer(
|
|
|
65
130
|
state: DiscoveryState,
|
|
66
131
|
options: { forceBuild?: boolean; cacheDir?: string } = {},
|
|
67
132
|
) {
|
|
133
|
+
// Install the Node ESM loader hook before any module evaluation so
|
|
134
|
+
// `cloudflare:*` specifiers in externalized/loader-delegated modules
|
|
135
|
+
// (e.g. packages plugin-rsc marks as external) resolve to stubs
|
|
136
|
+
// instead of crashing Node's native loader.
|
|
137
|
+
ensureCloudflareProtocolLoaderRegistered();
|
|
68
138
|
const { default: rsc } = await import("@vitejs/plugin-rsc");
|
|
139
|
+
// Mirror the user's resolution config + plugins so discovery (and the
|
|
140
|
+
// prerender/static rendering that shares this runner) resolves modules the
|
|
141
|
+
// same way the real environment does. Falls back to the legacy alias-only
|
|
142
|
+
// behavior if configResolved hasn't populated the parity slice yet.
|
|
143
|
+
const runnerConfig = state.userRunnerConfig;
|
|
144
|
+
const resolveConfig = runnerConfig?.resolve ?? {
|
|
145
|
+
alias: state.userResolveAlias,
|
|
146
|
+
};
|
|
147
|
+
const oxcConfig = runnerConfig?.oxc ?? {
|
|
148
|
+
jsx: { runtime: "automatic", importSource: "react" },
|
|
149
|
+
};
|
|
69
150
|
return createViteServer({
|
|
70
151
|
root: state.projectRoot,
|
|
71
152
|
configFile: false,
|
|
72
153
|
server: { middlewareMode: true },
|
|
73
154
|
appType: "custom",
|
|
74
155
|
logLevel: "silent",
|
|
75
|
-
resolve:
|
|
76
|
-
|
|
156
|
+
resolve: resolveConfig,
|
|
157
|
+
...(runnerConfig?.define ? { define: runnerConfig.define } : {}),
|
|
158
|
+
oxc: oxcConfig as any,
|
|
77
159
|
...(options.cacheDir && { cacheDir: options.cacheDir }),
|
|
78
160
|
plugins: [
|
|
79
161
|
rsc({
|
|
@@ -87,14 +169,124 @@ async function createTempRscServer(
|
|
|
87
169
|
...(options.forceBuild ? [hashClientRefs(state.projectRoot)] : []),
|
|
88
170
|
createVersionPlugin(),
|
|
89
171
|
createVirtualStubPlugin(),
|
|
172
|
+
createCloudflareProtocolStubPlugin(),
|
|
90
173
|
// Dev prerender must use dev-mode IDs (path-based) to match the workerd
|
|
91
174
|
// runtime. forceBuild produces hashed IDs for production bundle consistency.
|
|
92
175
|
exposeInternalIds(options.forceBuild ? { forceBuild: true } : undefined),
|
|
93
176
|
exposeRouterId(),
|
|
177
|
+
// Forwarded user resolution plugins (e.g. vite-tsconfig-paths). Stripped
|
|
178
|
+
// to resolveId/load and placed last so framework resolution runs first;
|
|
179
|
+
// Vite re-sorts by `enforce`, so `enforce: "pre"` resolvers still lead.
|
|
180
|
+
...state.userResolvePlugins,
|
|
94
181
|
],
|
|
95
182
|
});
|
|
96
183
|
}
|
|
97
184
|
|
|
185
|
+
// ============================================================================
|
|
186
|
+
// Build-Time Env Resolution
|
|
187
|
+
// ============================================================================
|
|
188
|
+
|
|
189
|
+
import type {
|
|
190
|
+
BuildEnvOption,
|
|
191
|
+
BuildEnvFactoryContext,
|
|
192
|
+
BuildEnvResult,
|
|
193
|
+
} from "./plugin-types.js";
|
|
194
|
+
|
|
195
|
+
/**
|
|
196
|
+
* Resolve the buildEnv option into a concrete { env, dispose? } result.
|
|
197
|
+
* Handles all four input shapes: false, "auto", factory, plain object.
|
|
198
|
+
*/
|
|
199
|
+
async function resolveBuildEnv(
|
|
200
|
+
option: BuildEnvOption | undefined,
|
|
201
|
+
factoryCtx: BuildEnvFactoryContext,
|
|
202
|
+
): Promise<BuildEnvResult | null> {
|
|
203
|
+
if (!option) return null;
|
|
204
|
+
|
|
205
|
+
if (option === "auto") {
|
|
206
|
+
if (factoryCtx.preset !== "cloudflare") {
|
|
207
|
+
throw new Error(
|
|
208
|
+
'[rango] buildEnv: "auto" is only supported with preset: "cloudflare". ' +
|
|
209
|
+
"Use a factory function or plain object for other presets.",
|
|
210
|
+
);
|
|
211
|
+
}
|
|
212
|
+
try {
|
|
213
|
+
// Resolve wrangler from the user's project root (not the router package)
|
|
214
|
+
const userRequire = createRequire(
|
|
215
|
+
resolve(factoryCtx.root, "package.json"),
|
|
216
|
+
);
|
|
217
|
+
const wranglerPath = userRequire.resolve("wrangler");
|
|
218
|
+
const { getPlatformProxy } = (await import(
|
|
219
|
+
pathToFileURL(wranglerPath).href
|
|
220
|
+
)) as {
|
|
221
|
+
getPlatformProxy: (opts?: any) => Promise<any>;
|
|
222
|
+
};
|
|
223
|
+
const proxy = await getPlatformProxy();
|
|
224
|
+
return {
|
|
225
|
+
env: proxy.env as Record<string, unknown>,
|
|
226
|
+
dispose: proxy.dispose,
|
|
227
|
+
};
|
|
228
|
+
} catch (err: any) {
|
|
229
|
+
throw new Error(
|
|
230
|
+
'[rango] buildEnv: "auto" requires wrangler to be installed.\n' +
|
|
231
|
+
`Install it with: pnpm add -D wrangler\n${err.message}`,
|
|
232
|
+
);
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
if (typeof option === "function") {
|
|
237
|
+
return await option(factoryCtx);
|
|
238
|
+
}
|
|
239
|
+
|
|
240
|
+
// Plain object
|
|
241
|
+
return { env: option };
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
/**
|
|
245
|
+
* Acquire build-time env bindings and store on discovery state.
|
|
246
|
+
* Returns true if env was acquired, false if buildEnv is disabled.
|
|
247
|
+
*/
|
|
248
|
+
async function acquireBuildEnv(
|
|
249
|
+
s: DiscoveryState,
|
|
250
|
+
command: "serve" | "build",
|
|
251
|
+
mode: string,
|
|
252
|
+
): Promise<boolean> {
|
|
253
|
+
const option = s.opts?.buildEnv;
|
|
254
|
+
if (!option) return false;
|
|
255
|
+
|
|
256
|
+
const result = await resolveBuildEnv(option, {
|
|
257
|
+
root: s.projectRoot,
|
|
258
|
+
mode,
|
|
259
|
+
command,
|
|
260
|
+
preset: s.opts?.preset ?? "node",
|
|
261
|
+
});
|
|
262
|
+
if (!result) return false;
|
|
263
|
+
|
|
264
|
+
s.resolvedBuildEnv = result.env;
|
|
265
|
+
s.buildEnvDispose = result.dispose ?? null;
|
|
266
|
+
// Bridge the resolved env into `cloudflare:workers`'s stubbed `env`
|
|
267
|
+
// export so user code that does `import { env } from "cloudflare:workers"`
|
|
268
|
+
// sees the real bindings proxy during discovery + prerender instead of
|
|
269
|
+
// an empty object. The stub reads this global at module-evaluation time.
|
|
270
|
+
(globalThis as Record<string, unknown>)[BUILD_ENV_GLOBAL_KEY] = result.env;
|
|
271
|
+
return true;
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
/**
|
|
275
|
+
* Release build-time env resources and clear state.
|
|
276
|
+
*/
|
|
277
|
+
async function releaseBuildEnv(s: DiscoveryState): Promise<void> {
|
|
278
|
+
if (s.buildEnvDispose) {
|
|
279
|
+
try {
|
|
280
|
+
await s.buildEnvDispose();
|
|
281
|
+
} catch (err: any) {
|
|
282
|
+
console.warn(`[rango] buildEnv dispose failed: ${err.message}`);
|
|
283
|
+
}
|
|
284
|
+
s.buildEnvDispose = null;
|
|
285
|
+
}
|
|
286
|
+
s.resolvedBuildEnv = undefined;
|
|
287
|
+
delete (globalThis as Record<string, unknown>)[BUILD_ENV_GLOBAL_KEY];
|
|
288
|
+
}
|
|
289
|
+
|
|
98
290
|
/**
|
|
99
291
|
* Plugin that discovers router instances at dev/build time via the RSC environment.
|
|
100
292
|
*
|
|
@@ -112,6 +304,8 @@ export function createRouterDiscoveryPlugin(
|
|
|
112
304
|
opts?: PluginOptions,
|
|
113
305
|
): Plugin {
|
|
114
306
|
const s = createDiscoveryState(entryPath, opts);
|
|
307
|
+
let viteCommand: "serve" | "build" = "build";
|
|
308
|
+
let viteMode = "production";
|
|
115
309
|
|
|
116
310
|
return {
|
|
117
311
|
name: "@rangojs/router:discovery",
|
|
@@ -122,58 +316,50 @@ export function createRouterDiscoveryPlugin(
|
|
|
122
316
|
__RANGO_DEBUG__: JSON.stringify(!!process.env.INTERNAL_RANGO_DEBUG),
|
|
123
317
|
},
|
|
124
318
|
};
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
manualChunks(id: string) {
|
|
132
|
-
if (s.resolvedPrerenderModules?.has(id)) {
|
|
133
|
-
return "__prerender-handlers";
|
|
134
|
-
}
|
|
135
|
-
if (s.resolvedStaticModules?.has(id)) {
|
|
136
|
-
return "__static-handlers";
|
|
137
|
-
}
|
|
138
|
-
},
|
|
139
|
-
},
|
|
140
|
-
},
|
|
141
|
-
},
|
|
142
|
-
},
|
|
143
|
-
};
|
|
144
|
-
}
|
|
319
|
+
// Prerender/static handler modules are bundled naturally with the
|
|
320
|
+
// rest of the RSC entry. A previous design forced them into dedicated
|
|
321
|
+
// __prerender-handlers / __static-handlers chunks via manualChunks,
|
|
322
|
+
// but Rollup hoisted all shared dependencies into those chunks,
|
|
323
|
+
// inflating them to ~1 MB with active runtime code. Handler code is
|
|
324
|
+
// evicted in closeBundle regardless of which chunk it lands in.
|
|
145
325
|
return config;
|
|
146
326
|
},
|
|
147
327
|
|
|
148
328
|
configResolved(config) {
|
|
149
329
|
s.projectRoot = config.root;
|
|
330
|
+
// Compile the optional discovery scan filter (glob include/exclude) now
|
|
331
|
+
// that the project root is known. findRouterFiles() below — and the
|
|
332
|
+
// build/HMR rediscovery paths — honor s.scanFilter.
|
|
333
|
+
s.scanFilter = opts?.discovery
|
|
334
|
+
? createScanFilter(s.projectRoot, opts.discovery)
|
|
335
|
+
: undefined;
|
|
150
336
|
s.isBuildMode = config.command === "build";
|
|
337
|
+
viteCommand = config.command as "serve" | "build";
|
|
338
|
+
viteMode = config.mode;
|
|
151
339
|
// Capture user's resolve aliases for the temp server
|
|
152
340
|
s.userResolveAlias = config.resolve.alias;
|
|
341
|
+
// Capture the data-only resolution config (resolve.*, define, oxc) and
|
|
342
|
+
// the user's resolution plugins (resolveId/load) so the discovery temp
|
|
343
|
+
// server resolves modules the same way the real environment does.
|
|
344
|
+
// Without this, both flavors of user resolution are absent during
|
|
345
|
+
// discovery/prerender/static rendering even though they apply at request
|
|
346
|
+
// time: third-party resolvers (e.g. vite-tsconfig-paths, forwarded as
|
|
347
|
+
// plugins) and Vite 8's native resolve.tsconfigPaths (forwarded in the
|
|
348
|
+
// data slice). See utils/forward-user-plugins.ts.
|
|
349
|
+
s.userRunnerConfig = pickForwardedRunnerConfig(config);
|
|
350
|
+
s.userResolvePlugins = selectForwardableResolvePlugins(
|
|
351
|
+
config.plugins as any,
|
|
352
|
+
);
|
|
153
353
|
// Node preset: pick up auto-discovered router path from the config() hook.
|
|
154
354
|
// The auto-discover plugin runs in config() using Vite's resolved root,
|
|
155
355
|
// populating the mutable ref before configResolved fires.
|
|
156
356
|
if (!s.resolvedEntryPath && opts?.routerPathRef?.path) {
|
|
157
357
|
s.resolvedEntryPath = opts.routerPathRef.path;
|
|
158
358
|
}
|
|
159
|
-
// Cloudflare preset:
|
|
160
|
-
// The @cloudflare/vite-plugin reads wrangler config (toml/json/jsonc)
|
|
161
|
-
// and sets optimizeDeps.entries on the RSC environment.
|
|
359
|
+
// Cloudflare preset: entry comes from the resolved RSC env config.
|
|
162
360
|
if (!s.resolvedEntryPath) {
|
|
163
|
-
const
|
|
164
|
-
|
|
165
|
-
if (typeof entries === "string") {
|
|
166
|
-
s.resolvedEntryPath = entries;
|
|
167
|
-
} else if (Array.isArray(entries) && entries.length > 0) {
|
|
168
|
-
s.resolvedEntryPath = entries[0];
|
|
169
|
-
}
|
|
170
|
-
}
|
|
171
|
-
// Compile include/exclude patterns into a scan filter
|
|
172
|
-
if (opts?.include || opts?.exclude) {
|
|
173
|
-
s.scanFilter = createScanFilter(s.projectRoot, {
|
|
174
|
-
include: opts.include,
|
|
175
|
-
exclude: opts.exclude,
|
|
176
|
-
});
|
|
361
|
+
const entry = resolveRscEntryFromConfig(config);
|
|
362
|
+
if (entry) s.resolvedEntryPath = entry;
|
|
177
363
|
}
|
|
178
364
|
// Generate combined named-routes.gen.ts from static source parsing.
|
|
179
365
|
// Runs before the dev server starts so the gen file exists immediately for IDE.
|
|
@@ -212,6 +398,17 @@ export function createRouterDiscoveryPlugin(
|
|
|
212
398
|
resolveDiscovery = resolve;
|
|
213
399
|
});
|
|
214
400
|
|
|
401
|
+
// Manifest-readiness gate + rediscovery scheduler.
|
|
402
|
+
// The virtual:rsc-router/routes-manifest module's `load()` hook
|
|
403
|
+
// awaits `s.discoveryDone`; the gate is reset on each discovery
|
|
404
|
+
// cycle so workerd's HMR reloads block until the new gen file is
|
|
405
|
+
// written. State machine + transitions are extracted into
|
|
406
|
+
// ./discovery/gate-state.ts and unit-tested there — see the
|
|
407
|
+
// module's JSDoc for the four-flag contract.
|
|
408
|
+
const gate = createDiscoveryGate(s, debugDiscovery);
|
|
409
|
+
const beginDiscoveryGate = gate.beginGate;
|
|
410
|
+
const resolveDiscoveryGate = gate.resolveGate;
|
|
411
|
+
|
|
215
412
|
// Compute dev server origin from resolved URLs (preferred) or config port (fallback).
|
|
216
413
|
// Called after discovery (or in the load hook) when the server may be listening.
|
|
217
414
|
const getDevServerOrigin = () =>
|
|
@@ -225,18 +422,113 @@ export function createRouterDiscoveryPlugin(
|
|
|
225
422
|
let prerenderTempServer: any = null;
|
|
226
423
|
let prerenderNodeRegistry: Map<string, any> | null = null;
|
|
227
424
|
|
|
228
|
-
// Clean up the temporary server when the dev server shuts down
|
|
425
|
+
// Clean up the temporary server and build env when the dev server shuts down
|
|
229
426
|
server.httpServer?.on("close", () => {
|
|
230
427
|
if (prerenderTempServer) {
|
|
231
428
|
prerenderTempServer.close().catch(() => {});
|
|
232
429
|
prerenderTempServer = null;
|
|
233
430
|
}
|
|
431
|
+
releaseBuildEnv(s).catch(() => {});
|
|
234
432
|
});
|
|
235
433
|
|
|
434
|
+
// Mirror the build-path contract (the buildStart hook below, which sets
|
|
435
|
+
// __rscRouterDiscoveryActive before running user modules):
|
|
436
|
+
// set __rscRouterDiscoveryActive before running user modules so any
|
|
437
|
+
// module-level router.reverse() calls return a placeholder instead
|
|
438
|
+
// of throwing. The temp Vite server's module runner has its own
|
|
439
|
+
// module context; the flag must be on globalThis to cross that
|
|
440
|
+
// boundary. Cleared in finally so the dev request handlers run with
|
|
441
|
+
// strict reverse() semantics afterwards.
|
|
442
|
+
async function importEntryAndRegistry(tempRscEnv: any): Promise<void> {
|
|
443
|
+
const flagAlreadySet = !!(globalThis as any).__rscRouterDiscoveryActive;
|
|
444
|
+
if (!flagAlreadySet) {
|
|
445
|
+
(globalThis as any).__rscRouterDiscoveryActive = true;
|
|
446
|
+
}
|
|
447
|
+
try {
|
|
448
|
+
debugDiscovery?.(
|
|
449
|
+
"importEntryAndRegistry: importing entry (flag=%s)",
|
|
450
|
+
(globalThis as any).__rscRouterDiscoveryActive ?? false,
|
|
451
|
+
);
|
|
452
|
+
await tempRscEnv.runner.import(s.resolvedEntryPath!);
|
|
453
|
+
debugDiscovery?.(
|
|
454
|
+
"importEntryAndRegistry: entry import OK, fetching RouterRegistry",
|
|
455
|
+
);
|
|
456
|
+
const serverMod = await tempRscEnv.runner.import(
|
|
457
|
+
"@rangojs/router/server",
|
|
458
|
+
);
|
|
459
|
+
prerenderNodeRegistry = serverMod.RouterRegistry;
|
|
460
|
+
debugDiscovery?.(
|
|
461
|
+
"importEntryAndRegistry: registry size=%d",
|
|
462
|
+
prerenderNodeRegistry?.size ?? 0,
|
|
463
|
+
);
|
|
464
|
+
} finally {
|
|
465
|
+
if (!flagAlreadySet) {
|
|
466
|
+
delete (globalThis as any).__rscRouterDiscoveryActive;
|
|
467
|
+
debugDiscovery?.(
|
|
468
|
+
"importEntryAndRegistry: cleared __rscRouterDiscoveryActive",
|
|
469
|
+
);
|
|
470
|
+
}
|
|
471
|
+
}
|
|
472
|
+
}
|
|
473
|
+
|
|
236
474
|
async function getOrCreateTempServer(): Promise<any | null> {
|
|
237
|
-
if
|
|
238
|
-
|
|
475
|
+
// Reuse path: if a temp server is already alive, prefer reusing
|
|
476
|
+
// it over orphaning the existing instance and spinning up a new
|
|
477
|
+
// one. This handles two cases:
|
|
478
|
+
//
|
|
479
|
+
// 1. Steady-state cache hit (cold-start completed, registry
|
|
480
|
+
// cached) — return the env immediately.
|
|
481
|
+
// 2. Recovery from a failed refresh: refreshTempRscEnv() may
|
|
482
|
+
// have invalidated and nulled the registry, then thrown
|
|
483
|
+
// during importEntryAndRegistry. Without reuse, the next
|
|
484
|
+
// call would `createTempRscServer` and overwrite the
|
|
485
|
+
// handle, leaking the previous server. Try to re-import on
|
|
486
|
+
// the existing runner first; only if THAT fails do we
|
|
487
|
+
// close the orphan and create new.
|
|
488
|
+
if (prerenderTempServer) {
|
|
489
|
+
const existingEnv = (prerenderTempServer.environments as any)?.rsc;
|
|
490
|
+
if (existingEnv?.runner) {
|
|
491
|
+
if (prerenderNodeRegistry) {
|
|
492
|
+
debugDiscovery?.(
|
|
493
|
+
"getOrCreateTempServer: cached temp runner reused",
|
|
494
|
+
);
|
|
495
|
+
return existingEnv;
|
|
496
|
+
}
|
|
497
|
+
// Server alive but registry missing — likely after a prior
|
|
498
|
+
// refresh's invalidate + import threw. Try to re-import.
|
|
499
|
+
debugDiscovery?.(
|
|
500
|
+
"getOrCreateTempServer: server alive but registry missing — re-importing",
|
|
501
|
+
);
|
|
502
|
+
try {
|
|
503
|
+
await importEntryAndRegistry(existingEnv);
|
|
504
|
+
return existingEnv;
|
|
505
|
+
} catch (err: any) {
|
|
506
|
+
debugDiscovery?.(
|
|
507
|
+
"getOrCreateTempServer: reuse import failed (%s) — closing orphan and creating fresh",
|
|
508
|
+
err?.message ?? String(err),
|
|
509
|
+
);
|
|
510
|
+
await prerenderTempServer.close().catch(() => {});
|
|
511
|
+
prerenderTempServer = null;
|
|
512
|
+
prerenderNodeRegistry = null;
|
|
513
|
+
// Fall through to create-new path below.
|
|
514
|
+
}
|
|
515
|
+
} else {
|
|
516
|
+
// Server reference exists but its rsc env is unhealthy
|
|
517
|
+
// (no runner). Close and recreate.
|
|
518
|
+
debugDiscovery?.(
|
|
519
|
+
"getOrCreateTempServer: existing server has no rsc.runner — closing and recreating",
|
|
520
|
+
);
|
|
521
|
+
await prerenderTempServer.close().catch(() => {});
|
|
522
|
+
prerenderTempServer = null;
|
|
523
|
+
prerenderNodeRegistry = null;
|
|
524
|
+
}
|
|
239
525
|
}
|
|
526
|
+
|
|
527
|
+
// Create path: no existing temp server (or just nullified above).
|
|
528
|
+
debugDiscovery?.(
|
|
529
|
+
"getOrCreateTempServer: creating new temp server, entry=%s",
|
|
530
|
+
s.resolvedEntryPath ?? "(unset)",
|
|
531
|
+
);
|
|
240
532
|
try {
|
|
241
533
|
prerenderTempServer = await createTempRscServer(s, {
|
|
242
534
|
cacheDir: "node_modules/.vite_prerender",
|
|
@@ -244,58 +536,189 @@ export function createRouterDiscoveryPlugin(
|
|
|
244
536
|
|
|
245
537
|
const tempRscEnv = (prerenderTempServer.environments as any)?.rsc;
|
|
246
538
|
if (tempRscEnv?.runner) {
|
|
247
|
-
await tempRscEnv
|
|
248
|
-
const serverMod = await tempRscEnv.runner.import(
|
|
249
|
-
"@rangojs/router/server",
|
|
250
|
-
);
|
|
251
|
-
prerenderNodeRegistry = serverMod.RouterRegistry;
|
|
539
|
+
await importEntryAndRegistry(tempRscEnv);
|
|
252
540
|
return tempRscEnv;
|
|
253
541
|
}
|
|
542
|
+
debugDiscovery?.(
|
|
543
|
+
"getOrCreateTempServer: tempRscEnv.runner unavailable",
|
|
544
|
+
);
|
|
254
545
|
} catch (err: any) {
|
|
255
|
-
|
|
256
|
-
|
|
546
|
+
debugDiscovery?.(
|
|
547
|
+
"getOrCreateTempServer: FAILED message=%s",
|
|
548
|
+
err.message,
|
|
257
549
|
);
|
|
550
|
+
console.warn(`[rango] Failed to create temp runner: ${err.message}`);
|
|
258
551
|
}
|
|
259
552
|
return null;
|
|
260
553
|
}
|
|
261
554
|
|
|
555
|
+
// Clear the package-level singleton registries that survive a Vite
|
|
556
|
+
// moduleGraph.invalidateAll(). createRouter() / createHostRouter()
|
|
557
|
+
// call .set(id, ...) on these Maps; for "router removed" or
|
|
558
|
+
// "router id changed" edits, the OLD entry would persist after
|
|
559
|
+
// re-import without an explicit .clear(), leaving ghost routes
|
|
560
|
+
// in discoverRouters' output.
|
|
561
|
+
//
|
|
562
|
+
// We import the same module the runner imports, so the .clear()
|
|
563
|
+
// here mutates the same Map the freshly re-imported entry will
|
|
564
|
+
// populate.
|
|
565
|
+
async function clearTempRegistries(tempRscEnv: any): Promise<void> {
|
|
566
|
+
try {
|
|
567
|
+
const serverMod = await tempRscEnv.runner.import(
|
|
568
|
+
"@rangojs/router/server",
|
|
569
|
+
);
|
|
570
|
+
if (typeof serverMod?.RouterRegistry?.clear === "function") {
|
|
571
|
+
serverMod.RouterRegistry.clear();
|
|
572
|
+
}
|
|
573
|
+
if (typeof serverMod?.HostRouterRegistry?.clear === "function") {
|
|
574
|
+
serverMod.HostRouterRegistry.clear();
|
|
575
|
+
}
|
|
576
|
+
debugDiscovery?.(
|
|
577
|
+
"clearTempRegistries: cleared RouterRegistry + HostRouterRegistry",
|
|
578
|
+
);
|
|
579
|
+
} catch (err: any) {
|
|
580
|
+
// Non-fatal: if the import fails here, importEntryAndRegistry
|
|
581
|
+
// below will fail loudly with the same root cause and the
|
|
582
|
+
// caller will surface it.
|
|
583
|
+
debugDiscovery?.(
|
|
584
|
+
"clearTempRegistries: import @rangojs/router/server failed (%s)",
|
|
585
|
+
err?.message ?? String(err),
|
|
586
|
+
);
|
|
587
|
+
}
|
|
588
|
+
}
|
|
589
|
+
|
|
590
|
+
// HMR refresh: keep the temp Vite server alive across HMR cycles and
|
|
591
|
+
// invalidate its module graph instead of close+recreate. Closing the
|
|
592
|
+
// temp server during workerd's first post-cold-start module-fetch
|
|
593
|
+
// window disrupted the main dev server's transport — the user-visible
|
|
594
|
+
// symptom was a `transport was disconnected, cannot call "fetchModule"`
|
|
595
|
+
// error on the first urls.tsx edit (workerd's cache was cold, so its
|
|
596
|
+
// eval was still in flight when our close() ran). Module-graph
|
|
597
|
+
// invalidation is the architecturally cleaner refresh: same Vite
|
|
598
|
+
// instance, same transport, fresh source.
|
|
599
|
+
//
|
|
600
|
+
// Falls back to close+recreate when neither the env-level nor
|
|
601
|
+
// server-level moduleGraph exposes invalidateAll() (defensive — Vite
|
|
602
|
+
// versions / preset configurations may differ in which graph carries
|
|
603
|
+
// the module-runner cache).
|
|
604
|
+
async function refreshTempRscEnv(): Promise<any | null> {
|
|
605
|
+
let tempRscEnv = await getOrCreateTempServer();
|
|
606
|
+
if (!tempRscEnv) return null;
|
|
607
|
+
|
|
608
|
+
// Module-runner cache is on the per-environment graph in Vite 6+;
|
|
609
|
+
// older / non-environments setups carry it on the server graph.
|
|
610
|
+
// Try env first, server second.
|
|
611
|
+
const envGraph = (tempRscEnv as any).moduleGraph;
|
|
612
|
+
const serverGraph = (prerenderTempServer as any)?.moduleGraph;
|
|
613
|
+
const target = envGraph?.invalidateAll
|
|
614
|
+
? envGraph
|
|
615
|
+
: serverGraph?.invalidateAll
|
|
616
|
+
? serverGraph
|
|
617
|
+
: null;
|
|
618
|
+
|
|
619
|
+
if (!target) {
|
|
620
|
+
// No invalidate method available — fall back to close+recreate.
|
|
621
|
+
// This preserves the previous behavior in case a Vite version
|
|
622
|
+
// doesn't expose invalidateAll on either graph.
|
|
623
|
+
debugDiscovery?.(
|
|
624
|
+
"refreshTempRscEnv: invalidateAll unavailable on env+server graphs, falling back to close+recreate",
|
|
625
|
+
);
|
|
626
|
+
if (prerenderTempServer) {
|
|
627
|
+
await prerenderTempServer.close().catch(() => {});
|
|
628
|
+
prerenderTempServer = null;
|
|
629
|
+
prerenderNodeRegistry = null;
|
|
630
|
+
}
|
|
631
|
+
return await getOrCreateTempServer();
|
|
632
|
+
}
|
|
633
|
+
|
|
634
|
+
debugDiscovery?.(
|
|
635
|
+
"refreshTempRscEnv: invalidating module graph (%s)",
|
|
636
|
+
envGraph?.invalidateAll ? "env" : "server",
|
|
637
|
+
);
|
|
638
|
+
target.invalidateAll();
|
|
639
|
+
// Drop the cached registry so importEntryAndRegistry re-reads it
|
|
640
|
+
// through the now-invalidated module runner.
|
|
641
|
+
prerenderNodeRegistry = null;
|
|
642
|
+
// Clear singleton Maps that Vite's moduleGraph invalidation can't
|
|
643
|
+
// reach (RouterRegistry / HostRouterRegistry). Without this, an
|
|
644
|
+
// edit that REMOVES a createRouter() call or CHANGES a router id
|
|
645
|
+
// would leave the old entry in the registry, and discoverRouters
|
|
646
|
+
// would still emit its routes alongside whatever the new source
|
|
647
|
+
// declares.
|
|
648
|
+
await clearTempRegistries(tempRscEnv);
|
|
649
|
+
await importEntryAndRegistry(tempRscEnv);
|
|
650
|
+
return tempRscEnv;
|
|
651
|
+
}
|
|
652
|
+
|
|
262
653
|
const discover = async () => {
|
|
654
|
+
const discoverStart = performance.now();
|
|
263
655
|
const rscEnv = (server.environments as any)?.rsc;
|
|
264
656
|
if (!rscEnv?.runner) {
|
|
265
657
|
// Cloudflare dev: no module runner available (workerd-based RSC env).
|
|
266
658
|
// Set devServerOrigin so the virtual module can inject __PRERENDER_DEV_URL
|
|
267
659
|
// for on-demand prerender via the /__rsc_prerender endpoint.
|
|
660
|
+
debugDiscovery?.(
|
|
661
|
+
"dev: cloudflare path start, __rscRouterDiscoveryActive=%s",
|
|
662
|
+
(globalThis as any).__rscRouterDiscoveryActive ?? false,
|
|
663
|
+
);
|
|
268
664
|
s.devServerOrigin = getDevServerOrigin();
|
|
269
665
|
|
|
270
666
|
// Create a temp Node.js server to run runtime discovery and generate
|
|
271
667
|
// named route types (static parser can't resolve factory calls).
|
|
272
668
|
try {
|
|
273
|
-
|
|
669
|
+
// Acquire build-time env bindings for dev prerender
|
|
670
|
+
await timed(debugDiscovery, "acquireBuildEnv", () =>
|
|
671
|
+
acquireBuildEnv(s, viteCommand, viteMode),
|
|
672
|
+
);
|
|
673
|
+
|
|
674
|
+
const tempRscEnv = await timed(
|
|
675
|
+
debugDiscovery,
|
|
676
|
+
"getOrCreateTempServer",
|
|
677
|
+
() => getOrCreateTempServer(),
|
|
678
|
+
);
|
|
274
679
|
if (tempRscEnv) {
|
|
275
|
-
await discoverRouters(
|
|
276
|
-
|
|
680
|
+
await timed(debugDiscovery, "discoverRouters (cloudflare)", () =>
|
|
681
|
+
discoverRouters(s, tempRscEnv),
|
|
682
|
+
);
|
|
683
|
+
timedSync(debugDiscovery, "writeRouteTypesFiles", () =>
|
|
684
|
+
writeRouteTypesFiles(s),
|
|
685
|
+
);
|
|
277
686
|
}
|
|
278
687
|
} catch (err: any) {
|
|
279
688
|
console.warn(
|
|
280
|
-
`[
|
|
689
|
+
`[rango] Cloudflare dev discovery failed: ${err.message}\n${err.stack}`,
|
|
281
690
|
);
|
|
282
691
|
}
|
|
283
692
|
|
|
693
|
+
debugDiscovery?.(
|
|
694
|
+
"dev discovery done (%sms)",
|
|
695
|
+
(performance.now() - discoverStart).toFixed(1),
|
|
696
|
+
);
|
|
284
697
|
resolveDiscovery!();
|
|
285
698
|
return;
|
|
286
699
|
}
|
|
287
700
|
|
|
288
701
|
try {
|
|
702
|
+
// Acquire build-time env bindings for dev prerender (Node.js path)
|
|
703
|
+
debugDiscovery?.("dev: node path start");
|
|
704
|
+
await timed(debugDiscovery, "acquireBuildEnv", () =>
|
|
705
|
+
acquireBuildEnv(s, viteCommand, viteMode),
|
|
706
|
+
);
|
|
707
|
+
|
|
289
708
|
// Set the readiness gate BEFORE discovery so early requests
|
|
290
709
|
// block until manifest is populated
|
|
291
|
-
const serverMod = await
|
|
292
|
-
|
|
710
|
+
const serverMod = await timed(
|
|
711
|
+
debugDiscovery,
|
|
712
|
+
"import @rangojs/router/server",
|
|
713
|
+
() => rscEnv.runner.import("@rangojs/router/server"),
|
|
293
714
|
);
|
|
294
715
|
if (serverMod?.setManifestReadyPromise) {
|
|
295
716
|
serverMod.setManifestReadyPromise(discoveryPromise);
|
|
296
717
|
}
|
|
297
718
|
|
|
298
|
-
await
|
|
719
|
+
await timed(debugDiscovery, "discoverRouters", () =>
|
|
720
|
+
discoverRouters(s, rscEnv),
|
|
721
|
+
);
|
|
299
722
|
|
|
300
723
|
// Store server origin for dev prerender endpoint (virtual module injection)
|
|
301
724
|
s.devServerOrigin = getDevServerOrigin();
|
|
@@ -305,24 +728,36 @@ export function createRouterDiscoveryPlugin(
|
|
|
305
728
|
// routes (e.g. Array.from loops) that the static parser cannot see.
|
|
306
729
|
// writeRouteTypesFiles() only writes when content changes, so this
|
|
307
730
|
// won't cause unnecessary HMR triggers.
|
|
308
|
-
writeRouteTypesFiles(
|
|
731
|
+
timedSync(debugDiscovery, "writeRouteTypesFiles", () =>
|
|
732
|
+
writeRouteTypesFiles(s),
|
|
733
|
+
);
|
|
309
734
|
|
|
310
735
|
// Populate the route map and per-router data in the RSC env
|
|
311
|
-
await propagateDiscoveryState(
|
|
736
|
+
await timed(debugDiscovery, "propagateDiscoveryState", () =>
|
|
737
|
+
propagateDiscoveryState(rscEnv),
|
|
738
|
+
);
|
|
312
739
|
} catch (err: any) {
|
|
313
740
|
console.warn(
|
|
314
|
-
`[
|
|
741
|
+
`[rango] Router discovery failed: ${err.message}\n${err.stack}`,
|
|
315
742
|
);
|
|
316
743
|
} finally {
|
|
744
|
+
debugDiscovery?.(
|
|
745
|
+
"dev discovery done (%sms)",
|
|
746
|
+
(performance.now() - discoverStart).toFixed(1),
|
|
747
|
+
);
|
|
317
748
|
resolveDiscovery!();
|
|
318
749
|
}
|
|
319
750
|
};
|
|
320
751
|
|
|
321
752
|
// Schedule after all plugins have finished configureServer.
|
|
322
|
-
//
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
753
|
+
// The gate (s.discoveryDone) is reset via beginDiscoveryGate() and
|
|
754
|
+
// resolved when discover() finishes, so the virtual manifest module's
|
|
755
|
+
// load() awaits the populated state.
|
|
756
|
+
beginDiscoveryGate();
|
|
757
|
+
setTimeout(
|
|
758
|
+
() => discover().then(resolveDiscoveryGate, resolveDiscoveryGate),
|
|
759
|
+
0,
|
|
760
|
+
);
|
|
326
761
|
|
|
327
762
|
// Dev-mode on-demand prerender endpoint.
|
|
328
763
|
// When workerd hits a prerender route, it fetches this endpoint instead of
|
|
@@ -362,24 +797,30 @@ export function createRouterDiscoveryPlugin(
|
|
|
362
797
|
if (s.mergedRouteTrie && serverMod.setRouteTrie) {
|
|
363
798
|
serverMod.setRouteTrie(s.mergedRouteTrie);
|
|
364
799
|
}
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
|
|
368
|
-
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
}
|
|
375
|
-
if (serverMod.setRouterPrecomputedEntries) {
|
|
376
|
-
for (const [routerId, entries] of s.perRouterPrecomputedMap) {
|
|
377
|
-
serverMod.setRouterPrecomputedEntries(routerId, entries);
|
|
378
|
-
}
|
|
800
|
+
const perRouterSetters: Array<[Map<string, any>, string]> = [
|
|
801
|
+
[s.perRouterManifestDataMap, "setRouterManifest"],
|
|
802
|
+
[s.perRouterTrieMap, "setRouterTrie"],
|
|
803
|
+
[s.perRouterPrecomputedMap, "setRouterPrecomputedEntries"],
|
|
804
|
+
];
|
|
805
|
+
for (const [map, fn] of perRouterSetters) {
|
|
806
|
+
const setter = serverMod[fn];
|
|
807
|
+
if (typeof setter !== "function") continue;
|
|
808
|
+
for (const [routerId, value] of map) setter(routerId, value);
|
|
379
809
|
}
|
|
380
810
|
};
|
|
381
811
|
|
|
382
812
|
server.middlewares.use("/__rsc_prerender", async (req: any, res: any) => {
|
|
813
|
+
const reqStart = debugDev ? performance.now() : 0;
|
|
814
|
+
const logResult = (status: number, note: string) => {
|
|
815
|
+
debugDev?.(
|
|
816
|
+
"/__rsc_prerender %s -> %d %s (%sms)",
|
|
817
|
+
req.url,
|
|
818
|
+
status,
|
|
819
|
+
note,
|
|
820
|
+
(performance.now() - reqStart).toFixed(1),
|
|
821
|
+
);
|
|
822
|
+
};
|
|
823
|
+
|
|
383
824
|
if (s.discoveryDone) await s.discoveryDone;
|
|
384
825
|
|
|
385
826
|
const url = new URL(req.url || "/", "http://localhost");
|
|
@@ -387,12 +828,36 @@ export function createRouterDiscoveryPlugin(
|
|
|
387
828
|
if (!pathname) {
|
|
388
829
|
res.statusCode = 400;
|
|
389
830
|
res.end("Missing pathname");
|
|
831
|
+
logResult(400, "missing pathname");
|
|
390
832
|
return;
|
|
391
833
|
}
|
|
392
834
|
|
|
393
|
-
//
|
|
394
|
-
//
|
|
395
|
-
|
|
835
|
+
// Import the user's entry module to force re-evaluation of any
|
|
836
|
+
// HMR-invalidated modules in the chain (entry → router → urls → handlers).
|
|
837
|
+
// This ensures createRouter() re-runs with updated handler code before
|
|
838
|
+
// we read RouterRegistry. Without this, edits to prerender handler files
|
|
839
|
+
// produce stale content because the old router instance remains registered.
|
|
840
|
+
const rscEnv = (server.environments as any)?.rsc;
|
|
841
|
+
let registry: Map<string, any> | null = null;
|
|
842
|
+
if (rscEnv?.runner && s.resolvedEntryPath) {
|
|
843
|
+
try {
|
|
844
|
+
await rscEnv.runner.import(s.resolvedEntryPath);
|
|
845
|
+
const serverMod = await rscEnv.runner.import(
|
|
846
|
+
"@rangojs/router/server",
|
|
847
|
+
);
|
|
848
|
+
registry = serverMod.RouterRegistry ?? null;
|
|
849
|
+
} catch (err: any) {
|
|
850
|
+
console.warn(
|
|
851
|
+
`[rango] Dev prerender module refresh failed: ${err.message}`,
|
|
852
|
+
);
|
|
853
|
+
res.statusCode = 500;
|
|
854
|
+
res.end(`Prerender handler error: ${err.message}`);
|
|
855
|
+
logResult(500, "module refresh failed");
|
|
856
|
+
return;
|
|
857
|
+
}
|
|
858
|
+
} else {
|
|
859
|
+
registry = mainRegistry;
|
|
860
|
+
}
|
|
396
861
|
|
|
397
862
|
if (!registry) {
|
|
398
863
|
// No main registry: the RSC env has no module runner (Cloudflare dev).
|
|
@@ -406,6 +871,7 @@ export function createRouterDiscoveryPlugin(
|
|
|
406
871
|
if (!registry || registry.size === 0) {
|
|
407
872
|
res.statusCode = 503;
|
|
408
873
|
res.end("Prerender runner not available");
|
|
874
|
+
logResult(503, "no registry");
|
|
409
875
|
return;
|
|
410
876
|
}
|
|
411
877
|
|
|
@@ -421,6 +887,8 @@ export function createRouterDiscoveryPlugin(
|
|
|
421
887
|
{},
|
|
422
888
|
undefined,
|
|
423
889
|
wantPassthrough,
|
|
890
|
+
s.resolvedBuildEnv,
|
|
891
|
+
true, // devMode: check getParams for passthrough routes
|
|
424
892
|
);
|
|
425
893
|
if (!result) continue;
|
|
426
894
|
if (result.passthrough) continue;
|
|
@@ -433,25 +901,26 @@ export function createRouterDiscoveryPlugin(
|
|
|
433
901
|
if (wantIntercept && result.interceptSegments?.length) {
|
|
434
902
|
payload = {
|
|
435
903
|
segments: [...result.segments, ...result.interceptSegments],
|
|
436
|
-
handles
|
|
437
|
-
|
|
438
|
-
|
|
439
|
-
},
|
|
904
|
+
// Pre-encoded MERGED handle string from the producer (handles are
|
|
905
|
+
// Flight-encoded so Promise/ReactNode values survive the wire).
|
|
906
|
+
handles: result.interceptHandles ?? "",
|
|
440
907
|
};
|
|
441
908
|
} else {
|
|
442
909
|
payload = { segments: result.segments, handles: result.handles };
|
|
443
910
|
}
|
|
444
911
|
res.end(JSON.stringify(payload));
|
|
912
|
+
logResult(200, `match ${result.routeName}`);
|
|
445
913
|
return;
|
|
446
914
|
} catch (err: any) {
|
|
447
915
|
console.warn(
|
|
448
|
-
`[
|
|
916
|
+
`[rango] Dev prerender failed for ${pathname}: ${err.message}`,
|
|
449
917
|
);
|
|
450
918
|
}
|
|
451
919
|
}
|
|
452
920
|
|
|
453
921
|
res.statusCode = 404;
|
|
454
922
|
res.end("No prerender match");
|
|
923
|
+
logResult(404, "no match");
|
|
455
924
|
});
|
|
456
925
|
|
|
457
926
|
// Watch url module and router files for changes and regenerate named-routes.gen.ts.
|
|
@@ -494,21 +963,135 @@ export function createRouterDiscoveryPlugin(
|
|
|
494
963
|
|
|
495
964
|
// Re-run runtime discovery so factory-generated routes that the
|
|
496
965
|
// static parser cannot see are refreshed after source changes.
|
|
497
|
-
|
|
966
|
+
// The state-machine concerns (queued/pending/gatePending) are
|
|
967
|
+
// owned by the gate created above (./discovery/gate-state.ts).
|
|
968
|
+
// Here we provide just the env-specific work.
|
|
498
969
|
const refreshRuntimeDiscovery = async () => {
|
|
499
970
|
const rscEnv = (server.environments as any)?.rsc;
|
|
500
|
-
|
|
501
|
-
|
|
971
|
+
const hasMainRunner = !!rscEnv?.runner;
|
|
972
|
+
// Cloudflare HMR has no main RSC runner (workerd is a separate
|
|
973
|
+
// runtime). When we have a populated runtime manifest from cold
|
|
974
|
+
// start, we can re-discover via the temp Node runner — the same
|
|
975
|
+
// mechanism getOrCreateTempServer() uses at startup. Without a
|
|
976
|
+
// populated manifest there's nothing useful to do, so bail
|
|
977
|
+
// before involving the gate machine at all.
|
|
978
|
+
if (!hasMainRunner && s.perRouterManifests.length === 0) return;
|
|
979
|
+
await gate.runRefreshCycle(async () => {
|
|
980
|
+
const hmrStart = performance.now();
|
|
981
|
+
try {
|
|
982
|
+
if (hasMainRunner) {
|
|
983
|
+
await timed(debugDiscovery, "hmr discoverRouters", () =>
|
|
984
|
+
discoverRouters(s, rscEnv),
|
|
985
|
+
);
|
|
986
|
+
timedSync(debugDiscovery, "hmr writeRouteTypesFiles", () =>
|
|
987
|
+
writeRouteTypesFiles(s),
|
|
988
|
+
);
|
|
989
|
+
await timed(debugDiscovery, "hmr propagateDiscoveryState", () =>
|
|
990
|
+
propagateDiscoveryState(rscEnv),
|
|
991
|
+
);
|
|
992
|
+
} else {
|
|
993
|
+
// Cloudflare HMR: invalidate the temp server's RSC module
|
|
994
|
+
// graph (or close+recreate as a fallback) so the runner
|
|
995
|
+
// re-reads the freshly edited source. Keeping the same
|
|
996
|
+
// Vite instance alive avoids disrupting workerd's transport
|
|
997
|
+
// during the first post-cold-start module-fetch window.
|
|
998
|
+
const tempRscEnv = await timed(
|
|
999
|
+
debugDiscovery,
|
|
1000
|
+
"hmr refreshTempRscEnv (cloudflare)",
|
|
1001
|
+
() => refreshTempRscEnv(),
|
|
1002
|
+
);
|
|
1003
|
+
if (!tempRscEnv) {
|
|
1004
|
+
throw new Error(
|
|
1005
|
+
"temp runner unavailable for cloudflare HMR rediscovery",
|
|
1006
|
+
);
|
|
1007
|
+
}
|
|
1008
|
+
await timed(
|
|
1009
|
+
debugDiscovery,
|
|
1010
|
+
"hmr discoverRouters (cloudflare)",
|
|
1011
|
+
() => discoverRouters(s, tempRscEnv),
|
|
1012
|
+
);
|
|
1013
|
+
timedSync(debugDiscovery, "hmr writeRouteTypesFiles", () =>
|
|
1014
|
+
writeRouteTypesFiles(s),
|
|
1015
|
+
);
|
|
1016
|
+
}
|
|
1017
|
+
if (s.lastDiscoveryError) {
|
|
1018
|
+
debugDiscovery?.(
|
|
1019
|
+
"hmr: cleared lastDiscoveryError (%s) after successful rediscovery",
|
|
1020
|
+
s.lastDiscoveryError.message,
|
|
1021
|
+
);
|
|
1022
|
+
s.lastDiscoveryError = null;
|
|
1023
|
+
}
|
|
1024
|
+
// Cloudflare dev: on a successful cycle drop the workerd runner's
|
|
1025
|
+
// cached worker-entry chain so the next request re-evaluates
|
|
1026
|
+
// createRouter() with the new routes. Fired here in the work path
|
|
1027
|
+
// (not the caller's .then()) so a queued follow-up cycle that
|
|
1028
|
+
// succeeds after an earlier failed cycle still reloads:
|
|
1029
|
+
// runRefreshCycle recurses queued work without awaiting it, so the
|
|
1030
|
+
// original call already resolved on the failed cycle. A failed
|
|
1031
|
+
// cycle throws above and never reaches here, so a broken edit
|
|
1032
|
+
// never reloads the worker onto bad source.
|
|
1033
|
+
if (rscEnv && !rscEnv.runner) forceCloudflareWorkerReload(rscEnv);
|
|
1034
|
+
} catch (err: any) {
|
|
1035
|
+
s.lastDiscoveryError = {
|
|
1036
|
+
message: err?.message ?? String(err),
|
|
1037
|
+
at: Date.now(),
|
|
1038
|
+
};
|
|
1039
|
+
console.warn(
|
|
1040
|
+
`[rango] Runtime re-discovery failed: ${err.message}`,
|
|
1041
|
+
);
|
|
1042
|
+
debugDiscovery?.(
|
|
1043
|
+
"hmr: lastDiscoveryError set (%s) — manifest preserved at last-good; recovery mode active (any in-scan source change will trigger rediscovery)",
|
|
1044
|
+
err?.message,
|
|
1045
|
+
);
|
|
1046
|
+
} finally {
|
|
1047
|
+
debugDiscovery?.(
|
|
1048
|
+
"hmr re-discovery done (%sms)",
|
|
1049
|
+
(performance.now() - hmrStart).toFixed(1),
|
|
1050
|
+
);
|
|
1051
|
+
}
|
|
1052
|
+
});
|
|
1053
|
+
};
|
|
1054
|
+
|
|
1055
|
+
// Cloudflare dev only. workerd serves every request through the
|
|
1056
|
+
// runner-worker singleton, which re-resolves the worker entry per
|
|
1057
|
+
// request via runner.import("virtual:cloudflare/worker-entry"). The
|
|
1058
|
+
// route table lives in the user's createRouter() instance, captured
|
|
1059
|
+
// when that entry chain (entry -> router -> urls) was last evaluated
|
|
1060
|
+
// and then cached in the runner's evaluatedModules. The route-file
|
|
1061
|
+
// watcher refreshes discovery + types on the Node side, but the worker
|
|
1062
|
+
// keeps serving the cached (stale) router: route-definition modules
|
|
1063
|
+
// have no import.meta.hot boundary, so Vite never sends the worker an
|
|
1064
|
+
// HMR update for them and the entry chain is never evicted.
|
|
1065
|
+
//
|
|
1066
|
+
// Fix: after discovery completes, (1) invalidate the worker env's
|
|
1067
|
+
// Node-side module graph, then (2) send a full-reload to the worker.
|
|
1068
|
+
// Step (2) alone is insufficient: the full-reload handler clears the
|
|
1069
|
+
// runner's evaluatedModules and re-imports entrypoints, but each
|
|
1070
|
+
// re-import fetches the module back through this Node-side graph, which
|
|
1071
|
+
// still holds the pre-edit transform of urls.tsx — so createRouter()
|
|
1072
|
+
// rebuilds the stale route table and the new route 404s/hits the
|
|
1073
|
+
// catch-all. Invalidating the graph forces a fresh transform on
|
|
1074
|
+
// re-fetch (the same mechanism refreshTempRscEnv uses for discovery),
|
|
1075
|
+
// so the re-import re-runs createRouter() with the new routes. This is
|
|
1076
|
+
// the programmatic equivalent of the dev-server "r + enter" restart,
|
|
1077
|
+
// scoped to the worker environment instead of tearing down the server.
|
|
1078
|
+
const forceCloudflareWorkerReload = (rscEnv: any) => {
|
|
1079
|
+
if (!rscEnv?.hot) return;
|
|
502
1080
|
try {
|
|
503
|
-
|
|
504
|
-
|
|
505
|
-
|
|
1081
|
+
const graph = rscEnv.moduleGraph;
|
|
1082
|
+
if (graph?.invalidateAll) {
|
|
1083
|
+
graph.invalidateAll();
|
|
1084
|
+
debugDiscovery?.("hmr: invalidated workerd rsc module graph");
|
|
1085
|
+
}
|
|
1086
|
+
rscEnv.hot.send({ type: "full-reload" });
|
|
1087
|
+
debugDiscovery?.(
|
|
1088
|
+
"hmr: forced workerd rsc env reload (full-reload)",
|
|
1089
|
+
);
|
|
506
1090
|
} catch (err: any) {
|
|
507
|
-
|
|
508
|
-
|
|
1091
|
+
debugDiscovery?.(
|
|
1092
|
+
"hmr: workerd reload failed: %s",
|
|
1093
|
+
err?.message ?? err,
|
|
509
1094
|
);
|
|
510
|
-
} finally {
|
|
511
|
-
runtimeRediscoveryInProgress = false;
|
|
512
1095
|
}
|
|
513
1096
|
};
|
|
514
1097
|
|
|
@@ -516,23 +1099,50 @@ export function createRouterDiscoveryPlugin(
|
|
|
516
1099
|
clearTimeout(routeChangeTimer);
|
|
517
1100
|
routeChangeTimer = setTimeout(() => {
|
|
518
1101
|
routeChangeTimer = undefined;
|
|
1102
|
+
const regenStart = debugDiscovery ? performance.now() : 0;
|
|
1103
|
+
const rscEnv = (server.environments as any)?.rsc;
|
|
1104
|
+
const skipStaticWrite =
|
|
1105
|
+
!rscEnv?.runner && s.perRouterManifests.length > 0;
|
|
519
1106
|
try {
|
|
520
|
-
|
|
521
|
-
|
|
522
|
-
|
|
1107
|
+
// In cloudflare dev with a populated runtime manifest, the
|
|
1108
|
+
// static parser produces a strictly smaller (and actively
|
|
1109
|
+
// wrong) gen file — supplementGenFilesWithRuntimeRoutes can
|
|
1110
|
+
// only restore factory-only prefixes, and apps with mixed
|
|
1111
|
+
// static+factory routes under shared prefixes (cf-stress)
|
|
1112
|
+
// collapse to the 19-route static view. Skip the static
|
|
1113
|
+
// write entirely; runtime rediscovery below will overwrite
|
|
1114
|
+
// the gen file with the authoritative manifest.
|
|
1115
|
+
if (skipStaticWrite) {
|
|
1116
|
+
debugDiscovery?.(
|
|
1117
|
+
"watcher: skipping static write (cloudflare HMR — runtime rediscovery owns gen file)",
|
|
1118
|
+
);
|
|
1119
|
+
} else {
|
|
1120
|
+
writeCombinedRouteTypesWithTracking(s);
|
|
1121
|
+
if (s.perRouterManifests.length > 0) {
|
|
1122
|
+
supplementGenFilesWithRuntimeRoutes(s);
|
|
1123
|
+
}
|
|
523
1124
|
}
|
|
524
1125
|
} catch (err: any) {
|
|
525
|
-
console.error(
|
|
526
|
-
`[rsc-router] Route regeneration error: ${err.message}`,
|
|
527
|
-
);
|
|
1126
|
+
console.error(`[rango] Route regeneration error: ${err.message}`);
|
|
528
1127
|
}
|
|
1128
|
+
debugDiscovery?.(
|
|
1129
|
+
"watcher: regenerated gen files (%sms)",
|
|
1130
|
+
(performance.now() - regenStart).toFixed(1),
|
|
1131
|
+
);
|
|
529
1132
|
// Async: re-run runtime discovery to refresh factory-generated
|
|
530
|
-
// routes that the static parser cannot resolve.
|
|
1133
|
+
// routes that the static parser cannot resolve. Resolves the
|
|
1134
|
+
// discovery gate when complete.
|
|
531
1135
|
if (s.perRouterManifests.length > 0) {
|
|
1136
|
+
// The cloudflare workerd reload fires inside refreshRuntimeDiscovery
|
|
1137
|
+
// on the successful cycle (see forceCloudflareWorkerReload call
|
|
1138
|
+
// there) so queued follow-up cycles also trigger it.
|
|
532
1139
|
refreshRuntimeDiscovery().catch((err: any) => {
|
|
533
1140
|
console.warn(
|
|
534
|
-
`[
|
|
1141
|
+
`[rango] Runtime re-discovery error: ${err.message}`,
|
|
535
1142
|
);
|
|
1143
|
+
// Even on error, unblock the gate so workerd's reload doesn't
|
|
1144
|
+
// hang indefinitely against the previous manifest.
|
|
1145
|
+
resolveDiscoveryGate();
|
|
536
1146
|
});
|
|
537
1147
|
}
|
|
538
1148
|
}, 100);
|
|
@@ -545,21 +1155,74 @@ export function createRouterDiscoveryPlugin(
|
|
|
545
1155
|
!filePath.endsWith(".tsx") &&
|
|
546
1156
|
!filePath.endsWith(".js") &&
|
|
547
1157
|
!filePath.endsWith(".jsx")
|
|
548
|
-
)
|
|
1158
|
+
) {
|
|
1159
|
+
if (s.lastDiscoveryError) {
|
|
1160
|
+
debugDiscovery?.(
|
|
1161
|
+
"watcher: skip non-source %s [LASTERR %s]",
|
|
1162
|
+
filePath,
|
|
1163
|
+
s.lastDiscoveryError.message,
|
|
1164
|
+
);
|
|
1165
|
+
}
|
|
549
1166
|
return;
|
|
1167
|
+
}
|
|
550
1168
|
// Apply scan filter as early-exit before reading file
|
|
551
|
-
if (s.scanFilter && !s.scanFilter(filePath))
|
|
1169
|
+
if (s.scanFilter && !s.scanFilter(filePath)) {
|
|
1170
|
+
if (s.lastDiscoveryError) {
|
|
1171
|
+
debugDiscovery?.(
|
|
1172
|
+
"watcher: skip scan-filter %s [LASTERR %s]",
|
|
1173
|
+
filePath,
|
|
1174
|
+
s.lastDiscoveryError.message,
|
|
1175
|
+
);
|
|
1176
|
+
}
|
|
1177
|
+
return;
|
|
1178
|
+
}
|
|
1179
|
+
// Recovery mode: when the previous HMR re-discovery failed, the
|
|
1180
|
+
// import graph is incomplete and the manifest is stuck at the
|
|
1181
|
+
// last-good state. The fix may land in a non-route file (e.g. a
|
|
1182
|
+
// helper imported by the router, a missing module being created,
|
|
1183
|
+
// or a "use client" component) that the narrow content sniff
|
|
1184
|
+
// would otherwise filter out. While in recovery, treat any
|
|
1185
|
+
// in-scan source change as a candidate for rediscovery; the
|
|
1186
|
+
// tighter filter resumes once discovery succeeds again.
|
|
1187
|
+
const inRecoveryMode = !!s.lastDiscoveryError;
|
|
552
1188
|
try {
|
|
553
1189
|
const source = readFileSync(filePath, "utf-8");
|
|
554
1190
|
const trimmed = source.trimStart();
|
|
555
|
-
|
|
1191
|
+
const isUseClient =
|
|
556
1192
|
trimmed.startsWith('"use client"') ||
|
|
557
|
-
trimmed.startsWith("'use client'")
|
|
558
|
-
)
|
|
559
|
-
|
|
560
|
-
|
|
561
|
-
|
|
562
|
-
|
|
1193
|
+
trimmed.startsWith("'use client'");
|
|
1194
|
+
if (!inRecoveryMode && isUseClient) return;
|
|
1195
|
+
// Cheap raw pre-check first; only when a candidate token is present
|
|
1196
|
+
// do we confirm it occurs in real code (not a comment/string) via a
|
|
1197
|
+
// single allocation-free code-region scan. Most saved files contain
|
|
1198
|
+
// neither token and skip the scan entirely. This avoids a comment or
|
|
1199
|
+
// string mention spuriously marking a file relevant and triggering an
|
|
1200
|
+
// unnecessary re-discovery on save.
|
|
1201
|
+
let hasUrls = source.includes("urls(");
|
|
1202
|
+
let hasCreateRouter = /\bcreateRouter\s*[<(]/.test(source);
|
|
1203
|
+
if (hasUrls) hasUrls = firstCodeMatchIndex(source, /urls\(/g) >= 0;
|
|
1204
|
+
if (hasCreateRouter) {
|
|
1205
|
+
hasCreateRouter =
|
|
1206
|
+
firstCodeMatchIndex(source, /\bcreateRouter\s*[<(]/g) >= 0;
|
|
1207
|
+
}
|
|
1208
|
+
if (!inRecoveryMode && !hasUrls && !hasCreateRouter) return;
|
|
1209
|
+
if (inRecoveryMode) {
|
|
1210
|
+
debugDiscovery?.(
|
|
1211
|
+
"watcher: recovery rediscovery for %s (urls=%s, router=%s, useClient=%s) [LASTERR %s]",
|
|
1212
|
+
filePath,
|
|
1213
|
+
hasUrls,
|
|
1214
|
+
hasCreateRouter,
|
|
1215
|
+
isUseClient,
|
|
1216
|
+
s.lastDiscoveryError!.message,
|
|
1217
|
+
);
|
|
1218
|
+
} else {
|
|
1219
|
+
debugDiscovery?.(
|
|
1220
|
+
"watcher: %s matches (urls=%s, router=%s)",
|
|
1221
|
+
filePath,
|
|
1222
|
+
hasUrls,
|
|
1223
|
+
hasCreateRouter,
|
|
1224
|
+
);
|
|
1225
|
+
}
|
|
563
1226
|
// Invalidate cache when a router file changes (new router added/removed)
|
|
564
1227
|
if (hasCreateRouter) {
|
|
565
1228
|
const nestedRouterConflict = findNestedRouterConflict([
|
|
@@ -574,8 +1237,27 @@ export function createRouterDiscoveryPlugin(
|
|
|
574
1237
|
}
|
|
575
1238
|
s.cachedRouterFiles = undefined;
|
|
576
1239
|
}
|
|
1240
|
+
// Note the event in the gate machine IMMEDIATELY (before the
|
|
1241
|
+
// 100ms debounce and any downstream HMR fanout). This sets
|
|
1242
|
+
// both `pendingEvents` (so refresh's finally holds the gate
|
|
1243
|
+
// through the tail window even if no rediscovery is queued)
|
|
1244
|
+
// and resets `discoveryDone` to a fresh pending promise (so
|
|
1245
|
+
// workerd reloads triggered by the same source change can't
|
|
1246
|
+
// observe a stale resolved gate from cold-start). Resolved
|
|
1247
|
+
// by the trailing refreshRuntimeDiscovery() cycle.
|
|
1248
|
+
if (s.perRouterManifests.length > 0) {
|
|
1249
|
+
gate.noteRouteEvent();
|
|
1250
|
+
}
|
|
577
1251
|
scheduleRouteRegeneration();
|
|
578
|
-
} catch {
|
|
1252
|
+
} catch (readErr: any) {
|
|
1253
|
+
if (s.lastDiscoveryError) {
|
|
1254
|
+
debugDiscovery?.(
|
|
1255
|
+
"watcher: read error %s: %s [LASTERR %s]",
|
|
1256
|
+
filePath,
|
|
1257
|
+
readErr?.message,
|
|
1258
|
+
s.lastDiscoveryError.message,
|
|
1259
|
+
);
|
|
1260
|
+
}
|
|
579
1261
|
// Ignore read errors for deleted/moved files
|
|
580
1262
|
}
|
|
581
1263
|
};
|
|
@@ -604,11 +1286,24 @@ export function createRouterDiscoveryPlugin(
|
|
|
604
1286
|
async buildStart() {
|
|
605
1287
|
if (!s.isBuildMode) return;
|
|
606
1288
|
// Only run once across environment builds
|
|
607
|
-
if (s.mergedRouteManifest !== null)
|
|
1289
|
+
if (s.mergedRouteManifest !== null) {
|
|
1290
|
+
debugDiscovery?.(
|
|
1291
|
+
"build: skip (already discovered, env=%s)",
|
|
1292
|
+
this.environment?.name ?? "?",
|
|
1293
|
+
);
|
|
1294
|
+
return;
|
|
1295
|
+
}
|
|
1296
|
+
const buildStartTime = performance.now();
|
|
1297
|
+
debugDiscovery?.("build: start (env=%s)", this.environment?.name ?? "?");
|
|
608
1298
|
resetStagedBuildAssets(s.projectRoot);
|
|
609
1299
|
s.prerenderManifestEntries = null;
|
|
610
1300
|
s.staticManifestEntries = null;
|
|
611
1301
|
|
|
1302
|
+
// Acquire build-time env bindings if configured
|
|
1303
|
+
await timed(debugDiscovery, "build acquireBuildEnv", () =>
|
|
1304
|
+
acquireBuildEnv(s, viteCommand, viteMode),
|
|
1305
|
+
);
|
|
1306
|
+
|
|
612
1307
|
let tempServer: any = null;
|
|
613
1308
|
// Signal to user-space code (e.g. reverse.ts) that build-time discovery
|
|
614
1309
|
// is active. Uses globalThis because the temp server's module runner
|
|
@@ -616,12 +1311,16 @@ export function createRouterDiscoveryPlugin(
|
|
|
616
1311
|
// between the vite plugin and user code loaded via runner.import().
|
|
617
1312
|
(globalThis as any).__rscRouterDiscoveryActive = true;
|
|
618
1313
|
try {
|
|
619
|
-
tempServer = await
|
|
1314
|
+
tempServer = await timed(
|
|
1315
|
+
debugDiscovery,
|
|
1316
|
+
"build createTempRscServer",
|
|
1317
|
+
() => createTempRscServer(s, { forceBuild: true }),
|
|
1318
|
+
);
|
|
620
1319
|
|
|
621
1320
|
const rscEnv = (tempServer.environments as any)?.rsc;
|
|
622
1321
|
if (!rscEnv?.runner) {
|
|
623
1322
|
console.warn(
|
|
624
|
-
"[
|
|
1323
|
+
"[rango] RSC environment runner not available during build, skipping manifest generation",
|
|
625
1324
|
);
|
|
626
1325
|
return;
|
|
627
1326
|
}
|
|
@@ -636,11 +1335,15 @@ export function createRouterDiscoveryPlugin(
|
|
|
636
1335
|
s.resolvedStaticModules = tempIdsPlugin.api.staticHandlerModules;
|
|
637
1336
|
}
|
|
638
1337
|
|
|
639
|
-
await
|
|
1338
|
+
await timed(debugDiscovery, "build discoverRouters", () =>
|
|
1339
|
+
discoverRouters(s, rscEnv),
|
|
1340
|
+
);
|
|
640
1341
|
// Update named-routes.gen.ts from runtime discovery.
|
|
641
1342
|
// The runtime manifest includes dynamically generated routes
|
|
642
1343
|
// that the static parser cannot extract from source code.
|
|
643
|
-
writeRouteTypesFiles(
|
|
1344
|
+
timedSync(debugDiscovery, "build writeRouteTypesFiles", () =>
|
|
1345
|
+
writeRouteTypesFiles(s),
|
|
1346
|
+
);
|
|
644
1347
|
} catch (err: any) {
|
|
645
1348
|
// Extract the user source file from the stack trace (skip internal frames)
|
|
646
1349
|
const sourceFile = err.stack
|
|
@@ -660,13 +1363,50 @@ export function createRouterDiscoveryPlugin(
|
|
|
660
1363
|
.filter(Boolean)
|
|
661
1364
|
.join("\n");
|
|
662
1365
|
throw new Error(
|
|
663
|
-
`[
|
|
1366
|
+
`[rango] Build-time router discovery failed:\n${details}`,
|
|
1367
|
+
{ cause: err },
|
|
664
1368
|
);
|
|
665
1369
|
} finally {
|
|
666
1370
|
delete (globalThis as any).__rscRouterDiscoveryActive;
|
|
667
1371
|
if (tempServer) {
|
|
668
|
-
await tempServer.close()
|
|
1372
|
+
await timed(debugDiscovery, "build tempServer.close", () =>
|
|
1373
|
+
tempServer.close(),
|
|
1374
|
+
);
|
|
669
1375
|
}
|
|
1376
|
+
await releaseBuildEnv(s);
|
|
1377
|
+
debugDiscovery?.(
|
|
1378
|
+
"build discovery done (%sms)",
|
|
1379
|
+
(performance.now() - buildStartTime).toFixed(1),
|
|
1380
|
+
);
|
|
1381
|
+
}
|
|
1382
|
+
},
|
|
1383
|
+
|
|
1384
|
+
// Suppress vite's HMR cascade for our own gen-file writes.
|
|
1385
|
+
//
|
|
1386
|
+
// After every cf HMR cycle, refreshTempRscEnv → writeRouteTypesFiles
|
|
1387
|
+
// writes the configured gen files (default `router.named-routes.gen.ts`,
|
|
1388
|
+
// but the source filenames and gen suffix are user-configurable). The
|
|
1389
|
+
// chokidar watcher then fires twice independently: our
|
|
1390
|
+
// `handleRouteFileChange` (already short-circuited by
|
|
1391
|
+
// `consumeSelfGenWrite` inside `maybeHandleGeneratedRouteFileMutation`),
|
|
1392
|
+
// AND vite's own HMR pipeline (which invalidates the gen file's
|
|
1393
|
+
// importers and triggers a second workerd full reload — visible to the
|
|
1394
|
+
// user as a duplicate "[Rango] HMR: version changed" on the client).
|
|
1395
|
+
//
|
|
1396
|
+
// `peekSelfGenWrite` is the authoritative filter: its map only contains
|
|
1397
|
+
// paths that `markSelfGenWrite` has registered, so it natively works
|
|
1398
|
+
// for any configured gen-file name. It is non-consuming so the chokidar
|
|
1399
|
+
// handler that fires later can still consume the same entry. Returning
|
|
1400
|
+
// [] tells vite "no modules invalidated by this change" — safe because
|
|
1401
|
+
// `s.perRouterManifests` is already up-to-date (the write that just
|
|
1402
|
+
// happened is the consequence of our just-completed rediscovery).
|
|
1403
|
+
handleHotUpdate(ctx) {
|
|
1404
|
+
if (peekSelfGenWrite(s, ctx.file)) {
|
|
1405
|
+
debugDiscovery?.(
|
|
1406
|
+
"handleHotUpdate: suppressing self-write HMR cascade for %s",
|
|
1407
|
+
ctx.file,
|
|
1408
|
+
);
|
|
1409
|
+
return [];
|
|
670
1410
|
}
|
|
671
1411
|
},
|
|
672
1412
|
|
|
@@ -690,19 +1430,38 @@ export function createRouterDiscoveryPlugin(
|
|
|
690
1430
|
// This is critical for Cloudflare dev where the worker runs in a separate
|
|
691
1431
|
// Miniflare process and can only receive manifest data via the virtual module.
|
|
692
1432
|
if (s.discoveryDone) {
|
|
693
|
-
await
|
|
1433
|
+
await timed(
|
|
1434
|
+
debugRoutes,
|
|
1435
|
+
"await discoveryDone (manifest)",
|
|
1436
|
+
() => s.discoveryDone,
|
|
1437
|
+
);
|
|
694
1438
|
}
|
|
695
|
-
|
|
1439
|
+
const code = await timed(
|
|
1440
|
+
debugRoutes,
|
|
1441
|
+
"generateRoutesManifestModule",
|
|
1442
|
+
() => generateRoutesManifestModule(s),
|
|
1443
|
+
);
|
|
1444
|
+
debugRoutes?.("manifest module emitted (%d bytes)", code?.length ?? 0);
|
|
1445
|
+
return code;
|
|
696
1446
|
}
|
|
697
1447
|
// Per-router virtual modules: pure data exports (no side effects).
|
|
698
1448
|
// ensureRouterManifest() imports the module and stores the data.
|
|
699
1449
|
const perRouterPrefix = "\0" + VIRTUAL_ROUTES_MANIFEST_ID + "/";
|
|
700
1450
|
if (id.startsWith(perRouterPrefix)) {
|
|
701
1451
|
if (s.discoveryDone) {
|
|
702
|
-
await
|
|
1452
|
+
await timed(
|
|
1453
|
+
debugRoutes,
|
|
1454
|
+
"await discoveryDone (per-router)",
|
|
1455
|
+
() => s.discoveryDone,
|
|
1456
|
+
);
|
|
703
1457
|
}
|
|
704
1458
|
const routerId = id.slice(perRouterPrefix.length);
|
|
705
|
-
|
|
1459
|
+
const code = await timed(
|
|
1460
|
+
debugRoutes,
|
|
1461
|
+
`generatePerRouterModule ${routerId}`,
|
|
1462
|
+
() => generatePerRouterModule(s, routerId),
|
|
1463
|
+
);
|
|
1464
|
+
return code;
|
|
706
1465
|
}
|
|
707
1466
|
// virtual:rsc-router/prerender-paths load handler removed
|
|
708
1467
|
return null;
|
|
@@ -712,6 +1471,7 @@ export function createRouterDiscoveryPlugin(
|
|
|
712
1471
|
// Used by closeBundle for handler code eviction and prerender data injection.
|
|
713
1472
|
generateBundle(_options: any, bundle: any) {
|
|
714
1473
|
if (this.environment?.name !== "rsc") return;
|
|
1474
|
+
const genStart = debugBuild ? performance.now() : 0;
|
|
715
1475
|
|
|
716
1476
|
// Record RSC entry chunk filename for closeBundle injection
|
|
717
1477
|
for (const [fileName, chunk] of Object.entries(bundle) as [
|
|
@@ -724,8 +1484,19 @@ export function createRouterDiscoveryPlugin(
|
|
|
724
1484
|
}
|
|
725
1485
|
}
|
|
726
1486
|
|
|
727
|
-
if (!s.resolvedPrerenderModules?.size && !s.resolvedStaticModules?.size)
|
|
1487
|
+
if (!s.resolvedPrerenderModules?.size && !s.resolvedStaticModules?.size) {
|
|
1488
|
+
debugBuild?.(
|
|
1489
|
+
"generateBundle (rsc): no handlers to scan (%sms)",
|
|
1490
|
+
(performance.now() - genStart).toFixed(1),
|
|
1491
|
+
);
|
|
728
1492
|
return;
|
|
1493
|
+
}
|
|
1494
|
+
|
|
1495
|
+
// Clear maps at the start of each RSC generateBundle pass.
|
|
1496
|
+
// Vite 6 multi-environment builds run RSC twice (analysis + production);
|
|
1497
|
+
// clearing prevents stale/duplicate records from the analysis pass.
|
|
1498
|
+
s.handlerChunkInfoMap.clear();
|
|
1499
|
+
s.staticHandlerChunkInfoMap.clear();
|
|
729
1500
|
|
|
730
1501
|
for (const [fileName, chunk] of Object.entries(bundle) as [
|
|
731
1502
|
string,
|
|
@@ -733,27 +1504,28 @@ export function createRouterDiscoveryPlugin(
|
|
|
733
1504
|
][]) {
|
|
734
1505
|
if (chunk.type !== "chunk") continue;
|
|
735
1506
|
|
|
736
|
-
//
|
|
737
|
-
if (
|
|
738
|
-
fileName.includes("__prerender-handlers") &&
|
|
739
|
-
s.resolvedPrerenderModules?.size
|
|
740
|
-
) {
|
|
1507
|
+
// Scan all chunks for handler exports (handlers may land in any chunk)
|
|
1508
|
+
if (s.resolvedPrerenderModules?.size) {
|
|
741
1509
|
const handlers = extractHandlerExportsFromChunk(
|
|
742
1510
|
chunk.code,
|
|
743
1511
|
s.resolvedPrerenderModules,
|
|
744
1512
|
"Prerender",
|
|
745
|
-
|
|
1513
|
+
false,
|
|
746
1514
|
);
|
|
747
1515
|
if (handlers.length > 0) {
|
|
748
|
-
|
|
1516
|
+
const existing = s.handlerChunkInfoMap.get(fileName);
|
|
1517
|
+
if (existing) {
|
|
1518
|
+
existing.exports.push(...handlers);
|
|
1519
|
+
} else {
|
|
1520
|
+
s.handlerChunkInfoMap.set(fileName, {
|
|
1521
|
+
fileName,
|
|
1522
|
+
exports: handlers,
|
|
1523
|
+
});
|
|
1524
|
+
}
|
|
749
1525
|
}
|
|
750
1526
|
}
|
|
751
1527
|
|
|
752
|
-
|
|
753
|
-
if (
|
|
754
|
-
fileName.includes("__static-handlers") &&
|
|
755
|
-
s.resolvedStaticModules?.size
|
|
756
|
-
) {
|
|
1528
|
+
if (s.resolvedStaticModules?.size) {
|
|
757
1529
|
const handlers = extractHandlerExportsFromChunk(
|
|
758
1530
|
chunk.code,
|
|
759
1531
|
s.resolvedStaticModules,
|
|
@@ -761,10 +1533,26 @@ export function createRouterDiscoveryPlugin(
|
|
|
761
1533
|
false,
|
|
762
1534
|
);
|
|
763
1535
|
if (handlers.length > 0) {
|
|
764
|
-
|
|
1536
|
+
const existing = s.staticHandlerChunkInfoMap.get(fileName);
|
|
1537
|
+
if (existing) {
|
|
1538
|
+
existing.exports.push(...handlers);
|
|
1539
|
+
} else {
|
|
1540
|
+
s.staticHandlerChunkInfoMap.set(fileName, {
|
|
1541
|
+
fileName,
|
|
1542
|
+
exports: handlers,
|
|
1543
|
+
});
|
|
1544
|
+
}
|
|
765
1545
|
}
|
|
766
1546
|
}
|
|
767
1547
|
}
|
|
1548
|
+
|
|
1549
|
+
debugBuild?.(
|
|
1550
|
+
"generateBundle (rsc): scanned %d chunks, %d prerender chunk(s), %d static chunk(s) (%sms)",
|
|
1551
|
+
Object.keys(bundle).length,
|
|
1552
|
+
s.handlerChunkInfoMap.size,
|
|
1553
|
+
s.staticHandlerChunkInfoMap.size,
|
|
1554
|
+
(performance.now() - genStart).toFixed(1),
|
|
1555
|
+
);
|
|
768
1556
|
},
|
|
769
1557
|
|
|
770
1558
|
// Build-time pre-rendering: evict handler code and inject collected prerender data.
|
|
@@ -778,7 +1566,9 @@ export function createRouterDiscoveryPlugin(
|
|
|
778
1566
|
// Only run for the RSC environment — other environments (client, ssr) have
|
|
779
1567
|
// no prerender/static data to process and would just do redundant file I/O.
|
|
780
1568
|
if (this.environment && this.environment.name !== "rsc") return;
|
|
781
|
-
postprocessBundle(
|
|
1569
|
+
timedSync(debugBuild, "closeBundle postprocessBundle", () =>
|
|
1570
|
+
postprocessBundle(s),
|
|
1571
|
+
);
|
|
782
1572
|
},
|
|
783
1573
|
},
|
|
784
1574
|
};
|