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.
Files changed (74) hide show
  1. package/README.md +48 -313
  2. package/dist/package.json +123 -13
  3. package/dist/plugin/bundle/deferredStaticGeneration.js +14 -39
  4. package/dist/plugin/bundle/manifests.js +30 -48
  5. package/dist/plugin/config/autoDiscover/resolveAutoDiscover.d.ts.map +1 -1
  6. package/dist/plugin/config/autoDiscover/resolveAutoDiscover.js +4 -1
  7. package/dist/plugin/config/envPrefixFromConfig.js +12 -7
  8. package/dist/plugin/config/getCondition.d.ts.map +1 -1
  9. package/dist/plugin/config/getCondition.js +7 -5
  10. package/dist/plugin/dev-server/virtualRscHmrPlugin.js +23 -23
  11. package/dist/plugin/environments/createBuildEventPlugin.js +88 -98
  12. package/dist/plugin/environments/createEnvironmentPlugin.js +222 -250
  13. package/dist/plugin/helpers/createRscRenderHelpers.js +33 -34
  14. package/dist/plugin/helpers/createSharedLoader.d.ts.map +1 -1
  15. package/dist/plugin/helpers/createSharedLoader.js +4 -2
  16. package/dist/plugin/helpers/headlessStreamReuseHandler.js +30 -22
  17. package/dist/plugin/helpers/headlessStreamState.js +15 -28
  18. package/dist/plugin/helpers/resolveComponent.d.ts.map +1 -1
  19. package/dist/plugin/helpers/resolveComponent.js +4 -2
  20. package/dist/plugin/index.client.d.ts +5 -0
  21. package/dist/plugin/index.client.d.ts.map +1 -0
  22. package/dist/plugin/index.client.js +4 -0
  23. package/dist/plugin/index.d.ts +4 -3
  24. package/dist/plugin/index.d.ts.map +1 -1
  25. package/dist/plugin/index.js +10 -5
  26. package/dist/plugin/index.server.d.ts +5 -0
  27. package/dist/plugin/index.server.d.ts.map +1 -0
  28. package/dist/plugin/index.server.js +4 -0
  29. package/dist/plugin/metrics/createWorkerStartupMetrics.js +31 -13
  30. package/dist/plugin/orchestrator/createPluginOrchestrator.client.js +41 -38
  31. package/dist/plugin/orchestrator/createPluginOrchestrator.server.js +43 -46
  32. package/dist/plugin/plugin.client.js +2 -2
  33. package/dist/plugin/plugin.server.js +2 -2
  34. package/dist/plugin/react-static/createBuildLoader.client.js +12 -6
  35. package/dist/plugin/react-static/createBuildLoader.server.js +255 -235
  36. package/dist/plugin/react-static/plugin.client.js +684 -770
  37. package/dist/plugin/react-static/plugin.server.js +517 -603
  38. package/dist/plugin/react-static/processCssFilesForPages.js +103 -88
  39. package/dist/plugin/react-static/renderPage.client.js +455 -529
  40. package/dist/plugin/react-static/renderPage.server.js +485 -508
  41. package/dist/plugin/react-static/renderPagesBatched.js +277 -275
  42. package/dist/plugin/react-static/rscToHtmlStream.client.js +48 -29
  43. package/dist/plugin/react-static/rscToHtmlStream.server.js +62 -37
  44. package/dist/plugin/react-static/temporaryReferences.server.js +11 -2
  45. package/dist/plugin/stream/createMainThreadHandlers.js +40 -31
  46. package/dist/plugin/stream/renderRscStream.server.d.ts.map +1 -1
  47. package/dist/plugin/stream/renderRscStream.server.js +127 -144
  48. package/dist/plugin/transformer/createTransformerPlugin.js +226 -265
  49. package/dist/plugin/utils/checkReactVersion.d.ts +7 -0
  50. package/dist/plugin/utils/checkReactVersion.d.ts.map +1 -0
  51. package/dist/plugin/utils/checkReactVersion.js +23 -0
  52. package/dist/plugin/utils/envUrls.node.js +12 -11
  53. package/dist/plugin/vendor/vendor-alias.js +84 -114
  54. package/dist/plugin/vendor/vendor.client.d.ts.map +1 -1
  55. package/dist/plugin/vendor/vendor.client.js +1 -3
  56. package/dist/plugin/worker/rsc/handleRscRender.d.ts.map +1 -1
  57. package/dist/plugin/worker/rsc/handleRscRender.js +3 -1
  58. package/dist/tsconfig.tsbuildinfo +1 -1
  59. package/package.json +123 -13
  60. package/plugin/config/autoDiscover/resolveAutoDiscover.ts +4 -0
  61. package/plugin/config/getCondition.ts +6 -4
  62. package/plugin/helpers/createRscRenderHelpers.ts +1 -1
  63. package/plugin/helpers/createSharedLoader.ts +6 -1
  64. package/plugin/helpers/resolveComponent.ts +6 -1
  65. package/plugin/index.client.ts +4 -0
  66. package/plugin/index.server.ts +4 -0
  67. package/plugin/index.ts +12 -5
  68. package/plugin/plugin.client.ts +1 -1
  69. package/plugin/plugin.server.ts +1 -1
  70. package/plugin/stream/renderRscStream.server.ts +3 -0
  71. package/plugin/utils/checkReactVersion.ts +28 -0
  72. package/plugin/vendor/vendor.client.ts +0 -2
  73. package/plugin/worker/rsc/handleRscRender.ts +2 -0
  74. package/scripts/generate-toc.mjs +27 -294
