vite-plugin-react-server 1.4.2 → 1.4.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (73) hide show
  1. package/README.md +48 -313
  2. package/dist/package.json +123 -13
  3. package/dist/plugin/bundle/deferredStaticGeneration.js +14 -39
  4. package/dist/plugin/bundle/manifests.js +30 -48
  5. package/dist/plugin/config/autoDiscover/resolveAutoDiscover.d.ts.map +1 -1
  6. package/dist/plugin/config/autoDiscover/resolveAutoDiscover.js +4 -1
  7. package/dist/plugin/config/envPrefixFromConfig.js +12 -7
  8. package/dist/plugin/config/getCondition.d.ts.map +1 -1
  9. package/dist/plugin/config/getCondition.js +7 -5
  10. package/dist/plugin/dev-server/virtualRscHmrPlugin.js +23 -23
  11. package/dist/plugin/environments/createBuildEventPlugin.js +88 -98
  12. package/dist/plugin/environments/createEnvironmentPlugin.js +222 -250
  13. package/dist/plugin/helpers/createRscRenderHelpers.js +33 -34
  14. package/dist/plugin/helpers/createSharedLoader.d.ts.map +1 -1
  15. package/dist/plugin/helpers/createSharedLoader.js +4 -2
  16. package/dist/plugin/helpers/headlessStreamReuseHandler.js +30 -22
  17. package/dist/plugin/helpers/headlessStreamState.js +15 -28
  18. package/dist/plugin/helpers/resolveComponent.d.ts.map +1 -1
  19. package/dist/plugin/helpers/resolveComponent.js +4 -2
  20. package/dist/plugin/index.client.d.ts +5 -0
  21. package/dist/plugin/index.client.d.ts.map +1 -0
  22. package/dist/plugin/index.client.js +4 -0
  23. package/dist/plugin/index.d.ts +4 -3
  24. package/dist/plugin/index.d.ts.map +1 -1
  25. package/dist/plugin/index.js +10 -5
  26. package/dist/plugin/index.server.d.ts +5 -0
  27. package/dist/plugin/index.server.d.ts.map +1 -0
  28. package/dist/plugin/index.server.js +4 -0
  29. package/dist/plugin/metrics/createWorkerStartupMetrics.js +31 -13
  30. package/dist/plugin/orchestrator/createPluginOrchestrator.client.js +41 -38
  31. package/dist/plugin/orchestrator/createPluginOrchestrator.server.js +43 -46
  32. package/dist/plugin/plugin.client.js +2 -2
  33. package/dist/plugin/plugin.server.js +2 -2
  34. package/dist/plugin/react-static/createBuildLoader.client.js +12 -6
  35. package/dist/plugin/react-static/createBuildLoader.server.js +255 -235
  36. package/dist/plugin/react-static/plugin.client.js +684 -770
  37. package/dist/plugin/react-static/plugin.server.js +517 -603
  38. package/dist/plugin/react-static/processCssFilesForPages.js +103 -88
  39. package/dist/plugin/react-static/renderPage.client.js +455 -529
  40. package/dist/plugin/react-static/renderPage.server.js +485 -508
  41. package/dist/plugin/react-static/renderPagesBatched.js +277 -275
  42. package/dist/plugin/react-static/rscToHtmlStream.client.js +48 -29
  43. package/dist/plugin/react-static/rscToHtmlStream.server.js +62 -37
  44. package/dist/plugin/react-static/temporaryReferences.server.js +11 -2
  45. package/dist/plugin/stream/createMainThreadHandlers.js +40 -31
  46. package/dist/plugin/stream/renderRscStream.server.d.ts.map +1 -1
  47. package/dist/plugin/stream/renderRscStream.server.js +127 -144
  48. package/dist/plugin/transformer/createTransformerPlugin.js +226 -265
  49. package/dist/plugin/utils/checkReactVersion.d.ts +7 -0
  50. package/dist/plugin/utils/checkReactVersion.d.ts.map +1 -0
  51. package/dist/plugin/utils/checkReactVersion.js +23 -0
  52. package/dist/plugin/utils/envUrls.node.js +12 -11
  53. package/dist/plugin/vendor/vendor-alias.js +84 -114
  54. package/dist/plugin/vendor/vendor.client.d.ts.map +1 -1
  55. package/dist/plugin/vendor/vendor.client.js +1 -3
  56. package/dist/plugin/worker/rsc/handleRscRender.d.ts.map +1 -1
  57. package/dist/plugin/worker/rsc/handleRscRender.js +3 -1
  58. package/dist/tsconfig.tsbuildinfo +1 -1
  59. package/package.json +123 -13
  60. package/plugin/config/autoDiscover/resolveAutoDiscover.ts +4 -0
  61. package/plugin/config/getCondition.ts +6 -4
  62. package/plugin/helpers/createSharedLoader.ts +6 -1
  63. package/plugin/helpers/resolveComponent.ts +6 -1
  64. package/plugin/index.client.ts +4 -0
  65. package/plugin/index.server.ts +4 -0
  66. package/plugin/index.ts +12 -5
  67. package/plugin/plugin.client.ts +1 -1
  68. package/plugin/plugin.server.ts +1 -1
  69. package/plugin/stream/renderRscStream.server.ts +3 -0
  70. package/plugin/utils/checkReactVersion.ts +28 -0
  71. package/plugin/vendor/vendor.client.ts +0 -2
  72. package/plugin/worker/rsc/handleRscRender.ts +2 -0
  73. package/scripts/generate-toc.mjs +27 -294
@@ -1,801 +1,715 @@
1
1
  /**
2
- * plugin.client.ts
3
- *
4
- * PURPOSE: Client-side static plugin for React Server Components
5
- *
6
- * This module:
7
- * 1. Handles static site generation in the client environment
8
- * 2. Uses RSC worker for RSC rendering and main-thread for HTML rendering
9
- * 3. Generates both RSC and HTML files for static pages
10
- * 4. Integrates with Vite's build process
11
- *
12
- * Feature parity with main react-static plugin, but in reverse. Uses rsc-worker to render rsc, and main thread for html.
13
- * This is not the default behavior, but is supported for testing and custom app development purposes.
14
- * Additionally, this can make it easier to use the --app flag to build all the modules + static generation at once.
2
+ * vite-plugin-react-server
3
+ * Copyright (c) Nico Brinkkemper
4
+ * MIT License
15
5
  */
16
- import { createLogger, } from "vite";
17
- import { resolveOptions } from "../config/resolveOptions.js";
18
- import { renderPagesBatched } from "./renderPagesBatched.js";
19
- import { performance } from "node:perf_hooks";
20
- import { renderPage } from "./renderPage.client.js";
21
- import { createWorker } from "../worker/createWorker.js";
22
- import { serializedOptions, serializeResolvedConfig, } from "../helpers/serializeUserOptions.js";
23
- import { getBundleManifest } from "../helpers/getBundleManifest.js";
24
- import { handleError } from "../error/handleError.js";
25
- import { shouldCausePanic } from "../error/panicThresholdHandler.js";
26
- import { configurePreviewServer } from "./configurePreviewServer.js";
27
- import { assertNonReactServer } from "../config/getCondition.js";
28
- import { envPrefixFromConfig } from "../config/envPrefixFromConfig.js";
29
- import { createWorkerStartupMetrics } from "../metrics/createWorkerStartupMetrics.js";
30
- import { processCssFilesForPages } from "./processCssFilesForPages.js";
31
- import { createBuildLoader } from "./createBuildLoader.client.js";
32
- import { getNodeEnv } from "../config/getNodeEnv.js";
33
- import { toError } from "../error/toError.js";
34
- import { addStaticManifest, manifests, getSharedManifestStore, } from "../bundle/manifests.js";
35
- import { deferStaticGeneration } from "../bundle/deferredStaticGeneration.js";
36
- import { resolveAutoDiscover } from "../config/index.js";
37
- import { join } from "node:path";
38
- import { baseURL } from "../utils/envUrls.node.js";
39
- import { tryManifest } from "../helpers/tryManifest.js";
40
- // cssCollector removed - using filesystem-based CSS processing
6
+ import { createLogger } from 'vite';
7
+ import { resolveOptions } from '../config/resolveOptions.js';
8
+ import { renderPagesBatched } from './renderPagesBatched.js';
9
+ import { performance } from 'node:perf_hooks';
10
+ import { renderPage } from './renderPage.client.js';
11
+ import { createWorker } from '../worker/createWorker.js';
12
+ import { serializeResolvedConfig, serializedOptions } from '../helpers/serializeUserOptions.js';
13
+ import { getBundleManifest } from '../helpers/getBundleManifest.js';
14
+ import { handleError } from '../error/handleError.js';
15
+ import { shouldCausePanic } from '../error/panicThresholdHandler.js';
16
+ import { configurePreviewServer } from './configurePreviewServer.js';
17
+ import { assertNonReactServer } from '../config/getCondition.js';
18
+ import { envPrefixFromConfig } from '../config/envPrefixFromConfig.js';
19
+ import { createWorkerStartupMetrics } from '../metrics/createWorkerStartupMetrics.js';
20
+ import { processCssFilesForPages } from './processCssFilesForPages.js';
21
+ import { createBuildLoader } from './createBuildLoader.client.js';
22
+ import { getNodeEnv } from '../config/getNodeEnv.js';
23
+ import { toError } from '../error/toError.js';
24
+ import { addStaticManifest, manifests, getSharedManifestStore } from '../bundle/manifests.js';
25
+ import { deferStaticGeneration } from '../bundle/deferredStaticGeneration.js';
26
+ import { resolveAutoDiscover } from '../config/autoDiscover/resolveAutoDiscover.js';
27
+ import { join } from 'node:path';
28
+ import { baseURL } from '../utils/envUrls.node.js';
29
+ import { tryManifest } from '../helpers/tryManifest.js';
30
+
41
31
  assertNonReactServer();
