vite-plugin-react-server 1.4.2 → 1.4.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +48 -313
- package/dist/package.json +123 -13
- package/dist/plugin/bundle/deferredStaticGeneration.js +14 -39
- package/dist/plugin/bundle/manifests.js +30 -48
- package/dist/plugin/config/autoDiscover/resolveAutoDiscover.d.ts.map +1 -1
- package/dist/plugin/config/autoDiscover/resolveAutoDiscover.js +4 -1
- package/dist/plugin/config/envPrefixFromConfig.js +12 -7
- package/dist/plugin/config/getCondition.d.ts.map +1 -1
- package/dist/plugin/config/getCondition.js +7 -5
- package/dist/plugin/dev-server/virtualRscHmrPlugin.js +23 -23
- package/dist/plugin/environments/createBuildEventPlugin.js +88 -98
- package/dist/plugin/environments/createEnvironmentPlugin.js +222 -250
- package/dist/plugin/helpers/createRscRenderHelpers.js +33 -34
- package/dist/plugin/helpers/createSharedLoader.d.ts.map +1 -1
- package/dist/plugin/helpers/createSharedLoader.js +4 -2
- package/dist/plugin/helpers/headlessStreamReuseHandler.js +30 -22
- package/dist/plugin/helpers/headlessStreamState.js +15 -28
- package/dist/plugin/helpers/resolveComponent.d.ts.map +1 -1
- package/dist/plugin/helpers/resolveComponent.js +4 -2
- package/dist/plugin/index.client.d.ts +5 -0
- package/dist/plugin/index.client.d.ts.map +1 -0
- package/dist/plugin/index.client.js +4 -0
- package/dist/plugin/index.d.ts +4 -3
- package/dist/plugin/index.d.ts.map +1 -1
- package/dist/plugin/index.js +10 -5
- package/dist/plugin/index.server.d.ts +5 -0
- package/dist/plugin/index.server.d.ts.map +1 -0
- package/dist/plugin/index.server.js +4 -0
- package/dist/plugin/metrics/createWorkerStartupMetrics.js +31 -13
- package/dist/plugin/orchestrator/createPluginOrchestrator.client.js +41 -38
- package/dist/plugin/orchestrator/createPluginOrchestrator.server.js +43 -46
- package/dist/plugin/plugin.client.js +2 -2
- package/dist/plugin/plugin.server.js +2 -2
- package/dist/plugin/react-static/createBuildLoader.client.js +12 -6
- package/dist/plugin/react-static/createBuildLoader.server.js +255 -235
- package/dist/plugin/react-static/plugin.client.js +684 -770
- package/dist/plugin/react-static/plugin.server.js +517 -603
- package/dist/plugin/react-static/processCssFilesForPages.js +103 -88
- package/dist/plugin/react-static/renderPage.client.js +455 -529
- package/dist/plugin/react-static/renderPage.server.js +485 -508
- package/dist/plugin/react-static/renderPagesBatched.js +277 -275
- package/dist/plugin/react-static/rscToHtmlStream.client.js +48 -29
- package/dist/plugin/react-static/rscToHtmlStream.server.js +62 -37
- package/dist/plugin/react-static/temporaryReferences.server.js +11 -2
- package/dist/plugin/stream/createMainThreadHandlers.js +40 -31
- package/dist/plugin/stream/renderRscStream.server.d.ts.map +1 -1
- package/dist/plugin/stream/renderRscStream.server.js +127 -144
- package/dist/plugin/transformer/createTransformerPlugin.js +226 -265
- package/dist/plugin/utils/checkReactVersion.d.ts +7 -0
- package/dist/plugin/utils/checkReactVersion.d.ts.map +1 -0
- package/dist/plugin/utils/checkReactVersion.js +23 -0
- package/dist/plugin/utils/envUrls.node.js +12 -11
- package/dist/plugin/vendor/vendor-alias.js +84 -114
- package/dist/plugin/vendor/vendor.client.d.ts.map +1 -1
- package/dist/plugin/vendor/vendor.client.js +1 -3
- package/dist/plugin/worker/rsc/handleRscRender.d.ts.map +1 -1
- package/dist/plugin/worker/rsc/handleRscRender.js +3 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +123 -13
- package/plugin/config/autoDiscover/resolveAutoDiscover.ts +4 -0
- package/plugin/config/getCondition.ts +6 -4
- package/plugin/helpers/createRscRenderHelpers.ts +1 -1
- package/plugin/helpers/createSharedLoader.ts +6 -1
- package/plugin/helpers/resolveComponent.ts +6 -1
- package/plugin/index.client.ts +4 -0
- package/plugin/index.server.ts +4 -0
- package/plugin/index.ts +12 -5
- package/plugin/plugin.client.ts +1 -1
- package/plugin/plugin.server.ts +1 -1
- package/plugin/stream/renderRscStream.server.ts +3 -0
- package/plugin/utils/checkReactVersion.ts +28 -0
- package/plugin/vendor/vendor.client.ts +0 -2
- package/plugin/worker/rsc/handleRscRender.ts +2 -0
- package/scripts/generate-toc.mjs +27 -294
|
@@ -1,533 +1,510 @@
|
|
|
1
1
|
/**
|
|
2
|
-
*
|
|
3
|
-
*
|
|
4
|
-
*
|
|
5
|
-
*
|
|
6
|
-
* ARCHITECTURE OVERVIEW:
|
|
7
|
-
*
|
|
8
|
-
* SERVER-SIDE vs CLIENT-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. Create headless RSC stream (for .rsc file)
|
|
14
|
-
* 2. Create full RSC stream (for HTML generation)
|
|
15
|
-
* 3. Create HTML transform stream that converts RSC to HTML
|
|
16
|
-
* 4. Both streams are piped to file writers
|
|
17
|
-
*
|
|
18
|
-
* SIMPLIFIED APPROACH:
|
|
19
|
-
* This implementation follows the same simple pattern as the client side,
|
|
20
|
-
* avoiding complex backpressure handling and race conditions.
|
|
2
|
+
* vite-plugin-react-server
|
|
3
|
+
* Copyright (c) Nico Brinkkemper
|
|
4
|
+
* MIT License
|
|
21
5
|
*/
|
|
22
|
-
import { createRenderMetrics } from
|
|
23
|
-
import { routeToURL } from
|
|
24
|
-
import { handleError } from
|
|
25
|
-
import { assertReactServer } from
|
|
26
|
-
import { renderRscStream } from
|
|
27
|
-
import { createMainThreadHandlers } from
|
|
28
|
-
import { createRscToHtmlStream } from
|
|
29
|
-
import { resolveComponent } from
|
|
30
|
-
import { resolvePageAndProps } from
|
|
31
|
-
import { Root
|
|
32
|
-
import { Html
|
|
33
|
-
import { createStreamMetrics } from
|
|
34
|
-
import { join } from
|
|
35
|
-
import {
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
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
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
}
|
|
131
|
-
// Load Root component
|
|
132
|
-
if (handlerOptions.rootPath) {
|
|
133
|
-
try {
|
|
134
|
-
const rootResult = await resolveComponent({
|
|
135
|
-
componentPath: handlerOptions.rootPath,
|
|
136
|
-
exportName: handlerOptions.rootExportName,
|
|
137
|
-
loader: handlerOptions.loader,
|
|
138
|
-
});
|
|
139
|
-
if (rootResult.type === "success") {
|
|
140
|
-
RootComponent = rootResult.component;
|
|
141
|
-
}
|
|
142
|
-
else {
|
|
143
|
-
handlerOptions.logger?.warn(`Failed to load Root component from ${handlerOptions.rootPath}: ${rootResult.error?.message || "Unknown error"}`);
|
|
144
|
-
}
|
|
145
|
-
}
|
|
146
|
-
catch (error) {
|
|
147
|
-
handlerOptions.logger?.warn(`Error loading Root component from ${handlerOptions.rootPath}: ${error instanceof Error ? error.message : String(error)}`);
|
|
148
|
-
}
|
|
149
|
-
}
|
|
150
|
-
// Load Html component
|
|
151
|
-
if (handlerOptions.htmlPath) {
|
|
152
|
-
try {
|
|
153
|
-
const htmlResult = await resolveComponent({
|
|
154
|
-
componentPath: handlerOptions.htmlPath,
|
|
155
|
-
exportName: handlerOptions.htmlExportName,
|
|
156
|
-
loader: handlerOptions.loader,
|
|
157
|
-
});
|
|
158
|
-
if (htmlResult.type === "success") {
|
|
159
|
-
HtmlComponent = htmlResult.component;
|
|
160
|
-
}
|
|
161
|
-
else {
|
|
162
|
-
handlerOptions.logger?.warn(`Failed to load Html component from ${handlerOptions.htmlPath}: ${htmlResult.error?.message || "Unknown error"}`);
|
|
163
|
-
}
|
|
164
|
-
}
|
|
165
|
-
catch (error) {
|
|
166
|
-
handlerOptions.logger?.warn(`Error loading Html component from ${handlerOptions.htmlPath}: ${error instanceof Error ? error.message : String(error)}`);
|
|
167
|
-
}
|
|
6
|
+
import { createRenderMetrics } from '../metrics/createRenderMetrics.js';
|
|
7
|
+
import { routeToURL } from '../utils/routeToURL.js';
|
|
8
|
+
import { handleError } from '../error/handleError.js';
|
|
9
|
+
import { assertReactServer } from '../config/getCondition.js';
|
|
10
|
+
import { renderRscStream } from '../stream/renderRscStream.server.js';
|
|
11
|
+
import { createMainThreadHandlers } from '../stream/createMainThreadHandlers.js';
|
|
12
|
+
import { createRscToHtmlStream } from './rscToHtmlStream.server.js';
|
|
13
|
+
import { resolveComponent } from '../helpers/resolveComponent.js';
|
|
14
|
+
import { resolvePageAndProps } from '../helpers/resolvePageAndProps.js';
|
|
15
|
+
import { Root } from '../components/root.js';
|
|
16
|
+
import { Html } from '../components/html.js';
|
|
17
|
+
import { createStreamMetrics } from '../metrics/createStreamMetrics.js';
|
|
18
|
+
import { join } from 'node:path';
|
|
19
|
+
import { trackHeadlessStreamError, createHeadlessStreamState, hasHeadlessStreamError } from '../helpers/headlessStreamState.js';
|
|
20
|
+
|
|
21
|
+
const renderPage = async function* renderPage2(handlerOptions) {
|
|
22
|
+
assertReactServer();
|
|
23
|
+
const baseDir = join(
|
|
24
|
+
handlerOptions.build.outDir,
|
|
25
|
+
handlerOptions.build.static
|
|
26
|
+
);
|
|
27
|
+
const routePath = handlerOptions.route.replace(/^\//, "");
|
|
28
|
+
const htmlMetrics = createRenderMetrics({
|
|
29
|
+
route: handlerOptions.route,
|
|
30
|
+
type: "html",
|
|
31
|
+
fromMainThread: false,
|
|
32
|
+
// Server: HTML rendered in worker
|
|
33
|
+
fromRscWorker: false,
|
|
34
|
+
fromHtmlWorker: true,
|
|
35
|
+
baseDir,
|
|
36
|
+
routePath,
|
|
37
|
+
fileName: handlerOptions.build.htmlOutputPath,
|
|
38
|
+
outputPath: join(baseDir, routePath, handlerOptions.build.htmlOutputPath)
|
|
39
|
+
});
|
|
40
|
+
const rscFullMetrics = createRenderMetrics({
|
|
41
|
+
route: handlerOptions.route,
|
|
42
|
+
type: "rsc-full",
|
|
43
|
+
fromMainThread: true,
|
|
44
|
+
// Server: RSC rendered on main thread
|
|
45
|
+
fromRscWorker: false,
|
|
46
|
+
fromHtmlWorker: false
|
|
47
|
+
});
|
|
48
|
+
const rscHeadlessMetrics = createRenderMetrics({
|
|
49
|
+
route: handlerOptions.route,
|
|
50
|
+
type: "rsc-headless",
|
|
51
|
+
fromMainThread: true,
|
|
52
|
+
// Server: RSC rendered on main thread
|
|
53
|
+
fromRscWorker: false,
|
|
54
|
+
fromHtmlWorker: false,
|
|
55
|
+
baseDir,
|
|
56
|
+
routePath,
|
|
57
|
+
fileName: handlerOptions.build.rscOutputPath,
|
|
58
|
+
outputPath: join(baseDir, routePath, handlerOptions.build.rscOutputPath)
|
|
59
|
+
});
|
|
60
|
+
let headlessRscHandler = null;
|
|
61
|
+
let fullRscHandler = null;
|
|
62
|
+
let htmlTransformStream = null;
|
|
63
|
+
let headlessStreamErrored = false;
|
|
64
|
+
let headlessError = null;
|
|
65
|
+
let htmlStreamErrored = false;
|
|
66
|
+
let htmlStreamError = null;
|
|
67
|
+
const headlessStreamState = createHeadlessStreamState();
|
|
68
|
+
try {
|
|
69
|
+
if (handlerOptions.verbose) {
|
|
70
|
+
handlerOptions.logger?.info(
|
|
71
|
+
`[renderPage.server] Server-side rendering for route: ${handlerOptions.route}`
|
|
72
|
+
);
|
|
73
|
+
}
|
|
74
|
+
if (!handlerOptions.url) {
|
|
75
|
+
handlerOptions.url = routeToURL(
|
|
76
|
+
handlerOptions.route,
|
|
77
|
+
handlerOptions.moduleBaseURL,
|
|
78
|
+
handlerOptions.build.rscOutputPath
|
|
79
|
+
);
|
|
80
|
+
}
|
|
81
|
+
let PageComponent = null;
|
|
82
|
+
let RootComponent = null;
|
|
83
|
+
let HtmlComponent = null;
|
|
84
|
+
let pageProps = {};
|
|
85
|
+
if (handlerOptions.pagePath) {
|
|
86
|
+
try {
|
|
87
|
+
const pageAndPropsResult = await resolvePageAndProps({
|
|
88
|
+
pagePath: handlerOptions.pagePath,
|
|
89
|
+
pageExportName: handlerOptions.pageExportName,
|
|
90
|
+
propsPath: handlerOptions.propsPath,
|
|
91
|
+
propsExportName: handlerOptions.propsExportName,
|
|
92
|
+
loader: handlerOptions.loader,
|
|
93
|
+
verbose: handlerOptions.verbose,
|
|
94
|
+
logger: handlerOptions.logger,
|
|
95
|
+
route: handlerOptions.route,
|
|
96
|
+
url: handlerOptions.url,
|
|
97
|
+
moduleBaseURL: handlerOptions.moduleBaseURL,
|
|
98
|
+
build: {
|
|
99
|
+
rscOutputPath: handlerOptions.build.rscOutputPath
|
|
100
|
+
}
|
|
101
|
+
});
|
|
102
|
+
if (pageAndPropsResult.type === "success") {
|
|
103
|
+
PageComponent = pageAndPropsResult.PageComponent;
|
|
104
|
+
pageProps = pageAndPropsResult.pageProps || {};
|
|
105
|
+
if (handlerOptions.verbose) {
|
|
106
|
+
handlerOptions.logger?.info(
|
|
107
|
+
`[renderPage.server] Successfully loaded page and props for route ${handlerOptions.route}: pageProps=${JSON.stringify(pageProps)}`
|
|
108
|
+
);
|
|
109
|
+
}
|
|
110
|
+
} else {
|
|
111
|
+
handlerOptions.logger?.warn(
|
|
112
|
+
`Failed to load page and props from ${handlerOptions.pagePath}: ${pageAndPropsResult.error?.message || "Unknown error"}`
|
|
113
|
+
);
|
|
168
114
|
}
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
115
|
+
} catch (error) {
|
|
116
|
+
handlerOptions.logger?.warn(
|
|
117
|
+
`Error loading page and props from ${handlerOptions.pagePath}: ${error instanceof Error ? error.message : String(error)}`
|
|
118
|
+
);
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
if (handlerOptions.rootPath) {
|
|
122
|
+
try {
|
|
123
|
+
const rootResult = await resolveComponent({
|
|
124
|
+
componentPath: handlerOptions.rootPath,
|
|
125
|
+
exportName: handlerOptions.rootExportName,
|
|
126
|
+
loader: handlerOptions.loader
|
|
127
|
+
});
|
|
128
|
+
if (rootResult.type === "success") {
|
|
129
|
+
RootComponent = rootResult.component;
|
|
130
|
+
} else {
|
|
131
|
+
handlerOptions.logger?.warn(
|
|
132
|
+
`Failed to load Root component from ${handlerOptions.rootPath}: ${rootResult.error?.message || "Unknown error"}`
|
|
133
|
+
);
|
|
172
134
|
}
|
|
173
|
-
|
|
174
|
-
|
|
135
|
+
} catch (error) {
|
|
136
|
+
handlerOptions.logger?.warn(
|
|
137
|
+
`Error loading Root component from ${handlerOptions.rootPath}: ${error instanceof Error ? error.message : String(error)}`
|
|
138
|
+
);
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
if (handlerOptions.htmlPath) {
|
|
142
|
+
try {
|
|
143
|
+
const htmlResult = await resolveComponent({
|
|
144
|
+
componentPath: handlerOptions.htmlPath,
|
|
145
|
+
exportName: handlerOptions.htmlExportName,
|
|
146
|
+
loader: handlerOptions.loader
|
|
147
|
+
});
|
|
148
|
+
if (htmlResult.type === "success") {
|
|
149
|
+
HtmlComponent = htmlResult.component;
|
|
150
|
+
} else {
|
|
151
|
+
handlerOptions.logger?.warn(
|
|
152
|
+
`Failed to load Html component from ${handlerOptions.htmlPath}: ${htmlResult.error?.message || "Unknown error"}`
|
|
153
|
+
);
|
|
175
154
|
}
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
155
|
+
} catch (error) {
|
|
156
|
+
handlerOptions.logger?.warn(
|
|
157
|
+
`Error loading Html component from ${handlerOptions.htmlPath}: ${error instanceof Error ? error.message : String(error)}`
|
|
158
|
+
);
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
if (!RootComponent) {
|
|
162
|
+
RootComponent = Root;
|
|
163
|
+
}
|
|
164
|
+
if (!HtmlComponent) {
|
|
165
|
+
HtmlComponent = Html;
|
|
166
|
+
}
|
|
167
|
+
if (!PageComponent || !RootComponent || !HtmlComponent) {
|
|
168
|
+
yield {
|
|
169
|
+
type: "error",
|
|
170
|
+
error: new Error(
|
|
171
|
+
`Component resolution failed: missing required components (Page: ${!!PageComponent}, Root: ${!!RootComponent}, Html: ${!!HtmlComponent})`
|
|
172
|
+
),
|
|
173
|
+
metrics: {
|
|
174
|
+
rscFull: rscFullMetrics,
|
|
175
|
+
rscHeadless: rscHeadlessMetrics,
|
|
176
|
+
html: htmlMetrics
|
|
188
177
|
}
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
178
|
+
};
|
|
179
|
+
return;
|
|
180
|
+
}
|
|
181
|
+
const uniqueId = handlerOptions.id ?? `${handlerOptions.route}?id=${Date.now()}-${Math.random().toString(36).substring(2, 11)}`;
|
|
182
|
+
const newHandlerOptions = {
|
|
183
|
+
...handlerOptions,
|
|
184
|
+
id: uniqueId,
|
|
185
|
+
url: `${handlerOptions.url}`,
|
|
186
|
+
route: `${handlerOptions.route}`,
|
|
187
|
+
PageComponent,
|
|
188
|
+
RootComponent,
|
|
189
|
+
HtmlComponent,
|
|
190
|
+
pageProps
|
|
191
|
+
};
|
|
192
|
+
if (handlerOptions.verbose) {
|
|
193
|
+
handlerOptions.logger?.info(
|
|
194
|
+
`[renderPage.server] Created newHandlerOptions for route ${handlerOptions.route} with pageProps: ${JSON.stringify(pageProps)}`
|
|
195
|
+
);
|
|
196
|
+
}
|
|
197
|
+
const headlessHandlers = createMainThreadHandlers(
|
|
198
|
+
handlerOptions,
|
|
199
|
+
(error, isPanic) => {
|
|
201
200
|
if (handlerOptions.verbose) {
|
|
202
|
-
|
|
201
|
+
handlerOptions.logger?.info(
|
|
202
|
+
`[renderPage.server] Headless stream error handler called for route ${handlerOptions.route}: ${error.message}, isPanic: ${isPanic}`
|
|
203
|
+
);
|
|
203
204
|
}
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
handlerOptions.logger?.info(`[renderPage.server] Headless stream error handler called for route ${handlerOptions.route}: ${error.message}, isPanic: ${isPanic}`);
|
|
208
|
-
}
|
|
209
|
-
// Track if the headless stream had errors
|
|
210
|
-
headlessStreamErrored = true;
|
|
211
|
-
headlessError = error instanceof Error ? error : new Error("Headless RSC stream failed");
|
|
212
|
-
// Track headless stream errors for conditional reuse logic (like RSC worker)
|
|
213
|
-
trackHeadlessStreamError(headlessStreamState, handlerOptions.route, headlessError);
|
|
214
|
-
if (handlerOptions.verbose) {
|
|
215
|
-
handlerOptions.logger?.info(`[renderPage.server] Stored headless stream error for route ${handlerOptions.route} in headlessStreamErrors map`);
|
|
216
|
-
}
|
|
217
|
-
// Store panic errors for later handling
|
|
218
|
-
if (isPanic) {
|
|
219
|
-
// For panic threshold "all_errors", the panic error will be handled by renderPages
|
|
220
|
-
if (handlerOptions.verbose) {
|
|
221
|
-
handlerOptions.logger?.info(`[renderPage.server] Panic error detected for route ${handlerOptions.route}, will be handled by renderPages`);
|
|
222
|
-
}
|
|
223
|
-
}
|
|
224
|
-
});
|
|
225
|
-
// Override onData to track metrics
|
|
226
|
-
headlessHandlers.onData = (_id, chunk) => {
|
|
227
|
-
rscHeadlessMetrics.chunks++;
|
|
228
|
-
rscHeadlessMetrics.streamMetrics.bytes += chunk.length;
|
|
229
|
-
};
|
|
230
|
-
headlessRscHandler = renderRscStream({
|
|
231
|
-
...newHandlerOptions,
|
|
232
|
-
htmlPath: '', // Headless RSC - no HTML wrapper
|
|
233
|
-
// If we expect errors, provide a safe Page component that doesn't throw
|
|
234
|
-
PageComponent: newHandlerOptions.PageComponent, // Use original for now, will be overridden if errors occur
|
|
235
|
-
}, headlessHandlers);
|
|
236
|
-
// Note: Panic errors will be yielded from the error handler when they occur
|
|
237
|
-
// No need to check shouldYieldPanicError here as it's set asynchronously
|
|
238
|
-
// Store PageComponent for reuse when headless stream completes (like RSC worker)
|
|
239
|
-
headlessRscHandler.rscStream.on('end', () => {
|
|
240
|
-
// Only store if this is a headless stream and no errors occurred (like RSC worker)
|
|
241
|
-
if (!hasHeadlessStreamError(headlessStreamState, handlerOptions.route)) {
|
|
242
|
-
headlessStreamState.elements.set(uniqueId, {
|
|
243
|
-
PageComponent: newHandlerOptions.PageComponent,
|
|
244
|
-
errored: false
|
|
245
|
-
});
|
|
246
|
-
if (handlerOptions.verbose) {
|
|
247
|
-
handlerOptions.logger?.info(`[renderPage.server] Stored PageComponent for headless stream ${uniqueId}`);
|
|
248
|
-
}
|
|
249
|
-
}
|
|
250
|
-
else {
|
|
251
|
-
if (handlerOptions.verbose) {
|
|
252
|
-
handlerOptions.logger?.info(`[renderPage.server] Headless stream errored for route ${handlerOptions.route}, not storing PageComponent for reuse`);
|
|
253
|
-
}
|
|
254
|
-
}
|
|
255
|
-
});
|
|
256
|
-
// Create full RSC handler (for HTML generation) - reuse headless stream elements if no errors
|
|
257
|
-
// For server-side, we create both streams in parallel like the client-side
|
|
258
|
-
let fullPanicError = null;
|
|
259
|
-
const fullHandlers = createMainThreadHandlers(handlerOptions, (error, isPanic) => {
|
|
260
|
-
// If this is a panic error, store it to be handled later
|
|
261
|
-
if (isPanic) {
|
|
262
|
-
fullPanicError = error instanceof Error ? error : new Error("Full RSC stream failed");
|
|
263
|
-
}
|
|
264
|
-
});
|
|
265
|
-
// Override onData to track metrics
|
|
266
|
-
fullHandlers.onData = (_id, chunk) => {
|
|
267
|
-
rscFullMetrics.chunks++;
|
|
268
|
-
rscFullMetrics.streamMetrics.bytes += chunk.length;
|
|
269
|
-
};
|
|
270
|
-
// Create full RSC handler options - use React.Fragment if headless stream had errors (like RSC worker)
|
|
271
|
-
// Check if there are any existing headless stream errors for this route
|
|
272
|
-
const hasExistingHeadlessError = hasHeadlessStreamError(headlessStreamState, handlerOptions.route);
|
|
273
|
-
const shouldUseFallback = headlessStreamErrored || hasExistingHeadlessError;
|
|
205
|
+
headlessStreamErrored = true;
|
|
206
|
+
headlessError = error instanceof Error ? error : new Error("Headless RSC stream failed");
|
|
207
|
+
trackHeadlessStreamError(headlessStreamState, handlerOptions.route, headlessError);
|
|
274
208
|
if (handlerOptions.verbose) {
|
|
275
|
-
|
|
209
|
+
handlerOptions.logger?.info(
|
|
210
|
+
`[renderPage.server] Stored headless stream error for route ${handlerOptions.route} in headlessStreamErrors map`
|
|
211
|
+
);
|
|
212
|
+
}
|
|
213
|
+
if (isPanic) {
|
|
214
|
+
if (handlerOptions.verbose) {
|
|
215
|
+
handlerOptions.logger?.info(
|
|
216
|
+
`[renderPage.server] Panic error detected for route ${handlerOptions.route}, will be handled by renderPages`
|
|
217
|
+
);
|
|
218
|
+
}
|
|
276
219
|
}
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
297
|
-
|
|
298
|
-
|
|
299
|
-
handlerOptions.logger?.info(`[renderPage.server] Headless stream completed successfully for route ${handlerOptions.route}`);
|
|
300
|
-
}
|
|
301
|
-
}
|
|
220
|
+
}
|
|
221
|
+
);
|
|
222
|
+
headlessHandlers.onData = (_id, chunk) => {
|
|
223
|
+
rscHeadlessMetrics.chunks++;
|
|
224
|
+
rscHeadlessMetrics.streamMetrics.bytes += chunk.length;
|
|
225
|
+
};
|
|
226
|
+
headlessRscHandler = renderRscStream(
|
|
227
|
+
{
|
|
228
|
+
...newHandlerOptions,
|
|
229
|
+
htmlPath: "",
|
|
230
|
+
// Headless RSC - no HTML wrapper
|
|
231
|
+
// If we expect errors, provide a safe Page component that doesn't throw
|
|
232
|
+
PageComponent: newHandlerOptions.PageComponent
|
|
233
|
+
// Use original for now, will be overridden if errors occur
|
|
234
|
+
},
|
|
235
|
+
headlessHandlers
|
|
236
|
+
);
|
|
237
|
+
headlessRscHandler.rscStream.on("end", () => {
|
|
238
|
+
if (!hasHeadlessStreamError(headlessStreamState, handlerOptions.route)) {
|
|
239
|
+
headlessStreamState.elements.set(uniqueId, {
|
|
240
|
+
PageComponent: newHandlerOptions.PageComponent,
|
|
241
|
+
errored: false
|
|
302
242
|
});
|
|
303
243
|
if (handlerOptions.verbose) {
|
|
304
|
-
|
|
244
|
+
handlerOptions.logger?.info(`[renderPage.server] Stored PageComponent for headless stream ${uniqueId}`);
|
|
305
245
|
}
|
|
306
|
-
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
yield {
|
|
310
|
-
type: "error",
|
|
311
|
-
error: fullPanicError,
|
|
312
|
-
metrics: {
|
|
313
|
-
rscFull: rscFullMetrics,
|
|
314
|
-
rscHeadless: rscHeadlessMetrics,
|
|
315
|
-
html: htmlMetrics,
|
|
316
|
-
},
|
|
317
|
-
};
|
|
318
|
-
return;
|
|
246
|
+
} else {
|
|
247
|
+
if (handlerOptions.verbose) {
|
|
248
|
+
handlerOptions.logger?.info(`[renderPage.server] Headless stream errored for route ${handlerOptions.route}, not storing PageComponent for reuse`);
|
|
319
249
|
}
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
moduleBaseURL: handlerOptions.moduleBaseURL,
|
|
329
|
-
projectRoot: handlerOptions.projectRoot,
|
|
330
|
-
build: handlerOptions.build,
|
|
331
|
-
panicThreshold: handlerOptions.panicThreshold,
|
|
332
|
-
verbose: handlerOptions.verbose,
|
|
333
|
-
signal: handlerOptions.signal,
|
|
334
|
-
logger: handlerOptions.logger,
|
|
335
|
-
htmlWorker: handlerOptions.htmlWorker,
|
|
336
|
-
clientPipeableStreamOptions: handlerOptions.clientPipeableStreamOptions,
|
|
337
|
-
onMetrics: handlerOptions.onMetrics,
|
|
338
|
-
htmlTimeout: handlerOptions.htmlTimeout || 15000,
|
|
339
|
-
rscStream: fullRscHandler.rscStream,
|
|
340
|
-
onError: (error, isPanic) => {
|
|
341
|
-
// Track HTML stream errors
|
|
342
|
-
htmlStreamErrored = true;
|
|
343
|
-
htmlStreamError = error;
|
|
344
|
-
if (isPanic) {
|
|
345
|
-
// This is a panic error, it should be yielded as an error result
|
|
346
|
-
if (handlerOptions.verbose) {
|
|
347
|
-
handlerOptions.logger?.error(`[renderPage.server] HTML stream panic error for route ${handlerOptions.route}: ${error.message}`);
|
|
348
|
-
}
|
|
349
|
-
}
|
|
350
|
-
else {
|
|
351
|
-
// For non-panic errors, just log them
|
|
352
|
-
if (handlerOptions.verbose) {
|
|
353
|
-
handlerOptions.logger?.warn(`[renderPage.server] HTML stream error for route ${handlerOptions.route}: ${error.message}`);
|
|
354
|
-
}
|
|
355
|
-
}
|
|
356
|
-
},
|
|
357
|
-
});
|
|
358
|
-
// Create stream wrappers for file writing - simplified like client side
|
|
359
|
-
const rscStreamWrapper = {
|
|
360
|
-
pipe: (destination) => {
|
|
361
|
-
const streamMetrics = createStreamMetrics();
|
|
362
|
-
streamMetrics.startTime = performance.now();
|
|
363
|
-
// Use the headless RSC stream directly for the .rsc file
|
|
364
|
-
const rscFileStream = headlessRscHandler.rscStream;
|
|
365
|
-
rscFileStream.on("data", (chunk) => {
|
|
366
|
-
streamMetrics.chunks++;
|
|
367
|
-
streamMetrics.bytes += chunk.length;
|
|
368
|
-
});
|
|
369
|
-
rscFileStream.on("end", () => {
|
|
370
|
-
streamMetrics.duration = performance.now() - streamMetrics.startTime;
|
|
371
|
-
streamMetrics.endTime = performance.now();
|
|
372
|
-
rscHeadlessMetrics.streamMetrics = streamMetrics;
|
|
373
|
-
rscHeadlessMetrics.chunkRate = streamMetrics.chunks / (streamMetrics.duration / 1000);
|
|
374
|
-
rscHeadlessMetrics.processingTime = streamMetrics.duration;
|
|
375
|
-
rscHeadlessMetrics.memoryUsage = process.memoryUsage();
|
|
376
|
-
rscHeadlessMetrics.chunks = streamMetrics.chunks;
|
|
377
|
-
});
|
|
378
|
-
rscFileStream.pipe(destination);
|
|
379
|
-
return destination;
|
|
380
|
-
},
|
|
381
|
-
abort: () => headlessRscHandler.abort(),
|
|
382
|
-
};
|
|
383
|
-
const htmlStreamWrapper = {
|
|
384
|
-
pipe: (destination) => {
|
|
385
|
-
// Use the HTML transform stream's pipe method directly (same as client side)
|
|
386
|
-
return htmlTransformStream.pipe(destination);
|
|
387
|
-
},
|
|
388
|
-
abort: () => {
|
|
389
|
-
htmlTransformStream.abort();
|
|
390
|
-
},
|
|
391
|
-
on: (event, listener) => {
|
|
392
|
-
// Forward error events from the HTML transform stream to the wrapper
|
|
393
|
-
if (event === 'error') {
|
|
394
|
-
// Access the actual stream from the transform result
|
|
395
|
-
const htmlStream = htmlTransformStream.htmlStream;
|
|
396
|
-
if (htmlStream && typeof htmlStream.on === 'function') {
|
|
397
|
-
htmlStream.on('error', listener);
|
|
398
|
-
}
|
|
399
|
-
}
|
|
400
|
-
return htmlStreamWrapper;
|
|
401
|
-
},
|
|
402
|
-
};
|
|
403
|
-
// Wait for HTML stream to complete or error before yielding success
|
|
404
|
-
// This ensures that any errors from the HTML stream are caught before we yield success
|
|
405
|
-
await new Promise((resolve, reject) => {
|
|
406
|
-
const timeout = setTimeout(() => {
|
|
407
|
-
reject(new Error(`HTML stream timeout for route ${handlerOptions.route}`));
|
|
408
|
-
}, handlerOptions.htmlTimeout || 15000);
|
|
409
|
-
// Check if HTML stream already errored
|
|
410
|
-
if (htmlStreamErrored) {
|
|
411
|
-
clearTimeout(timeout);
|
|
412
|
-
resolve(); // Let the panic threshold logic handle this at the renderPages level
|
|
413
|
-
return;
|
|
414
|
-
}
|
|
415
|
-
// Set up a flag to track if we've resolved
|
|
416
|
-
let resolved = false;
|
|
417
|
-
// Create a wrapper that resolves the promise when the stream completes
|
|
418
|
-
const originalPipe = htmlTransformStream.pipe;
|
|
419
|
-
htmlTransformStream.pipe = function (destination) {
|
|
420
|
-
const result = originalPipe.call(this, destination);
|
|
421
|
-
// Listen for the destination stream to end
|
|
422
|
-
destination.on('finish', () => {
|
|
423
|
-
if (!resolved) {
|
|
424
|
-
resolved = true;
|
|
425
|
-
clearTimeout(timeout);
|
|
426
|
-
resolve();
|
|
427
|
-
}
|
|
428
|
-
});
|
|
429
|
-
destination.on('error', (error) => {
|
|
430
|
-
if (!resolved) {
|
|
431
|
-
resolved = true;
|
|
432
|
-
clearTimeout(timeout);
|
|
433
|
-
reject(error);
|
|
434
|
-
}
|
|
435
|
-
});
|
|
436
|
-
return result;
|
|
437
|
-
};
|
|
438
|
-
// If we don't have a destination yet, resolve after a short delay
|
|
439
|
-
// This handles the case where the stream is created but not yet piped
|
|
440
|
-
setTimeout(() => {
|
|
441
|
-
if (!resolved) {
|
|
442
|
-
resolved = true;
|
|
443
|
-
clearTimeout(timeout);
|
|
444
|
-
resolve();
|
|
445
|
-
}
|
|
446
|
-
}, 100);
|
|
447
|
-
});
|
|
448
|
-
// Check for HTML stream errors after waiting for completion
|
|
449
|
-
if (htmlStreamErrored) {
|
|
450
|
-
yield {
|
|
451
|
-
type: "error",
|
|
452
|
-
error: htmlStreamError || new Error("HTML stream failed"),
|
|
453
|
-
metrics: {
|
|
454
|
-
rscFull: rscFullMetrics,
|
|
455
|
-
rscHeadless: rscHeadlessMetrics,
|
|
456
|
-
html: htmlMetrics,
|
|
457
|
-
},
|
|
458
|
-
};
|
|
459
|
-
return;
|
|
250
|
+
}
|
|
251
|
+
});
|
|
252
|
+
let fullPanicError = null;
|
|
253
|
+
const fullHandlers = createMainThreadHandlers(
|
|
254
|
+
handlerOptions,
|
|
255
|
+
(error, isPanic) => {
|
|
256
|
+
if (isPanic) {
|
|
257
|
+
fullPanicError = error instanceof Error ? error : new Error("Full RSC stream failed");
|
|
460
258
|
}
|
|
461
|
-
|
|
462
|
-
|
|
463
|
-
|
|
464
|
-
|
|
465
|
-
|
|
466
|
-
|
|
467
|
-
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
|
|
471
|
-
}
|
|
259
|
+
}
|
|
260
|
+
);
|
|
261
|
+
fullHandlers.onData = (_id, chunk) => {
|
|
262
|
+
rscFullMetrics.chunks++;
|
|
263
|
+
rscFullMetrics.streamMetrics.bytes += chunk.length;
|
|
264
|
+
};
|
|
265
|
+
const hasExistingHeadlessError = hasHeadlessStreamError(headlessStreamState, handlerOptions.route);
|
|
266
|
+
const shouldUseFallback = headlessStreamErrored || hasExistingHeadlessError;
|
|
267
|
+
if (handlerOptions.verbose) {
|
|
268
|
+
handlerOptions.logger?.info(
|
|
269
|
+
`[renderPage.server] Creating full RSC handler options for route ${handlerOptions.route}: headlessStreamErrored=${headlessStreamErrored}, hasExistingHeadlessError=${hasExistingHeadlessError}, shouldUseFallback=${shouldUseFallback}`
|
|
270
|
+
);
|
|
472
271
|
}
|
|
473
|
-
|
|
272
|
+
const SafePageComponent = (props) => {
|
|
273
|
+
const hasError = hasHeadlessStreamError(headlessStreamState, handlerOptions.route);
|
|
274
|
+
if (hasError) {
|
|
275
|
+
return null;
|
|
276
|
+
}
|
|
277
|
+
return newHandlerOptions.PageComponent(props);
|
|
278
|
+
};
|
|
279
|
+
const fullRscHandlerOptions = {
|
|
280
|
+
...newHandlerOptions,
|
|
281
|
+
htmlPath: void 0,
|
|
282
|
+
// Full RSC - include HTML wrapper
|
|
283
|
+
headlessStreamElements: headlessStreamState.elements,
|
|
284
|
+
// Pass the storage map for reuse
|
|
285
|
+
// Use SafePageComponent that returns null when there are headless stream errors
|
|
286
|
+
PageComponent: SafePageComponent
|
|
287
|
+
};
|
|
288
|
+
headlessRscHandler.rscStream.on("end", () => {
|
|
289
|
+
if (!hasHeadlessStreamError(headlessStreamState, handlerOptions.route)) {
|
|
474
290
|
if (handlerOptions.verbose) {
|
|
475
|
-
|
|
291
|
+
handlerOptions.logger?.info(
|
|
292
|
+
`[renderPage.server] Headless stream completed successfully for route ${handlerOptions.route}`
|
|
293
|
+
);
|
|
476
294
|
}
|
|
477
|
-
|
|
478
|
-
|
|
479
|
-
|
|
480
|
-
|
|
481
|
-
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
|
|
295
|
+
}
|
|
296
|
+
});
|
|
297
|
+
if (handlerOptions.verbose) {
|
|
298
|
+
handlerOptions.logger?.info(
|
|
299
|
+
`[renderPage.server] Created PageComponent that uses React.use() to consume headless stream for route ${handlerOptions.route}`
|
|
300
|
+
);
|
|
301
|
+
}
|
|
302
|
+
fullRscHandler = renderRscStream(fullRscHandlerOptions, fullHandlers);
|
|
303
|
+
if (fullPanicError) {
|
|
304
|
+
yield {
|
|
305
|
+
type: "error",
|
|
306
|
+
error: fullPanicError,
|
|
307
|
+
metrics: {
|
|
308
|
+
rscFull: rscFullMetrics,
|
|
309
|
+
rscHeadless: rscHeadlessMetrics,
|
|
310
|
+
html: htmlMetrics
|
|
485
311
|
}
|
|
486
|
-
|
|
487
|
-
|
|
312
|
+
};
|
|
313
|
+
return;
|
|
314
|
+
}
|
|
315
|
+
htmlTransformStream = createRscToHtmlStream({
|
|
316
|
+
id: handlerOptions.id,
|
|
317
|
+
worker: handlerOptions.worker,
|
|
318
|
+
route: handlerOptions.route,
|
|
319
|
+
url: handlerOptions.url,
|
|
320
|
+
moduleRootPath: handlerOptions.moduleRootPath,
|
|
321
|
+
moduleBasePath: handlerOptions.moduleBasePath,
|
|
322
|
+
moduleBaseURL: handlerOptions.moduleBaseURL,
|
|
323
|
+
projectRoot: handlerOptions.projectRoot,
|
|
324
|
+
build: handlerOptions.build,
|
|
325
|
+
panicThreshold: handlerOptions.panicThreshold,
|
|
326
|
+
verbose: handlerOptions.verbose,
|
|
327
|
+
signal: handlerOptions.signal,
|
|
328
|
+
logger: handlerOptions.logger,
|
|
329
|
+
htmlWorker: handlerOptions.htmlWorker,
|
|
330
|
+
clientPipeableStreamOptions: handlerOptions.clientPipeableStreamOptions,
|
|
331
|
+
onMetrics: handlerOptions.onMetrics,
|
|
332
|
+
htmlTimeout: handlerOptions.htmlTimeout || 15e3,
|
|
333
|
+
rscStream: fullRscHandler.rscStream,
|
|
334
|
+
onError: (error, isPanic) => {
|
|
335
|
+
htmlStreamErrored = true;
|
|
336
|
+
htmlStreamError = error;
|
|
337
|
+
if (isPanic) {
|
|
338
|
+
if (handlerOptions.verbose) {
|
|
339
|
+
handlerOptions.logger?.error(
|
|
340
|
+
`[renderPage.server] HTML stream panic error for route ${handlerOptions.route}: ${error.message}`
|
|
341
|
+
);
|
|
342
|
+
}
|
|
343
|
+
} else {
|
|
344
|
+
if (handlerOptions.verbose) {
|
|
345
|
+
handlerOptions.logger?.warn(
|
|
346
|
+
`[renderPage.server] HTML stream error for route ${handlerOptions.route}: ${error.message}`
|
|
347
|
+
);
|
|
348
|
+
}
|
|
349
|
+
}
|
|
350
|
+
}
|
|
351
|
+
});
|
|
352
|
+
const rscStreamWrapper = {
|
|
353
|
+
pipe: (destination) => {
|
|
354
|
+
const streamMetrics = createStreamMetrics();
|
|
355
|
+
streamMetrics.startTime = performance.now();
|
|
356
|
+
const rscFileStream = headlessRscHandler.rscStream;
|
|
357
|
+
rscFileStream.on("data", (chunk) => {
|
|
358
|
+
streamMetrics.chunks++;
|
|
359
|
+
streamMetrics.bytes += chunk.length;
|
|
360
|
+
});
|
|
361
|
+
rscFileStream.on("end", () => {
|
|
362
|
+
streamMetrics.duration = performance.now() - streamMetrics.startTime;
|
|
363
|
+
streamMetrics.endTime = performance.now();
|
|
364
|
+
rscHeadlessMetrics.streamMetrics = streamMetrics;
|
|
365
|
+
rscHeadlessMetrics.chunkRate = streamMetrics.chunks / (streamMetrics.duration / 1e3);
|
|
366
|
+
rscHeadlessMetrics.processingTime = streamMetrics.duration;
|
|
367
|
+
rscHeadlessMetrics.memoryUsage = process.memoryUsage();
|
|
368
|
+
rscHeadlessMetrics.chunks = streamMetrics.chunks;
|
|
369
|
+
});
|
|
370
|
+
rscFileStream.pipe(destination);
|
|
371
|
+
return destination;
|
|
372
|
+
},
|
|
373
|
+
abort: () => headlessRscHandler.abort()
|
|
374
|
+
};
|
|
375
|
+
const htmlStreamWrapper = {
|
|
376
|
+
pipe: (destination) => {
|
|
377
|
+
return htmlTransformStream.pipe(destination);
|
|
378
|
+
},
|
|
379
|
+
abort: () => {
|
|
380
|
+
htmlTransformStream.abort();
|
|
381
|
+
},
|
|
382
|
+
on: (event, listener) => {
|
|
383
|
+
if (event === "error") {
|
|
384
|
+
const htmlStream = htmlTransformStream.htmlStream;
|
|
385
|
+
if (htmlStream && typeof htmlStream.on === "function") {
|
|
386
|
+
htmlStream.on("error", listener);
|
|
387
|
+
}
|
|
488
388
|
}
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
493
|
-
|
|
494
|
-
|
|
389
|
+
return htmlStreamWrapper;
|
|
390
|
+
}
|
|
391
|
+
};
|
|
392
|
+
await new Promise((resolve, reject) => {
|
|
393
|
+
const timeout = setTimeout(() => {
|
|
394
|
+
reject(new Error(`HTML stream timeout for route ${handlerOptions.route}`));
|
|
395
|
+
}, handlerOptions.htmlTimeout || 15e3);
|
|
396
|
+
if (htmlStreamErrored) {
|
|
397
|
+
clearTimeout(timeout);
|
|
398
|
+
resolve();
|
|
399
|
+
return;
|
|
400
|
+
}
|
|
401
|
+
let resolved = false;
|
|
402
|
+
const originalPipe = htmlTransformStream.pipe;
|
|
403
|
+
htmlTransformStream.pipe = function(destination) {
|
|
404
|
+
const result = originalPipe.call(this, destination);
|
|
405
|
+
destination.on("finish", () => {
|
|
406
|
+
if (!resolved) {
|
|
407
|
+
resolved = true;
|
|
408
|
+
clearTimeout(timeout);
|
|
409
|
+
resolve();
|
|
410
|
+
}
|
|
411
|
+
});
|
|
412
|
+
destination.on("error", (error) => {
|
|
413
|
+
if (!resolved) {
|
|
414
|
+
resolved = true;
|
|
415
|
+
clearTimeout(timeout);
|
|
416
|
+
reject(error);
|
|
417
|
+
}
|
|
495
418
|
});
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
|
|
499
|
-
|
|
500
|
-
|
|
501
|
-
|
|
502
|
-
|
|
503
|
-
|
|
504
|
-
|
|
505
|
-
|
|
419
|
+
return result;
|
|
420
|
+
};
|
|
421
|
+
setTimeout(() => {
|
|
422
|
+
if (!resolved) {
|
|
423
|
+
resolved = true;
|
|
424
|
+
clearTimeout(timeout);
|
|
425
|
+
resolve();
|
|
426
|
+
}
|
|
427
|
+
}, 100);
|
|
428
|
+
});
|
|
429
|
+
if (htmlStreamErrored) {
|
|
430
|
+
yield {
|
|
431
|
+
type: "error",
|
|
432
|
+
error: htmlStreamError || new Error("HTML stream failed"),
|
|
433
|
+
metrics: {
|
|
434
|
+
rscFull: rscFullMetrics,
|
|
435
|
+
rscHeadless: rscHeadlessMetrics,
|
|
436
|
+
html: htmlMetrics
|
|
437
|
+
}
|
|
438
|
+
};
|
|
439
|
+
return;
|
|
440
|
+
}
|
|
441
|
+
yield {
|
|
442
|
+
type: "success",
|
|
443
|
+
html: htmlStreamWrapper,
|
|
444
|
+
rsc: rscStreamWrapper,
|
|
445
|
+
metrics: {
|
|
446
|
+
rscFull: rscFullMetrics,
|
|
447
|
+
rscHeadless: rscHeadlessMetrics,
|
|
448
|
+
html: htmlMetrics
|
|
449
|
+
}
|
|
450
|
+
};
|
|
451
|
+
} catch (err) {
|
|
452
|
+
if (handlerOptions.verbose) {
|
|
453
|
+
handlerOptions.logger?.error(`[renderPage.server] Error: ${JSON.stringify(err)}`);
|
|
454
|
+
}
|
|
455
|
+
try {
|
|
456
|
+
if (headlessRscHandler) headlessRscHandler.abort();
|
|
457
|
+
if (fullRscHandler) fullRscHandler.abort();
|
|
458
|
+
if (htmlTransformStream) htmlTransformStream.abort();
|
|
459
|
+
} catch (cleanupError) {
|
|
460
|
+
handlerOptions.logger?.warn(`Failed to cleanup streams on error: ${cleanupError}`);
|
|
461
|
+
}
|
|
462
|
+
const panicError = handleError({
|
|
463
|
+
error: err,
|
|
464
|
+
critical: false,
|
|
465
|
+
logger: handlerOptions.logger,
|
|
466
|
+
panicThreshold: handlerOptions.panicThreshold,
|
|
467
|
+
context: `RenderPage Error (${handlerOptions.route})`
|
|
468
|
+
});
|
|
469
|
+
if (panicError != null) {
|
|
470
|
+
yield {
|
|
471
|
+
type: "error",
|
|
472
|
+
error: panicError,
|
|
473
|
+
metrics: {
|
|
474
|
+
rscFull: rscFullMetrics,
|
|
475
|
+
rscHeadless: rscHeadlessMetrics,
|
|
476
|
+
html: htmlMetrics
|
|
506
477
|
}
|
|
507
|
-
|
|
508
|
-
|
|
509
|
-
|
|
510
|
-
|
|
511
|
-
|
|
512
|
-
|
|
513
|
-
|
|
514
|
-
|
|
515
|
-
|
|
516
|
-
|
|
517
|
-
|
|
518
|
-
|
|
519
|
-
|
|
520
|
-
|
|
521
|
-
|
|
522
|
-
|
|
523
|
-
|
|
524
|
-
|
|
525
|
-
|
|
526
|
-
|
|
527
|
-
|
|
528
|
-
|
|
529
|
-
|
|
530
|
-
|
|
478
|
+
};
|
|
479
|
+
} else {
|
|
480
|
+
yield {
|
|
481
|
+
type: "skip",
|
|
482
|
+
reason: err,
|
|
483
|
+
html: {
|
|
484
|
+
pipe: (destination) => {
|
|
485
|
+
destination.end();
|
|
486
|
+
return destination;
|
|
487
|
+
},
|
|
488
|
+
abort: () => {
|
|
489
|
+
}
|
|
490
|
+
},
|
|
491
|
+
rsc: {
|
|
492
|
+
pipe: (destination) => {
|
|
493
|
+
destination.end();
|
|
494
|
+
return destination;
|
|
495
|
+
},
|
|
496
|
+
abort: () => {
|
|
497
|
+
}
|
|
498
|
+
},
|
|
499
|
+
metrics: {
|
|
500
|
+
rscFull: rscFullMetrics,
|
|
501
|
+
rscHeadless: rscHeadlessMetrics,
|
|
502
|
+
html: htmlMetrics
|
|
531
503
|
}
|
|
504
|
+
};
|
|
532
505
|
}
|
|
506
|
+
}
|
|
533
507
|
};
|
|
508
|
+
|
|
509
|
+
export { renderPage };
|
|
510
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"renderPage.server.js","sources":["../../../plugin/react-static/renderPage.server.ts"],"sourcesContent":["/**\n * renderPage.server.ts\n *\n * PURPOSE: Server-side static page rendering for React Server Components\n *\n * ARCHITECTURE OVERVIEW:\n * \n * SERVER-SIDE vs CLIENT-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. Create headless RSC stream (for .rsc file)\n * 2. Create full RSC stream (for HTML generation)\n * 3. Create HTML transform stream that converts RSC to HTML\n * 4. Both streams are piped to file writers\n * \n * SIMPLIFIED APPROACH:\n * This implementation follows the same simple pattern as the client side,\n * avoiding complex backpressure handling and race conditions.\n */\n\nimport { createRenderMetrics } from \"../metrics/createRenderMetrics.js\";\nimport { routeToURL } from \"../utils/routeToURL.js\";\nimport type { RenderPageFn } from \"./types.js\";\nimport { handleError } from \"../error/handleError.js\";\nimport { assertReactServer } from \"../config/getCondition.js\";\n\nimport { renderRscStream } from \"../stream/renderRscStream.server.js\";\nimport { createMainThreadHandlers } from \"../stream/createMainThreadHandlers.js\";\nimport { createRscToHtmlStream } from \"./rscToHtmlStream.server.js\";\nimport { resolveComponent } from \"../helpers/resolveComponent.js\";\nimport { resolvePageAndProps } from \"../helpers/resolvePageAndProps.js\";\nimport { Root as DefaultRoot } from \"../components/root.js\";\nimport { Html as DefaultHtml } from \"../components/html.js\";\nimport { createStreamMetrics } from \"../metrics/createStreamMetrics.js\";\nimport { join } from \"node:path\";\nimport { createHeadlessStreamState, trackHeadlessStreamError, hasHeadlessStreamError } from \"../helpers/headlessStreamState.js\";\n\nexport const renderPage: RenderPageFn = async function* renderPage(\n  handlerOptions\n) {\n  // Ensure we're in the correct environment\n  assertReactServer();\n\n  // Create metrics upfront with proper types\n  const baseDir = join(\n    handlerOptions.build.outDir,\n    handlerOptions.build.static\n  );\n  const routePath = handlerOptions.route.replace(/^\\//, \"\");\n\n  const htmlMetrics = createRenderMetrics({\n    route: handlerOptions.route,\n    type: \"html\",\n    fromMainThread: false, // Server: HTML rendered in worker\n    fromRscWorker: false,\n    fromHtmlWorker: true,\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: true, // Server: RSC rendered on main thread\n    fromRscWorker: false,\n    fromHtmlWorker: false,\n  });\n  \n  const rscHeadlessMetrics = createRenderMetrics({\n    route: handlerOptions.route,\n    type: \"rsc-headless\",\n    fromMainThread: true, // Server: RSC rendered on main thread\n    fromRscWorker: false,\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\n  let headlessRscHandler: any = null;\n  let fullRscHandler: any = null;\n  let htmlTransformStream: any = null;\n  \n  // Error tracking variables for headless stream\n  let headlessStreamErrored = false;\n  let headlessError: Error | null = null;\n  \n  // Error tracking variables for HTML stream\n  let htmlStreamErrored = false;\n  let htmlStreamError: Error | null = null;\n\n  // Server-side stream reuse storage (similar to client-side headlessStreamElements)\n  const headlessStreamState = createHeadlessStreamState();\n\n  try {\n    if (handlerOptions.verbose) {\n      handlerOptions.logger?.info(\n        `[renderPage.server] Server-side rendering for route: ${handlerOptions.route}`\n      );\n    }\n\n    // Set URL if not provided\n    if (!handlerOptions.url) {\n      handlerOptions.url = routeToURL(\n        handlerOptions.route,\n        handlerOptions.moduleBaseURL,\n        handlerOptions.build.rscOutputPath\n      );\n    }\n\n    // Resolve components and props using the proper helper\n    let PageComponent: any = null;\n    let RootComponent: any = null;\n    let HtmlComponent: any = null;\n    let pageProps: any = {}; // Initialize as empty object - props function will populate it\n\n    // Use resolvePageAndProps helper to properly load page and props\n    if (handlerOptions.pagePath) {\n      try {\n        const pageAndPropsResult = await resolvePageAndProps({\n          pagePath: handlerOptions.pagePath,\n          pageExportName: handlerOptions.pageExportName,\n          propsPath: handlerOptions.propsPath,\n          propsExportName: handlerOptions.propsExportName,\n          loader: handlerOptions.loader,\n          verbose: handlerOptions.verbose,\n          logger: handlerOptions.logger,\n          route: handlerOptions.route,\n          url: handlerOptions.url,\n          moduleBaseURL: handlerOptions.moduleBaseURL,\n          build: {\n            rscOutputPath: handlerOptions.build.rscOutputPath,\n          },\n        });\n\n        if (pageAndPropsResult.type === \"success\") {\n          PageComponent = pageAndPropsResult.PageComponent;\n          // Always use the props returned from the props function\n          // Root components can handle empty props with their defaults\n          pageProps = pageAndPropsResult.pageProps || {};\n          \n          if (handlerOptions.verbose) {\n            handlerOptions.logger?.info(\n              `[renderPage.server] Successfully loaded page and props for route ${handlerOptions.route}: pageProps=${JSON.stringify(pageProps)}`\n            );\n          }\n        } else {\n          handlerOptions.logger?.warn(\n            `Failed to load page and props from ${handlerOptions.pagePath}: ${\n              pageAndPropsResult.error?.message || \"Unknown error\"\n            }`\n          );\n        }\n      } catch (error) {\n        handlerOptions.logger?.warn(\n          `Error loading page and props from ${handlerOptions.pagePath}: ${\n            error instanceof Error ? error.message : String(error)\n          }`\n        );\n      }\n    }\n\n    // Load Root component\n    if (handlerOptions.rootPath) {\n      try {\n        const rootResult = await resolveComponent({\n          componentPath: handlerOptions.rootPath,\n          exportName: handlerOptions.rootExportName,\n          loader: handlerOptions.loader,\n        });\n        if (rootResult.type === \"success\") {\n          RootComponent = rootResult.component;\n        } else {\n          handlerOptions.logger?.warn(\n            `Failed to load Root component from ${handlerOptions.rootPath}: ${\n              rootResult.error?.message || \"Unknown error\"\n            }`\n          );\n        }\n      } catch (error) {\n        handlerOptions.logger?.warn(\n          `Error loading Root component from ${handlerOptions.rootPath}: ${\n            error instanceof Error ? error.message : String(error)\n          }`\n        );\n      }\n    }\n\n    // Load Html component\n    if (handlerOptions.htmlPath) {\n      try {\n        const htmlResult = await resolveComponent({\n          componentPath: handlerOptions.htmlPath,\n          exportName: handlerOptions.htmlExportName,\n          loader: handlerOptions.loader,\n        });\n        if (htmlResult.type === \"success\") {\n          HtmlComponent = htmlResult.component;\n        } else {\n          handlerOptions.logger?.warn(\n            `Failed to load Html component from ${handlerOptions.htmlPath}: ${\n              htmlResult.error?.message || \"Unknown error\"\n            }`\n          );\n        }\n      } catch (error) {\n        handlerOptions.logger?.warn(\n          `Error loading Html component from ${handlerOptions.htmlPath}: ${\n            error instanceof Error ? error.message : String(error)\n          }`\n        );\n      }\n    }\n\n    // Use defaults if components are still not loaded\n    if (!RootComponent) {\n      RootComponent = DefaultRoot as any;\n    }\n    if (!HtmlComponent) {\n      HtmlComponent = DefaultHtml as any;\n    }\n\n    // Ensure we have all required components\n    if (!PageComponent || !RootComponent || !HtmlComponent) {\n      yield {\n        type: \"error\",\n        error: new Error(\n          `Component resolution failed: missing required components (Page: ${!!PageComponent}, Root: ${!!RootComponent}, Html: ${!!HtmlComponent})`\n        ),\n        metrics: {\n          rscFull: rscFullMetrics,\n          rscHeadless: rscHeadlessMetrics,\n          html: htmlMetrics,\n        },\n      };\n      return;\n    }\n\n    // Create handler options with resolved components and props\n    const uniqueId = handlerOptions.id ?? `${handlerOptions.route}?id=${Date.now()}-${Math.random().toString(36).substring(2, 11)}`;\n    const newHandlerOptions = {\n      ...handlerOptions,\n      id: uniqueId,\n      url: `${handlerOptions.url}`,\n      route: `${handlerOptions.route}`,\n      PageComponent,\n      RootComponent,\n      HtmlComponent,\n      pageProps,\n    };\n    \n    if (handlerOptions.verbose) {\n      handlerOptions.logger?.info(\n        `[renderPage.server] Created newHandlerOptions for route ${handlerOptions.route} with pageProps: ${JSON.stringify(pageProps)}`\n      );\n    }\n\n    // Create headless RSC handler (for .rsc file) - with proper error handling\n    const headlessHandlers = createMainThreadHandlers(\n      handlerOptions,\n      (error, isPanic) => {\n        if (handlerOptions.verbose) {\n          handlerOptions.logger?.info(\n            `[renderPage.server] Headless stream error handler called for route ${handlerOptions.route}: ${error.message}, isPanic: ${isPanic}`\n          );\n        }\n        \n        // Track if the headless stream had errors\n        headlessStreamErrored = true;\n        headlessError = error instanceof Error ? error : new Error(\"Headless RSC stream failed\");\n        \n        // Track headless stream errors for conditional reuse logic (like RSC worker)\n        trackHeadlessStreamError(headlessStreamState, handlerOptions.route, headlessError);\n        \n        if (handlerOptions.verbose) {\n          handlerOptions.logger?.info(\n            `[renderPage.server] Stored headless stream error for route ${handlerOptions.route} in headlessStreamErrors map`\n          );\n        }\n        \n        // Store panic errors for later handling\n        if (isPanic) {\n          // For panic threshold \"all_errors\", the panic error will be handled by renderPages\n          if (handlerOptions.verbose) {\n            handlerOptions.logger?.info(\n              `[renderPage.server] Panic error detected for route ${handlerOptions.route}, will be handled by renderPages`\n            );\n          }\n        }\n      }\n    );\n    \n    // Override onData to track metrics\n    headlessHandlers.onData = (_id, chunk) => {\n      rscHeadlessMetrics.chunks++;\n      rscHeadlessMetrics.streamMetrics.bytes += chunk.length;\n    };\n    \n    headlessRscHandler = renderRscStream(\n      {\n        ...newHandlerOptions,\n        htmlPath: '', // Headless RSC - no HTML wrapper\n        // If we expect errors, provide a safe Page component that doesn't throw\n        PageComponent: newHandlerOptions.PageComponent, // Use original for now, will be overridden if errors occur\n      },\n      headlessHandlers\n    );\n    \n    // Note: Panic errors will be yielded from the error handler when they occur\n    // No need to check shouldYieldPanicError here as it's set asynchronously\n\n    // Store PageComponent for reuse when headless stream completes (like RSC worker)\n    headlessRscHandler.rscStream.on('end', () => {\n      // Only store if this is a headless stream and no errors occurred (like RSC worker)\n      if (!hasHeadlessStreamError(headlessStreamState, handlerOptions.route)) {\n        headlessStreamState.elements.set(uniqueId, {\n          PageComponent: newHandlerOptions.PageComponent,\n          errored: false\n        });\n        if (handlerOptions.verbose) {\n          handlerOptions.logger?.info(`[renderPage.server] Stored PageComponent for headless stream ${uniqueId}`);\n        }\n      } else {\n        if (handlerOptions.verbose) {\n          handlerOptions.logger?.info(`[renderPage.server] Headless stream errored for route ${handlerOptions.route}, not storing PageComponent for reuse`);\n        }\n      }\n    });\n\n    // Create full RSC handler (for HTML generation) - reuse headless stream elements if no errors\n    // For server-side, we create both streams in parallel like the client-side\n    let fullPanicError: Error | null = null;\n    const fullHandlers = createMainThreadHandlers(\n      handlerOptions,\n      (error, isPanic) => {\n        // If this is a panic error, store it to be handled later\n        if (isPanic) {\n          fullPanicError = error instanceof Error ? error : new Error(\"Full RSC stream failed\");\n        }\n      }\n    );\n    \n    // Override onData to track metrics\n    fullHandlers.onData = (_id, chunk) => {\n      rscFullMetrics.chunks++;\n      rscFullMetrics.streamMetrics.bytes += chunk.length;\n    };\n    \n    // Create full RSC handler options - use React.Fragment if headless stream had errors (like RSC worker)\n    // Check if there are any existing headless stream errors for this route\n    const hasExistingHeadlessError = hasHeadlessStreamError(headlessStreamState, handlerOptions.route);\n    const shouldUseFallback = headlessStreamErrored || hasExistingHeadlessError;\n    \n    if (handlerOptions.verbose) {\n      handlerOptions.logger?.info(\n        `[renderPage.server] Creating full RSC handler options for route ${handlerOptions.route}: headlessStreamErrored=${headlessStreamErrored}, hasExistingHeadlessError=${hasExistingHeadlessError}, shouldUseFallback=${shouldUseFallback}`\n      );\n    }\n    \n    // Create a wrapper PageComponent that returns null if there are headless stream errors\n    const SafePageComponent = (props: any) => {\n      // Check if there are any headless stream errors for this route\n      const hasError = hasHeadlessStreamError(headlessStreamState, handlerOptions.route);\n      if (hasError) {\n        return null;\n      }\n      return newHandlerOptions.PageComponent(props);\n    };\n    \n    const fullRscHandlerOptions = {\n      ...newHandlerOptions,\n      htmlPath: undefined, // Full RSC - include HTML wrapper\n      headlessStreamElements: headlessStreamState.elements, // Pass the storage map for reuse\n      // Use SafePageComponent that returns null when there are headless stream errors\n      PageComponent: SafePageComponent,\n    };\n    \n    \n    // Create a PageComponent that uses React.use() to consume the headless stream and check for errors\n\n    // Store the headless stream elements for reuse (like RSC worker does)\n    \n    // Listen for the headless stream to complete and store its elements\n    headlessRscHandler.rscStream.on('end', () => {\n      if (!hasHeadlessStreamError(headlessStreamState, handlerOptions.route)) {\n        if (handlerOptions.verbose) {\n          handlerOptions.logger?.info(\n            `[renderPage.server] Headless stream completed successfully for route ${handlerOptions.route}`\n          );\n        }\n      }\n    });\n\n    \n    if (handlerOptions.verbose) {\n      handlerOptions.logger?.info(\n        `[renderPage.server] Created PageComponent that uses React.use() to consume headless stream for route ${handlerOptions.route}`\n      );\n    }\n    \n    fullRscHandler = renderRscStream(fullRscHandlerOptions, fullHandlers);\n    \n    // Check for panic error after creating the handler\n    if (fullPanicError) {\n      yield {\n        type: \"error\",\n        error: fullPanicError,\n        metrics: {\n          rscFull: rscFullMetrics,\n          rscHeadless: rscHeadlessMetrics,\n          html: htmlMetrics,\n        },\n      };\n      return;\n    }\n\n    // Create HTML transform stream - need createRscToHtmlStream for async server actions\n    htmlTransformStream = createRscToHtmlStream({\n      id: handlerOptions.id,\n      worker: handlerOptions.worker,\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      build: handlerOptions.build,\n      panicThreshold: handlerOptions.panicThreshold,\n      verbose: handlerOptions.verbose,\n      signal: handlerOptions.signal,\n      logger: handlerOptions.logger,\n      htmlWorker: handlerOptions.htmlWorker,\n      clientPipeableStreamOptions: handlerOptions.clientPipeableStreamOptions,\n      onMetrics: handlerOptions.onMetrics,\n      htmlTimeout: handlerOptions.htmlTimeout || 15000,\n      rscStream: fullRscHandler.rscStream,\n      onError: (error, isPanic) => {\n        // Track HTML stream errors\n        htmlStreamErrored = true;\n        htmlStreamError = error;\n        \n        if (isPanic) {\n          // This is a panic error, it should be yielded as an error result\n          if (handlerOptions.verbose) {\n            handlerOptions.logger?.error(\n              `[renderPage.server] HTML stream panic error for route ${handlerOptions.route}: ${error.message}`\n            );\n          }\n        } else {\n          // For non-panic errors, just log them\n          if (handlerOptions.verbose) {\n            handlerOptions.logger?.warn(\n              `[renderPage.server] HTML stream error for route ${handlerOptions.route}: ${error.message}`\n            );\n          }\n        }\n      },\n    });\n\n    // Create stream wrappers for file writing - simplified like client side\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 = headlessRscHandler.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: () => headlessRscHandler.abort(),\n    };\n\n    const htmlStreamWrapper = {\n      pipe: <Writable extends NodeJS.WritableStream>(destination: Writable) => {\n        // Use the HTML transform stream's pipe method directly (same as client side)\n        return htmlTransformStream.pipe(destination);\n      },\n      abort: () => {\n        htmlTransformStream.abort();\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    // Wait for HTML stream to complete or error before yielding success\n    // This ensures that any errors from the HTML stream are caught before we yield success\n    await new Promise<void>((resolve, reject) => {\n      const timeout = setTimeout(() => {\n        reject(new Error(`HTML stream timeout for route ${handlerOptions.route}`));\n      }, handlerOptions.htmlTimeout || 15000);\n\n      // Check if HTML stream already errored\n      if (htmlStreamErrored) {\n        clearTimeout(timeout);\n        resolve(); // Let the panic threshold logic handle this at the renderPages level\n        return;\n      }\n\n      // Set up a flag to track if we've resolved\n      let resolved = false;\n      \n      // Create a wrapper that resolves the promise when the stream completes\n      const originalPipe = htmlTransformStream.pipe;\n      htmlTransformStream.pipe = function(destination: any) {\n        const result = originalPipe.call(this, destination);\n        \n        // Listen for the destination stream to end\n        destination.on('finish', () => {\n          if (!resolved) {\n            resolved = true;\n            clearTimeout(timeout);\n            resolve();\n          }\n        });\n        \n        destination.on('error', (error: Error) => {\n          if (!resolved) {\n            resolved = true;\n            clearTimeout(timeout);\n            reject(error);\n          }\n        });\n        \n        return result;\n      };\n      \n      // If we don't have a destination yet, resolve after a short delay\n      // This handles the case where the stream is created but not yet piped\n      setTimeout(() => {\n        if (!resolved) {\n          resolved = true;\n          clearTimeout(timeout);\n          resolve();\n        }\n      }, 100);\n    });\n\n    // Check for HTML stream errors after waiting for completion\n    if (htmlStreamErrored) {\n      yield {\n        type: \"error\",\n        error: htmlStreamError || new Error(\"HTML stream failed\"),\n        metrics: {\n          rscFull: rscFullMetrics,\n          rscHeadless: rscHeadlessMetrics,\n          html: htmlMetrics,\n        },\n      };\n      return;\n    }\n\n    // Yield success result - simplified like client side\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\n  } catch (err) {\n    if (handlerOptions.verbose) {\n      handlerOptions.logger?.error(`[renderPage.server] Error: ${JSON.stringify(err)}`);\n    }\n\n    // Clean up any resources\n    try {\n      if (headlessRscHandler) headlessRscHandler.abort();\n      if (fullRscHandler) fullRscHandler.abort();\n      if (htmlTransformStream) htmlTransformStream.abort();\n    } catch (cleanupError: unknown) {\n      handlerOptions.logger?.warn(`Failed to cleanup streams on error: ${cleanupError}`);\n    }\n\n    const panicError = handleError({\n      error: err,\n      critical: false,\n      logger: handlerOptions.logger,\n      panicThreshold: handlerOptions.panicThreshold,\n      context: `RenderPage Error (${handlerOptions.route})`,\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      yield {\n        type: \"skip\",\n        reason: err,\n        html: {\n          pipe: <Writable extends NodeJS.WritableStream>(destination: Writable) => {\n            destination.end();\n            return destination;\n          },\n          abort: () => {},\n        },\n        rsc: {\n          pipe: <Writable extends NodeJS.WritableStream>(destination: Writable) => {\n            destination.end();\n            return destination;\n          },\n          abort: () => {},\n        },\n        metrics: {\n          rscFull: rscFullMetrics,\n          rscHeadless: rscHeadlessMetrics,\n          html: htmlMetrics,\n        },\n      };\n    }\n  }\n};"],"names":["renderPage","DefaultRoot","DefaultHtml"],"mappings":"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"}
|