vite-plugin-react-server 1.4.1 → 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 (146) 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/configureReactServer.server.d.ts.map +1 -1
  11. package/dist/plugin/dev-server/configureReactServer.server.js +5 -8
  12. package/dist/plugin/dev-server/plugin.client.d.ts.map +1 -1
  13. package/dist/plugin/dev-server/plugin.client.js +2 -1
  14. package/dist/plugin/dev-server/plugin.server.d.ts.map +1 -1
  15. package/dist/plugin/dev-server/plugin.server.js +2 -36
  16. package/dist/plugin/dev-server/virtualRscHmrPlugin.js +23 -23
  17. package/dist/plugin/environments/createBuildEventPlugin.js +88 -98
  18. package/dist/plugin/environments/createEnvironmentPlugin.js +222 -250
  19. package/dist/plugin/error/index.d.ts +1 -2
  20. package/dist/plugin/error/index.d.ts.map +1 -1
  21. package/dist/plugin/error/index.js +2 -3
  22. package/dist/plugin/error/panicThresholdHandler.js +14 -3
  23. package/dist/plugin/error/setupGlobalErrorHandler.d.ts.map +1 -1
  24. package/dist/plugin/error/setupGlobalErrorHandler.js +23 -16
  25. package/dist/plugin/helpers/createRscRenderHelpers.d.ts +0 -5
  26. package/dist/plugin/helpers/createRscRenderHelpers.d.ts.map +1 -1
  27. package/dist/plugin/helpers/createRscRenderHelpers.js +32 -55
  28. package/dist/plugin/helpers/createSharedLoader.d.ts.map +1 -1
  29. package/dist/plugin/helpers/createSharedLoader.js +4 -2
  30. package/dist/plugin/helpers/headlessStreamReuseHandler.js +30 -22
  31. package/dist/plugin/helpers/headlessStreamState.d.ts +0 -38
  32. package/dist/plugin/helpers/headlessStreamState.d.ts.map +1 -1
  33. package/dist/plugin/helpers/headlessStreamState.js +15 -76
  34. package/dist/plugin/helpers/index.d.ts +0 -3
  35. package/dist/plugin/helpers/index.d.ts.map +1 -1
  36. package/dist/plugin/helpers/index.js +1 -4
  37. package/dist/plugin/helpers/requestInfo.d.ts.map +1 -1
  38. package/dist/plugin/helpers/requestInfo.js +3 -3
  39. package/dist/plugin/helpers/resolveComponent.d.ts.map +1 -1
  40. package/dist/plugin/helpers/resolveComponent.js +4 -2
  41. package/dist/plugin/helpers/workerCleanup.d.ts +1 -12
  42. package/dist/plugin/helpers/workerCleanup.d.ts.map +1 -1
  43. package/dist/plugin/helpers/workerCleanup.js +1 -1
  44. package/dist/plugin/index.client.d.ts +5 -0
  45. package/dist/plugin/index.client.d.ts.map +1 -0
  46. package/dist/plugin/index.client.js +4 -0
  47. package/dist/plugin/index.d.ts +4 -3
  48. package/dist/plugin/index.d.ts.map +1 -1
  49. package/dist/plugin/index.js +10 -5
  50. package/dist/plugin/index.server.d.ts +5 -0
  51. package/dist/plugin/index.server.d.ts.map +1 -0
  52. package/dist/plugin/index.server.js +4 -0
  53. package/dist/plugin/loader/directives/index.d.ts +0 -1
  54. package/dist/plugin/loader/directives/index.d.ts.map +1 -1
  55. package/dist/plugin/loader/directives/index.js +1 -2
  56. package/dist/plugin/metrics/createWorkerStartupMetrics.js +31 -13
  57. package/dist/plugin/orchestrator/createPluginOrchestrator.client.js +41 -38
  58. package/dist/plugin/orchestrator/createPluginOrchestrator.server.js +43 -46
  59. package/dist/plugin/plugin.client.js +2 -2
  60. package/dist/plugin/plugin.server.js +2 -2
  61. package/dist/plugin/react-static/createBuildLoader.client.js +12 -6
  62. package/dist/plugin/react-static/createBuildLoader.server.js +255 -235
  63. package/dist/plugin/react-static/plugin.client.js +684 -770
  64. package/dist/plugin/react-static/plugin.server.js +517 -603
  65. package/dist/plugin/react-static/processCssFilesForPages.js +103 -88
  66. package/dist/plugin/react-static/renderPage.client.js +455 -529
  67. package/dist/plugin/react-static/renderPage.server.js +485 -508
  68. package/dist/plugin/react-static/renderPagesBatched.js +277 -275
  69. package/dist/plugin/react-static/rscToHtmlStream.client.js +48 -29
  70. package/dist/plugin/react-static/rscToHtmlStream.server.js +62 -37
  71. package/dist/plugin/react-static/temporaryReferences.server.js +11 -2
  72. package/dist/plugin/stream/createMainThreadHandlers.js +40 -31
  73. package/dist/plugin/stream/renderRscStream.server.d.ts.map +1 -1
  74. package/dist/plugin/stream/renderRscStream.server.js +127 -144
  75. package/dist/plugin/transformer/createTransformerPlugin.js +226 -265
  76. package/dist/plugin/utils/checkReactVersion.d.ts +7 -0
  77. package/dist/plugin/utils/checkReactVersion.d.ts.map +1 -0
  78. package/dist/plugin/utils/checkReactVersion.js +23 -0
  79. package/dist/plugin/utils/envUrls.node.js +12 -11
  80. package/dist/plugin/vendor/vendor-alias.js +84 -114
  81. package/dist/plugin/vendor/vendor.client.d.ts.map +1 -1
  82. package/dist/plugin/vendor/vendor.client.js +1 -3
  83. package/dist/plugin/worker/rsc/handleRscRender.d.ts.map +1 -1
  84. package/dist/plugin/worker/rsc/handleRscRender.js +3 -1
  85. package/dist/tsconfig.tsbuildinfo +1 -1
  86. package/package.json +123 -13
  87. package/plugin/config/autoDiscover/resolveAutoDiscover.ts +4 -0
  88. package/plugin/config/getCondition.ts +6 -4
  89. package/plugin/dev-server/configureReactServer.server.ts +7 -10
  90. package/plugin/dev-server/plugin.client.ts +2 -0
  91. package/plugin/dev-server/plugin.server.ts +2 -49
  92. package/plugin/error/index.ts +1 -2
  93. package/plugin/error/setupGlobalErrorHandler.ts +24 -25
  94. package/plugin/helpers/createRscRenderHelpers.ts +0 -29
  95. package/plugin/helpers/createSharedLoader.ts +6 -1
  96. package/plugin/helpers/headlessStreamState.ts +0 -69
  97. package/plugin/helpers/index.ts +0 -3
  98. package/plugin/helpers/requestInfo.ts +1 -2
  99. package/plugin/helpers/resolveComponent.ts +6 -1
  100. package/plugin/helpers/workerCleanup.ts +1 -38
  101. package/plugin/index.client.ts +4 -0
  102. package/plugin/index.server.ts +4 -0
  103. package/plugin/index.ts +12 -5
  104. package/plugin/loader/directives/index.ts +0 -1
  105. package/plugin/plugin.client.ts +1 -1
  106. package/plugin/plugin.server.ts +1 -1
  107. package/plugin/stream/renderRscStream.server.ts +3 -0
  108. package/plugin/transformer/README.md +1 -1
  109. package/plugin/utils/checkReactVersion.ts +28 -0
  110. package/plugin/vendor/vendor.client.ts +0 -2
  111. package/plugin/worker/html/README.md +1 -1
  112. package/plugin/worker/rsc/README.md +1 -1
  113. package/plugin/worker/rsc/handleRscRender.ts +2 -0
  114. package/scripts/generate-toc.mjs +27 -294
  115. package/dist/plugin/error/assertPanic.d.ts +0 -2
  116. package/dist/plugin/error/assertPanic.d.ts.map +0 -1
  117. package/dist/plugin/error/assertPanic.js +0 -15
  118. package/dist/plugin/error/directiveError.d.ts +0 -13
  119. package/dist/plugin/error/directiveError.d.ts.map +0 -1
  120. package/dist/plugin/error/directiveError.js +0 -21
  121. package/dist/plugin/error/enhanceError.d.ts +0 -14
  122. package/dist/plugin/error/enhanceError.d.ts.map +0 -1
  123. package/dist/plugin/error/enhanceError.js +0 -24
  124. package/dist/plugin/helpers/createSafePageComponent.d.ts +0 -36
  125. package/dist/plugin/helpers/createSafePageComponent.d.ts.map +0 -1
  126. package/dist/plugin/helpers/createSafePageComponent.js +0 -50
  127. package/dist/plugin/helpers/moduleResolver.d.ts +0 -25
  128. package/dist/plugin/helpers/moduleResolver.d.ts.map +0 -1
  129. package/dist/plugin/helpers/moduleResolver.js +0 -64
  130. package/dist/plugin/helpers/stashReturnValue.d.ts +0 -3
  131. package/dist/plugin/helpers/stashReturnValue.d.ts.map +0 -1
  132. package/dist/plugin/helpers/stashReturnValue.js +0 -23
  133. package/dist/plugin/helpers/workerManager.d.ts +0 -5
  134. package/dist/plugin/helpers/workerManager.d.ts.map +0 -1
  135. package/dist/plugin/helpers/workerManager.js +0 -18
  136. package/dist/plugin/loader/directives/collectExportsFromModule.d.ts +0 -6
  137. package/dist/plugin/loader/directives/collectExportsFromModule.d.ts.map +0 -1
  138. package/dist/plugin/loader/directives/collectExportsFromModule.js +0 -24
  139. package/plugin/error/assertPanic.ts +0 -9
  140. package/plugin/error/directiveError.ts +0 -29
  141. package/plugin/error/enhanceError.ts +0 -41
  142. package/plugin/helpers/createSafePageComponent.ts +0 -64
  143. package/plugin/helpers/moduleResolver.ts +0 -91
  144. package/plugin/helpers/stashReturnValue.ts +0 -19
  145. package/plugin/helpers/workerManager.ts +0 -16
  146. package/plugin/loader/directives/collectExportsFromModule.ts +0 -25
