@rangojs/router 0.0.0-experimental.002d056c
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 +9 -0
- package/README.md +899 -0
- package/dist/bin/rango.js +1606 -0
- package/dist/vite/index.js +5153 -0
- package/package.json +177 -0
- package/skills/breadcrumbs/SKILL.md +250 -0
- package/skills/cache-guide/SKILL.md +262 -0
- package/skills/caching/SKILL.md +253 -0
- package/skills/composability/SKILL.md +172 -0
- package/skills/debug-manifest/SKILL.md +112 -0
- package/skills/document-cache/SKILL.md +182 -0
- package/skills/fonts/SKILL.md +167 -0
- package/skills/hooks/SKILL.md +704 -0
- package/skills/host-router/SKILL.md +218 -0
- package/skills/intercept/SKILL.md +313 -0
- package/skills/layout/SKILL.md +310 -0
- package/skills/links/SKILL.md +239 -0
- package/skills/loader/SKILL.md +596 -0
- package/skills/middleware/SKILL.md +339 -0
- package/skills/mime-routes/SKILL.md +128 -0
- package/skills/parallel/SKILL.md +305 -0
- package/skills/prerender/SKILL.md +643 -0
- package/skills/rango/SKILL.md +118 -0
- package/skills/response-routes/SKILL.md +411 -0
- package/skills/route/SKILL.md +385 -0
- package/skills/router-setup/SKILL.md +439 -0
- package/skills/tailwind/SKILL.md +129 -0
- package/skills/theme/SKILL.md +79 -0
- package/skills/typesafety/SKILL.md +623 -0
- package/skills/use-cache/SKILL.md +324 -0
- package/src/__internal.ts +273 -0
- package/src/bin/rango.ts +321 -0
- package/src/browser/action-coordinator.ts +97 -0
- package/src/browser/action-response-classifier.ts +99 -0
- package/src/browser/event-controller.ts +899 -0
- package/src/browser/history-state.ts +80 -0
- package/src/browser/index.ts +18 -0
- package/src/browser/intercept-utils.ts +52 -0
- package/src/browser/link-interceptor.ts +141 -0
- package/src/browser/logging.ts +55 -0
- package/src/browser/merge-segment-loaders.ts +134 -0
- package/src/browser/navigation-bridge.ts +638 -0
- package/src/browser/navigation-client.ts +261 -0
- package/src/browser/navigation-store.ts +806 -0
- package/src/browser/navigation-transaction.ts +297 -0
- package/src/browser/network-error-handler.ts +61 -0
- package/src/browser/partial-update.ts +582 -0
- package/src/browser/prefetch/cache.ts +206 -0
- package/src/browser/prefetch/fetch.ts +145 -0
- package/src/browser/prefetch/observer.ts +65 -0
- package/src/browser/prefetch/policy.ts +48 -0
- package/src/browser/prefetch/queue.ts +128 -0
- package/src/browser/rango-state.ts +112 -0
- package/src/browser/react/Link.tsx +368 -0
- package/src/browser/react/NavigationProvider.tsx +413 -0
- package/src/browser/react/ScrollRestoration.tsx +94 -0
- package/src/browser/react/context.ts +59 -0
- package/src/browser/react/filter-segment-order.ts +11 -0
- package/src/browser/react/index.ts +52 -0
- package/src/browser/react/location-state-shared.ts +162 -0
- package/src/browser/react/location-state.ts +107 -0
- package/src/browser/react/mount-context.ts +37 -0
- package/src/browser/react/nonce-context.ts +23 -0
- package/src/browser/react/shallow-equal.ts +27 -0
- package/src/browser/react/use-action.ts +218 -0
- package/src/browser/react/use-client-cache.ts +58 -0
- package/src/browser/react/use-handle.ts +162 -0
- package/src/browser/react/use-href.tsx +40 -0
- package/src/browser/react/use-link-status.ts +135 -0
- package/src/browser/react/use-mount.ts +31 -0
- package/src/browser/react/use-navigation.ts +99 -0
- package/src/browser/react/use-params.ts +65 -0
- package/src/browser/react/use-pathname.ts +47 -0
- package/src/browser/react/use-router.ts +63 -0
- package/src/browser/react/use-search-params.ts +56 -0
- package/src/browser/react/use-segments.ts +171 -0
- package/src/browser/response-adapter.ts +73 -0
- package/src/browser/rsc-router.tsx +464 -0
- package/src/browser/scroll-restoration.ts +397 -0
- package/src/browser/segment-reconciler.ts +216 -0
- package/src/browser/segment-structure-assert.ts +83 -0
- package/src/browser/server-action-bridge.ts +667 -0
- package/src/browser/shallow.ts +40 -0
- package/src/browser/types.ts +547 -0
- package/src/browser/validate-redirect-origin.ts +29 -0
- package/src/build/generate-manifest.ts +438 -0
- package/src/build/generate-route-types.ts +36 -0
- package/src/build/index.ts +35 -0
- package/src/build/route-trie.ts +265 -0
- package/src/build/route-types/ast-helpers.ts +25 -0
- package/src/build/route-types/ast-route-extraction.ts +98 -0
- package/src/build/route-types/codegen.ts +102 -0
- package/src/build/route-types/include-resolution.ts +411 -0
- package/src/build/route-types/param-extraction.ts +48 -0
- package/src/build/route-types/per-module-writer.ts +128 -0
- package/src/build/route-types/router-processing.ts +479 -0
- package/src/build/route-types/scan-filter.ts +78 -0
- package/src/build/runtime-discovery.ts +231 -0
- package/src/cache/background-task.ts +34 -0
- package/src/cache/cache-key-utils.ts +44 -0
- package/src/cache/cache-policy.ts +125 -0
- package/src/cache/cache-runtime.ts +338 -0
- package/src/cache/cache-scope.ts +382 -0
- package/src/cache/cf/cf-cache-store.ts +982 -0
- package/src/cache/cf/index.ts +29 -0
- package/src/cache/document-cache.ts +369 -0
- package/src/cache/handle-capture.ts +81 -0
- package/src/cache/handle-snapshot.ts +41 -0
- package/src/cache/index.ts +44 -0
- package/src/cache/memory-segment-store.ts +328 -0
- package/src/cache/profile-registry.ts +73 -0
- package/src/cache/read-through-swr.ts +134 -0
- package/src/cache/segment-codec.ts +256 -0
- package/src/cache/taint.ts +98 -0
- package/src/cache/types.ts +342 -0
- package/src/client.rsc.tsx +85 -0
- package/src/client.tsx +601 -0
- package/src/component-utils.ts +76 -0
- package/src/components/DefaultDocument.tsx +27 -0
- package/src/context-var.ts +86 -0
- package/src/debug.ts +243 -0
- package/src/default-error-boundary.tsx +88 -0
- package/src/deps/browser.ts +8 -0
- package/src/deps/html-stream-client.ts +2 -0
- package/src/deps/html-stream-server.ts +2 -0
- package/src/deps/rsc.ts +10 -0
- package/src/deps/ssr.ts +2 -0
- package/src/errors.ts +365 -0
- package/src/handle.ts +135 -0
- package/src/handles/MetaTags.tsx +246 -0
- package/src/handles/breadcrumbs.ts +66 -0
- package/src/handles/index.ts +7 -0
- package/src/handles/meta.ts +264 -0
- package/src/host/cookie-handler.ts +165 -0
- package/src/host/errors.ts +97 -0
- package/src/host/index.ts +53 -0
- package/src/host/pattern-matcher.ts +214 -0
- package/src/host/router.ts +352 -0
- package/src/host/testing.ts +79 -0
- package/src/host/types.ts +146 -0
- package/src/host/utils.ts +25 -0
- package/src/href-client.ts +222 -0
- package/src/index.rsc.ts +233 -0
- package/src/index.ts +277 -0
- package/src/internal-debug.ts +11 -0
- package/src/loader.rsc.ts +89 -0
- package/src/loader.ts +64 -0
- package/src/network-error-thrower.tsx +23 -0
- package/src/outlet-context.ts +15 -0
- package/src/outlet-provider.tsx +45 -0
- package/src/prerender/param-hash.ts +37 -0
- package/src/prerender/store.ts +185 -0
- package/src/prerender.ts +463 -0
- package/src/reverse.ts +330 -0
- package/src/root-error-boundary.tsx +289 -0
- package/src/route-content-wrapper.tsx +196 -0
- package/src/route-definition/dsl-helpers.ts +934 -0
- package/src/route-definition/helper-factories.ts +200 -0
- package/src/route-definition/helpers-types.ts +430 -0
- package/src/route-definition/index.ts +52 -0
- package/src/route-definition/redirect.ts +93 -0
- package/src/route-definition.ts +1 -0
- package/src/route-map-builder.ts +281 -0
- package/src/route-name.ts +53 -0
- package/src/route-types.ts +259 -0
- package/src/router/content-negotiation.ts +116 -0
- package/src/router/debug-manifest.ts +72 -0
- package/src/router/error-handling.ts +287 -0
- package/src/router/find-match.ts +160 -0
- package/src/router/handler-context.ts +451 -0
- package/src/router/intercept-resolution.ts +397 -0
- package/src/router/lazy-includes.ts +236 -0
- package/src/router/loader-resolution.ts +420 -0
- package/src/router/logging.ts +251 -0
- package/src/router/manifest.ts +269 -0
- package/src/router/match-api.ts +620 -0
- package/src/router/match-context.ts +266 -0
- package/src/router/match-handlers.ts +440 -0
- package/src/router/match-middleware/background-revalidation.ts +223 -0
- package/src/router/match-middleware/cache-lookup.ts +634 -0
- package/src/router/match-middleware/cache-store.ts +295 -0
- package/src/router/match-middleware/index.ts +81 -0
- package/src/router/match-middleware/intercept-resolution.ts +306 -0
- package/src/router/match-middleware/segment-resolution.ts +193 -0
- package/src/router/match-pipelines.ts +179 -0
- package/src/router/match-result.ts +219 -0
- package/src/router/metrics.ts +282 -0
- package/src/router/middleware-cookies.ts +55 -0
- package/src/router/middleware-types.ts +222 -0
- package/src/router/middleware.ts +749 -0
- package/src/router/pattern-matching.ts +563 -0
- package/src/router/prerender-match.ts +402 -0
- package/src/router/preview-match.ts +170 -0
- package/src/router/revalidation.ts +289 -0
- package/src/router/router-context.ts +320 -0
- package/src/router/router-interfaces.ts +452 -0
- package/src/router/router-options.ts +592 -0
- package/src/router/router-registry.ts +24 -0
- package/src/router/segment-resolution/fresh.ts +570 -0
- package/src/router/segment-resolution/helpers.ts +263 -0
- package/src/router/segment-resolution/loader-cache.ts +198 -0
- package/src/router/segment-resolution/revalidation.ts +1242 -0
- package/src/router/segment-resolution/static-store.ts +67 -0
- package/src/router/segment-resolution.ts +21 -0
- package/src/router/segment-wrappers.ts +291 -0
- package/src/router/telemetry-otel.ts +299 -0
- package/src/router/telemetry.ts +300 -0
- package/src/router/timeout.ts +148 -0
- package/src/router/trie-matching.ts +239 -0
- package/src/router/types.ts +170 -0
- package/src/router.ts +1006 -0
- package/src/rsc/handler-context.ts +45 -0
- package/src/rsc/handler.ts +1089 -0
- package/src/rsc/helpers.ts +198 -0
- package/src/rsc/index.ts +36 -0
- package/src/rsc/loader-fetch.ts +209 -0
- package/src/rsc/manifest-init.ts +86 -0
- package/src/rsc/nonce.ts +32 -0
- package/src/rsc/origin-guard.ts +141 -0
- package/src/rsc/progressive-enhancement.ts +379 -0
- package/src/rsc/response-error.ts +37 -0
- package/src/rsc/response-route-handler.ts +347 -0
- package/src/rsc/rsc-rendering.ts +237 -0
- package/src/rsc/runtime-warnings.ts +42 -0
- package/src/rsc/server-action.ts +348 -0
- package/src/rsc/ssr-setup.ts +128 -0
- package/src/rsc/types.ts +263 -0
- package/src/search-params.ts +230 -0
- package/src/segment-system.tsx +454 -0
- package/src/server/context.ts +591 -0
- package/src/server/cookie-store.ts +190 -0
- package/src/server/fetchable-loader-store.ts +37 -0
- package/src/server/handle-store.ts +308 -0
- package/src/server/loader-registry.ts +133 -0
- package/src/server/request-context.ts +920 -0
- package/src/server/root-layout.tsx +10 -0
- package/src/server/tsconfig.json +14 -0
- package/src/server.ts +51 -0
- package/src/ssr/index.tsx +365 -0
- package/src/static-handler.ts +114 -0
- package/src/theme/ThemeProvider.tsx +297 -0
- package/src/theme/ThemeScript.tsx +61 -0
- package/src/theme/constants.ts +62 -0
- package/src/theme/index.ts +48 -0
- package/src/theme/theme-context.ts +44 -0
- package/src/theme/theme-script.ts +155 -0
- package/src/theme/types.ts +182 -0
- package/src/theme/use-theme.ts +44 -0
- package/src/types/boundaries.ts +158 -0
- package/src/types/cache-types.ts +198 -0
- package/src/types/error-types.ts +192 -0
- package/src/types/global-namespace.ts +100 -0
- package/src/types/handler-context.ts +687 -0
- package/src/types/index.ts +88 -0
- package/src/types/loader-types.ts +183 -0
- package/src/types/route-config.ts +170 -0
- package/src/types/route-entry.ts +109 -0
- package/src/types/segments.ts +148 -0
- package/src/types.ts +1 -0
- package/src/urls/include-helper.ts +197 -0
- package/src/urls/index.ts +53 -0
- package/src/urls/path-helper-types.ts +339 -0
- package/src/urls/path-helper.ts +329 -0
- package/src/urls/pattern-types.ts +95 -0
- package/src/urls/response-types.ts +106 -0
- package/src/urls/type-extraction.ts +372 -0
- package/src/urls/urls-function.ts +98 -0
- package/src/urls.ts +1 -0
- package/src/use-loader.tsx +354 -0
- package/src/vite/discovery/bundle-postprocess.ts +184 -0
- package/src/vite/discovery/discover-routers.ts +344 -0
- package/src/vite/discovery/prerender-collection.ts +385 -0
- package/src/vite/discovery/route-types-writer.ts +258 -0
- package/src/vite/discovery/self-gen-tracking.ts +47 -0
- package/src/vite/discovery/state.ts +108 -0
- package/src/vite/discovery/virtual-module-codegen.ts +203 -0
- package/src/vite/index.ts +16 -0
- package/src/vite/plugin-types.ts +48 -0
- package/src/vite/plugins/cjs-to-esm.ts +93 -0
- package/src/vite/plugins/client-ref-dedup.ts +115 -0
- package/src/vite/plugins/client-ref-hashing.ts +105 -0
- package/src/vite/plugins/expose-action-id.ts +363 -0
- package/src/vite/plugins/expose-id-utils.ts +287 -0
- package/src/vite/plugins/expose-ids/export-analysis.ts +296 -0
- package/src/vite/plugins/expose-ids/handler-transform.ts +179 -0
- package/src/vite/plugins/expose-ids/loader-transform.ts +74 -0
- package/src/vite/plugins/expose-ids/router-transform.ts +110 -0
- package/src/vite/plugins/expose-ids/types.ts +45 -0
- package/src/vite/plugins/expose-internal-ids.ts +569 -0
- package/src/vite/plugins/refresh-cmd.ts +65 -0
- package/src/vite/plugins/use-cache-transform.ts +323 -0
- package/src/vite/plugins/version-injector.ts +83 -0
- package/src/vite/plugins/version-plugin.ts +266 -0
- package/src/vite/plugins/version.d.ts +12 -0
- package/src/vite/plugins/virtual-entries.ts +123 -0
- package/src/vite/plugins/virtual-stub-plugin.ts +29 -0
- package/src/vite/rango.ts +445 -0
- package/src/vite/router-discovery.ts +777 -0
- package/src/vite/utils/ast-handler-extract.ts +517 -0
- package/src/vite/utils/banner.ts +36 -0
- package/src/vite/utils/bundle-analysis.ts +137 -0
- package/src/vite/utils/manifest-utils.ts +70 -0
- package/src/vite/utils/package-resolution.ts +121 -0
- package/src/vite/utils/prerender-utils.ts +189 -0
- package/src/vite/utils/shared-utils.ts +169 -0
|
@@ -0,0 +1,777 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Router Discovery Plugin
|
|
3
|
+
*
|
|
4
|
+
* Vite plugin that discovers router instances at dev/build time via the RSC
|
|
5
|
+
* environment. Delegates to extracted modules for discovery, route types
|
|
6
|
+
* generation, virtual module codegen, and bundle post-processing.
|
|
7
|
+
*/
|
|
8
|
+
|
|
9
|
+
import type { Plugin } from "vite";
|
|
10
|
+
import { createServer as createViteServer } from "vite";
|
|
11
|
+
import { resolve } from "node:path";
|
|
12
|
+
import { readFileSync } from "node:fs";
|
|
13
|
+
import {
|
|
14
|
+
formatNestedRouterConflictError,
|
|
15
|
+
findNestedRouterConflict,
|
|
16
|
+
findRouterFiles,
|
|
17
|
+
} from "../build/generate-route-types.js";
|
|
18
|
+
import { createVersionPlugin } from "./plugins/version-plugin.js";
|
|
19
|
+
import { createVirtualStubPlugin } from "./plugins/virtual-stub-plugin.js";
|
|
20
|
+
import {
|
|
21
|
+
exposeInternalIds,
|
|
22
|
+
exposeRouterId,
|
|
23
|
+
} from "./plugins/expose-internal-ids.js";
|
|
24
|
+
import { hashClientRefs } from "./plugins/client-ref-hashing.js";
|
|
25
|
+
import { extractHandlerExportsFromChunk } from "./utils/bundle-analysis.js";
|
|
26
|
+
import {
|
|
27
|
+
createDiscoveryState,
|
|
28
|
+
VIRTUAL_ROUTES_MANIFEST_ID,
|
|
29
|
+
type DiscoveryState,
|
|
30
|
+
type PluginOptions,
|
|
31
|
+
} from "./discovery/state.js";
|
|
32
|
+
import { consumeSelfGenWrite } from "./discovery/self-gen-tracking.js";
|
|
33
|
+
import { discoverRouters } from "./discovery/discover-routers.js";
|
|
34
|
+
import {
|
|
35
|
+
writeCombinedRouteTypesWithTracking,
|
|
36
|
+
writeRouteTypesFiles,
|
|
37
|
+
supplementGenFilesWithRuntimeRoutes,
|
|
38
|
+
} from "./discovery/route-types-writer.js";
|
|
39
|
+
import {
|
|
40
|
+
generateRoutesManifestModule,
|
|
41
|
+
generatePerRouterModule,
|
|
42
|
+
} from "./discovery/virtual-module-codegen.js";
|
|
43
|
+
import { postprocessBundle } from "./discovery/bundle-postprocess.js";
|
|
44
|
+
import { resetStagedBuildAssets } from "./utils/prerender-utils.js";
|
|
45
|
+
|
|
46
|
+
export { VIRTUAL_ROUTES_MANIFEST_ID };
|
|
47
|
+
|
|
48
|
+
// ============================================================================
|
|
49
|
+
// Temp Server Factory
|
|
50
|
+
// ============================================================================
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Create a minimal Vite server for router discovery.
|
|
54
|
+
*
|
|
55
|
+
* Both dev-mode prerender and build-mode discovery need a temp RSC server
|
|
56
|
+
* to import user router files via module runner. This factory centralizes
|
|
57
|
+
* the shared config and the mode-specific differences:
|
|
58
|
+
* - Dev: path-based IDs (no forceBuild), separate cacheDir
|
|
59
|
+
* - Build: hashed IDs (forceBuild), hashClientRefs for production bundles
|
|
60
|
+
*
|
|
61
|
+
* Returns the ViteDevServer instance. Callers access .environments.rsc as needed.
|
|
62
|
+
*/
|
|
63
|
+
async function createTempRscServer(
|
|
64
|
+
state: DiscoveryState,
|
|
65
|
+
options: { forceBuild?: boolean; cacheDir?: string } = {},
|
|
66
|
+
) {
|
|
67
|
+
const { default: rsc } = await import("@vitejs/plugin-rsc");
|
|
68
|
+
return createViteServer({
|
|
69
|
+
root: state.projectRoot,
|
|
70
|
+
configFile: false,
|
|
71
|
+
server: { middlewareMode: true },
|
|
72
|
+
appType: "custom",
|
|
73
|
+
logLevel: "silent",
|
|
74
|
+
resolve: { alias: state.userResolveAlias },
|
|
75
|
+
esbuild: { jsx: "automatic", jsxImportSource: "react" },
|
|
76
|
+
...(options.cacheDir && { cacheDir: options.cacheDir }),
|
|
77
|
+
plugins: [
|
|
78
|
+
rsc({
|
|
79
|
+
entries: {
|
|
80
|
+
client: "virtual:entry-client",
|
|
81
|
+
ssr: "virtual:entry-ssr",
|
|
82
|
+
rsc: state.resolvedEntryPath!,
|
|
83
|
+
},
|
|
84
|
+
}),
|
|
85
|
+
// hashClientRefs only in build mode — production bundles need hashed refs
|
|
86
|
+
...(options.forceBuild ? [hashClientRefs(state.projectRoot)] : []),
|
|
87
|
+
createVersionPlugin(),
|
|
88
|
+
createVirtualStubPlugin(),
|
|
89
|
+
// Dev prerender must use dev-mode IDs (path-based) to match the workerd
|
|
90
|
+
// runtime. forceBuild produces hashed IDs for production bundle consistency.
|
|
91
|
+
exposeInternalIds(options.forceBuild ? { forceBuild: true } : undefined),
|
|
92
|
+
exposeRouterId(),
|
|
93
|
+
],
|
|
94
|
+
});
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
/**
|
|
98
|
+
* Plugin that discovers router instances at dev/build time via the RSC environment.
|
|
99
|
+
*
|
|
100
|
+
* Uses `server.environments.rsc.runner.import()` to load the user's router file
|
|
101
|
+
* with full TS/TSX compilation. This triggers `createRouter()` which populates
|
|
102
|
+
* the `RouterRegistry`. The plugin then generates manifests for each router.
|
|
103
|
+
*
|
|
104
|
+
* In dev mode, this runs in `configureServer` (post-middleware setup).
|
|
105
|
+
* In build mode, this will run in `buildStart` (future).
|
|
106
|
+
*
|
|
107
|
+
* @internal
|
|
108
|
+
*/
|
|
109
|
+
export function createRouterDiscoveryPlugin(
|
|
110
|
+
entryPath: string | undefined,
|
|
111
|
+
opts?: PluginOptions,
|
|
112
|
+
): Plugin {
|
|
113
|
+
const s = createDiscoveryState(entryPath, opts);
|
|
114
|
+
|
|
115
|
+
return {
|
|
116
|
+
name: "@rangojs/router:discovery",
|
|
117
|
+
|
|
118
|
+
config() {
|
|
119
|
+
const config: any = {
|
|
120
|
+
define: {
|
|
121
|
+
__RANGO_DEBUG__: JSON.stringify(!!process.env.INTERNAL_RANGO_DEBUG),
|
|
122
|
+
},
|
|
123
|
+
};
|
|
124
|
+
if (opts?.enableBuildPrerender) {
|
|
125
|
+
config.environments = {
|
|
126
|
+
rsc: {
|
|
127
|
+
build: {
|
|
128
|
+
rollupOptions: {
|
|
129
|
+
output: {
|
|
130
|
+
manualChunks(id: string) {
|
|
131
|
+
if (s.resolvedPrerenderModules?.has(id)) {
|
|
132
|
+
return "__prerender-handlers";
|
|
133
|
+
}
|
|
134
|
+
if (s.resolvedStaticModules?.has(id)) {
|
|
135
|
+
return "__static-handlers";
|
|
136
|
+
}
|
|
137
|
+
},
|
|
138
|
+
},
|
|
139
|
+
},
|
|
140
|
+
},
|
|
141
|
+
},
|
|
142
|
+
};
|
|
143
|
+
}
|
|
144
|
+
return config;
|
|
145
|
+
},
|
|
146
|
+
|
|
147
|
+
configResolved(config) {
|
|
148
|
+
s.projectRoot = config.root;
|
|
149
|
+
s.isBuildMode = config.command === "build";
|
|
150
|
+
// Capture user's resolve aliases for the temp server
|
|
151
|
+
s.userResolveAlias = config.resolve.alias;
|
|
152
|
+
// Node preset: pick up auto-discovered router path from the config() hook.
|
|
153
|
+
// The auto-discover plugin runs in config() using Vite's resolved root,
|
|
154
|
+
// populating the mutable ref before configResolved fires.
|
|
155
|
+
if (!s.resolvedEntryPath && opts?.routerPathRef?.path) {
|
|
156
|
+
s.resolvedEntryPath = opts.routerPathRef.path;
|
|
157
|
+
}
|
|
158
|
+
// Cloudflare preset: read entry from resolved environment config.
|
|
159
|
+
// The @cloudflare/vite-plugin reads wrangler config (toml/json/jsonc)
|
|
160
|
+
// and sets optimizeDeps.entries on the RSC environment.
|
|
161
|
+
if (!s.resolvedEntryPath) {
|
|
162
|
+
const rscEnvConfig = (config.environments as any)?.["rsc"];
|
|
163
|
+
const entries = rscEnvConfig?.optimizeDeps?.entries;
|
|
164
|
+
if (typeof entries === "string") {
|
|
165
|
+
s.resolvedEntryPath = entries;
|
|
166
|
+
} else if (Array.isArray(entries) && entries.length > 0) {
|
|
167
|
+
s.resolvedEntryPath = entries[0];
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
// Generate combined named-routes.gen.ts from static source parsing.
|
|
171
|
+
// Runs before the dev server starts so the gen file exists immediately for IDE.
|
|
172
|
+
// In build mode, the runtime discovery in buildStart produces the definitive
|
|
173
|
+
// named-routes.gen.ts (including dynamically generated routes).
|
|
174
|
+
// preserveIfLarger prevents overwriting a previously generated complete
|
|
175
|
+
// file with a partial one.
|
|
176
|
+
if (opts?.staticRouteTypesGeneration !== false) {
|
|
177
|
+
s.cachedRouterFiles = findRouterFiles(s.projectRoot, s.scanFilter);
|
|
178
|
+
writeCombinedRouteTypesWithTracking(s, { preserveIfLarger: true });
|
|
179
|
+
}
|
|
180
|
+
// Resolve prerenderHandlerModules and staticHandlerModules from the consolidated IDs plugin's API.
|
|
181
|
+
if (opts?.enableBuildPrerender) {
|
|
182
|
+
const idsPlugin = config.plugins.find(
|
|
183
|
+
(p: any) => p.name === "@rangojs/router:expose-internal-ids",
|
|
184
|
+
);
|
|
185
|
+
s.resolvedPrerenderModules = (
|
|
186
|
+
idsPlugin?.api as any
|
|
187
|
+
)?.prerenderHandlerModules;
|
|
188
|
+
s.resolvedStaticModules = (idsPlugin?.api as any)?.staticHandlerModules;
|
|
189
|
+
}
|
|
190
|
+
},
|
|
191
|
+
|
|
192
|
+
// Dev mode: discover routers and populate manifest in memory.
|
|
193
|
+
// Skipped in build mode (buildStart handles it).
|
|
194
|
+
configureServer(server) {
|
|
195
|
+
if (s.isBuildMode) return;
|
|
196
|
+
// Skip if this is a temp server created by buildStart
|
|
197
|
+
if ((globalThis as any).__rscRouterDiscoveryActive) return;
|
|
198
|
+
s.devServer = server;
|
|
199
|
+
|
|
200
|
+
// Discovery promise that the handler can await if requests arrive
|
|
201
|
+
// before discovery completes
|
|
202
|
+
let resolveDiscovery: () => void;
|
|
203
|
+
const discoveryPromise = new Promise<void>((resolve) => {
|
|
204
|
+
resolveDiscovery = resolve;
|
|
205
|
+
});
|
|
206
|
+
|
|
207
|
+
// Compute dev server origin from resolved URLs (preferred) or config port (fallback).
|
|
208
|
+
// Called after discovery (or in the load hook) when the server may be listening.
|
|
209
|
+
const getDevServerOrigin = () =>
|
|
210
|
+
server.resolvedUrls?.local?.[0]?.replace(/\/$/, "") ||
|
|
211
|
+
`http://localhost:${server.config.server.port || 5173}`;
|
|
212
|
+
|
|
213
|
+
// Shared temp server for Cloudflare dev (no module runner in workerd).
|
|
214
|
+
// Used by both discover() (route type generation) and the prerender
|
|
215
|
+
// middleware (on-demand prerender evaluation). Created lazily, closed on
|
|
216
|
+
// server shutdown.
|
|
217
|
+
let prerenderTempServer: any = null;
|
|
218
|
+
let prerenderNodeRegistry: Map<string, any> | null = null;
|
|
219
|
+
|
|
220
|
+
// Clean up the temporary server when the dev server shuts down
|
|
221
|
+
server.httpServer?.on("close", () => {
|
|
222
|
+
if (prerenderTempServer) {
|
|
223
|
+
prerenderTempServer.close().catch(() => {});
|
|
224
|
+
prerenderTempServer = null;
|
|
225
|
+
}
|
|
226
|
+
});
|
|
227
|
+
|
|
228
|
+
async function getOrCreateTempServer(): Promise<any | null> {
|
|
229
|
+
if (prerenderNodeRegistry) {
|
|
230
|
+
return (prerenderTempServer.environments as any)?.rsc ?? null;
|
|
231
|
+
}
|
|
232
|
+
try {
|
|
233
|
+
prerenderTempServer = await createTempRscServer(s, {
|
|
234
|
+
cacheDir: "node_modules/.vite_prerender",
|
|
235
|
+
});
|
|
236
|
+
|
|
237
|
+
const tempRscEnv = (prerenderTempServer.environments as any)?.rsc;
|
|
238
|
+
if (tempRscEnv?.runner) {
|
|
239
|
+
await tempRscEnv.runner.import(s.resolvedEntryPath!);
|
|
240
|
+
const serverMod = await tempRscEnv.runner.import(
|
|
241
|
+
"@rangojs/router/server",
|
|
242
|
+
);
|
|
243
|
+
prerenderNodeRegistry = serverMod.RouterRegistry;
|
|
244
|
+
return tempRscEnv;
|
|
245
|
+
}
|
|
246
|
+
} catch (err: any) {
|
|
247
|
+
console.warn(
|
|
248
|
+
`[rsc-router] Failed to create temp runner: ${err.message}`,
|
|
249
|
+
);
|
|
250
|
+
}
|
|
251
|
+
return null;
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
const discover = async () => {
|
|
255
|
+
const rscEnv = (server.environments as any)?.rsc;
|
|
256
|
+
if (!rscEnv?.runner) {
|
|
257
|
+
// Cloudflare dev: no module runner available (workerd-based RSC env).
|
|
258
|
+
// Set devServerOrigin so the virtual module can inject __PRERENDER_DEV_URL
|
|
259
|
+
// for on-demand prerender via the /__rsc_prerender endpoint.
|
|
260
|
+
s.devServerOrigin = getDevServerOrigin();
|
|
261
|
+
|
|
262
|
+
// Create a temp Node.js server to run runtime discovery and generate
|
|
263
|
+
// named route types (static parser can't resolve factory calls).
|
|
264
|
+
try {
|
|
265
|
+
const tempRscEnv = await getOrCreateTempServer();
|
|
266
|
+
if (tempRscEnv) {
|
|
267
|
+
await discoverRouters(s, tempRscEnv);
|
|
268
|
+
writeRouteTypesFiles(s);
|
|
269
|
+
}
|
|
270
|
+
} catch (err: any) {
|
|
271
|
+
console.warn(
|
|
272
|
+
`[rsc-router] Cloudflare dev discovery failed: ${err.message}\n${err.stack}`,
|
|
273
|
+
);
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
resolveDiscovery!();
|
|
277
|
+
return;
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
try {
|
|
281
|
+
// Set the readiness gate BEFORE discovery so early requests
|
|
282
|
+
// block until manifest is populated
|
|
283
|
+
const serverMod = await rscEnv.runner.import(
|
|
284
|
+
"@rangojs/router/server",
|
|
285
|
+
);
|
|
286
|
+
if (serverMod?.setManifestReadyPromise) {
|
|
287
|
+
serverMod.setManifestReadyPromise(discoveryPromise);
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
await discoverRouters(s, rscEnv);
|
|
291
|
+
|
|
292
|
+
// Store server origin for dev prerender endpoint (virtual module injection)
|
|
293
|
+
s.devServerOrigin = getDevServerOrigin();
|
|
294
|
+
|
|
295
|
+
// Update named-routes.gen.ts from runtime discovery.
|
|
296
|
+
// The runtime manifest is the source of truth: it evaluates dynamic
|
|
297
|
+
// routes (e.g. Array.from loops) that the static parser cannot see.
|
|
298
|
+
// writeRouteTypesFiles() only writes when content changes, so this
|
|
299
|
+
// won't cause unnecessary HMR triggers.
|
|
300
|
+
writeRouteTypesFiles(s);
|
|
301
|
+
|
|
302
|
+
// Populate the route map and per-router data in the RSC env
|
|
303
|
+
await propagateDiscoveryState(rscEnv);
|
|
304
|
+
} catch (err: any) {
|
|
305
|
+
console.warn(
|
|
306
|
+
`[rsc-router] Router discovery failed: ${err.message}\n${err.stack}`,
|
|
307
|
+
);
|
|
308
|
+
} finally {
|
|
309
|
+
resolveDiscovery!();
|
|
310
|
+
}
|
|
311
|
+
};
|
|
312
|
+
|
|
313
|
+
// Schedule after all plugins have finished configureServer.
|
|
314
|
+
// Store the promise so the virtual module's load hook can await it.
|
|
315
|
+
s.discoveryDone = new Promise<void>((resolve) => {
|
|
316
|
+
setTimeout(() => discover().then(resolve, resolve), 0);
|
|
317
|
+
});
|
|
318
|
+
|
|
319
|
+
// Dev-mode on-demand prerender endpoint.
|
|
320
|
+
// When workerd hits a prerender route, it fetches this endpoint instead of
|
|
321
|
+
// trying to run node:fs-dependent handlers in the Cloudflare environment.
|
|
322
|
+
//
|
|
323
|
+
// Node.js preset: uses the main server's RSC environment directly (router
|
|
324
|
+
// instances are already discovered and have matchForPrerender).
|
|
325
|
+
// Cloudflare preset: lazily creates a Node.js temp server because the main
|
|
326
|
+
// RSC environment uses workerd where node:fs can't access the host filesystem.
|
|
327
|
+
|
|
328
|
+
// Registry from the main server's RSC environment (populated by discoverRouters)
|
|
329
|
+
let mainRegistry: Map<string, any> | null = null;
|
|
330
|
+
|
|
331
|
+
// Push discovery state (manifest, trie, precomputed entries) to the
|
|
332
|
+
// server module so runtime request handling uses the current routes.
|
|
333
|
+
// Shared by initial discovery and HMR-triggered re-discovery.
|
|
334
|
+
const propagateDiscoveryState = async (rscEnv: any) => {
|
|
335
|
+
const serverMod = await rscEnv.runner.import("@rangojs/router/server");
|
|
336
|
+
if (!serverMod) return;
|
|
337
|
+
// Clear stale per-router and global route data before repopulating.
|
|
338
|
+
// Without this, removed routers/routes survive in the per-router maps
|
|
339
|
+
// and shrunk precomputed entries or tries are never purged.
|
|
340
|
+
if (serverMod.clearAllRouterData) {
|
|
341
|
+
serverMod.clearAllRouterData();
|
|
342
|
+
}
|
|
343
|
+
mainRegistry = serverMod.RouterRegistry ?? null;
|
|
344
|
+
if (s.mergedRouteManifest && serverMod.setCachedManifest) {
|
|
345
|
+
serverMod.setCachedManifest(s.mergedRouteManifest);
|
|
346
|
+
}
|
|
347
|
+
if (
|
|
348
|
+
s.mergedPrecomputedEntries &&
|
|
349
|
+
s.mergedPrecomputedEntries.length > 0 &&
|
|
350
|
+
serverMod.setPrecomputedEntries
|
|
351
|
+
) {
|
|
352
|
+
serverMod.setPrecomputedEntries(s.mergedPrecomputedEntries);
|
|
353
|
+
}
|
|
354
|
+
if (s.mergedRouteTrie && serverMod.setRouteTrie) {
|
|
355
|
+
serverMod.setRouteTrie(s.mergedRouteTrie);
|
|
356
|
+
}
|
|
357
|
+
if (serverMod.setRouterManifest) {
|
|
358
|
+
for (const [routerId, manifest] of s.perRouterManifestDataMap) {
|
|
359
|
+
serverMod.setRouterManifest(routerId, manifest);
|
|
360
|
+
}
|
|
361
|
+
}
|
|
362
|
+
if (serverMod.setRouterTrie) {
|
|
363
|
+
for (const [routerId, trie] of s.perRouterTrieMap) {
|
|
364
|
+
serverMod.setRouterTrie(routerId, trie);
|
|
365
|
+
}
|
|
366
|
+
}
|
|
367
|
+
if (serverMod.setRouterPrecomputedEntries) {
|
|
368
|
+
for (const [routerId, entries] of s.perRouterPrecomputedMap) {
|
|
369
|
+
serverMod.setRouterPrecomputedEntries(routerId, entries);
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
};
|
|
373
|
+
|
|
374
|
+
server.middlewares.use("/__rsc_prerender", async (req: any, res: any) => {
|
|
375
|
+
if (s.discoveryDone) await s.discoveryDone;
|
|
376
|
+
|
|
377
|
+
const url = new URL(req.url || "/", "http://localhost");
|
|
378
|
+
const pathname = url.searchParams.get("pathname");
|
|
379
|
+
if (!pathname) {
|
|
380
|
+
res.statusCode = 400;
|
|
381
|
+
res.end("Missing pathname");
|
|
382
|
+
return;
|
|
383
|
+
}
|
|
384
|
+
|
|
385
|
+
// Prefer the main server's registry (Node.js preset: module runner available).
|
|
386
|
+
// Fall back to a temp server for Cloudflare where the main RSC env uses workerd.
|
|
387
|
+
let registry = mainRegistry;
|
|
388
|
+
|
|
389
|
+
if (!registry) {
|
|
390
|
+
// No main registry: the RSC env has no module runner (Cloudflare dev).
|
|
391
|
+
// Lazily create a Node.js temp server for prerender evaluation.
|
|
392
|
+
if (!prerenderNodeRegistry) {
|
|
393
|
+
await getOrCreateTempServer();
|
|
394
|
+
}
|
|
395
|
+
registry = prerenderNodeRegistry;
|
|
396
|
+
}
|
|
397
|
+
|
|
398
|
+
if (!registry || registry.size === 0) {
|
|
399
|
+
res.statusCode = 503;
|
|
400
|
+
res.end("Prerender runner not available");
|
|
401
|
+
return;
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
const wantIntercept = url.searchParams.get("intercept") === "1";
|
|
405
|
+
const wantRouteName = url.searchParams.get("routeName");
|
|
406
|
+
const wantPassthrough = url.searchParams.get("passthrough") === "1";
|
|
407
|
+
|
|
408
|
+
for (const [, routerInstance] of registry) {
|
|
409
|
+
if (!routerInstance.matchForPrerender) continue;
|
|
410
|
+
try {
|
|
411
|
+
const result = await routerInstance.matchForPrerender(
|
|
412
|
+
pathname,
|
|
413
|
+
{},
|
|
414
|
+
undefined,
|
|
415
|
+
wantPassthrough,
|
|
416
|
+
);
|
|
417
|
+
if (!result) continue;
|
|
418
|
+
if (result.passthrough) continue;
|
|
419
|
+
// When routeName is specified, only accept a match for that route.
|
|
420
|
+
// This prevents returning the wrong entry when multiple routers
|
|
421
|
+
// have prerenderable routes sharing the same pathname.
|
|
422
|
+
if (wantRouteName && result.routeName !== wantRouteName) continue;
|
|
423
|
+
res.setHeader("content-type", "application/json");
|
|
424
|
+
let payload: Record<string, unknown>;
|
|
425
|
+
if (wantIntercept && result.interceptSegments?.length) {
|
|
426
|
+
payload = {
|
|
427
|
+
segments: [...result.segments, ...result.interceptSegments],
|
|
428
|
+
handles: {
|
|
429
|
+
...result.handles,
|
|
430
|
+
...(result.interceptHandles || {}),
|
|
431
|
+
},
|
|
432
|
+
};
|
|
433
|
+
} else {
|
|
434
|
+
payload = { segments: result.segments, handles: result.handles };
|
|
435
|
+
}
|
|
436
|
+
res.end(JSON.stringify(payload));
|
|
437
|
+
return;
|
|
438
|
+
} catch (err: any) {
|
|
439
|
+
console.warn(
|
|
440
|
+
`[rsc-router] Dev prerender failed for ${pathname}: ${err.message}`,
|
|
441
|
+
);
|
|
442
|
+
}
|
|
443
|
+
}
|
|
444
|
+
|
|
445
|
+
res.statusCode = 404;
|
|
446
|
+
res.end("No prerender match");
|
|
447
|
+
});
|
|
448
|
+
|
|
449
|
+
// Watch url module and router files for changes and regenerate named-routes.gen.ts.
|
|
450
|
+
// Process files containing urls( or createRouter( to update the combined route map.
|
|
451
|
+
if (opts?.staticRouteTypesGeneration !== false) {
|
|
452
|
+
const isGeneratedRouteFile = (filePath: string): boolean =>
|
|
453
|
+
filePath.endsWith(".gen.ts") &&
|
|
454
|
+
(filePath.includes("named-routes.gen.ts") ||
|
|
455
|
+
filePath.includes("urls.gen.ts"));
|
|
456
|
+
|
|
457
|
+
const regenerateGeneratedRouteFiles = () => {
|
|
458
|
+
if (s.perRouterManifests.length > 0) {
|
|
459
|
+
writeRouteTypesFiles(s);
|
|
460
|
+
} else {
|
|
461
|
+
writeCombinedRouteTypesWithTracking(s);
|
|
462
|
+
}
|
|
463
|
+
};
|
|
464
|
+
|
|
465
|
+
const maybeHandleGeneratedRouteFileMutation = (
|
|
466
|
+
filePath: string,
|
|
467
|
+
): boolean => {
|
|
468
|
+
if (!isGeneratedRouteFile(filePath)) return false;
|
|
469
|
+
if (consumeSelfGenWrite(s, filePath)) return true;
|
|
470
|
+
// In Cloudflare dev (no module runner), perRouterManifests is never
|
|
471
|
+
// refreshed after HMR so regenerateGeneratedRouteFiles() would use
|
|
472
|
+
// stale data and revert user edits. Source files own route state;
|
|
473
|
+
// gen files are derived output. Skip regeneration and let the next
|
|
474
|
+
// source-file change rebuild them from the static parser.
|
|
475
|
+
const hasRunner = !!(server.environments as any)?.rsc?.runner;
|
|
476
|
+
if (!hasRunner) return true;
|
|
477
|
+
regenerateGeneratedRouteFiles();
|
|
478
|
+
return true;
|
|
479
|
+
};
|
|
480
|
+
|
|
481
|
+
// Debounce timer for batching rapid route-file changes (e.g. afterEach
|
|
482
|
+
// restoring two files in quick succession). The cheap checks (extension,
|
|
483
|
+
// scanFilter, content sniff) run synchronously to gate non-route files;
|
|
484
|
+
// only the expensive regeneration is debounced.
|
|
485
|
+
let routeChangeTimer: ReturnType<typeof setTimeout> | undefined;
|
|
486
|
+
|
|
487
|
+
// Re-run runtime discovery so factory-generated routes that the
|
|
488
|
+
// static parser cannot see are refreshed after source changes.
|
|
489
|
+
let runtimeRediscoveryInProgress = false;
|
|
490
|
+
const refreshRuntimeDiscovery = async () => {
|
|
491
|
+
const rscEnv = (server.environments as any)?.rsc;
|
|
492
|
+
if (!rscEnv?.runner || runtimeRediscoveryInProgress) return;
|
|
493
|
+
runtimeRediscoveryInProgress = true;
|
|
494
|
+
try {
|
|
495
|
+
await discoverRouters(s, rscEnv);
|
|
496
|
+
writeRouteTypesFiles(s);
|
|
497
|
+
await propagateDiscoveryState(rscEnv);
|
|
498
|
+
} catch (err: any) {
|
|
499
|
+
console.warn(
|
|
500
|
+
`[rsc-router] Runtime re-discovery failed: ${err.message}`,
|
|
501
|
+
);
|
|
502
|
+
} finally {
|
|
503
|
+
runtimeRediscoveryInProgress = false;
|
|
504
|
+
}
|
|
505
|
+
};
|
|
506
|
+
|
|
507
|
+
const scheduleRouteRegeneration = () => {
|
|
508
|
+
clearTimeout(routeChangeTimer);
|
|
509
|
+
routeChangeTimer = setTimeout(() => {
|
|
510
|
+
routeChangeTimer = undefined;
|
|
511
|
+
try {
|
|
512
|
+
writeCombinedRouteTypesWithTracking(s);
|
|
513
|
+
if (s.perRouterManifests.length > 0) {
|
|
514
|
+
supplementGenFilesWithRuntimeRoutes(s);
|
|
515
|
+
}
|
|
516
|
+
} catch (err: any) {
|
|
517
|
+
console.error(
|
|
518
|
+
`[rsc-router] Route regeneration error: ${err.message}`,
|
|
519
|
+
);
|
|
520
|
+
}
|
|
521
|
+
// Async: re-run runtime discovery to refresh factory-generated
|
|
522
|
+
// routes that the static parser cannot resolve.
|
|
523
|
+
if (s.perRouterManifests.length > 0) {
|
|
524
|
+
refreshRuntimeDiscovery().catch((err: any) => {
|
|
525
|
+
console.warn(
|
|
526
|
+
`[rsc-router] Runtime re-discovery error: ${err.message}`,
|
|
527
|
+
);
|
|
528
|
+
});
|
|
529
|
+
}
|
|
530
|
+
}, 100);
|
|
531
|
+
};
|
|
532
|
+
|
|
533
|
+
const handleRouteFileChange = (filePath: string) => {
|
|
534
|
+
if (maybeHandleGeneratedRouteFileMutation(filePath)) return;
|
|
535
|
+
if (
|
|
536
|
+
!filePath.endsWith(".ts") &&
|
|
537
|
+
!filePath.endsWith(".tsx") &&
|
|
538
|
+
!filePath.endsWith(".js") &&
|
|
539
|
+
!filePath.endsWith(".jsx")
|
|
540
|
+
)
|
|
541
|
+
return;
|
|
542
|
+
// Apply scan filter as early-exit before reading file
|
|
543
|
+
if (s.scanFilter && !s.scanFilter(filePath)) return;
|
|
544
|
+
try {
|
|
545
|
+
const source = readFileSync(filePath, "utf-8");
|
|
546
|
+
const trimmed = source.trimStart();
|
|
547
|
+
if (
|
|
548
|
+
trimmed.startsWith('"use client"') ||
|
|
549
|
+
trimmed.startsWith("'use client'")
|
|
550
|
+
)
|
|
551
|
+
return;
|
|
552
|
+
const hasUrls = source.includes("urls(");
|
|
553
|
+
const hasCreateRouter = /\bcreateRouter\s*[<(]/.test(source);
|
|
554
|
+
if (!hasUrls && !hasCreateRouter) return;
|
|
555
|
+
// Invalidate cache when a router file changes (new router added/removed)
|
|
556
|
+
if (hasCreateRouter) {
|
|
557
|
+
const nestedRouterConflict = findNestedRouterConflict([
|
|
558
|
+
...(s.cachedRouterFiles ?? []),
|
|
559
|
+
resolve(filePath),
|
|
560
|
+
]);
|
|
561
|
+
if (nestedRouterConflict) {
|
|
562
|
+
server.config.logger.error(
|
|
563
|
+
formatNestedRouterConflictError(nestedRouterConflict),
|
|
564
|
+
);
|
|
565
|
+
return;
|
|
566
|
+
}
|
|
567
|
+
s.cachedRouterFiles = undefined;
|
|
568
|
+
}
|
|
569
|
+
scheduleRouteRegeneration();
|
|
570
|
+
} catch {
|
|
571
|
+
// Ignore read errors for deleted/moved files
|
|
572
|
+
}
|
|
573
|
+
};
|
|
574
|
+
|
|
575
|
+
// Handle both "add" and "change" events: editors with atomic saves
|
|
576
|
+
// (unlink + rename) emit "add" instead of "change", and chokidar's
|
|
577
|
+
// polling mode on CI Linux can also emit "add" for overwrites.
|
|
578
|
+
server.watcher.on("add", handleRouteFileChange);
|
|
579
|
+
server.watcher.on("change", handleRouteFileChange);
|
|
580
|
+
|
|
581
|
+
// Regenerate gen files when they are deleted (e.g. manual cleanup).
|
|
582
|
+
// Same no-runner guard as change/add: stale perRouterManifests would
|
|
583
|
+
// reintroduce reverted content.
|
|
584
|
+
server.watcher.on("unlink", (filePath) => {
|
|
585
|
+
if (!isGeneratedRouteFile(filePath)) return;
|
|
586
|
+
const hasRunner = !!(server.environments as any)?.rsc?.runner;
|
|
587
|
+
if (!hasRunner) return;
|
|
588
|
+
regenerateGeneratedRouteFiles();
|
|
589
|
+
});
|
|
590
|
+
}
|
|
591
|
+
},
|
|
592
|
+
|
|
593
|
+
// Build mode: create a temporary Vite dev server to access the RSC
|
|
594
|
+
// environment's module runner, then discover routers and generate manifests.
|
|
595
|
+
// The manifest data is stored for the virtual module's load hook.
|
|
596
|
+
async buildStart() {
|
|
597
|
+
if (!s.isBuildMode) return;
|
|
598
|
+
// Only run once across environment builds
|
|
599
|
+
if (s.mergedRouteManifest !== null) return;
|
|
600
|
+
resetStagedBuildAssets(s.projectRoot);
|
|
601
|
+
s.prerenderManifestEntries = null;
|
|
602
|
+
s.staticManifestEntries = null;
|
|
603
|
+
|
|
604
|
+
let tempServer: any = null;
|
|
605
|
+
// Signal to user-space code (e.g. reverse.ts) that build-time discovery
|
|
606
|
+
// is active. Uses globalThis because the temp server's module runner
|
|
607
|
+
// creates a separate module context — there is no shared import path
|
|
608
|
+
// between the vite plugin and user code loaded via runner.import().
|
|
609
|
+
(globalThis as any).__rscRouterDiscoveryActive = true;
|
|
610
|
+
try {
|
|
611
|
+
tempServer = await createTempRscServer(s, { forceBuild: true });
|
|
612
|
+
|
|
613
|
+
const rscEnv = (tempServer.environments as any)?.rsc;
|
|
614
|
+
if (!rscEnv?.runner) {
|
|
615
|
+
console.warn(
|
|
616
|
+
"[rsc-router] RSC environment runner not available during build, skipping manifest generation",
|
|
617
|
+
);
|
|
618
|
+
return;
|
|
619
|
+
}
|
|
620
|
+
|
|
621
|
+
// Point resolvedStaticModules at the temp server's expose-internal-ids
|
|
622
|
+
// plugin so that discoverRouters() can access the static handler module
|
|
623
|
+
// map after the temp server's transforms populate it.
|
|
624
|
+
const tempIdsPlugin = (tempServer as any).config?.plugins?.find(
|
|
625
|
+
(p: any) => p.name === "@rangojs/router:expose-internal-ids",
|
|
626
|
+
);
|
|
627
|
+
if (tempIdsPlugin?.api?.staticHandlerModules) {
|
|
628
|
+
s.resolvedStaticModules = tempIdsPlugin.api.staticHandlerModules;
|
|
629
|
+
}
|
|
630
|
+
|
|
631
|
+
await discoverRouters(s, rscEnv);
|
|
632
|
+
// Update named-routes.gen.ts from runtime discovery.
|
|
633
|
+
// The runtime manifest includes dynamically generated routes
|
|
634
|
+
// that the static parser cannot extract from source code.
|
|
635
|
+
writeRouteTypesFiles(s);
|
|
636
|
+
} catch (err: any) {
|
|
637
|
+
// Extract the user source file from the stack trace (skip internal frames)
|
|
638
|
+
const sourceFile = err.stack
|
|
639
|
+
?.split("\n")
|
|
640
|
+
.find(
|
|
641
|
+
(line: string) =>
|
|
642
|
+
line.includes(s.projectRoot) && !line.includes("node_modules"),
|
|
643
|
+
)
|
|
644
|
+
?.match(/\(([^)]+)\)/)?.[1];
|
|
645
|
+
// Extract the route name from "Unknown route: <name>" errors
|
|
646
|
+
const routeName = err.message?.match(/Unknown route: (.+)/)?.[1];
|
|
647
|
+
const details = [
|
|
648
|
+
routeName ? ` Route name: ${routeName}` : null,
|
|
649
|
+
sourceFile ? ` File: ${sourceFile}` : null,
|
|
650
|
+
err.stack ? ` Stack:\n${err.stack}` : null,
|
|
651
|
+
]
|
|
652
|
+
.filter(Boolean)
|
|
653
|
+
.join("\n");
|
|
654
|
+
throw new Error(
|
|
655
|
+
`[rsc-router] Build-time router discovery failed:\n${details}`,
|
|
656
|
+
);
|
|
657
|
+
} finally {
|
|
658
|
+
delete (globalThis as any).__rscRouterDiscoveryActive;
|
|
659
|
+
if (tempServer) {
|
|
660
|
+
await tempServer.close();
|
|
661
|
+
}
|
|
662
|
+
}
|
|
663
|
+
},
|
|
664
|
+
|
|
665
|
+
// Virtual module: provides the pre-generated route manifest as a JS module
|
|
666
|
+
// that calls setCachedManifest() at import time.
|
|
667
|
+
resolveId(id) {
|
|
668
|
+
if (id === VIRTUAL_ROUTES_MANIFEST_ID) {
|
|
669
|
+
return "\0" + VIRTUAL_ROUTES_MANIFEST_ID;
|
|
670
|
+
}
|
|
671
|
+
// Per-router virtual modules: virtual:rsc-router/routes-manifest/<routerId>
|
|
672
|
+
if (id.startsWith(VIRTUAL_ROUTES_MANIFEST_ID + "/")) {
|
|
673
|
+
return "\0" + id;
|
|
674
|
+
}
|
|
675
|
+
// virtual:rsc-router/prerender-paths removed: prerender data is served through the worker
|
|
676
|
+
return null;
|
|
677
|
+
},
|
|
678
|
+
|
|
679
|
+
async load(id) {
|
|
680
|
+
if (id === "\0" + VIRTUAL_ROUTES_MANIFEST_ID) {
|
|
681
|
+
// In dev mode, wait for discovery to complete before emitting module content.
|
|
682
|
+
// This is critical for Cloudflare dev where the worker runs in a separate
|
|
683
|
+
// Miniflare process and can only receive manifest data via the virtual module.
|
|
684
|
+
if (s.discoveryDone) {
|
|
685
|
+
await s.discoveryDone;
|
|
686
|
+
}
|
|
687
|
+
return generateRoutesManifestModule(s);
|
|
688
|
+
}
|
|
689
|
+
// Per-router virtual modules: pure data exports (no side effects).
|
|
690
|
+
// ensureRouterManifest() imports the module and stores the data.
|
|
691
|
+
const perRouterPrefix = "\0" + VIRTUAL_ROUTES_MANIFEST_ID + "/";
|
|
692
|
+
if (id.startsWith(perRouterPrefix)) {
|
|
693
|
+
if (s.discoveryDone) {
|
|
694
|
+
await s.discoveryDone;
|
|
695
|
+
}
|
|
696
|
+
const routerId = id.slice(perRouterPrefix.length);
|
|
697
|
+
return generatePerRouterModule(s, routerId);
|
|
698
|
+
}
|
|
699
|
+
// virtual:rsc-router/prerender-paths load handler removed
|
|
700
|
+
return null;
|
|
701
|
+
},
|
|
702
|
+
|
|
703
|
+
// Record handler chunk metadata and RSC entry filename during RSC build.
|
|
704
|
+
// Used by closeBundle for handler code eviction and prerender data injection.
|
|
705
|
+
generateBundle(_options: any, bundle: any) {
|
|
706
|
+
if (this.environment?.name !== "rsc") return;
|
|
707
|
+
|
|
708
|
+
// Record RSC entry chunk filename for closeBundle injection
|
|
709
|
+
for (const [fileName, chunk] of Object.entries(bundle) as [
|
|
710
|
+
string,
|
|
711
|
+
any,
|
|
712
|
+
][]) {
|
|
713
|
+
if (chunk.type === "chunk" && chunk.isEntry) {
|
|
714
|
+
s.rscEntryFileName = fileName;
|
|
715
|
+
break;
|
|
716
|
+
}
|
|
717
|
+
}
|
|
718
|
+
|
|
719
|
+
if (!s.resolvedPrerenderModules?.size && !s.resolvedStaticModules?.size)
|
|
720
|
+
return;
|
|
721
|
+
|
|
722
|
+
for (const [fileName, chunk] of Object.entries(bundle) as [
|
|
723
|
+
string,
|
|
724
|
+
any,
|
|
725
|
+
][]) {
|
|
726
|
+
if (chunk.type !== "chunk") continue;
|
|
727
|
+
|
|
728
|
+
// Prerender handlers chunk
|
|
729
|
+
if (
|
|
730
|
+
fileName.includes("__prerender-handlers") &&
|
|
731
|
+
s.resolvedPrerenderModules?.size
|
|
732
|
+
) {
|
|
733
|
+
const handlers = extractHandlerExportsFromChunk(
|
|
734
|
+
chunk.code,
|
|
735
|
+
s.resolvedPrerenderModules,
|
|
736
|
+
"Prerender",
|
|
737
|
+
true,
|
|
738
|
+
);
|
|
739
|
+
if (handlers.length > 0) {
|
|
740
|
+
s.handlerChunkInfo = { fileName, exports: handlers };
|
|
741
|
+
}
|
|
742
|
+
}
|
|
743
|
+
|
|
744
|
+
// Static handlers chunk
|
|
745
|
+
if (
|
|
746
|
+
fileName.includes("__static-handlers") &&
|
|
747
|
+
s.resolvedStaticModules?.size
|
|
748
|
+
) {
|
|
749
|
+
const handlers = extractHandlerExportsFromChunk(
|
|
750
|
+
chunk.code,
|
|
751
|
+
s.resolvedStaticModules,
|
|
752
|
+
"Static",
|
|
753
|
+
false,
|
|
754
|
+
);
|
|
755
|
+
if (handlers.length > 0) {
|
|
756
|
+
s.staticHandlerChunkInfo = { fileName, exports: handlers };
|
|
757
|
+
}
|
|
758
|
+
}
|
|
759
|
+
}
|
|
760
|
+
},
|
|
761
|
+
|
|
762
|
+
// Build-time pre-rendering: evict handler code and inject collected prerender data.
|
|
763
|
+
// Collection now happens in-process during discoverRouters() via RSC runner.
|
|
764
|
+
// closeBundle only needs to evict handlers and inject the in-memory data.
|
|
765
|
+
closeBundle: {
|
|
766
|
+
order: "post" as const,
|
|
767
|
+
sequential: true,
|
|
768
|
+
async handler(this: any) {
|
|
769
|
+
if (!s.isBuildMode) return;
|
|
770
|
+
// Only run for the RSC environment — other environments (client, ssr) have
|
|
771
|
+
// no prerender/static data to process and would just do redundant file I/O.
|
|
772
|
+
if (this.environment && this.environment.name !== "rsc") return;
|
|
773
|
+
postprocessBundle(s);
|
|
774
|
+
},
|
|
775
|
+
},
|
|
776
|
+
};
|
|
777
|
+
}
|