@@ -1,555 +1,481 @@
1
1
  /**
2
- * renderPage.client.ts
3
- *
4
- * PURPOSE: Client-side static page rendering for React Server Components
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 "../metrics/createRenderMetrics.js";
42
- import { routeToURL } from "../utils/routeToURL.js";
43
- import { handleError } from "../error/handleError.js";
44
- import { assertNonReactServer } from "../config/getCondition.js";
45
- import { createRscStream } from "../stream/createRscStream.client.js";
46
- import { resolveComponents } from "../helpers/resolveComponents.client.js";
47
- import { join } from "node:path";
48
- import { createStreamMetrics } from "../metrics/createStreamMetrics.js";
49
- import { performance } from "node:perf_hooks";
50
- import { createRscToHtmlStream } from "./rscToHtmlStream.client.js";
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
- * Client version of renderPage that uses the react-client pattern
54
- * This works in REVERSE from the server plugin:
55
- * - Server: Main thread (RSC) + HTML worker (HTML)
56
- * - Client: RSC worker (RSC) + Main thread (HTML)
57
- */
58
- export const renderPage = async function* _renderPageClient(handlerOptions) {
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
- handlerOptions.logger?.info(`[renderPage.client] onEvent callback exists: ${!!handlerOptions.onEvent}`);
61
- handlerOptions.logger?.info(`[renderPage.client] onMetrics callback exists: ${!!handlerOptions.onMetrics}`);
153
+ handlerOptions.logger?.info(
154
+ `[renderPage.client] Client-side rendering for route: ${handlerOptions.route}`
155
+ );
62
156
  }