@@ -1,623 +1,537 @@
1
1
  /**
2
- * plugin.ts
3
- *
4
- * PURPOSE: Main Vite plugin for React Server Components (RSC) static site generation
5
- *
6
- * This module:
7
- * 1. Orchestrates the entire static site generation process
8
- * 2. Manages the lifecycle of the RSC rendering process
9
- * 3. Handles file writing for both initial page loads and client-side navigation
10
- * - Writes .html files for initial page loads (complete HTML document)
11
- * - Writes .rsc files for client-side navigation (RSC content only)
12
- * 4. Provides hooks for Vite to integrate with the build process
13
- * 5. Manages worker threads for parallel rendering
14
- * 6. Handles error reporting and metrics collection
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 { createBuildLoader } from "./createBuildLoader.server.js";
19
- import { renderPagesBatched } from "./renderPagesBatched.js";
20
- import { renderPages as renderPagesSequential } from "./renderPages.js";
21
- import { getBundleManifest } from "../helpers/getBundleManifest.js";
22
- import { createWorker } from "../worker/createWorker.js";
23
- import { serializedOptions, serializeResolvedConfig, } from "../helpers/serializeUserOptions.js";
24
- import { performance } from "node:perf_hooks";
25
- import { baseURL } from "../utils/envUrls.node.js";
26
- import { handleError } from "../error/handleError.js";
27
- import { shouldCausePanic } from "../error/panicThresholdHandler.js";
28
- import { renderPage } from "./renderPage.server.js";
29
- import { temporaryReferences } from "./temporaryReferences.server.js";
30
- import { configurePreviewServer } from "./configurePreviewServer.js";
31
- import { envPrefixFromConfig } from "../config/envPrefixFromConfig.js";
32
- import { processCssFilesForPages } from "./processCssFilesForPages.js";
33
- import { createWorkerStartupMetrics } from "../metrics/createWorkerStartupMetrics.js";
34
- import { tryManifest } from "../helpers/tryManifest.js";
35
- import { join } from "node:path";
36
- import { resolveAutoDiscover } from "../config/autoDiscover/resolveAutoDiscover.js";
37
- import { assertReactServer } from "../config/getCondition.js";
38
- import { toError } from "../error/toError.js";
6
+ import { createLogger } from 'vite';
7
+ import { resolveOptions } from '../config/resolveOptions.js';
8
+ import { createBuildLoader } from './createBuildLoader.server.js';
9
+ import { renderPagesBatched } from './renderPagesBatched.js';
10
+ import { renderPages } from './renderPages.js';
11
+ import { getBundleManifest } from '../helpers/getBundleManifest.js';
12
+ import { createWorker } from '../worker/createWorker.js';
13
+ import { serializedOptions, serializeResolvedConfig } from '../helpers/serializeUserOptions.js';
14
+ import { performance } from 'node:perf_hooks';
15
+ import { baseURL } from '../utils/envUrls.node.js';
16
+ import { handleError } from '../error/handleError.js';
17
+ import { shouldCausePanic } from '../error/panicThresholdHandler.js';
18
+ import { renderPage } from './renderPage.server.js';
19
+ import { temporaryReferences } from './temporaryReferences.server.js';
20
+ import { configurePreviewServer } from './configurePreviewServer.js';
21
+ import { envPrefixFromConfig } from '../config/envPrefixFromConfig.js';
22
+ import { processCssFilesForPages } from './processCssFilesForPages.js';
23
+ import { createWorkerStartupMetrics } from '../metrics/createWorkerStartupMetrics.js';
24
+ import { tryManifest } from '../helpers/tryManifest.js';
25
+ import { join } from 'node:path';
26
+ import { resolveAutoDiscover } from '../config/autoDiscover/resolveAutoDiscover.js';
27
+ import { assertReactServer } from '../config/getCondition.js';
28
+ import { toError } from '../error/toError.js';
29
+
39
30
  assertReactServer();
40
- /**
41
- * Main entrypoint for the static plugin.
42
- *
43
- * This plugin is responsible for:
44
- * 1. Orchestrating the static site generation process
45
- * 2. Handling the lifecycle of the RSC rendering process (main thread)
46
- * 3. Writing .html files for initial page loads (complete HTML document)
47
- * 4. Writing .rsc files for client-side navigation (RSC content only)
48
- * 5. Managing worker threads for parallel rendering (html worker)
49
- * 6. Handling error reporting and metrics collection
50
- */
51
- // Global worker instance to prevent duplicate creation across plugin instances
52
31
  let globalWorker;
