vite-plugin-react-server 1.4.2 → 1.4.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +48 -313
- package/dist/package.json +123 -13
- package/dist/plugin/bundle/deferredStaticGeneration.js +14 -39
- package/dist/plugin/bundle/manifests.js +30 -48
- package/dist/plugin/config/autoDiscover/resolveAutoDiscover.d.ts.map +1 -1
- package/dist/plugin/config/autoDiscover/resolveAutoDiscover.js +4 -1
- package/dist/plugin/config/envPrefixFromConfig.js +12 -7
- package/dist/plugin/config/getCondition.d.ts.map +1 -1
- package/dist/plugin/config/getCondition.js +7 -5
- package/dist/plugin/dev-server/virtualRscHmrPlugin.js +23 -23
- package/dist/plugin/environments/createBuildEventPlugin.js +88 -98
- package/dist/plugin/environments/createEnvironmentPlugin.js +222 -250
- package/dist/plugin/helpers/createRscRenderHelpers.js +33 -34
- package/dist/plugin/helpers/createSharedLoader.d.ts.map +1 -1
- package/dist/plugin/helpers/createSharedLoader.js +4 -2
- package/dist/plugin/helpers/headlessStreamReuseHandler.js +30 -22
- package/dist/plugin/helpers/headlessStreamState.js +15 -28
- package/dist/plugin/helpers/resolveComponent.d.ts.map +1 -1
- package/dist/plugin/helpers/resolveComponent.js +4 -2
- package/dist/plugin/index.client.d.ts +5 -0
- package/dist/plugin/index.client.d.ts.map +1 -0
- package/dist/plugin/index.client.js +4 -0
- package/dist/plugin/index.d.ts +4 -3
- package/dist/plugin/index.d.ts.map +1 -1
- package/dist/plugin/index.js +10 -5
- package/dist/plugin/index.server.d.ts +5 -0
- package/dist/plugin/index.server.d.ts.map +1 -0
- package/dist/plugin/index.server.js +4 -0
- package/dist/plugin/metrics/createWorkerStartupMetrics.js +31 -13
- package/dist/plugin/orchestrator/createPluginOrchestrator.client.js +41 -38
- package/dist/plugin/orchestrator/createPluginOrchestrator.server.js +43 -46
- package/dist/plugin/plugin.client.js +2 -2
- package/dist/plugin/plugin.server.js +2 -2
- package/dist/plugin/react-static/createBuildLoader.client.js +12 -6
- package/dist/plugin/react-static/createBuildLoader.server.js +255 -235
- package/dist/plugin/react-static/plugin.client.js +684 -770
- package/dist/plugin/react-static/plugin.server.js +517 -603
- package/dist/plugin/react-static/processCssFilesForPages.js +103 -88
- package/dist/plugin/react-static/renderPage.client.js +455 -529
- package/dist/plugin/react-static/renderPage.server.js +485 -508
- package/dist/plugin/react-static/renderPagesBatched.js +277 -275
- package/dist/plugin/react-static/rscToHtmlStream.client.js +48 -29
- package/dist/plugin/react-static/rscToHtmlStream.server.js +62 -37
- package/dist/plugin/react-static/temporaryReferences.server.js +11 -2
- package/dist/plugin/stream/createMainThreadHandlers.js +40 -31
- package/dist/plugin/stream/renderRscStream.server.d.ts.map +1 -1
- package/dist/plugin/stream/renderRscStream.server.js +127 -144
- package/dist/plugin/transformer/createTransformerPlugin.js +226 -265
- package/dist/plugin/utils/checkReactVersion.d.ts +7 -0
- package/dist/plugin/utils/checkReactVersion.d.ts.map +1 -0
- package/dist/plugin/utils/checkReactVersion.js +23 -0
- package/dist/plugin/utils/envUrls.node.js +12 -11
- package/dist/plugin/vendor/vendor-alias.js +84 -114
- package/dist/plugin/vendor/vendor.client.d.ts.map +1 -1
- package/dist/plugin/vendor/vendor.client.js +1 -3
- package/dist/plugin/worker/rsc/handleRscRender.d.ts.map +1 -1
- package/dist/plugin/worker/rsc/handleRscRender.js +3 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +123 -13
- package/plugin/config/autoDiscover/resolveAutoDiscover.ts +4 -0
- package/plugin/config/getCondition.ts +6 -4
- package/plugin/helpers/createRscRenderHelpers.ts +1 -1
- package/plugin/helpers/createSharedLoader.ts +6 -1
- package/plugin/helpers/resolveComponent.ts +6 -1
- package/plugin/index.client.ts +4 -0
- package/plugin/index.server.ts +4 -0
- package/plugin/index.ts +12 -5
- package/plugin/plugin.client.ts +1 -1
- package/plugin/plugin.server.ts +1 -1
- package/plugin/stream/renderRscStream.server.ts +3 -0
- package/plugin/utils/checkReactVersion.ts +28 -0
- package/plugin/vendor/vendor.client.ts +0 -2
- package/plugin/worker/rsc/handleRscRender.ts +2 -0
- package/scripts/generate-toc.mjs +27 -294
|
@@ -1,623 +1,537 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* plugin
|
|
3
|
-
*
|
|
4
|
-
*
|
|
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
|
|
17
|
-
import { resolveOptions } from
|
|
18
|
-
import { createBuildLoader } from
|
|
19
|
-
import { renderPagesBatched } from
|
|
20
|
-
import { renderPages
|
|
21
|
-
import { getBundleManifest } from
|
|
22
|
-
import { createWorker } from
|
|
23
|
-
import { serializedOptions, serializeResolvedConfig
|
|
24
|
-
import { performance } from
|
|
25
|
-
import { baseURL } from
|
|
26
|
-
import { handleError } from
|
|
27
|
-
import { shouldCausePanic } from
|
|
28
|
-
import { renderPage } from
|
|
29
|
-
import { temporaryReferences } from
|
|
30
|
-
import { configurePreviewServer } from
|
|
31
|
-
import { envPrefixFromConfig } from
|
|
32
|
-
import { processCssFilesForPages } from
|
|
33
|
-
import { createWorkerStartupMetrics } from
|
|
34
|
-
import { tryManifest } from
|
|
35
|
-
import { join } from
|
|
36
|
-
import { resolveAutoDiscover } from
|
|
37
|
-
import { assertReactServer } from
|
|
38
|
-
import { toError } from
|
|
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
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
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
|
-
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
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
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
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
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
logger = this.environment.logger;
|
|
270
|
+
});
|
|
271
|
+
if (workerResult.type === "error") {
|
|
272
|
+
if (workerResult.error != null) {
|
|
273
|
+
throw workerResult.error;
|
|
105
274
|
}
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
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
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
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
|
-
|
|
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
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
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
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
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
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
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
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
236
|
-
|
|
237
|
-
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
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
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
|
|
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
|
-
|
|
525
|
-
|
|
526
|
-
|
|
527
|
-
|
|
528
|
-
|
|
529
|
-
|
|
530
|
-
|
|
531
|
-
|
|
532
|
-
|
|
533
|
-
|
|
534
|
-
|
|
535
|
-
|
|
536
|
-
|
|
537
|
-
|
|
538
|
-
|
|
539
|
-
|
|
540
|
-
|
|
541
|
-
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
|
|
546
|
-
|
|
547
|
-
|
|
548
|
-
|
|
549
|
-
|
|
550
|
-
|
|
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
|
-
|
|
555
|
-
|
|
556
|
-
|
|
557
|
-
|
|
558
|
-
|
|
559
|
-
|
|
560
|
-
|
|
561
|
-
|
|
562
|
-
|
|
563
|
-
|
|
564
|
-
|
|
565
|
-
|
|
566
|
-
|
|
567
|
-
|
|
568
|
-
|
|
569
|
-
|
|
570
|
-
|
|
571
|
-
|
|
572
|
-
|
|
573
|
-
|
|
574
|
-
|
|
575
|
-
|
|
576
|
-
|
|
577
|
-
|
|
578
|
-
|
|
579
|
-
|
|
580
|
-
|
|
581
|
-
|
|
582
|
-
|
|
583
|
-
|
|
584
|
-
|
|
585
|
-
|
|
586
|
-
|
|
587
|
-
|
|
588
|
-
|
|
589
|
-
|
|
590
|
-
|
|
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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"}
|