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,555 +1,481 @@
|
|
|
1
1
|
/**
|
|
2
|
-
*
|
|
3
|
-
*
|
|
4
|
-
*
|
|
5
|
-
*
|
|
6
|
-
* ARCHITECTURE OVERVIEW:
|
|
7
|
-
*
|
|
8
|
-
* CLIENT-SIDE vs SERVER-SIDE:
|
|
9
|
-
* - Server-side: RSC generation in main thread, HTML generation in worker
|
|
10
|
-
* - Client-side: RSC generation in worker, HTML generation in main thread
|
|
11
|
-
*
|
|
12
|
-
* FLOW:
|
|
13
|
-
* 1. RSC Worker generates RSC content with HTML wrapper
|
|
14
|
-
* 2. RSC content is buffered to allow dual consumption
|
|
15
|
-
* 3. Buffered RSC stream is consumed twice:
|
|
16
|
-
* - For RSC file writing (index.rsc)
|
|
17
|
-
* - For HTML transformation (index.html)
|
|
18
|
-
* 4. HTML transform processes RSC content in main thread
|
|
19
|
-
* 5. Both files are written to filesystem
|
|
20
|
-
*
|
|
21
|
-
* KEY INSIGHT: Node.js streams can only be consumed once, so we buffer the RSC
|
|
22
|
-
* content to allow it to be used for both RSC file generation and HTML transformation.
|
|
23
|
-
* This follows the pattern from collectRscContent.ts.
|
|
24
|
-
*
|
|
25
|
-
* HELPER FUNCTIONS:
|
|
26
|
-
* - createBufferedRscStream: Creates a buffered stream for dual consumption
|
|
27
|
-
* - createRscToHtmlStream: Transforms RSC content to HTML in main thread
|
|
28
|
-
*
|
|
29
|
-
* USAGE:
|
|
30
|
-
* ```typescript
|
|
31
|
-
* const result = await renderPage({
|
|
32
|
-
* route: "/",
|
|
33
|
-
* pagePath: "src/page/page.tsx",
|
|
34
|
-
* // ... other options
|
|
35
|
-
* });
|
|
36
|
-
*
|
|
37
|
-
* // result.html.pipe(htmlFileWriter);
|
|
38
|
-
* // result.rsc.pipe(rscFileWriter);
|
|
39
|
-
* ```
|
|
2
|
+
* vite-plugin-react-server
|
|
3
|
+
* Copyright (c) Nico Brinkkemper
|
|
4
|
+
* MIT License
|
|
40
5
|
*/
|
|
41
|
-
import { createRenderMetrics } from
|
|
42
|
-
import { routeToURL } from
|
|
43
|
-
import { handleError } from
|
|
44
|
-
import { assertNonReactServer } from
|
|
45
|
-
import { createRscStream } from
|
|
46
|
-
import { resolveComponents } from
|
|
47
|
-
import { join } from
|
|
48
|
-
import { createStreamMetrics } from
|
|
49
|
-
import { performance } from
|
|
50
|
-
import { createRscToHtmlStream } from
|
|
6
|
+
import { createRenderMetrics } from '../metrics/createRenderMetrics.js';
|
|
7
|
+
import { routeToURL } from '../utils/routeToURL.js';
|
|
8
|
+
import { handleError } from '../error/handleError.js';
|
|
9
|
+
import { assertNonReactServer } from '../config/getCondition.js';
|
|
10
|
+
import { createRscStream } from '../stream/createRscStream.client.js';
|
|
11
|
+
import { resolveComponents } from '../helpers/resolveComponents.client.js';
|
|
12
|
+
import { join } from 'node:path';
|
|
13
|
+
import { createStreamMetrics } from '../metrics/createStreamMetrics.js';
|
|
14
|
+
import { performance } from 'node:perf_hooks';
|
|
15
|
+
import { createRscToHtmlStream } from './rscToHtmlStream.client.js';
|
|
16
|
+
|
|
51
17
|
assertNonReactServer();
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
18
|
+
const renderPage = async function* _renderPageClient(handlerOptions) {
|
|
19
|
+
if (handlerOptions.verbose) {
|
|
20
|
+
handlerOptions.logger?.info(
|
|
21
|
+
`[renderPage.client] onEvent callback exists: ${!!handlerOptions.onEvent}`
|
|
22
|
+
);
|
|
23
|
+
handlerOptions.logger?.info(
|
|
24
|
+
`[renderPage.client] onMetrics callback exists: ${!!handlerOptions.onMetrics}`
|
|
25
|
+
);
|
|
26
|
+
}
|
|
27
|
+
let hasYielded = false;
|
|
28
|
+
let errorResult = null;
|
|
29
|
+
const wrappedOnEvent = (event) => {
|
|
30
|
+
if (handlerOptions.onEvent) {
|
|
31
|
+
handlerOptions.onEvent(event);
|
|
32
|
+
}
|
|
33
|
+
if (event.type === "route.error" && !hasYielded) {
|
|
34
|
+
hasYielded = true;
|
|
35
|
+
const panicError = handleError({
|
|
36
|
+
error: event.data.error,
|
|
37
|
+
logger: handlerOptions.logger,
|
|
38
|
+
panicThreshold: event.data.panicThreshold,
|
|
39
|
+
context: `route.error (${event.data.route})`
|
|
40
|
+
});
|
|
41
|
+
if (panicError != null) {
|
|
42
|
+
errorResult = {
|
|
43
|
+
type: "error",
|
|
44
|
+
error: panicError,
|
|
45
|
+
metrics: {
|
|
46
|
+
rscHeadless: { duration: 0, chunks: 0, bytes: 0 },
|
|
47
|
+
html: { duration: 0, chunks: 0, bytes: 0 }
|
|
48
|
+
}
|
|
49
|
+
};
|
|
50
|
+
} else {
|
|
51
|
+
errorResult = {
|
|
52
|
+
type: "skip",
|
|
53
|
+
reason: event.data.error.message || "Non-panic error occurred",
|
|
54
|
+
html: { duration: 0, chunks: 0, bytes: 0 },
|
|
55
|
+
rsc: { duration: 0, chunks: 0, bytes: 0 },
|
|
56
|
+
metrics: {
|
|
57
|
+
rscHeadless: { duration: 0, chunks: 0, bytes: 0 },
|
|
58
|
+
html: { duration: 0, chunks: 0, bytes: 0 }
|
|
59
|
+
}
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
};
|
|
64
|
+
if (!handlerOptions.pagePath && !handlerOptions.PageComponent) {
|
|
65
|
+
const emptyStreamWrapper = {
|
|
66
|
+
pipe: (destination) => {
|
|
67
|
+
destination.end();
|
|
68
|
+
return destination;
|
|
69
|
+
},
|
|
70
|
+
abort: () => {
|
|
71
|
+
}
|
|
72
|
+
};
|
|
73
|
+
yield {
|
|
74
|
+
type: "skip",
|
|
75
|
+
reason: "No pagePath and no PageComponent provided",
|
|
76
|
+
html: emptyStreamWrapper,
|
|
77
|
+
rsc: emptyStreamWrapper,
|
|
78
|
+
metrics: {
|
|
79
|
+
rscFull: createRenderMetrics({
|
|
80
|
+
route: handlerOptions.route,
|
|
81
|
+
type: "rsc-full",
|
|
82
|
+
fromMainThread: false,
|
|
83
|
+
fromRscWorker: true,
|
|
84
|
+
fromHtmlWorker: false
|
|
85
|
+
}),
|
|
86
|
+
rscHeadless: createRenderMetrics({
|
|
87
|
+
route: handlerOptions.route,
|
|
88
|
+
type: "rsc-headless",
|
|
89
|
+
fromMainThread: false,
|
|
90
|
+
fromRscWorker: true,
|
|
91
|
+
fromHtmlWorker: false
|
|
92
|
+
}),
|
|
93
|
+
html: createRenderMetrics({
|
|
94
|
+
route: handlerOptions.route,
|
|
95
|
+
type: "html",
|
|
96
|
+
fromMainThread: true,
|
|
97
|
+
fromRscWorker: false,
|
|
98
|
+
fromHtmlWorker: false
|
|
99
|
+
})
|
|
100
|
+
}
|
|
101
|
+
};
|
|
102
|
+
return;
|
|
103
|
+
}
|
|
104
|
+
if (!handlerOptions.url) {
|
|
105
|
+
handlerOptions.url = routeToURL(
|
|
106
|
+
handlerOptions.route,
|
|
107
|
+
handlerOptions.moduleBaseURL,
|
|
108
|
+
handlerOptions.build.rscOutputPath
|
|
109
|
+
);
|
|
110
|
+
}
|
|
111
|
+
const baseDir = join(
|
|
112
|
+
handlerOptions.build.outDir,
|
|
113
|
+
handlerOptions.build.static
|
|
114
|
+
);
|
|
115
|
+
const routePath = handlerOptions.route.replace(/^\//, "");
|
|
116
|
+
const htmlMetrics = createRenderMetrics({
|
|
117
|
+
route: handlerOptions.route,
|
|
118
|
+
type: "html",
|
|
119
|
+
fromMainThread: true,
|
|
120
|
+
// Client: HTML rendered on main thread
|
|
121
|
+
fromRscWorker: false,
|
|
122
|
+
fromHtmlWorker: false,
|
|
123
|
+
baseDir,
|
|
124
|
+
routePath,
|
|
125
|
+
fileName: handlerOptions.build.htmlOutputPath,
|
|
126
|
+
outputPath: join(baseDir, routePath, handlerOptions.build.htmlOutputPath)
|
|
127
|
+
});
|
|
128
|
+
const rscFullMetrics = createRenderMetrics({
|
|
129
|
+
route: handlerOptions.route,
|
|
130
|
+
type: "rsc-full",
|
|
131
|
+
fromMainThread: false,
|
|
132
|
+
fromRscWorker: true,
|
|
133
|
+
// Client: RSC rendered on RSC worker
|
|
134
|
+
fromHtmlWorker: false
|
|
135
|
+
});
|
|
136
|
+
const rscHeadlessMetrics = createRenderMetrics({
|
|
137
|
+
route: handlerOptions.route,
|
|
138
|
+
type: "rsc-headless",
|
|
139
|
+
fromMainThread: false,
|
|
140
|
+
fromRscWorker: true,
|
|
141
|
+
// Client: RSC rendered on RSC worker
|
|
142
|
+
fromHtmlWorker: false,
|
|
143
|
+
baseDir,
|
|
144
|
+
routePath,
|
|
145
|
+
fileName: handlerOptions.build.rscOutputPath,
|
|
146
|
+
outputPath: join(baseDir, routePath, handlerOptions.build.rscOutputPath)
|
|
147
|
+
});
|
|
148
|
+
let headlessRscStream = null;
|
|
149
|
+
let fullRscStream = null;
|
|
150
|
+
let htmlHandler = null;
|
|
151
|
+
try {
|
|
59
152
|
if (handlerOptions.verbose) {
|
|
60
|
-
|
|
61
|
-
|
|
153
|
+
handlerOptions.logger?.info(
|
|
154
|
+
`[renderPage.client] Client-side rendering for route: ${handlerOptions.route}`
|
|
155
|
+
);
|
|
62
156
|
}
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
if (handlerOptions.onEvent) {
|
|
70
|
-
handlerOptions.onEvent(event);
|
|
71
|
-
}
|
|
72
|
-
// Handle route.error events by storing result for later yielding
|
|
73
|
-
if (event.type === "route.error" && !hasYielded) {
|
|
74
|
-
hasYielded = true;
|
|
75
|
-
// Check if this should cause a panic
|
|
76
|
-
const panicError = handleError({
|
|
77
|
-
error: event.data.error,
|
|
78
|
-
logger: handlerOptions.logger,
|
|
79
|
-
panicThreshold: event.data.panicThreshold,
|
|
80
|
-
context: `route.error (${event.data.route})`,
|
|
81
|
-
});
|
|
82
|
-
if (panicError != null) {
|
|
83
|
-
// This is a panic error, store error result
|
|
84
|
-
errorResult = {
|
|
85
|
-
type: "error",
|
|
86
|
-
error: panicError,
|
|
87
|
-
metrics: {
|
|
88
|
-
rscHeadless: { duration: 0, chunks: 0, bytes: 0 },
|
|
89
|
-
html: { duration: 0, chunks: 0, bytes: 0 },
|
|
90
|
-
},
|
|
91
|
-
};
|
|
92
|
-
}
|
|
93
|
-
else {
|
|
94
|
-
// This is a non-panic error, store skip result
|
|
95
|
-
errorResult = {
|
|
96
|
-
type: "skip",
|
|
97
|
-
reason: event.data.error.message || "Non-panic error occurred",
|
|
98
|
-
html: { duration: 0, chunks: 0, bytes: 0 },
|
|
99
|
-
rsc: { duration: 0, chunks: 0, bytes: 0 },
|
|
100
|
-
metrics: {
|
|
101
|
-
rscHeadless: { duration: 0, chunks: 0, bytes: 0 },
|
|
102
|
-
html: { duration: 0, chunks: 0, bytes: 0 },
|
|
103
|
-
},
|
|
104
|
-
};
|
|
105
|
-
}
|
|
106
|
-
}
|
|
157
|
+
const resolvePathWithManifest = (path, manifest2) => {
|
|
158
|
+
const entry = manifest2[path];
|
|
159
|
+
if (entry && entry.file) {
|
|
160
|
+
return entry.file;
|
|
161
|
+
}
|
|
162
|
+
return path;
|
|
107
163
|
};
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
164
|
+
const manifest = handlerOptions.manifest || {};
|
|
165
|
+
const resolvedPagePath = handlerOptions.pagePath ? resolvePathWithManifest(handlerOptions.pagePath, manifest) : void 0;
|
|
166
|
+
const resolvedPropsPath = handlerOptions.propsPath ? resolvePathWithManifest(handlerOptions.propsPath, manifest) : void 0;
|
|
167
|
+
const resolvedRootPath = handlerOptions.rootPath ? resolvePathWithManifest(handlerOptions.rootPath, manifest) : void 0;
|
|
168
|
+
const resolvedHtmlPath = handlerOptions.htmlPath ? resolvePathWithManifest(handlerOptions.htmlPath, manifest) : void 0;
|
|
169
|
+
if (handlerOptions.verbose) {
|
|
170
|
+
handlerOptions.logger?.info(`[renderPage.client] Resolved paths for route ${handlerOptions.route}:`);
|
|
171
|
+
handlerOptions.logger?.info(` page: ${handlerOptions.pagePath} -> ${resolvedPagePath}`);
|
|
172
|
+
handlerOptions.logger?.info(` props: ${handlerOptions.propsPath} -> ${resolvedPropsPath}`);
|
|
173
|
+
handlerOptions.logger?.info(` root: ${handlerOptions.rootPath} -> ${resolvedRootPath}`);
|
|
174
|
+
handlerOptions.logger?.info(` html: ${handlerOptions.htmlPath} -> ${resolvedHtmlPath}`);
|
|
175
|
+
handlerOptions.logger?.info(` manifest keys: ${Object.keys(manifest).join(", ")}`);
|
|
176
|
+
handlerOptions.logger?.info(` HTML path issue: htmlPath='${handlerOptions.htmlPath}', resolved='${resolvedHtmlPath}', manifest has Html entry: ${!!manifest[handlerOptions.htmlPath || ""]}`);
|
|
177
|
+
handlerOptions.logger?.info(` About to pass htmlPath='${resolvedHtmlPath}' to RSC stream`);
|
|
178
|
+
}
|
|
179
|
+
const worker = handlerOptions.worker ?? handlerOptions.rscWorker;
|
|
180
|
+
if (!worker) {
|
|
181
|
+
throw new Error("RSC worker is required for client-side component resolution");
|
|
182
|
+
}
|
|
183
|
+
try {
|
|
184
|
+
await resolveComponents({
|
|
185
|
+
route: handlerOptions.route,
|
|
186
|
+
pagePath: resolvedPagePath,
|
|
187
|
+
propsPath: resolvedPropsPath,
|
|
188
|
+
rootPath: resolvedRootPath,
|
|
189
|
+
htmlPath: resolvedHtmlPath,
|
|
190
|
+
pageExportName: handlerOptions.pageExportName,
|
|
191
|
+
propsExportName: handlerOptions.propsExportName,
|
|
192
|
+
rootExportName: handlerOptions.rootExportName,
|
|
193
|
+
htmlExportName: handlerOptions.htmlExportName,
|
|
194
|
+
worker,
|
|
195
|
+
rscWorker: worker,
|
|
196
|
+
onMetrics: handlerOptions.onMetrics,
|
|
197
|
+
logger: handlerOptions.logger,
|
|
198
|
+
verbose: handlerOptions.verbose
|
|
199
|
+
});
|
|
200
|
+
} catch (componentResolutionError) {
|
|
201
|
+
const error = componentResolutionError instanceof Error ? componentResolutionError : new Error(String(componentResolutionError));
|
|
202
|
+
const panicError = handleError({
|
|
203
|
+
error,
|
|
204
|
+
critical: false,
|
|
205
|
+
logger: handlerOptions.logger,
|
|
206
|
+
panicThreshold: handlerOptions.panicThreshold,
|
|
207
|
+
context: `Component resolution failed for route ${handlerOptions.route}`
|
|
208
|
+
});
|
|
209
|
+
if (panicError) {
|
|
120
210
|
yield {
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
type: "rsc-full",
|
|
129
|
-
fromMainThread: false,
|
|
130
|
-
fromRscWorker: true,
|
|
131
|
-
fromHtmlWorker: false,
|
|
132
|
-
}),
|
|
133
|
-
rscHeadless: createRenderMetrics({
|
|
134
|
-
route: handlerOptions.route,
|
|
135
|
-
type: "rsc-headless",
|
|
136
|
-
fromMainThread: false,
|
|
137
|
-
fromRscWorker: true,
|
|
138
|
-
fromHtmlWorker: false,
|
|
139
|
-
}),
|
|
140
|
-
html: createRenderMetrics({
|
|
141
|
-
route: handlerOptions.route,
|
|
142
|
-
type: "html",
|
|
143
|
-
fromMainThread: true,
|
|
144
|
-
fromRscWorker: false,
|
|
145
|
-
fromHtmlWorker: false,
|
|
146
|
-
}),
|
|
147
|
-
},
|
|
211
|
+
type: "error",
|
|
212
|
+
error: panicError,
|
|
213
|
+
metrics: {
|
|
214
|
+
rscFull: rscFullMetrics,
|
|
215
|
+
rscHeadless: rscHeadlessMetrics,
|
|
216
|
+
html: htmlMetrics
|
|
217
|
+
}
|
|
148
218
|
};
|
|
149
219
|
return;
|
|
220
|
+
}
|
|
221
|
+
handlerOptions.logger?.warn(
|
|
222
|
+
`[renderPage.client] Component resolution failed for route ${handlerOptions.route}, continuing with client-only HTML: ${error.message}`
|
|
223
|
+
);
|
|
224
|
+
const clientOnlyHtmlStreamWrapper = {
|
|
225
|
+
pipe: (destination) => {
|
|
226
|
+
const minimalHtml = `<!DOCTYPE html><html><head><link rel="expect" href="#«R»" blocking="render"/></head><body><div id="root"></div><template id="«R»"></template></body></html>`;
|
|
227
|
+
destination.write(minimalHtml);
|
|
228
|
+
destination.end();
|
|
229
|
+
return destination;
|
|
230
|
+
},
|
|
231
|
+
abort: () => {
|
|
232
|
+
}
|
|
233
|
+
};
|
|
234
|
+
const emptyRscStreamWrapper = {
|
|
235
|
+
pipe: (destination) => {
|
|
236
|
+
destination.end();
|
|
237
|
+
return destination;
|
|
238
|
+
},
|
|
239
|
+
abort: () => {
|
|
240
|
+
}
|
|
241
|
+
};
|
|
242
|
+
yield {
|
|
243
|
+
type: "skip",
|
|
244
|
+
reason: error,
|
|
245
|
+
html: clientOnlyHtmlStreamWrapper,
|
|
246
|
+
rsc: emptyRscStreamWrapper,
|
|
247
|
+
metrics: {
|
|
248
|
+
rscFull: rscFullMetrics,
|
|
249
|
+
rscHeadless: rscHeadlessMetrics,
|
|
250
|
+
html: htmlMetrics
|
|
251
|
+
}
|
|
252
|
+
};
|
|
253
|
+
return;
|
|
150
254
|
}
|
|
151
|
-
|
|
152
|
-
|
|
255
|
+
const newHandlerOptions = {
|
|
256
|
+
...handlerOptions,
|
|
257
|
+
// Pass page paths to the RSC worker so it knows what to render
|
|
258
|
+
pagePath: resolvedPagePath,
|
|
259
|
+
propsPath: resolvedPropsPath,
|
|
260
|
+
rootPath: resolvedRootPath,
|
|
261
|
+
htmlPath: resolvedHtmlPath
|
|
262
|
+
};
|
|
263
|
+
if (handlerOptions.verbose) {
|
|
264
|
+
handlerOptions.logger?.info(
|
|
265
|
+
`[renderPage.client] handlerOptions.clientPipeableStreamOptions: ${JSON.stringify(handlerOptions.clientPipeableStreamOptions)}`
|
|
266
|
+
);
|
|
267
|
+
handlerOptions.logger?.info(
|
|
268
|
+
`[renderPage.client] newHandlerOptions.clientPipeableStreamOptions: ${JSON.stringify(newHandlerOptions.clientPipeableStreamOptions)}`
|
|
269
|
+
);
|
|
270
|
+
handlerOptions.logger?.info(
|
|
271
|
+
`[renderPage.client] newHandlerOptions page paths: pagePath=${newHandlerOptions.pagePath}, propsPath=${newHandlerOptions.propsPath}, rootPath=${newHandlerOptions.rootPath}, htmlPath=${newHandlerOptions.htmlPath}`
|
|
272
|
+
);
|
|
153
273
|
}
|
|
154
|
-
const
|
|
155
|
-
const
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
274
|
+
const uniqueId = handlerOptions.id ?? `${handlerOptions.route}-${Date.now()}-${Math.random().toString(36).substring(2, 11)}`;
|
|
275
|
+
const headlessRscStreamLocal = createRscStream({
|
|
276
|
+
...newHandlerOptions,
|
|
277
|
+
id: `${handlerOptions.route}-headless-${uniqueId}`,
|
|
278
|
+
rscTimeout: handlerOptions.rscTimeout || 5e3,
|
|
279
|
+
onMetrics: handlerOptions.onMetrics,
|
|
280
|
+
// Headless RSC stream: page content only (for .rsc file)
|
|
281
|
+
htmlPath: "",
|
|
282
|
+
// No HTML wrapper - just page content
|
|
283
|
+
pagePath: newHandlerOptions.pagePath || "",
|
|
284
|
+
// Ensure pagePath is always a string
|
|
285
|
+
url: newHandlerOptions.url || "",
|
|
286
|
+
// Ensure url is always a string
|
|
287
|
+
pageProps: newHandlerOptions.pageProps || {},
|
|
288
|
+
// Ensure pageProps is always an object
|
|
289
|
+
onEvent: wrappedOnEvent
|
|
167
290
|
});
|
|
168
|
-
const
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
291
|
+
const fullRscStreamLocal = createRscStream({
|
|
292
|
+
...newHandlerOptions,
|
|
293
|
+
id: `${handlerOptions.route}-full-${uniqueId}`,
|
|
294
|
+
rscTimeout: handlerOptions.rscTimeout || 5e3,
|
|
295
|
+
onMetrics: handlerOptions.onMetrics,
|
|
296
|
+
// Full RSC stream: include HTML wrapper (for HTML generation)
|
|
297
|
+
// Pass through the resolved htmlPath so custom Html components work in client mode
|
|
298
|
+
htmlPath: resolvedHtmlPath,
|
|
299
|
+
pagePath: newHandlerOptions.pagePath || "",
|
|
300
|
+
// Ensure pagePath is always a string
|
|
301
|
+
url: newHandlerOptions.url || "",
|
|
302
|
+
// Ensure url is always a string
|
|
303
|
+
pageProps: newHandlerOptions.pageProps || {},
|
|
304
|
+
// Ensure pageProps is always an object
|
|
305
|
+
// Reuse headless stream elements - the worker will handle this with the unique ID
|
|
306
|
+
reuseHeadlessStreamId: headlessRscStreamLocal.id,
|
|
307
|
+
onEvent: wrappedOnEvent
|
|
174
308
|
});
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
309
|
+
headlessRscStream = headlessRscStreamLocal;
|
|
310
|
+
fullRscStream = fullRscStreamLocal;
|
|
311
|
+
if (handlerOptions.verbose) {
|
|
312
|
+
handlerOptions.logger?.info(
|
|
313
|
+
`[renderPage.client] Creating HTML transform stream with clientPipeableStreamOptions: ${JSON.stringify(newHandlerOptions.clientPipeableStreamOptions)}`
|
|
314
|
+
);
|
|
315
|
+
}
|
|
316
|
+
const htmlTransformStream = createRscToHtmlStream({
|
|
317
|
+
...newHandlerOptions,
|
|
318
|
+
htmlTimeout: handlerOptions.htmlTimeout || 15e3,
|
|
319
|
+
route: handlerOptions.route,
|
|
320
|
+
logger: handlerOptions.logger,
|
|
321
|
+
verbose: handlerOptions.verbose,
|
|
322
|
+
rscStream: fullRscStreamLocal.rscStream
|
|
185
323
|
});
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
return entry.file;
|
|
201
|
-
}
|
|
202
|
-
return path;
|
|
203
|
-
};
|
|
204
|
-
// Use manifest for page component resolution (client version works in reverse)
|
|
205
|
-
const manifest = handlerOptions.manifest || {};
|
|
206
|
-
const resolvedPagePath = handlerOptions.pagePath ? resolvePathWithManifest(handlerOptions.pagePath, manifest) : undefined;
|
|
207
|
-
const resolvedPropsPath = handlerOptions.propsPath ? resolvePathWithManifest(handlerOptions.propsPath, manifest) : undefined;
|
|
208
|
-
const resolvedRootPath = handlerOptions.rootPath ? resolvePathWithManifest(handlerOptions.rootPath, manifest) : undefined;
|
|
209
|
-
const resolvedHtmlPath = handlerOptions.htmlPath ? resolvePathWithManifest(handlerOptions.htmlPath, manifest) : undefined;
|
|
210
|
-
if (handlerOptions.verbose) {
|
|
211
|
-
handlerOptions.logger?.info(`[renderPage.client] Resolved paths for route ${handlerOptions.route}:`);
|
|
212
|
-
handlerOptions.logger?.info(` page: ${handlerOptions.pagePath} -> ${resolvedPagePath}`);
|
|
213
|
-
handlerOptions.logger?.info(` props: ${handlerOptions.propsPath} -> ${resolvedPropsPath}`);
|
|
214
|
-
handlerOptions.logger?.info(` root: ${handlerOptions.rootPath} -> ${resolvedRootPath}`);
|
|
215
|
-
handlerOptions.logger?.info(` html: ${handlerOptions.htmlPath} -> ${resolvedHtmlPath}`);
|
|
216
|
-
handlerOptions.logger?.info(` manifest keys: ${Object.keys(manifest).join(', ')}`);
|
|
217
|
-
handlerOptions.logger?.info(` HTML path issue: htmlPath='${handlerOptions.htmlPath}', resolved='${resolvedHtmlPath}', manifest has Html entry: ${!!manifest[handlerOptions.htmlPath || '']}`);
|
|
218
|
-
handlerOptions.logger?.info(` About to pass htmlPath='${resolvedHtmlPath}' to RSC stream`);
|
|
219
|
-
}
|
|
220
|
-
const worker = handlerOptions.worker ?? handlerOptions.rscWorker;
|
|
221
|
-
// Step 2: Resolve components using the RSC worker with built paths
|
|
222
|
-
// This separates component resolution from RSC generation, making the
|
|
223
|
-
// subsequent RSC render completely synchronous
|
|
224
|
-
if (!worker) {
|
|
225
|
-
throw new Error("RSC worker is required for client-side component resolution");
|
|
226
|
-
}
|
|
227
|
-
// Preload components in the worker for faster subsequent RSC stream generation
|
|
228
|
-
try {
|
|
229
|
-
await resolveComponents({
|
|
230
|
-
route: handlerOptions.route,
|
|
231
|
-
pagePath: resolvedPagePath,
|
|
232
|
-
propsPath: resolvedPropsPath,
|
|
233
|
-
rootPath: resolvedRootPath,
|
|
234
|
-
htmlPath: resolvedHtmlPath,
|
|
235
|
-
pageExportName: handlerOptions.pageExportName,
|
|
236
|
-
propsExportName: handlerOptions.propsExportName,
|
|
237
|
-
rootExportName: handlerOptions.rootExportName,
|
|
238
|
-
htmlExportName: handlerOptions.htmlExportName,
|
|
239
|
-
worker: worker,
|
|
240
|
-
rscWorker: worker,
|
|
241
|
-
onMetrics: handlerOptions.onMetrics,
|
|
242
|
-
logger: handlerOptions.logger,
|
|
243
|
-
verbose: handlerOptions.verbose,
|
|
244
|
-
});
|
|
245
|
-
}
|
|
246
|
-
catch (componentResolutionError) {
|
|
247
|
-
// Handle component resolution failures gracefully
|
|
248
|
-
const error = componentResolutionError instanceof Error
|
|
249
|
-
? componentResolutionError
|
|
250
|
-
: new Error(String(componentResolutionError));
|
|
251
|
-
// Check if this component resolution error should cause a panic based on panicThreshold
|
|
252
|
-
const panicError = handleError({
|
|
253
|
-
error,
|
|
254
|
-
critical: false,
|
|
255
|
-
logger: handlerOptions.logger,
|
|
256
|
-
panicThreshold: handlerOptions.panicThreshold,
|
|
257
|
-
context: `Component resolution failed for route ${handlerOptions.route}`,
|
|
258
|
-
});
|
|
259
|
-
// If this should cause a panic, yield error and return
|
|
260
|
-
if (panicError) {
|
|
261
|
-
yield {
|
|
262
|
-
type: "error",
|
|
263
|
-
error: panicError,
|
|
264
|
-
metrics: {
|
|
265
|
-
rscFull: rscFullMetrics,
|
|
266
|
-
rscHeadless: rscHeadlessMetrics,
|
|
267
|
-
html: htmlMetrics,
|
|
268
|
-
},
|
|
269
|
-
};
|
|
270
|
-
return;
|
|
271
|
-
}
|
|
272
|
-
// Otherwise, treat this as a non-critical error and continue with client-only HTML
|
|
273
|
-
// This allows the build to complete with a client-only page
|
|
274
|
-
handlerOptions.logger?.warn(`[renderPage.client] Component resolution failed for route ${handlerOptions.route}, continuing with client-only HTML: ${error.message}`);
|
|
275
|
-
// Create a client-only HTML stream wrapper with minimal HTML
|
|
276
|
-
const clientOnlyHtmlStreamWrapper = {
|
|
277
|
-
pipe: (destination) => {
|
|
278
|
-
// Write a minimal client-only HTML structure
|
|
279
|
-
const minimalHtml = `<!DOCTYPE html><html><head><link rel="expect" href="#«R»" blocking="render"/></head><body><div id="root"></div><template id="«R»"></template></body></html>`;
|
|
280
|
-
destination.write(minimalHtml);
|
|
281
|
-
destination.end();
|
|
282
|
-
return destination;
|
|
283
|
-
},
|
|
284
|
-
abort: () => {
|
|
285
|
-
// No cleanup needed for simple HTML string
|
|
286
|
-
},
|
|
287
|
-
};
|
|
288
|
-
// Create an empty RSC stream wrapper
|
|
289
|
-
const emptyRscStreamWrapper = {
|
|
290
|
-
pipe: (destination) => {
|
|
291
|
-
// No RSC content for failed component resolution
|
|
292
|
-
destination.end();
|
|
293
|
-
return destination;
|
|
294
|
-
},
|
|
295
|
-
abort: () => {
|
|
296
|
-
// No cleanup needed
|
|
297
|
-
},
|
|
298
|
-
};
|
|
299
|
-
// Yield skip result with client-only HTML and empty RSC
|
|
300
|
-
yield {
|
|
301
|
-
type: "skip",
|
|
302
|
-
reason: error,
|
|
303
|
-
html: clientOnlyHtmlStreamWrapper,
|
|
304
|
-
rsc: emptyRscStreamWrapper,
|
|
305
|
-
metrics: {
|
|
306
|
-
rscFull: rscFullMetrics,
|
|
307
|
-
rscHeadless: rscHeadlessMetrics,
|
|
308
|
-
html: htmlMetrics,
|
|
309
|
-
},
|
|
310
|
-
};
|
|
311
|
-
return;
|
|
312
|
-
}
|
|
313
|
-
// Step 2: Create handler options
|
|
314
|
-
// Components are now preloaded in the worker, so we can use the original handler options
|
|
315
|
-
const newHandlerOptions = {
|
|
316
|
-
...handlerOptions,
|
|
317
|
-
// Pass page paths to the RSC worker so it knows what to render
|
|
318
|
-
pagePath: resolvedPagePath,
|
|
319
|
-
propsPath: resolvedPropsPath,
|
|
320
|
-
rootPath: resolvedRootPath,
|
|
321
|
-
htmlPath: resolvedHtmlPath,
|
|
322
|
-
};
|
|
323
|
-
if (handlerOptions.verbose) {
|
|
324
|
-
handlerOptions.logger?.info(`[renderPage.client] handlerOptions.clientPipeableStreamOptions: ${JSON.stringify(handlerOptions.clientPipeableStreamOptions)}`);
|
|
325
|
-
handlerOptions.logger?.info(`[renderPage.client] newHandlerOptions.clientPipeableStreamOptions: ${JSON.stringify(newHandlerOptions.clientPipeableStreamOptions)}`);
|
|
326
|
-
handlerOptions.logger?.info(`[renderPage.client] newHandlerOptions page paths: pagePath=${newHandlerOptions.pagePath}, propsPath=${newHandlerOptions.propsPath}, rootPath=${newHandlerOptions.rootPath}, htmlPath=${newHandlerOptions.htmlPath}`);
|
|
327
|
-
}
|
|
328
|
-
// Component resolution is already measured in resolveComponents
|
|
329
|
-
// No need to measure module resolution time here anymore
|
|
330
|
-
// Create headless RSC stream first (for .rsc file)
|
|
331
|
-
const uniqueId = handlerOptions.id ?? `${handlerOptions.route}-${Date.now()}-${Math.random().toString(36).substring(2, 11)}`;
|
|
332
|
-
const headlessRscStreamLocal = createRscStream({
|
|
333
|
-
...newHandlerOptions,
|
|
334
|
-
id: `${handlerOptions.route}-headless-${uniqueId}`,
|
|
335
|
-
rscTimeout: handlerOptions.rscTimeout || 5000,
|
|
336
|
-
onMetrics: handlerOptions.onMetrics,
|
|
337
|
-
// Headless RSC stream: page content only (for .rsc file)
|
|
338
|
-
htmlPath: '', // No HTML wrapper - just page content
|
|
339
|
-
pagePath: newHandlerOptions.pagePath || '', // Ensure pagePath is always a string
|
|
340
|
-
url: newHandlerOptions.url || '', // Ensure url is always a string
|
|
341
|
-
pageProps: newHandlerOptions.pageProps || {}, // Ensure pageProps is always an object
|
|
342
|
-
onEvent: wrappedOnEvent,
|
|
324
|
+
htmlHandler = {
|
|
325
|
+
htmlStream: htmlTransformStream,
|
|
326
|
+
abort: () => {
|
|
327
|
+
htmlTransformStream.abort();
|
|
328
|
+
}
|
|
329
|
+
};
|
|
330
|
+
const rscStreamWrapper = {
|
|
331
|
+
pipe: (destination) => {
|
|
332
|
+
const streamMetrics = createStreamMetrics();
|
|
333
|
+
streamMetrics.startTime = performance.now();
|
|
334
|
+
const rscFileStream = headlessRscStream.rscStream;
|
|
335
|
+
rscFileStream.on("data", (chunk) => {
|
|
336
|
+
streamMetrics.chunks++;
|
|
337
|
+
streamMetrics.bytes += chunk.length;
|
|
343
338
|
});
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
351
|
-
|
|
352
|
-
htmlPath: resolvedHtmlPath,
|
|
353
|
-
pagePath: newHandlerOptions.pagePath || '', // Ensure pagePath is always a string
|
|
354
|
-
url: newHandlerOptions.url || '', // Ensure url is always a string
|
|
355
|
-
pageProps: newHandlerOptions.pageProps || {}, // Ensure pageProps is always an object
|
|
356
|
-
// Reuse headless stream elements - the worker will handle this with the unique ID
|
|
357
|
-
reuseHeadlessStreamId: headlessRscStreamLocal.id,
|
|
358
|
-
onEvent: wrappedOnEvent,
|
|
339
|
+
rscFileStream.on("end", () => {
|
|
340
|
+
streamMetrics.duration = performance.now() - streamMetrics.startTime;
|
|
341
|
+
streamMetrics.endTime = performance.now();
|
|
342
|
+
rscHeadlessMetrics.streamMetrics = streamMetrics;
|
|
343
|
+
rscHeadlessMetrics.chunkRate = streamMetrics.chunks / (streamMetrics.duration / 1e3);
|
|
344
|
+
rscHeadlessMetrics.processingTime = streamMetrics.duration;
|
|
345
|
+
rscHeadlessMetrics.memoryUsage = process.memoryUsage();
|
|
346
|
+
rscHeadlessMetrics.chunks = streamMetrics.chunks;
|
|
359
347
|
});
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
|
|
364
|
-
|
|
365
|
-
|
|
348
|
+
rscFileStream.pipe(destination);
|
|
349
|
+
return destination;
|
|
350
|
+
},
|
|
351
|
+
abort: () => headlessRscStream.abort()
|
|
352
|
+
};
|
|
353
|
+
const htmlStreamWrapper = {
|
|
354
|
+
pipe: (destination) => {
|
|
366
355
|
if (handlerOptions.verbose) {
|
|
367
|
-
|
|
356
|
+
handlerOptions.logger?.info(
|
|
357
|
+
`[renderPage.client] Piping HTML stream to destination for route: ${handlerOptions.route}`
|
|
358
|
+
);
|
|
368
359
|
}
|
|
369
|
-
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
|
|
374
|
-
|
|
375
|
-
verbose: handlerOptions.verbose,
|
|
376
|
-
rscStream: fullRscStreamLocal.rscStream,
|
|
377
|
-
});
|
|
378
|
-
htmlHandler = {
|
|
379
|
-
htmlStream: htmlTransformStream,
|
|
380
|
-
abort: () => {
|
|
381
|
-
htmlTransformStream.abort();
|
|
382
|
-
}
|
|
383
|
-
};
|
|
384
|
-
// Create stream wrappers for file writing
|
|
385
|
-
const rscStreamWrapper = {
|
|
386
|
-
pipe: (destination) => {
|
|
387
|
-
const streamMetrics = createStreamMetrics();
|
|
388
|
-
streamMetrics.startTime = performance.now();
|
|
389
|
-
// Use the headless RSC stream directly for the .rsc file
|
|
390
|
-
const rscFileStream = headlessRscStream.rscStream;
|
|
391
|
-
rscFileStream.on("data", (chunk) => {
|
|
392
|
-
streamMetrics.chunks++;
|
|
393
|
-
streamMetrics.bytes += chunk.length;
|
|
394
|
-
});
|
|
395
|
-
rscFileStream.on("end", () => {
|
|
396
|
-
streamMetrics.duration = performance.now() - streamMetrics.startTime;
|
|
397
|
-
streamMetrics.endTime = performance.now();
|
|
398
|
-
rscHeadlessMetrics.streamMetrics = streamMetrics;
|
|
399
|
-
rscHeadlessMetrics.chunkRate = streamMetrics.chunks / (streamMetrics.duration / 1000);
|
|
400
|
-
rscHeadlessMetrics.processingTime = streamMetrics.duration;
|
|
401
|
-
rscHeadlessMetrics.memoryUsage = process.memoryUsage();
|
|
402
|
-
rscHeadlessMetrics.chunks = streamMetrics.chunks;
|
|
403
|
-
});
|
|
404
|
-
rscFileStream.pipe(destination);
|
|
405
|
-
return destination;
|
|
406
|
-
},
|
|
407
|
-
abort: () => headlessRscStream.abort(),
|
|
408
|
-
};
|
|
409
|
-
const htmlStreamWrapper = {
|
|
410
|
-
pipe: (destination) => {
|
|
411
|
-
if (handlerOptions.verbose) {
|
|
412
|
-
handlerOptions.logger?.info(`[renderPage.client] Piping HTML stream to destination for route: ${handlerOptions.route}`);
|
|
413
|
-
}
|
|
414
|
-
// Use the HTML transform stream's pipe method directly (same as server side)
|
|
415
|
-
return htmlTransformStream.pipe(destination);
|
|
416
|
-
},
|
|
417
|
-
abort: () => {
|
|
418
|
-
fullRscStream.abort();
|
|
419
|
-
if (htmlHandler.abort) {
|
|
420
|
-
htmlHandler.abort();
|
|
421
|
-
}
|
|
422
|
-
},
|
|
423
|
-
on: (event, listener) => {
|
|
424
|
-
// Forward error events from the HTML transform stream to the wrapper
|
|
425
|
-
if (event === 'error') {
|
|
426
|
-
// Access the actual stream from the transform result
|
|
427
|
-
const htmlStream = htmlTransformStream.htmlStream;
|
|
428
|
-
if (htmlStream && typeof htmlStream.on === 'function') {
|
|
429
|
-
htmlStream.on('error', listener);
|
|
430
|
-
}
|
|
431
|
-
}
|
|
432
|
-
return htmlStreamWrapper;
|
|
433
|
-
},
|
|
434
|
-
};
|
|
435
|
-
// Don't emit initial metrics - wait for file writes to complete
|
|
436
|
-
// The onMetrics callback will be called after both file.write.done events
|
|
437
|
-
// Check if we have an error result to yield (with timeout protection)
|
|
438
|
-
// Wait a short time for any pending route.error events
|
|
439
|
-
await new Promise(resolve => setTimeout(resolve, 100));
|
|
440
|
-
if (errorResult) {
|
|
441
|
-
yield errorResult;
|
|
442
|
-
return;
|
|
360
|
+
return htmlTransformStream.pipe(destination);
|
|
361
|
+
},
|
|
362
|
+
abort: () => {
|
|
363
|
+
fullRscStream.abort();
|
|
364
|
+
if (htmlHandler.abort) {
|
|
365
|
+
htmlHandler.abort();
|
|
443
366
|
}
|
|
444
|
-
|
|
445
|
-
|
|
446
|
-
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
|
|
451
|
-
|
|
452
|
-
|
|
453
|
-
|
|
367
|
+
},
|
|
368
|
+
on: (event, listener) => {
|
|
369
|
+
if (event === "error") {
|
|
370
|
+
const htmlStream = htmlTransformStream.htmlStream;
|
|
371
|
+
if (htmlStream && typeof htmlStream.on === "function") {
|
|
372
|
+
htmlStream.on("error", listener);
|
|
373
|
+
}
|
|
374
|
+
}
|
|
375
|
+
return htmlStreamWrapper;
|
|
376
|
+
}
|
|
377
|
+
};
|
|
378
|
+
await new Promise((resolve) => setTimeout(resolve, 100));
|
|
379
|
+
if (errorResult) {
|
|
380
|
+
yield errorResult;
|
|
381
|
+
return;
|
|
454
382
|
}
|
|
455
|
-
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
459
|
-
|
|
460
|
-
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
383
|
+
yield {
|
|
384
|
+
type: "success",
|
|
385
|
+
html: htmlStreamWrapper,
|
|
386
|
+
rsc: rscStreamWrapper,
|
|
387
|
+
metrics: {
|
|
388
|
+
rscFull: rscFullMetrics,
|
|
389
|
+
rscHeadless: rscHeadlessMetrics,
|
|
390
|
+
html: htmlMetrics
|
|
391
|
+
}
|
|
392
|
+
};
|
|
393
|
+
} catch (error) {
|
|
394
|
+
try {
|
|
395
|
+
if (headlessRscStream) headlessRscStream.abort();
|
|
396
|
+
if (fullRscStream) fullRscStream.abort();
|
|
397
|
+
if (htmlHandler?.abort) htmlHandler.abort();
|
|
398
|
+
} catch (cleanupError) {
|
|
399
|
+
handlerOptions.logger?.warn(`Failed to cleanup streams on error: ${cleanupError}`);
|
|
400
|
+
}
|
|
401
|
+
const panicError = handleError({
|
|
402
|
+
error,
|
|
403
|
+
logger: handlerOptions.logger,
|
|
404
|
+
panicThreshold: handlerOptions.panicThreshold
|
|
405
|
+
});
|
|
406
|
+
if (panicError != null) {
|
|
407
|
+
yield {
|
|
408
|
+
type: "error",
|
|
409
|
+
error: panicError,
|
|
410
|
+
metrics: {
|
|
411
|
+
rscFull: rscFullMetrics,
|
|
412
|
+
rscHeadless: rscHeadlessMetrics,
|
|
413
|
+
html: htmlMetrics
|
|
464
414
|
}
|
|
465
|
-
|
|
466
|
-
|
|
415
|
+
};
|
|
416
|
+
} else {
|
|
417
|
+
const fallbackRscStream = createRscStream({
|
|
418
|
+
...handlerOptions,
|
|
419
|
+
url: `${handlerOptions.url}`,
|
|
420
|
+
route: `${handlerOptions.route}`,
|
|
421
|
+
cssFiles: handlerOptions.cssFiles || /* @__PURE__ */ new Map(),
|
|
422
|
+
globalCss: handlerOptions.globalCss || /* @__PURE__ */ new Map(),
|
|
423
|
+
id: `${handlerOptions.route}-fallback-${Date.now()}`,
|
|
424
|
+
rscTimeout: handlerOptions.rscTimeout || 5e3,
|
|
425
|
+
onMetrics: handlerOptions.onMetrics,
|
|
426
|
+
// Use React.Fragment as fallback (same as server environment)
|
|
427
|
+
pagePath: "",
|
|
428
|
+
// This will cause the default page to be used, but we'll override it
|
|
429
|
+
pageProps: {}
|
|
430
|
+
// Ensure pageProps is always an object
|
|
431
|
+
});
|
|
432
|
+
const fallbackHtmlStream = createRscToHtmlStream({
|
|
433
|
+
id: handlerOptions.id,
|
|
434
|
+
route: handlerOptions.route,
|
|
435
|
+
url: handlerOptions.url,
|
|
436
|
+
moduleRootPath: handlerOptions.moduleRootPath,
|
|
437
|
+
moduleBasePath: handlerOptions.moduleBasePath,
|
|
438
|
+
moduleBaseURL: handlerOptions.moduleBaseURL,
|
|
439
|
+
projectRoot: handlerOptions.projectRoot,
|
|
440
|
+
panicThreshold: handlerOptions.panicThreshold,
|
|
441
|
+
verbose: handlerOptions.verbose,
|
|
442
|
+
signal: handlerOptions.signal,
|
|
443
|
+
logger: handlerOptions.logger,
|
|
444
|
+
htmlTimeout: handlerOptions.htmlTimeout,
|
|
445
|
+
clientPipeableStreamOptions: handlerOptions.clientPipeableStreamOptions,
|
|
446
|
+
onMetrics: handlerOptions.onMetrics,
|
|
447
|
+
build: handlerOptions.build
|
|
448
|
+
});
|
|
449
|
+
const clientOnlyHtmlStreamWrapper = {
|
|
450
|
+
pipe: (destination) => {
|
|
451
|
+
return fallbackHtmlStream.pipe(destination);
|
|
452
|
+
},
|
|
453
|
+
abort: () => {
|
|
454
|
+
fallbackRscStream.abort();
|
|
467
455
|
}
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
}
|
|
474
|
-
|
|
475
|
-
yield {
|
|
476
|
-
type: "error",
|
|
477
|
-
error: panicError,
|
|
478
|
-
metrics: {
|
|
479
|
-
rscFull: rscFullMetrics,
|
|
480
|
-
rscHeadless: rscHeadlessMetrics,
|
|
481
|
-
html: htmlMetrics,
|
|
482
|
-
},
|
|
483
|
-
};
|
|
456
|
+
};
|
|
457
|
+
const emptyRscStreamWrapper = {
|
|
458
|
+
pipe: (destination) => {
|
|
459
|
+
destination.end();
|
|
460
|
+
return destination;
|
|
461
|
+
},
|
|
462
|
+
abort: () => {
|
|
484
463
|
}
|
|
485
|
-
|
|
486
|
-
|
|
487
|
-
|
|
488
|
-
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
493
|
-
|
|
494
|
-
|
|
495
|
-
id: `${handlerOptions.route}-fallback-${Date.now()}`,
|
|
496
|
-
rscTimeout: handlerOptions.rscTimeout || 5000,
|
|
497
|
-
onMetrics: handlerOptions.onMetrics,
|
|
498
|
-
// Use React.Fragment as fallback (same as server environment)
|
|
499
|
-
pagePath: '', // This will cause the default page to be used, but we'll override it
|
|
500
|
-
pageProps: {}, // Ensure pageProps is always an object
|
|
501
|
-
});
|
|
502
|
-
// Create HTML stream that processes the fallback RSC stream to ensure performance timing script is injected
|
|
503
|
-
const fallbackHtmlStream = createRscToHtmlStream({
|
|
504
|
-
id: handlerOptions.id,
|
|
505
|
-
route: handlerOptions.route,
|
|
506
|
-
url: handlerOptions.url,
|
|
507
|
-
moduleRootPath: handlerOptions.moduleRootPath,
|
|
508
|
-
moduleBasePath: handlerOptions.moduleBasePath,
|
|
509
|
-
moduleBaseURL: handlerOptions.moduleBaseURL,
|
|
510
|
-
projectRoot: handlerOptions.projectRoot,
|
|
511
|
-
panicThreshold: handlerOptions.panicThreshold,
|
|
512
|
-
verbose: handlerOptions.verbose,
|
|
513
|
-
signal: handlerOptions.signal,
|
|
514
|
-
logger: handlerOptions.logger,
|
|
515
|
-
htmlTimeout: handlerOptions.htmlTimeout,
|
|
516
|
-
clientPipeableStreamOptions: handlerOptions.clientPipeableStreamOptions,
|
|
517
|
-
onMetrics: handlerOptions.onMetrics,
|
|
518
|
-
build: handlerOptions.build,
|
|
519
|
-
});
|
|
520
|
-
// Create a wrapper that pipes the fallback RSC stream through the HTML transform
|
|
521
|
-
const clientOnlyHtmlStreamWrapper = {
|
|
522
|
-
pipe: (destination) => {
|
|
523
|
-
// Pipe the fallback RSC stream through the HTML transform to ensure performance timing script is injected
|
|
524
|
-
return fallbackHtmlStream.pipe(destination);
|
|
525
|
-
},
|
|
526
|
-
abort: () => {
|
|
527
|
-
// Clean up the fallback RSC stream
|
|
528
|
-
fallbackRscStream.abort();
|
|
529
|
-
},
|
|
530
|
-
};
|
|
531
|
-
// Create an empty RSC stream wrapper
|
|
532
|
-
const emptyRscStreamWrapper = {
|
|
533
|
-
pipe: (destination) => {
|
|
534
|
-
// No RSC content for skipped routes
|
|
535
|
-
destination.end();
|
|
536
|
-
return destination;
|
|
537
|
-
},
|
|
538
|
-
abort: () => {
|
|
539
|
-
// No cleanup needed
|
|
540
|
-
},
|
|
541
|
-
};
|
|
542
|
-
yield {
|
|
543
|
-
type: "skip",
|
|
544
|
-
reason: error,
|
|
545
|
-
html: clientOnlyHtmlStreamWrapper,
|
|
546
|
-
rsc: emptyRscStreamWrapper,
|
|
547
|
-
metrics: {
|
|
548
|
-
rscFull: rscFullMetrics,
|
|
549
|
-
rscHeadless: rscHeadlessMetrics,
|
|
550
|
-
html: htmlMetrics,
|
|
551
|
-
},
|
|
552
|
-
};
|
|
464
|
+
};
|
|
465
|
+
yield {
|
|
466
|
+
type: "skip",
|
|
467
|
+
reason: error,
|
|
468
|
+
html: clientOnlyHtmlStreamWrapper,
|
|
469
|
+
rsc: emptyRscStreamWrapper,
|
|
470
|
+
metrics: {
|
|
471
|
+
rscFull: rscFullMetrics,
|
|
472
|
+
rscHeadless: rscHeadlessMetrics,
|
|
473
|
+
html: htmlMetrics
|
|
553
474
|
}
|
|
475
|
+
};
|
|
554
476
|
}
|
|
477
|
+
}
|
|
555
478
|
};
|
|
479
|
+
|
|
480
|
+
export { renderPage };
|
|
481
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"renderPage.client.js","sources":["../../../plugin/react-static/renderPage.client.ts"],"sourcesContent":["/**\n * renderPage.client.ts\n *\n * PURPOSE: Client-side static page rendering for React Server Components\n *\n * ARCHITECTURE OVERVIEW:\n * \n * CLIENT-SIDE vs SERVER-SIDE:\n * - Server-side: RSC generation in main thread, HTML generation in worker\n * - Client-side: RSC generation in worker, HTML generation in main thread\n * \n * FLOW:\n * 1. RSC Worker generates RSC content with HTML wrapper\n * 2. RSC content is buffered to allow dual consumption\n * 3. Buffered RSC stream is consumed twice:\n *    - For RSC file writing (index.rsc)\n *    - For HTML transformation (index.html)\n * 4. HTML transform processes RSC content in main thread\n * 5. Both files are written to filesystem\n * \n * KEY INSIGHT: Node.js streams can only be consumed once, so we buffer the RSC\n * content to allow it to be used for both RSC file generation and HTML transformation.\n * This follows the pattern from collectRscContent.ts.\n * \n * HELPER FUNCTIONS:\n * - createBufferedRscStream: Creates a buffered stream for dual consumption\n * - createRscToHtmlStream: Transforms RSC content to HTML in main thread\n * \n * USAGE:\n * ```typescript\n * const result = await renderPage({\n *   route: \"/\",\n *   pagePath: \"src/page/page.tsx\",\n *   // ... other options\n * });\n * \n * // result.html.pipe(htmlFileWriter);\n * // result.rsc.pipe(rscFileWriter);\n * ```\n */\n\nimport { createRenderMetrics } from \"../metrics/createRenderMetrics.js\";\nimport type { RenderMetrics } from \"../metrics/types.js\";\nimport { routeToURL } from \"../utils/routeToURL.js\";\nimport type { RenderPageFn } from \"./types.js\";\nimport { handleError } from \"../error/handleError.js\";\nimport { assertNonReactServer } from \"../config/getCondition.js\";\n\nimport { createRscStream } from \"../stream/createRscStream.client.js\";\nimport { resolveComponents } from \"../helpers/resolveComponents.client.js\";\n\nimport { join } from \"node:path\";\n\nimport { createStreamMetrics } from \"../metrics/createStreamMetrics.js\";\nimport { performance } from \"node:perf_hooks\";\nimport { createRscToHtmlStream } from \"./rscToHtmlStream.client.js\";\n\n\n\nassertNonReactServer();\n\n/**\n * Client version of renderPage that uses the react-client pattern\n * This works in REVERSE from the server plugin:\n * - Server: Main thread (RSC) + HTML worker (HTML)\n * - Client: RSC worker (RSC) + Main thread (HTML)\n */\nexport const renderPage: RenderPageFn = async function* _renderPageClient(\n  handlerOptions\n) {\n  if (handlerOptions.verbose) {\n    handlerOptions.logger?.info(\n      `[renderPage.client] onEvent callback exists: ${!!handlerOptions.onEvent}`\n    );\n    handlerOptions.logger?.info(\n      `[renderPage.client] onMetrics callback exists: ${!!handlerOptions.onMetrics}`\n    );\n  }\n\n  // Track if we've yielded a result to prevent multiple yields\n  let hasYielded = false;\n  let errorResult: any = null;\n\n  // Create a wrapper around onEvent to handle route.error events\n  const wrappedOnEvent = (event: any) => {\n    // Call the original onEvent first\n    if (handlerOptions.onEvent) {\n      handlerOptions.onEvent(event);\n    }\n    \n    // Handle route.error events by storing result for later yielding\n    if (event.type === \"route.error\" && !hasYielded) {\n      hasYielded = true;\n      \n      // Check if this should cause a panic\n      const panicError = handleError({\n        error: event.data.error,\n        logger: handlerOptions.logger,\n        panicThreshold: event.data.panicThreshold,\n        context: `route.error (${event.data.route})`,\n      });\n      \n      if (panicError != null) {\n        // This is a panic error, store error result\n        errorResult = {\n          type: \"error\",\n          error: panicError,\n          metrics: {\n            rscHeadless: { duration: 0, chunks: 0, bytes: 0 },\n            html: { duration: 0, chunks: 0, bytes: 0 },\n          },\n        };\n      } else {\n        // This is a non-panic error, store skip result\n        errorResult = {\n          type: \"skip\",\n          reason: event.data.error.message || \"Non-panic error occurred\",\n          html: { duration: 0, chunks: 0, bytes: 0 },\n          rsc: { duration: 0, chunks: 0, bytes: 0 },\n          metrics: {\n            rscHeadless: { duration: 0, chunks: 0, bytes: 0 },\n            html: { duration: 0, chunks: 0, bytes: 0 },\n          },\n        };\n      }\n    }\n  };\n\n  // Skip if no pagePath AND no PageComponent provided (fallback case)\n  if (!handlerOptions.pagePath && !handlerOptions.PageComponent) {\n    // Create empty stream wrappers for skip case\n    const emptyStreamWrapper = {\n      pipe: <Writable extends NodeJS.WritableStream>(destination: Writable) => {\n        destination.end();\n        return destination;\n      },\n      abort: () => {\n        // No cleanup needed\n      },\n    };\n\n    yield {\n      type: \"skip\",\n      reason: \"No pagePath and no PageComponent provided\",\n      html: emptyStreamWrapper,\n      rsc: emptyStreamWrapper,\n      metrics: {\n        rscFull: createRenderMetrics({\n          route: handlerOptions.route,\n          type: \"rsc-full\",\n          fromMainThread: false,\n          fromRscWorker: true,\n          fromHtmlWorker: false,\n        }) as RenderMetrics & { type: \"rsc-full\" },\n        rscHeadless: createRenderMetrics({\n          route: handlerOptions.route,\n          type: \"rsc-headless\",\n          fromMainThread: false,\n          fromRscWorker: true,\n          fromHtmlWorker: false,\n        }) as RenderMetrics & { type: \"rsc-headless\" },\n        html: createRenderMetrics({\n          route: handlerOptions.route,\n          type: \"html\",\n          fromMainThread: true,\n          fromRscWorker: false,\n          fromHtmlWorker: false,\n        }) as RenderMetrics & { type: \"html\" },\n      },\n    };\n    return;\n  }\n\n  if (!handlerOptions.url) {\n    handlerOptions.url = routeToURL(\n      handlerOptions.route,\n      handlerOptions.moduleBaseURL,\n      handlerOptions.build.rscOutputPath\n    );\n  }\n\n  const baseDir = join(\n    handlerOptions.build.outDir,\n    handlerOptions.build.static\n  );\n  const routePath = handlerOptions.route.replace(/^\\//, \"\");\n\n  // Create metrics upfront with proper types - REVERSE from server\n  const htmlMetrics = createRenderMetrics({\n    route: handlerOptions.route,\n    type: \"html\",\n    fromMainThread: true, // Client: HTML rendered on main thread\n    fromRscWorker: false,\n    fromHtmlWorker: false,\n    baseDir,\n    routePath,\n    fileName: handlerOptions.build.htmlOutputPath,\n    outputPath: join(baseDir, routePath, handlerOptions.build.htmlOutputPath),\n  });\n  \n  const rscFullMetrics = createRenderMetrics({\n    route: handlerOptions.route,\n    type: \"rsc-full\",\n    fromMainThread: false,\n    fromRscWorker: true, // Client: RSC rendered on RSC worker\n    fromHtmlWorker: false,\n  });\n  \n  const rscHeadlessMetrics = createRenderMetrics({\n    route: handlerOptions.route,\n    type: \"rsc-headless\",\n    fromMainThread: false,\n    fromRscWorker: true, // Client: RSC rendered on RSC worker\n    fromHtmlWorker: false,\n    baseDir,\n    routePath,\n    fileName: handlerOptions.build.rscOutputPath,\n    outputPath: join(baseDir, routePath, handlerOptions.build.rscOutputPath),\n  });\n\n  // Declare variables outside try block so they can be accessed in catch block\n  let headlessRscStream: any = null;\n  let fullRscStream: any = null;\n  let htmlHandler: any = null;\n  \n\n\n  try {\n    if (handlerOptions.verbose) {\n      handlerOptions.logger?.info(\n        `[renderPage.client] Client-side rendering for route: ${handlerOptions.route}`\n      );\n    }\n\n    // Step 1: Resolve paths to built paths using the server manifest\n    // The client version needs to use the server manifest to get the built paths\n    // for the page components, not the static manifest\n    const resolvePathWithManifest = (path: string, manifest: any): string => {\n      const entry = manifest[path];\n      if (entry && entry.file) {\n        return entry.file;\n      }\n      return path;\n    };\n\n    // Use manifest for page component resolution (client version works in reverse)\n    const manifest = handlerOptions.manifest || {};\n    const resolvedPagePath = handlerOptions.pagePath ? resolvePathWithManifest(handlerOptions.pagePath, manifest) : undefined;\n    const resolvedPropsPath = handlerOptions.propsPath ? resolvePathWithManifest(handlerOptions.propsPath, manifest) : undefined;\n    const resolvedRootPath = handlerOptions.rootPath ? resolvePathWithManifest(handlerOptions.rootPath, manifest) : undefined;\n    const resolvedHtmlPath = handlerOptions.htmlPath ? resolvePathWithManifest(handlerOptions.htmlPath, manifest) : undefined;\n\n    if (handlerOptions.verbose) {\n      handlerOptions.logger?.info(`[renderPage.client] Resolved paths for route ${handlerOptions.route}:`);\n      handlerOptions.logger?.info(`  page: ${handlerOptions.pagePath} -> ${resolvedPagePath}`);\n      handlerOptions.logger?.info(`  props: ${handlerOptions.propsPath} -> ${resolvedPropsPath}`);\n      handlerOptions.logger?.info(`  root: ${handlerOptions.rootPath} -> ${resolvedRootPath}`);\n      handlerOptions.logger?.info(`  html: ${handlerOptions.htmlPath} -> ${resolvedHtmlPath}`);\n      handlerOptions.logger?.info(`  manifest keys: ${Object.keys(manifest).join(', ')}`);\n      handlerOptions.logger?.info(`  HTML path issue: htmlPath='${handlerOptions.htmlPath}', resolved='${resolvedHtmlPath}', manifest has Html entry: ${!!manifest[handlerOptions.htmlPath || '']}`);\n      handlerOptions.logger?.info(`  About to pass htmlPath='${resolvedHtmlPath}' to RSC stream`);\n    }\n    const worker = handlerOptions.worker ?? handlerOptions.rscWorker;\n\n    // Step 2: Resolve components using the RSC worker with built paths\n    // This separates component resolution from RSC generation, making the\n    // subsequent RSC render completely synchronous\n    if (!worker) {\n      throw new Error(\"RSC worker is required for client-side component resolution\");\n    }\n    \n    // Preload components in the worker for faster subsequent RSC stream generation\n    try {\n      await resolveComponents({\n        route: handlerOptions.route,\n        pagePath: resolvedPagePath,\n        propsPath: resolvedPropsPath,\n        rootPath: resolvedRootPath,\n        htmlPath: resolvedHtmlPath,\n        pageExportName: handlerOptions.pageExportName,\n        propsExportName: handlerOptions.propsExportName,\n        rootExportName: handlerOptions.rootExportName,\n        htmlExportName: handlerOptions.htmlExportName,\n        worker: worker,\n        rscWorker: worker,\n        onMetrics: handlerOptions.onMetrics,\n        logger: handlerOptions.logger,\n        verbose: handlerOptions.verbose,\n      });\n    } catch (componentResolutionError) {\n      // Handle component resolution failures gracefully\n      const error = componentResolutionError instanceof Error \n        ? componentResolutionError \n        : new Error(String(componentResolutionError));\n      \n      // Check if this component resolution error should cause a panic based on panicThreshold\n      const panicError = handleError({\n        error,\n        critical: false,\n        logger: handlerOptions.logger,\n        panicThreshold: handlerOptions.panicThreshold,\n        context: `Component resolution failed for route ${handlerOptions.route}`,\n      });\n      \n             // If this should cause a panic, yield error and return\n       if (panicError) {\n         yield {\n           type: \"error\",\n           error: panicError,\n           metrics: {\n             rscFull: rscFullMetrics,\n             rscHeadless: rscHeadlessMetrics,\n             html: htmlMetrics,\n           },\n         };\n         return;\n       }\n       \n       // Otherwise, treat this as a non-critical error and continue with client-only HTML\n       // This allows the build to complete with a client-only page\n       handlerOptions.logger?.warn(\n         `[renderPage.client] Component resolution failed for route ${handlerOptions.route}, continuing with client-only HTML: ${error.message}`\n       );\n       \n       // Create a client-only HTML stream wrapper with minimal HTML\n       const clientOnlyHtmlStreamWrapper = {\n         pipe: <Writable extends NodeJS.WritableStream>(destination: Writable) => {\n           // Write a minimal client-only HTML structure\n           const minimalHtml = `<!DOCTYPE html><html><head><link rel=\"expect\" href=\"#«R»\" blocking=\"render\"/></head><body><div id=\"root\"></div><template id=\"«R»\"></template></body></html>`;\n           destination.write(minimalHtml);\n           destination.end();\n           return destination;\n         },\n         abort: () => {\n           // No cleanup needed for simple HTML string\n         },\n       };\n       \n       // Create an empty RSC stream wrapper\n       const emptyRscStreamWrapper = {\n         pipe: <Writable extends NodeJS.WritableStream>(destination: Writable) => {\n           // No RSC content for failed component resolution\n           destination.end();\n           return destination;\n         },\n         abort: () => {\n           // No cleanup needed\n         },\n       };\n       \n       // Yield skip result with client-only HTML and empty RSC\n       yield {\n         type: \"skip\",\n         reason: error,\n         html: clientOnlyHtmlStreamWrapper,\n         rsc: emptyRscStreamWrapper,\n         metrics: {\n           rscFull: rscFullMetrics,\n           rscHeadless: rscHeadlessMetrics,\n           html: htmlMetrics,\n         },\n       };\n       return;\n    }\n\n    // Step 2: Create handler options\n    // Components are now preloaded in the worker, so we can use the original handler options\n    const newHandlerOptions = {\n      ...handlerOptions,\n      // Pass page paths to the RSC worker so it knows what to render\n      pagePath: resolvedPagePath,\n      propsPath: resolvedPropsPath,\n      rootPath: resolvedRootPath,\n      htmlPath: resolvedHtmlPath,\n    };\n\n    if (handlerOptions.verbose) {\n      handlerOptions.logger?.info(\n        `[renderPage.client] handlerOptions.clientPipeableStreamOptions: ${JSON.stringify(handlerOptions.clientPipeableStreamOptions)}`\n      );\n      handlerOptions.logger?.info(\n        `[renderPage.client] newHandlerOptions.clientPipeableStreamOptions: ${JSON.stringify(newHandlerOptions.clientPipeableStreamOptions)}`\n      );\n      handlerOptions.logger?.info(\n        `[renderPage.client] newHandlerOptions page paths: pagePath=${newHandlerOptions.pagePath}, propsPath=${newHandlerOptions.propsPath}, rootPath=${newHandlerOptions.rootPath}, htmlPath=${newHandlerOptions.htmlPath}`\n      );\n    }\n\n    // Component resolution is already measured in resolveComponents\n    // No need to measure module resolution time here anymore\n\n    // Create headless RSC stream first (for .rsc file)\n    const uniqueId = handlerOptions.id ?? `${handlerOptions.route}-${Date.now()}-${Math.random().toString(36).substring(2, 11)}`;\n    \n    const headlessRscStreamLocal = createRscStream({\n      ...newHandlerOptions,\n      id: `${handlerOptions.route}-headless-${uniqueId}`,\n      rscTimeout: handlerOptions.rscTimeout || 5000,\n      onMetrics: handlerOptions.onMetrics,\n      // Headless RSC stream: page content only (for .rsc file)\n      htmlPath: '', // No HTML wrapper - just page content\n      pagePath: newHandlerOptions.pagePath || '', // Ensure pagePath is always a string\n      url: newHandlerOptions.url || '', // Ensure url is always a string\n      pageProps: newHandlerOptions.pageProps || {}, // Ensure pageProps is always an object\n      onEvent: wrappedOnEvent,\n    });\n\n    // Create full RSC stream that reuses the headless stream elements\n    const fullRscStreamLocal = createRscStream({\n      ...newHandlerOptions,\n      id: `${handlerOptions.route}-full-${uniqueId}`,\n      rscTimeout: handlerOptions.rscTimeout || 5000,\n      onMetrics: handlerOptions.onMetrics,\n      // Full RSC stream: include HTML wrapper (for HTML generation)\n      // Pass through the resolved htmlPath so custom Html components work in client mode\n      htmlPath: resolvedHtmlPath,\n      pagePath: newHandlerOptions.pagePath || '', // Ensure pagePath is always a string\n      url: newHandlerOptions.url || '', // Ensure url is always a string\n      pageProps: newHandlerOptions.pageProps || {}, // Ensure pageProps is always an object\n      // Reuse headless stream elements - the worker will handle this with the unique ID\n      reuseHeadlessStreamId: headlessRscStreamLocal.id,\n      onEvent: wrappedOnEvent,\n    });\n\n    // Assign to the outer variables\n    headlessRscStream = headlessRscStreamLocal;\n    fullRscStream = fullRscStreamLocal;\n\n    // The headless stream will be consumed naturally by the file writing\n    // The full stream will reuse the headless stream elements for HTML generation\n\n    // Step 3: Create HTML transform stream\n    if (handlerOptions.verbose) {\n      handlerOptions.logger?.info(\n        `[renderPage.client] Creating HTML transform stream with clientPipeableStreamOptions: ${JSON.stringify(newHandlerOptions.clientPipeableStreamOptions)}`\n      );\n    }\n    // Create HTML stream using the full RSC stream (which reuses headless stream elements)\n    const htmlTransformStream = createRscToHtmlStream({\n      ...newHandlerOptions,\n      htmlTimeout: handlerOptions.htmlTimeout || 15000,\n      route: handlerOptions.route,\n      logger: handlerOptions.logger,\n      verbose: handlerOptions.verbose,\n      rscStream: fullRscStreamLocal.rscStream,\n    });\n\n    htmlHandler = {\n      htmlStream: htmlTransformStream,\n      abort: () => {\n        htmlTransformStream.abort();\n      }\n    };\n\n    // Create stream wrappers for file writing\n    const rscStreamWrapper = {\n      pipe: <Writable extends NodeJS.WritableStream>(destination: Writable) => {\n        const streamMetrics = createStreamMetrics();\n        streamMetrics.startTime = performance.now();\n\n        // Use the headless RSC stream directly for the .rsc file\n        const rscFileStream = headlessRscStream.rscStream;\n\n        rscFileStream.on(\"data\", (chunk: Buffer) => {\n          streamMetrics.chunks++;\n          streamMetrics.bytes += chunk.length;\n        });\n\n        rscFileStream.on(\"end\", () => {\n          streamMetrics.duration = performance.now() - streamMetrics.startTime;\n          streamMetrics.endTime = performance.now();\n\n          rscHeadlessMetrics.streamMetrics = streamMetrics;\n          rscHeadlessMetrics.chunkRate = streamMetrics.chunks / (streamMetrics.duration / 1000);\n          rscHeadlessMetrics.processingTime = streamMetrics.duration;\n          rscHeadlessMetrics.memoryUsage = process.memoryUsage();\n          rscHeadlessMetrics.chunks = streamMetrics.chunks;\n        });\n\n        rscFileStream.pipe(destination);\n        return destination;\n      },\n      abort: () => headlessRscStream.abort(),\n    };\n\n    const htmlStreamWrapper = {\n      pipe: <Writable extends NodeJS.WritableStream>(destination: Writable) => {\n        if (handlerOptions.verbose) {\n          handlerOptions.logger?.info(\n            `[renderPage.client] Piping HTML stream to destination for route: ${handlerOptions.route}`\n          );\n        }\n        \n        // Use the HTML transform stream's pipe method directly (same as server side)\n        return htmlTransformStream.pipe(destination);\n      },\n      abort: () => {\n        fullRscStream.abort();\n        if (htmlHandler.abort) {\n          htmlHandler.abort();\n        }\n      },\n      on: (event: string, listener: (...args: any[]) => void) => {\n        // Forward error events from the HTML transform stream to the wrapper\n        if (event === 'error') {\n          // Access the actual stream from the transform result\n          const htmlStream = (htmlTransformStream as any).htmlStream;\n          if (htmlStream && typeof htmlStream.on === 'function') {\n            htmlStream.on('error', listener);\n          }\n        }\n        return htmlStreamWrapper;\n      },\n    };\n\n    // Don't emit initial metrics - wait for file writes to complete\n    // The onMetrics callback will be called after both file.write.done events\n\n    // Check if we have an error result to yield (with timeout protection)\n    // Wait a short time for any pending route.error events\n    await new Promise(resolve => setTimeout(resolve, 100));\n    \n    if (errorResult) {\n      yield errorResult;\n      return;\n    }\n\n    yield {\n      type: \"success\",\n      html: htmlStreamWrapper,\n      rsc: rscStreamWrapper,\n      metrics: {\n        rscFull: rscFullMetrics,\n        rscHeadless: rscHeadlessMetrics,\n        html: htmlMetrics,\n      },\n    } as const;\n  } catch (error) {\n    // Clean up resources\n    try {\n      if (headlessRscStream) headlessRscStream.abort();\n      if (fullRscStream) fullRscStream.abort();\n      if (htmlHandler?.abort) htmlHandler.abort();\n    } catch (cleanupError: unknown) {\n      handlerOptions.logger?.warn(`Failed to cleanup streams on error: ${cleanupError}`);\n    }\n\n    const panicError = handleError({\n      error,\n      logger: handlerOptions.logger,\n      context: \"renderPageClient\",\n      panicThreshold: handlerOptions.panicThreshold,\n    });\n\n    if (panicError != null) {\n      yield {\n        type: \"error\",\n        error: panicError,\n        metrics: {\n          rscFull: rscFullMetrics,\n          rscHeadless: rscHeadlessMetrics,\n          html: htmlMetrics,\n        },\n      };\n    } else {\n      // For non-panic errors, we still want to write the HTML file (client-only)\n      // but skip the RSC file since there was a server error\n      \n      // Create a fallback RSC stream with React.Fragment (same as server environment)\n      const fallbackRscStream = createRscStream({\n        ...handlerOptions,\n        url: `${handlerOptions.url}`,\n        route: `${handlerOptions.route}`,\n        cssFiles: handlerOptions.cssFiles || new Map(),\n        globalCss: handlerOptions.globalCss || new Map(),\n        id: `${handlerOptions.route}-fallback-${Date.now()}`,\n        rscTimeout: handlerOptions.rscTimeout || 5000,\n        onMetrics: handlerOptions.onMetrics,\n        // Use React.Fragment as fallback (same as server environment)\n        pagePath: '', // This will cause the default page to be used, but we'll override it\n        pageProps: {}, // Ensure pageProps is always an object\n      });\n      \n      // Create HTML stream that processes the fallback RSC stream to ensure performance timing script is injected\n      const fallbackHtmlStream = createRscToHtmlStream({\n        id: handlerOptions.id,\n        route: handlerOptions.route,\n        url: handlerOptions.url,\n        moduleRootPath: handlerOptions.moduleRootPath,\n        moduleBasePath: handlerOptions.moduleBasePath,\n        moduleBaseURL: handlerOptions.moduleBaseURL,\n        projectRoot: handlerOptions.projectRoot,\n        panicThreshold: handlerOptions.panicThreshold,\n        verbose: handlerOptions.verbose,\n        signal: handlerOptions.signal,\n        logger: handlerOptions.logger,\n        htmlTimeout: handlerOptions.htmlTimeout,\n        clientPipeableStreamOptions: handlerOptions.clientPipeableStreamOptions,\n        onMetrics: handlerOptions.onMetrics,\n        build: handlerOptions.build,\n      });\n      \n      // Create a wrapper that pipes the fallback RSC stream through the HTML transform\n      const clientOnlyHtmlStreamWrapper = {\n        pipe: <Writable extends NodeJS.WritableStream>(destination: Writable) => {\n          // Pipe the fallback RSC stream through the HTML transform to ensure performance timing script is injected\n          return fallbackHtmlStream.pipe(destination);\n        },\n        abort: () => {\n          // Clean up the fallback RSC stream\n          fallbackRscStream.abort();\n        },\n      };\n      \n      // Create an empty RSC stream wrapper\n      const emptyRscStreamWrapper = {\n        pipe: <Writable extends NodeJS.WritableStream>(destination: Writable) => {\n          // No RSC content for skipped routes\n          destination.end();\n          return destination;\n        },\n        abort: () => {\n          // No cleanup needed\n        },\n      };\n      \n      yield {\n        type: \"skip\",\n        reason: error,\n        html: clientOnlyHtmlStreamWrapper,\n        rsc: emptyRscStreamWrapper,\n        metrics: {\n          rscFull: rscFullMetrics,\n          rscHeadless: rscHeadlessMetrics,\n          html: htmlMetrics,\n        },\n      };\n    }\n  }\n}; "],"names":["manifest"],"mappings":"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"}
|