53
- export const reactStaticPlugin = function _reactStaticPlugin(options) {
54
- let worker;
55
- let logger;
56
- let resolvedConfig;
57
- let autoDiscoveredFiles = null;
58
- let serverManifest = undefined;
59
- let configEnv;
60
- const timing = {
61
- start: performance.now(),
62
- configResolved: 0,
63
- buildStart: 0,
64
- renderStart: 0,
65
- };
66
- const resolvedOptions = resolveOptions(options);
67
- if (resolvedOptions.type === "error") {
68
- throw resolvedOptions.error;
69
- }
70
- const userOptions = resolvedOptions.userOptions;
71
- return {
72
- name: "vite:plugin-react-server/server-static",
73
- enforce: "post",
74
- api: {
75
- meta: { timing },
76
- },
77
- async config(_config, viteConfigEnv) {
78
- configEnv = viteConfigEnv;
79
- },
80
- applyToEnvironment(partialEnvironment) {
81
- if (["server"].includes(partialEnvironment.name)) {
82
- return true;
32
+ const reactStaticPlugin = function _reactStaticPlugin(options) {
33
+ let worker;
34
+ let logger;
35
+ let resolvedConfig;
36
+ let autoDiscoveredFiles = null;
37
+ let serverManifest = void 0;
38
+ let configEnv;
39
+ const timing = {
40
+ start: performance.now(),
41
+ configResolved: 0,
42
+ buildStart: 0,
43
+ renderStart: 0
44
+ };
45
+ const resolvedOptions = resolveOptions(options);
46
+ if (resolvedOptions.type === "error") {
47
+ throw resolvedOptions.error;
48
+ }
49
+ const userOptions = resolvedOptions.userOptions;
50
+ return {
51
+ name: "vite:plugin-react-server/server-static",
52
+ enforce: "post",
53
+ api: {
54
+ meta: { timing }
55
+ },
56
+ async config(_config, viteConfigEnv) {
57
+ configEnv = viteConfigEnv;
58
+ },
59
+ applyToEnvironment(partialEnvironment) {
60
+ if (["server"].includes(
61
+ partialEnvironment.name
62
+ )) {
63
+ return true;
64
+ }
65
+ return false;
66
+ },
67
+ async configResolved(config) {
68
+ resolvedConfig = config;
69
+ if (!logger) {
70
+ logger = config.customLogger ?? createLogger();
71
+ }
72
+ const autoDiscoverResult = await resolveAutoDiscover({
73
+ config,
74
+ configEnv,
75
+ userOptions,
76
+ logger
77
+ });
78
+ if (autoDiscoverResult.type === "error") {
79
+ throw autoDiscoverResult.error;
80
+ }
81
+ autoDiscoveredFiles = autoDiscoverResult.autoDiscoveredFiles;
82
+ },
83
+ async buildStart() {
84
+ if (!logger) {
85
+ logger = this.environment.logger;
86
+ }
87
+ timing.buildStart = performance.now();
88
+ if (userOptions.onEvent && autoDiscoveredFiles) {
89
+ try {
90
+ userOptions.onEvent({
91
+ type: "build.start",
92
+ data: {
93
+ pages: Array.from(autoDiscoveredFiles.urlMap.keys()),
94
+ files: autoDiscoveredFiles
83
95
  }
84
- return false;
85
- },
86
- async configResolved(config) {
87
- resolvedConfig = config;
88
- if (!logger) {
89
- logger = config.customLogger ?? createLogger();
96
+ });
97
+ } catch (error) {
98
+ const panicError = handleError({
99
+ error,
100
+ logger,
101
+ panicThreshold: userOptions.panicThreshold});
102
+ if (panicError != null) {
103
+ worker?.terminate();
104
+ throw panicError;
105
+ } else {
106
+ this.error(new Error("Failed to emit build.start event"));
107
+ }
108
+ }
109
+ }
110
+ },
111
+ // the preview server helps to view the generated static folder, but only when the static plugin is enabled
112
+ // if no build.pages, then the preview server will instead use default vite preview server
113
+ // it works the same under both conditions
114
+ async configurePreviewServer(server) {
115
+ logger = server.config.customLogger || server.config.logger;
116
+ configurePreviewServer({
117
+ server,
118
+ userOptions
119
+ });
120
+ },
121
+ async renderStart() {
122
+ timing.renderStart = performance.now();
123
+ },
124
+ generateBundle(_options, bundle) {
125
+ if (this.environment.name === "server") {
126
+ const keysToDelete = [];
127
+ for (const [key, chunk] of Object.entries(bundle)) {
128
+ if (chunk.type === "chunk") {
129
+ const isVirtual = chunk.fileName?.includes("_virtual") || key.includes("_virtual") || chunk.facadeModuleId?.includes("_virtual") || chunk.moduleIds?.some((id) => id.includes("_virtual"));
130
+ const isDynamicImportHelper = chunk.fileName?.includes("dynamic-import-helper") || key.includes("dynamic-import-helper");
131
+ if (isVirtual && !isDynamicImportHelper) {
132
+ keysToDelete.push(key);
133
+ if (userOptions.verbose) {
134
+ logger?.info(`[plugin.server] Filtered out virtual file: ${chunk.fileName || key} (moduleId: ${chunk.facadeModuleId || chunk.moduleIds?.[0]})`);
135
+ }
90
136
  }
91
- const autoDiscoverResult = await resolveAutoDiscover({
92
- config: config,
93
- configEnv: configEnv,
94
- userOptions,
95
- logger,
96
- });
97
- if (autoDiscoverResult.type === "error") {
98
- throw autoDiscoverResult.error;
137
+ }
138
+ }
139
+ for (const key of keysToDelete) {
140
+ delete bundle[key];
141
+ }
142
+ }
143
+ },
144
+ async writeBundle(_options, bundle) {
145
+ if (this.environment.name !== "server") {
146
+ if (userOptions.verbose) {
147
+ logger?.info(`[plugin.server] Skipping static generation for environment: ${this.environment.name}`);
148
+ }
149
+ return;
150
+ }
151
+ let panicError = null;
152
+ let bundleManifest = void 0;
153
+ if (!logger) {
154
+ logger = this.environment.logger;
155
+ }
156
+ try {
157
+ bundleManifest = getBundleManifest({
158
+ bundle,
159
+ normalizer: userOptions.normalizer
160
+ });
161
+ const manifestPath = typeof resolvedConfig.build.manifest === "string" ? resolvedConfig.build.manifest : ".vite/manifest.json";
162
+ if (!bundleManifest[manifestPath] || !("source" in bundleManifest[manifestPath])) {
163
+ throw new Error("Server manifest not found");
164
+ }
165
+ serverManifest = JSON.parse(
166
+ bundleManifest[manifestPath].source
167
+ );
168
+ if (!serverManifest) {
169
+ throw new Error("Failed to parse server manifest");
170
+ }
171
+ } catch (error) {
172
+ const panicError2 = handleError({
173
+ error,
174
+ logger,
175
+ panicThreshold: userOptions.panicThreshold});
176
+ if (panicError2 != null) {
177
+ throw panicError2;
178
+ } else {
179
+ throw new Error("Failed to get bundle manifest");
180
+ }
181
+ }
182
+ try {
183
+ const staticManifestResult = await tryManifest({
184
+ root: userOptions.projectRoot,
185
+ outDir: join(userOptions.build.outDir, userOptions.build.static),
186
+ manifestPath: resolvedConfig.build.manifest,
187
+ ssrManifest: false
188
+ });
189
+ if (staticManifestResult.type === "error") {
190
+ throw staticManifestResult.error;
191
+ }
192
+ const staticManifest = staticManifestResult.manifest;
193
+ const buildLoader = createBuildLoader(
194
+ {
195
+ userOptions,
196
+ serverManifest: serverManifest ?? {},
197
+ staticManifest
198
+ },
199
+ bundle,
200
+ temporaryReferences,
201
+ logger
202
+ );
203
+ const { cssFilesByPage, globalCss } = processCssFilesForPages({
204
+ userOptions,
205
+ autoDiscoveredFiles,
206
+ serverManifest,
207
+ staticManifest,
208
+ bundle,
209
+ logger
210
+ });
211
+ if (userOptions.verbose) {
212
+ logger.info(
213
+ `[plugin.server] cssFilesByPage size: ${cssFilesByPage.size}`
214
+ );
215
+ for (const [route, cssMap] of cssFilesByPage.entries()) {
216
+ logger.info(
217
+ `[plugin.server] Route ${route}: ${cssMap.size} CSS files`
218
+ );
219
+ }
220
+ }
221
+ const indexHtml = staticManifest?.["index.html"]?.file;
222
+ const serverPipeableStreamOptions = {
223
+ ...userOptions.serverPipeableStreamOptions,
224
+ bootstrapModules: [
225
+ ...indexHtml ? [baseURL(indexHtml)] : [],
226
+ ...userOptions.serverPipeableStreamOptions?.bootstrapModules ?? []
227
+ ]
228
+ };
229
+ userOptions.serverPipeableStreamOptions = serverPipeableStreamOptions;
230
+ const clientPipeableStreamOptions = {
231
+ ...userOptions.clientPipeableStreamOptions,
232
+ bootstrapModules: [
233
+ ...indexHtml ? [baseURL(indexHtml)] : [],
234
+ ...userOptions.clientPipeableStreamOptions?.bootstrapModules ?? []
235
+ ]
236
+ };
237
+ const routes = !autoDiscoveredFiles ? [] : Array.from(autoDiscoveredFiles.urlMap.keys());
238
+ if (routes.length === 0) {
239
+ logger?.info(
240
+ "[plugin.server] No pages to generate, skipping static generation"
241
+ );
242
+ return;
243
+ }
244
+ const serializedUserOptions = serializedOptions(
245
+ userOptions,
246
+ autoDiscoveredFiles
247
+ );
248
+ if (globalWorker) {
249
+ logger?.warn("[plugin.server] Global worker already exists, reusing existing worker");
250
+ worker = globalWorker;
251
+ } else {
252
+ const workerStartTime = performance.now();
253
+ const viteEnvPrefix = envPrefixFromConfig(resolvedConfig);
254
+ const routeCount = autoDiscoveredFiles?.urlMap.size ?? 0;
255
+ const maxListeners = routeCount + 1;
256
+ const workerResult = await createWorker({
257
+ projectRoot: userOptions.projectRoot,
258
+ workerPath: userOptions.htmlWorkerPath,
259
+ currentCondition: "react-server",
260
+ reverseCondition: "react-client",
261
+ // HTML worker needs react-client for react-dom/server
262
+ maxListeners,
263
+ envPrefix: viteEnvPrefix,
264
+ logger,
265
+ workerData: {
266
+ resolvedConfig: serializeResolvedConfig(resolvedConfig),
267
+ userOptions: serializedUserOptions,
268
+ configEnv
99
269
  }
100
- autoDiscoveredFiles = autoDiscoverResult.autoDiscoveredFiles;
101
- },
102
- async buildStart() {
103
- if (!logger) {
104
- logger = this.environment.logger;
270
+ });
271
+ if (workerResult.type === "error") {
272
+ if (workerResult.error != null) {
273
+ throw workerResult.error;
105
274
  }
106
- timing.buildStart = performance.now();
107
- if (userOptions.onEvent && autoDiscoveredFiles) {
108
- try {
109
- userOptions.onEvent({
110
- type: "build.start",
111
- data: {
112
- pages: Array.from(autoDiscoveredFiles.urlMap.keys()),
113
- files: autoDiscoveredFiles,
114
- },
115
- });
116
- }
117
- catch (error) {
118
- const panicError = handleError({
119
- error,
120
- logger: logger,
121
- panicThreshold: userOptions.panicThreshold,
122
- context: "buildStart",
123
- });
124
- if (panicError != null) {
125
- worker?.terminate();
126
- throw panicError;
127
- }
128
- else {
129
- this.error(new Error("Failed to emit build.start event"));
130
- }
131
- }
275
+ throw new Error("React static plugin failed to create worker");
276
+ } else if (workerResult.type === "skip") {
277
+ logger.info("Worker not created, skipping static build");
278
+ return;
279
+ } else {
280
+ worker = workerResult.worker;
281
+ const workerStartupTime = performance.now() - workerStartTime;
282
+ if (userOptions.onMetrics) {
283
+ const workerStartupMetric = createWorkerStartupMetrics({
284
+ route: "/",
285
+ // Worker startup is global, not route-specific
286
+ workerType: "html",
287
+ // This is the HTML worker for server-side static generation
288
+ startupTime: workerStartupTime,
289
+ fromMainThread: true,
290
+ fromRscWorker: false,
291
+ fromHtmlWorker: false,
292
+ description: `HTML worker startup for server-side static generation`
293
+ });
294
+ if (this.environment.name === "server") {
295
+ userOptions.onMetrics(workerStartupMetric);
296
+ }
132
297
  }
133
- },
134
- // the preview server helps to view the generated static folder, but only when the static plugin is enabled
135
- // if no build.pages, then the preview server will instead use default vite preview server
136
- // it works the same under both conditions
137
- async configurePreviewServer(server) {
138
- logger = server.config.customLogger || server.config.logger;
139
- configurePreviewServer({
140
- server,
141
- userOptions,
298
+ globalWorker = worker;
299
+ }
300
+ }
301
+ const { onEvent, ...handlerOptions } = userOptions;
302
+ if (typeof userOptions.onEvent === "function") {
303
+ try {
304
+ const r = userOptions.onEvent({
305
+ type: "build.ssg.start",
306
+ data: {
307
+ pages: Array.from(autoDiscoveredFiles?.urlMap.keys() ?? []),
308
+ options: null,
309
+ // No specific rollup output options for static generation
310
+ bundle
311
+ }
142
312
  });
143
- },
144
- async renderStart() {
145
- timing.renderStart = performance.now();
146
- },
147
- generateBundle(_options, bundle) {
148
- // Filter out unnecessary _virtual files from the bundle
149
- // Keep dynamic-import-helper.js since it's needed for dynamic imports
150
- // Note: Static builds are handled by plugin.client.ts, this only handles server builds
151
- if (this.environment.name === "server") {
152
- const keysToDelete = [];
153
- for (const [key, chunk] of Object.entries(bundle)) {
154
- if (chunk.type === "chunk") {
155
- // Check fileName, key, moduleIds, and facadeModuleId for _virtual
156
- const isVirtual = chunk.fileName?.includes("_virtual") ||
157
- key.includes("_virtual") ||
158
- chunk.facadeModuleId?.includes("_virtual") ||
159
- chunk.moduleIds?.some(id => id.includes("_virtual"));
160
- // Keep dynamic-import-helper.js - it's needed for dynamic imports
161
- const isDynamicImportHelper = chunk.fileName?.includes("dynamic-import-helper") ||
162
- key.includes("dynamic-import-helper");
163
- if (isVirtual && !isDynamicImportHelper) {
164
- keysToDelete.push(key);
165
- if (userOptions.verbose) {
166
- logger?.info(`[plugin.server] Filtered out virtual file: ${chunk.fileName || key} (moduleId: ${chunk.facadeModuleId || chunk.moduleIds?.[0]})`);
167
- }
168
- }
169
- }
170
- }
171
- // Delete after iteration to avoid modifying while iterating
172
- for (const key of keysToDelete) {
173
- delete bundle[key];
174
- }
175
- }
176
- },
177
- async writeBundle(_options, bundle) {
178
- // Only execute static generation for the server environment
179
- if (this.environment.name !== "server") {
180
- if (userOptions.verbose) {
181
- logger?.info(`[plugin.server] Skipping static generation for environment: ${this.environment.name}`);
182
- }
183
- return;
313
+ if (r != null && typeof r === "object" && "then" in r) {
314
+ await r;
184
315
  }
185
- let panicError = null;
186
- let bundleManifest = undefined;
187
- if (!logger) {
188
- logger = this.environment.logger;
316
+ } catch (error) {
317
+ const eventPanicError = handleError({
318
+ error,
319
+ logger,
320
+ panicThreshold: userOptions.panicThreshold,
321
+ context: "onEvent(build.ssg.start)"
322
+ });
323
+ if (eventPanicError != null) {
324
+ throw eventPanicError;
325
+ } else {
326
+ throw new Error("Failed to emit build.ssg.start event");
189
327
  }
190
- // handle the bundle manifest
191
- try {
192
- bundleManifest = getBundleManifest({
193
- bundle,
194
- normalizer: userOptions.normalizer,
195
- });
196
- // make sure that we have a manifest
197
- const manifestPath = typeof resolvedConfig.build.manifest === "string"
198
- ? resolvedConfig.build.manifest
199
- : ".vite/manifest.json";
200
- if (!bundleManifest[manifestPath] ||
201
- !("source" in bundleManifest[manifestPath])) {
202
- throw new Error("Server manifest not found");
203
- }
204
- // parse the manifest
205
- serverManifest = JSON.parse(bundleManifest[manifestPath].source);
206
- // make sure that we have a manifest
207
- if (!serverManifest) {
208
- throw new Error("Failed to parse server manifest");
209
- }
328
+ }
329
+ }
330
+ const renderMode = userOptions.build?.renderMode ?? "parallel";
331
+ const renderPages$1 = renderMode === "sequential" ? renderPages : renderPagesBatched;
332
+ if (userOptions.verbose) {
333
+ logger.info(`[static] Using ${renderMode} rendering${renderMode === "parallel" ? ` (batch size: ${userOptions.build?.batchSize ?? 8})` : ""}`);
334
+ }
335
+ const renderPagesGenerator = renderPages$1(
336
+ routes,
337
+ {
338
+ ...handlerOptions,
339
+ loader: buildLoader,
340
+ worker,
341
+ htmlWorker: worker,
342
+ // Pass the HTML worker for HTML generation
343
+ logger,
344
+ // Pass global CSS to downstream renderer
345
+ globalCss,
346
+ // Pass abort signal to cancel operations when errors occur
347
+ signal: AbortSignal.timeout(handlerOptions.htmlTimeout),
348
+ onEvent,
349
+ serverPipeableStreamOptions,
350
+ clientPipeableStreamOptions,
351
+ manifest: serverManifest ?? {},
352
+ staticManifest,
353
+ // Pass static manifest for path resolution
354
+ autoDiscoveredFiles,
355
+ cssFilesByPage,
356
+ batchSize: userOptions.build?.batchSize
357
+ },
358
+ renderPage
359
+ );
360
+ let finalResult;
361
+ try {
362
+ for await (const result of renderPagesGenerator) {
363
+ if (result.type === "error") {
364
+ throw result.error;
210
365
  }
211
- catch (error) {
212
- const panicError = handleError({
213
- error,
214
- logger: logger,
215
- panicThreshold: userOptions.panicThreshold,
216
- context: "writeBundle(bundleManifest)",
217
- });
218
- if (panicError != null) {
219
- throw panicError;
220
- }
221
- else {
222
- throw new Error("Failed to get bundle manifest");
223
- }
366
+ if (result.type === "success" && result.failedRoutes && result.failedRoutes.size > 0) {
367
+ const firstError = result.failedRoutes.values().next().value;
368
+ if (firstError != null && shouldCausePanic(firstError, { panicThreshold: userOptions.panicThreshold })) {
369
+ throw firstError;
370
+ }
371
+ for (const [route, error] of result.failedRoutes) {
372
+ const err = error instanceof Error ? error : toError(error);
373
+ this.warn(
374
+ new Error("Failed to render route: " + route + "\n" + err.message + "\n" + err.stack, { cause: err })
375
+ );
376
+ }
224
377
  }
225
- try {
226
- const staticManifestResult = await tryManifest({
227
- root: userOptions.projectRoot,
228
- outDir: join(userOptions.build.outDir, userOptions.build.static),
229
- manifestPath: resolvedConfig.build.manifest,
230
- ssrManifest: false,
231
- });
232
- if (staticManifestResult.type === "error") {
233
- throw staticManifestResult.error;
234
- }
235
- const staticManifest = staticManifestResult.manifest;
236
- // Don't create helper file - let resolveVirtualAndNodeModules shim handle it
237
- // Same approach as client environment - no special file needed
238
- const buildLoader = createBuildLoader({
239
- userOptions: userOptions,
240
- serverManifest: serverManifest ?? {},
241
- staticManifest: staticManifest,
242
- }, bundle, temporaryReferences, logger);
243
- // Create CSS props for each CSS file
244
- const { cssFilesByPage, globalCss } = processCssFilesForPages({
245
- userOptions,
246
- autoDiscoveredFiles,
247
- serverManifest,
248
- staticManifest,
249
- bundle,
250
- logger,
251
- });
252
- if (userOptions.verbose) {
253
- logger.info(`[plugin.server] cssFilesByPage size: ${cssFilesByPage.size}`);
254
- for (const [route, cssMap] of cssFilesByPage.entries()) {
255
- logger.info(`[plugin.server] Route ${route}: ${cssMap.size} CSS files`);
256
- }
257
- }
258
- const indexHtml = staticManifest?.["index.html"]?.file;
259
- const serverPipeableStreamOptions = {
260
- ...userOptions.serverPipeableStreamOptions,
261
- bootstrapModules: [
262
- ...(indexHtml ? [baseURL(indexHtml)] : []),
263
- ...(userOptions.serverPipeableStreamOptions?.bootstrapModules ??
264
- []),
265
- ],
266
- };
267
- userOptions.serverPipeableStreamOptions = serverPipeableStreamOptions;
268
- const clientPipeableStreamOptions = {
269
- ...userOptions.clientPipeableStreamOptions,
270
- bootstrapModules: [
271
- ...(indexHtml ? [baseURL(indexHtml)] : []),
272
- ...(userOptions.clientPipeableStreamOptions?.bootstrapModules ??
273
- []),
274
- ],
275
- };
276
- // Get routes for worker configuration
277
- const routes = !autoDiscoveredFiles
278
- ? []
279
- : Array.from(autoDiscoveredFiles.urlMap.keys());
280
- // If no pages to generate, skip static generation entirely (including worker creation)
281
- if (routes.length === 0) {
282
- logger?.info("[plugin.server] No pages to generate, skipping static generation");
283
- return;
284
- }
285
- const serializedUserOptions = serializedOptions(userOptions, autoDiscoveredFiles);
286
- // Create HTML worker for HTML generation
287
- // IMPORTANT: We create a new worker for each page render to ensure completely clean state
288
- // This prevents race conditions where worker state persists between renders
289
- // Guard against duplicate worker creation if plugin is instantiated multiple times
290
- if (globalWorker) {
291
- logger?.warn("[plugin.server] Global worker already exists, reusing existing worker");
292
- worker = globalWorker;
293
- }
294
- else {
295
- const workerStartTime = performance.now();
296
- const viteEnvPrefix = envPrefixFromConfig(resolvedConfig);
297
- const routeCount = autoDiscoveredFiles?.urlMap.size ?? 0;
298
- const maxListeners = routeCount + 1;
299
- const workerResult = await createWorker({
300
- projectRoot: userOptions.projectRoot,
301
- workerPath: userOptions.htmlWorkerPath,
302
- currentCondition: "react-server",
303
- reverseCondition: "react-client", // HTML worker needs react-client for react-dom/server
304
- maxListeners: maxListeners,
305
- envPrefix: viteEnvPrefix,
306
- logger: logger,
307
- workerData: {
308
- resolvedConfig: serializeResolvedConfig(resolvedConfig),
309
- userOptions: serializedUserOptions,
310
- configEnv,
311
- },
312
- });
313
- if (workerResult.type === "error") {
314
- if (workerResult.error != null) {
315
- throw workerResult.error;
316
- }
317
- throw new Error("React static plugin failed to create worker");
318
- }
319
- else if (workerResult.type === "skip") {
320
- logger.info("Worker not created, skipping static build");
321
- return;
322
- }
323
- else {
324
- worker = workerResult.worker;
325
- // Emit worker startup metric after worker is created
326
- const workerStartupTime = performance.now() - workerStartTime;
327
- if (userOptions.onMetrics) {
328
- const workerStartupMetric = createWorkerStartupMetrics({
329
- route: "/", // Worker startup is global, not route-specific
330
- workerType: "html", // This is the HTML worker for server-side static generation
331
- startupTime: workerStartupTime,
332
- fromMainThread: true,
333
- fromRscWorker: false,
334
- fromHtmlWorker: false,
335
- description: `HTML worker startup for server-side static generation`,
336
- });
337
- // Only emit metrics from the server environment to prevent duplicates
338
- if (this.environment.name === "server") {
339
- userOptions.onMetrics(workerStartupMetric);
340
- }
341
- }
342
- // Store the worker globally to prevent duplicate creation
343
- globalWorker = worker;
344
- }
345
- }
346
- // No RSC worker needed for static generation - main thread runs with react-server conditions
347
- // Render pages - component resolution now happens per-route in renderPage
348
- const { onEvent, ...handlerOptions } = userOptions;
349
- // Emit the static site generation start event
350
- if (typeof userOptions.onEvent === "function") {
351
- try {
352
- const r = userOptions.onEvent({
353
- type: "build.ssg.start",
354
- data: {
355
- pages: Array.from(autoDiscoveredFiles?.urlMap.keys() ?? []),
356
- options: null, // No specific rollup output options for static generation
357
- bundle: bundle,
358
- },
359
- });
360
- if (r != null && typeof r === "object" && "then" in r) {
361
- await r;
362
- }
363
- }
364
- catch (error) {
365
- const eventPanicError = handleError({
366
- error,
367
- logger: logger,
368
- panicThreshold: userOptions.panicThreshold,
369
- context: "onEvent(build.ssg.start)",
370
- });
371
- if (eventPanicError != null) {
372
- throw eventPanicError; // Re-throw to abort the build
373
- }
374
- else {
375
- throw new Error("Failed to emit build.ssg.start event");
376
- }
377
- }
378
- }
379
- // Select render mode based on build config
380
- const renderMode = userOptions.build?.renderMode ?? "parallel";
381
- const renderPages = renderMode === "sequential" ? renderPagesSequential : renderPagesBatched;
382
- if (userOptions.verbose) {
383
- logger.info(`[static] Using ${renderMode} rendering${renderMode === "parallel" ? ` (batch size: ${userOptions.build?.batchSize ?? 8})` : ""}`);
384
- }
385
- // this will render the routes
386
- const renderPagesGenerator = renderPages(routes, {
387
- ...handlerOptions,
388
- loader: buildLoader,
389
- worker: worker,
390
- htmlWorker: worker, // Pass the HTML worker for HTML generation
391
- logger: logger,
392
- // Pass global CSS to downstream renderer
393
- globalCss,
394
- // Pass abort signal to cancel operations when errors occur
395
- signal: AbortSignal.timeout(handlerOptions.htmlTimeout),
396
- onEvent: onEvent,
397
- serverPipeableStreamOptions: serverPipeableStreamOptions,
398
- clientPipeableStreamOptions: clientPipeableStreamOptions,
399
- manifest: serverManifest ?? {},
400
- staticManifest: staticManifest, // Pass static manifest for path resolution
401
- autoDiscoveredFiles: autoDiscoveredFiles,
402
- cssFilesByPage: cssFilesByPage,
403
- batchSize: userOptions.build?.batchSize,
404
- }, renderPage);
405
- // Process render results
406
- let finalResult;
407
- try {
408
- for await (const result of renderPagesGenerator) {
409
- // Handle error results immediately
410
- if (result.type === "error") {
411
- throw result.error;
412
- }
413
- // Handle failed routes based on panic threshold
414
- if (result.type === "success" &&
415
- result.failedRoutes &&
416
- result.failedRoutes.size > 0) {
417
- // Use centralized panic threshold logic
418
- const firstError = result.failedRoutes.values().next().value;
419
- if (firstError != null && shouldCausePanic(firstError, { panicThreshold: userOptions.panicThreshold })) {
420
- // This should cause a panic, throw the error
421
- throw firstError;
422
- }
423
- // For non-panic errors, log warnings but continue
424
- for (const [route, error] of result.failedRoutes) {
425
- const err = error instanceof Error ? error : toError(error);
426
- this.warn(new Error("Failed to render route: " + route + "\n" + err.message + "\n" + err.stack, { cause: err }));
427
- }
428
- }
429
- finalResult = result;
430
- }
431
- }
432
- catch (renderError) {
433
- // Handle render errors with panic threshold logic
434
- const renderPanicError = handleError({
435
- error: renderError,
436
- logger: logger,
437
- panicThreshold: userOptions.panicThreshold,
438
- context: "renderPages",
439
- });
440
- if (renderPanicError != null) {
441
- throw renderPanicError;
442
- }
443
- throw renderError;
444
- }
445
- if (!finalResult) {
446
- throw new Error("No render result produced");
447
- }
448
- // Calculate duration from timing
449
- const duration = Math.round(performance.now() - (timing.renderStart || timing.start));
450
- this.info(`Rendered ${finalResult.completedRoutes.size} pages in ${duration}ms`);
451
- // Emit the static site generation completion event once
452
- if (typeof userOptions.onEvent === "function") {
453
- try {
454
- const r = userOptions.onEvent({
455
- type: "build.ssg.end",
456
- data: {
457
- pages: Array.from(autoDiscoveredFiles?.urlMap.keys() ?? []),
458
- options: null, // No specific rollup output options for static generation
459
- bundle: bundle,
460
- },
461
- });
462
- if (r != null && typeof r === "object" && "then" in r) {
463
- await r;
464
- }
465
- }
466
- catch (error) {
467
- if (error != null) {
468
- throw error; // Re-throw to abort the build
469
- }
470
- else {
471
- throw new Error("Failed to emit build.ssg.end event");
472
- }
473
- }
474
- }
475
- if (process.env["NODE_ENV"] !== "production") {
476
- this.warn(`THIS BUILD IS NOT INTENDED FOR PRODUCTION (${process.env["NODE_ENV"]})`);
477
- }
478
- // Update timing
479
- timing.render =
480
- performance.now() - (timing.renderStart ?? timing.start);
481
- }
482
- catch (error) {
483
- panicError = handleError({
484
- error,
485
- logger: logger,
486
- panicThreshold: userOptions.panicThreshold,
487
- context: "writeBundle",
488
- });
489
- // Let the finally block handle additional cleanup
490
- }
491
- finally {
492
- // Reset any cached state to prevent issues in subsequent builds
493
- autoDiscoveredFiles = null;
494
- serverManifest = undefined;
495
- // Clean up worker if it exists
496
- if (worker) {
497
- try {
498
- worker.removeAllListeners();
499
- worker.terminate();
500
- }
501
- catch (terminateError) {
502
- // Ignore termination errors
503
- }
504
- worker = undefined;
505
- // Reset global worker since it's been terminated
506
- globalWorker = undefined;
507
- }
378
+ finalResult = result;
379
+ }
380
+ } catch (renderError) {
381
+ const renderPanicError = handleError({
382
+ error: renderError,
383
+ logger,
384
+ panicThreshold: userOptions.panicThreshold,
385
+ context: "renderPages"
386
+ });
387
+ if (renderPanicError != null) {
388
+ throw renderPanicError;
389
+ }
390
+ throw renderError;
391
+ }
392
+ if (!finalResult) {
393
+ throw new Error("No render result produced");
394
+ }
395
+ const duration = Math.round(
396
+ performance.now() - (timing.renderStart || timing.start)
397
+ );
398
+ this.info(
399
+ `Rendered ${finalResult.completedRoutes.size} pages in ${duration}ms`
400
+ );
401
+ if (typeof userOptions.onEvent === "function") {
402
+ try {
403
+ const r = userOptions.onEvent({
404
+ type: "build.ssg.end",
405
+ data: {
406
+ pages: Array.from(autoDiscoveredFiles?.urlMap.keys() ?? []),
407
+ options: null,
408
+ // No specific rollup output options for static generation
409
+ bundle
410
+ }
411
+ });
412
+ if (r != null && typeof r === "object" && "then" in r) {
413
+ await r;
508
414
  }
509
- if (panicError != null) {
510
- // Ensure we have a proper Error object that can have properties set on it
511
- const errorToThrow = panicError instanceof Error
512
- ? panicError
513
- : new Error(String(panicError));
514
- // Create a new Error object to avoid the "code" property issue
515
- const finalError = new Error(errorToThrow.message);
516
- finalError.stack = errorToThrow.stack;
517
- finalError.cause = errorToThrow.cause;
518
- // Copy any additional properties that might be needed
519
- if (errorToThrow.name)
520
- finalError.name = errorToThrow.name;
521
- throw finalError;
415
+ } catch (error) {
416
+ if (error != null) {
417
+ throw error;
418
+ } else {
419
+ throw new Error("Failed to emit build.ssg.end event");
522
420
  }
523
- },
524
- async closeBundle() {
525
- // Clean up _virtual files after build completes
526
- // These are Vite's internal virtual modules and aren't needed in the final output
527
- if (this.environment.name === "server") {
528
- try {
529
- const { existsSync } = await import("node:fs");
530
- const { join, resolve } = await import("node:path");
531
- // Use the resolved output directory from the environment config
532
- const resolvedOutDir = this.environment.config.build?.outDir
533
- ? resolve(this.environment.config.root || userOptions.projectRoot, this.environment.config.build.outDir)
534
- : resolve(userOptions.projectRoot, userOptions.build.outDir);
535
- // Don't clean up server/_virtual - we need dynamic-import-helper.js for runtime
536
- // Only clean up static/_virtual if it exists (shouldn't, but just in case)
537
- const staticOutDir = join(resolvedOutDir, userOptions.build.static || "static");
538
- const staticVirtualDir = join(staticOutDir, "_virtual");
539
- if (existsSync(staticVirtualDir)) {
540
- const { rmSync } = await import("node:fs");
541
- rmSync(staticVirtualDir, { recursive: true, force: true });
542
- if (userOptions.verbose) {
543
- logger?.info(`[plugin.server] Cleaned up _virtual directory: ${staticVirtualDir}`);
544
- }
545
- }
546
- }
547
- catch (error) {
548
- // Non-critical - log but don't fail the build
549
- if (userOptions.verbose) {
550
- logger?.warn(`[plugin.server] Failed to clean up _virtual directory: ${error}`);
551
- }
552
- }
421
+ }
422
+ }
423
+ if (process.env["NODE_ENV"] !== "production") {
424
+ this.warn(
425
+ `THIS BUILD IS NOT INTENDED FOR PRODUCTION (${process.env["NODE_ENV"]})`
426
+ );
427
+ }
428
+ timing.render = performance.now() - (timing.renderStart ?? timing.start);
429
+ } catch (error) {
430
+ panicError = handleError({
431
+ error,
432
+ logger,
433
+ panicThreshold: userOptions.panicThreshold});
434
+ } finally {
435
+ autoDiscoveredFiles = null;
436
+ serverManifest = void 0;
437
+ if (worker) {
438
+ try {
439
+ worker.removeAllListeners();
440
+ worker.terminate();
441
+ } catch (terminateError) {
442
+ }
443
+ worker = void 0;
444
+ globalWorker = void 0;
445
+ }
446
+ }
447
+ if (panicError != null) {
448
+ const errorToThrow = panicError instanceof Error ? panicError : new Error(String(panicError));
449
+ const finalError = new Error(errorToThrow.message);
450
+ finalError.stack = errorToThrow.stack;
451
+ finalError.cause = errorToThrow.cause;
452
+ if (errorToThrow.name) finalError.name = errorToThrow.name;
453
+ throw finalError;
454
+ }
455
+ },
456
+ async closeBundle() {
457
+ if (this.environment.name === "server") {
458
+ try {
459
+ const { existsSync } = await import('node:fs');
460
+ const { join: join2, resolve } = await import('node:path');
461
+ 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);
462
+ const staticOutDir = join2(resolvedOutDir, userOptions.build.static || "static");
463
+ const staticVirtualDir = join2(staticOutDir, "_virtual");
464
+ if (existsSync(staticVirtualDir)) {
465
+ const { rmSync } = await import('node:fs');
466
+ rmSync(staticVirtualDir, { recursive: true, force: true });
467
+ if (userOptions.verbose) {
468
+ logger?.info(`[plugin.server] Cleaned up _virtual directory: ${staticVirtualDir}`);
553
469
  }
554
- // Graceful worker shutdown - only at the end of the entire build process
555
- if (worker) {
556
- try {
557
- await Promise.race([
558
- new Promise((resolve, reject) => {
559
- const timeout = setTimeout(() => {
560
- reject(new Error("Worker shutdown timeout"));
561
- }, userOptions.workerShutdownTimeout);
562
- const backupTimeout = setTimeout(() => {
563
- reject(new Error("Worker shutdown backup timeout"));
564
- }, Math.floor(userOptions.workerShutdownTimeout * 0.6)); // 60% of main timeout
565
- const messageHandler = (message) => {
566
- if (message.type === "SHUTDOWN_COMPLETE") {
567
- if (userOptions.verbose) {
568
- logger.info("Worker shutdown complete");
569
- }
570
- clearTimeout(timeout);
571
- clearTimeout(backupTimeout);
572
- worker?.removeListener("message", messageHandler);
573
- // Remove all other event listeners as well
574
- worker?.removeAllListeners();
575
- resolve();
576
- }
577
- else if (message.type === "CLEANUP_COMPLETE") {
578
- // Handle cleanup complete messages during shutdown - this is normal
579
- if (userOptions.verbose) {
580
- logger.info("Worker cleanup completed during shutdown");
581
- }
582
- // Don't resolve here - wait for SHUTDOWN_COMPLETE
583
- }
584
- else {
585
- if (userOptions.verbose) {
586
- logger.info("Worker is still busy, received message " + message?.type);
587
- }
588
- }
589
- };
590
- worker?.on("message", messageHandler);
591
- // Send shutdown message
592
- worker?.postMessage({
593
- type: "SHUTDOWN",
594
- id: "*",
595
- });
596
- }),
597
- ]);
598
- }
599
- catch (error) {
600
- // If shutdown protocol fails, force terminate
601
- this.warn("Worker shutdown protocol failed, forcing termination: " +
602
- (error instanceof Error ? error.message : String(error)));
603
- // Don't try to clean up listeners in error case - just force terminate
604
- }
605
- finally {
606
- // Always force cleanup and termination
607
- if (worker) {
608
- try {
609
- worker.removeAllListeners();
610
- worker.terminate();
611
- }
612
- catch (terminateError) {
613
- // Ignore termination errors
614
- }
615
- worker = undefined;
616
- // Reset global worker since it's been terminated
617
- globalWorker = undefined;
618
- }
470
+ }
471
+ } catch (error) {
472
+ if (userOptions.verbose) {
473
+ logger?.warn(`[plugin.server] Failed to clean up _virtual directory: ${error}`);
474
+ }
475
+ }
476
+ }
477
+ if (worker) {
478
+ try {
479
+ await Promise.race([
480
+ new Promise((resolve, reject) => {
481
+ const timeout = setTimeout(() => {
482
+ reject(new Error("Worker shutdown timeout"));
483
+ }, userOptions.workerShutdownTimeout);
484
+ const backupTimeout = setTimeout(() => {
485
+ reject(new Error("Worker shutdown backup timeout"));
486
+ }, Math.floor(userOptions.workerShutdownTimeout * 0.6));
487
+ const messageHandler = (message) => {
488
+ if (message.type === "SHUTDOWN_COMPLETE") {
489
+ if (userOptions.verbose) {
490
+ logger.info("Worker shutdown complete");
491
+ }
492
+ clearTimeout(timeout);
493
+ clearTimeout(backupTimeout);
494
+ worker?.removeListener("message", messageHandler);
495
+ worker?.removeAllListeners();
496
+ resolve();
497
+ } else if (message.type === "CLEANUP_COMPLETE") {
498
+ if (userOptions.verbose) {
499
+ logger.info("Worker cleanup completed during shutdown");
500
+ }
501
+ } else {
502
+ if (userOptions.verbose) {
503
+ logger.info(
504
+ "Worker is still busy, received message " + message?.type
505
+ );
506
+ }
619
507
  }
508
+ };
509
+ worker?.on("message", messageHandler);
510
+ worker?.postMessage({
511
+ type: "SHUTDOWN",
512
+ id: "*"
513
+ });
514
+ })
515
+ ]);
516
+ } catch (error) {
517
+ this.warn(
518
+ "Worker shutdown protocol failed, forcing termination: " + (error instanceof Error ? error.message : String(error))
519
+ );
520
+ } finally {
521
+ if (worker) {
522
+ try {
523
+ worker.removeAllListeners();
524
+ worker.terminate();
525
+ } catch (terminateError) {
620
526
  }
621
- },
622
- };
527
+ worker = void 0;
528
+ globalWorker = void 0;
529
+ }
530
+ }
531
+ }
532
+ }
533
+ };
623
534
  };
535
+
536
+ export { reactStaticPlugin };
537
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"plugin.server.js","sources":["../../../plugin/react-static/plugin.server.ts"],"sourcesContent":["/**\n * plugin.ts\n *\n * PURPOSE: Main Vite plugin for React Server Components (RSC) static site generation\n *\n * This module:\n * 1. Orchestrates the entire static site generation process\n * 2. Manages the lifecycle of the RSC rendering process\n * 3. Handles file writing for both initial page loads and client-side navigation\n *    - Writes .html files for initial page loads (complete HTML document)\n *    - Writes .rsc files for client-side navigation (RSC content only)\n * 4. Provides hooks for Vite to integrate with the build process\n * 5. Manages worker threads for parallel rendering\n * 6. Handles error reporting and metrics collection\n */\n\nimport type { Worker } from \"node:worker_threads\";\nimport {\n  type ConfigEnv,\n  type Logger,\n  type Manifest,\n  type ManifestChunk,\n  type ResolvedConfig,\n  createLogger,\n} from \"vite\";\nimport { resolveOptions } from \"../config/resolveOptions.js\";\nimport { createBuildLoader } from \"./createBuildLoader.server.js\";\nimport type {\n  BuildTiming,\n  RenderPagesResult,\n  AutoDiscoveredFiles,\n  VitePluginFn,\n} from \"../types.js\";\nimport { renderPagesBatched } from \"./renderPagesBatched.js\";\nimport { renderPages as renderPagesSequential } from \"./renderPages.js\";\nimport { getBundleManifest } from \"../helpers/getBundleManifest.js\";\nimport { createWorker } from \"../worker/createWorker.js\";\nimport {\n  serializedOptions,\n  serializeResolvedConfig,\n} from \"../helpers/serializeUserOptions.js\";\nimport { performance } from \"node:perf_hooks\";\nimport { baseURL } from \"../utils/envUrls.node.js\";\nimport { handleError } from \"../error/handleError.js\";\nimport { shouldCausePanic } from \"../error/panicThresholdHandler.js\";\nimport { renderPage } from \"./renderPage.server.js\";\nimport { temporaryReferences } from \"./temporaryReferences.server.js\";\nimport { configurePreviewServer } from \"./configurePreviewServer.js\";\nimport { envPrefixFromConfig } from \"../config/envPrefixFromConfig.js\";\n\nimport { processCssFilesForPages } from \"./processCssFilesForPages.js\";\nimport { createWorkerStartupMetrics } from \"../metrics/createWorkerStartupMetrics.js\";\nimport { tryManifest } from \"../helpers/tryManifest.js\";\nimport { join } from \"node:path\";\nimport { resolveAutoDiscover } from \"../config/autoDiscover/resolveAutoDiscover.js\";\nimport { assertReactServer } from \"../config/getCondition.js\";\nimport { toError } from \"../error/toError.js\";\n\nassertReactServer();\n\n/**\n * Main entrypoint for the static plugin.\n *\n * This plugin is responsible for:\n * 1. Orchestrating the static site generation process\n * 2. Handling the lifecycle of the RSC rendering process (main thread)\n * 3. Writing .html files for initial page loads (complete HTML document)\n * 4. Writing .rsc files for client-side navigation (RSC content only)\n * 5. Managing worker threads for parallel rendering (html worker)\n * 6. Handling error reporting and metrics collection\n */\n// Global worker instance to prevent duplicate creation across plugin instances\nlet globalWorker: Worker | undefined;\n\nexport const reactStaticPlugin: VitePluginFn = function _reactStaticPlugin(\n  options\n) {\n  let worker: Worker | undefined;\n  let logger: Logger;\n  let resolvedConfig: ResolvedConfig;\n  let autoDiscoveredFiles: AutoDiscoveredFiles | null = null;\n  let serverManifest: Manifest | undefined = undefined;\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/server-static\",\n    enforce: \"post\",\n    api: {\n      meta: { timing },\n    },\n    async config(_config, viteConfigEnv) {\n      configEnv = viteConfigEnv;\n    },\n    applyToEnvironment(partialEnvironment) {\n\n      if (\n        [\"server\"].includes(\n          partialEnvironment.name as \"client\" | \"server\" | \"ssr\"\n        )\n      ) {\n        return true;\n      }\n      return false;\n    },\n    async configResolved(config) {\n      resolvedConfig = config;\n      if (!logger) {\n        logger = config.customLogger ?? createLogger();\n      }\n      const autoDiscoverResult = await resolveAutoDiscover({\n        config: config,\n        configEnv: configEnv!,\n        userOptions,\n        logger,\n      });\n      if (autoDiscoverResult.type === \"error\") {\n        throw autoDiscoverResult.error;\n      }\n      autoDiscoveredFiles = autoDiscoverResult.autoDiscoveredFiles;\n    },\n    async buildStart() {\n      if (!logger) {\n        logger = this.environment.logger;\n      }\n      timing.buildStart = performance.now();\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            worker?.terminate();\n            throw panicError;\n          } else {\n            this.error(new Error(\"Failed to emit build.start event\"));\n          }\n        }\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    async renderStart() {\n      timing.renderStart = performance.now();\n    },\n    generateBundle(_options, bundle) {\n      // Filter out unnecessary _virtual files from the bundle\n      // Keep dynamic-import-helper.js since it's needed for dynamic imports\n      // Note: Static builds are handled by plugin.client.ts, this only handles server builds\n      if (this.environment.name === \"server\") {\n        const keysToDelete: string[] = [];\n        for (const [key, chunk] of Object.entries(bundle)) {\n          if (chunk.type === \"chunk\") {\n            // Check fileName, key, moduleIds, and facadeModuleId for _virtual\n            const isVirtual = \n              chunk.fileName?.includes(\"_virtual\") ||\n              key.includes(\"_virtual\") ||\n              chunk.facadeModuleId?.includes(\"_virtual\") ||\n              chunk.moduleIds?.some(id => id.includes(\"_virtual\"));\n            \n            // Keep dynamic-import-helper.js - it's needed for dynamic imports\n            const isDynamicImportHelper = \n              chunk.fileName?.includes(\"dynamic-import-helper\") ||\n              key.includes(\"dynamic-import-helper\");\n            \n            if (isVirtual && !isDynamicImportHelper) {\n              keysToDelete.push(key);\n              if (userOptions.verbose) {\n                logger?.info(`[plugin.server] Filtered out virtual file: ${chunk.fileName || key} (moduleId: ${chunk.facadeModuleId || chunk.moduleIds?.[0]})`);\n              }\n            }\n          }\n        }\n        // Delete after iteration to avoid modifying while iterating\n        for (const key of keysToDelete) {\n          delete bundle[key];\n        }\n      }\n    },\n\n    async writeBundle(_options, bundle) {\n      // Only execute static generation for the server environment\n      if (this.environment.name !== \"server\") {\n        if (userOptions.verbose) {\n          logger?.info(`[plugin.server] Skipping static generation for environment: ${this.environment.name}`);\n        }\n        return;\n      }\n      \n      let panicError: Error | null = null;\n      let bundleManifest:\n        | {\n            [key: string]: ManifestChunk & {\n              source: string;\n            };\n          }\n        | undefined = undefined;\n      if (!logger) {\n        logger = this.environment.logger;\n      }\n\n      // handle the bundle manifest\n      try {\n        bundleManifest = getBundleManifest<false>({\n          bundle,\n          normalizer: userOptions.normalizer,\n        });\n\n        // make sure that we have a manifest\n        const manifestPath =\n          typeof resolvedConfig.build.manifest === \"string\"\n            ? resolvedConfig.build.manifest\n            : \".vite/manifest.json\";\n        if (\n          !bundleManifest[manifestPath] ||\n          !(\"source\" in bundleManifest[manifestPath])\n        ) {\n          throw new Error(\"Server manifest not found\");\n        }\n\n        // parse the manifest\n        serverManifest = JSON.parse(\n          bundleManifest[manifestPath].source as string\n        );\n\n        // make sure that we have a manifest\n        if (!serverManifest) {\n          throw new Error(\"Failed to parse server manifest\");\n        }\n      } catch (error) {\n        const panicError = handleError({\n          error,\n          logger: logger,\n          panicThreshold: userOptions.panicThreshold,\n          context: \"writeBundle(bundleManifest)\",\n        });\n        if (panicError != null) {\n          throw panicError;\n        } else {\n          throw new Error(\"Failed to get bundle manifest\");\n        }\n      }\n\n      try {\n        const staticManifestResult = await tryManifest({\n          root: userOptions.projectRoot,\n          outDir: join(userOptions.build.outDir, userOptions.build.static),\n          manifestPath: resolvedConfig.build.manifest,\n          ssrManifest: false,\n        });\n        if (staticManifestResult.type === \"error\") {\n          throw staticManifestResult.error;\n        }\n        const staticManifest = staticManifestResult.manifest;\n        \n        // Don't create helper file - let resolveVirtualAndNodeModules shim handle it\n        // Same approach as client environment - no special file needed\n        \n        const buildLoader = createBuildLoader(\n          {\n            userOptions: userOptions,\n            serverManifest: serverManifest ?? {},\n            staticManifest: staticManifest,\n          },\n          bundle,\n          temporaryReferences,\n          logger\n        );\n        // Create CSS props for each CSS file\n        const { cssFilesByPage, globalCss } = processCssFilesForPages({\n          userOptions,\n          autoDiscoveredFiles,\n          serverManifest,\n          staticManifest,\n          bundle,\n          logger,\n        });\n\n        if (userOptions.verbose) {\n          logger.info(\n            `[plugin.server] cssFilesByPage size: ${cssFilesByPage.size}`\n          );\n          for (const [route, cssMap] of cssFilesByPage.entries()) {\n            logger.info(\n              `[plugin.server] Route ${route}: ${cssMap.size} CSS files`\n            );\n            \n          }\n        }\n\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        // Get routes for worker configuration\n        const routes = !autoDiscoveredFiles\n          ? []\n          : Array.from(autoDiscoveredFiles!.urlMap.keys());\n\n        // If no pages to generate, skip static generation entirely (including worker creation)\n        if (routes.length === 0) {\n          logger?.info(\n            \"[plugin.server] No pages to generate, skipping static generation\"\n          );\n          return;\n        }\n\n        const serializedUserOptions = serializedOptions(\n          userOptions,\n          autoDiscoveredFiles!\n        );\n        // Create HTML worker for HTML generation\n        // IMPORTANT: We create a new worker for each page render to ensure completely clean state\n        // This prevents race conditions where worker state persists between renders\n        // Guard against duplicate worker creation if plugin is instantiated multiple times\n        if (globalWorker) {\n          logger?.warn(\"[plugin.server] Global worker already exists, reusing existing worker\");\n          worker = globalWorker;\n        } else {\n          const workerStartTime = performance.now();\n          const viteEnvPrefix = envPrefixFromConfig(resolvedConfig);\n          const routeCount = autoDiscoveredFiles?.urlMap.size ?? 0;\n          const maxListeners = routeCount + 1;\n          const workerResult = await createWorker({\n            projectRoot: userOptions.projectRoot,\n            workerPath: userOptions.htmlWorkerPath,\n            currentCondition: \"react-server\",\n            reverseCondition: \"react-client\", // HTML worker needs react-client for react-dom/server\n            maxListeners: maxListeners,\n            envPrefix: viteEnvPrefix,\n            logger: logger,\n            workerData: {\n              resolvedConfig: serializeResolvedConfig(resolvedConfig),\n              userOptions: serializedUserOptions,\n              configEnv,\n            },\n          });\n          if (workerResult.type === \"error\") {\n            if (workerResult.error != null) {\n              throw workerResult.error;\n            }\n            throw new Error(\"React static plugin failed to create worker\");\n          } else if (workerResult.type === \"skip\") {\n            logger.info(\"Worker not created, skipping static build\");\n            return;\n          } else {\n            worker = workerResult.worker;\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: \"html\", // This is the HTML worker for server-side static generation\n                startupTime: workerStartupTime,\n                fromMainThread: true,\n                fromRscWorker: false,\n                fromHtmlWorker: false,\n                description: `HTML worker startup for server-side static generation`,\n              });\n              // Only emit metrics from the server environment to prevent duplicates\n              if (this.environment.name === \"server\") {\n                userOptions.onMetrics(workerStartupMetric);\n              }\n            }\n            // Store the worker globally to prevent duplicate creation\n            globalWorker = worker;\n          }\n        }\n\n        // No RSC worker needed for static generation - main thread runs with react-server conditions\n        // Render pages - component resolution now happens per-route in renderPage\n        const { onEvent, ...handlerOptions } = userOptions;\n\n        // Emit the static site generation start event\n        if (typeof userOptions.onEvent === \"function\") {\n          try {\n            const r = userOptions.onEvent({\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: bundle,\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            } else {\n              throw new Error(\"Failed to emit build.ssg.start event\");\n            }\n          }\n        }\n\n        // Select render mode based on build config\n        const renderMode = userOptions.build?.renderMode ?? \"parallel\";\n        const renderPages = renderMode === \"sequential\" ? renderPagesSequential : renderPagesBatched;\n\n        if (userOptions.verbose) {\n          logger.info(`[static] Using ${renderMode} rendering${renderMode === \"parallel\" ? ` (batch size: ${userOptions.build?.batchSize ?? 8})` : \"\"}`);\n        }\n\n        // this will render the routes\n        const renderPagesGenerator = renderPages(\n          routes,\n          {\n            ...handlerOptions,\n            loader: buildLoader,\n            worker: worker,\n            htmlWorker: worker, // Pass the HTML worker for HTML generation\n            logger: logger,\n            // Pass global CSS to downstream renderer\n            globalCss,\n            // Pass abort signal to cancel operations when errors occur\n            signal: AbortSignal.timeout(handlerOptions.htmlTimeout),\n            onEvent: onEvent,\n            serverPipeableStreamOptions: serverPipeableStreamOptions,\n            clientPipeableStreamOptions: clientPipeableStreamOptions,\n            manifest: serverManifest ?? {},\n            staticManifest: staticManifest, // Pass static manifest for path resolution\n            autoDiscoveredFiles: autoDiscoveredFiles!,\n            cssFilesByPage: cssFilesByPage,\n            batchSize: userOptions.build?.batchSize,\n          },\n          renderPage\n        );\n\n        // Process render results\n        let finalResult: RenderPagesResult | undefined;\n        try {\n          for await (const result of renderPagesGenerator) {\n            // Handle error results immediately\n            if (result.type === \"error\") {\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\n              const firstError = result.failedRoutes.values().next().value;\n              if (firstError != null && shouldCausePanic(firstError, { panicThreshold: userOptions.panicThreshold })) {\n                // This should cause a panic, throw the error\n                throw firstError;\n              }\n              // For non-panic errors, log warnings but continue\n              for (const [route, error] of result.failedRoutes) {\n                const err = error instanceof Error ? error : toError(error);\n                this.warn(\n                  new Error(\"Failed to render route: \" + route + \"\\n\" + err.message + \"\\n\" + err.stack, { cause: err })\n                );\n              }\n            }\n\n            finalResult = result;\n          }\n        } catch (renderError) {\n          // Handle render errors with panic threshold logic\n          const renderPanicError = handleError({\n            error: renderError,\n            logger: logger,\n            panicThreshold: userOptions.panicThreshold,\n            context: \"renderPages\",\n          });\n          if (renderPanicError != null) {\n            throw renderPanicError;\n          }\n          throw renderError;\n        }\n\n        if (!finalResult) {\n          throw new Error(\"No render result produced\");\n        }\n        // Calculate duration from timing\n        const duration = Math.round(\n          performance.now() - (timing.renderStart || timing.start)\n        );\n\n        this.info(\n          `Rendered ${finalResult.completedRoutes.size} pages in ${duration}ms`\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: bundle,\n              },\n            });\n            if (r != null && typeof r === \"object\" && \"then\" in r) {\n              await (r as Promise<any>);\n            }\n          } catch (error) {\n            if (error != null) {\n              throw error; // Re-throw to abort the build\n            } else {\n              throw new Error(\"Failed to emit build.ssg.end event\");\n            }\n          }\n        }\n\n        if (process.env[\"NODE_ENV\"] !== \"production\") {\n          this.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      } catch (error) {\n        panicError = handleError({\n          error,\n          logger: logger,\n          panicThreshold: userOptions.panicThreshold,\n          context: \"writeBundle\",\n        });\n\n        // Let the finally block handle additional cleanup\n      } finally {\n        // Reset any cached state to prevent issues in subsequent builds\n        autoDiscoveredFiles = null;\n        serverManifest = undefined;\n        \n        // Clean up worker if it exists\n        if (worker) {\n          try {\n            worker.removeAllListeners();\n            worker.terminate();\n          } catch (terminateError) {\n            // Ignore termination errors\n          }\n          worker = undefined;\n          // Reset global worker since it's been terminated\n          globalWorker = 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    },\n\n    async closeBundle() {\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 (this.environment.name === \"server\") {\n        try {\n          const { 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          // Don't clean up server/_virtual - we need dynamic-import-helper.js for runtime\n          // Only clean up static/_virtual if it exists (shouldn't, but just in case)\n          const staticOutDir = join(resolvedOutDir, userOptions.build.static || \"static\");\n          const staticVirtualDir = join(staticOutDir, \"_virtual\");\n          if (existsSync(staticVirtualDir)) {\n            const { rmSync } = await import(\"node:fs\");\n            rmSync(staticVirtualDir, { recursive: true, force: true });\n            if (userOptions.verbose) {\n              logger?.info(`[plugin.server] Cleaned up _virtual directory: ${staticVirtualDir}`);\n            }\n          }\n        } catch (error) {\n          // Non-critical - log but don't fail the build\n          if (userOptions.verbose) {\n            logger?.warn(`[plugin.server] Failed to clean up _virtual directory: ${error}`);\n          }\n        }\n      }\n\n      // Graceful worker shutdown - only at the end of the entire build process\n      if (worker) {\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)); // 60% of main timeout\n\n              const messageHandler = (message: any) => {\n                if (message.type === \"SHUTDOWN_COMPLETE\") {\n                  if (userOptions.verbose) {\n                    logger.info(\"Worker shutdown complete\");\n                  }\n                  clearTimeout(timeout);\n                  clearTimeout(backupTimeout);\n                  worker?.removeListener(\"message\", messageHandler);\n                  // Remove all other event listeners as well\n                  worker?.removeAllListeners();\n                  resolve();\n                } else if (message.type === \"CLEANUP_COMPLETE\") {\n                  // Handle cleanup complete messages during shutdown - this is normal\n                  if (userOptions.verbose) {\n                    logger.info(\"Worker cleanup completed during shutdown\");\n                  }\n                  // Don't resolve here - wait for SHUTDOWN_COMPLETE\n                } else {\n                  if (userOptions.verbose) {\n                    logger.info(\n                      \"Worker is still busy, received message \" + message?.type\n                    );\n                  }\n                }\n              };\n\n              worker?.on(\"message\", messageHandler);\n\n              // Send shutdown message\n              worker?.postMessage({\n                type: \"SHUTDOWN\",\n                id: \"*\",\n              });\n            }),\n          ]);\n        } catch (error) {\n          // If shutdown protocol fails, force terminate\n          this.warn(\n            \"Worker shutdown protocol failed, forcing termination: \" +\n              (error instanceof Error ? error.message : String(error))\n          );\n          // Don't try to clean up listeners in error case - just force terminate\n        } finally {\n          // Always force cleanup and termination\n          if (worker) {\n            try {\n              worker.removeAllListeners();\n              worker.terminate();\n            } catch (terminateError) {\n              // Ignore termination errors\n            }\n            worker = undefined;\n            // Reset global worker since it's been terminated\n            globalWorker = undefined;\n          }\n        }\n      }\n    },\n  } as const;\n};\n"],"names":["panicError","renderPages","renderPagesSequential","join"],"mappings":"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"}