42
- /**
43
- * plugin.client.ts
44
- *
45
- * PURPOSE: Client-side static plugin for React Server Components
46
- *
47
- * This module:
48
- * 1. Handles static site generation in the client environment
49
- * 2. Uses RSC worker for RSC rendering and main-thread for HTML rendering
50
- * 3. Generates both RSC and HTML files for static pages
51
- * 4. Integrates with Vite's build process
52
- *
53
- * @param options
54
- * @returns
55
- */
56
- export const reactStaticPlugin = function _reactStaticPlugin(options) {
57
- let logger;
58
- let autoDiscoveredFiles = null;
59
- let rscWorker = undefined;
60
- let resolvedConfig = null;
61
- let serverManifest = undefined;
62
- let staticBundle = undefined;
63
- let serverBundle = undefined;
64
- let configEnv;
65
- const timing = {
66
- start: performance.now(),
67
- configResolved: 0,
68
- buildStart: 0,
69
- renderStart: 0,
70
- };
71
- const resolvedOptions = resolveOptions(options);
72
- if (resolvedOptions.type === "error") {
73
- throw resolvedOptions.error;
74
- }
75
- const userOptions = resolvedOptions.userOptions;
76
- return {
77
- name: "vite:plugin-react-server/client-static",
78
- enforce: "post",
79
- apply: "build", // Apply to build mode
80
- api: {
81
- meta: { timing },
82
- },
83
- async config(_config, viteConfigEnv) {
84
- configEnv = viteConfigEnv;
85
- },
86
- applyToEnvironment(partialEnvironment) {
87
- // Client static plugin should apply to static environment (browser/ESM builds)
88
- // This is where we want to bundle everything and filter out _virtual files
89
- // Apply to both "static" and "client" environments - we'll handle which one runs static generation in closeBundle
90
- const envName = partialEnvironment.name;
91
- if (["static", "client"].includes(envName)) {
92
- return true;
32
+ const reactStaticPlugin = function _reactStaticPlugin(options) {
33
+ let logger;
34
+ let autoDiscoveredFiles = null;
35
+ let rscWorker = void 0;
36
+ let resolvedConfig = null;
37
+ let serverManifest = void 0;
38
+ let staticBundle = void 0;
39
+ let serverBundle = void 0;
40
+ let configEnv;
41
+ const timing = {
42
+ start: performance.now(),
43
+ configResolved: 0,
44
+ buildStart: 0,
45
+ renderStart: 0
46
+ };
47
+ const resolvedOptions = resolveOptions(options);
48
+ if (resolvedOptions.type === "error") {
49
+ throw resolvedOptions.error;
50
+ }
51
+ const userOptions = resolvedOptions.userOptions;
52
+ return {
53
+ name: "vite:plugin-react-server/client-static",
54
+ enforce: "post",
55
+ apply: "build",
56
+ // Apply to build mode
57
+ api: {
58
+ meta: { timing }
59
+ },
60
+ async config(_config, viteConfigEnv) {
61
+ configEnv = viteConfigEnv;
62
+ },
63
+ applyToEnvironment(partialEnvironment) {
64
+ const envName = partialEnvironment.name;
65
+ if (["static", "client"].includes(envName)) {
66
+ return true;
67
+ }
68
+ return false;
69
+ },
70
+ async configResolved(config) {
71
+ timing.configResolved = performance.now();
72
+ logger = config.customLogger || createLogger();
73
+ resolvedConfig = config;
74
+ const autoDiscoverResult = await resolveAutoDiscover({
75
+ config,
76
+ configEnv: configEnv || {
77
+ mode: config.mode,
78
+ command: config.command},
79
+ userOptions,
80
+ logger
81
+ });
82
+ if (autoDiscoverResult.type === "error") {
83
+ throw autoDiscoverResult.error;
84
+ }
85
+ autoDiscoveredFiles = autoDiscoverResult.autoDiscoveredFiles;
86
+ if (userOptions.verbose) {
87
+ logger?.info(`Auto-discovery ${autoDiscoverResult.type === "success" ? "completed" : "skipped"}`);
88
+ }
89
+ },
90
+ async buildStart() {
91
+ timing.buildStart = performance.now();
92
+ if (userOptions.verbose) {
93
+ logger?.info("[react-static-client] Build started");
94
+ }
95
+ if (userOptions.onEvent && autoDiscoveredFiles) {
96
+ try {
97
+ userOptions.onEvent({
98
+ type: "build.start",
99
+ data: {
100
+ pages: Array.from(autoDiscoveredFiles.urlMap.keys()),
101
+ files: autoDiscoveredFiles
102
+ }
103
+ });
104
+ } catch (error) {
105
+ const panicError = handleError({
106
+ error,
107
+ logger,
108
+ panicThreshold: userOptions.panicThreshold});
109
+ if (panicError != null) {
110
+ rscWorker?.terminate();
111
+ throw panicError;
112
+ }
113
+ }
114
+ }
115
+ },
116
+ async renderStart() {
117
+ timing.renderStart = performance.now();
118
+ if (userOptions.verbose) {
119
+ logger?.info("[react-static-client] Render started");
120
+ }
121
+ },
122
+ // the preview server helps to view the generated static folder, but only when the static plugin is enabled
123
+ // if no build.pages, then the preview server will instead use default vite preview server
124
+ // it works the same under both conditions
125
+ async configurePreviewServer(server) {
126
+ logger = server.config.customLogger || server.config.logger;
127
+ configurePreviewServer({
128
+ server,
129
+ userOptions
130
+ });
131
+ },
132
+ async writeBundle(_options, bundle) {
133
+ try {
134
+ if (!autoDiscoveredFiles?.urlMap) {
135
+ return;
136
+ }
137
+ const bundleManifest = getBundleManifest({
138
+ bundle,
139
+ normalizer: userOptions.normalizer
140
+ });
141
+ if (this.environment.name === "static") {
142
+ addStaticManifest(bundleManifest);
143
+ staticBundle = bundle;
144
+ } else if (this.environment.name === "client") {
145
+ if (manifests.static) {
146
+ const staticManifest = manifests.static;
147
+ for (const [, chunk] of Object.entries(bundle)) {
148
+ if (chunk.type === "chunk" && chunk.fileName) {
149
+ const normalized = userOptions.normalizer(chunk.fileName);
150
+ let value = normalized[1];
151
+ if (value.startsWith(userOptions.moduleBasePath)) {
152
+ value = value.slice(userOptions.moduleBasePath.length);
153
+ }
154
+ const entry = staticManifest[value];
155
+ if (entry && entry.file !== chunk.fileName) {
156
+ chunk.fileName = entry.file;
157
+ }
158
+ }
93
159
  }
94
- return false;
95
- },
96
- async configResolved(config) {
97
- timing.configResolved = performance.now();
98
- logger = config.customLogger || createLogger();
99
- resolvedConfig = config;
100
- // Perform auto-discovery to populate autoDiscoveredFiles
101
- const autoDiscoverResult = await resolveAutoDiscover({
102
- config: config,
160
+ }
161
+ } else if (this.environment.name === "server") {
162
+ serverBundle = bundle;
163
+ }
164
+ return;
165
+ } catch (error) {
166
+ const panicError = handleError({
167
+ error,
168
+ logger,
169
+ panicThreshold: userOptions.panicThreshold});
170
+ if (panicError != null) {
171
+ throw panicError;
172
+ }
173
+ }
174
+ },
175
+ async closeBundle() {
176
+ const envName = this.environment.name;
177
+ const isSsr = this.environment.config.build?.ssr === true;
178
+ if (userOptions.verbose) {
179
+ logger?.info(`[react-static-client] closeBundle called for environment: ${envName}, ssr: ${isSsr}`);
180
+ }
181
+ if (envName === "ssr" || envName === "server" || isSsr) {
182
+ if (userOptions.verbose) {
183
+ logger?.info(`[react-static-client] Skipping static generation for environment: ${envName} (ssr: ${isSsr})`);
184
+ }
185
+ return;
186
+ }
187
+ if (envName === "static" || envName === "client" && !isSsr) {
188
+ try {
189
+ const { rmSync, existsSync } = await import('node:fs');
190
+ const { join: join2, resolve } = await import('node:path');
191
+ const resolvedOutDir = this.environment.config.build?.outDir ? resolve(this.environment.config.root || userOptions.projectRoot, this.environment.config.build.outDir) : resolve(userOptions.projectRoot, userOptions.build.outDir);
192
+ const outputDirs = [
193
+ join2(resolvedOutDir, userOptions.build.static || "static"),
194
+ join2(resolvedOutDir, userOptions.build.client || "client")
195
+ ];
196
+ for (const outDir of outputDirs) {
197
+ const virtualDir = join2(outDir, "_virtual");
198
+ if (existsSync(virtualDir)) {
199
+ rmSync(virtualDir, { recursive: true, force: true });
200
+ if (userOptions.verbose) {
201
+ logger?.info(`[react-static-client] Cleaned up _virtual directory: ${virtualDir}`);
202
+ }
203
+ }
204
+ }
205
+ } catch (error) {
206
+ if (userOptions.verbose) {
207
+ logger?.warn(`[react-static-client] Failed to clean up _virtual directory: ${error}`);
208
+ }
209
+ }
210
+ }
211
+ if (envName === "ssr" || envName === "server" || isSsr) {
212
+ if (userOptions.verbose) {
213
+ logger?.info(`[react-static-client] Skipping static generation - not in static environment (${envName}, ssr: ${isSsr})`);
214
+ }
215
+ return;
216
+ }
217
+ const closeBundleContext = this;
218
+ deferStaticGeneration(async () => {
219
+ try {
220
+ if (!autoDiscoveredFiles) {
221
+ if (userOptions.verbose) {
222
+ logger?.warn("[react-static-client] autoDiscoveredFiles not set, attempting to re-discover");
223
+ }
224
+ const { getStashedUserOptions, getEnvironmentId } = await import('../config/stashedOptionsState.js');
225
+ const { getCondition } = await import('../config/getCondition.js');
226
+ const envId = getEnvironmentId(getCondition(), resolvedConfig?.mode || "production");
227
+ const stashedOptions = getStashedUserOptions(envId);
228
+ if (stashedOptions && resolvedConfig) {
229
+ const autoDiscoverResult = await resolveAutoDiscover({
230
+ config: resolvedConfig,
103
231
  configEnv: configEnv || {
104
- mode: config.mode,
105
- command: config.command,
106
- isSsrBuild: false,
107
- isPreview: false,
232
+ mode: resolvedConfig.mode || "production",
233
+ command: resolvedConfig.command || "build",
234
+ isSsrBuild: false,
235
+ isPreview: false
108
236
  },
109
237
  userOptions,
110
- logger,
111
- });
112
- if (autoDiscoverResult.type === "error") {
113
- throw autoDiscoverResult.error;
238
+ logger
239
+ });
240
+ if (autoDiscoverResult.type === "success") {
241
+ autoDiscoveredFiles = autoDiscoverResult.autoDiscoveredFiles;
242
+ if (userOptions.verbose) {
243
+ logger?.info(`[react-static-client] Re-discovered ${autoDiscoveredFiles.urlMap.size} pages`);
244
+ }
245
+ } else {
246
+ if (userOptions.verbose) {
247
+ logger?.warn(`[react-static-client] Failed to re-discover pages: ${autoDiscoverResult.error}`);
248
+ }
249
+ }
114
250
  }
115
- autoDiscoveredFiles = autoDiscoverResult.autoDiscoveredFiles;
251
+ }
252
+ if (!autoDiscoveredFiles?.urlMap || autoDiscoveredFiles?.urlMap.size === 0) {
116
253
  if (userOptions.verbose) {
117
- logger?.info(`Auto-discovery ${autoDiscoverResult.type === "success" ? "completed" : "skipped"}`);
254
+ logger?.warn(`[react-static-client] No pages to generate - urlMap is empty (size: ${autoDiscoveredFiles?.urlMap?.size || 0})`);
255
+ logger?.warn(`[react-static-client] autoDiscoveredFiles exists: ${!!autoDiscoveredFiles}, urlMap exists: ${!!autoDiscoveredFiles?.urlMap}`);
118
256
  }
119
- },
120
- async buildStart() {
121
- timing.buildStart = performance.now();
257
+ return;
258
+ }
259
+ if (userOptions.verbose) {
260
+ logger?.info(`[react-static-client] Starting static generation with ${autoDiscoveredFiles.urlMap.size} pages`);
261
+ }
262
+ try {
122
263
  if (userOptions.verbose) {
123
- logger?.info("[react-static-client] Build started");
264
+ logger?.info(`[react-static-client] Attempting to get server manifest from shared state`);
124
265
  }
125
- if (userOptions.onEvent && autoDiscoveredFiles) {
126
- try {
127
- userOptions.onEvent({
128
- type: "build.start",
129
- data: {
130
- pages: Array.from(autoDiscoveredFiles.urlMap.keys()),
131
- files: autoDiscoveredFiles,
132
- },
133
- });
134
- }
135
- catch (error) {
136
- const panicError = handleError({
137
- error,
138
- logger: logger,
139
- panicThreshold: userOptions.panicThreshold,
140
- context: "buildStart",
141
- });
142
- if (panicError != null) {
143
- rscWorker?.terminate();
144
- throw panicError;
145
- }
146
- }
266
+ const sharedState = getSharedManifestStore(closeBundleContext);
267
+ if (sharedState.server) {
268
+ serverManifest = sharedState.server;
269
+ if (userOptions.verbose) {
270
+ logger?.info(`[react-static-client] Got server manifest from shared state`);
271
+ }
272
+ } else {
273
+ throw new Error("No server manifest in shared state");
147
274
  }
148
- },
149
- async renderStart() {
150
- timing.renderStart = performance.now();
275
+ } catch (error) {
151
276
  if (userOptions.verbose) {
152
- logger?.info("[react-static-client] Render started");
277
+ logger?.info(`[react-static-client] Failed to get server manifest from shared state, trying filesystem: ${error}`);
153
278
  }
154
- },
155
- // the preview server helps to view the generated static folder, but only when the static plugin is enabled
156
- // if no build.pages, then the preview server will instead use default vite preview server
157
- // it works the same under both conditions
158
- async configurePreviewServer(server) {
159
- logger = server.config.customLogger || server.config.logger;
160
- configurePreviewServer({
161
- server,
162
- userOptions,
279
+ const serverManifestPath = join(
280
+ userOptions.build.outDir,
281
+ userOptions.build.server
282
+ );
283
+ const manifestPath = typeof resolvedConfig?.build.manifest === "string" ? resolvedConfig.build.manifest : ".vite/manifest.json";
284
+ if (userOptions.verbose) {
285
+ logger?.info(`[react-static-client] Loading server manifest from: ${join(serverManifestPath, manifestPath)}`);
286
+ }
287
+ const serverManifestResult = await tryManifest({
288
+ root: userOptions.projectRoot,
289
+ outDir: serverManifestPath,
290
+ manifestPath,
291
+ ssrManifest: false
292
+ });
293
+ if (serverManifestResult.type === "error") {
294
+ if (userOptions.verbose) {
295
+ logger?.warn(`[react-static-client] Failed to load server manifest: ${serverManifestResult.error}`);
296
+ }
297
+ serverManifest = {};
298
+ if (userOptions.verbose) {
299
+ logger?.warn(`[react-static-client] Using empty server manifest as fallback`);
300
+ }
301
+ } else if (serverManifestResult.type === "skip") {
302
+ if (userOptions.verbose) {
303
+ logger?.warn(`[react-static-client] Server manifest not found, using empty manifest as fallback`);
304
+ }
305
+ serverManifest = {};
306
+ } else {
307
+ serverManifest = serverManifestResult.manifest;
308
+ if (userOptions.verbose) {
309
+ logger?.info(`[react-static-client] Loaded server manifest from filesystem`);
310
+ }
311
+ }
312
+ }
313
+ const staticManifestResult = await tryManifest({
314
+ root: userOptions.projectRoot,
315
+ outDir: join(userOptions.build.outDir, userOptions.build.static),
316
+ manifestPath: resolvedConfig?.build.manifest ?? ".vite/manifest.json",
317
+ ssrManifest: false
318
+ });
319
+ if (staticManifestResult.type === "error") {
320
+ throw staticManifestResult.error;
321
+ }
322
+ const staticManifest = staticManifestResult.manifest;
323
+ const indexHtml = staticManifest?.["index.html"]?.file;
324
+ const serverPipeableStreamOptions = {
325
+ ...userOptions.serverPipeableStreamOptions,
326
+ bootstrapModules: [
327
+ ...indexHtml ? [baseURL(indexHtml)] : [],
328
+ ...userOptions.serverPipeableStreamOptions?.bootstrapModules ?? []
329
+ ]
330
+ };
331
+ userOptions.serverPipeableStreamOptions = serverPipeableStreamOptions;
332
+ const clientPipeableStreamOptions = {
333
+ ...userOptions.clientPipeableStreamOptions,
334
+ bootstrapModules: [
335
+ ...indexHtml ? [baseURL(indexHtml)] : [],
336
+ ...userOptions.clientPipeableStreamOptions?.bootstrapModules ?? []
337
+ ]
338
+ };
339
+ const { cssFilesByPage, globalCss } = processCssFilesForPages({
340
+ userOptions,
341
+ autoDiscoveredFiles,
342
+ serverManifest,
343
+ staticManifest,
344
+ bundle: staticBundle || {},
345
+ logger
346
+ });
347
+ if (userOptions.verbose) {
348
+ for (const [route, cssMap] of cssFilesByPage.entries()) {
349
+ logger.info(
350
+ `[react-static-client] Route ${route}: ${cssMap.size} CSS files`
351
+ );
352
+ for (const [key, value] of cssMap.entries()) {
353
+ logger.info(
354
+ `[react-static-client] CSS file: ${key} -> ${value.as} (${value.children ? "inline" : "link"})`
355
+ );
356
+ }
357
+ }
358
+ }
359
+ const routes = Array.from(
360
+ autoDiscoveredFiles.urlMap.keys()
361
+ );
362
+ if (routes.length === 0) {
363
+ if (userOptions.verbose) {
364
+ logger?.info(
365
+ "[react-static-client] No pages to generate, skipping static generation"
366
+ );
367
+ }
368
+ return;
369
+ }
370
+ if (userOptions.verbose) {
371
+ logger?.info(`[react-static-client] Creating build loader`);
372
+ }
373
+ const buildLoader = createBuildLoader();
374
+ if (userOptions.verbose) {
375
+ logger?.info(`[react-static-client] Build loader created`);
376
+ }
377
+ if (userOptions.verbose) {
378
+ logger?.info(
379
+ `[react-static-client] Creating RSC worker with path: ${userOptions.rscWorkerPath}`
380
+ );
381
+ }
382
+ const workerStartTime = performance.now();
383
+ let rscWorkerResult;
384
+ try {
385
+ rscWorkerResult = await createWorker({
386
+ projectRoot: userOptions.projectRoot,
387
+ workerPath: userOptions.rscWorkerPath,
388
+ currentCondition: "react-client",
389
+ reverseCondition: "react-server",
390
+ maxListeners: Math.max(routes.length * 3, 10),
391
+ // Account for multiple listeners per route
392
+ envPrefix: envPrefixFromConfig(resolvedConfig),
393
+ logger,
394
+ verbose: userOptions.verbose,
395
+ mode: getNodeEnv(),
396
+ workerData: {
397
+ userOptions: serializedOptions(userOptions, autoDiscoveredFiles),
398
+ resolvedConfig: serializeResolvedConfig(resolvedConfig),
399
+ configEnv: (() => {
400
+ const fallback = resolvedConfig ? {
401
+ command: resolvedConfig.command,
402
+ mode: resolvedConfig.mode,
403
+ isSsrBuild: false,
404
+ isPreview: false
405
+ } : void 0;
406
+ const finalConfigEnv = configEnv || fallback;
407
+ return finalConfigEnv;
408
+ })(),
409
+ serverManifest: serverManifest || {},
410
+ // Use server manifest for page component resolution
411
+ bundle: staticBundle || {},
412
+ // Use static bundle (client build) for page component resolution
413
+ staticBundle: staticBundle || {},
414
+ // Pass static bundle separately for path resolution
415
+ id: "static-client-rsc-worker"
416
+ }
417
+ });
418
+ } catch (workerError) {
419
+ if (userOptions.verbose) {
420
+ logger?.error(`[react-static-client] Error creating RSC worker: ${workerError}`);
421
+ }
422
+ throw workerError;
423
+ }
424
+ if (rscWorkerResult.type !== "success") {
425
+ const err = rscWorkerResult.error ?? new Error(`Failed to create RSC worker`);
426
+ if (userOptions.verbose) {
427
+ logger?.error(
428
+ `[react-static-client] RSC worker creation failed, throwing error`,
429
+ { error: err }
430
+ );
431
+ }
432
+ throw err;
433
+ }
434
+ rscWorker = rscWorkerResult.worker;
435
+ if (userOptions.verbose) {
436
+ logger?.info(`[react-static-client] RSC worker created successfully`);
437
+ }
438
+ const workerStartupTime = performance.now() - workerStartTime;
439
+ if (userOptions.onMetrics) {
440
+ const workerStartupMetric = createWorkerStartupMetrics({
441
+ route: "/",
442
+ // Worker startup is global, not route-specific
443
+ workerType: "rsc",
444
+ // This is the RSC worker for client-side static generation
445
+ startupTime: workerStartupTime,
446
+ fromMainThread: true,
447
+ fromRscWorker: false,
448
+ fromHtmlWorker: false,
449
+ description: `RSC worker startup for client-side static generation`
163
450
  });
164
- },
165
- async writeBundle(_options, bundle) {
166
- // Capture manifests from all environments
451
+ userOptions.onMetrics(workerStartupMetric);
452
+ }
453
+ const { onEvent, onMetrics, ...handlerOptions } = userOptions;
454
+ if (userOptions.verbose) {
455
+ logger?.info(`[react-static-client] Extracted onEvent: ${typeof onEvent}, userOptions.onEvent: ${typeof userOptions.onEvent}`);
456
+ }
457
+ if (!serverBundle && onEvent) {
458
+ const originalOnEvent = onEvent;
459
+ const tempOnEvent = (event) => {
460
+ if (event.type === "build.writeBundle.server") {
461
+ serverBundle = event.data.bundle;
462
+ logger?.info(
463
+ "[react-static-client] Captured server bundle from build event"
464
+ );
465
+ }
466
+ originalOnEvent(event);
467
+ };
468
+ userOptions.onEvent = tempOnEvent;
469
+ }
470
+ const eventHandler = onEvent || userOptions.onEvent;
471
+ if (typeof eventHandler === "function") {
167
472
  try {
168
- if (!autoDiscoveredFiles?.urlMap) {
169
- return;
473
+ if (userOptions.verbose) {
474
+ logger?.info(`[react-static-client] Emitting build.ssg.start event`);
475
+ }
476
+ const r = eventHandler({
477
+ type: "build.ssg.start",
478
+ data: {
479
+ pages: Array.from(autoDiscoveredFiles?.urlMap.keys() ?? []),
480
+ options: null,
481
+ // No specific rollup output options for static generation
482
+ bundle: staticBundle || {}
170
483
  }
171
- const bundleManifest = getBundleManifest({
172
- bundle,
173
- normalizer: userOptions.normalizer,
174
- });
175
- // Store manifest based on environment
176
- if (this.environment.name === "static") {
177
- // Store in global manifest store for environment plugin access
178
- addStaticManifest(bundleManifest);
179
- staticBundle = bundle;
484
+ });
485
+ if (r != null && typeof r === "object" && "then" in r) {
486
+ await r;
487
+ }
488
+ } catch (error) {
489
+ const eventPanicError = handleError({
490
+ error,
491
+ logger,
492
+ panicThreshold: userOptions.panicThreshold,
493
+ context: "onEvent(build.ssg.start)"
494
+ });
495
+ if (eventPanicError != null) {
496
+ throw eventPanicError;
497
+ }
498
+ }
499
+ } else if (userOptions.verbose) {
500
+ logger?.warn(`[react-static-client] No onEvent handler available to emit build.ssg.start`);
501
+ }
502
+ const renderPagesGenerator = renderPagesBatched(
503
+ routes,
504
+ {
505
+ ...handlerOptions,
506
+ // Use the clean options instead of the original handlerOptions
507
+ worker: rscWorker,
508
+ // Pass the RSC worker for RSC rendering only
509
+ rscWorker,
510
+ // Pass the RSC worker for RSC rendering only
511
+ loader: buildLoader,
512
+ // Use proper build loader instead of no-op
513
+ logger,
514
+ autoDiscoveredFiles,
515
+ cssFilesByPage,
516
+ // Pass CSS files by page
517
+ serverPipeableStreamOptions,
518
+ // Pass server options to RSC worker
519
+ clientPipeableStreamOptions,
520
+ // Pass client options to RSC worker
521
+ globalCss,
522
+ // Pass global CSS
523
+ manifest: serverManifest || {},
524
+ // Server manifest for RSC worker
525
+ staticManifest,
526
+ // Static manifest for consistent module IDs
527
+ onEvent,
528
+ onMetrics
529
+ // Pass through the onMetrics callback (metric watcher)
530
+ },
531
+ renderPage
532
+ );
533
+ let finalResult = void 0;
534
+ try {
535
+ for await (const result of renderPagesGenerator) {
536
+ if (result.type === "error") {
537
+ if (userOptions.verbose) {
538
+ logger?.error(`[react-static-client] Render error: ${result.error}`);
180
539
  }
181
- else if (this.environment.name === "client") {
182
- // Client build manifest (SSR modules) - stored globally now
183
- if (manifests.static) {
184
- const staticManifest = manifests.static;
185
- // Update bundle filenames to match static manifest
186
- for (const [, chunk] of Object.entries(bundle)) {
187
- if (chunk.type === "chunk" && chunk.fileName) {
188
- const normalized = userOptions.normalizer(chunk.fileName);
189
- let value = normalized[1];
190
- if (value.startsWith(userOptions.moduleBasePath)) {
191
- value = value.slice(userOptions.moduleBasePath.length);
192
- }
193
- const entry = staticManifest[value];
194
- if (entry && entry.file !== chunk.fileName) {
195
- // Update the filename to match static manifest
196
- chunk.fileName = entry.file;
197
- }
198
- }
199
- }
200
- }
540
+ throw result.error;
541
+ }
542
+ if (result.type === "success" && result.failedRoutes && result.failedRoutes.size > 0) {
543
+ const firstError = result.failedRoutes.values().next().value;
544
+ if (firstError != null && shouldCausePanic(firstError, {
545
+ panicThreshold: userOptions.panicThreshold
546
+ })) {
547
+ throw firstError;
201
548
  }
202
- else if (this.environment.name === "server") {
203
- // Server build manifest (server components) - stored globally now
204
- serverBundle = bundle;
549
+ for (const [route, error] of result.failedRoutes) {
550
+ const err = error instanceof Error ? error : toError(error);
551
+ closeBundleContext.warn(
552
+ new Error(
553
+ "Failed to render route: " + route + "\n" + err.message + "\n" + err.stack,
554
+ { cause: err }
555
+ )
556
+ );
205
557
  }
206
- // Skip the static generation here - it will happen in closeBundle
207
- return;
558
+ }
559
+ finalResult = result;
208
560
  }
209
- catch (error) {
210
- const panicError = handleError({
211
- error,
212
- logger: logger,
213
- panicThreshold: userOptions.panicThreshold,
214
- context: "writeBundle",
215
- });
216
- if (panicError != null) {
217
- throw panicError;
218
- }
219
- }
220
- },
221
- async closeBundle() {
222
- const envName = this.environment.name;
223
- const isSsr = this.environment.config.build?.ssr === true;
561
+ } catch (renderError) {
224
562
  if (userOptions.verbose) {
225
- logger?.info(`[react-static-client] closeBundle called for environment: ${envName}, ssr: ${isSsr}`);
563
+ logger?.error(`[react-static-client] Error during renderPages: ${renderError}`);
226
564
  }
227
- // Only run static generation in the non-SSR client environment (static builds)
228
- // Skip SSR client builds and server builds
229
- if (envName === "ssr" || envName === "server" || isSsr) {
230
- if (userOptions.verbose) {
231
- logger?.info(`[react-static-client] Skipping static generation for environment: ${envName} (ssr: ${isSsr})`);
232
- }
233
- return;
565
+ throw renderError;
566
+ }
567
+ if (!finalResult) {
568
+ const errorMsg = "No render result produced";
569
+ if (userOptions.verbose) {
570
+ logger?.error(`[react-static-client] ${errorMsg}`);
234
571
  }
235
- // Clean up _virtual files after build completes
236
- // These are Vite's internal virtual modules and aren't needed in the final output
237
- if (envName === "static" || (envName === "client" && !isSsr)) {
238
- try {
239
- const { rmSync, existsSync } = await import("node:fs");
240
- const { join, resolve } = await import("node:path");
241
- // Use the resolved output directory from the environment config
242
- const resolvedOutDir = this.environment.config.build?.outDir
243
- ? resolve(this.environment.config.root || userOptions.projectRoot, this.environment.config.build.outDir)
244
- : resolve(userOptions.projectRoot, userOptions.build.outDir);
245
- // Clean up _virtual from client/static output directories only
246
- // Don't clean up server/_virtual since we need dynamic-import-helper.js there
247
- const outputDirs = [
248
- join(resolvedOutDir, userOptions.build.static || "static"),
249
- join(resolvedOutDir, userOptions.build.client || "client"),
250
- ];
251
- for (const outDir of outputDirs) {
252
- const virtualDir = join(outDir, "_virtual");
253
- if (existsSync(virtualDir)) {
254
- rmSync(virtualDir, { recursive: true, force: true });
255
- if (userOptions.verbose) {
256
- logger?.info(`[react-static-client] Cleaned up _virtual directory: ${virtualDir}`);
257
- }
258
- }
259
- }
260
- }
261
- catch (error) {
262
- // Non-critical - log but don't fail the build
263
- if (userOptions.verbose) {
264
- logger?.warn(`[react-static-client] Failed to clean up _virtual directory: ${error}`);
265
- }
572
+ throw new Error(errorMsg);
573
+ }
574
+ if (userOptions.verbose) {
575
+ logger?.info(`[react-static-client] Render completed: ${finalResult.completedRoutes.size} pages, ${finalResult.failedRoutes?.size || 0} failed`);
576
+ }
577
+ const duration = Math.round(
578
+ performance.now() - (timing.renderStart || timing.start)
579
+ );
580
+ closeBundleContext.info(
581
+ `Rendered ${finalResult.completedRoutes.size} pages in ${duration}ms`
582
+ );
583
+ if (process.env["NODE_ENV"] !== "production") {
584
+ closeBundleContext.warn(
585
+ `THIS BUILD IS NOT INTENDED FOR PRODUCTION (${process.env["NODE_ENV"]})`
586
+ );
587
+ }
588
+ timing.render = performance.now() - (timing.renderStart ?? timing.start);
589
+ if (userOptions.verbose) {
590
+ logger?.info("[react-static-client] Static generation completed");
591
+ }
592
+ if (typeof userOptions.onEvent === "function") {
593
+ try {
594
+ const r = userOptions.onEvent({
595
+ type: "build.ssg.end",
596
+ data: {
597
+ pages: Array.from(autoDiscoveredFiles?.urlMap.keys() ?? []),
598
+ options: null,
599
+ // No specific rollup output options for static generation
600
+ bundle: staticBundle || {}
266
601
  }
602
+ });
603
+ if (r != null && typeof r === "object" && "then" in r) {
604
+ await r;
605
+ }
606
+ } catch (error) {
607
+ const eventPanicError = handleError({
608
+ error,
609
+ logger,
610
+ panicThreshold: userOptions.panicThreshold,
611
+ context: "onEvent(build.ssg.end)"
612
+ });
613
+ if (eventPanicError != null) {
614
+ throw eventPanicError;
615
+ }
267
616
  }
268
- // This runs after all writeBundle hooks are complete
269
- // Run static generation in the non-SSR client environment (static builds)
270
- // This could be "static" or "client" depending on how environments are configured
271
- if (envName === "ssr" || envName === "server" || isSsr) {
272
- if (userOptions.verbose) {
273
- logger?.info(`[react-static-client] Skipping static generation - not in static environment (${envName}, ssr: ${isSsr})`);
274
- }
275
- return;
617
+ }
618
+ } catch (error) {
619
+ const panicError = handleError({
620
+ error,
621
+ logger,
622
+ panicThreshold: userOptions.panicThreshold
623
+ });
624
+ if (rscWorker) {
625
+ const workerToCleanup = rscWorker;
626
+ try {
627
+ await Promise.race([
628
+ new Promise((resolve) => {
629
+ const timeoutId = setTimeout(() => {
630
+ workerToCleanup.removeAllListeners();
631
+ workerToCleanup.terminate();
632
+ resolve();
633
+ }, 1e3);
634
+ const messageHandler = (message) => {
635
+ if (message.type === "SHUTDOWN_COMPLETE") {
636
+ clearTimeout(timeoutId);
637
+ workerToCleanup.removeListener("message", messageHandler);
638
+ resolve();
639
+ }
640
+ };
641
+ workerToCleanup.on("message", messageHandler);
642
+ workerToCleanup.postMessage({ type: "SHUTDOWN" });
643
+ })
644
+ ]);
645
+ rscWorker = void 0;
646
+ } catch (cleanupError) {
647
+ logger.warn(`Failed to cleanup worker on error: ${cleanupError}`);
648
+ try {
649
+ workerToCleanup.removeAllListeners();
650
+ workerToCleanup.terminate();
651
+ } catch (terminateError) {
652
+ }
653
+ rscWorker = void 0;
276
654
  }
277
- // Defer static generation to run after ALL environments complete their builds.
278
- // This is necessary because we need the server manifest (from server env's writeBundle)
279
- // to resolve function-based component paths like Root: (url) => 'src/CustomRoot.tsx'.
280
- // The buildApp hook in createEnvironmentPlugin will call runDeferredStaticGeneration().
281
- const closeBundleContext = this;
282
- deferStaticGeneration(async () => {
655
+ }
656
+ if (panicError != null) {
657
+ const errorToThrow = panicError instanceof Error ? panicError : new Error(String(panicError));
658
+ const finalError = new Error(errorToThrow.message);
659
+ finalError.stack = errorToThrow.stack;
660
+ finalError.cause = errorToThrow.cause;
661
+ if (errorToThrow.name) finalError.name = errorToThrow.name;
662
+ throw finalError;
663
+ }
664
+ } finally {
665
+ if (rscWorker) {
666
+ try {
667
+ await Promise.race([
668
+ new Promise((resolve, reject) => {
669
+ const timeout = setTimeout(() => {
670
+ reject(new Error("Worker shutdown timeout"));
671
+ }, userOptions.workerShutdownTimeout);
672
+ const backupTimeout = setTimeout(() => {
673
+ reject(new Error("Worker shutdown backup timeout"));
674
+ }, Math.floor(userOptions.workerShutdownTimeout * 0.6));
675
+ const shutdownMessageHandler = (message) => {
676
+ if (message.type === "SHUTDOWN_COMPLETE") {
677
+ clearTimeout(timeout);
678
+ clearTimeout(backupTimeout);
679
+ rscWorker?.removeListener(
680
+ "message",
681
+ shutdownMessageHandler
682
+ );
683
+ rscWorker?.removeAllListeners();
684
+ resolve();
685
+ }
686
+ };
687
+ rscWorker?.on("message", shutdownMessageHandler);
688
+ rscWorker?.postMessage({
689
+ type: "SHUTDOWN",
690
+ id: "*"
691
+ });
692
+ })
693
+ ]);
694
+ } catch {
695
+ } finally {
696
+ if (rscWorker) {
283
697
  try {
284
- // Re-check autoDiscoveredFiles - it might not be set if configResolved didn't run
285
- // or if it was cleared. Try to get it from stashed options if needed
286
- if (!autoDiscoveredFiles) {
287
- if (userOptions.verbose) {
288
- logger?.warn("[react-static-client] autoDiscoveredFiles not set, attempting to re-discover");
289
- }
290
- const { getStashedUserOptions, getEnvironmentId } = await import("../config/stashedOptionsState.js");
291
- const { getCondition } = await import("../config/getCondition.js");
292
- const envId = getEnvironmentId(getCondition(), resolvedConfig?.mode || "production");
293
- const stashedOptions = getStashedUserOptions(envId);
294
- if (stashedOptions && resolvedConfig) {
295
- // Try to re-run auto-discovery if we have the config
296
- const autoDiscoverResult = await resolveAutoDiscover({
297
- config: resolvedConfig,
298
- configEnv: configEnv || {
299
- mode: resolvedConfig.mode || "production",
300
- command: resolvedConfig.command || "build",
301
- isSsrBuild: false,
302
- isPreview: false,
303
- },
304
- userOptions,
305
- logger,
306
- });
307
- if (autoDiscoverResult.type === "success") {
308
- autoDiscoveredFiles = autoDiscoverResult.autoDiscoveredFiles;
309
- if (userOptions.verbose) {
310
- logger?.info(`[react-static-client] Re-discovered ${autoDiscoveredFiles.urlMap.size} pages`);
311
- }
312
- }
313
- else {
314
- if (userOptions.verbose) {
315
- logger?.warn(`[react-static-client] Failed to re-discover pages: ${autoDiscoverResult.error}`);
316
- }
317
- }
318
- }
319
- }
320
- if (!autoDiscoveredFiles?.urlMap ||
321
- autoDiscoveredFiles?.urlMap.size === 0) {
322
- if (userOptions.verbose) {
323
- logger?.warn(`[react-static-client] No pages to generate - urlMap is empty (size: ${autoDiscoveredFiles?.urlMap?.size || 0})`);
324
- logger?.warn(`[react-static-client] autoDiscoveredFiles exists: ${!!autoDiscoveredFiles}, urlMap exists: ${!!autoDiscoveredFiles?.urlMap}`);
325
- }
326
- return;
327
- }
328
- if (userOptions.verbose) {
329
- logger?.info(`[react-static-client] Starting static generation with ${autoDiscoveredFiles.urlMap.size} pages`);
330
- }
331
- // Check if we can access the shared manifest store
332
- try {
333
- if (userOptions.verbose) {
334
- logger?.info(`[react-static-client] Attempting to get server manifest from shared state`);
335
- }
336
- const sharedState = getSharedManifestStore(closeBundleContext);
337
- if (sharedState.server) {
338
- serverManifest = sharedState.server;
339
- if (userOptions.verbose) {
340
- logger?.info(`[react-static-client] Got server manifest from shared state`);
341
- }
342
- }
343
- else {
344
- throw new Error("No server manifest in shared state");
345
- }
346
- }
347
- catch (error) {
348
- if (userOptions.verbose) {
349
- logger?.info(`[react-static-client] Failed to get server manifest from shared state, trying filesystem: ${error}`);
350
- }
351
- const serverManifestPath = join(userOptions.build.outDir, userOptions.build.server);
352
- const manifestPath = (typeof resolvedConfig?.build.manifest === "string"
353
- ? resolvedConfig.build.manifest
354
- : ".vite/manifest.json");
355
- if (userOptions.verbose) {
356
- logger?.info(`[react-static-client] Loading server manifest from: ${join(serverManifestPath, manifestPath)}`);
357
- }
358
- const serverManifestResult = await tryManifest({
359
- root: userOptions.projectRoot,
360
- outDir: serverManifestPath,
361
- manifestPath: manifestPath,
362
- ssrManifest: false,
363
- });
364
- if (serverManifestResult.type === "error") {
365
- if (userOptions.verbose) {
366
- logger?.warn(`[react-static-client] Failed to load server manifest: ${serverManifestResult.error}`);
367
- }
368
- // Use empty manifest as fallback - static generation can proceed without it
369
- serverManifest = {};
370
- if (userOptions.verbose) {
371
- logger?.warn(`[react-static-client] Using empty server manifest as fallback`);
372
- }
373
- }
374
- else if (serverManifestResult.type === "skip") {
375
- if (userOptions.verbose) {
376
- logger?.warn(`[react-static-client] Server manifest not found, using empty manifest as fallback`);
377
- }
378
- // Use empty manifest as fallback - static generation can proceed without it
379
- serverManifest = {};
380
- }
381
- else {
382
- serverManifest = serverManifestResult.manifest;
383
- if (userOptions.verbose) {
384
- logger?.info(`[react-static-client] Loaded server manifest from filesystem`);
385
- }
386
- }
387
- }
388
- // Load static manifest from filesystem for CSS path mapping
389
- const staticManifestResult = await tryManifest({
390
- root: userOptions.projectRoot,
391
- outDir: join(userOptions.build.outDir, userOptions.build.static),
392
- manifestPath: resolvedConfig?.build.manifest ?? ".vite/manifest.json",
393
- ssrManifest: false,
394
- });
395
- if (staticManifestResult.type === "error") {
396
- throw staticManifestResult.error;
397
- }
398
- const staticManifest = staticManifestResult.manifest;
399
- // Construct bootstrapModules like the server plugin does
400
- const indexHtml = staticManifest?.["index.html"]?.file;
401
- const serverPipeableStreamOptions = {
402
- ...userOptions.serverPipeableStreamOptions,
403
- bootstrapModules: [
404
- ...(indexHtml ? [baseURL(indexHtml)] : []),
405
- ...(userOptions.serverPipeableStreamOptions?.bootstrapModules ??
406
- []),
407
- ],
408
- };
409
- userOptions.serverPipeableStreamOptions = serverPipeableStreamOptions;
410
- const clientPipeableStreamOptions = {
411
- ...userOptions.clientPipeableStreamOptions,
412
- bootstrapModules: [
413
- ...(indexHtml ? [baseURL(indexHtml)] : []),
414
- ...(userOptions.clientPipeableStreamOptions?.bootstrapModules ??
415
- []),
416
- ],
417
- };
418
- // Create CSS props for each CSS file (same as server-static)
419
- const { cssFilesByPage, globalCss } = processCssFilesForPages({
420
- userOptions,
421
- autoDiscoveredFiles,
422
- serverManifest,
423
- staticManifest,
424
- bundle: staticBundle || {},
425
- logger,
426
- });
427
- if (userOptions.verbose) {
428
- for (const [route, cssMap] of cssFilesByPage.entries()) {
429
- logger.info(`[react-static-client] Route ${route}: ${cssMap.size} CSS files`);
430
- for (const [key, value] of cssMap.entries()) {
431
- logger.info(`[react-static-client] CSS file: ${key} -> ${value.as} (${value.children ? "inline" : "link"})`);
432
- }
433
- }
434
- }
435
- const routes = Array.from(autoDiscoveredFiles.urlMap.keys());
436
- // If no pages to generate, skip static generation
437
- if (routes.length === 0) {
438
- if (userOptions.verbose) {
439
- logger?.info("[react-static-client] No pages to generate, skipping static generation");
440
- }
441
- return;
442
- }
443
- // Use the static manifest to ensure consistent module IDs between RSC stream and client build
444
- // The static manifest contains the correct hashes that should be used for both builds
445
- // (staticManifest already loaded above)
446
- // Create a build loader for client mode (reuse server's sophisticated loader)
447
- if (userOptions.verbose) {
448
- logger?.info(`[react-static-client] Creating build loader`);
449
- }
450
- const buildLoader = createBuildLoader();
451
- if (userOptions.verbose) {
452
- logger?.info(`[react-static-client] Build loader created`);
453
- }
454
- // Create an RSC worker for generating RSC content
455
- if (userOptions.verbose) {
456
- logger?.info(`[react-static-client] Creating RSC worker with path: ${userOptions.rscWorkerPath}`);
457
- }
458
- const workerStartTime = performance.now();
459
- let rscWorkerResult;
460
- try {
461
- rscWorkerResult = await createWorker({
462
- projectRoot: userOptions.projectRoot,
463
- workerPath: userOptions.rscWorkerPath,
464
- currentCondition: "react-client",
465
- reverseCondition: "react-server",
466
- maxListeners: Math.max(routes.length * 3, 10), // Account for multiple listeners per route
467
- envPrefix: envPrefixFromConfig(resolvedConfig),
468
- logger: logger,
469
- verbose: userOptions.verbose,
470
- mode: getNodeEnv(),
471
- workerData: {
472
- userOptions: serializedOptions(userOptions, autoDiscoveredFiles),
473
- resolvedConfig: serializeResolvedConfig(resolvedConfig),
474
- configEnv: (() => {
475
- const fallback = resolvedConfig
476
- ? {
477
- command: resolvedConfig.command,
478
- mode: resolvedConfig.mode,
479
- isSsrBuild: false,
480
- isPreview: false,
481
- }
482
- : undefined;
483
- const finalConfigEnv = configEnv || fallback;
484
- return finalConfigEnv;
485
- })(),
486
- serverManifest: serverManifest || {}, // Use server manifest for page component resolution
487
- bundle: staticBundle || {}, // Use static bundle (client build) for page component resolution
488
- staticBundle: staticBundle || {}, // Pass static bundle separately for path resolution
489
- id: "static-client-rsc-worker",
490
- },
491
- });
492
- }
493
- catch (workerError) {
494
- if (userOptions.verbose) {
495
- logger?.error(`[react-static-client] Error creating RSC worker: ${workerError}`);
496
- }
497
- throw workerError;
498
- }
499
- if (rscWorkerResult.type !== "success") {
500
- const err = rscWorkerResult.error ?? new Error(`Failed to create RSC worker`);
501
- if (userOptions.verbose) {
502
- logger?.error(`[react-static-client] RSC worker creation failed, throwing error`, { error: err });
503
- }
504
- throw err;
505
- }
506
- rscWorker = rscWorkerResult.worker;
507
- if (userOptions.verbose) {
508
- logger?.info(`[react-static-client] RSC worker created successfully`);
509
- }
510
- // Emit worker startup metric after worker is created
511
- const workerStartupTime = performance.now() - workerStartTime;
512
- if (userOptions.onMetrics) {
513
- const workerStartupMetric = createWorkerStartupMetrics({
514
- route: "/", // Worker startup is global, not route-specific
515
- workerType: "rsc", // This is the RSC worker for client-side static generation
516
- startupTime: workerStartupTime,
517
- fromMainThread: true,
518
- fromRscWorker: false,
519
- fromHtmlWorker: false,
520
- description: `RSC worker startup for client-side static generation`,
521
- });
522
- userOptions.onMetrics(workerStartupMetric);
523
- }
524
- // Render pages using client-side renderer with RSC worker only
525
- const { onEvent, onMetrics, ...handlerOptions } = userOptions;
526
- if (userOptions.verbose) {
527
- logger?.info(`[react-static-client] Extracted onEvent: ${typeof onEvent}, userOptions.onEvent: ${typeof userOptions.onEvent}`);
528
- }
529
- // Capture server bundle from onEvent if not already captured
530
- if (!serverBundle && onEvent) {
531
- // Create a temporary event handler to capture the server bundle
532
- const originalOnEvent = onEvent;
533
- const tempOnEvent = (event) => {
534
- if (event.type === "build.writeBundle.server") {
535
- serverBundle = event.data.bundle;
536
- logger?.info("[react-static-client] Captured server bundle from build event");
537
- }
538
- // Call the original event handler
539
- originalOnEvent(event);
540
- };
541
- // Replace the onEvent temporarily to capture the server bundle
542
- userOptions.onEvent = tempOnEvent;
543
- }
544
- // Emit the static site generation start event
545
- // Use the extracted onEvent if available, otherwise fall back to userOptions.onEvent
546
- const eventHandler = onEvent || userOptions.onEvent;
547
- if (typeof eventHandler === "function") {
548
- try {
549
- if (userOptions.verbose) {
550
- logger?.info(`[react-static-client] Emitting build.ssg.start event`);
551
- }
552
- const r = eventHandler({
553
- type: "build.ssg.start",
554
- data: {
555
- pages: Array.from(autoDiscoveredFiles?.urlMap.keys() ?? []),
556
- options: null, // No specific rollup output options for static generation
557
- bundle: staticBundle || {},
558
- },
559
- });
560
- if (r != null && typeof r === "object" && "then" in r) {
561
- await r;
562
- }
563
- }
564
- catch (error) {
565
- const eventPanicError = handleError({
566
- error,
567
- logger: logger,
568
- panicThreshold: userOptions.panicThreshold,
569
- context: "onEvent(build.ssg.start)",
570
- });
571
- if (eventPanicError != null) {
572
- throw eventPanicError; // Re-throw to abort the build
573
- }
574
- }
575
- }
576
- else if (userOptions.verbose) {
577
- logger?.warn(`[react-static-client] No onEvent handler available to emit build.ssg.start`);
578
- }
579
- const renderPagesGenerator = renderPagesBatched(routes, {
580
- ...handlerOptions, // Use the clean options instead of the original handlerOptions
581
- worker: rscWorker, // Pass the RSC worker for RSC rendering only
582
- rscWorker: rscWorker, // Pass the RSC worker for RSC rendering only
583
- loader: buildLoader, // Use proper build loader instead of no-op
584
- logger: logger,
585
- autoDiscoveredFiles: autoDiscoveredFiles,
586
- cssFilesByPage: cssFilesByPage, // Pass CSS files by page
587
- serverPipeableStreamOptions: serverPipeableStreamOptions, // Pass server options to RSC worker
588
- clientPipeableStreamOptions: clientPipeableStreamOptions, // Pass client options to RSC worker
589
- globalCss: globalCss, // Pass global CSS
590
- manifest: serverManifest || {}, // Server manifest for RSC worker
591
- staticManifest: staticManifest, // Static manifest for consistent module IDs
592
- onEvent: onEvent,
593
- onMetrics: onMetrics, // Pass through the onMetrics callback (metric watcher)
594
- }, renderPage);
595
- // Process the rendered pages
596
- let finalResult = undefined;
597
- try {
598
- for await (const result of renderPagesGenerator) {
599
- if (result.type === "error") {
600
- if (userOptions.verbose) {
601
- logger?.error(`[react-static-client] Render error: ${result.error}`);
602
- }
603
- throw result.error;
604
- }
605
- // Handle failed routes based on panic threshold
606
- if (result.type === "success" &&
607
- result.failedRoutes &&
608
- result.failedRoutes.size > 0) {
609
- // Use centralized panic threshold logic (same as server plugin)
610
- const firstError = result.failedRoutes.values().next().value;
611
- if (firstError != null &&
612
- shouldCausePanic(firstError, {
613
- panicThreshold: userOptions.panicThreshold,
614
- })) {
615
- // This should cause a panic, throw the error
616
- throw firstError;
617
- }
618
- // For other panic thresholds, log warnings but continue
619
- for (const [route, error] of result.failedRoutes) {
620
- const err = error instanceof Error ? error : toError(error);
621
- closeBundleContext.warn(new Error("Failed to render route: " +
622
- route +
623
- "\n" +
624
- err.message +
625
- "\n" +
626
- err.stack, { cause: err }));
627
- }
628
- }
629
- finalResult = result;
630
- }
631
- }
632
- catch (renderError) {
633
- if (userOptions.verbose) {
634
- logger?.error(`[react-static-client] Error during renderPages: ${renderError}`);
635
- }
636
- throw renderError;
637
- }
638
- if (!finalResult) {
639
- const errorMsg = "No render result produced";
640
- if (userOptions.verbose) {
641
- logger?.error(`[react-static-client] ${errorMsg}`);
642
- }
643
- throw new Error(errorMsg);
644
- }
645
- if (userOptions.verbose) {
646
- logger?.info(`[react-static-client] Render completed: ${finalResult.completedRoutes.size} pages, ${finalResult.failedRoutes?.size || 0} failed`);
647
- }
648
- // File writes are handled by renderPages, no need to do them here
649
- // Calculate duration from timing
650
- const duration = Math.round(performance.now() - (timing.renderStart || timing.start));
651
- closeBundleContext.info(`Rendered ${finalResult.completedRoutes.size} pages in ${duration}ms`);
652
- if (process.env["NODE_ENV"] !== "production") {
653
- closeBundleContext.warn(`THIS BUILD IS NOT INTENDED FOR PRODUCTION (${process.env["NODE_ENV"]})`);
654
- }
655
- // Update timing
656
- timing.render =
657
- performance.now() - (timing.renderStart ?? timing.start);
658
- if (userOptions.verbose) {
659
- logger?.info("[react-static-client] Static generation completed");
660
- }
661
- // Emit the static site generation completion event once
662
- if (typeof userOptions.onEvent === "function") {
663
- try {
664
- const r = userOptions.onEvent({
665
- type: "build.ssg.end",
666
- data: {
667
- pages: Array.from(autoDiscoveredFiles?.urlMap.keys() ?? []),
668
- options: null, // No specific rollup output options for static generation
669
- bundle: staticBundle || {},
670
- },
671
- });
672
- if (r != null && typeof r === "object" && "then" in r) {
673
- await r;
674
- }
675
- }
676
- catch (error) {
677
- const eventPanicError = handleError({
678
- error,
679
- logger: logger,
680
- panicThreshold: userOptions.panicThreshold,
681
- context: "onEvent(build.ssg.end)",
682
- });
683
- if (eventPanicError != null) {
684
- throw eventPanicError; // Re-throw to abort the build
685
- }
686
- }
687
- }
688
- }
689
- catch (error) {
690
- const panicError = handleError({
691
- error,
692
- context: "react-static-client",
693
- logger,
694
- panicThreshold: userOptions.panicThreshold,
695
- });
696
- // Ensure graceful shutdown on error
697
- if (rscWorker) {
698
- const workerToCleanup = rscWorker;
699
- try {
700
- // Use graceful shutdown protocol even on error
701
- await Promise.race([
702
- new Promise((resolve) => {
703
- const timeoutId = setTimeout(() => {
704
- workerToCleanup.removeAllListeners();
705
- workerToCleanup.terminate();
706
- resolve();
707
- }, 1000); // 1 second timeout for graceful shutdown
708
- const messageHandler = (message) => {
709
- if (message.type === "SHUTDOWN_COMPLETE") {
710
- clearTimeout(timeoutId);
711
- workerToCleanup.removeListener("message", messageHandler);
712
- resolve();
713
- }
714
- };
715
- workerToCleanup.on("message", messageHandler);
716
- workerToCleanup.postMessage({ type: "SHUTDOWN" });
717
- }),
718
- ]);
719
- rscWorker = undefined;
720
- }
721
- catch (cleanupError) {
722
- logger.warn(`Failed to cleanup worker on error: ${cleanupError}`);
723
- // Force terminate if graceful shutdown fails
724
- try {
725
- workerToCleanup.removeAllListeners();
726
- workerToCleanup.terminate();
727
- }
728
- catch (terminateError) {
729
- // Ignore termination errors
730
- }
731
- rscWorker = undefined;
732
- }
733
- }
734
- if (panicError != null) {
735
- // Ensure we have a proper Error object that can have properties set on it
736
- const errorToThrow = panicError instanceof Error
737
- ? panicError
738
- : new Error(String(panicError));
739
- // Create a new Error object to avoid the "code" property issue
740
- const finalError = new Error(errorToThrow.message);
741
- finalError.stack = errorToThrow.stack;
742
- finalError.cause = errorToThrow.cause;
743
- // Copy any additional properties that might be needed
744
- if (errorToThrow.name)
745
- finalError.name = errorToThrow.name;
746
- throw finalError;
747
- }
748
- }
749
- finally {
750
- // Graceful worker shutdown — runs on both success and error paths
751
- if (rscWorker) {
752
- try {
753
- await Promise.race([
754
- new Promise((resolve, reject) => {
755
- const timeout = setTimeout(() => {
756
- reject(new Error("Worker shutdown timeout"));
757
- }, userOptions.workerShutdownTimeout);
758
- const backupTimeout = setTimeout(() => {
759
- reject(new Error("Worker shutdown backup timeout"));
760
- }, Math.floor(userOptions.workerShutdownTimeout * 0.6));
761
- const shutdownMessageHandler = (message) => {
762
- if (message.type === "SHUTDOWN_COMPLETE") {
763
- clearTimeout(timeout);
764
- clearTimeout(backupTimeout);
765
- rscWorker?.removeListener("message", shutdownMessageHandler);
766
- rscWorker?.removeAllListeners();
767
- resolve();
768
- }
769
- };
770
- rscWorker?.on("message", shutdownMessageHandler);
771
- rscWorker?.postMessage({
772
- type: "SHUTDOWN",
773
- id: "*",
774
- });
775
- }),
776
- ]);
777
- }
778
- catch {
779
- // Shutdown protocol failed — force terminate below
780
- }
781
- finally {
782
- if (rscWorker) {
783
- try {
784
- rscWorker.removeAllListeners();
785
- rscWorker.terminate();
786
- }
787
- catch {
788
- // Ignore termination errors
789
- }
790
- rscWorker = undefined;
791
- }
792
- }
793
- }
794
- // Reset any cached state to prevent issues in subsequent builds
795
- autoDiscoveredFiles = null;
796
- serverManifest = undefined;
698
+ rscWorker.removeAllListeners();
699
+ rscWorker.terminate();
700
+ } catch {
797
701
  }
798
- }); // end deferStaticGeneration
799
- },
800
- };
702
+ rscWorker = void 0;
703
+ }
704
+ }
705
+ }
706
+ autoDiscoveredFiles = null;
707
+ serverManifest = void 0;
708
+ }
709
+ });
710
+ }
711
+ };
801
712
  };
713
+
714
+ export { reactStaticPlugin };
715
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"plugin.client.js","sources":["../../../plugin/react-static/plugin.client.ts"],"sourcesContent":["/**\n * plugin.client.ts\n *\n * PURPOSE: Client-side static plugin for React Server Components\n *\n * This module:\n * 1. Handles static site generation in the client environment\n * 2. Uses RSC worker for RSC rendering and main-thread for HTML rendering\n * 3. Generates both RSC and HTML files for static pages\n * 4. Integrates with Vite's build process\n *\n * Feature parity with main react-static plugin, but in reverse. Uses rsc-worker to render rsc, and main thread for html.\n * This is not the default behavior, but is supported for testing and custom app development purposes.\n * Additionally, this can make it easier to use the --app flag to build all the modules + static generation at once.\n */\n\nimport {\n  createLogger,\n  type ResolvedConfig,\n  type Manifest,\n  type ConfigEnv,\n} from \"vite\";\nimport { resolveOptions } from \"../config/resolveOptions.js\";\nimport type {\n  BuildTiming,\n  VitePluginFn,\n  AutoDiscoveredFiles,\n} from \"../types.js\";\nimport type { OutputBundle } from \"rollup\";\nimport { renderPagesBatched } from \"./renderPagesBatched.js\";\nimport { performance } from \"node:perf_hooks\";\nimport { renderPage } from \"./renderPage.client.js\";\n\nimport { createWorker } from \"../worker/createWorker.js\";\nimport {\n  serializedOptions,\n  serializeResolvedConfig,\n} from \"../helpers/serializeUserOptions.js\";\nimport { getBundleManifest } from \"../helpers/getBundleManifest.js\";\n\nimport { handleError } from \"../error/handleError.js\";\nimport { shouldCausePanic } from \"../error/panicThresholdHandler.js\";\nimport { configurePreviewServer } from \"./configurePreviewServer.js\";\nimport { assertNonReactServer } from \"../config/getCondition.js\";\nimport { envPrefixFromConfig } from \"../config/envPrefixFromConfig.js\";\nimport { createWorkerStartupMetrics } from \"../metrics/createWorkerStartupMetrics.js\";\nimport { processCssFilesForPages } from \"./processCssFilesForPages.js\";\nimport { createBuildLoader } from \"./createBuildLoader.client.js\";\nimport { getNodeEnv } from \"../config/getNodeEnv.js\";\nimport { toError } from \"../error/toError.js\";\nimport {\n  addStaticManifest,\n  manifests,\n  getSharedManifestStore,\n} from \"../bundle/manifests.js\";\nimport { deferStaticGeneration } from \"../bundle/deferredStaticGeneration.js\";\nimport type { Worker } from \"node:worker_threads\";\nimport { resolveAutoDiscover } from \"../config/index.js\";\nimport { join } from \"node:path\";\n\nimport { baseURL } from \"../utils/envUrls.node.js\";\nimport { tryManifest } from \"../helpers/tryManifest.js\";\n// cssCollector removed - using filesystem-based CSS processing\n\nassertNonReactServer();\n\n/**\n * plugin.client.ts\n *\n * PURPOSE: Client-side static plugin for React Server Components\n *\n * This module:\n * 1. Handles static site generation in the client environment\n * 2. Uses RSC worker for RSC rendering and main-thread for HTML rendering\n * 3. Generates both RSC and HTML files for static pages\n * 4. Integrates with Vite's build process\n *\n * @param options\n * @returns\n */\nexport const reactStaticPlugin: VitePluginFn = function _reactStaticPlugin(\n  options\n) {\n  let logger: ReturnType<typeof createLogger>;\n  let autoDiscoveredFiles: AutoDiscoveredFiles | null = null;\n  let rscWorker: Worker | undefined = undefined;\n  let resolvedConfig: ResolvedConfig | null = null;\n  let serverManifest: Manifest | undefined = undefined;\n  let staticBundle: OutputBundle | undefined = undefined;\n  let serverBundle: OutputBundle | undefined = undefined;\n\n  let configEnv: ConfigEnv | undefined;\n  const timing: BuildTiming = {\n    start: performance.now(),\n    configResolved: 0,\n    buildStart: 0,\n    renderStart: 0,\n  };\n\n  const resolvedOptions = resolveOptions(options);\n  if (resolvedOptions.type === \"error\") {\n    throw resolvedOptions.error;\n  }\n  const userOptions = resolvedOptions.userOptions;\n\n  return {\n    name: \"vite:plugin-react-server/client-static\",\n    enforce: \"post\",\n    apply: \"build\", // Apply to build mode\n    api: {\n      meta: { timing },\n    },\n    async config(_config, viteConfigEnv) {\n      configEnv = viteConfigEnv;\n    },\n    applyToEnvironment(partialEnvironment) {\n      // Client static plugin should apply to static environment (browser/ESM builds)\n      // This is where we want to bundle everything and filter out _virtual files\n      // Apply to both \"static\" and \"client\" environments - we'll handle which one runs static generation in closeBundle\n      const envName = partialEnvironment.name as \"client\" | \"server\" | \"ssr\" | \"static\";\n      if (\n        [\"static\", \"client\"].includes(envName)\n      ) {\n        return true;\n      }\n      return false;\n    },\n\n    async configResolved(config) {\n      timing.configResolved = performance.now();\n      logger = config.customLogger || createLogger();\n      resolvedConfig = config;\n\n      // Perform auto-discovery to populate autoDiscoveredFiles\n      const autoDiscoverResult = await resolveAutoDiscover({\n        config: config,\n        configEnv: configEnv || {\n          mode: config.mode,\n          command: config.command,\n          isSsrBuild: false,\n          isPreview: false,\n        },\n        userOptions,\n        logger,\n      });\n      if (autoDiscoverResult.type === \"error\") {\n        throw autoDiscoverResult.error;\n      }\n      autoDiscoveredFiles = autoDiscoverResult.autoDiscoveredFiles;\n      if(userOptions.verbose) {\n        logger?.info(`Auto-discovery ${autoDiscoverResult.type === \"success\" ? \"completed\" : \"skipped\"}`);\n      }\n    },\n\n    async buildStart() {\n      timing.buildStart = performance.now();\n      if(userOptions.verbose) {\n        logger?.info(\"[react-static-client] Build started\");\n      }\n\n      if (userOptions.onEvent && autoDiscoveredFiles) {\n        try {\n          userOptions.onEvent({\n            type: \"build.start\",\n            data: {\n              pages: Array.from(autoDiscoveredFiles.urlMap.keys()),\n              files: autoDiscoveredFiles,\n            },\n          });\n        } catch (error) {\n          const panicError = handleError({\n            error,\n            logger: logger,\n            panicThreshold: userOptions.panicThreshold,\n            context: \"buildStart\",\n          });\n          if (panicError != null) {\n            rscWorker?.terminate();\n          throw panicError;\n          }\n        }\n      }\n    },\n\n    async renderStart() {\n      timing.renderStart = performance.now();\n      if(userOptions.verbose) { \n        logger?.info(\"[react-static-client] Render started\");\n      }\n    },\n\n    // the preview server helps to view the generated static folder, but only when the static plugin is enabled\n    // if no build.pages, then the preview server will instead use default vite preview server\n    // it works the same under both conditions\n    async configurePreviewServer(server) {\n      logger = server.config.customLogger || server.config.logger;\n      configurePreviewServer({\n        server,\n        userOptions,\n      });\n    },\n\n\n\n    async writeBundle(_options, bundle) {\n\n      // Capture manifests from all environments\n      try {\n        if (!autoDiscoveredFiles?.urlMap) {\n          return;\n        }\n\n        const bundleManifest = getBundleManifest<false>({\n          bundle,\n          normalizer: userOptions.normalizer,\n        });\n\n        // Store manifest based on environment\n        if (this.environment.name === \"static\") {\n          // Store in global manifest store for environment plugin access\n          addStaticManifest(bundleManifest);\n\n          staticBundle = bundle;\n        } else if (this.environment.name === \"client\") {\n          // Client build manifest (SSR modules) - stored globally now\n\n          if (manifests.static) {\n            const staticManifest = manifests.static;\n\n            // Update bundle filenames to match static manifest\n            for (const [, chunk] of Object.entries(bundle)) {\n              if (chunk.type === \"chunk\" && chunk.fileName) {\n                const normalized = userOptions.normalizer(chunk.fileName);\n                let value = normalized[1];\n                if (value.startsWith(userOptions.moduleBasePath)) {\n                  value = value.slice(userOptions.moduleBasePath.length);\n                }\n\n                const entry = staticManifest[value];\n                if (entry && entry.file !== chunk.fileName) {\n                  // Update the filename to match static manifest\n                  chunk.fileName = entry.file;\n                }\n              }\n            }\n          }\n        } else if (this.environment.name === \"server\") {\n          // Server build manifest (server components) - stored globally now\n          serverBundle = bundle;\n        }\n\n        // Skip the static generation here - it will happen in closeBundle\n        return;\n      } catch (error) {\n        const panicError = handleError({\n          error,\n          logger: logger,\n          panicThreshold: userOptions.panicThreshold,\n          context: \"writeBundle\",\n        });\n        if (panicError != null) {\n          throw panicError;\n        }\n      }\n    },\n\n    async closeBundle() {\n      const envName = this.environment.name;\n      const isSsr = this.environment.config.build?.ssr === true;\n      \n      if (userOptions.verbose) {\n        logger?.info(`[react-static-client] closeBundle called for environment: ${envName}, ssr: ${isSsr}`);\n      }\n      \n      // Only run static generation in the non-SSR client environment (static builds)\n      // Skip SSR client builds and server builds\n      if (envName === \"ssr\" || envName === \"server\" || isSsr) {\n        if (userOptions.verbose) {\n          logger?.info(`[react-static-client] Skipping static generation for environment: ${envName} (ssr: ${isSsr})`);\n        }\n        return;\n      }\n\n      // Clean up _virtual files after build completes\n      // These are Vite's internal virtual modules and aren't needed in the final output\n      if (envName === \"static\" || (envName === \"client\" && !isSsr)) {\n        try {\n          const { rmSync, existsSync } = await import(\"node:fs\");\n          const { join, resolve } = await import(\"node:path\");\n          \n          // Use the resolved output directory from the environment config\n          const resolvedOutDir = this.environment.config.build?.outDir \n            ? resolve(this.environment.config.root || userOptions.projectRoot, this.environment.config.build.outDir)\n            : resolve(userOptions.projectRoot, userOptions.build.outDir);\n          \n          // Clean up _virtual from client/static output directories only\n          // Don't clean up server/_virtual since we need dynamic-import-helper.js there\n          const outputDirs = [\n            join(resolvedOutDir, userOptions.build.static || \"static\"),\n            join(resolvedOutDir, userOptions.build.client || \"client\"),\n          ];\n          \n          for (const outDir of outputDirs) {\n            const virtualDir = join(outDir, \"_virtual\");\n            if (existsSync(virtualDir)) {\n              rmSync(virtualDir, { recursive: true, force: true });\n              if (userOptions.verbose) {\n                logger?.info(`[react-static-client] Cleaned up _virtual directory: ${virtualDir}`);\n              }\n            }\n          }\n        } catch (error) {\n          // Non-critical - log but don't fail the build\n          if (userOptions.verbose) {\n            logger?.warn(`[react-static-client] Failed to clean up _virtual directory: ${error}`);\n          }\n        }\n      }\n\n      // This runs after all writeBundle hooks are complete\n      // Run static generation in the non-SSR client environment (static builds)\n      // This could be \"static\" or \"client\" depending on how environments are configured\n      if (envName === \"ssr\" || envName === \"server\" || isSsr) {\n        if (userOptions.verbose) {\n          logger?.info(`[react-static-client] Skipping static generation - not in static environment (${envName}, ssr: ${isSsr})`);\n        }\n        return;\n      }\n\n      // Defer static generation to run after ALL environments complete their builds.\n      // This is necessary because we need the server manifest (from server env's writeBundle)\n      // to resolve function-based component paths like Root: (url) => 'src/CustomRoot.tsx'.\n      // The buildApp hook in createEnvironmentPlugin will call runDeferredStaticGeneration().\n      const closeBundleContext = this;\n      deferStaticGeneration(async () => {\n\n      try {\n        // Re-check autoDiscoveredFiles - it might not be set if configResolved didn't run\n        // or if it was cleared. Try to get it from stashed options if needed\n        if (!autoDiscoveredFiles) {\n          if (userOptions.verbose) {\n            logger?.warn(\"[react-static-client] autoDiscoveredFiles not set, attempting to re-discover\");\n          }\n          const { getStashedUserOptions, getEnvironmentId } = await import(\"../config/stashedOptionsState.js\");\n          const { getCondition } = await import(\"../config/getCondition.js\");\n          const envId = getEnvironmentId(getCondition(), resolvedConfig?.mode || \"production\");\n          const stashedOptions = getStashedUserOptions(envId);\n          if (stashedOptions && resolvedConfig) {\n            // Try to re-run auto-discovery if we have the config\n            const autoDiscoverResult = await resolveAutoDiscover({\n              config: resolvedConfig,\n              configEnv: configEnv || {\n                mode: resolvedConfig.mode || \"production\",\n                command: resolvedConfig.command || \"build\",\n                isSsrBuild: false,\n                isPreview: false,\n              },\n              userOptions,\n              logger,\n            });\n            if (autoDiscoverResult.type === \"success\") {\n              autoDiscoveredFiles = autoDiscoverResult.autoDiscoveredFiles;\n              if (userOptions.verbose) {\n                logger?.info(`[react-static-client] Re-discovered ${autoDiscoveredFiles.urlMap.size} pages`);\n              }\n            } else {\n              if (userOptions.verbose) {\n                logger?.warn(`[react-static-client] Failed to re-discover pages: ${autoDiscoverResult.error}`);\n              }\n            }\n          }\n        }\n\n        if (\n          !autoDiscoveredFiles?.urlMap ||\n          autoDiscoveredFiles?.urlMap.size === 0\n        ) {\n          if (userOptions.verbose) {\n            logger?.warn(`[react-static-client] No pages to generate - urlMap is empty (size: ${autoDiscoveredFiles?.urlMap?.size || 0})`);\n            logger?.warn(`[react-static-client] autoDiscoveredFiles exists: ${!!autoDiscoveredFiles}, urlMap exists: ${!!autoDiscoveredFiles?.urlMap}`);\n          }\n          return;\n        }\n\n        if (userOptions.verbose) {\n          logger?.info(`[react-static-client] Starting static generation with ${autoDiscoveredFiles.urlMap.size} pages`);\n        }\n\n        // Check if we can access the shared manifest store\n        try {\n          if (userOptions.verbose) {\n            logger?.info(`[react-static-client] Attempting to get server manifest from shared state`);\n          }\n          const sharedState = getSharedManifestStore(closeBundleContext);\n          if (sharedState.server) {\n            serverManifest = sharedState.server;\n            if (userOptions.verbose) {\n              logger?.info(`[react-static-client] Got server manifest from shared state`);\n            }\n          } else {\n            throw new Error(\"No server manifest in shared state\");\n          }\n        } catch (error) {\n          if (userOptions.verbose) {\n            logger?.info(`[react-static-client] Failed to get server manifest from shared state, trying filesystem: ${error}`);\n          }\n          const serverManifestPath = join(\n            userOptions.build.outDir,\n            userOptions.build.server\n          );\n          const manifestPath =\n            (typeof resolvedConfig?.build.manifest === \"string\" \n              ? resolvedConfig.build.manifest \n              : \".vite/manifest.json\");\n\n          if (userOptions.verbose) {\n            logger?.info(`[react-static-client] Loading server manifest from: ${join(serverManifestPath, manifestPath)}`);\n          }\n\n          const serverManifestResult = await tryManifest({\n            root: userOptions.projectRoot,\n            outDir: serverManifestPath,\n            manifestPath: manifestPath,\n            ssrManifest: false,\n          });\n\n          if (serverManifestResult.type === \"error\") {\n            if (userOptions.verbose) {\n              logger?.warn(`[react-static-client] Failed to load server manifest: ${serverManifestResult.error}`);\n            }\n            // Use empty manifest as fallback - static generation can proceed without it\n            serverManifest = {};\n            if (userOptions.verbose) {\n              logger?.warn(`[react-static-client] Using empty server manifest as fallback`);\n            }\n          } else if (serverManifestResult.type === \"skip\") {\n            if (userOptions.verbose) {\n              logger?.warn(`[react-static-client] Server manifest not found, using empty manifest as fallback`);\n            }\n            // Use empty manifest as fallback - static generation can proceed without it\n            serverManifest = {};\n          } else {\n            serverManifest = serverManifestResult.manifest;\n            if (userOptions.verbose) {\n              logger?.info(`[react-static-client] Loaded server manifest from filesystem`);\n            }\n          }\n        }\n\n        // Load static manifest from filesystem for CSS path mapping\n\n        const staticManifestResult = await tryManifest({\n          root: userOptions.projectRoot,\n          outDir: join(userOptions.build.outDir, userOptions.build.static),\n          manifestPath: resolvedConfig?.build.manifest ?? \".vite/manifest.json\",\n          ssrManifest: false,\n        });\n        if (staticManifestResult.type === \"error\") {\n          throw staticManifestResult.error;\n        }\n        const staticManifest = staticManifestResult.manifest;\n\n        // Construct bootstrapModules like the server plugin does\n        const indexHtml = staticManifest?.[\"index.html\"]?.file;\n        const serverPipeableStreamOptions = {\n          ...userOptions.serverPipeableStreamOptions,\n          bootstrapModules: [\n            ...(indexHtml ? [baseURL(indexHtml)] : []),\n            ...(userOptions.serverPipeableStreamOptions?.bootstrapModules ??\n              []),\n          ],\n        };\n        userOptions.serverPipeableStreamOptions = serverPipeableStreamOptions;\n        const clientPipeableStreamOptions = {\n          ...userOptions.clientPipeableStreamOptions,\n          bootstrapModules: [\n            ...(indexHtml ? [baseURL(indexHtml)] : []),\n            ...(userOptions.clientPipeableStreamOptions?.bootstrapModules ??\n              []),\n          ],\n        };\n        // Create CSS props for each CSS file (same as server-static)\n        const { cssFilesByPage, globalCss } = processCssFilesForPages({\n          userOptions,\n          autoDiscoveredFiles,\n          serverManifest,\n          staticManifest,\n          bundle: staticBundle || {},\n          logger,\n        });\n\n        if (userOptions.verbose) {\n          for (const [route, cssMap] of cssFilesByPage.entries()) {\n            logger.info(\n              `[react-static-client] Route ${route}: ${cssMap.size} CSS files`\n            );\n            for (const [key, value] of cssMap.entries()) {\n              logger.info(\n                `[react-static-client]   CSS file: ${key} -> ${value.as} (${\n                  value.children ? \"inline\" : \"link\"\n                })`\n              );\n            }\n          }\n        }\n\n        const routes = Array.from(\n          autoDiscoveredFiles.urlMap.keys()\n        ) as string[];\n\n        // If no pages to generate, skip static generation\n        if (routes.length === 0) {\n          if (userOptions.verbose) {\n            logger?.info(\n              \"[react-static-client] No pages to generate, skipping static generation\"\n            );\n          }\n          return;\n        }\n\n        // Use the static manifest to ensure consistent module IDs between RSC stream and client build\n        // The static manifest contains the correct hashes that should be used for both builds\n        // (staticManifest already loaded above)\n\n        // Create a build loader for client mode (reuse server's sophisticated loader)\n        if (userOptions.verbose) {\n          logger?.info(`[react-static-client] Creating build loader`);\n        }\n        const buildLoader = createBuildLoader();\n        if (userOptions.verbose) {\n          logger?.info(`[react-static-client] Build loader created`);\n        }\n\n        // Create an RSC worker for generating RSC content\n        if (userOptions.verbose) {\n          logger?.info(\n            `[react-static-client] Creating RSC worker with path: ${userOptions.rscWorkerPath}`\n          );\n        }\n\n        const workerStartTime = performance.now();\n        let rscWorkerResult;\n        try {\n          rscWorkerResult = await createWorker({\n            projectRoot: userOptions.projectRoot,\n            workerPath: userOptions.rscWorkerPath,\n            currentCondition: \"react-client\",\n            reverseCondition: \"react-server\",\n            maxListeners: Math.max(routes.length * 3, 10), // Account for multiple listeners per route\n            envPrefix: envPrefixFromConfig(resolvedConfig as any),\n            logger: logger,\n            verbose: userOptions.verbose,\n            mode: getNodeEnv(),\n            workerData: {\n              userOptions: serializedOptions(userOptions, autoDiscoveredFiles),\n              resolvedConfig: serializeResolvedConfig(resolvedConfig as any),\n              configEnv: (() => {\n                const fallback = resolvedConfig\n                  ? {\n                      command: resolvedConfig.command,\n                      mode: resolvedConfig.mode,\n                      isSsrBuild: false,\n                      isPreview: false,\n                    }\n                  : undefined;\n                const finalConfigEnv = configEnv || fallback;\n\n                return finalConfigEnv;\n              })(),\n              serverManifest: serverManifest || {}, // Use server manifest for page component resolution\n              bundle: staticBundle || {}, // Use static bundle (client build) for page component resolution\n              staticBundle: staticBundle || {}, // Pass static bundle separately for path resolution\n\n              id: \"static-client-rsc-worker\",\n            },\n          });\n        } catch (workerError) {\n          if (userOptions.verbose) {\n            logger?.error(`[react-static-client] Error creating RSC worker: ${workerError}`);\n          }\n          throw workerError;\n        }\n\n        if (rscWorkerResult.type !== \"success\") {\n          const err = rscWorkerResult.error ?? new Error(`Failed to create RSC worker`);\n          if (userOptions.verbose) {\n            logger?.error(\n              `[react-static-client] RSC worker creation failed, throwing error`, { error: err }\n            );\n          }\n          throw err;\n        }\n\n        rscWorker = rscWorkerResult.worker;\n        if (userOptions.verbose) {\n          logger?.info(`[react-static-client] RSC worker created successfully`);\n        }\n\n        // Emit worker startup metric after worker is created\n        const workerStartupTime = performance.now() - workerStartTime;\n        if (userOptions.onMetrics) {\n          const workerStartupMetric = createWorkerStartupMetrics({\n            route: \"/\", // Worker startup is global, not route-specific\n            workerType: \"rsc\", // This is the RSC worker for client-side static generation\n            startupTime: workerStartupTime,\n            fromMainThread: true,\n            fromRscWorker: false,\n            fromHtmlWorker: false,\n            description: `RSC worker startup for client-side static generation`,\n          });\n          userOptions.onMetrics(workerStartupMetric);\n        }\n\n        // Render pages using client-side renderer with RSC worker only\n        const { onEvent, onMetrics, ...handlerOptions } = userOptions;\n\n        if (userOptions.verbose) {\n          logger?.info(`[react-static-client] Extracted onEvent: ${typeof onEvent}, userOptions.onEvent: ${typeof userOptions.onEvent}`);\n        }\n\n        // Capture server bundle from onEvent if not already captured\n        if (!serverBundle && onEvent) {\n          // Create a temporary event handler to capture the server bundle\n          const originalOnEvent = onEvent;\n          const tempOnEvent = (event: any) => {\n            if (event.type === \"build.writeBundle.server\") {\n              serverBundle = event.data.bundle;\n              logger?.info(\n                \"[react-static-client] Captured server bundle from build event\"\n              );\n            }\n            // Call the original event handler\n            originalOnEvent(event);\n          };\n\n          // Replace the onEvent temporarily to capture the server bundle\n          userOptions.onEvent = tempOnEvent;\n        }\n\n        // Emit the static site generation start event\n        // Use the extracted onEvent if available, otherwise fall back to userOptions.onEvent\n        const eventHandler = onEvent || userOptions.onEvent;\n        if (typeof eventHandler === \"function\") {\n          try {\n            if (userOptions.verbose) {\n              logger?.info(`[react-static-client] Emitting build.ssg.start event`);\n            }\n            const r = eventHandler({\n              type: \"build.ssg.start\",\n              data: {\n                pages: Array.from(autoDiscoveredFiles?.urlMap.keys() ?? []),\n                options: null as any, // No specific rollup output options for static generation\n                bundle: staticBundle || {},\n              },\n            });\n            if (r != null && typeof r === \"object\" && \"then\" in r) {\n              await (r as Promise<any>);\n            }\n          } catch (error) {\n            const eventPanicError = handleError({\n              error,\n              logger: logger,\n              panicThreshold: userOptions.panicThreshold,\n              context: \"onEvent(build.ssg.start)\",\n            });\n            if (eventPanicError != null) {\n              throw eventPanicError; // Re-throw to abort the build\n            }\n          }\n        } else if (userOptions.verbose) {\n          logger?.warn(`[react-static-client] No onEvent handler available to emit build.ssg.start`);\n        }\n\n        const renderPagesGenerator = renderPagesBatched(\n          routes,\n          {\n            ...handlerOptions, // Use the clean options instead of the original handlerOptions\n            worker: rscWorker, // Pass the RSC worker for RSC rendering only\n            rscWorker: rscWorker, // Pass the RSC worker for RSC rendering only\n            loader: buildLoader, // Use proper build loader instead of no-op\n            logger: logger,\n            autoDiscoveredFiles: autoDiscoveredFiles,\n            cssFilesByPage: cssFilesByPage, // Pass CSS files by page\n            serverPipeableStreamOptions: serverPipeableStreamOptions, // Pass server options to RSC worker\n            clientPipeableStreamOptions: clientPipeableStreamOptions, // Pass client options to RSC worker\n            globalCss: globalCss, // Pass global CSS\n            manifest: serverManifest || {}, // Server manifest for RSC worker\n            staticManifest: staticManifest, // Static manifest for consistent module IDs\n            onEvent: onEvent,\n            onMetrics: onMetrics, // Pass through the onMetrics callback (metric watcher)\n          },\n          renderPage\n        );\n\n        // Process the rendered pages\n        let finalResult: any = undefined;\n        try {\n          for await (const result of renderPagesGenerator) {\n            if (result.type === \"error\") {\n              if (userOptions.verbose) {\n                logger?.error(`[react-static-client] Render error: ${result.error}`);\n              }\n              throw result.error;\n            }\n\n          // Handle failed routes based on panic threshold\n          if (\n            result.type === \"success\" &&\n            result.failedRoutes &&\n            result.failedRoutes.size > 0\n          ) {\n            // Use centralized panic threshold logic (same as server plugin)\n            const firstError = result.failedRoutes.values().next().value;\n            if (\n              firstError != null &&\n              shouldCausePanic(firstError, {\n                panicThreshold: userOptions.panicThreshold,\n              })\n            ) {\n              // This should cause a panic, throw the error\n              throw firstError;\n            }\n            // For other panic thresholds, log warnings but continue\n            for (const [route, error] of result.failedRoutes) {\n              const err = error instanceof Error ? error : toError(error);\n              closeBundleContext.warn(\n                new Error(\n                  \"Failed to render route: \" +\n                    route +\n                    \"\\n\" +\n                    err.message +\n                    \"\\n\" +\n                    err.stack,\n                  { cause: err }\n                )\n              );\n            }\n          }\n\n            finalResult = result;\n          }\n        } catch (renderError) {\n          if (userOptions.verbose) {\n            logger?.error(`[react-static-client] Error during renderPages: ${renderError}`);\n          }\n          throw renderError;\n        }\n\n        if (!finalResult) {\n          const errorMsg = \"No render result produced\";\n          if (userOptions.verbose) {\n            logger?.error(`[react-static-client] ${errorMsg}`);\n          }\n          throw new Error(errorMsg);\n        }\n\n        if (userOptions.verbose) {\n          logger?.info(`[react-static-client] Render completed: ${finalResult.completedRoutes.size} pages, ${finalResult.failedRoutes?.size || 0} failed`);\n        }\n\n        // File writes are handled by renderPages, no need to do them here\n\n        // Calculate duration from timing\n        const duration = Math.round(\n          performance.now() - (timing.renderStart || timing.start)\n        );\n\n        closeBundleContext.info(\n          `Rendered ${finalResult.completedRoutes.size} pages in ${duration}ms`\n        );\n\n        if (process.env[\"NODE_ENV\"] !== \"production\") {\n          closeBundleContext.warn(\n            `THIS BUILD IS NOT INTENDED FOR PRODUCTION (${process.env[\"NODE_ENV\"]})`\n          );\n        }\n\n        // Update timing\n        timing.render =\n          performance.now() - (timing.renderStart ?? timing.start);\n\n        if (userOptions.verbose) {\n          logger?.info(\"[react-static-client] Static generation completed\");\n        }\n\n        // Emit the static site generation completion event once\n        if (typeof userOptions.onEvent === \"function\") {\n          try {\n            const r = userOptions.onEvent({\n              type: \"build.ssg.end\",\n              data: {\n                pages: Array.from(autoDiscoveredFiles?.urlMap.keys() ?? []),\n                options: null as any, // No specific rollup output options for static generation\n                bundle: staticBundle || {},\n              },\n            });\n            if (r != null && typeof r === \"object\" && \"then\" in r) {\n              await (r as Promise<any>);\n            }\n          } catch (error) {\n            const eventPanicError = handleError({\n              error,\n              logger: logger,\n              panicThreshold: userOptions.panicThreshold,\n              context: \"onEvent(build.ssg.end)\",\n            });\n            if (eventPanicError != null) {\n              throw eventPanicError; // Re-throw to abort the build\n            }\n          }\n        }\n      } catch (error) {\n        const panicError = handleError({\n          error,\n          context: \"react-static-client\",\n          logger,\n          panicThreshold: userOptions.panicThreshold,\n        });\n\n        // Ensure graceful shutdown on error\n        if (rscWorker) {\n          const workerToCleanup = rscWorker;\n          try {\n            // Use graceful shutdown protocol even on error\n            await Promise.race([\n              new Promise<void>((resolve) => {\n                const timeoutId = setTimeout(() => {\n                  workerToCleanup.removeAllListeners();\n                  workerToCleanup.terminate();\n                  resolve();\n                }, 1000); // 1 second timeout for graceful shutdown\n\n                const messageHandler = (message: any) => {\n                  if (message.type === \"SHUTDOWN_COMPLETE\") {\n                    clearTimeout(timeoutId);\n                    workerToCleanup.removeListener(\"message\", messageHandler);\n                    resolve();\n                  }\n                };\n                workerToCleanup.on(\"message\", messageHandler);\n                workerToCleanup.postMessage({ type: \"SHUTDOWN\" });\n              }),\n            ]);\n            rscWorker = undefined;\n          } catch (cleanupError) {\n            logger.warn(`Failed to cleanup worker on error: ${cleanupError}`);\n            // Force terminate if graceful shutdown fails\n            try {\n              workerToCleanup.removeAllListeners();\n              workerToCleanup.terminate();\n            } catch (terminateError) {\n              // Ignore termination errors\n            }\n            rscWorker = undefined;\n          }\n        }\n\n        if (panicError != null) {\n          // Ensure we have a proper Error object that can have properties set on it\n          const errorToThrow =\n            panicError instanceof Error\n              ? panicError\n              : new Error(String(panicError));\n\n          // Create a new Error object to avoid the \"code\" property issue\n          const finalError = new Error(errorToThrow.message);\n          finalError.stack = errorToThrow.stack;\n          finalError.cause = errorToThrow.cause;\n\n          // Copy any additional properties that might be needed\n          if (errorToThrow.name) finalError.name = errorToThrow.name;\n\n        throw finalError;\n        }\n      } finally {\n        // Graceful worker shutdown — runs on both success and error paths\n        if (rscWorker) {\n          try {\n            await Promise.race([\n              new Promise<void>((resolve, reject) => {\n                const timeout = setTimeout(() => {\n                  reject(new Error(\"Worker shutdown timeout\"));\n                }, userOptions.workerShutdownTimeout);\n\n                const backupTimeout = setTimeout(() => {\n                  reject(new Error(\"Worker shutdown backup timeout\"));\n                }, Math.floor(userOptions.workerShutdownTimeout * 0.6));\n\n                const shutdownMessageHandler = (message: any) => {\n                  if (message.type === \"SHUTDOWN_COMPLETE\") {\n                    clearTimeout(timeout);\n                    clearTimeout(backupTimeout);\n                    rscWorker?.removeListener(\n                      \"message\",\n                      shutdownMessageHandler\n                    );\n                    rscWorker?.removeAllListeners();\n                    resolve();\n                  }\n                };\n\n                rscWorker?.on(\"message\", shutdownMessageHandler);\n                rscWorker?.postMessage({\n                  type: \"SHUTDOWN\",\n                  id: \"*\",\n                });\n              }),\n            ]);\n          } catch {\n            // Shutdown protocol failed — force terminate below\n          } finally {\n            if (rscWorker) {\n              try {\n                (rscWorker as Worker).removeAllListeners();\n                (rscWorker as Worker).terminate();\n              } catch {\n                // Ignore termination errors\n              }\n              rscWorker = undefined;\n            }\n          }\n        }\n\n        // Reset any cached state to prevent issues in subsequent builds\n        autoDiscoveredFiles = null;\n        serverManifest = undefined;\n      }\n\n      }); // end deferStaticGeneration\n    },\n  };\n};\n"],"names":["join"],"mappings":"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"}