vite-plugin-react-server 1.4.2 → 1.4.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +48 -313
- package/dist/package.json +123 -13
- package/dist/plugin/bundle/deferredStaticGeneration.js +14 -39
- package/dist/plugin/bundle/manifests.js +30 -48
- package/dist/plugin/config/autoDiscover/resolveAutoDiscover.d.ts.map +1 -1
- package/dist/plugin/config/autoDiscover/resolveAutoDiscover.js +4 -1
- package/dist/plugin/config/envPrefixFromConfig.js +12 -7
- package/dist/plugin/config/getCondition.d.ts.map +1 -1
- package/dist/plugin/config/getCondition.js +7 -5
- package/dist/plugin/dev-server/virtualRscHmrPlugin.js +23 -23
- package/dist/plugin/environments/createBuildEventPlugin.js +88 -98
- package/dist/plugin/environments/createEnvironmentPlugin.js +222 -250
- package/dist/plugin/helpers/createRscRenderHelpers.js +33 -34
- package/dist/plugin/helpers/createSharedLoader.d.ts.map +1 -1
- package/dist/plugin/helpers/createSharedLoader.js +4 -2
- package/dist/plugin/helpers/headlessStreamReuseHandler.js +30 -22
- package/dist/plugin/helpers/headlessStreamState.js +15 -28
- package/dist/plugin/helpers/resolveComponent.d.ts.map +1 -1
- package/dist/plugin/helpers/resolveComponent.js +4 -2
- package/dist/plugin/index.client.d.ts +5 -0
- package/dist/plugin/index.client.d.ts.map +1 -0
- package/dist/plugin/index.client.js +4 -0
- package/dist/plugin/index.d.ts +4 -3
- package/dist/plugin/index.d.ts.map +1 -1
- package/dist/plugin/index.js +10 -5
- package/dist/plugin/index.server.d.ts +5 -0
- package/dist/plugin/index.server.d.ts.map +1 -0
- package/dist/plugin/index.server.js +4 -0
- package/dist/plugin/metrics/createWorkerStartupMetrics.js +31 -13
- package/dist/plugin/orchestrator/createPluginOrchestrator.client.js +41 -38
- package/dist/plugin/orchestrator/createPluginOrchestrator.server.js +43 -46
- package/dist/plugin/plugin.client.js +2 -2
- package/dist/plugin/plugin.server.js +2 -2
- package/dist/plugin/react-static/createBuildLoader.client.js +12 -6
- package/dist/plugin/react-static/createBuildLoader.server.js +255 -235
- package/dist/plugin/react-static/plugin.client.js +684 -770
- package/dist/plugin/react-static/plugin.server.js +517 -603
- package/dist/plugin/react-static/processCssFilesForPages.js +103 -88
- package/dist/plugin/react-static/renderPage.client.js +455 -529
- package/dist/plugin/react-static/renderPage.server.js +485 -508
- package/dist/plugin/react-static/renderPagesBatched.js +277 -275
- package/dist/plugin/react-static/rscToHtmlStream.client.js +48 -29
- package/dist/plugin/react-static/rscToHtmlStream.server.js +62 -37
- package/dist/plugin/react-static/temporaryReferences.server.js +11 -2
- package/dist/plugin/stream/createMainThreadHandlers.js +40 -31
- package/dist/plugin/stream/renderRscStream.server.d.ts.map +1 -1
- package/dist/plugin/stream/renderRscStream.server.js +127 -144
- package/dist/plugin/transformer/createTransformerPlugin.js +226 -265
- package/dist/plugin/utils/checkReactVersion.d.ts +7 -0
- package/dist/plugin/utils/checkReactVersion.d.ts.map +1 -0
- package/dist/plugin/utils/checkReactVersion.js +23 -0
- package/dist/plugin/utils/envUrls.node.js +12 -11
- package/dist/plugin/vendor/vendor-alias.js +84 -114
- package/dist/plugin/vendor/vendor.client.d.ts.map +1 -1
- package/dist/plugin/vendor/vendor.client.js +1 -3
- package/dist/plugin/worker/rsc/handleRscRender.d.ts.map +1 -1
- package/dist/plugin/worker/rsc/handleRscRender.js +3 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +123 -13
- package/plugin/config/autoDiscover/resolveAutoDiscover.ts +4 -0
- package/plugin/config/getCondition.ts +6 -4
- package/plugin/helpers/createSharedLoader.ts +6 -1
- package/plugin/helpers/resolveComponent.ts +6 -1
- package/plugin/index.client.ts +4 -0
- package/plugin/index.server.ts +4 -0
- package/plugin/index.ts +12 -5
- package/plugin/plugin.client.ts +1 -1
- package/plugin/plugin.server.ts +1 -1
- package/plugin/stream/renderRscStream.server.ts +3 -0
- package/plugin/utils/checkReactVersion.ts +28 -0
- package/plugin/vendor/vendor.client.ts +0 -2
- package/plugin/worker/rsc/handleRscRender.ts +2 -0
- package/scripts/generate-toc.mjs +27 -294
|
@@ -1,295 +1,297 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
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
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
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
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
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
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
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
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
179
|
-
|
|
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
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
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
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
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
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
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
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
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
|
-
|
|
286
|
-
|
|
287
|
-
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
|
|
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;;;;"}
|