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,295 +1,297 @@
1
- import { handleError } from "../error/handleError.js";
2
- import { fileWriter } from "./fileWriter.js";
3
- import { createRenderMetrics } from "../metrics/createRenderMetrics.js";
4
- import { createStreamMetrics } from "../metrics/createStreamMetrics.js";
1
+ /**
2
+ * vite-plugin-react-server
3
+ * Copyright (c) Nico Brinkkemper
4
+ * MIT License
5
+ */
6
+ import { handleError } from '../error/handleError.js';
7
+ import { fileWriter } from './fileWriter.js';
8
+ import { createRenderMetrics } from '../metrics/createRenderMetrics.js';
9
+ import { createStreamMetrics } from '../metrics/createStreamMetrics.js';
10
+
5
11
  const DEFAULT_BATCH_SIZE = 8;
6
12
  function resolvePathWithManifest(path, manifest) {
7
- const entry = manifest[path];
8
- if (entry && entry.file) {
9
- return entry.file;
10
- }
11
- return path;
13
+ const entry = manifest[path];
14
+ if (entry && entry.file) {
15
+ return entry.file;
16
+ }
17
+ return path;
12
18
  }
13
- /**
14
- * Renders a single route completely, consuming all yields from renderPage
15
- * and writing the RSC and HTML files. Collects metrics and handles events
16
- * identically to the sequential renderPages.
17
- */
18
19
  async function renderSingleRoute(route, handlerOptions, renderPage, manifest, failedRoutes) {
19
- const { autoDiscoveredFiles, cssFilesByPage, ...options } = handlerOptions;
20
- const { page, props, root, html } = autoDiscoveredFiles.urlMap?.get(route) || {};
21
- if (!page) {
22
- return { route, results: [], error: new Error(`No page found for route ${route}`) };
23
- }
24
- try {
25
- const resolvedPagePath = page ? resolvePathWithManifest(page, manifest) : undefined;
26
- const resolvedPropsPath = props ? resolvePathWithManifest(props, manifest) : undefined;
27
- const resolvedRootPath = root ? resolvePathWithManifest(root, manifest) : undefined;
28
- const resolvedHtmlPath = html ? resolvePathWithManifest(html, manifest) : undefined;
29
- // Store results for metrics tracking
30
- const routeResults = new Map();
31
- // Create onEvent wrapper that handles route.error and metrics collection
32
- // This mirrors the sequential renderPages behavior exactly
33
- const wrapperOnEvent = (event) => {
34
- // Call the original onEvent first
35
- if (options.onEvent) {
36
- options.onEvent(event);
37
- }
38
- // Handle route.error events
39
- if (event.type === "route.error") {
40
- const detectedPanicError = handleError({
41
- error: event.data.error,
42
- logger: options.logger,
43
- panicThreshold: event.data.panicThreshold,
44
- context: `route.error (${event.data.route})`,
45
- });
46
- if (detectedPanicError != null) {
47
- options.logger?.error(`[renderPagesBatched] Panic error for route ${event.data.route}: ${event.data.error.message}`);
48
- failedRoutes.set(event.data.route, event.data.error);
49
- }
50
- else {
51
- options.logger?.warn(`[renderPagesBatched] Non-panic error for route ${event.data.route}: ${event.data.error.message}`);
52
- }
20
+ const { autoDiscoveredFiles, cssFilesByPage, ...options } = handlerOptions;
21
+ const { page, props, root, html } = autoDiscoveredFiles.urlMap?.get(route) || {};
22
+ if (!page) {
23
+ return { route, results: [], error: new Error(`No page found for route ${route}`) };
24
+ }
25
+ try {
26
+ const resolvedPagePath = page ? resolvePathWithManifest(page, manifest) : void 0;
27
+ const resolvedPropsPath = props ? resolvePathWithManifest(props, manifest) : void 0;
28
+ const resolvedRootPath = root ? resolvePathWithManifest(root, manifest) : void 0;
29
+ const resolvedHtmlPath = html ? resolvePathWithManifest(html, manifest) : void 0;
30
+ const routeResults = /* @__PURE__ */ new Map();
31
+ const wrapperOnEvent = (event) => {
32
+ if (options.onEvent) {
33
+ options.onEvent(event);
34
+ }
35
+ if (event.type === "route.error") {
36
+ const detectedPanicError = handleError({
37
+ error: event.data.error,
38
+ logger: options.logger,
39
+ panicThreshold: event.data.panicThreshold,
40
+ context: `route.error (${event.data.route})`
41
+ });
42
+ if (detectedPanicError != null) {
43
+ options.logger?.error(
44
+ `[renderPagesBatched] Panic error for route ${event.data.route}: ${event.data.error.message}`
45
+ );
46
+ failedRoutes.set(event.data.route, event.data.error);
47
+ } else {
48
+ options.logger?.warn(
49
+ `[renderPagesBatched] Non-panic error for route ${event.data.route}: ${event.data.error.message}`
50
+ );
51
+ }
52
+ }
53
+ if (event.type === "file.write.done" && event.data.route === route) {
54
+ const routeResult = routeResults.get(route);
55
+ if (routeResult && routeResult.type === "success") {
56
+ if (event.data.fileType === "html") {
57
+ const endTime = performance.now();
58
+ const htmlMetrics = createRenderMetrics({
59
+ route,
60
+ type: routeResult.metrics.html.type,
61
+ fromMainThread: routeResult.metrics.html.fromMainThread,
62
+ fromRscWorker: routeResult.metrics.html.fromRscWorker,
63
+ fromHtmlWorker: routeResult.metrics.html.fromHtmlWorker,
64
+ fileSize: event.data.content.length,
65
+ chunks: event.data.chunks || 0,
66
+ processingTime: endTime - routeResult.metrics.html.streamMetrics.startTime,
67
+ chunkRate: (event.data.chunks || 0) / ((endTime - routeResult.metrics.html.streamMetrics.startTime) / 1e3),
68
+ fileName: event.data.fileName,
69
+ outputPath: event.data.path,
70
+ baseDir: event.data.baseDir,
71
+ routePath: event.data.routePath,
72
+ streamMetrics: createStreamMetrics({
73
+ ...routeResult.metrics.html.streamMetrics,
74
+ chunks: event.data.chunks || 0,
75
+ bytes: event.data.content.length,
76
+ duration: endTime - routeResult.metrics.html.streamMetrics.startTime,
77
+ endTime
78
+ })
79
+ });
80
+ if (options.onMetrics) {
81
+ options.onMetrics(htmlMetrics);
53
82
  }
54
- // Handle metrics collection on file.write.done
55
- if (event.type === "file.write.done" && event.data.route === route) {
56
- const routeResult = routeResults.get(route);
57
- if (routeResult && routeResult.type === "success") {
58
- if (event.data.fileType === "html") {
59
- const endTime = performance.now();
60
- const htmlMetrics = createRenderMetrics({
61
- route: route,
62
- type: routeResult.metrics.html.type,
63
- fromMainThread: routeResult.metrics.html.fromMainThread,
64
- fromRscWorker: routeResult.metrics.html.fromRscWorker,
65
- fromHtmlWorker: routeResult.metrics.html.fromHtmlWorker,
66
- fileSize: event.data.content.length,
67
- chunks: event.data.chunks || 0,
68
- processingTime: endTime - routeResult.metrics.html.streamMetrics.startTime,
69
- chunkRate: (event.data.chunks || 0) / ((endTime - routeResult.metrics.html.streamMetrics.startTime) / 1000),
70
- fileName: event.data.fileName,
71
- outputPath: event.data.path,
72
- baseDir: event.data.baseDir,
73
- routePath: event.data.routePath,
74
- streamMetrics: createStreamMetrics({
75
- ...routeResult.metrics.html.streamMetrics,
76
- chunks: event.data.chunks || 0,
77
- bytes: event.data.content.length,
78
- duration: endTime - routeResult.metrics.html.streamMetrics.startTime,
79
- endTime: endTime,
80
- }),
81
- });
82
- if (options.onMetrics) {
83
- options.onMetrics(htmlMetrics);
84
- }
85
- // Also emit RSC Full metrics if available
86
- if (routeResult.metrics?.rscFull) {
87
- const rscFullEndTime = performance.now();
88
- const rscFullMetrics = createRenderMetrics({
89
- route: route,
90
- type: routeResult.metrics.rscFull.type,
91
- fromMainThread: routeResult.metrics.rscFull.fromMainThread,
92
- fromRscWorker: routeResult.metrics.rscFull.fromRscWorker,
93
- fromHtmlWorker: routeResult.metrics.rscFull.fromHtmlWorker,
94
- processingTime: rscFullEndTime - routeResult.metrics.rscFull.streamMetrics.startTime,
95
- chunks: routeResult.metrics.rscFull.streamMetrics.chunks,
96
- chunkRate: routeResult.metrics.rscFull.streamMetrics.chunks / ((rscFullEndTime - routeResult.metrics.rscFull.streamMetrics.startTime) / 1000),
97
- fileName: event.data.fileName,
98
- outputPath: event.data.path,
99
- baseDir: event.data.baseDir,
100
- routePath: event.data.routePath,
101
- streamMetrics: createStreamMetrics({
102
- ...routeResult.metrics.rscFull.streamMetrics,
103
- duration: rscFullEndTime - routeResult.metrics.rscFull.streamMetrics.startTime,
104
- endTime: rscFullEndTime,
105
- }),
106
- });
107
- if (options.onMetrics) {
108
- options.onMetrics(rscFullMetrics);
109
- }
110
- }
111
- }
112
- else if (event.data.fileType === "rsc") {
113
- const rscEndTime = performance.now();
114
- const rscMetrics = createRenderMetrics({
115
- route: route,
116
- type: routeResult.metrics.rscHeadless.type,
117
- fromMainThread: routeResult.metrics.rscHeadless.fromMainThread,
118
- fromRscWorker: routeResult.metrics.rscHeadless.fromRscWorker,
119
- fromHtmlWorker: routeResult.metrics.rscHeadless.fromHtmlWorker,
120
- fileSize: event.data.content.length,
121
- chunks: event.data.chunks || 0,
122
- processingTime: rscEndTime - routeResult.metrics.rscHeadless.streamMetrics.startTime,
123
- chunkRate: (event.data.chunks || 0) / ((rscEndTime - routeResult.metrics.rscHeadless.streamMetrics.startTime) / 1000),
124
- fileName: event.data.fileName,
125
- outputPath: event.data.path,
126
- baseDir: event.data.baseDir,
127
- routePath: event.data.routePath,
128
- streamMetrics: createStreamMetrics({
129
- ...routeResult.metrics.rscHeadless.streamMetrics,
130
- chunks: event.data.chunks || 0,
131
- bytes: event.data.content.length,
132
- duration: rscEndTime - routeResult.metrics.rscHeadless.streamMetrics.startTime,
133
- endTime: rscEndTime,
134
- }),
135
- });
136
- if (options.onMetrics) {
137
- options.onMetrics(rscMetrics);
138
- }
139
- }
140
- }
83
+ if (routeResult.metrics?.rscFull) {
84
+ const rscFullEndTime = performance.now();
85
+ const rscFullMetrics = createRenderMetrics({
86
+ route,
87
+ type: routeResult.metrics.rscFull.type,
88
+ fromMainThread: routeResult.metrics.rscFull.fromMainThread,
89
+ fromRscWorker: routeResult.metrics.rscFull.fromRscWorker,
90
+ fromHtmlWorker: routeResult.metrics.rscFull.fromHtmlWorker,
91
+ processingTime: rscFullEndTime - routeResult.metrics.rscFull.streamMetrics.startTime,
92
+ chunks: routeResult.metrics.rscFull.streamMetrics.chunks,
93
+ chunkRate: routeResult.metrics.rscFull.streamMetrics.chunks / ((rscFullEndTime - routeResult.metrics.rscFull.streamMetrics.startTime) / 1e3),
94
+ fileName: event.data.fileName,
95
+ outputPath: event.data.path,
96
+ baseDir: event.data.baseDir,
97
+ routePath: event.data.routePath,
98
+ streamMetrics: createStreamMetrics({
99
+ ...routeResult.metrics.rscFull.streamMetrics,
100
+ duration: rscFullEndTime - routeResult.metrics.rscFull.streamMetrics.startTime,
101
+ endTime: rscFullEndTime
102
+ })
103
+ });
104
+ if (options.onMetrics) {
105
+ options.onMetrics(rscFullMetrics);
106
+ }
141
107
  }
142
- };
143
- const routeHandlerOptions = {
144
- ...options,
145
- manifest,
146
- route,
147
- pagePath: resolvedPagePath,
148
- propsPath: resolvedPropsPath,
149
- rootPath: resolvedRootPath,
150
- htmlPath: resolvedHtmlPath,
151
- cssFiles: cssFilesByPage?.get(route) ?? new Map(),
152
- globalCss: options.globalCss ?? new Map(),
153
- id: `${route}-${Date.now()}-${Math.random().toString(36).substring(2, 11)}`,
154
- onEvent: wrapperOnEvent,
155
- };
156
- const pageRenderer = renderPage(routeHandlerOptions);
157
- const results = [];
158
- let routeError;
159
- // Consume all yields from the page renderer and write files
160
- for await (const result of pageRenderer) {
161
- results.push(result);
162
- if (result.type === "error" && result.error) {
163
- routeError = result.error instanceof Error ? result.error : new Error(String(result.error));
108
+ } else if (event.data.fileType === "rsc") {
109
+ const rscEndTime = performance.now();
110
+ const rscMetrics = createRenderMetrics({
111
+ route,
112
+ type: routeResult.metrics.rscHeadless.type,
113
+ fromMainThread: routeResult.metrics.rscHeadless.fromMainThread,
114
+ fromRscWorker: routeResult.metrics.rscHeadless.fromRscWorker,
115
+ fromHtmlWorker: routeResult.metrics.rscHeadless.fromHtmlWorker,
116
+ fileSize: event.data.content.length,
117
+ chunks: event.data.chunks || 0,
118
+ processingTime: rscEndTime - routeResult.metrics.rscHeadless.streamMetrics.startTime,
119
+ chunkRate: (event.data.chunks || 0) / ((rscEndTime - routeResult.metrics.rscHeadless.streamMetrics.startTime) / 1e3),
120
+ fileName: event.data.fileName,
121
+ outputPath: event.data.path,
122
+ baseDir: event.data.baseDir,
123
+ routePath: event.data.routePath,
124
+ streamMetrics: createStreamMetrics({
125
+ ...routeResult.metrics.rscHeadless.streamMetrics,
126
+ chunks: event.data.chunks || 0,
127
+ bytes: event.data.content.length,
128
+ duration: rscEndTime - routeResult.metrics.rscHeadless.streamMetrics.startTime,
129
+ endTime: rscEndTime
130
+ })
131
+ });
132
+ if (options.onMetrics) {
133
+ options.onMetrics(rscMetrics);
164
134
  }
165
- if (result.type === "success" || result.type === "skip") {
166
- // Store result for metrics tracking (wrapperOnEvent needs this)
167
- routeResults.set(route, result);
168
- const rscWritePromise = fileWriter(result.rsc, "rsc", { ...options, route, onEvent: wrapperOnEvent, logger: options.logger }, options.signal);
169
- const htmlWritePromise = fileWriter(result.html, "html", { ...options, route, onEvent: wrapperOnEvent, logger: options.logger }, options.signal);
170
- await Promise.all([rscWritePromise, htmlWritePromise]);
171
- }
172
- }
173
- if (routeError) {
174
- return { route, results, error: routeError };
135
+ }
175
136
  }
176
- return { route, results };
137
+ }
138
+ };
139
+ const routeHandlerOptions = {
140
+ ...options,
141
+ manifest,
142
+ route,
143
+ pagePath: resolvedPagePath,
144
+ propsPath: resolvedPropsPath,
145
+ rootPath: resolvedRootPath,
146
+ htmlPath: resolvedHtmlPath,
147
+ cssFiles: cssFilesByPage?.get(route) ?? /* @__PURE__ */ new Map(),
148
+ globalCss: options.globalCss ?? /* @__PURE__ */ new Map(),
149
+ id: `${route}-${Date.now()}-${Math.random().toString(36).substring(2, 11)}`,
150
+ onEvent: wrapperOnEvent
151
+ };
152
+ const pageRenderer = renderPage(routeHandlerOptions);
153
+ const results = [];
154
+ let routeError;
155
+ for await (const result of pageRenderer) {
156
+ results.push(result);
157
+ if (result.type === "error" && result.error) {
158
+ routeError = result.error instanceof Error ? result.error : new Error(String(result.error));
159
+ }
160
+ if (result.type === "success" || result.type === "skip") {
161
+ routeResults.set(route, result);
162
+ const rscWritePromise = fileWriter(
163
+ result.rsc,
164
+ "rsc",
165
+ { ...options, route, onEvent: wrapperOnEvent, logger: options.logger },
166
+ options.signal
167
+ );
168
+ const htmlWritePromise = fileWriter(
169
+ result.html,
170
+ "html",
171
+ { ...options, route, onEvent: wrapperOnEvent, logger: options.logger },
172
+ options.signal
173
+ );
174
+ await Promise.all([rscWritePromise, htmlWritePromise]);
175
+ }
177
176
  }
178
- catch (error) {
179
- return { route, results: [], error: error };
177
+ if (routeError) {
178
+ return { route, results, error: routeError };
180
179
  }
180
+ return { route, results };
181
+ } catch (error) {
182
+ return { route, results: [], error };
183
+ }
181
184
  }
182
- /**
183
- * Splits array into chunks of specified size
184
- */
185
185
  function chunk(array, size) {
186
- const chunks = [];
187
- for (let i = 0; i < array.length; i += size) {
188
- chunks.push(array.slice(i, i + size));
189
- }
190
- return chunks;
186
+ const chunks = [];
187
+ for (let i = 0; i < array.length; i += size) {
188
+ chunks.push(array.slice(i, i + size));
189
+ }
190
+ return chunks;
191
191
  }
192
- /**
193
- * Batched version of renderPages that renders pages in parallel batches
194
- */
195
- export const renderPagesBatched = (routes, handlerOptions, renderPage) => {
196
- const { autoDiscoveredFiles, manifest = {}, ...options } = handlerOptions;
197
- const batchSize = options.batchSize ?? DEFAULT_BATCH_SIZE;
198
- const completedRoutes = new Set();
199
- const failedRoutes = new Map();
200
- const results = new Map();
201
- if (!autoDiscoveredFiles?.urlMap) {
202
- return (async function* _renderPagesBatched() {
192
+ const renderPagesBatched = (routes, handlerOptions, renderPage) => {
193
+ const {
194
+ autoDiscoveredFiles,
195
+ manifest = {},
196
+ ...options
197
+ } = handlerOptions;
198
+ const batchSize = options.batchSize ?? DEFAULT_BATCH_SIZE;
199
+ const completedRoutes = /* @__PURE__ */ new Set();
200
+ const failedRoutes = /* @__PURE__ */ new Map();
201
+ const results = /* @__PURE__ */ new Map();
202
+ if (!autoDiscoveredFiles?.urlMap) {
203
+ return async function* _renderPagesBatched() {
204
+ const errorResult = {
205
+ type: "error",
206
+ error: new Error("No urlMap provided to renderPages"),
207
+ route: "",
208
+ failedRoutes: /* @__PURE__ */ new Map(),
209
+ completedRoutes: /* @__PURE__ */ new Set(),
210
+ results: /* @__PURE__ */ new Map()
211
+ };
212
+ yield errorResult;
213
+ return errorResult;
214
+ }();
215
+ }
216
+ return async function* _renderPagesBatched() {
217
+ const routeArray = Array.from(routes);
218
+ const batches = chunk(routeArray, batchSize);
219
+ if (options.verbose) {
220
+ options.logger?.info(
221
+ `[renderPagesBatched] Rendering ${routeArray.length} pages in ${batches.length} batches of ${batchSize}`
222
+ );
223
+ }
224
+ for (const batch of batches) {
225
+ if (options.signal?.aborted) {
226
+ const abortResult = {
227
+ type: "error",
228
+ error: options.signal.reason || new Error("Build aborted"),
229
+ route: batch[0] || "",
230
+ failedRoutes,
231
+ completedRoutes,
232
+ results
233
+ };
234
+ yield abortResult;
235
+ return abortResult;
236
+ }
237
+ const batchPromises = batch.map(
238
+ (route) => renderSingleRoute(route, handlerOptions, renderPage, manifest, failedRoutes)
239
+ );
240
+ const batchResults = await Promise.all(batchPromises);
241
+ for (const { route, results: pageResults, error } of batchResults) {
242
+ if (error) {
243
+ const panicError = handleError({
244
+ error,
245
+ logger: options.logger,
246
+ panicThreshold: options.panicThreshold});
247
+ if (panicError != null) {
248
+ failedRoutes.set(route, error);
249
+ options.logger?.error(`[renderPagesBatched] Panic error for route ${route}: ${error.message}`);
203
250
  const errorResult = {
204
- type: "error",
205
- error: new Error("No urlMap provided to renderPages"),
206
- route: "",
207
- failedRoutes: new Map(),
208
- completedRoutes: new Set(),
209
- results: new Map(),
251
+ type: "error",
252
+ error,
253
+ route,
254
+ failedRoutes,
255
+ completedRoutes,
256
+ results
210
257
  };
211
258
  yield errorResult;
212
259
  return errorResult;
213
- })();
214
- }
215
- return (async function* _renderPagesBatched() {
216
- const routeArray = Array.from(routes);
217
- const batches = chunk(routeArray, batchSize);
218
- if (options.verbose) {
219
- options.logger?.info(`[renderPagesBatched] Rendering ${routeArray.length} pages in ${batches.length} batches of ${batchSize}`);
220
- }
221
- for (const batch of batches) {
222
- // Check for abort signal
223
- if (options.signal?.aborted) {
224
- const abortResult = {
225
- type: "error",
226
- error: options.signal.reason || new Error("Build aborted"),
227
- route: batch[0] || "",
228
- failedRoutes,
229
- completedRoutes,
230
- results,
231
- };
232
- yield abortResult;
233
- return abortResult;
234
- }
235
- // Render all pages in this batch concurrently
236
- const batchPromises = batch.map(route => renderSingleRoute(route, handlerOptions, renderPage, manifest, failedRoutes));
237
- const batchResults = await Promise.all(batchPromises);
238
- // Process results from this batch
239
- for (const { route, results: pageResults, error } of batchResults) {
240
- if (error) {
241
- const panicError = handleError({
242
- error,
243
- logger: options.logger,
244
- panicThreshold: options.panicThreshold,
245
- context: `renderPagesBatched(${route})`,
246
- });
247
- if (panicError != null) {
248
- failedRoutes.set(route, error);
249
- options.logger?.error(`[renderPagesBatched] Panic error for route ${route}: ${error.message}`);
250
- const errorResult = {
251
- type: "error",
252
- error,
253
- route,
254
- failedRoutes,
255
- completedRoutes,
256
- results,
257
- };
258
- yield errorResult;
259
- return errorResult;
260
- }
261
- else {
262
- options.logger?.warn(`[renderPagesBatched] Non-panic error for route ${route}: ${error.message}`);
263
- }
264
- }
265
- else {
266
- completedRoutes.add(route);
267
- for (const result of pageResults) {
268
- if (result.type === "success" || result.type === "skip") {
269
- results.set(route, result);
270
- yield {
271
- type: "success",
272
- route,
273
- failedRoutes,
274
- completedRoutes,
275
- results,
276
- };
277
- }
278
- }
279
- }
280
- }
281
- if (options.verbose) {
282
- options.logger?.info(`[renderPagesBatched] Completed batch: ${completedRoutes.size}/${routeArray.length} pages`);
260
+ } else {
261
+ options.logger?.warn(`[renderPagesBatched] Non-panic error for route ${route}: ${error.message}`);
262
+ }
263
+ } else {
264
+ completedRoutes.add(route);
265
+ for (const result of pageResults) {
266
+ if (result.type === "success" || result.type === "skip") {
267
+ results.set(route, result);
268
+ yield {
269
+ type: "success",
270
+ route,
271
+ failedRoutes,
272
+ completedRoutes,
273
+ results
274
+ };
283
275
  }
276
+ }
284
277
  }
285
- // Final success result
286
- const finalResult = {
287
- type: "success",
288
- route: "",
289
- failedRoutes,
290
- completedRoutes,
291
- results,
292
- };
293
- return finalResult;
294
- })();
278
+ }
279
+ if (options.verbose) {
280
+ options.logger?.info(
281
+ `[renderPagesBatched] Completed batch: ${completedRoutes.size}/${routeArray.length} pages`
282
+ );
283
+ }
284
+ }
285
+ const finalResult = {
286
+ type: "success",
287
+ route: "",
288
+ failedRoutes,
289
+ completedRoutes,
290
+ results
291
+ };
292
+ return finalResult;
293
+ }();
295
294
  };
295
+
296
+ export { renderPagesBatched };
297
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"renderPagesBatched.js","sources":["../../../plugin/react-static/renderPagesBatched.ts"],"sourcesContent":["/**\n * renderPagesBatched.ts\n *\n * Batched version of renderPages that renders multiple pages concurrently.\n * Uses Promise.all on batches to parallelize rendering while preserving\n * the async generator interface for compatibility.\n */\nimport type { RenderPagesResult, RenderPageResult } from \"../types.js\";\nimport type { RenderPagesFn, RenderPageFn, RenderPagesHandlerOptions } from \"./types.js\";\nimport { handleError } from \"../error/handleError.js\";\nimport { fileWriter } from \"./fileWriter.js\";\nimport type { Manifest } from \"vite\";\nimport { createRenderMetrics } from \"../metrics/createRenderMetrics.js\";\nimport { createStreamMetrics } from \"../metrics/createStreamMetrics.js\";\n\nconst DEFAULT_BATCH_SIZE = 8;\n\nfunction resolvePathWithManifest(path: string, manifest: Manifest): string {\n  const entry = manifest[path];\n  if (entry && entry.file) {\n    return entry.file;\n  }\n  return path;\n}\n\n/**\n * Renders a single route completely, consuming all yields from renderPage\n * and writing the RSC and HTML files. Collects metrics and handles events\n * identically to the sequential renderPages.\n */\nasync function renderSingleRoute(\n  route: string,\n  handlerOptions: RenderPagesHandlerOptions,\n  renderPage: RenderPageFn,\n  manifest: Manifest,\n  failedRoutes: Map<string, unknown>,\n): Promise<{ route: string; results: RenderPageResult[]; error?: Error }> {\n  const { autoDiscoveredFiles, cssFilesByPage, ...options } = handlerOptions;\n  const { page, props, root, html } = autoDiscoveredFiles.urlMap?.get(route) || {};\n  \n  if (!page) {\n    return { route, results: [], error: new Error(`No page found for route ${route}`) };\n  }\n\n  try {\n    const resolvedPagePath = page ? resolvePathWithManifest(page, manifest) : undefined;\n    const resolvedPropsPath = props ? resolvePathWithManifest(props, manifest) : undefined;\n    const resolvedRootPath = root ? resolvePathWithManifest(root, manifest) : undefined;\n    const resolvedHtmlPath = html ? resolvePathWithManifest(html, manifest) : undefined;\n\n    // Store results for metrics tracking\n    const routeResults = new Map<string, RenderPageResult>();\n\n    // Create onEvent wrapper that handles route.error and metrics collection\n    // This mirrors the sequential renderPages behavior exactly\n    const wrapperOnEvent = (event: any) => {\n      // Call the original onEvent first\n      if (options.onEvent) {\n        options.onEvent(event);\n      }\n\n      // Handle route.error events\n      if (event.type === \"route.error\") {\n        const detectedPanicError = handleError({\n          error: event.data.error,\n          logger: options.logger,\n          panicThreshold: event.data.panicThreshold,\n          context: `route.error (${event.data.route})`,\n        });\n        \n        if (detectedPanicError != null) {\n          options.logger?.error(\n            `[renderPagesBatched] Panic error for route ${event.data.route}: ${event.data.error.message}`\n          );\n          failedRoutes.set(event.data.route, event.data.error);\n        } else {\n          options.logger?.warn(\n            `[renderPagesBatched] Non-panic error for route ${event.data.route}: ${event.data.error.message}`\n          );\n        }\n      }\n\n      // Handle metrics collection on file.write.done\n      if (event.type === \"file.write.done\" && event.data.route === route) {\n        const routeResult = routeResults.get(route);\n        if (routeResult && routeResult.type === \"success\") {\n          if (event.data.fileType === \"html\") {\n            const endTime = performance.now();\n            const htmlMetrics = createRenderMetrics({\n              route: route,\n              type: routeResult.metrics.html.type,\n              fromMainThread: routeResult.metrics.html.fromMainThread,\n              fromRscWorker: routeResult.metrics.html.fromRscWorker,\n              fromHtmlWorker: routeResult.metrics.html.fromHtmlWorker,\n              fileSize: event.data.content.length,\n              chunks: event.data.chunks || 0,\n              processingTime: endTime - routeResult.metrics.html.streamMetrics.startTime,\n              chunkRate: (event.data.chunks || 0) / ((endTime - routeResult.metrics.html.streamMetrics.startTime) / 1000),\n              fileName: event.data.fileName,\n              outputPath: event.data.path,\n              baseDir: event.data.baseDir,\n              routePath: event.data.routePath,\n              streamMetrics: createStreamMetrics({\n                ...routeResult.metrics.html.streamMetrics,\n                chunks: event.data.chunks || 0,\n                bytes: event.data.content.length,\n                duration: endTime - routeResult.metrics.html.streamMetrics.startTime,\n                endTime: endTime,\n              }),\n            });\n\n            if (options.onMetrics) {\n              options.onMetrics(htmlMetrics);\n            }\n\n            // Also emit RSC Full metrics if available\n            if (routeResult.metrics?.rscFull) {\n              const rscFullEndTime = performance.now();\n              const rscFullMetrics = createRenderMetrics({\n                route: route,\n                type: routeResult.metrics.rscFull.type,\n                fromMainThread: routeResult.metrics.rscFull.fromMainThread,\n                fromRscWorker: routeResult.metrics.rscFull.fromRscWorker,\n                fromHtmlWorker: routeResult.metrics.rscFull.fromHtmlWorker,\n                processingTime: rscFullEndTime - routeResult.metrics.rscFull.streamMetrics.startTime,\n                chunks: routeResult.metrics.rscFull.streamMetrics.chunks,\n                chunkRate: routeResult.metrics.rscFull.streamMetrics.chunks / ((rscFullEndTime - routeResult.metrics.rscFull.streamMetrics.startTime) / 1000),\n                fileName: event.data.fileName,\n                outputPath: event.data.path,\n                baseDir: event.data.baseDir,\n                routePath: event.data.routePath,\n                streamMetrics: createStreamMetrics({\n                  ...routeResult.metrics.rscFull.streamMetrics,\n                  duration: rscFullEndTime - routeResult.metrics.rscFull.streamMetrics.startTime,\n                  endTime: rscFullEndTime,\n                }),\n              });\n\n              if (options.onMetrics) {\n                options.onMetrics(rscFullMetrics);\n              }\n            }\n          } else if (event.data.fileType === \"rsc\") {\n            const rscEndTime = performance.now();\n            const rscMetrics = createRenderMetrics({\n              route: route,\n              type: routeResult.metrics.rscHeadless.type,\n              fromMainThread: routeResult.metrics.rscHeadless.fromMainThread,\n              fromRscWorker: routeResult.metrics.rscHeadless.fromRscWorker,\n              fromHtmlWorker: routeResult.metrics.rscHeadless.fromHtmlWorker,\n              fileSize: event.data.content.length,\n              chunks: event.data.chunks || 0,\n              processingTime: rscEndTime - routeResult.metrics.rscHeadless.streamMetrics.startTime,\n              chunkRate: (event.data.chunks || 0) / ((rscEndTime - routeResult.metrics.rscHeadless.streamMetrics.startTime) / 1000),\n              fileName: event.data.fileName,\n              outputPath: event.data.path,\n              baseDir: event.data.baseDir,\n              routePath: event.data.routePath,\n              streamMetrics: createStreamMetrics({\n                ...routeResult.metrics.rscHeadless.streamMetrics,\n                chunks: event.data.chunks || 0,\n                bytes: event.data.content.length,\n                duration: rscEndTime - routeResult.metrics.rscHeadless.streamMetrics.startTime,\n                endTime: rscEndTime,\n              }),\n            });\n\n            if (options.onMetrics) {\n              options.onMetrics(rscMetrics);\n            }\n          }\n        }\n      }\n    };\n\n    const routeHandlerOptions = {\n      ...options,\n      manifest,\n      route,\n      pagePath: resolvedPagePath as string,\n      propsPath: resolvedPropsPath as string,\n      rootPath: resolvedRootPath as string,\n      htmlPath: resolvedHtmlPath as string,\n      cssFiles: cssFilesByPage?.get(route) ?? new Map(),\n      globalCss: options.globalCss ?? new Map(),\n      id: `${route}-${Date.now()}-${Math.random().toString(36).substring(2, 11)}`,\n      onEvent: wrapperOnEvent,\n    };\n\n    const pageRenderer = renderPage(routeHandlerOptions);\n    const results: RenderPageResult[] = [];\n    let routeError: Error | undefined;\n\n    // Consume all yields from the page renderer and write files\n    for await (const result of pageRenderer) {\n      results.push(result);\n      \n      if (result.type === \"error\" && result.error) {\n        routeError = result.error instanceof Error ? result.error : new Error(String(result.error));\n      }\n      \n      if (result.type === \"success\" || result.type === \"skip\") {\n        // Store result for metrics tracking (wrapperOnEvent needs this)\n        routeResults.set(route, result);\n\n        const rscWritePromise = fileWriter(\n          result.rsc as any,\n          \"rsc\",\n          { ...options, route, onEvent: wrapperOnEvent, logger: options.logger },\n          options.signal\n        );\n\n        const htmlWritePromise = fileWriter(\n          result.html as any,\n          \"html\",\n          { ...options, route, onEvent: wrapperOnEvent, logger: options.logger },\n          options.signal\n        );\n\n        await Promise.all([rscWritePromise, htmlWritePromise]);\n      }\n    }\n\n    if (routeError) {\n      return { route, results, error: routeError };\n    }\n\n    return { route, results };\n  } catch (error) {\n    return { route, results: [], error: error as Error };\n  }\n}\n\n/**\n * Splits array into chunks of specified size\n */\nfunction chunk<T>(array: T[], size: number): T[][] {\n  const chunks: T[][] = [];\n  for (let i = 0; i < array.length; i += size) {\n    chunks.push(array.slice(i, i + size));\n  }\n  return chunks;\n}\n\n/**\n * Batched version of renderPages that renders pages in parallel batches\n */\nexport const renderPagesBatched: RenderPagesFn = (\n  routes,\n  handlerOptions,\n  renderPage\n) => {\n  const {\n    autoDiscoveredFiles,\n    manifest = {},\n    ...options\n  } = handlerOptions;\n\n  const batchSize = (options as any).batchSize ?? DEFAULT_BATCH_SIZE;\n  const completedRoutes = new Set<string>();\n  const failedRoutes = new Map<string, unknown>();\n  const results = new Map<string, RenderPageResult>();\n\n  if (!autoDiscoveredFiles?.urlMap) {\n    return (async function* _renderPagesBatched(): AsyncGenerator<RenderPagesResult, RenderPagesResult, unknown> {\n      const errorResult: RenderPagesResult = {\n        type: \"error\",\n        error: new Error(\"No urlMap provided to renderPages\"),\n        route: \"\",\n        failedRoutes: new Map(),\n        completedRoutes: new Set(),\n        results: new Map(),\n      };\n      yield errorResult;\n      return errorResult;\n    })();\n  }\n\n  return (async function* _renderPagesBatched(): AsyncGenerator<RenderPagesResult, RenderPagesResult, unknown> {\n    const routeArray = Array.from(routes);\n    const batches = chunk(routeArray, batchSize);\n    \n    if (options.verbose) {\n      options.logger?.info(\n        `[renderPagesBatched] Rendering ${routeArray.length} pages in ${batches.length} batches of ${batchSize}`\n      );\n    }\n\n    for (const batch of batches) {\n      // Check for abort signal\n      if (options.signal?.aborted) {\n        const abortResult: RenderPagesResult = {\n          type: \"error\",\n          error: options.signal.reason || new Error(\"Build aborted\"),\n          route: batch[0] || \"\",\n          failedRoutes,\n          completedRoutes,\n          results,\n        };\n        yield abortResult;\n        return abortResult;\n      }\n\n      // Render all pages in this batch concurrently\n      const batchPromises = batch.map(route => \n        renderSingleRoute(route, handlerOptions, renderPage, manifest, failedRoutes)\n      );\n\n      const batchResults = await Promise.all(batchPromises);\n\n      // Process results from this batch\n      for (const { route, results: pageResults, error } of batchResults) {\n        if (error) {\n          const panicError = handleError({\n            error,\n            logger: options.logger,\n            panicThreshold: options.panicThreshold,\n            context: `renderPagesBatched(${route})`,\n          });\n\n          if (panicError != null) {\n            failedRoutes.set(route, error);\n            options.logger?.error(`[renderPagesBatched] Panic error for route ${route}: ${error.message}`);\n            const errorResult: RenderPagesResult = {\n              type: \"error\",\n              error,\n              route,\n              failedRoutes,\n              completedRoutes,\n              results,\n            };\n            yield errorResult;\n            return errorResult;\n          } else {\n            options.logger?.warn(`[renderPagesBatched] Non-panic error for route ${route}: ${error.message}`);\n          }\n        } else {\n          completedRoutes.add(route);\n          \n          for (const result of pageResults) {\n            if (result.type === \"success\" || result.type === \"skip\") {\n              results.set(route, result);\n              yield {\n                type: \"success\",\n                route,\n                failedRoutes,\n                completedRoutes,\n                results,\n              } satisfies RenderPagesResult;\n            }\n          }\n        }\n      }\n\n      if (options.verbose) {\n        options.logger?.info(\n          `[renderPagesBatched] Completed batch: ${completedRoutes.size}/${routeArray.length} pages`\n        );\n      }\n    }\n\n    // Final success result\n    const finalResult: RenderPagesResult = {\n      type: \"success\",\n      route: \"\",\n      failedRoutes,\n      completedRoutes,\n      results,\n    };\n\n    return finalResult;\n  })();\n};\n"],"names":[],"mappings":";;;;;;;;;;AAeA,MAAM,kBAAqB,GAAA,CAAA;AAE3B,SAAS,uBAAA,CAAwB,MAAc,QAA4B,EAAA;AACzE,EAAM,MAAA,KAAA,GAAQ,SAAS,IAAI,CAAA;AAC3B,EAAI,IAAA,KAAA,IAAS,MAAM,IAAM,EAAA;AACvB,IAAA,OAAO,KAAM,CAAA,IAAA;AAAA;AAEf,EAAO,OAAA,IAAA;AACT;AAOA,eAAe,iBACb,CAAA,KAAA,EACA,cACA,EAAA,UAAA,EACA,UACA,YACwE,EAAA;AACxE,EAAA,MAAM,EAAE,mBAAA,EAAqB,cAAgB,EAAA,GAAG,SAAY,GAAA,cAAA;AAC5D,EAAM,MAAA,EAAE,IAAM,EAAA,KAAA,EAAO,IAAM,EAAA,IAAA,EAAS,GAAA,mBAAA,CAAoB,MAAQ,EAAA,GAAA,CAAI,KAAK,CAAA,IAAK,EAAC;AAE/E,EAAA,IAAI,CAAC,IAAM,EAAA;AACT,IAAO,OAAA,EAAE,KAAO,EAAA,OAAA,EAAS,EAAC,EAAG,KAAO,EAAA,IAAI,KAAM,CAAA,CAAA,wBAAA,EAA2B,KAAK,CAAA,CAAE,CAAE,EAAA;AAAA;AAGpF,EAAI,IAAA;AACF,IAAA,MAAM,gBAAmB,GAAA,IAAA,GAAO,uBAAwB,CAAA,IAAA,EAAM,QAAQ,CAAI,GAAA,KAAA,CAAA;AAC1E,IAAA,MAAM,iBAAoB,GAAA,KAAA,GAAQ,uBAAwB,CAAA,KAAA,EAAO,QAAQ,CAAI,GAAA,KAAA,CAAA;AAC7E,IAAA,MAAM,gBAAmB,GAAA,IAAA,GAAO,uBAAwB,CAAA,IAAA,EAAM,QAAQ,CAAI,GAAA,KAAA,CAAA;AAC1E,IAAA,MAAM,gBAAmB,GAAA,IAAA,GAAO,uBAAwB,CAAA,IAAA,EAAM,QAAQ,CAAI,GAAA,KAAA,CAAA;AAG1E,IAAM,MAAA,YAAA,uBAAmB,GAA8B,EAAA;AAIvD,IAAM,MAAA,cAAA,GAAiB,CAAC,KAAe,KAAA;AAErC,MAAA,IAAI,QAAQ,OAAS,EAAA;AACnB,QAAA,OAAA,CAAQ,QAAQ,KAAK,CAAA;AAAA;AAIvB,MAAI,IAAA,KAAA,CAAM,SAAS,aAAe,EAAA;AAChC,QAAA,MAAM,qBAAqB,WAAY,CAAA;AAAA,UACrC,KAAA,EAAO,MAAM,IAAK,CAAA,KAAA;AAAA,UAClB,QAAQ,OAAQ,CAAA,MAAA;AAAA,UAChB,cAAA,EAAgB,MAAM,IAAK,CAAA,cAAA;AAAA,UAC3B,OAAS,EAAA,CAAA,aAAA,EAAgB,KAAM,CAAA,IAAA,CAAK,KAAK,CAAA,CAAA;AAAA,SAC1C,CAAA;AAED,QAAA,IAAI,sBAAsB,IAAM,EAAA;AAC9B,UAAA,OAAA,CAAQ,MAAQ,EAAA,KAAA;AAAA,YACd,CAAA,2CAAA,EAA8C,MAAM,IAAK,CAAA,KAAK,KAAK,KAAM,CAAA,IAAA,CAAK,MAAM,OAAO,CAAA;AAAA,WAC7F;AACA,UAAA,YAAA,CAAa,IAAI,KAAM,CAAA,IAAA,CAAK,KAAO,EAAA,KAAA,CAAM,KAAK,KAAK,CAAA;AAAA,SAC9C,MAAA;AACL,UAAA,OAAA,CAAQ,MAAQ,EAAA,IAAA;AAAA,YACd,CAAA,+CAAA,EAAkD,MAAM,IAAK,CAAA,KAAK,KAAK,KAAM,CAAA,IAAA,CAAK,MAAM,OAAO,CAAA;AAAA,WACjG;AAAA;AACF;AAIF,MAAA,IAAI,MAAM,IAAS,KAAA,iBAAA,IAAqB,KAAM,CAAA,IAAA,CAAK,UAAU,KAAO,EAAA;AAClE,QAAM,MAAA,WAAA,GAAc,YAAa,CAAA,GAAA,CAAI,KAAK,CAAA;AAC1C,QAAI,IAAA,WAAA,IAAe,WAAY,CAAA,IAAA,KAAS,SAAW,EAAA;AACjD,UAAI,IAAA,KAAA,CAAM,IAAK,CAAA,QAAA,KAAa,MAAQ,EAAA;AAClC,YAAM,MAAA,OAAA,GAAU,YAAY,GAAI,EAAA;AAChC,YAAA,MAAM,cAAc,mBAAoB,CAAA;AAAA,cACtC,KAAA;AAAA,cACA,IAAA,EAAM,WAAY,CAAA,OAAA,CAAQ,IAAK,CAAA,IAAA;AAAA,cAC/B,cAAA,EAAgB,WAAY,CAAA,OAAA,CAAQ,IAAK,CAAA,cAAA;AAAA,cACzC,aAAA,EAAe,WAAY,CAAA,OAAA,CAAQ,IAAK,CAAA,aAAA;AAAA,cACxC,cAAA,EAAgB,WAAY,CAAA,OAAA,CAAQ,IAAK,CAAA,cAAA;AAAA,cACzC,QAAA,EAAU,KAAM,CAAA,IAAA,CAAK,OAAQ,CAAA,MAAA;AAAA,cAC7B,MAAA,EAAQ,KAAM,CAAA,IAAA,CAAK,MAAU,IAAA,CAAA;AAAA,cAC7B,cAAgB,EAAA,OAAA,GAAU,WAAY,CAAA,OAAA,CAAQ,KAAK,aAAc,CAAA,SAAA;AAAA,cACjE,SAAA,EAAA,CAAY,KAAM,CAAA,IAAA,CAAK,MAAU,IAAA,CAAA,KAAA,CAAO,UAAU,WAAY,CAAA,OAAA,CAAQ,IAAK,CAAA,aAAA,CAAc,SAAa,IAAA,GAAA,CAAA;AAAA,cACtG,QAAA,EAAU,MAAM,IAAK,CAAA,QAAA;AAAA,cACrB,UAAA,EAAY,MAAM,IAAK,CAAA,IAAA;AAAA,cACvB,OAAA,EAAS,MAAM,IAAK,CAAA,OAAA;AAAA,cACpB,SAAA,EAAW,MAAM,IAAK,CAAA,SAAA;AAAA,cACtB,eAAe,mBAAoB,CAAA;AAAA,gBACjC,GAAG,WAAY,CAAA,OAAA,CAAQ,IAAK,CAAA,aAAA;AAAA,gBAC5B,MAAA,EAAQ,KAAM,CAAA,IAAA,CAAK,MAAU,IAAA,CAAA;AAAA,gBAC7B,KAAA,EAAO,KAAM,CAAA,IAAA,CAAK,OAAQ,CAAA,MAAA;AAAA,gBAC1B,QAAU,EAAA,OAAA,GAAU,WAAY,CAAA,OAAA,CAAQ,KAAK,aAAc,CAAA,SAAA;AAAA,gBAC3D;AAAA,eACD;AAAA,aACF,CAAA;AAED,YAAA,IAAI,QAAQ,SAAW,EAAA;AACrB,cAAA,OAAA,CAAQ,UAAU,WAAW,CAAA;AAAA;AAI/B,YAAI,IAAA,WAAA,CAAY,SAAS,OAAS,EAAA;AAChC,cAAM,MAAA,cAAA,GAAiB,YAAY,GAAI,EAAA;AACvC,cAAA,MAAM,iBAAiB,mBAAoB,CAAA;AAAA,gBACzC,KAAA;AAAA,gBACA,IAAA,EAAM,WAAY,CAAA,OAAA,CAAQ,OAAQ,CAAA,IAAA;AAAA,gBAClC,cAAA,EAAgB,WAAY,CAAA,OAAA,CAAQ,OAAQ,CAAA,cAAA;AAAA,gBAC5C,aAAA,EAAe,WAAY,CAAA,OAAA,CAAQ,OAAQ,CAAA,aAAA;AAAA,gBAC3C,cAAA,EAAgB,WAAY,CAAA,OAAA,CAAQ,OAAQ,CAAA,cAAA;AAAA,gBAC5C,cAAgB,EAAA,cAAA,GAAiB,WAAY,CAAA,OAAA,CAAQ,QAAQ,aAAc,CAAA,SAAA;AAAA,gBAC3E,MAAQ,EAAA,WAAA,CAAY,OAAQ,CAAA,OAAA,CAAQ,aAAc,CAAA,MAAA;AAAA,gBAClD,SAAA,EAAW,WAAY,CAAA,OAAA,CAAQ,OAAQ,CAAA,aAAA,CAAc,MAAW,IAAA,CAAA,cAAA,GAAiB,WAAY,CAAA,OAAA,CAAQ,OAAQ,CAAA,aAAA,CAAc,SAAa,IAAA,GAAA,CAAA;AAAA,gBACxI,QAAA,EAAU,MAAM,IAAK,CAAA,QAAA;AAAA,gBACrB,UAAA,EAAY,MAAM,IAAK,CAAA,IAAA;AAAA,gBACvB,OAAA,EAAS,MAAM,IAAK,CAAA,OAAA;AAAA,gBACpB,SAAA,EAAW,MAAM,IAAK,CAAA,SAAA;AAAA,gBACtB,eAAe,mBAAoB,CAAA;AAAA,kBACjC,GAAG,WAAY,CAAA,OAAA,CAAQ,OAAQ,CAAA,aAAA;AAAA,kBAC/B,QAAU,EAAA,cAAA,GAAiB,WAAY,CAAA,OAAA,CAAQ,QAAQ,aAAc,CAAA,SAAA;AAAA,kBACrE,OAAS,EAAA;AAAA,iBACV;AAAA,eACF,CAAA;AAED,cAAA,IAAI,QAAQ,SAAW,EAAA;AACrB,gBAAA,OAAA,CAAQ,UAAU,cAAc,CAAA;AAAA;AAClC;AACF,WACS,MAAA,IAAA,KAAA,CAAM,IAAK,CAAA,QAAA,KAAa,KAAO,EAAA;AACxC,YAAM,MAAA,UAAA,GAAa,YAAY,GAAI,EAAA;AACnC,YAAA,MAAM,aAAa,mBAAoB,CAAA;AAAA,cACrC,KAAA;AAAA,cACA,IAAA,EAAM,WAAY,CAAA,OAAA,CAAQ,WAAY,CAAA,IAAA;AAAA,cACtC,cAAA,EAAgB,WAAY,CAAA,OAAA,CAAQ,WAAY,CAAA,cAAA;AAAA,cAChD,aAAA,EAAe,WAAY,CAAA,OAAA,CAAQ,WAAY,CAAA,aAAA;AAAA,cAC/C,cAAA,EAAgB,WAAY,CAAA,OAAA,CAAQ,WAAY,CAAA,cAAA;AAAA,cAChD,QAAA,EAAU,KAAM,CAAA,IAAA,CAAK,OAAQ,CAAA,MAAA;AAAA,cAC7B,MAAA,EAAQ,KAAM,CAAA,IAAA,CAAK,MAAU,IAAA,CAAA;AAAA,cAC7B,cAAgB,EAAA,UAAA,GAAa,WAAY,CAAA,OAAA,CAAQ,YAAY,aAAc,CAAA,SAAA;AAAA,cAC3E,SAAA,EAAA,CAAY,KAAM,CAAA,IAAA,CAAK,MAAU,IAAA,CAAA,KAAA,CAAO,aAAa,WAAY,CAAA,OAAA,CAAQ,WAAY,CAAA,aAAA,CAAc,SAAa,IAAA,GAAA,CAAA;AAAA,cAChH,QAAA,EAAU,MAAM,IAAK,CAAA,QAAA;AAAA,cACrB,UAAA,EAAY,MAAM,IAAK,CAAA,IAAA;AAAA,cACvB,OAAA,EAAS,MAAM,IAAK,CAAA,OAAA;AAAA,cACpB,SAAA,EAAW,MAAM,IAAK,CAAA,SAAA;AAAA,cACtB,eAAe,mBAAoB,CAAA;AAAA,gBACjC,GAAG,WAAY,CAAA,OAAA,CAAQ,WAAY,CAAA,aAAA;AAAA,gBACnC,MAAA,EAAQ,KAAM,CAAA,IAAA,CAAK,MAAU,IAAA,CAAA;AAAA,gBAC7B,KAAA,EAAO,KAAM,CAAA,IAAA,CAAK,OAAQ,CAAA,MAAA;AAAA,gBAC1B,QAAU,EAAA,UAAA,GAAa,WAAY,CAAA,OAAA,CAAQ,YAAY,aAAc,CAAA,SAAA;AAAA,gBACrE,OAAS,EAAA;AAAA,eACV;AAAA,aACF,CAAA;AAED,YAAA,IAAI,QAAQ,SAAW,EAAA;AACrB,cAAA,OAAA,CAAQ,UAAU,UAAU,CAAA;AAAA;AAC9B;AACF;AACF;AACF,KACF;AAEA,IAAA,MAAM,mBAAsB,GAAA;AAAA,MAC1B,GAAG,OAAA;AAAA,MACH,QAAA;AAAA,MACA,KAAA;AAAA,MACA,QAAU,EAAA,gBAAA;AAAA,MACV,SAAW,EAAA,iBAAA;AAAA,MACX,QAAU,EAAA,gBAAA;AAAA,MACV,QAAU,EAAA,gBAAA;AAAA,MACV,UAAU,cAAgB,EAAA,GAAA,CAAI,KAAK,CAAA,wBAAS,GAAI,EAAA;AAAA,MAChD,SAAW,EAAA,OAAA,CAAQ,SAAa,oBAAA,IAAI,GAAI,EAAA;AAAA,MACxC,IAAI,CAAG,EAAA,KAAK,CAAI,CAAA,EAAA,IAAA,CAAK,KAAK,CAAA,CAAA,EAAI,IAAK,CAAA,MAAA,GAAS,QAAS,CAAA,EAAE,EAAE,SAAU,CAAA,CAAA,EAAG,EAAE,CAAC,CAAA,CAAA;AAAA,MACzE,OAAS,EAAA;AAAA,KACX;AAEA,IAAM,MAAA,YAAA,GAAe,WAAW,mBAAmB,CAAA;AACnD,IAAA,MAAM,UAA8B,EAAC;AACrC,IAAI,IAAA,UAAA;AAGJ,IAAA,WAAA,MAAiB,UAAU,YAAc,EAAA;AACvC,MAAA,OAAA,CAAQ,KAAK,MAAM,CAAA;AAEnB,MAAA,IAAI,MAAO,CAAA,IAAA,KAAS,OAAW,IAAA,MAAA,CAAO,KAAO,EAAA;AAC3C,QAAa,UAAA,GAAA,MAAA,CAAO,KAAiB,YAAA,KAAA,GAAQ,MAAO,CAAA,KAAA,GAAQ,IAAI,KAAM,CAAA,MAAA,CAAO,MAAO,CAAA,KAAK,CAAC,CAAA;AAAA;AAG5F,MAAA,IAAI,MAAO,CAAA,IAAA,KAAS,SAAa,IAAA,MAAA,CAAO,SAAS,MAAQ,EAAA;AAEvD,QAAa,YAAA,CAAA,GAAA,CAAI,OAAO,MAAM,CAAA;AAE9B,QAAA,MAAM,eAAkB,GAAA,UAAA;AAAA,UACtB,MAAO,CAAA,GAAA;AAAA,UACP,KAAA;AAAA,UACA,EAAE,GAAG,OAAS,EAAA,KAAA,EAAO,SAAS,cAAgB,EAAA,MAAA,EAAQ,QAAQ,MAAO,EAAA;AAAA,UACrE,OAAQ,CAAA;AAAA,SACV;AAEA,QAAA,MAAM,gBAAmB,GAAA,UAAA;AAAA,UACvB,MAAO,CAAA,IAAA;AAAA,UACP,MAAA;AAAA,UACA,EAAE,GAAG,OAAS,EAAA,KAAA,EAAO,SAAS,cAAgB,EAAA,MAAA,EAAQ,QAAQ,MAAO,EAAA;AAAA,UACrE,OAAQ,CAAA;AAAA,SACV;AAEA,QAAA,MAAM,OAAQ,CAAA,GAAA,CAAI,CAAC,eAAA,EAAiB,gBAAgB,CAAC,CAAA;AAAA;AACvD;AAGF,IAAA,IAAI,UAAY,EAAA;AACd,MAAA,OAAO,EAAE,KAAA,EAAO,OAAS,EAAA,KAAA,EAAO,UAAW,EAAA;AAAA;AAG7C,IAAO,OAAA,EAAE,OAAO,OAAQ,EAAA;AAAA,WACjB,KAAO,EAAA;AACd,IAAA,OAAO,EAAE,KAAA,EAAO,OAAS,EAAA,IAAI,KAAsB,EAAA;AAAA;AAEvD;AAKA,SAAS,KAAA,CAAS,OAAY,IAAqB,EAAA;AACjD,EAAA,MAAM,SAAgB,EAAC;AACvB,EAAA,KAAA,IAAS,IAAI,CAAG,EAAA,CAAA,GAAI,KAAM,CAAA,MAAA,EAAQ,KAAK,IAAM,EAAA;AAC3C,IAAA,MAAA,CAAO,KAAK,KAAM,CAAA,KAAA,CAAM,CAAG,EAAA,CAAA,GAAI,IAAI,CAAC,CAAA;AAAA;AAEtC,EAAO,OAAA,MAAA;AACT;AAKO,MAAM,kBAAoC,GAAA,CAC/C,MACA,EAAA,cAAA,EACA,UACG,KAAA;AACH,EAAM,MAAA;AAAA,IACJ,mBAAA;AAAA,IACA,WAAW,EAAC;AAAA,IACZ,GAAG;AAAA,GACD,GAAA,cAAA;AAEJ,EAAM,MAAA,SAAA,GAAa,QAAgB,SAAa,IAAA,kBAAA;AAChD,EAAM,MAAA,eAAA,uBAAsB,GAAY,EAAA;AACxC,EAAM,MAAA,YAAA,uBAAmB,GAAqB,EAAA;AAC9C,EAAM,MAAA,OAAA,uBAAc,GAA8B,EAAA;AAElD,EAAI,IAAA,CAAC,qBAAqB,MAAQ,EAAA;AAChC,IAAA,OAAQ,gBAAgB,mBAAqF,GAAA;AAC3G,MAAA,MAAM,WAAiC,GAAA;AAAA,QACrC,IAAM,EAAA,OAAA;AAAA,QACN,KAAA,EAAO,IAAI,KAAA,CAAM,mCAAmC,CAAA;AAAA,QACpD,KAAO,EAAA,EAAA;AAAA,QACP,YAAA,sBAAkB,GAAI,EAAA;AAAA,QACtB,eAAA,sBAAqB,GAAI,EAAA;AAAA,QACzB,OAAA,sBAAa,GAAI;AAAA,OACnB;AACA,MAAM,MAAA,WAAA;AACN,MAAO,OAAA,WAAA;AAAA,KACN,EAAA;AAAA;AAGL,EAAA,OAAQ,gBAAgB,mBAAqF,GAAA;AAC3G,IAAM,MAAA,UAAA,GAAa,KAAM,CAAA,IAAA,CAAK,MAAM,CAAA;AACpC,IAAM,MAAA,OAAA,GAAU,KAAM,CAAA,UAAA,EAAY,SAAS,CAAA;AAE3C,IAAA,IAAI,QAAQ,OAAS,EAAA;AACnB,MAAA,OAAA,CAAQ,MAAQ,EAAA,IAAA;AAAA,QACd,kCAAkC,UAAW,CAAA,MAAM,aAAa,OAAQ,CAAA,MAAM,eAAe,SAAS,CAAA;AAAA,OACxG;AAAA;AAGF,IAAA,KAAA,MAAW,SAAS,OAAS,EAAA;AAE3B,MAAI,IAAA,OAAA,CAAQ,QAAQ,OAAS,EAAA;AAC3B,QAAA,MAAM,WAAiC,GAAA;AAAA,UACrC,IAAM,EAAA,OAAA;AAAA,UACN,OAAO,OAAQ,CAAA,MAAA,CAAO,MAAU,IAAA,IAAI,MAAM,eAAe,CAAA;AAAA,UACzD,KAAA,EAAO,KAAM,CAAA,CAAC,CAAK,IAAA,EAAA;AAAA,UACnB,YAAA;AAAA,UACA,eAAA;AAAA,UACA;AAAA,SACF;AACA,QAAM,MAAA,WAAA;AACN,QAAO,OAAA,WAAA;AAAA;AAIT,MAAA,MAAM,gBAAgB,KAAM,CAAA,GAAA;AAAA,QAAI,WAC9B,iBAAkB,CAAA,KAAA,EAAO,cAAgB,EAAA,UAAA,EAAY,UAAU,YAAY;AAAA,OAC7E;AAEA,MAAA,MAAM,YAAe,GAAA,MAAM,OAAQ,CAAA,GAAA,CAAI,aAAa,CAAA;AAGpD,MAAA,KAAA,MAAW,EAAE,KAAO,EAAA,OAAA,EAAS,WAAa,EAAA,KAAA,MAAW,YAAc,EAAA;AACjE,QAAA,IAAI,KAAO,EAAA;AACT,UAAA,MAAM,aAAa,WAAY,CAAA;AAAA,YAC7B,KAAA;AAAA,YACA,QAAQ,OAAQ,CAAA,MAAA;AAAA,YAChB,gBAAgB,OAAQ,CAAA,cAE1B,CAAC,CAAA;AAED,UAAA,IAAI,cAAc,IAAM,EAAA;AACtB,YAAa,YAAA,CAAA,GAAA,CAAI,OAAO,KAAK,CAAA;AAC7B,YAAA,OAAA,CAAQ,QAAQ,KAAM,CAAA,CAAA,2CAAA,EAA8C,KAAK,CAAK,EAAA,EAAA,KAAA,CAAM,OAAO,CAAE,CAAA,CAAA;AAC7F,YAAA,MAAM,WAAiC,GAAA;AAAA,cACrC,IAAM,EAAA,OAAA;AAAA,cACN,KAAA;AAAA,cACA,KAAA;AAAA,cACA,YAAA;AAAA,cACA,eAAA;AAAA,cACA;AAAA,aACF;AACA,YAAM,MAAA,WAAA;AACN,YAAO,OAAA,WAAA;AAAA,WACF,MAAA;AACL,YAAA,OAAA,CAAQ,QAAQ,IAAK,CAAA,CAAA,+CAAA,EAAkD,KAAK,CAAK,EAAA,EAAA,KAAA,CAAM,OAAO,CAAE,CAAA,CAAA;AAAA;AAClG,SACK,MAAA;AACL,UAAA,eAAA,CAAgB,IAAI,KAAK,CAAA;AAEzB,UAAA,KAAA,MAAW,UAAU,WAAa,EAAA;AAChC,YAAA,IAAI,MAAO,CAAA,IAAA,KAAS,SAAa,IAAA,MAAA,CAAO,SAAS,MAAQ,EAAA;AACvD,cAAQ,OAAA,CAAA,GAAA,CAAI,OAAO,MAAM,CAAA;AACzB,cAAM,MAAA;AAAA,gBACJ,IAAM,EAAA,SAAA;AAAA,gBACN,KAAA;AAAA,gBACA,YAAA;AAAA,gBACA,eAAA;AAAA,gBACA;AAAA,eACF;AAAA;AACF;AACF;AACF;AAGF,MAAA,IAAI,QAAQ,OAAS,EAAA;AACnB,QAAA,OAAA,CAAQ,MAAQ,EAAA,IAAA;AAAA,UACd,CAAyC,sCAAA,EAAA,eAAA,CAAgB,IAAI,CAAA,CAAA,EAAI,WAAW,MAAM,CAAA,MAAA;AAAA,SACpF;AAAA;AACF;AAIF,IAAA,MAAM,WAAiC,GAAA;AAAA,MACrC,IAAM,EAAA,SAAA;AAAA,MACN,KAAO,EAAA,EAAA;AAAA,MACP,YAAA;AAAA,MACA,eAAA;AAAA,MACA;AAAA,KACF;AAEA,IAAO,OAAA,WAAA;AAAA,GACN,EAAA;AACL;;;;"}