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