63
- // Track if we've yielded a result to prevent multiple yields
64
- let hasYielded = false;
65
- let errorResult = null;
66
- // Create a wrapper around onEvent to handle route.error events
67
- const wrappedOnEvent = (event) => {
68
- // Call the original onEvent first
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
- // Skip if no pagePath AND no PageComponent provided (fallback case)
109
- if (!handlerOptions.pagePath && !handlerOptions.PageComponent) {
110
- // Create empty stream wrappers for skip case
111
- const emptyStreamWrapper = {
112
- pipe: (destination) => {
113
- destination.end();
114
- return destination;
115
- },
116
- abort: () => {
117
- // No cleanup needed
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
- type: "skip",
122
- reason: "No pagePath and no PageComponent provided",
123
- html: emptyStreamWrapper,
124
- rsc: emptyStreamWrapper,
125
- metrics: {
126
- rscFull: createRenderMetrics({
127
- route: handlerOptions.route,
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
- if (!handlerOptions.url) {
152
- handlerOptions.url = routeToURL(handlerOptions.route, handlerOptions.moduleBaseURL, handlerOptions.build.rscOutputPath);
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 baseDir = join(handlerOptions.build.outDir, handlerOptions.build.static);
155
- const routePath = handlerOptions.route.replace(/^\//, "");
156
- // Create metrics upfront with proper types - REVERSE from server
157
- const htmlMetrics = createRenderMetrics({
158
- route: handlerOptions.route,
159
- type: "html",
160
- fromMainThread: true, // Client: HTML rendered on main thread
161
- fromRscWorker: false,
162
- fromHtmlWorker: false,
163
- baseDir,
164
- routePath,
165
- fileName: handlerOptions.build.htmlOutputPath,
166
- outputPath: join(baseDir, routePath, handlerOptions.build.htmlOutputPath),
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 rscFullMetrics = createRenderMetrics({
169
- route: handlerOptions.route,
170
- type: "rsc-full",
171
- fromMainThread: false,
172
- fromRscWorker: true, // Client: RSC rendered on RSC worker
173
- fromHtmlWorker: false,
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
- const rscHeadlessMetrics = createRenderMetrics({
176
- route: handlerOptions.route,
177
- type: "rsc-headless",
178
- fromMainThread: false,
179
- fromRscWorker: true, // Client: RSC rendered on RSC worker
180
- fromHtmlWorker: false,
181
- baseDir,
182
- routePath,
183
- fileName: handlerOptions.build.rscOutputPath,
184
- outputPath: join(baseDir, routePath, handlerOptions.build.rscOutputPath),
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
- // Declare variables outside try block so they can be accessed in catch block
187
- let headlessRscStream = null;
188
- let fullRscStream = null;
189
- let htmlHandler = null;
190
- try {
191
- if (handlerOptions.verbose) {
192
- handlerOptions.logger?.info(`[renderPage.client] Client-side rendering for route: ${handlerOptions.route}`);
193
- }
194
- // Step 1: Resolve paths to built paths using the server manifest
195
- // The client version needs to use the server manifest to get the built paths
196
- // for the page components, not the static manifest
197
- const resolvePathWithManifest = (path, manifest) => {
198
- const entry = manifest[path];
199
- if (entry && entry.file) {
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
- // Create full RSC stream that reuses the headless stream elements
345
- const fullRscStreamLocal = createRscStream({
346
- ...newHandlerOptions,
347
- id: `${handlerOptions.route}-full-${uniqueId}`,
348
- rscTimeout: handlerOptions.rscTimeout || 5000,
349
- onMetrics: handlerOptions.onMetrics,
350
- // Full RSC stream: include HTML wrapper (for HTML generation)
351
- // Pass through the resolved htmlPath so custom Html components work in client mode
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
- // Assign to the outer variables
361
- headlessRscStream = headlessRscStreamLocal;
362
- fullRscStream = fullRscStreamLocal;
363
- // The headless stream will be consumed naturally by the file writing
364
- // The full stream will reuse the headless stream elements for HTML generation
365
- // Step 3: Create HTML transform stream
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
- handlerOptions.logger?.info(`[renderPage.client] Creating HTML transform stream with clientPipeableStreamOptions: ${JSON.stringify(newHandlerOptions.clientPipeableStreamOptions)}`);
356
+ handlerOptions.logger?.info(
357
+ `[renderPage.client] Piping HTML stream to destination for route: ${handlerOptions.route}`
358
+ );
368
359
  }
369
- // Create HTML stream using the full RSC stream (which reuses headless stream elements)
370
- const htmlTransformStream = createRscToHtmlStream({
371
- ...newHandlerOptions,
372
- htmlTimeout: handlerOptions.htmlTimeout || 15000,
373
- route: handlerOptions.route,
374
- logger: handlerOptions.logger,
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
- yield {
445
- type: "success",
446
- html: htmlStreamWrapper,
447
- rsc: rscStreamWrapper,
448
- metrics: {
449
- rscFull: rscFullMetrics,
450
- rscHeadless: rscHeadlessMetrics,
451
- html: htmlMetrics,
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
- catch (error) {
456
- // Clean up resources
457
- try {
458
- if (headlessRscStream)
459
- headlessRscStream.abort();
460
- if (fullRscStream)
461
- fullRscStream.abort();
462
- if (htmlHandler?.abort)
463
- htmlHandler.abort();
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
- catch (cleanupError) {
466
- handlerOptions.logger?.warn(`Failed to cleanup streams on error: ${cleanupError}`);
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
- const panicError = handleError({
469
- error,
470
- logger: handlerOptions.logger,
471
- context: "renderPageClient",
472
- panicThreshold: handlerOptions.panicThreshold,
473
- });
474
- if (panicError != null) {
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
- else {
486
- // For non-panic errors, we still want to write the HTML file (client-only)
487
- // but skip the RSC file since there was a server error
488
- // Create a fallback RSC stream with React.Fragment (same as server environment)
489
- const fallbackRscStream = createRscStream({
490
- ...handlerOptions,
491
- url: `${handlerOptions.url}`,
492
- route: `${handlerOptions.route}`,
493
- cssFiles: handlerOptions.cssFiles || new Map(),
494
- globalCss: handlerOptions.globalCss || new Map(),
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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"}