autotel-devtools 4.0.0 → 5.0.0
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 +55 -0
- package/dist/cli.cjs +97 -19
- package/dist/cli.cjs.map +1 -1
- package/dist/cli.js +97 -19
- package/dist/cli.js.map +1 -1
- package/dist/index.cjs +71 -2
- package/dist/index.cjs.map +1 -1
- package/dist/index.js +71 -2
- package/dist/index.js.map +1 -1
- package/dist/server/index.cjs +11 -1
- package/dist/server/index.cjs.map +1 -1
- package/dist/server/index.js +11 -1
- package/dist/server/index.js.map +1 -1
- package/package.json +2 -2
package/README.md
CHANGED
|
@@ -151,6 +151,61 @@ Options:
|
|
|
151
151
|
- `--host, -H` - Host to bind to (default: 127.0.0.1)
|
|
152
152
|
- `--title, -t` - Dashboard title
|
|
153
153
|
|
|
154
|
+
When bound to a loopback host, the receiver listens on **both** `127.0.0.1`
|
|
155
|
+
and `::1`, so a client connecting via `localhost` reaches it regardless of how
|
|
156
|
+
the OS resolves `localhost` (macOS prefers IPv6 `::1`). The startup banner
|
|
157
|
+
prints every address it bound; if a family can't be bound you get a warning,
|
|
158
|
+
not a silent black hole.
|
|
159
|
+
|
|
160
|
+
## Behind a dev-server proxy
|
|
161
|
+
|
|
162
|
+
If your app's dev server proxies `/v1/traces` to the receiver, two classic
|
|
163
|
+
bugs make spans vanish with **no error** — both worth knowing:
|
|
164
|
+
|
|
165
|
+
```ts
|
|
166
|
+
// Express / http-proxy-middleware
|
|
167
|
+
import { createProxyMiddleware } from 'http-proxy-middleware';
|
|
168
|
+
|
|
169
|
+
app.use(
|
|
170
|
+
'/v1/traces',
|
|
171
|
+
createProxyMiddleware({
|
|
172
|
+
// (a) Express strips the mount prefix before calling middleware, so the
|
|
173
|
+
// proxy would otherwise forward "/" instead of "/v1/traces".
|
|
174
|
+
pathRewrite: () => '/v1/traces',
|
|
175
|
+
// (b) Use 127.0.0.1, NOT localhost. On macOS `localhost` resolves to ::1;
|
|
176
|
+
// pin the family so you reach the receiver deterministically.
|
|
177
|
+
target: 'http://127.0.0.1:4318',
|
|
178
|
+
changeOrigin: true,
|
|
179
|
+
}),
|
|
180
|
+
);
|
|
181
|
+
```
|
|
182
|
+
|
|
183
|
+
> Symptom of either bug: the browser shows the request leaving (200/no error),
|
|
184
|
+
> but the receiver stays empty. Always verify on the **receiver** side (below),
|
|
185
|
+
> not just that the browser tried to send.
|
|
186
|
+
|
|
187
|
+
## Verifying ingestion in tests
|
|
188
|
+
|
|
189
|
+
The receiver exposes an HTTP read-back so a test can assert the collector
|
|
190
|
+
**actually received** spans — instead of asserting "the client tried to send",
|
|
191
|
+
which a browser-level route intercept can fake (it fulfils the request before
|
|
192
|
+
it reaches any server):
|
|
193
|
+
|
|
194
|
+
```bash
|
|
195
|
+
GET /v1/traces # → { traces: [...], count: N } what the receiver has
|
|
196
|
+
DELETE /v1/traces # clear captured telemetry (reset between tests)
|
|
197
|
+
```
|
|
198
|
+
|
|
199
|
+
```ts
|
|
200
|
+
// Playwright / integration test — bypass any page.route() intercept and ask
|
|
201
|
+
// the collector directly.
|
|
202
|
+
await fetch(`${RECEIVER}/v1/traces`, { method: 'DELETE' }); // reset
|
|
203
|
+
await runTheUserFlow(); // app emits spans
|
|
204
|
+
await expect
|
|
205
|
+
.poll(async () => (await (await fetch(`${RECEIVER}/v1/traces`)).json()).count)
|
|
206
|
+
.toBeGreaterThan(0);
|
|
207
|
+
```
|
|
208
|
+
|
|
154
209
|
## License
|
|
155
210
|
|
|
156
211
|
MIT
|
package/dist/cli.cjs
CHANGED
|
@@ -700,7 +700,7 @@ function getWidgetJs() {
|
|
|
700
700
|
function attachDevtoolsRoutes(httpServer, devtools) {
|
|
701
701
|
httpServer.on("request", async (req, res) => {
|
|
702
702
|
res.setHeader("Access-Control-Allow-Origin", "*");
|
|
703
|
-
res.setHeader("Access-Control-Allow-Methods", "GET, POST, OPTIONS");
|
|
703
|
+
res.setHeader("Access-Control-Allow-Methods", "GET, POST, DELETE, OPTIONS");
|
|
704
704
|
res.setHeader("Access-Control-Allow-Headers", "Content-Type");
|
|
705
705
|
if (req.method === "OPTIONS") {
|
|
706
706
|
res.writeHead(204);
|
|
@@ -723,6 +723,16 @@ function attachDevtoolsRoutes(httpServer, devtools) {
|
|
|
723
723
|
sendJson(res, 200, { ok: true, clients: devtools.clientCount });
|
|
724
724
|
return;
|
|
725
725
|
}
|
|
726
|
+
if (req.method === "GET" && url === "/v1/traces") {
|
|
727
|
+
const data = devtools.getCurrentData();
|
|
728
|
+
sendJson(res, 200, { traces: data.traces, count: data.traces.length });
|
|
729
|
+
return;
|
|
730
|
+
}
|
|
731
|
+
if (req.method === "DELETE" && url === "/v1/traces") {
|
|
732
|
+
devtools.clearData();
|
|
733
|
+
sendJson(res, 200, { cleared: true });
|
|
734
|
+
return;
|
|
735
|
+
}
|
|
726
736
|
if (req.method === "POST" && url === "/v1/traces") {
|
|
727
737
|
try {
|
|
728
738
|
const payload = await readJsonBody(req);
|
|
@@ -758,6 +768,54 @@ function attachDevtoolsRoutes(httpServer, devtools) {
|
|
|
758
768
|
sendJson(res, 404, { error: "Not found" });
|
|
759
769
|
});
|
|
760
770
|
}
|
|
771
|
+
var LOOPBACK = /* @__PURE__ */ new Set(["localhost", "127.0.0.1", "::1"]);
|
|
772
|
+
function formatAddress(host, port) {
|
|
773
|
+
return host.includes(":") ? `[${host}]:${port}` : `${host}:${port}`;
|
|
774
|
+
}
|
|
775
|
+
function listenLoopbackDualStack(args) {
|
|
776
|
+
const { primary, port, host, attachSecondary } = args;
|
|
777
|
+
let sibling;
|
|
778
|
+
const ready = new Promise(
|
|
779
|
+
(resolve3) => {
|
|
780
|
+
const addresses = [];
|
|
781
|
+
const warnings = [];
|
|
782
|
+
const primaryHost = host === "localhost" ? "127.0.0.1" : host;
|
|
783
|
+
primary.listen(port, primaryHost, () => {
|
|
784
|
+
const addr = primary.address();
|
|
785
|
+
const resolvedPort = addr && typeof addr === "object" ? addr.port : port;
|
|
786
|
+
addresses.push(formatAddress(primaryHost, resolvedPort));
|
|
787
|
+
if (!LOOPBACK.has(host)) {
|
|
788
|
+
resolve3({ addresses, warnings });
|
|
789
|
+
return;
|
|
790
|
+
}
|
|
791
|
+
const siblingHost = primaryHost === "::1" ? "127.0.0.1" : "::1";
|
|
792
|
+
const s = http.createServer();
|
|
793
|
+
attachSecondary(s);
|
|
794
|
+
const onError = (e) => {
|
|
795
|
+
s.close();
|
|
796
|
+
warnings.push(
|
|
797
|
+
`could not also bind ${formatAddress(siblingHost, resolvedPort)} (${e.message}); clients using the ${siblingHost === "::1" ? "IPv6" : "IPv4"} form of "localhost" may not connect.`
|
|
798
|
+
);
|
|
799
|
+
resolve3({ addresses, warnings });
|
|
800
|
+
};
|
|
801
|
+
s.once("error", onError);
|
|
802
|
+
s.listen(resolvedPort, siblingHost, () => {
|
|
803
|
+
s.off("error", onError);
|
|
804
|
+
sibling = s;
|
|
805
|
+
addresses.push(formatAddress(siblingHost, resolvedPort));
|
|
806
|
+
resolve3({ addresses, warnings });
|
|
807
|
+
});
|
|
808
|
+
});
|
|
809
|
+
}
|
|
810
|
+
);
|
|
811
|
+
return {
|
|
812
|
+
ready,
|
|
813
|
+
closeSibling: () => new Promise((res) => {
|
|
814
|
+
if (!sibling) return res();
|
|
815
|
+
sibling.close(() => res());
|
|
816
|
+
})
|
|
817
|
+
};
|
|
818
|
+
}
|
|
761
819
|
|
|
762
820
|
// src/cli.ts
|
|
763
821
|
function printHelp() {
|
|
@@ -775,13 +833,15 @@ Options:
|
|
|
775
833
|
-v, --version Show version number
|
|
776
834
|
|
|
777
835
|
Endpoints:
|
|
778
|
-
GET
|
|
779
|
-
GET
|
|
780
|
-
POST
|
|
781
|
-
|
|
782
|
-
|
|
783
|
-
|
|
784
|
-
|
|
836
|
+
GET / Web devtools UI (fullpage)
|
|
837
|
+
GET /widget.js Widget bundle (embed in your app)
|
|
838
|
+
POST /v1/traces Receive OTLP JSON trace data
|
|
839
|
+
GET /v1/traces Read back received traces (verify ingestion in tests)
|
|
840
|
+
DELETE /v1/traces Clear captured telemetry (test reset)
|
|
841
|
+
POST /v1/logs Receive OTLP JSON log data
|
|
842
|
+
POST /v1/metrics Receive OTLP JSON metric data
|
|
843
|
+
WS /ws WebSocket stream for real-time updates
|
|
844
|
+
GET /healthz Health check
|
|
785
845
|
|
|
786
846
|
Examples:
|
|
787
847
|
npx autotel-devtools
|
|
@@ -853,29 +913,47 @@ async function main() {
|
|
|
853
913
|
const httpServer = http.createServer();
|
|
854
914
|
const wsServer = new DevtoolsServer({ server: httpServer, verbose: true });
|
|
855
915
|
attachDevtoolsRoutes(httpServer, wsServer);
|
|
856
|
-
|
|
857
|
-
|
|
858
|
-
|
|
916
|
+
const listeners = listenLoopbackDualStack({
|
|
917
|
+
primary: httpServer,
|
|
918
|
+
port: options.port,
|
|
919
|
+
host: options.host,
|
|
920
|
+
attachSecondary: (s) => attachDevtoolsRoutes(s, wsServer)
|
|
921
|
+
});
|
|
922
|
+
const { addresses, warnings } = await listeners.ready;
|
|
923
|
+
const uiBase = `http://${options.host === "localhost" ? "127.0.0.1" : options.host}:${options.port}`;
|
|
924
|
+
const title = options.title || "autotel-devtools";
|
|
925
|
+
process.stdout.write(`
|
|
859
926
|
${title}
|
|
860
927
|
|
|
861
928
|
`);
|
|
862
|
-
|
|
929
|
+
process.stdout.write(` Listening: ${addresses.join(" + ")}
|
|
930
|
+
`);
|
|
931
|
+
process.stdout.write(` UI: ${uiBase}
|
|
863
932
|
`);
|
|
864
|
-
|
|
933
|
+
process.stdout.write(` Widget: <script src="${uiBase}/widget.js"></script>
|
|
865
934
|
`);
|
|
866
|
-
|
|
935
|
+
process.stdout.write(` WebSocket: ${uiBase.replace("http", "ws")}/ws
|
|
867
936
|
`);
|
|
868
|
-
|
|
937
|
+
process.stdout.write(` OTLP: ${uiBase}/v1/traces
|
|
869
938
|
|
|
870
939
|
`);
|
|
871
|
-
|
|
940
|
+
process.stdout.write(` Set OTEL_EXPORTER_OTLP_PROTOCOL=http/json
|
|
872
941
|
`);
|
|
873
|
-
|
|
942
|
+
process.stdout.write(` Set OTEL_EXPORTER_OTLP_ENDPOINT=${uiBase}
|
|
874
943
|
|
|
875
944
|
`);
|
|
876
|
-
}
|
|
945
|
+
process.stdout.write(` Verify ingestion: curl -s ${uiBase}/v1/traces
|
|
946
|
+
|
|
947
|
+
`);
|
|
948
|
+
for (const w of warnings) {
|
|
949
|
+
process.stdout.write(` \u26A0 ${w}
|
|
950
|
+
`);
|
|
951
|
+
}
|
|
952
|
+
if (warnings.length > 0) process.stdout.write("\n");
|
|
877
953
|
const shutdown = () => {
|
|
878
|
-
wsServer.close().
|
|
954
|
+
Promise.all([wsServer.close(), listeners.closeSibling()]).then(
|
|
955
|
+
() => process.exit(0)
|
|
956
|
+
);
|
|
879
957
|
};
|
|
880
958
|
process.on("SIGINT", shutdown);
|
|
881
959
|
process.on("SIGTERM", shutdown);
|
package/dist/cli.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/server/error-aggregator.ts","../src/server/telemetry-limits.ts","../src/server/server.ts","../src/server/resource-utils.ts","../src/server/otlp.ts","../src/server/http.ts","../src/cli.ts"],"names":["createServer","WebSocketServer","WebSocket","resolve","dirname","fileURLToPath","existsSync","readFileSync"],"mappings":";;;;;;;;;;;AAsDO,IAAM,kBAAN,MAAsB;AAAA,EACnB,WAAA,uBAA2C,GAAA,EAAI;AAAA,EAC/C,OAAA;AAAA,EAER,WAAA,CAAY,OAAA,GAAkC,EAAC,EAAG;AAChD,IAAA,IAAA,CAAK,OAAA,GAAU;AAAA,MACb,SAAA,EAAW,QAAQ,SAAA,IAAa,GAAA;AAAA,MAChC,iBAAA,EAAmB,QAAQ,iBAAA,IAAqB,EAAA;AAAA,MAChD,gBAAA,EAAkB,QAAQ,gBAAA,IAAoB,CAAA;AAAA,MAC9C,yBAAA,EAA2B,QAAQ,yBAAA,IAA6B;AAAA,KAClE;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,SAAS,UAAA,EAAyC;AAChD,IAAA,MAAM,WAAA,GAAc,IAAA,CAAK,mBAAA,CAAoB,UAAU,CAAA;AACvD,IAAA,MAAM,QAAA,GAAW,IAAA,CAAK,WAAA,CAAY,GAAA,CAAI,WAAW,CAAA;AAEjD,IAAA,IAAI,QAAA,EAAU;AAEZ,MAAA,QAAA,CAAS,KAAA,EAAA;AACT,MAAA,QAAA,CAAS,WAAW,UAAA,CAAW,SAAA;AAG/B,MAAA,IAAI,CAAC,QAAA,CAAS,cAAA,CAAe,QAAA,CAAS,UAAA,CAAW,OAAO,CAAA,EAAG;AACzD,QAAA,QAAA,CAAS,cAAA,CAAe,IAAA,CAAK,UAAA,CAAW,OAAO,CAAA;AAC/C,QAAA,IAAI,QAAA,CAAS,cAAA,CAAe,MAAA,GAAS,IAAA,CAAK,QAAQ,iBAAA,EAAmB;AACnE,UAAA,QAAA,CAAS,eAAe,KAAA,EAAM;AAAA,QAChC;AAAA,MACF;AAGA,MAAA,IAAI,CAAC,QAAA,CAAS,aAAA,CAAc,QAAA,CAAS,UAAA,CAAW,QAAQ,CAAA,EAAG;AACzD,QAAA,QAAA,CAAS,aAAA,CAAc,IAAA,CAAK,UAAA,CAAW,QAAQ,CAAA;AAC/C,QAAA,IAAI,QAAA,CAAS,aAAA,CAAc,MAAA,GAAS,IAAA,CAAK,QAAQ,gBAAA,EAAkB;AACjE,UAAA,QAAA,CAAS,cAAc,KAAA,EAAM;AAAA,QAC/B;AAAA,MACF;AAEA,MAAA,OAAO,QAAA;AAAA,IACT;AAGA,IAAA,MAAM,QAAA,GAAuB;AAAA,MAC3B,WAAA;AAAA,MACA,IAAA,EAAM,WAAW,KAAA,CAAM,IAAA;AAAA,MACvB,OAAA,EAAS,WAAW,KAAA,CAAM,OAAA;AAAA,MAC1B,UAAA,EAAY,IAAA,CAAK,mBAAA,CAAoB,UAAA,CAAW,MAAM,UAAU,CAAA;AAAA,MAChE,KAAA,EAAO,CAAA;AAAA,MACP,WAAW,UAAA,CAAW,SAAA;AAAA,MACtB,UAAU,UAAA,CAAW,SAAA;AAAA,MACrB,cAAA,EAAgB,CAAC,UAAA,CAAW,OAAO,CAAA;AAAA,MACnC,aAAA,EAAe,CAAC,UAAA,CAAW,QAAQ,CAAA;AAAA,MACnC,SAAS,UAAA,CAAW,OAAA;AAAA,MACpB,YAAY,UAAA,CAAW;AAAA,KACzB;AAGA,IAAA,IAAI,IAAA,CAAK,WAAA,CAAY,IAAA,IAAQ,IAAA,CAAK,QAAQ,SAAA,EAAW;AACnD,MAAA,IAAA,CAAK,gBAAA,EAAiB;AAAA,IACxB;AAEA,IAAA,IAAA,CAAK,WAAA,CAAY,GAAA,CAAI,WAAA,EAAa,QAAQ,CAAA;AAC1C,IAAA,OAAO,QAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,mBAAmB,KAAA,EAAgC;AACjD,IAAA,MAAM,cAA4B,EAAC;AAEnC,IAAA,KAAA,MAAW,IAAA,IAAQ,MAAM,KAAA,EAAO;AAC9B,MAAA,IAAI,IAAA,CAAK,MAAA,CAAO,IAAA,KAAS,OAAA,EAAS;AAChC,QAAA,MAAM,UAAA,GAAa,IAAA,CAAK,oBAAA,CAAqB,IAAA,EAAM,KAAK,CAAA;AACxD,QAAA,IAAI,UAAA,EAAY;AACd,UAAA,MAAM,KAAA,GAAQ,IAAA,CAAK,QAAA,CAAS,UAAU,CAAA;AACtC,UAAA,WAAA,CAAY,KAAK,KAAK,CAAA;AAAA,QACxB;AAAA,MACF;AAAA,IACF;AAEA,IAAA,OAAO,WAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKQ,oBAAA,CACN,MACA,KAAA,EACwB;AAExB,IAAA,MAAM,cAAA,GAAiB,KAAK,MAAA,EAAQ,IAAA,CAAK,CAAC,CAAA,KAAM,CAAA,CAAE,SAAS,WAAW,CAAA;AACtE,IAAA,MAAM,SAAA,GACH,IAAA,CAAK,UAAA,CAAW,gBAAgB,CAAA,IAChC,IAAA,CAAK,UAAA,CAAW,YAAY,CAAA,IAC5B,cAAA,EAAgB,UAAA,GAAa,gBAAgB,CAAA,IAC9C,OAAA;AAEF,IAAA,MAAM,YAAA,GACJ,IAAA,CAAK,MAAA,CAAO,OAAA,IACX,IAAA,CAAK,UAAA,CAAW,mBAAmB,CAAA,IACnC,IAAA,CAAK,UAAA,CAAW,eAAe,CAAA,IAChC,eAAA;AAEF,IAAA,MAAM,UAAA,GACH,IAAA,CAAK,UAAA,CAAW,sBAAsB,CAAA,IACtC,IAAA,CAAK,UAAA,CAAW,iBAAiB,CAAA,IAClC,IAAA,CAAK,sBAAA,CAAuB,IAAI,CAAA;AAElC,IAAA,OAAO;AAAA,MACL,SAAS,KAAA,CAAM,OAAA;AAAA,MACf,QAAQ,IAAA,CAAK,MAAA;AAAA,MACb,UAAU,IAAA,CAAK,IAAA;AAAA,MACf,SAAS,KAAA,CAAM,OAAA;AAAA,MACf,WAAW,IAAA,CAAK,OAAA;AAAA,MAChB,KAAA,EAAO;AAAA,QACL,IAAA,EAAM,SAAA;AAAA,QACN,OAAA,EAAS,YAAA;AAAA,QACT;AAAA,OACF;AAAA,MACA,UAAA,EAAY,IAAA,CAAK,yBAAA,CAA0B,IAAA,CAAK,UAAU;AAAA,KAC5D;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKQ,uBAAuB,IAAA,EAAoC;AACjE,IAAA,IAAI,CAAC,IAAA,CAAK,MAAA,EAAQ,OAAO,MAAA;AAEzB,IAAA,MAAM,cAAA,GAAiB,KAAK,MAAA,CAAO,IAAA,CAAK,CAAC,CAAA,KAAM,CAAA,CAAE,SAAS,WAAW,CAAA;AACrE,IAAA,IAAI,gBAAgB,UAAA,EAAY;AAC9B,MAAA,OACG,eAAe,UAAA,CAAW,sBAAsB,CAAA,IAChD,cAAA,CAAe,WAAW,iBAAiB,CAAA;AAAA,IAEhD;AAEA,IAAA,OAAO,MAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKQ,0BACN,UAAA,EACyB;AACzB,IAAA,MAAM,WAAoC,EAAC;AAC3C,IAAA,MAAM,QAAA,GAAW;AAAA,MACf,aAAA;AAAA,MACA,UAAA;AAAA,MACA,YAAA;AAAA,MACA,kBAAA;AAAA,MACA,WAAA;AAAA,MACA,cAAA;AAAA,MACA,YAAA;AAAA,MACA,aAAA;AAAA,MACA,eAAA;AAAA,MACA,eAAA;AAAA,MACA,SAAA;AAAA,MACA;AAAA,KACF;AAEA,IAAA,KAAA,MAAW,OAAO,QAAA,EAAU;AAC1B,MAAA,IAAI,OAAO,UAAA,EAAY;AACrB,QAAA,QAAA,CAAS,GAAG,CAAA,GAAI,UAAA,CAAW,GAAG,CAAA;AAAA,MAChC;AAAA,IACF;AAEA,IAAA,OAAO,QAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOQ,oBAAoB,UAAA,EAAqC;AAC/D,IAAA,MAAM,KAAA,GAAkB,CAAC,UAAA,CAAW,KAAA,CAAM,IAAI,CAAA;AAE9C,IAAA,IAAI,UAAA,CAAW,MAAM,UAAA,EAAY;AAC/B,MAAA,MAAM,SAAS,IAAA,CAAK,kBAAA;AAAA,QAClB,WAAW,KAAA,CAAM,UAAA;AAAA,QACjB,KAAK,OAAA,CAAQ;AAAA,OACf;AACA,MAAA,KAAA,CAAM,IAAA,CAAK,GAAG,MAAM,CAAA;AAAA,IACtB,CAAA,MAAO;AAEL,MAAA,KAAA,CAAM,KAAK,IAAA,CAAK,gBAAA,CAAiB,UAAA,CAAW,KAAA,CAAM,OAAO,CAAC,CAAA;AAAA,IAC5D;AAGA,IAAA,OAAO,IAAA,CAAK,UAAA,CAAW,KAAA,CAAM,IAAA,CAAK,GAAG,CAAC,CAAA;AAAA,EACxC;AAAA;AAAA;AAAA;AAAA,EAKQ,kBAAA,CAAmB,YAAoB,KAAA,EAAyB;AACtE,IAAA,MAAM,KAAA,GAAQ,UAAA,CAAW,KAAA,CAAM,IAAI,CAAA;AACnC,IAAA,MAAM,SAAmB,EAAC;AAE1B,IAAA,KAAA,MAAW,QAAQ,KAAA,EAAO;AACxB,MAAA,IAAI,MAAA,CAAO,UAAU,KAAA,EAAO;AAG5B,MAAA,MAAM,OAAA,GAAU,KAAK,IAAA,EAAK;AAG1B,MAAA,MAAM,SAAA,GAAY,OAAA,CAAQ,KAAA,CAAM,oCAAoC,CAAA;AACpE,MAAA,IAAI,SAAA,EAAW;AACb,QAAA,MAAA,CAAO,IAAA,CAAK,CAAA,EAAG,SAAA,CAAU,CAAC,CAAC,CAAA,CAAA,EAAI,IAAA,CAAK,iBAAA,CAAkB,SAAA,CAAU,CAAC,CAAC,CAAC,CAAA,CAAE,CAAA;AACrE,QAAA;AAAA,MACF;AAGA,MAAA,MAAM,SAAA,GAAY,OAAA,CAAQ,KAAA,CAAM,wBAAwB,CAAA;AACxD,MAAA,IAAI,SAAA,EAAW;AACb,QAAA,MAAA,CAAO,IAAA,CAAK,aAAa,IAAA,CAAK,iBAAA,CAAkB,UAAU,CAAC,CAAC,CAAC,CAAA,CAAE,CAAA;AAC/D,QAAA;AAAA,MACF;AAGA,MAAA,MAAM,YAAA,GAAe,OAAA,CAAQ,KAAA,CAAM,yBAAyB,CAAA;AAC5D,MAAA,IAAI,YAAA,EAAc;AAChB,QAAA,MAAA,CAAO,IAAA;AAAA,UACL,CAAA,EAAG,YAAA,CAAa,CAAC,CAAC,CAAA,CAAA,EAAI,KAAK,iBAAA,CAAkB,YAAA,CAAa,CAAC,CAAC,CAAC,CAAA;AAAA,SAC/D;AACA,QAAA;AAAA,MACF;AAAA,IACF;AAEA,IAAA,OAAO,MAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKQ,kBAAkB,QAAA,EAA0B;AAElD,IAAA,MAAM,mBAAmB,QAAA,CAAS,KAAA;AAAA,MAChC;AAAA,KACF;AACA,IAAA,IAAI,gBAAA,EAAkB;AACpB,MAAA,OAAO,CAAA,MAAA,EAAS,gBAAA,CAAiB,CAAC,CAAC,CAAA,CAAA;AAAA,IACrC;AAGA,IAAA,OAAO,QAAA,CACJ,OAAA,CAAQ,aAAA,EAAe,MAAM,EAC7B,OAAA,CAAQ,cAAA,EAAgB,OAAO,CAAA,CAC/B,QAAQ,aAAA,EAAe,MAAM,CAAA,CAC7B,OAAA,CAAQ,cAAc,EAAE,CAAA;AAAA,EAC7B;AAAA;AAAA;AAAA;AAAA,EAKQ,iBAAiB,OAAA,EAAyB;AAChD,IAAA,OACE,OAAA,CAEG,UAAA;AAAA,MACC,gEAAA;AAAA,MACA;AAAA,MAGD,UAAA,CAAW,qBAAA,EAAuB,MAAM,CAAA,CAExC,UAAA,CAAW,YAAY,KAAK,CAAA,CAE5B,WAAW,UAAA,EAAY,SAAS,EAChC,UAAA,CAAW,UAAA,EAAY,SAAS,CAAA,CAEhC,KAAA,CAAM,GAAG,GAAG,CAAA;AAAA,EAEnB;AAAA;AAAA;AAAA;AAAA,EAKQ,oBAAoB,UAAA,EAAyC;AACnE,IAAA,IAAI,CAAC,YAAY,OAAO,MAAA;AAExB,IAAA,MAAM,QAAQ,UAAA,CAAW,KAAA,CAAM,IAAI,CAAA,CAAE,KAAA,CAAM,GAAG,EAAE,CAAA;AAChD,IAAA,OAAO,KAAA,CAAM,KAAK,IAAI,CAAA;AAAA,EACxB;AAAA;AAAA;AAAA;AAAA,EAKQ,WAAW,GAAA,EAAqB;AACtC,IAAA,IAAI,IAAA,GAAO,CAAA;AACX,IAAA,KAAA,IAAS,CAAA,GAAI,CAAA,EAAG,CAAA,GAAI,GAAA,CAAI,QAAQ,CAAA,EAAA,EAAK;AACnC,MAAA,MAAM,IAAA,GAAO,GAAA,CAAI,UAAA,CAAW,CAAC,CAAA;AAC7B,MAAA,IAAA,GAAA,CAAQ,IAAA,IAAQ,KAAK,IAAA,GAAO,IAAA;AAC5B,MAAA,IAAA,GAAO,IAAA,GAAO,IAAA;AAAA,IAChB;AACA,IAAA,OAAO,IAAA,CAAK,IAAI,IAAI,CAAA,CAAE,SAAS,EAAE,CAAA,CAAE,QAAA,CAAS,CAAA,EAAG,GAAG,CAAA;AAAA,EACpD;AAAA;AAAA;AAAA;AAAA,EAKQ,gBAAA,GAAyB;AAC/B,IAAA,IAAI,MAAA,GAA2D,IAAA;AAE/D,IAAA,KAAA,MAAW,CAAC,WAAA,EAAa,KAAK,CAAA,IAAK,KAAK,WAAA,EAAa;AACnD,MAAA,IAAI,CAAC,MAAA,IAAU,KAAA,CAAM,QAAA,GAAW,OAAO,QAAA,EAAU;AAC/C,QAAA,MAAA,GAAS,EAAE,WAAA,EAAa,QAAA,EAAU,KAAA,CAAM,QAAA,EAAS;AAAA,MACnD;AAAA,IACF;AAEA,IAAA,IAAI,MAAA,EAAQ;AACV,MAAA,IAAA,CAAK,WAAA,CAAY,MAAA,CAAO,MAAA,CAAO,WAAW,CAAA;AAAA,IAC5C;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,cAAA,GAA+B;AAC7B,IAAA,OAAO,CAAC,GAAG,IAAA,CAAK,WAAA,CAAY,MAAA,EAAQ,CAAA,CAAE,IAAA;AAAA,MACpC,CAAC,CAAA,EAAG,CAAA,KAAM,CAAA,CAAE,WAAW,CAAA,CAAE;AAAA,KAC3B;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,yBAAA,GAA0C;AACxC,IAAA,OAAO,CAAC,GAAG,IAAA,CAAK,WAAA,CAAY,MAAA,EAAQ,CAAA,CAAE,IAAA;AAAA,MACpC,CAAC,CAAA,EAAG,CAAA,KAAM,CAAA,CAAE,QAAQ,CAAA,CAAE;AAAA,KACxB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,cAAc,WAAA,EAA6C;AACzD,IAAA,OAAO,IAAA,CAAK,WAAA,CAAY,GAAA,CAAI,WAAW,CAAA;AAAA,EACzC;AAAA;AAAA;AAAA;AAAA,EAKA,wBAAwB,OAAA,EAA+B;AACrD,IAAA,OAAO,IAAA,CAAK,gBAAe,CAAE,MAAA,CAAO,CAAC,CAAA,KAAM,CAAA,CAAE,YAAY,OAAO,CAAA;AAAA,EAClE;AAAA;AAAA;AAAA;AAAA,EAKA,kBAAA,GAA6B;AAC3B,IAAA,IAAI,KAAA,GAAQ,CAAA;AACZ,IAAA,KAAA,MAAW,KAAA,IAAS,IAAA,CAAK,WAAA,CAAY,MAAA,EAAO,EAAG;AAC7C,MAAA,KAAA,IAAS,KAAA,CAAM,KAAA;AAAA,IACjB;AACA,IAAA,OAAO,KAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,QAAA,GAKE;AACA,IAAA,MAAM,GAAA,GAAM,KAAK,GAAA,EAAI;AACrB,IAAA,MAAM,UAAA,GAAa,GAAA,GAAM,EAAA,GAAK,EAAA,GAAK,GAAA;AAEnC,IAAA,IAAI,YAAA,GAAe,CAAA;AACnB,IAAA,MAAM,SAAA,uBAAgB,GAAA,EAAoB;AAE1C,IAAA,KAAA,MAAW,KAAA,IAAS,IAAA,CAAK,WAAA,CAAY,MAAA,EAAO,EAAG;AAC7C,MAAA,IAAI,KAAA,CAAM,WAAW,UAAA,EAAY;AAC/B,QAAA,YAAA,IAAgB,KAAA,CAAM,KAAA;AAAA,MACxB;AACA,MAAA,SAAA,CAAU,GAAA,CAAI,KAAA,CAAM,IAAA,EAAA,CAAO,SAAA,CAAU,GAAA,CAAI,MAAM,IAAI,CAAA,IAAK,CAAA,IAAK,KAAA,CAAM,KAAK,CAAA;AAAA,IAC1E;AAEA,IAAA,MAAM,aAAA,GAAgB,CAAC,GAAG,SAAA,CAAU,OAAA,EAAS,CAAA,CAC1C,GAAA,CAAI,CAAC,CAAC,IAAA,EAAM,KAAK,CAAA,MAAO,EAAE,IAAA,EAAM,KAAA,EAAM,CAAE,CAAA,CACxC,IAAA,CAAK,CAAC,CAAA,EAAG,CAAA,KAAM,CAAA,CAAE,KAAA,GAAQ,CAAA,CAAE,KAAK,CAAA,CAChC,KAAA,CAAM,GAAG,CAAC,CAAA;AAEb,IAAA,OAAO;AAAA,MACL,WAAA,EAAa,KAAK,WAAA,CAAY,IAAA;AAAA,MAC9B,WAAA,EAAa,KAAK,kBAAA,EAAmB;AAAA,MACrC,YAAA;AAAA,MACA;AAAA,KACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,KAAA,GAAc;AACZ,IAAA,IAAA,CAAK,YAAY,KAAA,EAAM;AAAA,EACzB;AAAA;AAAA;AAAA;AAAA,EAKA,eAAe,QAAA,EAA0B;AACvC,IAAA,MAAM,MAAA,GAAS,IAAA,CAAK,GAAA,EAAI,GAAI,QAAA;AAC5B,IAAA,IAAI,OAAA,GAAU,CAAA;AAEd,IAAA,KAAA,MAAW,CAAC,WAAA,EAAa,KAAK,CAAA,IAAK,KAAK,WAAA,EAAa;AACnD,MAAA,IAAI,KAAA,CAAM,WAAW,MAAA,EAAQ;AAC3B,QAAA,IAAA,CAAK,WAAA,CAAY,OAAO,WAAW,CAAA;AACnC,QAAA,OAAA,EAAA;AAAA,MACF;AAAA,IACF;AAEA,IAAA,OAAO,OAAA;AAAA,EACT;AACF,CAAA;;;AC7cA,IAAM,YAAA,GAAe,GAAA;AAErB,SAAS,WAAW,KAAA,EAA+C;AACjE,EAAA,IAAI,CAAC,OAAO,OAAO,MAAA;AACnB,EAAA,MAAM,MAAA,GAAS,MAAA,CAAO,QAAA,CAAS,KAAA,EAAO,EAAE,CAAA;AACxC,EAAA,OAAO,OAAO,QAAA,CAAS,MAAM,CAAA,IAAK,MAAA,GAAS,IAAI,MAAA,GAAS,MAAA;AAC1D;AAEO,SAAS,sBAAA,CACd,IAAA,GAAmC,EAAC,EACnB;AACjB,EAAA,MAAM,GAAA,GAAM,IAAA,CAAK,GAAA,IAAO,OAAA,CAAQ,GAAA;AAChC,EAAA,MAAM,QAAA,GAAW,KAAK,UAAA,IAAc,YAAA;AAEpC,EAAA,OAAO;AAAA,IACL,eACE,IAAA,CAAK,aAAA,IACL,UAAA,CAAW,GAAA,CAAI,uBAAuB,CAAA,IACtC,QAAA;AAAA,IACF,aACE,IAAA,CAAK,WAAA,IACL,UAAA,CAAW,GAAA,CAAI,qBAAqB,CAAA,IACpC,QAAA;AAAA,IACF,gBACE,IAAA,CAAK,cAAA,IACL,UAAA,CAAW,GAAA,CAAI,wBAAwB,CAAA,IACvC;AAAA,GACJ;AACF;AAEO,SAAS,eAAA,CAAmB,KAAA,EAAY,IAAA,EAAS,KAAA,EAAoB;AAC1E,EAAA,IAAI,KAAA,IAAS,CAAA,EAAG,OAAO,EAAC;AACxB,EAAA,MAAM,IAAA,GAAO,CAAC,GAAG,KAAA,EAAO,IAAI,CAAA;AAC5B,EAAA,OAAO,IAAA,CAAK,SAAS,KAAA,GAAQ,IAAA,CAAK,MAAM,IAAA,CAAK,MAAA,GAAS,KAAK,CAAA,GAAI,IAAA;AACjE;AAEO,SAAS,mBAAA,CACd,KAAA,EACA,QAAA,EACA,KAAA,EACK;AACL,EAAA,IAAI,KAAA,IAAS,KAAK,QAAA,CAAS,MAAA,KAAW,GAAG,OAAO,KAAA,IAAS,CAAA,GAAI,EAAC,GAAI,KAAA;AAClE,EAAA,MAAM,IAAA,GAAO,CAAC,GAAG,KAAA,EAAO,GAAG,QAAQ,CAAA;AACnC,EAAA,OAAO,IAAA,CAAK,SAAS,KAAA,GAAQ,IAAA,CAAK,MAAM,IAAA,CAAK,MAAA,GAAS,KAAK,CAAA,GAAI,IAAA;AACjE;;;AC/BO,IAAM,iBAAN,MAAqB;AAAA,EAClB,GAAA;AAAA,EACA,OAAA,uBAAc,GAAA,EAAe;AAAA,EAC7B,UAAA;AAAA,EACA,SAAsB,EAAC;AAAA,EACvB,OAAkB,EAAC;AAAA,EACnB,UAAwB,EAAC;AAAA,EACzB,eAAA,GAAkB,IAAI,eAAA,EAAgB;AAAA,EACtC,MAAA;AAAA,EACA,OAAA;AAAA,EACA,KAAA;AAAA,EAER,WAAA,CAAY,OAAA,GAAiC,EAAC,EAAG;AAC/C,IAAA,IAAA,CAAK,MAAA,GAAS,uBAAuB,OAAO,CAAA;AAC5C,IAAA,IAAA,CAAK,OAAA,GAAU,QAAQ,OAAA,IAAW,KAAA;AAClC,IAAA,IAAA,CAAK,KAAA,GAAQ,QAAQ,IAAA,IAAQ,IAAA;AAE7B,IAAA,IAAA,CAAK,UAAA,GAAa,OAAA,CAAQ,MAAA,IAAUA,iBAAA,EAAa;AACjD,IAAA,IAAA,CAAK,GAAA,GAAM,IAAIC,kBAAA,CAAgB,EAAE,MAAA,EAAQ,IAAA,CAAK,UAAA,EAAY,IAAA,EAAM,OAAA,CAAQ,IAAA,IAAQ,KAAA,EAAO,CAAA;AAEvF,IAAA,IAAA,CAAK,GAAA,CAAI,EAAA,CAAG,YAAA,EAAc,CAAC,EAAA,KAAO;AAChC,MAAA,IAAA,CAAK,OAAA,CAAQ,IAAI,EAAE,CAAA;AACnB,MAAA,IAAA,CAAK,GAAA,CAAI,CAAA,kBAAA,EAAqB,IAAA,CAAK,OAAA,CAAQ,IAAI,CAAA,OAAA,CAAS,CAAA;AAGxD,MAAA,MAAM,IAAA,GAAO,KAAK,cAAA,EAAe;AACjC,MAAA,IAAI,IAAA,CAAK,MAAA,CAAO,MAAA,GAAS,CAAA,IAAK,IAAA,CAAK,IAAA,CAAK,MAAA,GAAS,CAAA,IAAK,IAAA,CAAK,MAAA,CAAO,MAAA,GAAS,CAAA,EAAG;AAC5E,QAAA,EAAA,CAAG,IAAA,CAAK,IAAA,CAAK,SAAA,CAAU,IAAI,CAAC,CAAA;AAAA,MAC9B;AAEA,MAAA,EAAA,CAAG,EAAA,CAAG,SAAS,MAAM;AACnB,QAAA,IAAA,CAAK,OAAA,CAAQ,OAAO,EAAE,CAAA;AACtB,QAAA,IAAA,CAAK,GAAA,CAAI,CAAA,qBAAA,EAAwB,IAAA,CAAK,OAAA,CAAQ,IAAI,CAAA,OAAA,CAAS,CAAA;AAAA,MAC7D,CAAC,CAAA;AAAA,IACH,CAAC,CAAA;AAGD,IAAA,IAAI,CAAC,QAAQ,MAAA,EAAQ;AACnB,MAAA,IAAA,CAAK,UAAA,CAAW,MAAA,CAAO,IAAA,CAAK,KAAA,EAAO,MAAM;AACvC,QAAA,MAAM,IAAA,GAAO,IAAA,CAAK,UAAA,CAAW,OAAA,EAAQ;AACrC,QAAA,IAAI,QAAQ,OAAO,IAAA,KAAS,QAAA,EAAU,IAAA,CAAK,QAAQ,IAAA,CAAK,IAAA;AACxD,QAAA,IAAA,CAAK,GAAA,CAAI,CAAA,mCAAA,EAAsC,IAAA,CAAK,KAAK,CAAA,CAAE,CAAA;AAAA,MAC7D,CAAC,CAAA;AAAA,IACH;AAAA,EACF;AAAA,EAEA,IAAI,IAAA,GAAe;AACjB,IAAA,MAAM,IAAA,GAAO,IAAA,CAAK,UAAA,CAAW,OAAA,EAAQ;AACrC,IAAA,IAAI,IAAA,IAAQ,OAAO,IAAA,KAAS,QAAA,SAAiB,IAAA,CAAK,IAAA;AAClD,IAAA,OAAO,IAAA,CAAK,KAAA;AAAA,EACd;AAAA,EAEA,IAAI,WAAA,GAAsB;AACxB,IAAA,OAAO,KAAK,OAAA,CAAQ,IAAA;AAAA,EACtB;AAAA,EAEA,SAAS,KAAA,EAAwB;AAE/B,IAAA,MAAM,QAAA,GAAW,KAAK,MAAA,CAAO,IAAA,CAAK,OAAK,CAAA,CAAE,OAAA,KAAY,MAAM,OAAO,CAAA;AAClE,IAAA,IAAI,QAAA,EAAU;AACZ,MAAA,MAAM,eAAA,GAAkB,IAAI,GAAA,CAAI,QAAA,CAAS,MAAM,GAAA,CAAI,CAAA,CAAA,KAAK,CAAA,CAAE,MAAM,CAAC,CAAA;AACjE,MAAA,KAAA,MAAW,IAAA,IAAQ,MAAM,KAAA,EAAO;AAC9B,QAAA,IAAI,CAAC,eAAA,CAAgB,GAAA,CAAI,IAAA,CAAK,MAAM,CAAA,EAAG;AACrC,UAAA,QAAA,CAAS,KAAA,CAAM,KAAK,IAAI,CAAA;AAAA,QAC1B;AAAA,MACF;AACA,MAAA,QAAA,CAAS,YAAY,IAAA,CAAK,GAAA,CAAI,QAAA,CAAS,SAAA,EAAW,MAAM,SAAS,CAAA;AACjE,MAAA,QAAA,CAAS,UAAU,IAAA,CAAK,GAAA,CAAI,QAAA,CAAS,OAAA,EAAS,MAAM,OAAO,CAAA;AAC3D,MAAA,QAAA,CAAS,QAAA,GAAW,QAAA,CAAS,OAAA,GAAU,QAAA,CAAS,SAAA;AAChD,MAAA,IAAI,KAAA,CAAM,MAAA,KAAW,OAAA,EAAS,QAAA,CAAS,MAAA,GAAS,OAAA;AAAA,IAClD,CAAA,MAAO;AACL,MAAA,IAAA,CAAK,MAAA,GAAS,eAAA;AAAA,QACZ,IAAA,CAAK,MAAA;AAAA,QACL,KAAA;AAAA,QACA,KAAK,MAAA,CAAO;AAAA,OACd;AAAA,IACF;AAEA,IAAA,IAAA,CAAK,eAAA,CAAgB,mBAAmB,KAAK,CAAA;AAC7C,IAAA,IAAA,CAAK,UAAU,EAAE,MAAA,EAAQ,CAAC,KAAK,GAAG,OAAA,EAAS,EAAC,EAAG,IAAA,EAAM,EAAC,EAAG,MAAA,EAAQ,KAAK,eAAA,CAAgB,cAAA,IAAkB,CAAA;AAAA,EAC1G;AAAA,EAEA,UAAU,MAAA,EAA2B;AACnC,IAAA,KAAA,MAAW,KAAA,IAAS,MAAA,EAAQ,IAAA,CAAK,QAAA,CAAS,KAAK,CAAA;AAAA,EACjD;AAAA,EAEA,OAAO,GAAA,EAAoB;AACzB,IAAA,IAAA,CAAK,OAAO,eAAA,CAAgB,IAAA,CAAK,MAAM,GAAA,EAAK,IAAA,CAAK,OAAO,WAAW,CAAA;AACnE,IAAA,IAAA,CAAK,SAAA,CAAU,EAAE,MAAA,EAAQ,IAAI,OAAA,EAAS,EAAC,EAAG,IAAA,EAAM,CAAC,GAAG,CAAA,EAAG,MAAA,EAAQ,IAAI,CAAA;AAAA,EACrE;AAAA,EAEA,QAAQ,IAAA,EAAuB;AAC7B,IAAA,IAAA,CAAK,OAAO,mBAAA,CAAoB,IAAA,CAAK,MAAM,IAAA,EAAM,IAAA,CAAK,OAAO,WAAW,CAAA;AACxE,IAAA,IAAA,CAAK,SAAA,CAAU,EAAE,MAAA,EAAQ,EAAC,EAAG,OAAA,EAAS,EAAC,EAAG,IAAA,EAAM,MAAA,EAAQ,EAAC,EAAG,CAAA;AAAA,EAC9D;AAAA,EAEA,UAAU,MAAA,EAA0B;AAClC,IAAA,IAAA,CAAK,OAAA,GAAU,eAAA;AAAA,MACb,IAAA,CAAK,OAAA;AAAA,MACL,MAAA;AAAA,MACA,KAAK,MAAA,CAAO;AAAA,KACd;AACA,IAAA,IAAA,CAAK,SAAA,CAAU,EAAE,MAAA,EAAQ,IAAI,OAAA,EAAS,CAAC,MAAM,CAAA,EAAG,MAAM,EAAC,EAAG,MAAA,EAAQ,IAAI,CAAA;AAAA,EACxE;AAAA,EAEA,cAAA,GAA+B;AAC7B,IAAA,OAAO;AAAA,MACL,QAAQ,IAAA,CAAK,MAAA;AAAA,MACb,SAAS,IAAA,CAAK,OAAA;AAAA,MACd,MAAM,IAAA,CAAK,IAAA;AAAA,MACX,MAAA,EAAQ,IAAA,CAAK,eAAA,CAAgB,cAAA;AAAe,KAC9C;AAAA,EACF;AAAA,EAEA,SAAA,GAAkB;AAChB,IAAA,IAAA,CAAK,SAAS,EAAC;AACf,IAAA,IAAA,CAAK,OAAO,EAAC;AACb,IAAA,IAAA,CAAK,UAAU,EAAC;AAChB,IAAA,IAAA,CAAK,gBAAgB,KAAA,EAAM;AAAA,EAC7B;AAAA,EAEQ,UAAU,IAAA,EAA0B;AAC1C,IAAA,MAAM,GAAA,GAAM,IAAA,CAAK,SAAA,CAAU,IAAI,CAAA;AAC/B,IAAA,KAAA,MAAW,MAAA,IAAU,KAAK,OAAA,EAAS;AACjC,MAAA,IAAI,MAAA,CAAO,UAAA,KAAeC,YAAA,CAAU,IAAA,EAAM;AACxC,QAAA,MAAA,CAAO,KAAK,GAAG,CAAA;AAAA,MACjB;AAAA,IACF;AAAA,EACF;AAAA,EAEQ,IAAI,OAAA,EAAuB;AACjC,IAAA,IAAI,KAAK,OAAA,EAAS,OAAA,CAAQ,GAAA,CAAI,CAAA,mBAAA,EAAsB,OAAO,CAAA,CAAE,CAAA;AAAA,EAC/D;AAAA,EAEA,MAAM,KAAA,GAAuB;AAC3B,IAAA,KAAA,MAAW,MAAA,IAAU,IAAA,CAAK,OAAA,EAAS,MAAA,CAAO,KAAA,EAAM;AAChD,IAAA,IAAA,CAAK,QAAQ,KAAA,EAAM;AACnB,IAAA,IAAA,CAAK,IAAI,KAAA,EAAM;AACf,IAAA,MAAM,IAAI,OAAA,CAAc,CAACC,QAAAA,KAAY,IAAA,CAAK,WAAW,KAAA,CAAM,MAAMA,QAAAA,EAAS,CAAC,CAAA;AAAA,EAC7E;AACF,CAAA;;;ACzKO,SAAS,eAAA,CACd,QAAA,EACA,QAAA,GAAW,SAAA,EACH;AACR,EAAA,IAAI,CAAC,UAAU,OAAO,QAAA;AAEtB,EAAA,MAAM,UAAA,GAAa;AAAA,IACjB,SAAS,cAAc,CAAA;AAAA,IACvB,SAAS,mBAAmB,CAAA;AAAA,IAC5B,SAAS,6BAA6B,CAAA;AAAA,IACtC,SAAS,WAAW,CAAA;AAAA,IACpB,SAAS,gBAAgB,CAAA;AAAA,IACzB,SAAS,yBAAyB;AAAA,GACpC;AAEA,EAAA,KAAA,MAAW,aAAa,UAAA,EAAY;AAClC,IAAA,IAAI,OAAO,SAAA,KAAc,QAAA,IAAY,UAAU,IAAA,EAAK,CAAE,SAAS,CAAA,EAAG;AAChE,MAAA,OAAO,SAAA;AAAA,IACT;AAAA,EACF;AAEA,EAAA,OAAO,QAAA;AACT;;;ACLA,SAAS,iBAAiB,CAAA,EAA2B;AACnD,EAAA,IAAI,CAAC,GAAG,OAAO,MAAA;AACf,EAAA,IAAI,CAAA,CAAE,WAAA,KAAgB,MAAA,EAAW,OAAO,CAAA,CAAE,WAAA;AAC1C,EAAA,IAAI,CAAA,CAAE,SAAA,KAAc,MAAA,EAAW,OAAO,CAAA,CAAE,SAAA;AACxC,EAAA,IAAI,CAAA,CAAE,QAAA,KAAa,MAAA,EAAW,OAAO,OAAO,CAAA,CAAE,QAAA,KAAa,QAAA,GAAW,MAAA,CAAO,CAAA,CAAE,QAAQ,CAAA,GAAI,CAAA,CAAE,QAAA;AAC7F,EAAA,IAAI,CAAA,CAAE,WAAA,KAAgB,MAAA,EAAW,OAAO,CAAA,CAAE,WAAA;AAC1C,EAAA,IAAI,CAAA,CAAE,UAAA,KAAe,MAAA,EAAW,OAAO,CAAA,CAAE,UAAA;AACzC,EAAA,IAAI,CAAA,CAAE,YAAY,MAAA,EAAQ,OAAO,EAAE,UAAA,CAAW,MAAA,CAAO,IAAI,gBAAgB,CAAA;AACzE,EAAA,IAAI,EAAE,WAAA,EAAa,MAAA,SAAe,iBAAA,CAAkB,CAAA,CAAE,YAAY,MAAM,CAAA;AACxE,EAAA,OAAO,MAAA;AACT;AAEA,SAAS,kBAAkB,KAAA,EAAiD;AAC1E,EAAA,MAAM,MAA+B,EAAC;AACtC,EAAA,IAAI,CAAC,OAAO,OAAO,GAAA;AACnB,EAAA,KAAA,MAAW,EAAE,GAAA,EAAK,KAAA,EAAM,IAAK,KAAA,EAAO;AAClC,IAAA,GAAA,CAAI,GAAG,CAAA,GAAI,gBAAA,CAAiB,KAAK,CAAA;AAAA,EACnC;AACA,EAAA,OAAO,GAAA;AACT;AAEA,SAAS,SAAS,IAAA,EAAuB;AACvC,EAAA,IAAI,CAAC,MAAM,OAAO,CAAA;AAClB,EAAA,OAAO,MAAA,CAAO,MAAA,CAAO,IAAI,CAAA,GAAI,QAAU,CAAA;AACzC;AAEA,IAAM,aAAA,GAA2D;AAAA,EAC/D,CAAA,EAAG,UAAA;AAAA,EAAY,CAAA,EAAG,UAAA;AAAA,EAAY,CAAA,EAAG,QAAA;AAAA,EAAU,CAAA,EAAG,QAAA;AAAA,EAAU,CAAA,EAAG,UAAA;AAAA,EAAY,CAAA,EAAG,UAAA;AAAA,EAC1E,kBAAA,EAAoB,UAAA;AAAA,EAAY,gBAAA,EAAkB,QAAA;AAAA,EAClD,gBAAA,EAAkB,QAAA;AAAA,EAAU,kBAAA,EAAoB,UAAA;AAAA,EAAY,kBAAA,EAAoB;AAClF,CAAA;AAEA,SAAS,eAAe,EAAA,EAAqB;AAC3C,EAAA,IAAI,CAAC,IAAI,OAAO,EAAA;AAGhB,EAAA,MAAM,YAAA,GAAe,oBAAoB,IAAA,CAAK,EAAE,KAAK,CAAE,cAAA,CAAe,KAAK,EAAE,CAAA;AAC7E,EAAA,MAAM,gBAAA,GAAmB,YAAA,KAAiB,EAAA,CAAG,MAAA,KAAW,EAAA,IAAM,EAAA,CAAG,MAAA,KAAW,EAAA,IAAM,EAAA,CAAG,MAAA,KAAW,EAAA,IAAM,EAAA,CAAG,MAAA,KAAW,EAAA,CAAA;AACpH,EAAA,IAAI,gBAAA,EAAkB;AACpB,IAAA,IAAI;AACF,MAAA,MAAM,KAAA,GAAQ,MAAA,CAAO,IAAA,CAAK,EAAA,EAAI,QAAQ,CAAA;AACtC,MAAA,OAAO,KAAA,CAAM,SAAS,KAAK,CAAA;AAAA,IAC7B,CAAA,CAAA,MAAQ;AAAA,IAAqB;AAAA,EAC/B;AACA,EAAA,OAAO,EAAA;AACT;AAEO,SAAS,gBAAgB,OAAA,EAA+B;AAC7D,EAAA,IAAI,CAAC,OAAA,IAAW,OAAO,OAAA,KAAY,QAAA,SAAiB,EAAC;AACrD,EAAA,MAAM,EAAE,eAAc,GAAI,OAAA;AAC1B,EAAA,IAAI,CAAC,MAAM,OAAA,CAAQ,aAAa,KAAK,aAAA,CAAc,MAAA,KAAW,CAAA,EAAG,OAAO,EAAC;AAEzE,EAAA,MAAM,QAAA,uBAAe,GAAA,EAAoD;AAEzE,EAAA,KAAA,MAAW,MAAM,aAAA,EAAe;AAC9B,IAAA,MAAM,aAAA,GAAgB,iBAAA,CAAkB,EAAA,CAAG,QAAA,EAAU,UAAU,CAAA;AAC/D,IAAA,MAAM,OAAA,GAAU,MAAA,CAAO,aAAA,CAAc,cAAc,KAAK,SAAS,CAAA;AACjE,IAAA,MAAM,UAAA,GAAa,EAAA,CAAG,UAAA,IAAc,EAAC;AAErC,IAAA,KAAA,MAAW,MAAM,UAAA,EAAY;AAC3B,MAAA,KAAA,MAAW,IAAA,IAAQ,EAAA,CAAG,KAAA,IAAS,EAAC,EAAG;AACjC,QAAA,MAAM,OAAA,GAAU,cAAA,CAAe,IAAA,CAAK,OAAO,CAAA;AAC3C,QAAA,IAAI,CAAC,OAAA,EAAS;AAEd,QAAA,MAAM,OAAA,GAAU,QAAA,CAAS,IAAA,CAAK,iBAAiB,CAAA;AAC/C,QAAA,MAAM,KAAA,GAAQ,QAAA,CAAS,IAAA,CAAK,eAAe,CAAA;AAC3C,QAAA,MAAM,UAAA,GAAa,KAAK,MAAA,EAAQ,IAAA;AAChC,QAAA,IAAI,MAAA,GAAqC,OAAA;AACzC,QAAA,IAAI,UAAA,KAAe,CAAA,IAAK,UAAA,KAAe,gBAAA,EAAkB,MAAA,GAAS,IAAA;AAClE,QAAA,IAAI,UAAA,KAAe,CAAA,IAAK,UAAA,KAAe,mBAAA,EAAqB,MAAA,GAAS,OAAA;AAErE,QAAA,MAAM,QAAA,GAAqB;AAAA,UACzB,OAAA;AAAA,UACA,MAAA,EAAQ,cAAA,CAAe,IAAA,CAAK,MAAM,CAAA;AAAA,UAClC,YAAA,EAAc,cAAA,CAAe,IAAA,CAAK,YAAY,CAAA,IAAK,MAAA;AAAA,UACnD,IAAA,EAAM,KAAK,IAAA,IAAQ,SAAA;AAAA,UACnB,IAAA,EAAM,aAAA,CAAc,IAAA,CAAK,IAAA,IAAQ,CAAC,CAAA,IAAK,UAAA;AAAA,UACvC,SAAA,EAAW,OAAA;AAAA,UACX,OAAA,EAAS,KAAA;AAAA,UACT,UAAU,KAAA,GAAQ,OAAA;AAAA,UAClB,UAAA,EAAY,EAAE,GAAG,aAAA,EAAe,GAAG,iBAAA,CAAkB,IAAA,CAAK,UAAU,CAAA,EAAE;AAAA,UACtE,QAAQ,EAAE,IAAA,EAAM,QAAQ,OAAA,EAAS,IAAA,CAAK,QAAQ,OAAA,EAAQ;AAAA,UACtD,SAAS,IAAA,CAAK,MAAA,IAAU,EAAC,EAAG,GAAA,CAAI,CAAC,CAAA,MAAY;AAAA,YAC3C,IAAA,EAAM,EAAE,IAAA,IAAQ,EAAA;AAAA,YAChB,SAAA,EAAW,QAAA,CAAS,CAAA,CAAE,YAAY,CAAA;AAAA,YAClC,UAAA,EAAY,iBAAA,CAAkB,CAAA,CAAE,UAAU;AAAA,WAC5C,CAAE;AAAA,SACJ;AAEA,QAAA,MAAM,QAAA,GAAW,QAAA,CAAS,GAAA,CAAI,OAAO,CAAA;AACrC,QAAA,IAAI,QAAA,EAAU;AACZ,UAAA,QAAA,CAAS,KAAA,CAAM,KAAK,QAAQ,CAAA;AAAA,QAC9B,CAAA,MAAO;AACL,UAAA,QAAA,CAAS,GAAA,CAAI,SAAS,EAAE,KAAA,EAAO,CAAC,QAAQ,CAAA,EAAG,SAAS,CAAA;AAAA,QACtD;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAEA,EAAA,MAAM,SAAsB,EAAC;AAC7B,EAAA,KAAA,MAAW,CAAC,OAAA,EAAS,EAAE,OAAO,OAAA,EAAS,KAAK,QAAA,EAAU;AACpD,IAAA,MAAM,MAAA,GAAS,MAAM,IAAA,CAAK,CAAC,GAAG,CAAA,KAAM,CAAA,CAAE,SAAA,GAAY,CAAA,CAAE,SAAS,CAAA;AAC7D,IAAA,MAAM,QAAA,GAAW,OAAO,IAAA,CAAK,CAAA,CAAA,KAAK,CAAC,CAAA,CAAE,YAAY,CAAA,IAAK,MAAA,CAAO,CAAC,CAAA;AAC9D,IAAA,MAAM,SAAA,GAAY,KAAK,GAAA,CAAI,GAAG,OAAO,GAAA,CAAI,CAAA,CAAA,KAAK,CAAA,CAAE,SAAS,CAAC,CAAA;AAC1D,IAAA,MAAM,OAAA,GAAU,KAAK,GAAA,CAAI,GAAG,OAAO,GAAA,CAAI,CAAA,CAAA,KAAK,CAAA,CAAE,OAAO,CAAC,CAAA;AACtD,IAAA,MAAM,WAAW,MAAA,CAAO,IAAA,CAAK,OAAK,CAAA,CAAE,MAAA,CAAO,SAAS,OAAO,CAAA;AAE3D,IAAA,MAAA,CAAO,IAAA,CAAK;AAAA,MACV,OAAA;AAAA,MACA,aAAA,EAAe,OAAA,CAAQ,KAAA,CAAM,CAAA,EAAG,EAAE,CAAA;AAAA,MAClC,QAAA;AAAA,MACA,KAAA,EAAO,MAAA;AAAA,MACP,SAAA;AAAA,MACA,OAAA;AAAA,MACA,UAAU,OAAA,GAAU,SAAA;AAAA,MACpB,MAAA,EAAQ,WAAW,OAAA,GAAU,IAAA;AAAA,MAC7B;AAAA,KACD,CAAA;AAAA,EACH;AAEA,EAAA,OAAO,MAAA;AACT;AAEO,SAAS,cAAc,OAAA,EAA6B;AACzD,EAAA,IAAI,CAAC,OAAA,IAAW,OAAO,OAAA,KAAY,QAAA,SAAiB,EAAC;AACrD,EAAA,MAAM,EAAE,cAAa,GAAI,OAAA;AACzB,EAAA,IAAI,CAAC,KAAA,CAAM,OAAA,CAAQ,YAAY,CAAA,SAAU,EAAC;AAE1C,EAAA,MAAM,OAAkB,EAAC;AACzB,EAAA,KAAA,MAAW,MAAM,YAAA,EAAc;AAC7B,IAAA,MAAM,aAAA,GAAgB,iBAAA,CAAkB,EAAA,CAAG,QAAA,EAAU,UAAU,CAAA;AAC/D,IAAA,KAAA,MAAW,EAAA,IAAM,EAAA,CAAG,SAAA,IAAa,EAAC,EAAG;AACnC,MAAA,KAAA,MAAW,GAAA,IAAO,EAAA,CAAG,UAAA,IAAc,EAAC,EAAG;AACrC,QAAA,MAAM,SAAA,GAAY,QAAA,CAAS,GAAA,CAAI,YAAA,IAAgB,IAAI,oBAAoB,CAAA;AACvE,QAAA,MAAM,OAAA,GAAU,cAAA,CAAe,GAAA,CAAI,OAAO,CAAA,IAAK,MAAA;AAC/C,QAAA,MAAM,MAAA,GAAS,cAAA,CAAe,GAAA,CAAI,MAAM,CAAA,IAAK,MAAA;AAC7C,QAAA,MAAM,OAAO,GAAA,CAAI,IAAA,GAAO,gBAAA,CAAiB,GAAA,CAAI,IAAI,CAAA,GAAI,EAAA;AAErD,QAAA,IAAA,CAAK,IAAA,CAAK;AAAA,UACR,EAAA,EAAI,CAAA,EAAG,OAAA,IAAW,UAAU,CAAA,CAAA,EAAI,MAAA,IAAU,SAAS,CAAA,CAAA,EAAI,SAAS,CAAA,CAAA,EAAI,GAAA,CAAI,cAAA,IAAkB,CAAC,CAAA,CAAA;AAAA,UAC3F,OAAA;AAAA,UACA,MAAA;AAAA,UACA,YAAA,EAAc,gBAAgB,aAAwC,CAAA;AAAA,UACtE,cAAc,GAAA,CAAI,YAAA;AAAA,UAClB,gBAAgB,GAAA,CAAI,cAAA;AAAA,UACpB,IAAA,EAAM,OAAO,IAAA,KAAS,QAAA,GAAW,IAAA,GAAQ,IAAA;AAAA,UACzC,SAAA;AAAA,UACA,UAAA,EAAY,iBAAA,CAAkB,GAAA,CAAI,UAAU,CAAA;AAAA,UAC5C,QAAA,EAAU;AAAA,SACX,CAAA;AAAA,MACH;AAAA,IACF;AAAA,EACF;AAEA,EAAA,OAAO,IAAA;AACT;AAEO,SAAS,iBAAiB,OAAA,EAA0B;AACzD,EAAA,IAAI,CAAC,OAAA,IAAW,OAAO,OAAA,KAAY,UAAU,OAAO,CAAA;AACpD,EAAA,MAAM,EAAE,iBAAgB,GAAI,OAAA;AAC5B,EAAA,IAAI,CAAC,KAAA,CAAM,OAAA,CAAQ,eAAe,GAAG,OAAO,CAAA;AAC5C,EAAA,IAAI,KAAA,GAAQ,CAAA;AACZ,EAAA,KAAA,MAAW,MAAM,eAAA,EAAiB;AAChC,IAAA,KAAA,MAAW,EAAA,IAAM,EAAA,CAAG,YAAA,IAAgB,EAAC,EAAG;AACtC,MAAA,KAAA,IAAA,CAAU,EAAA,CAAG,OAAA,IAAW,EAAC,EAAG,MAAA;AAAA,IAC9B;AAAA,EACF;AACA,EAAA,OAAO,KAAA;AACT;AAEA,eAAsB,aAAa,GAAA,EAAwC;AACzE,EAAA,OAAO,IAAI,OAAA,CAAQ,CAACA,QAAAA,EAAS,MAAA,KAAW;AACtC,IAAA,MAAM,SAAmB,EAAC;AAC1B,IAAA,GAAA,CAAI,GAAG,MAAA,EAAQ,CAAC,UAAkB,MAAA,CAAO,IAAA,CAAK,KAAK,CAAC,CAAA;AACpD,IAAA,GAAA,CAAI,EAAA,CAAG,OAAO,MAAM;AAClB,MAAA,IAAI;AACF,QAAAA,QAAAA,CAAQ,KAAK,KAAA,CAAM,MAAA,CAAO,OAAO,MAAM,CAAA,CAAE,QAAA,EAAU,CAAC,CAAA;AAAA,MACtD,CAAA,CAAA,MAAQ;AACN,QAAA,MAAA,CAAO,IAAI,KAAA,CAAM,cAAc,CAAC,CAAA;AAAA,MAClC;AAAA,IACF,CAAC,CAAA;AACD,IAAA,GAAA,CAAI,EAAA,CAAG,SAAS,MAAM,CAAA;AAAA,EACxB,CAAC,CAAA;AACH;AAEO,SAAS,QAAA,CAAS,GAAA,EAAqB,MAAA,EAAgB,IAAA,EAAqC;AACjG,EAAA,MAAM,IAAA,GAAO,IAAA,CAAK,SAAA,CAAU,IAAI,CAAA;AAChC,EAAA,GAAA,CAAI,SAAA,CAAU,MAAA,EAAQ,EAAE,cAAA,EAAgB,kBAAA,EAAoB,kBAAkB,MAAA,CAAO,UAAA,CAAW,IAAI,CAAA,EAAG,CAAA;AACvG,EAAA,GAAA,CAAI,IAAI,IAAI,CAAA;AACd;;;ACjMA,SAAS,eAAA,GAA0B;AACjC,EAAA,IAAI,GAAA,GAAMC,YAAA,CAAQC,iBAAA,CAAc,yPAAe,CAAC,CAAA;AAChD,EAAA,KAAA,IAAS,CAAA,GAAI,CAAA,EAAG,CAAA,GAAI,CAAA,EAAG,CAAA,EAAA,EAAK;AAC1B,IAAA,IAAIC,cAAWH,YAAA,CAAQ,GAAA,EAAK,cAAc,CAAC,GAAG,OAAO,GAAA;AACrD,IAAA,GAAA,GAAMC,aAAQ,GAAG,CAAA;AAAA,EACnB;AACA,EAAA,OAAO,GAAA;AACT;AAEA,IAAM,aAAA,GAAgB,CAAA,4UAAA,CAAA;AAEtB,IAAI,cAAA,GAAgC,IAAA;AACpC,SAAS,WAAA,GAAsB;AAC7B,EAAA,IAAI,CAAC,cAAA,EAAgB;AACnB,IAAA,MAAM,UAAU,eAAA,EAAgB;AAChC,IAAA,MAAM,UAAA,GAAa;AAAA,MACjBD,YAAA,CAAQ,OAAA,EAAS,MAAA,EAAQ,kBAAkB,CAAA;AAAA,MAC3CA,YAAA,CAAQ,SAAS,kBAAkB;AAAA,KACrC;AACA,IAAA,KAAA,MAAW,aAAa,UAAA,EAAY;AAClC,MAAA,IAAI;AACF,QAAA,cAAA,GAAiBI,eAAA,CAAa,WAAW,MAAM,CAAA;AAC/C,QAAA;AAAA,MACF,CAAA,CAAA,MAAQ;AAAA,MAAiB;AAAA,IAC3B;AACA,IAAA,IAAI,CAAC,cAAA,EAAgB;AACnB,MAAA,cAAA,GAAiB,mDAAA;AAAA,IACnB;AAAA,EACF;AACA,EAAA,OAAO,cAAA;AACT;AAEO,SAAS,oBAAA,CAAqB,YAAoB,QAAA,EAAgC;AACvF,EAAA,UAAA,CAAW,EAAA,CAAG,SAAA,EAAW,OAAO,GAAA,EAAsB,GAAA,KAAwB;AAE5E,IAAA,GAAA,CAAI,SAAA,CAAU,+BAA+B,GAAG,CAAA;AAChD,IAAA,GAAA,CAAI,SAAA,CAAU,gCAAgC,oBAAoB,CAAA;AAClE,IAAA,GAAA,CAAI,SAAA,CAAU,gCAAgC,cAAc,CAAA;AAE5D,IAAA,IAAI,GAAA,CAAI,WAAW,SAAA,EAAW;AAC5B,MAAA,GAAA,CAAI,UAAU,GAAG,CAAA;AACjB,MAAA,GAAA,CAAI,GAAA,EAAI;AACR,MAAA;AAAA,IACF;AAEA,IAAA,MAAM,GAAA,GAAM,IAAI,GAAA,IAAO,GAAA;AAGvB,IAAA,IAAI,GAAA,CAAI,MAAA,KAAW,KAAA,IAAS,GAAA,KAAQ,GAAA,EAAK;AACvC,MAAA,GAAA,CAAI,SAAA,CAAU,GAAA,EAAK,EAAE,cAAA,EAAgB,0BAAA,EAA4B,kBAAkB,MAAA,CAAO,UAAA,CAAW,aAAa,CAAA,EAAG,CAAA;AACrH,MAAA,GAAA,CAAI,IAAI,aAAa,CAAA;AACrB,MAAA;AAAA,IACF;AAGA,IAAA,IAAI,IAAI,MAAA,KAAW,KAAA,IAAS,GAAA,CAAI,UAAA,CAAW,YAAY,CAAA,EAAG;AACxD,MAAA,MAAM,KAAK,WAAA,EAAY;AACvB,MAAA,GAAA,CAAI,SAAA,CAAU,GAAA,EAAK,EAAE,cAAA,EAAgB,uCAAA,EAAyC,kBAAkB,MAAA,CAAO,UAAA,CAAW,EAAE,CAAA,EAAG,CAAA;AACvH,MAAA,GAAA,CAAI,IAAI,EAAE,CAAA;AACV,MAAA;AAAA,IACF;AAGA,IAAA,IAAI,GAAA,CAAI,MAAA,KAAW,KAAA,IAAS,GAAA,KAAQ,UAAA,EAAY;AAC9C,MAAA,QAAA,CAAS,GAAA,EAAK,KAAK,EAAE,EAAA,EAAI,MAAM,OAAA,EAAS,QAAA,CAAS,aAAa,CAAA;AAC9D,MAAA;AAAA,IACF;AAGA,IAAA,IAAI,GAAA,CAAI,MAAA,KAAW,MAAA,IAAU,GAAA,KAAQ,YAAA,EAAc;AACjD,MAAA,IAAI;AACF,QAAA,MAAM,OAAA,GAAU,MAAM,YAAA,CAAa,GAAG,CAAA;AACtC,QAAA,MAAM,MAAA,GAAS,gBAAgB,OAAO,CAAA;AACtC,QAAA,QAAA,CAAS,UAAU,MAAM,CAAA;AACzB,QAAA,QAAA,CAAS,KAAK,GAAA,EAAK,EAAE,cAAA,EAAgB,MAAA,CAAO,QAAQ,CAAA;AAAA,MACtD,SAAS,CAAA,EAAG;AACV,QAAA,QAAA,CAAS,GAAA,EAAK,GAAA,EAAK,EAAE,KAAA,EAAO,mBAAA,EAAqB,OAAA,EAAS,CAAA,YAAa,KAAA,GAAQ,CAAA,CAAE,OAAA,GAAU,MAAA,CAAO,CAAC,GAAG,CAAA;AAAA,MACxG;AACA,MAAA;AAAA,IACF;AAGA,IAAA,IAAI,GAAA,CAAI,MAAA,KAAW,MAAA,IAAU,GAAA,KAAQ,UAAA,EAAY;AAC/C,MAAA,IAAI;AACF,QAAA,MAAM,OAAA,GAAU,MAAM,YAAA,CAAa,GAAG,CAAA;AACtC,QAAA,MAAM,IAAA,GAAO,cAAc,OAAO,CAAA;AAClC,QAAA,QAAA,CAAS,QAAQ,IAAI,CAAA;AACrB,QAAA,QAAA,CAAS,KAAK,GAAA,EAAK,EAAE,YAAA,EAAc,IAAA,CAAK,QAAQ,CAAA;AAAA,MAClD,SAAS,CAAA,EAAG;AACV,QAAA,QAAA,CAAS,GAAA,EAAK,GAAA,EAAK,EAAE,KAAA,EAAO,mBAAA,EAAqB,OAAA,EAAS,CAAA,YAAa,KAAA,GAAQ,CAAA,CAAE,OAAA,GAAU,MAAA,CAAO,CAAC,GAAG,CAAA;AAAA,MACxG;AACA,MAAA;AAAA,IACF;AAGA,IAAA,IAAI,GAAA,CAAI,MAAA,KAAW,MAAA,IAAU,GAAA,KAAQ,aAAA,EAAe;AAClD,MAAA,IAAI;AACF,QAAA,MAAM,OAAA,GAAU,MAAM,YAAA,CAAa,GAAG,CAAA;AACtC,QAAA,MAAM,KAAA,GAAQ,iBAAiB,OAAO,CAAA;AACtC,QAAA,QAAA,CAAS,GAAA,EAAK,GAAA,EAAK,EAAE,eAAA,EAAiB,OAAO,CAAA;AAAA,MAC/C,SAAS,CAAA,EAAG;AACV,QAAA,QAAA,CAAS,GAAA,EAAK,GAAA,EAAK,EAAE,KAAA,EAAO,mBAAA,EAAqB,OAAA,EAAS,CAAA,YAAa,KAAA,GAAQ,CAAA,CAAE,OAAA,GAAU,MAAA,CAAO,CAAC,GAAG,CAAA;AAAA,MACxG;AACA,MAAA;AAAA,IACF;AAEA,IAAA,QAAA,CAAS,GAAA,EAAK,GAAA,EAAK,EAAE,KAAA,EAAO,aAAa,CAAA;AAAA,EAC3C,CAAC,CAAA;AACH;;;AC1GA,SAAS,SAAA,GAAkB;AACzB,EAAA,OAAA,CAAQ,MAAA,CAAO,KAAA;AAAA,IACb,CAAA;;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;;AAAA;AAAA;;AAAA;AAAA;;AAAA;AAAA;;AAAA;AAAA,GAkCF;AACF;AAEA,SAAS,YAAA,GAAqB;AAC5B,EAAA,IAAI;AACF,IAAA,MAAM,GAAA,GAAMH,YAAAA,CAAQC,iBAAAA,CAAc,yPAAe,CAAC,CAAA;AAClD,IAAA,MAAM,OAAA,GAAUF,YAAAA,CAAQ,GAAA,EAAK,IAAA,EAAM,cAAc,CAAA;AACjD,IAAA,MAAM,MAAM,IAAA,CAAK,KAAA,CAAMI,eAAAA,CAAa,OAAA,EAAS,MAAM,CAAC,CAAA;AACpD,IAAA,OAAA,CAAQ,MAAA,CAAO,KAAA,CAAM,CAAA,EAAG,GAAA,CAAI,OAAO;AAAA,CAAI,CAAA;AAAA,EACzC,CAAA,CAAA,MAAQ;AACN,IAAA,OAAA,CAAQ,MAAA,CAAO,MAAM,WAAW,CAAA;AAAA,EAClC;AACF;AAEA,SAAS,UAAU,IAAA,EAAmC;AACpD,EAAA,MAAM,OAAA,GAAsB;AAAA,IAC1B,IAAA,EAAM,MAAA,CAAO,OAAA,CAAQ,GAAA,CAAI,yBAAyB,IAAI,CAAA;AAAA,IACtD,IAAA,EAAM,OAAA,CAAQ,GAAA,CAAI,qBAAA,IAAyB,WAAA;AAAA,IAC3C,KAAA,EAAO,QAAQ,GAAA,CAAI;AAAA,GACrB;AAEA,EAAA,KAAA,IAAS,CAAA,GAAI,CAAA,EAAG,CAAA,GAAI,IAAA,CAAK,QAAQ,CAAA,EAAA,EAAK;AACpC,IAAA,MAAM,GAAA,GAAM,KAAK,CAAC,CAAA;AAClB,IAAA,MAAM,IAAA,GAAO,IAAA,CAAK,CAAA,GAAI,CAAC,CAAA;AACvB,IAAA,IAAI,GAAA,KAAQ,QAAA,IAAY,GAAA,KAAQ,IAAA,EAAM;AAAE,MAAA,SAAA,EAAU;AAAG,MAAA,OAAO,IAAA;AAAA,IAAK;AACjE,IAAA,IAAI,GAAA,KAAQ,WAAA,IAAe,GAAA,KAAQ,IAAA,EAAM;AAAE,MAAA,YAAA,EAAa;AAAG,MAAA,OAAO,IAAA;AAAA,IAAK;AACvE,IAAA,IAAA,CAAK,GAAA,KAAQ,QAAA,IAAY,GAAA,KAAQ,IAAA,KAAS,IAAA,EAAM;AAAE,MAAA,OAAA,CAAQ,IAAA,GAAO,OAAO,IAAI,CAAA;AAAG,MAAA,CAAA,EAAA;AAAK,MAAA;AAAA,IAAS;AAC7F,IAAA,IAAA,CAAK,GAAA,KAAQ,QAAA,IAAY,GAAA,KAAQ,IAAA,KAAS,IAAA,EAAM;AAAE,MAAA,OAAA,CAAQ,IAAA,GAAO,IAAA;AAAM,MAAA,CAAA,EAAA;AAAK,MAAA;AAAA,IAAS;AACrF,IAAA,IAAA,CAAK,GAAA,KAAQ,SAAA,IAAa,GAAA,KAAQ,IAAA,KAAS,IAAA,EAAM;AAAE,MAAA,OAAA,CAAQ,KAAA,GAAQ,IAAA;AAAM,MAAA,CAAA,EAAA;AAAK,MAAA;AAAA,IAAS;AAAA,EACzF;AAEA,EAAA,OAAO,OAAA;AACT;AAEA,eAAe,IAAA,GAAsB;AACnC,EAAA,MAAM,UAAU,SAAA,CAAU,OAAA,CAAQ,IAAA,CAAK,KAAA,CAAM,CAAC,CAAC,CAAA;AAC/C,EAAA,IAAI,CAAC,OAAA,EAAS;AAAE,IAAA,OAAA,CAAQ,KAAK,CAAC,CAAA;AAAA,EAAE;AAEhC,EAAA,MAAM,aAAaP,iBAAAA,EAAa;AAChC,EAAA,MAAM,QAAA,GAAW,IAAI,cAAA,CAAe,EAAE,QAAQ,UAAA,EAAY,OAAA,EAAS,MAAM,CAAA;AACzE,EAAA,oBAAA,CAAqB,YAAY,QAAQ,CAAA;AAEzC,EAAA,UAAA,CAAW,MAAA,CAAO,OAAA,CAAQ,IAAA,EAAM,OAAA,CAAQ,MAAM,MAAM;AAClD,IAAA,MAAM,KAAA,GAAQ,QAAQ,KAAA,IAAS,kBAAA;AAC/B,IAAA,OAAA,CAAQ,OAAO,KAAA,CAAM;AAAA,EAAA,EAAO,KAAK;;AAAA,CAAM,CAAA;AACvC,IAAA,OAAA,CAAQ,OAAO,KAAA,CAAM,CAAA,oBAAA,EAAuB,QAAQ,IAAI,CAAA,CAAA,EAAI,QAAQ,IAAI;AAAA,CAAI,CAAA;AAC5E,IAAA,OAAA,CAAQ,OAAO,KAAA,CAAM,CAAA,iCAAA,EAAoC,QAAQ,IAAI,CAAA,CAAA,EAAI,QAAQ,IAAI,CAAA;AAAA,CAAyB,CAAA;AAC9G,IAAA,OAAA,CAAQ,OAAO,KAAA,CAAM,CAAA,kBAAA,EAAqB,QAAQ,IAAI,CAAA,CAAA,EAAI,QAAQ,IAAI,CAAA;AAAA,CAAO,CAAA;AAC7E,IAAA,OAAA,CAAQ,OAAO,KAAA,CAAM,CAAA,oBAAA,EAAuB,QAAQ,IAAI,CAAA,CAAA,EAAI,QAAQ,IAAI,CAAA;;AAAA,CAAgB,CAAA;AACxF,IAAA,OAAA,CAAQ,OAAO,KAAA,CAAM,CAAA;AAAA,CAA+C,CAAA;AACpE,IAAA,OAAA,CAAQ,OAAO,KAAA,CAAM,CAAA,yCAAA,EAA4C,QAAQ,IAAI,CAAA,CAAA,EAAI,QAAQ,IAAI;;AAAA,CAAM,CAAA;AAAA,EACrG,CAAC,CAAA;AAED,EAAA,MAAM,WAAW,MAAM;AACrB,IAAA,QAAA,CAAS,OAAM,CAAE,IAAA,CAAK,MAAM,OAAA,CAAQ,IAAA,CAAK,CAAC,CAAC,CAAA;AAAA,EAC7C,CAAA;AACA,EAAA,OAAA,CAAQ,EAAA,CAAG,UAAU,QAAQ,CAAA;AAC7B,EAAA,OAAA,CAAQ,EAAA,CAAG,WAAW,QAAQ,CAAA;AAChC;AAEA,IAAA,EAAK,CAAE,KAAA,CAAM,CAAC,KAAA,KAAU;AACtB,EAAA,OAAA,CAAQ,MAAA,CAAO,MAAM,CAAA,oCAAA,EAAuC,KAAA,YAAiB,QAAQ,KAAA,CAAM,OAAA,GAAU,MAAA,CAAO,KAAK,CAAC;AAAA,CAAI,CAAA;AACtH,EAAA,OAAA,CAAQ,KAAK,CAAC,CAAA;AAChB,CAAC,CAAA","file":"cli.cjs","sourcesContent":["/**\n * Error Aggregator\n *\n * Groups similar errors together based on stack trace fingerprinting.\n * Tracks error frequency, first/last occurrence, and affected traces.\n *\n * @example\n * ```typescript\n * const aggregator = new ErrorAggregator({ maxGroups: 100 });\n *\n * // Add errors from spans\n * aggregator.addError({\n * traceId: '123',\n * spanId: '456',\n * spanName: 'api.createUser',\n * service: 'user-service',\n * timestamp: Date.now(),\n * error: {\n * type: 'ValidationError',\n * message: 'Invalid email format',\n * stackTrace: 'Error: Invalid email...'\n * }\n * });\n *\n * // Get aggregated error groups\n * const groups = aggregator.getErrorGroups();\n * ```\n */\n\nimport type { ErrorGroup, ErrorOccurrence, SpanData, TraceData } from './types';\n\nexport interface ErrorAggregatorOptions {\n /**\n * Maximum number of error groups to track (default: 100)\n * Oldest groups are evicted when limit is reached\n */\n maxGroups?: number;\n\n /**\n * Maximum number of affected traces to keep per group (default: 10)\n */\n maxAffectedTraces?: number;\n\n /**\n * Maximum number of affected span names to keep per group (default: 5)\n */\n maxAffectedSpans?: number;\n\n /**\n * Number of stack frames to use for fingerprinting (default: 5)\n */\n stackFramesForFingerprint?: number;\n}\n\nexport class ErrorAggregator {\n private errorGroups: Map<string, ErrorGroup> = new Map();\n private options: Required<ErrorAggregatorOptions>;\n\n constructor(options: ErrorAggregatorOptions = {}) {\n this.options = {\n maxGroups: options.maxGroups ?? 100,\n maxAffectedTraces: options.maxAffectedTraces ?? 10,\n maxAffectedSpans: options.maxAffectedSpans ?? 5,\n stackFramesForFingerprint: options.stackFramesForFingerprint ?? 5,\n };\n }\n\n /**\n * Add an error occurrence to the aggregator\n */\n addError(occurrence: ErrorOccurrence): ErrorGroup {\n const fingerprint = this.generateFingerprint(occurrence);\n const existing = this.errorGroups.get(fingerprint);\n\n if (existing) {\n // Update existing group\n existing.count++;\n existing.lastSeen = occurrence.timestamp;\n\n // Add trace ID if not already present (keep last N)\n if (!existing.affectedTraces.includes(occurrence.traceId)) {\n existing.affectedTraces.push(occurrence.traceId);\n if (existing.affectedTraces.length > this.options.maxAffectedTraces) {\n existing.affectedTraces.shift();\n }\n }\n\n // Add span name if not already present\n if (!existing.affectedSpans.includes(occurrence.spanName)) {\n existing.affectedSpans.push(occurrence.spanName);\n if (existing.affectedSpans.length > this.options.maxAffectedSpans) {\n existing.affectedSpans.shift();\n }\n }\n\n return existing;\n }\n\n // Create new group\n const newGroup: ErrorGroup = {\n fingerprint,\n type: occurrence.error.type,\n message: occurrence.error.message,\n stackTrace: this.normalizeStackTrace(occurrence.error.stackTrace),\n count: 1,\n firstSeen: occurrence.timestamp,\n lastSeen: occurrence.timestamp,\n affectedTraces: [occurrence.traceId],\n affectedSpans: [occurrence.spanName],\n service: occurrence.service,\n attributes: occurrence.attributes,\n };\n\n // Evict oldest group if at capacity\n if (this.errorGroups.size >= this.options.maxGroups) {\n this.evictOldestGroup();\n }\n\n this.errorGroups.set(fingerprint, newGroup);\n return newGroup;\n }\n\n /**\n * Extract errors from a trace and add them to the aggregator\n */\n addErrorsFromTrace(trace: TraceData): ErrorGroup[] {\n const addedGroups: ErrorGroup[] = [];\n\n for (const span of trace.spans) {\n if (span.status.code === 'ERROR') {\n const occurrence = this.extractErrorFromSpan(span, trace);\n if (occurrence) {\n const group = this.addError(occurrence);\n addedGroups.push(group);\n }\n }\n }\n\n return addedGroups;\n }\n\n /**\n * Extract error occurrence from a span\n */\n private extractErrorFromSpan(\n span: SpanData,\n trace: TraceData,\n ): ErrorOccurrence | null {\n // Try to get error info from span attributes or events\n const exceptionEvent = span.events?.find((e) => e.name === 'exception');\n const errorType =\n (span.attributes['exception.type'] as string) ||\n (span.attributes['error.type'] as string) ||\n (exceptionEvent?.attributes?.['exception.type'] as string) ||\n 'Error';\n\n const errorMessage =\n span.status.message ||\n (span.attributes['exception.message'] as string) ||\n (span.attributes['error.message'] as string) ||\n 'Unknown error';\n\n const stackTrace =\n (span.attributes['exception.stacktrace'] as string) ||\n (span.attributes['exception.stack'] as string) ||\n this.extractStackFromEvents(span);\n\n return {\n traceId: trace.traceId,\n spanId: span.spanId,\n spanName: span.name,\n service: trace.service,\n timestamp: span.endTime,\n error: {\n type: errorType,\n message: errorMessage,\n stackTrace,\n },\n attributes: this.extractRelevantAttributes(span.attributes),\n };\n }\n\n /**\n * Extract stack trace from span events (exception events)\n */\n private extractStackFromEvents(span: SpanData): string | undefined {\n if (!span.events) return undefined;\n\n const exceptionEvent = span.events.find((e) => e.name === 'exception');\n if (exceptionEvent?.attributes) {\n return (\n (exceptionEvent.attributes['exception.stacktrace'] as string) ||\n (exceptionEvent.attributes['exception.stack'] as string)\n );\n }\n\n return undefined;\n }\n\n /**\n * Extract relevant attributes for error context\n */\n private extractRelevantAttributes(\n attributes: Record<string, unknown>,\n ): Record<string, unknown> {\n const relevant: Record<string, unknown> = {};\n const keepKeys = [\n 'http.method',\n 'http.url',\n 'http.route',\n 'http.status_code',\n 'db.system',\n 'db.operation',\n 'rpc.method',\n 'rpc.service',\n 'code.function',\n 'code.filepath',\n 'user.id',\n 'operation.name',\n ];\n\n for (const key of keepKeys) {\n if (key in attributes) {\n relevant[key] = attributes[key];\n }\n }\n\n return relevant;\n }\n\n /**\n * Generate a fingerprint for error grouping\n *\n * Uses error type + first N stack frames (normalized)\n */\n private generateFingerprint(occurrence: ErrorOccurrence): string {\n const parts: string[] = [occurrence.error.type];\n\n if (occurrence.error.stackTrace) {\n const frames = this.extractStackFrames(\n occurrence.error.stackTrace,\n this.options.stackFramesForFingerprint,\n );\n parts.push(...frames);\n } else {\n // Fallback to error message if no stack trace\n parts.push(this.normalizeMessage(occurrence.error.message));\n }\n\n // Simple hash function\n return this.simpleHash(parts.join('|'));\n }\n\n /**\n * Extract and normalize stack frames from a stack trace\n */\n private extractStackFrames(stackTrace: string, count: number): string[] {\n const lines = stackTrace.split('\\n');\n const frames: string[] = [];\n\n for (const line of lines) {\n if (frames.length >= count) break;\n\n // Match common stack trace patterns\n const trimmed = line.trim();\n\n // Node.js style: \"at functionName (file:line:col)\"\n const nodeMatch = trimmed.match(/^at\\s+(.+?)\\s+\\((.+?):(\\d+):\\d+\\)$/);\n if (nodeMatch) {\n frames.push(`${nodeMatch[1]}@${this.normalizeFilePath(nodeMatch[2])}`);\n continue;\n }\n\n // Anonymous function style: \"at file:line:col\"\n const anonMatch = trimmed.match(/^at\\s+(.+?):(\\d+):\\d+$/);\n if (anonMatch) {\n frames.push(`anonymous@${this.normalizeFilePath(anonMatch[1])}`);\n continue;\n }\n\n // Browser style: \"functionName@file:line:col\"\n const browserMatch = trimmed.match(/^(.+?)@(.+?):(\\d+):\\d+$/);\n if (browserMatch) {\n frames.push(\n `${browserMatch[1]}@${this.normalizeFilePath(browserMatch[2])}`,\n );\n continue;\n }\n }\n\n return frames;\n }\n\n /**\n * Normalize file path by removing absolute path prefixes and node_modules paths\n */\n private normalizeFilePath(filePath: string): string {\n // Remove node_modules paths (keep package name)\n const nodeModulesMatch = filePath.match(\n /node_modules\\/(@[^/]+\\/[^/]+|[^/]+)/,\n );\n if (nodeModulesMatch) {\n return `[npm]/${nodeModulesMatch[1]}`;\n }\n\n // Remove common absolute path prefixes\n return filePath\n .replace(/^.*?\\/src\\//, 'src/')\n .replace(/^.*?\\/dist\\//, 'dist/')\n .replace(/^.*?\\/lib\\//, 'lib/')\n .replace(/^file:\\/\\//, '');\n }\n\n /**\n * Normalize error message by removing dynamic parts\n */\n private normalizeMessage(message: string): string {\n return (\n message\n // Remove UUIDs\n .replaceAll(\n /[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}/gi,\n '[UUID]',\n )\n // Remove hex IDs\n .replaceAll(/\\b[0-9a-f]{16,}\\b/gi, '[ID]')\n // Remove numbers\n .replaceAll(/\\b\\d+\\b/g, '[N]')\n // Remove quoted strings\n .replaceAll(/\"[^\"]*\"/g, '\"[STR]\"')\n .replaceAll(/'[^']*'/g, \"'[STR]'\")\n // Truncate long messages\n .slice(0, 200)\n );\n }\n\n /**\n * Normalize stack trace for display\n */\n private normalizeStackTrace(stackTrace?: string): string | undefined {\n if (!stackTrace) return undefined;\n\n const lines = stackTrace.split('\\n').slice(0, 10); // Keep first 10 lines\n return lines.join('\\n');\n }\n\n /**\n * Simple hash function for fingerprinting\n */\n private simpleHash(str: string): string {\n let hash = 0;\n for (let i = 0; i < str.length; i++) {\n const char = str.charCodeAt(i);\n hash = (hash << 5) - hash + char;\n hash = hash & hash; // Convert to 32-bit integer\n }\n return Math.abs(hash).toString(16).padStart(8, '0');\n }\n\n /**\n * Evict the oldest error group\n */\n private evictOldestGroup(): void {\n let oldest: { fingerprint: string; lastSeen: number } | null = null;\n\n for (const [fingerprint, group] of this.errorGroups) {\n if (!oldest || group.lastSeen < oldest.lastSeen) {\n oldest = { fingerprint, lastSeen: group.lastSeen };\n }\n }\n\n if (oldest) {\n this.errorGroups.delete(oldest.fingerprint);\n }\n }\n\n /**\n * Get all error groups, sorted by most recent\n */\n getErrorGroups(): ErrorGroup[] {\n return [...this.errorGroups.values()].sort(\n (a, b) => b.lastSeen - a.lastSeen,\n );\n }\n\n /**\n * Get error groups sorted by count (most frequent first)\n */\n getErrorGroupsByFrequency(): ErrorGroup[] {\n return [...this.errorGroups.values()].sort(\n (a, b) => b.count - a.count,\n );\n }\n\n /**\n * Get a specific error group by fingerprint\n */\n getErrorGroup(fingerprint: string): ErrorGroup | undefined {\n return this.errorGroups.get(fingerprint);\n }\n\n /**\n * Get error groups for a specific service\n */\n getErrorGroupsByService(service: string): ErrorGroup[] {\n return this.getErrorGroups().filter((g) => g.service === service);\n }\n\n /**\n * Get total error count across all groups\n */\n getTotalErrorCount(): number {\n let total = 0;\n for (const group of this.errorGroups.values()) {\n total += group.count;\n }\n return total;\n }\n\n /**\n * Get error statistics\n */\n getStats(): {\n totalGroups: number;\n totalErrors: number;\n recentErrors: number;\n topErrorTypes: Array<{ type: string; count: number }>;\n } {\n const now = Date.now();\n const oneHourAgo = now - 60 * 60 * 1000;\n\n let recentErrors = 0;\n const typeCount = new Map<string, number>();\n\n for (const group of this.errorGroups.values()) {\n if (group.lastSeen > oneHourAgo) {\n recentErrors += group.count;\n }\n typeCount.set(group.type, (typeCount.get(group.type) || 0) + group.count);\n }\n\n const topErrorTypes = [...typeCount.entries()]\n .map(([type, count]) => ({ type, count }))\n .sort((a, b) => b.count - a.count)\n .slice(0, 5);\n\n return {\n totalGroups: this.errorGroups.size,\n totalErrors: this.getTotalErrorCount(),\n recentErrors,\n topErrorTypes,\n };\n }\n\n /**\n * Clear all error groups\n */\n clear(): void {\n this.errorGroups.clear();\n }\n\n /**\n * Clear old error groups (not seen in given time window)\n */\n clearOlderThan(maxAgeMs: number): number {\n const cutoff = Date.now() - maxAgeMs;\n let cleared = 0;\n\n for (const [fingerprint, group] of this.errorGroups) {\n if (group.lastSeen < cutoff) {\n this.errorGroups.delete(fingerprint);\n cleared++;\n }\n }\n\n return cleared;\n }\n}\n","import type { DevtoolsData } from './types'\n\nexport interface TelemetryLimits {\n maxTraceCount: number\n maxLogCount: number\n maxMetricCount: number\n}\n\nexport interface ResolveTelemetryLimitsArgs {\n maxHistory?: number\n maxTraceCount?: number\n maxLogCount?: number\n maxMetricCount?: number\n env?: NodeJS.ProcessEnv\n}\n\nconst defaultLimit = 100\n\nfunction parseLimit(value: string | undefined): number | undefined {\n if (!value) return undefined\n const parsed = Number.parseInt(value, 10)\n return Number.isFinite(parsed) && parsed > 0 ? parsed : undefined\n}\n\nexport function resolveTelemetryLimits(\n args: ResolveTelemetryLimitsArgs = {},\n): TelemetryLimits {\n const env = args.env ?? process.env\n const fallback = args.maxHistory ?? defaultLimit\n\n return {\n maxTraceCount:\n args.maxTraceCount ??\n parseLimit(env.AUTOTEL_MAX_TRACE_COUNT) ??\n fallback,\n maxLogCount:\n args.maxLogCount ??\n parseLimit(env.AUTOTEL_MAX_LOG_COUNT) ??\n fallback,\n maxMetricCount:\n args.maxMetricCount ??\n parseLimit(env.AUTOTEL_MAX_METRIC_COUNT) ??\n fallback,\n }\n}\n\nexport function appendWithLimit<T>(items: T[], item: T, limit: number): T[] {\n if (limit <= 0) return []\n const next = [...items, item]\n return next.length > limit ? next.slice(next.length - limit) : next\n}\n\nexport function appendManyWithLimit<T>(\n items: T[],\n incoming: T[],\n limit: number,\n): T[] {\n if (limit <= 0 || incoming.length === 0) return limit <= 0 ? [] : items\n const next = [...items, ...incoming]\n return next.length > limit ? next.slice(next.length - limit) : next\n}\n\nexport function applyTelemetryLimits(\n data: DevtoolsData,\n limits: TelemetryLimits,\n): DevtoolsData {\n return {\n ...data,\n traces: data.traces.slice(-limits.maxTraceCount),\n logs: data.logs.slice(-limits.maxLogCount),\n metrics: data.metrics.slice(-limits.maxMetricCount),\n }\n}\n","// src/server/server.ts\nimport { WebSocketServer, WebSocket } from 'ws'\nimport type { Server as HTTPServer } from 'node:http'\nimport { createServer } from 'node:http'\nimport { ErrorAggregator } from './error-aggregator'\nimport type {\n TraceData,\n LogData,\n MetricData,\n DevtoolsData,\n} from './types'\nimport {\n appendManyWithLimit,\n appendWithLimit,\n resolveTelemetryLimits,\n type TelemetryLimits,\n} from './telemetry-limits'\n\nexport interface DevtoolsServerOptions {\n port?: number\n server?: HTTPServer\n path?: string\n verbose?: boolean\n maxHistory?: number\n maxTraceCount?: number\n maxLogCount?: number\n maxMetricCount?: number\n}\n\nexport class DevtoolsServer {\n private wss: WebSocketServer\n private clients = new Set<WebSocket>()\n private httpServer: HTTPServer\n private traces: TraceData[] = []\n private logs: LogData[] = []\n private metrics: MetricData[] = []\n private errorAggregator = new ErrorAggregator()\n private limits: TelemetryLimits\n private verbose: boolean\n private _port: number\n\n constructor(options: DevtoolsServerOptions = {}) {\n this.limits = resolveTelemetryLimits(options)\n this.verbose = options.verbose ?? false\n this._port = options.port ?? 4318\n\n this.httpServer = options.server ?? createServer()\n this.wss = new WebSocketServer({ server: this.httpServer, path: options.path ?? '/ws' })\n\n this.wss.on('connection', (ws) => {\n this.clients.add(ws)\n this.log(`Client connected (${this.clients.size} total)`)\n\n // Send history to late-connecting clients\n const data = this.getCurrentData()\n if (data.traces.length > 0 || data.logs.length > 0 || data.errors.length > 0) {\n ws.send(JSON.stringify(data))\n }\n\n ws.on('close', () => {\n this.clients.delete(ws)\n this.log(`Client disconnected (${this.clients.size} total)`)\n })\n })\n\n // Only start listening if no external server was provided\n if (!options.server) {\n this.httpServer.listen(this._port, () => {\n const addr = this.httpServer.address()\n if (addr && typeof addr === 'object') this._port = addr.port\n this.log(`WebSocket server listening on port ${this._port}`)\n })\n }\n }\n\n get port(): number {\n const addr = this.httpServer.address()\n if (addr && typeof addr === 'object') return addr.port\n return this._port\n }\n\n get clientCount(): number {\n return this.clients.size\n }\n\n addTrace(trace: TraceData): void {\n // Merge if trace already exists (out-of-order spans)\n const existing = this.traces.find(t => t.traceId === trace.traceId)\n if (existing) {\n const existingSpanIds = new Set(existing.spans.map(s => s.spanId))\n for (const span of trace.spans) {\n if (!existingSpanIds.has(span.spanId)) {\n existing.spans.push(span)\n }\n }\n existing.startTime = Math.min(existing.startTime, trace.startTime)\n existing.endTime = Math.max(existing.endTime, trace.endTime)\n existing.duration = existing.endTime - existing.startTime\n if (trace.status === 'ERROR') existing.status = 'ERROR'\n } else {\n this.traces = appendWithLimit(\n this.traces,\n trace,\n this.limits.maxTraceCount,\n )\n }\n\n this.errorAggregator.addErrorsFromTrace(trace)\n this.broadcast({ traces: [trace], metrics: [], logs: [], errors: this.errorAggregator.getErrorGroups() })\n }\n\n addTraces(traces: TraceData[]): void {\n for (const trace of traces) this.addTrace(trace)\n }\n\n addLog(log: LogData): void {\n this.logs = appendWithLimit(this.logs, log, this.limits.maxLogCount)\n this.broadcast({ traces: [], metrics: [], logs: [log], errors: [] })\n }\n\n addLogs(logs: LogData[]): void {\n this.logs = appendManyWithLimit(this.logs, logs, this.limits.maxLogCount)\n this.broadcast({ traces: [], metrics: [], logs, errors: [] })\n }\n\n addMetric(metric: MetricData): void {\n this.metrics = appendWithLimit(\n this.metrics,\n metric,\n this.limits.maxMetricCount,\n )\n this.broadcast({ traces: [], metrics: [metric], logs: [], errors: [] })\n }\n\n getCurrentData(): DevtoolsData {\n return {\n traces: this.traces,\n metrics: this.metrics,\n logs: this.logs,\n errors: this.errorAggregator.getErrorGroups(),\n }\n }\n\n clearData(): void {\n this.traces = []\n this.logs = []\n this.metrics = []\n this.errorAggregator.clear()\n }\n\n private broadcast(data: DevtoolsData): void {\n const msg = JSON.stringify(data)\n for (const client of this.clients) {\n if (client.readyState === WebSocket.OPEN) {\n client.send(msg)\n }\n }\n }\n\n private log(message: string): void {\n if (this.verbose) console.log(`[autotel-devtools] ${message}`)\n }\n\n async close(): Promise<void> {\n for (const client of this.clients) client.close()\n this.clients.clear()\n this.wss.close()\n await new Promise<void>((resolve) => this.httpServer.close(() => resolve()))\n }\n}\n","export function getResourceName(\n resource: Record<string, unknown> | undefined,\n fallback = 'unknown',\n): string {\n if (!resource) return fallback\n\n const candidates = [\n resource['service.name'],\n resource['service.namespace'],\n resource['deployment.environment.name'],\n resource['host.name'],\n resource['container.name'],\n resource['process.executable.name'],\n ]\n\n for (const candidate of candidates) {\n if (typeof candidate === 'string' && candidate.trim().length > 0) {\n return candidate\n }\n }\n\n return fallback\n}\n","// src/server/otlp.ts\nimport type { IncomingMessage, ServerResponse } from 'node:http'\nimport type { SpanData, TraceData, LogData } from './types'\nimport { getResourceName } from './resource-utils'\n\ntype OtlpAnyValue = {\n stringValue?: string\n boolValue?: boolean\n intValue?: string | number\n doubleValue?: number\n bytesValue?: string\n arrayValue?: { values?: OtlpAnyValue[] }\n kvlistValue?: { values?: OtlpKeyValue[] }\n}\n\ntype OtlpKeyValue = { key: string; value?: OtlpAnyValue }\n\nfunction resolveOtlpValue(v?: OtlpAnyValue): unknown {\n if (!v) return undefined\n if (v.stringValue !== undefined) return v.stringValue\n if (v.boolValue !== undefined) return v.boolValue\n if (v.intValue !== undefined) return typeof v.intValue === 'string' ? Number(v.intValue) : v.intValue\n if (v.doubleValue !== undefined) return v.doubleValue\n if (v.bytesValue !== undefined) return v.bytesValue\n if (v.arrayValue?.values) return v.arrayValue.values.map(resolveOtlpValue)\n if (v.kvlistValue?.values) return flattenAttributes(v.kvlistValue.values)\n return undefined\n}\n\nfunction flattenAttributes(attrs?: OtlpKeyValue[]): Record<string, unknown> {\n const out: Record<string, unknown> = {}\n if (!attrs) return out\n for (const { key, value } of attrs) {\n out[key] = resolveOtlpValue(value)\n }\n return out\n}\n\nfunction nanoToMs(nano?: string): number {\n if (!nano) return 0\n return Number(BigInt(nano) / 1_000_000n)\n}\n\nconst SPAN_KIND_MAP: Record<number | string, SpanData['kind']> = {\n 0: 'INTERNAL', 1: 'INTERNAL', 2: 'SERVER', 3: 'CLIENT', 4: 'PRODUCER', 5: 'CONSUMER',\n SPAN_KIND_INTERNAL: 'INTERNAL', SPAN_KIND_SERVER: 'SERVER',\n SPAN_KIND_CLIENT: 'CLIENT', SPAN_KIND_PRODUCER: 'PRODUCER', SPAN_KIND_CONSUMER: 'CONSUMER',\n}\n\nfunction normalizeHexId(id?: string): string {\n if (!id) return ''\n // Only attempt base64 decode for strings that look like base64-encoded binary IDs\n // (length 24 or 28 chars for 16/32-byte IDs, valid base64 chars, not plain hex)\n const isBase64Like = /^[A-Za-z0-9+/=]+$/.test(id) && !(/^[0-9a-f]+$/i.test(id))\n const isLikelyBase64Id = isBase64Like && (id.length === 24 || id.length === 28 || id.length === 44 || id.length === 48)\n if (isLikelyBase64Id) {\n try {\n const bytes = Buffer.from(id, 'base64')\n return bytes.toString('hex')\n } catch { /* fall through */ }\n }\n return id\n}\n\nexport function parseOtlpTraces(payload: unknown): TraceData[] {\n if (!payload || typeof payload !== 'object') return []\n const { resourceSpans } = payload as any\n if (!Array.isArray(resourceSpans) || resourceSpans.length === 0) return []\n\n const traceMap = new Map<string, { spans: SpanData[]; service: string }>()\n\n for (const rs of resourceSpans) {\n const resourceAttrs = flattenAttributes(rs.resource?.attributes)\n const service = String(resourceAttrs['service.name'] || 'unknown')\n const scopeSpans = rs.scopeSpans || []\n\n for (const ss of scopeSpans) {\n for (const span of ss.spans || []) {\n const traceId = normalizeHexId(span.traceId)\n if (!traceId) continue\n\n const startMs = nanoToMs(span.startTimeUnixNano)\n const endMs = nanoToMs(span.endTimeUnixNano)\n const statusCode = span.status?.code\n let status: SpanData['status']['code'] = 'UNSET'\n if (statusCode === 1 || statusCode === 'STATUS_CODE_OK') status = 'OK'\n if (statusCode === 2 || statusCode === 'STATUS_CODE_ERROR') status = 'ERROR'\n\n const spanData: SpanData = {\n traceId,\n spanId: normalizeHexId(span.spanId),\n parentSpanId: normalizeHexId(span.parentSpanId) || undefined,\n name: span.name || 'unknown',\n kind: SPAN_KIND_MAP[span.kind ?? 0] || 'INTERNAL',\n startTime: startMs,\n endTime: endMs,\n duration: endMs - startMs,\n attributes: { ...resourceAttrs, ...flattenAttributes(span.attributes) } as Record<string, any>,\n status: { code: status, message: span.status?.message },\n events: (span.events || []).map((e: any) => ({\n name: e.name || '',\n timestamp: nanoToMs(e.timeUnixNano),\n attributes: flattenAttributes(e.attributes) as Record<string, any>,\n })),\n }\n\n const existing = traceMap.get(traceId)\n if (existing) {\n existing.spans.push(spanData)\n } else {\n traceMap.set(traceId, { spans: [spanData], service })\n }\n }\n }\n }\n\n const traces: TraceData[] = []\n for (const [traceId, { spans, service }] of traceMap) {\n const sorted = spans.sort((a, b) => a.startTime - b.startTime)\n const rootSpan = sorted.find(s => !s.parentSpanId) || sorted[0]\n const startTime = Math.min(...sorted.map(s => s.startTime))\n const endTime = Math.max(...sorted.map(s => s.endTime))\n const hasError = sorted.some(s => s.status.code === 'ERROR')\n\n traces.push({\n traceId,\n correlationId: traceId.slice(0, 16),\n rootSpan,\n spans: sorted,\n startTime,\n endTime,\n duration: endTime - startTime,\n status: hasError ? 'ERROR' : 'OK',\n service,\n })\n }\n\n return traces\n}\n\nexport function parseOtlpLogs(payload: unknown): LogData[] {\n if (!payload || typeof payload !== 'object') return []\n const { resourceLogs } = payload as any\n if (!Array.isArray(resourceLogs)) return []\n\n const logs: LogData[] = []\n for (const rl of resourceLogs) {\n const resourceAttrs = flattenAttributes(rl.resource?.attributes)\n for (const sl of rl.scopeLogs || []) {\n for (const rec of sl.logRecords || []) {\n const timestamp = nanoToMs(rec.timeUnixNano || rec.observedTimeUnixNano)\n const traceId = normalizeHexId(rec.traceId) || undefined\n const spanId = normalizeHexId(rec.spanId) || undefined\n const body = rec.body ? resolveOtlpValue(rec.body) : ''\n\n logs.push({\n id: `${traceId || 'no-trace'}:${spanId || 'no-span'}:${timestamp}:${rec.severityNumber || 0}`,\n traceId,\n spanId,\n resourceName: getResourceName(resourceAttrs as Record<string, unknown>),\n severityText: rec.severityText,\n severityNumber: rec.severityNumber,\n body: typeof body === 'string' ? body : (body as Record<string, unknown>),\n timestamp,\n attributes: flattenAttributes(rec.attributes) as Record<string, unknown>,\n resource: resourceAttrs as Record<string, unknown>,\n })\n }\n }\n }\n\n return logs\n}\n\nexport function countOtlpMetrics(payload: unknown): number {\n if (!payload || typeof payload !== 'object') return 0\n const { resourceMetrics } = payload as any\n if (!Array.isArray(resourceMetrics)) return 0\n let count = 0\n for (const rm of resourceMetrics) {\n for (const sm of rm.scopeMetrics || []) {\n count += (sm.metrics || []).length\n }\n }\n return count\n}\n\nexport async function readJsonBody(req: IncomingMessage): Promise<unknown> {\n return new Promise((resolve, reject) => {\n const chunks: Buffer[] = []\n req.on('data', (chunk: Buffer) => chunks.push(chunk))\n req.on('end', () => {\n try {\n resolve(JSON.parse(Buffer.concat(chunks).toString()))\n } catch {\n reject(new Error('Invalid JSON'))\n }\n })\n req.on('error', reject)\n })\n}\n\nexport function sendJson(res: ServerResponse, status: number, data: Record<string, unknown>): void {\n const body = JSON.stringify(data)\n res.writeHead(status, { 'Content-Type': 'application/json', 'Content-Length': Buffer.byteLength(body) })\n res.end(body)\n}\n","// src/server/http.ts\nimport { createServer, type IncomingMessage, type ServerResponse, type Server } from 'node:http'\nimport { readFileSync, existsSync } from 'node:fs'\nimport { resolve, dirname } from 'node:path'\nimport { fileURLToPath } from 'node:url'\nimport { parseOtlpTraces, parseOtlpLogs, countOtlpMetrics, readJsonBody, sendJson } from './otlp'\nimport type { DevtoolsServer } from './server'\n\nexport interface HttpServerOptions {\n port?: number\n host?: string\n}\n\nfunction findPackageRoot(): string {\n let dir = dirname(fileURLToPath(import.meta.url))\n for (let i = 0; i < 5; i++) {\n if (existsSync(resolve(dir, 'package.json'))) return dir\n dir = dirname(dir)\n }\n return dir\n}\n\nconst FULLPAGE_HTML = `<!DOCTYPE html><html><head><meta charset=\"UTF-8\"><meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\"><title>autotel-devtools</title><style>*{margin:0;padding:0;box-sizing:border-box}html,body{height:100%;width:100%;overflow:hidden}</style></head><body><script src=\"/widget.js?mode=fullpage\"></script></body></html>`\n\nlet cachedWidgetJs: string | null = null\nfunction getWidgetJs(): string {\n if (!cachedWidgetJs) {\n const pkgRoot = findPackageRoot()\n const candidates = [\n resolve(pkgRoot, 'dist', 'widget.global.js'),\n resolve(pkgRoot, 'widget.global.js'),\n ]\n for (const candidate of candidates) {\n try {\n cachedWidgetJs = readFileSync(candidate, 'utf8')\n break\n } catch { /* try next */ }\n }\n if (!cachedWidgetJs) {\n cachedWidgetJs = '// widget bundle not found - run pnpm build first'\n }\n }\n return cachedWidgetJs\n}\n\nexport function attachDevtoolsRoutes(httpServer: Server, devtools: DevtoolsServer): void {\n httpServer.on('request', async (req: IncomingMessage, res: ServerResponse) => {\n // CORS headers\n res.setHeader('Access-Control-Allow-Origin', '*')\n res.setHeader('Access-Control-Allow-Methods', 'GET, POST, OPTIONS')\n res.setHeader('Access-Control-Allow-Headers', 'Content-Type')\n\n if (req.method === 'OPTIONS') {\n res.writeHead(204)\n res.end()\n return\n }\n\n const url = req.url || '/'\n\n // GET / — fullpage HTML\n if (req.method === 'GET' && url === '/') {\n res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8', 'Content-Length': Buffer.byteLength(FULLPAGE_HTML) })\n res.end(FULLPAGE_HTML)\n return\n }\n\n // GET /widget.js — widget bundle\n if (req.method === 'GET' && url.startsWith('/widget.js')) {\n const js = getWidgetJs()\n res.writeHead(200, { 'Content-Type': 'application/javascript; charset=utf-8', 'Content-Length': Buffer.byteLength(js) })\n res.end(js)\n return\n }\n\n // GET /healthz\n if (req.method === 'GET' && url === '/healthz') {\n sendJson(res, 200, { ok: true, clients: devtools.clientCount })\n return\n }\n\n // POST /v1/traces\n if (req.method === 'POST' && url === '/v1/traces') {\n try {\n const payload = await readJsonBody(req)\n const traces = parseOtlpTraces(payload)\n devtools.addTraces(traces)\n sendJson(res, 200, { acceptedTraces: traces.length })\n } catch (e) {\n sendJson(res, 400, { error: 'Invalid OTLP JSON', message: e instanceof Error ? e.message : String(e) })\n }\n return\n }\n\n // POST /v1/logs\n if (req.method === 'POST' && url === '/v1/logs') {\n try {\n const payload = await readJsonBody(req)\n const logs = parseOtlpLogs(payload)\n devtools.addLogs(logs)\n sendJson(res, 200, { acceptedLogs: logs.length })\n } catch (e) {\n sendJson(res, 400, { error: 'Invalid OTLP JSON', message: e instanceof Error ? e.message : String(e) })\n }\n return\n }\n\n // POST /v1/metrics\n if (req.method === 'POST' && url === '/v1/metrics') {\n try {\n const payload = await readJsonBody(req)\n const count = countOtlpMetrics(payload)\n sendJson(res, 200, { acceptedMetrics: count })\n } catch (e) {\n sendJson(res, 400, { error: 'Invalid OTLP JSON', message: e instanceof Error ? e.message : String(e) })\n }\n return\n }\n\n sendJson(res, 404, { error: 'Not found' })\n })\n}\n\nexport function createDevtoolsHttpServer(devtools: DevtoolsServer, _options: HttpServerOptions = {}): Server {\n const server = createServer()\n attachDevtoolsRoutes(server, devtools)\n return server\n}\n","#!/usr/bin/env node\n// src/cli.ts\nimport { createServer } from 'node:http'\nimport { readFileSync } from 'node:fs'\nimport { resolve, dirname } from 'node:path'\nimport { fileURLToPath } from 'node:url'\nimport { DevtoolsServer } from './server/server'\nimport { attachDevtoolsRoutes } from './server/http'\n\ninterface CliOptions {\n port: number\n host: string\n title?: string\n}\n\nfunction printHelp(): void {\n process.stdout.write(\n `autotel-devtools - Standalone OTLP receiver with web devtools UI\n\nUsage: autotel-devtools [options]\n\nOptions:\n -p, --port <port> Port to listen on (default: 4318, env: AUTOTEL_DEVTOOLS_PORT)\n -H, --host <host> Host to bind to (default: 127.0.0.1, env: AUTOTEL_DEVTOOLS_HOST)\n -t, --title <title> Dashboard title (env: AUTOTEL_DEVTOOLS_TITLE)\n Env limits: AUTOTEL_MAX_TRACE_COUNT, AUTOTEL_MAX_LOG_COUNT, AUTOTEL_MAX_METRIC_COUNT\n -h, --help Show this help message\n -v, --version Show version number\n\nEndpoints:\n GET / Web devtools UI (fullpage)\n GET /widget.js Widget bundle (embed in your app)\n POST /v1/traces Receive OTLP JSON trace data\n POST /v1/logs Receive OTLP JSON log data\n POST /v1/metrics Receive OTLP JSON metric data\n WS /ws WebSocket stream for real-time updates\n GET /healthz Health check\n\nExamples:\n npx autotel-devtools\n npx autotel-devtools -p 4319\n\nThen point your app:\n OTEL_EXPORTER_OTLP_PROTOCOL=http/json OTEL_EXPORTER_OTLP_ENDPOINT=http://localhost:4318 node app.js\n\nView in browser:\n http://localhost:4318\n\nOr embed widget in your app:\n <script src=\"http://localhost:4318/widget.js\"></script>\n` + '\\n',\n )\n}\n\nfunction printVersion(): void {\n try {\n const dir = dirname(fileURLToPath(import.meta.url))\n const pkgPath = resolve(dir, '..', 'package.json')\n const pkg = JSON.parse(readFileSync(pkgPath, 'utf8'))\n process.stdout.write(`${pkg.version}\\n`)\n } catch {\n process.stdout.write('unknown\\n')\n }\n}\n\nfunction parseArgs(argv: string[]): CliOptions | null {\n const options: CliOptions = {\n port: Number(process.env.AUTOTEL_DEVTOOLS_PORT || 4318),\n host: process.env.AUTOTEL_DEVTOOLS_HOST || '127.0.0.1',\n title: process.env.AUTOTEL_DEVTOOLS_TITLE,\n }\n\n for (let i = 0; i < argv.length; i++) {\n const arg = argv[i]\n const next = argv[i + 1]\n if (arg === '--help' || arg === '-h') { printHelp(); return null }\n if (arg === '--version' || arg === '-v') { printVersion(); return null }\n if ((arg === '--port' || arg === '-p') && next) { options.port = Number(next); i++; continue }\n if ((arg === '--host' || arg === '-H') && next) { options.host = next; i++; continue }\n if ((arg === '--title' || arg === '-t') && next) { options.title = next; i++; continue }\n }\n\n return options\n}\n\nasync function main(): Promise<void> {\n const options = parseArgs(process.argv.slice(2))\n if (!options) { process.exit(0) }\n\n const httpServer = createServer()\n const wsServer = new DevtoolsServer({ server: httpServer, verbose: true })\n attachDevtoolsRoutes(httpServer, wsServer)\n\n httpServer.listen(options.port, options.host, () => {\n const title = options.title || 'autotel-devtools'\n process.stdout.write(`\\n ${title}\\n\\n`)\n process.stdout.write(` UI: http://${options.host}:${options.port}\\n`)\n process.stdout.write(` Widget: <script src=\"http://${options.host}:${options.port}/widget.js\"></script>\\n`)\n process.stdout.write(` WebSocket: ws://${options.host}:${options.port}/ws\\n`)\n process.stdout.write(` OTLP: http://${options.host}:${options.port}/v1/traces\\n\\n`)\n process.stdout.write(` Set OTEL_EXPORTER_OTLP_PROTOCOL=http/json\\n`)\n process.stdout.write(` Set OTEL_EXPORTER_OTLP_ENDPOINT=http://${options.host}:${options.port}\\n\\n`)\n })\n\n const shutdown = () => {\n wsServer.close().then(() => process.exit(0))\n }\n process.on('SIGINT', shutdown)\n process.on('SIGTERM', shutdown)\n}\n\nmain().catch((error) => {\n process.stderr.write(`[autotel-devtools] failed to start: ${error instanceof Error ? error.message : String(error)}\\n`)\n process.exit(1)\n})\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/server/error-aggregator.ts","../src/server/telemetry-limits.ts","../src/server/server.ts","../src/server/resource-utils.ts","../src/server/otlp.ts","../src/server/http.ts","../src/server/listen.ts","../src/cli.ts"],"names":["createServer","WebSocketServer","WebSocket","resolve","dirname","fileURLToPath","existsSync","readFileSync"],"mappings":";;;;;;;;;;;AAsDO,IAAM,kBAAN,MAAsB;AAAA,EACnB,WAAA,uBAA2C,GAAA,EAAI;AAAA,EAC/C,OAAA;AAAA,EAER,WAAA,CAAY,OAAA,GAAkC,EAAC,EAAG;AAChD,IAAA,IAAA,CAAK,OAAA,GAAU;AAAA,MACb,SAAA,EAAW,QAAQ,SAAA,IAAa,GAAA;AAAA,MAChC,iBAAA,EAAmB,QAAQ,iBAAA,IAAqB,EAAA;AAAA,MAChD,gBAAA,EAAkB,QAAQ,gBAAA,IAAoB,CAAA;AAAA,MAC9C,yBAAA,EAA2B,QAAQ,yBAAA,IAA6B;AAAA,KAClE;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,SAAS,UAAA,EAAyC;AAChD,IAAA,MAAM,WAAA,GAAc,IAAA,CAAK,mBAAA,CAAoB,UAAU,CAAA;AACvD,IAAA,MAAM,QAAA,GAAW,IAAA,CAAK,WAAA,CAAY,GAAA,CAAI,WAAW,CAAA;AAEjD,IAAA,IAAI,QAAA,EAAU;AAEZ,MAAA,QAAA,CAAS,KAAA,EAAA;AACT,MAAA,QAAA,CAAS,WAAW,UAAA,CAAW,SAAA;AAG/B,MAAA,IAAI,CAAC,QAAA,CAAS,cAAA,CAAe,QAAA,CAAS,UAAA,CAAW,OAAO,CAAA,EAAG;AACzD,QAAA,QAAA,CAAS,cAAA,CAAe,IAAA,CAAK,UAAA,CAAW,OAAO,CAAA;AAC/C,QAAA,IAAI,QAAA,CAAS,cAAA,CAAe,MAAA,GAAS,IAAA,CAAK,QAAQ,iBAAA,EAAmB;AACnE,UAAA,QAAA,CAAS,eAAe,KAAA,EAAM;AAAA,QAChC;AAAA,MACF;AAGA,MAAA,IAAI,CAAC,QAAA,CAAS,aAAA,CAAc,QAAA,CAAS,UAAA,CAAW,QAAQ,CAAA,EAAG;AACzD,QAAA,QAAA,CAAS,aAAA,CAAc,IAAA,CAAK,UAAA,CAAW,QAAQ,CAAA;AAC/C,QAAA,IAAI,QAAA,CAAS,aAAA,CAAc,MAAA,GAAS,IAAA,CAAK,QAAQ,gBAAA,EAAkB;AACjE,UAAA,QAAA,CAAS,cAAc,KAAA,EAAM;AAAA,QAC/B;AAAA,MACF;AAEA,MAAA,OAAO,QAAA;AAAA,IACT;AAGA,IAAA,MAAM,QAAA,GAAuB;AAAA,MAC3B,WAAA;AAAA,MACA,IAAA,EAAM,WAAW,KAAA,CAAM,IAAA;AAAA,MACvB,OAAA,EAAS,WAAW,KAAA,CAAM,OAAA;AAAA,MAC1B,UAAA,EAAY,IAAA,CAAK,mBAAA,CAAoB,UAAA,CAAW,MAAM,UAAU,CAAA;AAAA,MAChE,KAAA,EAAO,CAAA;AAAA,MACP,WAAW,UAAA,CAAW,SAAA;AAAA,MACtB,UAAU,UAAA,CAAW,SAAA;AAAA,MACrB,cAAA,EAAgB,CAAC,UAAA,CAAW,OAAO,CAAA;AAAA,MACnC,aAAA,EAAe,CAAC,UAAA,CAAW,QAAQ,CAAA;AAAA,MACnC,SAAS,UAAA,CAAW,OAAA;AAAA,MACpB,YAAY,UAAA,CAAW;AAAA,KACzB;AAGA,IAAA,IAAI,IAAA,CAAK,WAAA,CAAY,IAAA,IAAQ,IAAA,CAAK,QAAQ,SAAA,EAAW;AACnD,MAAA,IAAA,CAAK,gBAAA,EAAiB;AAAA,IACxB;AAEA,IAAA,IAAA,CAAK,WAAA,CAAY,GAAA,CAAI,WAAA,EAAa,QAAQ,CAAA;AAC1C,IAAA,OAAO,QAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,mBAAmB,KAAA,EAAgC;AACjD,IAAA,MAAM,cAA4B,EAAC;AAEnC,IAAA,KAAA,MAAW,IAAA,IAAQ,MAAM,KAAA,EAAO;AAC9B,MAAA,IAAI,IAAA,CAAK,MAAA,CAAO,IAAA,KAAS,OAAA,EAAS;AAChC,QAAA,MAAM,UAAA,GAAa,IAAA,CAAK,oBAAA,CAAqB,IAAA,EAAM,KAAK,CAAA;AACxD,QAAA,IAAI,UAAA,EAAY;AACd,UAAA,MAAM,KAAA,GAAQ,IAAA,CAAK,QAAA,CAAS,UAAU,CAAA;AACtC,UAAA,WAAA,CAAY,KAAK,KAAK,CAAA;AAAA,QACxB;AAAA,MACF;AAAA,IACF;AAEA,IAAA,OAAO,WAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKQ,oBAAA,CACN,MACA,KAAA,EACwB;AAExB,IAAA,MAAM,cAAA,GAAiB,KAAK,MAAA,EAAQ,IAAA,CAAK,CAAC,CAAA,KAAM,CAAA,CAAE,SAAS,WAAW,CAAA;AACtE,IAAA,MAAM,SAAA,GACH,IAAA,CAAK,UAAA,CAAW,gBAAgB,CAAA,IAChC,IAAA,CAAK,UAAA,CAAW,YAAY,CAAA,IAC5B,cAAA,EAAgB,UAAA,GAAa,gBAAgB,CAAA,IAC9C,OAAA;AAEF,IAAA,MAAM,YAAA,GACJ,IAAA,CAAK,MAAA,CAAO,OAAA,IACX,IAAA,CAAK,UAAA,CAAW,mBAAmB,CAAA,IACnC,IAAA,CAAK,UAAA,CAAW,eAAe,CAAA,IAChC,eAAA;AAEF,IAAA,MAAM,UAAA,GACH,IAAA,CAAK,UAAA,CAAW,sBAAsB,CAAA,IACtC,IAAA,CAAK,UAAA,CAAW,iBAAiB,CAAA,IAClC,IAAA,CAAK,sBAAA,CAAuB,IAAI,CAAA;AAElC,IAAA,OAAO;AAAA,MACL,SAAS,KAAA,CAAM,OAAA;AAAA,MACf,QAAQ,IAAA,CAAK,MAAA;AAAA,MACb,UAAU,IAAA,CAAK,IAAA;AAAA,MACf,SAAS,KAAA,CAAM,OAAA;AAAA,MACf,WAAW,IAAA,CAAK,OAAA;AAAA,MAChB,KAAA,EAAO;AAAA,QACL,IAAA,EAAM,SAAA;AAAA,QACN,OAAA,EAAS,YAAA;AAAA,QACT;AAAA,OACF;AAAA,MACA,UAAA,EAAY,IAAA,CAAK,yBAAA,CAA0B,IAAA,CAAK,UAAU;AAAA,KAC5D;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKQ,uBAAuB,IAAA,EAAoC;AACjE,IAAA,IAAI,CAAC,IAAA,CAAK,MAAA,EAAQ,OAAO,MAAA;AAEzB,IAAA,MAAM,cAAA,GAAiB,KAAK,MAAA,CAAO,IAAA,CAAK,CAAC,CAAA,KAAM,CAAA,CAAE,SAAS,WAAW,CAAA;AACrE,IAAA,IAAI,gBAAgB,UAAA,EAAY;AAC9B,MAAA,OACG,eAAe,UAAA,CAAW,sBAAsB,CAAA,IAChD,cAAA,CAAe,WAAW,iBAAiB,CAAA;AAAA,IAEhD;AAEA,IAAA,OAAO,MAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKQ,0BACN,UAAA,EACyB;AACzB,IAAA,MAAM,WAAoC,EAAC;AAC3C,IAAA,MAAM,QAAA,GAAW;AAAA,MACf,aAAA;AAAA,MACA,UAAA;AAAA,MACA,YAAA;AAAA,MACA,kBAAA;AAAA,MACA,WAAA;AAAA,MACA,cAAA;AAAA,MACA,YAAA;AAAA,MACA,aAAA;AAAA,MACA,eAAA;AAAA,MACA,eAAA;AAAA,MACA,SAAA;AAAA,MACA;AAAA,KACF;AAEA,IAAA,KAAA,MAAW,OAAO,QAAA,EAAU;AAC1B,MAAA,IAAI,OAAO,UAAA,EAAY;AACrB,QAAA,QAAA,CAAS,GAAG,CAAA,GAAI,UAAA,CAAW,GAAG,CAAA;AAAA,MAChC;AAAA,IACF;AAEA,IAAA,OAAO,QAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOQ,oBAAoB,UAAA,EAAqC;AAC/D,IAAA,MAAM,KAAA,GAAkB,CAAC,UAAA,CAAW,KAAA,CAAM,IAAI,CAAA;AAE9C,IAAA,IAAI,UAAA,CAAW,MAAM,UAAA,EAAY;AAC/B,MAAA,MAAM,SAAS,IAAA,CAAK,kBAAA;AAAA,QAClB,WAAW,KAAA,CAAM,UAAA;AAAA,QACjB,KAAK,OAAA,CAAQ;AAAA,OACf;AACA,MAAA,KAAA,CAAM,IAAA,CAAK,GAAG,MAAM,CAAA;AAAA,IACtB,CAAA,MAAO;AAEL,MAAA,KAAA,CAAM,KAAK,IAAA,CAAK,gBAAA,CAAiB,UAAA,CAAW,KAAA,CAAM,OAAO,CAAC,CAAA;AAAA,IAC5D;AAGA,IAAA,OAAO,IAAA,CAAK,UAAA,CAAW,KAAA,CAAM,IAAA,CAAK,GAAG,CAAC,CAAA;AAAA,EACxC;AAAA;AAAA;AAAA;AAAA,EAKQ,kBAAA,CAAmB,YAAoB,KAAA,EAAyB;AACtE,IAAA,MAAM,KAAA,GAAQ,UAAA,CAAW,KAAA,CAAM,IAAI,CAAA;AACnC,IAAA,MAAM,SAAmB,EAAC;AAE1B,IAAA,KAAA,MAAW,QAAQ,KAAA,EAAO;AACxB,MAAA,IAAI,MAAA,CAAO,UAAU,KAAA,EAAO;AAG5B,MAAA,MAAM,OAAA,GAAU,KAAK,IAAA,EAAK;AAG1B,MAAA,MAAM,SAAA,GAAY,OAAA,CAAQ,KAAA,CAAM,oCAAoC,CAAA;AACpE,MAAA,IAAI,SAAA,EAAW;AACb,QAAA,MAAA,CAAO,IAAA,CAAK,CAAA,EAAG,SAAA,CAAU,CAAC,CAAC,CAAA,CAAA,EAAI,IAAA,CAAK,iBAAA,CAAkB,SAAA,CAAU,CAAC,CAAC,CAAC,CAAA,CAAE,CAAA;AACrE,QAAA;AAAA,MACF;AAGA,MAAA,MAAM,SAAA,GAAY,OAAA,CAAQ,KAAA,CAAM,wBAAwB,CAAA;AACxD,MAAA,IAAI,SAAA,EAAW;AACb,QAAA,MAAA,CAAO,IAAA,CAAK,aAAa,IAAA,CAAK,iBAAA,CAAkB,UAAU,CAAC,CAAC,CAAC,CAAA,CAAE,CAAA;AAC/D,QAAA;AAAA,MACF;AAGA,MAAA,MAAM,YAAA,GAAe,OAAA,CAAQ,KAAA,CAAM,yBAAyB,CAAA;AAC5D,MAAA,IAAI,YAAA,EAAc;AAChB,QAAA,MAAA,CAAO,IAAA;AAAA,UACL,CAAA,EAAG,YAAA,CAAa,CAAC,CAAC,CAAA,CAAA,EAAI,KAAK,iBAAA,CAAkB,YAAA,CAAa,CAAC,CAAC,CAAC,CAAA;AAAA,SAC/D;AACA,QAAA;AAAA,MACF;AAAA,IACF;AAEA,IAAA,OAAO,MAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKQ,kBAAkB,QAAA,EAA0B;AAElD,IAAA,MAAM,mBAAmB,QAAA,CAAS,KAAA;AAAA,MAChC;AAAA,KACF;AACA,IAAA,IAAI,gBAAA,EAAkB;AACpB,MAAA,OAAO,CAAA,MAAA,EAAS,gBAAA,CAAiB,CAAC,CAAC,CAAA,CAAA;AAAA,IACrC;AAGA,IAAA,OAAO,QAAA,CACJ,OAAA,CAAQ,aAAA,EAAe,MAAM,EAC7B,OAAA,CAAQ,cAAA,EAAgB,OAAO,CAAA,CAC/B,QAAQ,aAAA,EAAe,MAAM,CAAA,CAC7B,OAAA,CAAQ,cAAc,EAAE,CAAA;AAAA,EAC7B;AAAA;AAAA;AAAA;AAAA,EAKQ,iBAAiB,OAAA,EAAyB;AAChD,IAAA,OACE,OAAA,CAEG,UAAA;AAAA,MACC,gEAAA;AAAA,MACA;AAAA,MAGD,UAAA,CAAW,qBAAA,EAAuB,MAAM,CAAA,CAExC,UAAA,CAAW,YAAY,KAAK,CAAA,CAE5B,WAAW,UAAA,EAAY,SAAS,EAChC,UAAA,CAAW,UAAA,EAAY,SAAS,CAAA,CAEhC,KAAA,CAAM,GAAG,GAAG,CAAA;AAAA,EAEnB;AAAA;AAAA;AAAA;AAAA,EAKQ,oBAAoB,UAAA,EAAyC;AACnE,IAAA,IAAI,CAAC,YAAY,OAAO,MAAA;AAExB,IAAA,MAAM,QAAQ,UAAA,CAAW,KAAA,CAAM,IAAI,CAAA,CAAE,KAAA,CAAM,GAAG,EAAE,CAAA;AAChD,IAAA,OAAO,KAAA,CAAM,KAAK,IAAI,CAAA;AAAA,EACxB;AAAA;AAAA;AAAA;AAAA,EAKQ,WAAW,GAAA,EAAqB;AACtC,IAAA,IAAI,IAAA,GAAO,CAAA;AACX,IAAA,KAAA,IAAS,CAAA,GAAI,CAAA,EAAG,CAAA,GAAI,GAAA,CAAI,QAAQ,CAAA,EAAA,EAAK;AACnC,MAAA,MAAM,IAAA,GAAO,GAAA,CAAI,UAAA,CAAW,CAAC,CAAA;AAC7B,MAAA,IAAA,GAAA,CAAQ,IAAA,IAAQ,KAAK,IAAA,GAAO,IAAA;AAC5B,MAAA,IAAA,GAAO,IAAA,GAAO,IAAA;AAAA,IAChB;AACA,IAAA,OAAO,IAAA,CAAK,IAAI,IAAI,CAAA,CAAE,SAAS,EAAE,CAAA,CAAE,QAAA,CAAS,CAAA,EAAG,GAAG,CAAA;AAAA,EACpD;AAAA;AAAA;AAAA;AAAA,EAKQ,gBAAA,GAAyB;AAC/B,IAAA,IAAI,MAAA,GAA2D,IAAA;AAE/D,IAAA,KAAA,MAAW,CAAC,WAAA,EAAa,KAAK,CAAA,IAAK,KAAK,WAAA,EAAa;AACnD,MAAA,IAAI,CAAC,MAAA,IAAU,KAAA,CAAM,QAAA,GAAW,OAAO,QAAA,EAAU;AAC/C,QAAA,MAAA,GAAS,EAAE,WAAA,EAAa,QAAA,EAAU,KAAA,CAAM,QAAA,EAAS;AAAA,MACnD;AAAA,IACF;AAEA,IAAA,IAAI,MAAA,EAAQ;AACV,MAAA,IAAA,CAAK,WAAA,CAAY,MAAA,CAAO,MAAA,CAAO,WAAW,CAAA;AAAA,IAC5C;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,cAAA,GAA+B;AAC7B,IAAA,OAAO,CAAC,GAAG,IAAA,CAAK,WAAA,CAAY,MAAA,EAAQ,CAAA,CAAE,IAAA;AAAA,MACpC,CAAC,CAAA,EAAG,CAAA,KAAM,CAAA,CAAE,WAAW,CAAA,CAAE;AAAA,KAC3B;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,yBAAA,GAA0C;AACxC,IAAA,OAAO,CAAC,GAAG,IAAA,CAAK,WAAA,CAAY,MAAA,EAAQ,CAAA,CAAE,IAAA;AAAA,MACpC,CAAC,CAAA,EAAG,CAAA,KAAM,CAAA,CAAE,QAAQ,CAAA,CAAE;AAAA,KACxB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,cAAc,WAAA,EAA6C;AACzD,IAAA,OAAO,IAAA,CAAK,WAAA,CAAY,GAAA,CAAI,WAAW,CAAA;AAAA,EACzC;AAAA;AAAA;AAAA;AAAA,EAKA,wBAAwB,OAAA,EAA+B;AACrD,IAAA,OAAO,IAAA,CAAK,gBAAe,CAAE,MAAA,CAAO,CAAC,CAAA,KAAM,CAAA,CAAE,YAAY,OAAO,CAAA;AAAA,EAClE;AAAA;AAAA;AAAA;AAAA,EAKA,kBAAA,GAA6B;AAC3B,IAAA,IAAI,KAAA,GAAQ,CAAA;AACZ,IAAA,KAAA,MAAW,KAAA,IAAS,IAAA,CAAK,WAAA,CAAY,MAAA,EAAO,EAAG;AAC7C,MAAA,KAAA,IAAS,KAAA,CAAM,KAAA;AAAA,IACjB;AACA,IAAA,OAAO,KAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,QAAA,GAKE;AACA,IAAA,MAAM,GAAA,GAAM,KAAK,GAAA,EAAI;AACrB,IAAA,MAAM,UAAA,GAAa,GAAA,GAAM,EAAA,GAAK,EAAA,GAAK,GAAA;AAEnC,IAAA,IAAI,YAAA,GAAe,CAAA;AACnB,IAAA,MAAM,SAAA,uBAAgB,GAAA,EAAoB;AAE1C,IAAA,KAAA,MAAW,KAAA,IAAS,IAAA,CAAK,WAAA,CAAY,MAAA,EAAO,EAAG;AAC7C,MAAA,IAAI,KAAA,CAAM,WAAW,UAAA,EAAY;AAC/B,QAAA,YAAA,IAAgB,KAAA,CAAM,KAAA;AAAA,MACxB;AACA,MAAA,SAAA,CAAU,GAAA,CAAI,KAAA,CAAM,IAAA,EAAA,CAAO,SAAA,CAAU,GAAA,CAAI,MAAM,IAAI,CAAA,IAAK,CAAA,IAAK,KAAA,CAAM,KAAK,CAAA;AAAA,IAC1E;AAEA,IAAA,MAAM,aAAA,GAAgB,CAAC,GAAG,SAAA,CAAU,OAAA,EAAS,CAAA,CAC1C,GAAA,CAAI,CAAC,CAAC,IAAA,EAAM,KAAK,CAAA,MAAO,EAAE,IAAA,EAAM,KAAA,EAAM,CAAE,CAAA,CACxC,IAAA,CAAK,CAAC,CAAA,EAAG,CAAA,KAAM,CAAA,CAAE,KAAA,GAAQ,CAAA,CAAE,KAAK,CAAA,CAChC,KAAA,CAAM,GAAG,CAAC,CAAA;AAEb,IAAA,OAAO;AAAA,MACL,WAAA,EAAa,KAAK,WAAA,CAAY,IAAA;AAAA,MAC9B,WAAA,EAAa,KAAK,kBAAA,EAAmB;AAAA,MACrC,YAAA;AAAA,MACA;AAAA,KACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,KAAA,GAAc;AACZ,IAAA,IAAA,CAAK,YAAY,KAAA,EAAM;AAAA,EACzB;AAAA;AAAA;AAAA;AAAA,EAKA,eAAe,QAAA,EAA0B;AACvC,IAAA,MAAM,MAAA,GAAS,IAAA,CAAK,GAAA,EAAI,GAAI,QAAA;AAC5B,IAAA,IAAI,OAAA,GAAU,CAAA;AAEd,IAAA,KAAA,MAAW,CAAC,WAAA,EAAa,KAAK,CAAA,IAAK,KAAK,WAAA,EAAa;AACnD,MAAA,IAAI,KAAA,CAAM,WAAW,MAAA,EAAQ;AAC3B,QAAA,IAAA,CAAK,WAAA,CAAY,OAAO,WAAW,CAAA;AACnC,QAAA,OAAA,EAAA;AAAA,MACF;AAAA,IACF;AAEA,IAAA,OAAO,OAAA;AAAA,EACT;AACF,CAAA;;;AC7cA,IAAM,YAAA,GAAe,GAAA;AAErB,SAAS,WAAW,KAAA,EAA+C;AACjE,EAAA,IAAI,CAAC,OAAO,OAAO,MAAA;AACnB,EAAA,MAAM,MAAA,GAAS,MAAA,CAAO,QAAA,CAAS,KAAA,EAAO,EAAE,CAAA;AACxC,EAAA,OAAO,OAAO,QAAA,CAAS,MAAM,CAAA,IAAK,MAAA,GAAS,IAAI,MAAA,GAAS,MAAA;AAC1D;AAEO,SAAS,sBAAA,CACd,IAAA,GAAmC,EAAC,EACnB;AACjB,EAAA,MAAM,GAAA,GAAM,IAAA,CAAK,GAAA,IAAO,OAAA,CAAQ,GAAA;AAChC,EAAA,MAAM,QAAA,GAAW,KAAK,UAAA,IAAc,YAAA;AAEpC,EAAA,OAAO;AAAA,IACL,eACE,IAAA,CAAK,aAAA,IACL,UAAA,CAAW,GAAA,CAAI,uBAAuB,CAAA,IACtC,QAAA;AAAA,IACF,aACE,IAAA,CAAK,WAAA,IACL,UAAA,CAAW,GAAA,CAAI,qBAAqB,CAAA,IACpC,QAAA;AAAA,IACF,gBACE,IAAA,CAAK,cAAA,IACL,UAAA,CAAW,GAAA,CAAI,wBAAwB,CAAA,IACvC;AAAA,GACJ;AACF;AAEO,SAAS,eAAA,CAAmB,KAAA,EAAY,IAAA,EAAS,KAAA,EAAoB;AAC1E,EAAA,IAAI,KAAA,IAAS,CAAA,EAAG,OAAO,EAAC;AACxB,EAAA,MAAM,IAAA,GAAO,CAAC,GAAG,KAAA,EAAO,IAAI,CAAA;AAC5B,EAAA,OAAO,IAAA,CAAK,SAAS,KAAA,GAAQ,IAAA,CAAK,MAAM,IAAA,CAAK,MAAA,GAAS,KAAK,CAAA,GAAI,IAAA;AACjE;AAEO,SAAS,mBAAA,CACd,KAAA,EACA,QAAA,EACA,KAAA,EACK;AACL,EAAA,IAAI,KAAA,IAAS,KAAK,QAAA,CAAS,MAAA,KAAW,GAAG,OAAO,KAAA,IAAS,CAAA,GAAI,EAAC,GAAI,KAAA;AAClE,EAAA,MAAM,IAAA,GAAO,CAAC,GAAG,KAAA,EAAO,GAAG,QAAQ,CAAA;AACnC,EAAA,OAAO,IAAA,CAAK,SAAS,KAAA,GAAQ,IAAA,CAAK,MAAM,IAAA,CAAK,MAAA,GAAS,KAAK,CAAA,GAAI,IAAA;AACjE;;;AC/BO,IAAM,iBAAN,MAAqB;AAAA,EAClB,GAAA;AAAA,EACA,OAAA,uBAAc,GAAA,EAAe;AAAA,EAC7B,UAAA;AAAA,EACA,SAAsB,EAAC;AAAA,EACvB,OAAkB,EAAC;AAAA,EACnB,UAAwB,EAAC;AAAA,EACzB,eAAA,GAAkB,IAAI,eAAA,EAAgB;AAAA,EACtC,MAAA;AAAA,EACA,OAAA;AAAA,EACA,KAAA;AAAA,EAER,WAAA,CAAY,OAAA,GAAiC,EAAC,EAAG;AAC/C,IAAA,IAAA,CAAK,MAAA,GAAS,uBAAuB,OAAO,CAAA;AAC5C,IAAA,IAAA,CAAK,OAAA,GAAU,QAAQ,OAAA,IAAW,KAAA;AAClC,IAAA,IAAA,CAAK,KAAA,GAAQ,QAAQ,IAAA,IAAQ,IAAA;AAE7B,IAAA,IAAA,CAAK,UAAA,GAAa,OAAA,CAAQ,MAAA,IAAUA,iBAAA,EAAa;AACjD,IAAA,IAAA,CAAK,GAAA,GAAM,IAAIC,kBAAA,CAAgB,EAAE,MAAA,EAAQ,IAAA,CAAK,UAAA,EAAY,IAAA,EAAM,OAAA,CAAQ,IAAA,IAAQ,KAAA,EAAO,CAAA;AAEvF,IAAA,IAAA,CAAK,GAAA,CAAI,EAAA,CAAG,YAAA,EAAc,CAAC,EAAA,KAAO;AAChC,MAAA,IAAA,CAAK,OAAA,CAAQ,IAAI,EAAE,CAAA;AACnB,MAAA,IAAA,CAAK,GAAA,CAAI,CAAA,kBAAA,EAAqB,IAAA,CAAK,OAAA,CAAQ,IAAI,CAAA,OAAA,CAAS,CAAA;AAGxD,MAAA,MAAM,IAAA,GAAO,KAAK,cAAA,EAAe;AACjC,MAAA,IAAI,IAAA,CAAK,MAAA,CAAO,MAAA,GAAS,CAAA,IAAK,IAAA,CAAK,IAAA,CAAK,MAAA,GAAS,CAAA,IAAK,IAAA,CAAK,MAAA,CAAO,MAAA,GAAS,CAAA,EAAG;AAC5E,QAAA,EAAA,CAAG,IAAA,CAAK,IAAA,CAAK,SAAA,CAAU,IAAI,CAAC,CAAA;AAAA,MAC9B;AAEA,MAAA,EAAA,CAAG,EAAA,CAAG,SAAS,MAAM;AACnB,QAAA,IAAA,CAAK,OAAA,CAAQ,OAAO,EAAE,CAAA;AACtB,QAAA,IAAA,CAAK,GAAA,CAAI,CAAA,qBAAA,EAAwB,IAAA,CAAK,OAAA,CAAQ,IAAI,CAAA,OAAA,CAAS,CAAA;AAAA,MAC7D,CAAC,CAAA;AAAA,IACH,CAAC,CAAA;AAGD,IAAA,IAAI,CAAC,QAAQ,MAAA,EAAQ;AACnB,MAAA,IAAA,CAAK,UAAA,CAAW,MAAA,CAAO,IAAA,CAAK,KAAA,EAAO,MAAM;AACvC,QAAA,MAAM,IAAA,GAAO,IAAA,CAAK,UAAA,CAAW,OAAA,EAAQ;AACrC,QAAA,IAAI,QAAQ,OAAO,IAAA,KAAS,QAAA,EAAU,IAAA,CAAK,QAAQ,IAAA,CAAK,IAAA;AACxD,QAAA,IAAA,CAAK,GAAA,CAAI,CAAA,mCAAA,EAAsC,IAAA,CAAK,KAAK,CAAA,CAAE,CAAA;AAAA,MAC7D,CAAC,CAAA;AAAA,IACH;AAAA,EACF;AAAA,EAEA,IAAI,IAAA,GAAe;AACjB,IAAA,MAAM,IAAA,GAAO,IAAA,CAAK,UAAA,CAAW,OAAA,EAAQ;AACrC,IAAA,IAAI,IAAA,IAAQ,OAAO,IAAA,KAAS,QAAA,SAAiB,IAAA,CAAK,IAAA;AAClD,IAAA,OAAO,IAAA,CAAK,KAAA;AAAA,EACd;AAAA,EAEA,IAAI,WAAA,GAAsB;AACxB,IAAA,OAAO,KAAK,OAAA,CAAQ,IAAA;AAAA,EACtB;AAAA,EAEA,SAAS,KAAA,EAAwB;AAE/B,IAAA,MAAM,QAAA,GAAW,KAAK,MAAA,CAAO,IAAA,CAAK,OAAK,CAAA,CAAE,OAAA,KAAY,MAAM,OAAO,CAAA;AAClE,IAAA,IAAI,QAAA,EAAU;AACZ,MAAA,MAAM,eAAA,GAAkB,IAAI,GAAA,CAAI,QAAA,CAAS,MAAM,GAAA,CAAI,CAAA,CAAA,KAAK,CAAA,CAAE,MAAM,CAAC,CAAA;AACjE,MAAA,KAAA,MAAW,IAAA,IAAQ,MAAM,KAAA,EAAO;AAC9B,QAAA,IAAI,CAAC,eAAA,CAAgB,GAAA,CAAI,IAAA,CAAK,MAAM,CAAA,EAAG;AACrC,UAAA,QAAA,CAAS,KAAA,CAAM,KAAK,IAAI,CAAA;AAAA,QAC1B;AAAA,MACF;AACA,MAAA,QAAA,CAAS,YAAY,IAAA,CAAK,GAAA,CAAI,QAAA,CAAS,SAAA,EAAW,MAAM,SAAS,CAAA;AACjE,MAAA,QAAA,CAAS,UAAU,IAAA,CAAK,GAAA,CAAI,QAAA,CAAS,OAAA,EAAS,MAAM,OAAO,CAAA;AAC3D,MAAA,QAAA,CAAS,QAAA,GAAW,QAAA,CAAS,OAAA,GAAU,QAAA,CAAS,SAAA;AAChD,MAAA,IAAI,KAAA,CAAM,MAAA,KAAW,OAAA,EAAS,QAAA,CAAS,MAAA,GAAS,OAAA;AAAA,IAClD,CAAA,MAAO;AACL,MAAA,IAAA,CAAK,MAAA,GAAS,eAAA;AAAA,QACZ,IAAA,CAAK,MAAA;AAAA,QACL,KAAA;AAAA,QACA,KAAK,MAAA,CAAO;AAAA,OACd;AAAA,IACF;AAEA,IAAA,IAAA,CAAK,eAAA,CAAgB,mBAAmB,KAAK,CAAA;AAC7C,IAAA,IAAA,CAAK,UAAU,EAAE,MAAA,EAAQ,CAAC,KAAK,GAAG,OAAA,EAAS,EAAC,EAAG,IAAA,EAAM,EAAC,EAAG,MAAA,EAAQ,KAAK,eAAA,CAAgB,cAAA,IAAkB,CAAA;AAAA,EAC1G;AAAA,EAEA,UAAU,MAAA,EAA2B;AACnC,IAAA,KAAA,MAAW,KAAA,IAAS,MAAA,EAAQ,IAAA,CAAK,QAAA,CAAS,KAAK,CAAA;AAAA,EACjD;AAAA,EAEA,OAAO,GAAA,EAAoB;AACzB,IAAA,IAAA,CAAK,OAAO,eAAA,CAAgB,IAAA,CAAK,MAAM,GAAA,EAAK,IAAA,CAAK,OAAO,WAAW,CAAA;AACnE,IAAA,IAAA,CAAK,SAAA,CAAU,EAAE,MAAA,EAAQ,IAAI,OAAA,EAAS,EAAC,EAAG,IAAA,EAAM,CAAC,GAAG,CAAA,EAAG,MAAA,EAAQ,IAAI,CAAA;AAAA,EACrE;AAAA,EAEA,QAAQ,IAAA,EAAuB;AAC7B,IAAA,IAAA,CAAK,OAAO,mBAAA,CAAoB,IAAA,CAAK,MAAM,IAAA,EAAM,IAAA,CAAK,OAAO,WAAW,CAAA;AACxE,IAAA,IAAA,CAAK,SAAA,CAAU,EAAE,MAAA,EAAQ,EAAC,EAAG,OAAA,EAAS,EAAC,EAAG,IAAA,EAAM,MAAA,EAAQ,EAAC,EAAG,CAAA;AAAA,EAC9D;AAAA,EAEA,UAAU,MAAA,EAA0B;AAClC,IAAA,IAAA,CAAK,OAAA,GAAU,eAAA;AAAA,MACb,IAAA,CAAK,OAAA;AAAA,MACL,MAAA;AAAA,MACA,KAAK,MAAA,CAAO;AAAA,KACd;AACA,IAAA,IAAA,CAAK,SAAA,CAAU,EAAE,MAAA,EAAQ,IAAI,OAAA,EAAS,CAAC,MAAM,CAAA,EAAG,MAAM,EAAC,EAAG,MAAA,EAAQ,IAAI,CAAA;AAAA,EACxE;AAAA,EAEA,cAAA,GAA+B;AAC7B,IAAA,OAAO;AAAA,MACL,QAAQ,IAAA,CAAK,MAAA;AAAA,MACb,SAAS,IAAA,CAAK,OAAA;AAAA,MACd,MAAM,IAAA,CAAK,IAAA;AAAA,MACX,MAAA,EAAQ,IAAA,CAAK,eAAA,CAAgB,cAAA;AAAe,KAC9C;AAAA,EACF;AAAA,EAEA,SAAA,GAAkB;AAChB,IAAA,IAAA,CAAK,SAAS,EAAC;AACf,IAAA,IAAA,CAAK,OAAO,EAAC;AACb,IAAA,IAAA,CAAK,UAAU,EAAC;AAChB,IAAA,IAAA,CAAK,gBAAgB,KAAA,EAAM;AAAA,EAC7B;AAAA,EAEQ,UAAU,IAAA,EAA0B;AAC1C,IAAA,MAAM,GAAA,GAAM,IAAA,CAAK,SAAA,CAAU,IAAI,CAAA;AAC/B,IAAA,KAAA,MAAW,MAAA,IAAU,KAAK,OAAA,EAAS;AACjC,MAAA,IAAI,MAAA,CAAO,UAAA,KAAeC,YAAA,CAAU,IAAA,EAAM;AACxC,QAAA,MAAA,CAAO,KAAK,GAAG,CAAA;AAAA,MACjB;AAAA,IACF;AAAA,EACF;AAAA,EAEQ,IAAI,OAAA,EAAuB;AACjC,IAAA,IAAI,KAAK,OAAA,EAAS,OAAA,CAAQ,GAAA,CAAI,CAAA,mBAAA,EAAsB,OAAO,CAAA,CAAE,CAAA;AAAA,EAC/D;AAAA,EAEA,MAAM,KAAA,GAAuB;AAC3B,IAAA,KAAA,MAAW,MAAA,IAAU,IAAA,CAAK,OAAA,EAAS,MAAA,CAAO,KAAA,EAAM;AAChD,IAAA,IAAA,CAAK,QAAQ,KAAA,EAAM;AACnB,IAAA,IAAA,CAAK,IAAI,KAAA,EAAM;AACf,IAAA,MAAM,IAAI,OAAA,CAAc,CAACC,QAAAA,KAAY,IAAA,CAAK,WAAW,KAAA,CAAM,MAAMA,QAAAA,EAAS,CAAC,CAAA;AAAA,EAC7E;AACF,CAAA;;;ACzKO,SAAS,eAAA,CACd,QAAA,EACA,QAAA,GAAW,SAAA,EACH;AACR,EAAA,IAAI,CAAC,UAAU,OAAO,QAAA;AAEtB,EAAA,MAAM,UAAA,GAAa;AAAA,IACjB,SAAS,cAAc,CAAA;AAAA,IACvB,SAAS,mBAAmB,CAAA;AAAA,IAC5B,SAAS,6BAA6B,CAAA;AAAA,IACtC,SAAS,WAAW,CAAA;AAAA,IACpB,SAAS,gBAAgB,CAAA;AAAA,IACzB,SAAS,yBAAyB;AAAA,GACpC;AAEA,EAAA,KAAA,MAAW,aAAa,UAAA,EAAY;AAClC,IAAA,IAAI,OAAO,SAAA,KAAc,QAAA,IAAY,UAAU,IAAA,EAAK,CAAE,SAAS,CAAA,EAAG;AAChE,MAAA,OAAO,SAAA;AAAA,IACT;AAAA,EACF;AAEA,EAAA,OAAO,QAAA;AACT;;;ACLA,SAAS,iBAAiB,CAAA,EAA2B;AACnD,EAAA,IAAI,CAAC,GAAG,OAAO,MAAA;AACf,EAAA,IAAI,CAAA,CAAE,WAAA,KAAgB,MAAA,EAAW,OAAO,CAAA,CAAE,WAAA;AAC1C,EAAA,IAAI,CAAA,CAAE,SAAA,KAAc,MAAA,EAAW,OAAO,CAAA,CAAE,SAAA;AACxC,EAAA,IAAI,CAAA,CAAE,QAAA,KAAa,MAAA,EAAW,OAAO,OAAO,CAAA,CAAE,QAAA,KAAa,QAAA,GAAW,MAAA,CAAO,CAAA,CAAE,QAAQ,CAAA,GAAI,CAAA,CAAE,QAAA;AAC7F,EAAA,IAAI,CAAA,CAAE,WAAA,KAAgB,MAAA,EAAW,OAAO,CAAA,CAAE,WAAA;AAC1C,EAAA,IAAI,CAAA,CAAE,UAAA,KAAe,MAAA,EAAW,OAAO,CAAA,CAAE,UAAA;AACzC,EAAA,IAAI,CAAA,CAAE,YAAY,MAAA,EAAQ,OAAO,EAAE,UAAA,CAAW,MAAA,CAAO,IAAI,gBAAgB,CAAA;AACzE,EAAA,IAAI,EAAE,WAAA,EAAa,MAAA,SAAe,iBAAA,CAAkB,CAAA,CAAE,YAAY,MAAM,CAAA;AACxE,EAAA,OAAO,MAAA;AACT;AAEA,SAAS,kBAAkB,KAAA,EAAiD;AAC1E,EAAA,MAAM,MAA+B,EAAC;AACtC,EAAA,IAAI,CAAC,OAAO,OAAO,GAAA;AACnB,EAAA,KAAA,MAAW,EAAE,GAAA,EAAK,KAAA,EAAM,IAAK,KAAA,EAAO;AAClC,IAAA,GAAA,CAAI,GAAG,CAAA,GAAI,gBAAA,CAAiB,KAAK,CAAA;AAAA,EACnC;AACA,EAAA,OAAO,GAAA;AACT;AAEA,SAAS,SAAS,IAAA,EAAuB;AACvC,EAAA,IAAI,CAAC,MAAM,OAAO,CAAA;AAClB,EAAA,OAAO,MAAA,CAAO,MAAA,CAAO,IAAI,CAAA,GAAI,QAAU,CAAA;AACzC;AAEA,IAAM,aAAA,GAA2D;AAAA,EAC/D,CAAA,EAAG,UAAA;AAAA,EAAY,CAAA,EAAG,UAAA;AAAA,EAAY,CAAA,EAAG,QAAA;AAAA,EAAU,CAAA,EAAG,QAAA;AAAA,EAAU,CAAA,EAAG,UAAA;AAAA,EAAY,CAAA,EAAG,UAAA;AAAA,EAC1E,kBAAA,EAAoB,UAAA;AAAA,EAAY,gBAAA,EAAkB,QAAA;AAAA,EAClD,gBAAA,EAAkB,QAAA;AAAA,EAAU,kBAAA,EAAoB,UAAA;AAAA,EAAY,kBAAA,EAAoB;AAClF,CAAA;AAEA,SAAS,eAAe,EAAA,EAAqB;AAC3C,EAAA,IAAI,CAAC,IAAI,OAAO,EAAA;AAGhB,EAAA,MAAM,YAAA,GAAe,oBAAoB,IAAA,CAAK,EAAE,KAAK,CAAE,cAAA,CAAe,KAAK,EAAE,CAAA;AAC7E,EAAA,MAAM,gBAAA,GAAmB,YAAA,KAAiB,EAAA,CAAG,MAAA,KAAW,EAAA,IAAM,EAAA,CAAG,MAAA,KAAW,EAAA,IAAM,EAAA,CAAG,MAAA,KAAW,EAAA,IAAM,EAAA,CAAG,MAAA,KAAW,EAAA,CAAA;AACpH,EAAA,IAAI,gBAAA,EAAkB;AACpB,IAAA,IAAI;AACF,MAAA,MAAM,KAAA,GAAQ,MAAA,CAAO,IAAA,CAAK,EAAA,EAAI,QAAQ,CAAA;AACtC,MAAA,OAAO,KAAA,CAAM,SAAS,KAAK,CAAA;AAAA,IAC7B,CAAA,CAAA,MAAQ;AAAA,IAAqB;AAAA,EAC/B;AACA,EAAA,OAAO,EAAA;AACT;AAEO,SAAS,gBAAgB,OAAA,EAA+B;AAC7D,EAAA,IAAI,CAAC,OAAA,IAAW,OAAO,OAAA,KAAY,QAAA,SAAiB,EAAC;AACrD,EAAA,MAAM,EAAE,eAAc,GAAI,OAAA;AAC1B,EAAA,IAAI,CAAC,MAAM,OAAA,CAAQ,aAAa,KAAK,aAAA,CAAc,MAAA,KAAW,CAAA,EAAG,OAAO,EAAC;AAEzE,EAAA,MAAM,QAAA,uBAAe,GAAA,EAAoD;AAEzE,EAAA,KAAA,MAAW,MAAM,aAAA,EAAe;AAC9B,IAAA,MAAM,aAAA,GAAgB,iBAAA,CAAkB,EAAA,CAAG,QAAA,EAAU,UAAU,CAAA;AAC/D,IAAA,MAAM,OAAA,GAAU,MAAA,CAAO,aAAA,CAAc,cAAc,KAAK,SAAS,CAAA;AACjE,IAAA,MAAM,UAAA,GAAa,EAAA,CAAG,UAAA,IAAc,EAAC;AAErC,IAAA,KAAA,MAAW,MAAM,UAAA,EAAY;AAC3B,MAAA,KAAA,MAAW,IAAA,IAAQ,EAAA,CAAG,KAAA,IAAS,EAAC,EAAG;AACjC,QAAA,MAAM,OAAA,GAAU,cAAA,CAAe,IAAA,CAAK,OAAO,CAAA;AAC3C,QAAA,IAAI,CAAC,OAAA,EAAS;AAEd,QAAA,MAAM,OAAA,GAAU,QAAA,CAAS,IAAA,CAAK,iBAAiB,CAAA;AAC/C,QAAA,MAAM,KAAA,GAAQ,QAAA,CAAS,IAAA,CAAK,eAAe,CAAA;AAC3C,QAAA,MAAM,UAAA,GAAa,KAAK,MAAA,EAAQ,IAAA;AAChC,QAAA,IAAI,MAAA,GAAqC,OAAA;AACzC,QAAA,IAAI,UAAA,KAAe,CAAA,IAAK,UAAA,KAAe,gBAAA,EAAkB,MAAA,GAAS,IAAA;AAClE,QAAA,IAAI,UAAA,KAAe,CAAA,IAAK,UAAA,KAAe,mBAAA,EAAqB,MAAA,GAAS,OAAA;AAErE,QAAA,MAAM,QAAA,GAAqB;AAAA,UACzB,OAAA;AAAA,UACA,MAAA,EAAQ,cAAA,CAAe,IAAA,CAAK,MAAM,CAAA;AAAA,UAClC,YAAA,EAAc,cAAA,CAAe,IAAA,CAAK,YAAY,CAAA,IAAK,MAAA;AAAA,UACnD,IAAA,EAAM,KAAK,IAAA,IAAQ,SAAA;AAAA,UACnB,IAAA,EAAM,aAAA,CAAc,IAAA,CAAK,IAAA,IAAQ,CAAC,CAAA,IAAK,UAAA;AAAA,UACvC,SAAA,EAAW,OAAA;AAAA,UACX,OAAA,EAAS,KAAA;AAAA,UACT,UAAU,KAAA,GAAQ,OAAA;AAAA,UAClB,UAAA,EAAY,EAAE,GAAG,aAAA,EAAe,GAAG,iBAAA,CAAkB,IAAA,CAAK,UAAU,CAAA,EAAE;AAAA,UACtE,QAAQ,EAAE,IAAA,EAAM,QAAQ,OAAA,EAAS,IAAA,CAAK,QAAQ,OAAA,EAAQ;AAAA,UACtD,SAAS,IAAA,CAAK,MAAA,IAAU,EAAC,EAAG,GAAA,CAAI,CAAC,CAAA,MAAY;AAAA,YAC3C,IAAA,EAAM,EAAE,IAAA,IAAQ,EAAA;AAAA,YAChB,SAAA,EAAW,QAAA,CAAS,CAAA,CAAE,YAAY,CAAA;AAAA,YAClC,UAAA,EAAY,iBAAA,CAAkB,CAAA,CAAE,UAAU;AAAA,WAC5C,CAAE;AAAA,SACJ;AAEA,QAAA,MAAM,QAAA,GAAW,QAAA,CAAS,GAAA,CAAI,OAAO,CAAA;AACrC,QAAA,IAAI,QAAA,EAAU;AACZ,UAAA,QAAA,CAAS,KAAA,CAAM,KAAK,QAAQ,CAAA;AAAA,QAC9B,CAAA,MAAO;AACL,UAAA,QAAA,CAAS,GAAA,CAAI,SAAS,EAAE,KAAA,EAAO,CAAC,QAAQ,CAAA,EAAG,SAAS,CAAA;AAAA,QACtD;AAAA,MACF;AAAA,IACF;AAAA,EACF;AAEA,EAAA,MAAM,SAAsB,EAAC;AAC7B,EAAA,KAAA,MAAW,CAAC,OAAA,EAAS,EAAE,OAAO,OAAA,EAAS,KAAK,QAAA,EAAU;AACpD,IAAA,MAAM,MAAA,GAAS,MAAM,IAAA,CAAK,CAAC,GAAG,CAAA,KAAM,CAAA,CAAE,SAAA,GAAY,CAAA,CAAE,SAAS,CAAA;AAC7D,IAAA,MAAM,QAAA,GAAW,OAAO,IAAA,CAAK,CAAA,CAAA,KAAK,CAAC,CAAA,CAAE,YAAY,CAAA,IAAK,MAAA,CAAO,CAAC,CAAA;AAC9D,IAAA,MAAM,SAAA,GAAY,KAAK,GAAA,CAAI,GAAG,OAAO,GAAA,CAAI,CAAA,CAAA,KAAK,CAAA,CAAE,SAAS,CAAC,CAAA;AAC1D,IAAA,MAAM,OAAA,GAAU,KAAK,GAAA,CAAI,GAAG,OAAO,GAAA,CAAI,CAAA,CAAA,KAAK,CAAA,CAAE,OAAO,CAAC,CAAA;AACtD,IAAA,MAAM,WAAW,MAAA,CAAO,IAAA,CAAK,OAAK,CAAA,CAAE,MAAA,CAAO,SAAS,OAAO,CAAA;AAE3D,IAAA,MAAA,CAAO,IAAA,CAAK;AAAA,MACV,OAAA;AAAA,MACA,aAAA,EAAe,OAAA,CAAQ,KAAA,CAAM,CAAA,EAAG,EAAE,CAAA;AAAA,MAClC,QAAA;AAAA,MACA,KAAA,EAAO,MAAA;AAAA,MACP,SAAA;AAAA,MACA,OAAA;AAAA,MACA,UAAU,OAAA,GAAU,SAAA;AAAA,MACpB,MAAA,EAAQ,WAAW,OAAA,GAAU,IAAA;AAAA,MAC7B;AAAA,KACD,CAAA;AAAA,EACH;AAEA,EAAA,OAAO,MAAA;AACT;AAEO,SAAS,cAAc,OAAA,EAA6B;AACzD,EAAA,IAAI,CAAC,OAAA,IAAW,OAAO,OAAA,KAAY,QAAA,SAAiB,EAAC;AACrD,EAAA,MAAM,EAAE,cAAa,GAAI,OAAA;AACzB,EAAA,IAAI,CAAC,KAAA,CAAM,OAAA,CAAQ,YAAY,CAAA,SAAU,EAAC;AAE1C,EAAA,MAAM,OAAkB,EAAC;AACzB,EAAA,KAAA,MAAW,MAAM,YAAA,EAAc;AAC7B,IAAA,MAAM,aAAA,GAAgB,iBAAA,CAAkB,EAAA,CAAG,QAAA,EAAU,UAAU,CAAA;AAC/D,IAAA,KAAA,MAAW,EAAA,IAAM,EAAA,CAAG,SAAA,IAAa,EAAC,EAAG;AACnC,MAAA,KAAA,MAAW,GAAA,IAAO,EAAA,CAAG,UAAA,IAAc,EAAC,EAAG;AACrC,QAAA,MAAM,SAAA,GAAY,QAAA,CAAS,GAAA,CAAI,YAAA,IAAgB,IAAI,oBAAoB,CAAA;AACvE,QAAA,MAAM,OAAA,GAAU,cAAA,CAAe,GAAA,CAAI,OAAO,CAAA,IAAK,MAAA;AAC/C,QAAA,MAAM,MAAA,GAAS,cAAA,CAAe,GAAA,CAAI,MAAM,CAAA,IAAK,MAAA;AAC7C,QAAA,MAAM,OAAO,GAAA,CAAI,IAAA,GAAO,gBAAA,CAAiB,GAAA,CAAI,IAAI,CAAA,GAAI,EAAA;AAErD,QAAA,IAAA,CAAK,IAAA,CAAK;AAAA,UACR,EAAA,EAAI,CAAA,EAAG,OAAA,IAAW,UAAU,CAAA,CAAA,EAAI,MAAA,IAAU,SAAS,CAAA,CAAA,EAAI,SAAS,CAAA,CAAA,EAAI,GAAA,CAAI,cAAA,IAAkB,CAAC,CAAA,CAAA;AAAA,UAC3F,OAAA;AAAA,UACA,MAAA;AAAA,UACA,YAAA,EAAc,gBAAgB,aAAwC,CAAA;AAAA,UACtE,cAAc,GAAA,CAAI,YAAA;AAAA,UAClB,gBAAgB,GAAA,CAAI,cAAA;AAAA,UACpB,IAAA,EAAM,OAAO,IAAA,KAAS,QAAA,GAAW,IAAA,GAAQ,IAAA;AAAA,UACzC,SAAA;AAAA,UACA,UAAA,EAAY,iBAAA,CAAkB,GAAA,CAAI,UAAU,CAAA;AAAA,UAC5C,QAAA,EAAU;AAAA,SACX,CAAA;AAAA,MACH;AAAA,IACF;AAAA,EACF;AAEA,EAAA,OAAO,IAAA;AACT;AAEO,SAAS,iBAAiB,OAAA,EAA0B;AACzD,EAAA,IAAI,CAAC,OAAA,IAAW,OAAO,OAAA,KAAY,UAAU,OAAO,CAAA;AACpD,EAAA,MAAM,EAAE,iBAAgB,GAAI,OAAA;AAC5B,EAAA,IAAI,CAAC,KAAA,CAAM,OAAA,CAAQ,eAAe,GAAG,OAAO,CAAA;AAC5C,EAAA,IAAI,KAAA,GAAQ,CAAA;AACZ,EAAA,KAAA,MAAW,MAAM,eAAA,EAAiB;AAChC,IAAA,KAAA,MAAW,EAAA,IAAM,EAAA,CAAG,YAAA,IAAgB,EAAC,EAAG;AACtC,MAAA,KAAA,IAAA,CAAU,EAAA,CAAG,OAAA,IAAW,EAAC,EAAG,MAAA;AAAA,IAC9B;AAAA,EACF;AACA,EAAA,OAAO,KAAA;AACT;AAEA,eAAsB,aAAa,GAAA,EAAwC;AACzE,EAAA,OAAO,IAAI,OAAA,CAAQ,CAACA,QAAAA,EAAS,MAAA,KAAW;AACtC,IAAA,MAAM,SAAmB,EAAC;AAC1B,IAAA,GAAA,CAAI,GAAG,MAAA,EAAQ,CAAC,UAAkB,MAAA,CAAO,IAAA,CAAK,KAAK,CAAC,CAAA;AACpD,IAAA,GAAA,CAAI,EAAA,CAAG,OAAO,MAAM;AAClB,MAAA,IAAI;AACF,QAAAA,QAAAA,CAAQ,KAAK,KAAA,CAAM,MAAA,CAAO,OAAO,MAAM,CAAA,CAAE,QAAA,EAAU,CAAC,CAAA;AAAA,MACtD,CAAA,CAAA,MAAQ;AACN,QAAA,MAAA,CAAO,IAAI,KAAA,CAAM,cAAc,CAAC,CAAA;AAAA,MAClC;AAAA,IACF,CAAC,CAAA;AACD,IAAA,GAAA,CAAI,EAAA,CAAG,SAAS,MAAM,CAAA;AAAA,EACxB,CAAC,CAAA;AACH;AAEO,SAAS,QAAA,CAAS,GAAA,EAAqB,MAAA,EAAgB,IAAA,EAAqC;AACjG,EAAA,MAAM,IAAA,GAAO,IAAA,CAAK,SAAA,CAAU,IAAI,CAAA;AAChC,EAAA,GAAA,CAAI,SAAA,CAAU,MAAA,EAAQ,EAAE,cAAA,EAAgB,kBAAA,EAAoB,kBAAkB,MAAA,CAAO,UAAA,CAAW,IAAI,CAAA,EAAG,CAAA;AACvG,EAAA,GAAA,CAAI,IAAI,IAAI,CAAA;AACd;;;ACjMA,SAAS,eAAA,GAA0B;AACjC,EAAA,IAAI,GAAA,GAAMC,YAAA,CAAQC,iBAAA,CAAc,yPAAe,CAAC,CAAA;AAChD,EAAA,KAAA,IAAS,CAAA,GAAI,CAAA,EAAG,CAAA,GAAI,CAAA,EAAG,CAAA,EAAA,EAAK;AAC1B,IAAA,IAAIC,cAAWH,YAAA,CAAQ,GAAA,EAAK,cAAc,CAAC,GAAG,OAAO,GAAA;AACrD,IAAA,GAAA,GAAMC,aAAQ,GAAG,CAAA;AAAA,EACnB;AACA,EAAA,OAAO,GAAA;AACT;AAEA,IAAM,aAAA,GAAgB,CAAA,4UAAA,CAAA;AAEtB,IAAI,cAAA,GAAgC,IAAA;AACpC,SAAS,WAAA,GAAsB;AAC7B,EAAA,IAAI,CAAC,cAAA,EAAgB;AACnB,IAAA,MAAM,UAAU,eAAA,EAAgB;AAChC,IAAA,MAAM,UAAA,GAAa;AAAA,MACjBD,YAAA,CAAQ,OAAA,EAAS,MAAA,EAAQ,kBAAkB,CAAA;AAAA,MAC3CA,YAAA,CAAQ,SAAS,kBAAkB;AAAA,KACrC;AACA,IAAA,KAAA,MAAW,aAAa,UAAA,EAAY;AAClC,MAAA,IAAI;AACF,QAAA,cAAA,GAAiBI,eAAA,CAAa,WAAW,MAAM,CAAA;AAC/C,QAAA;AAAA,MACF,CAAA,CAAA,MAAQ;AAAA,MAAiB;AAAA,IAC3B;AACA,IAAA,IAAI,CAAC,cAAA,EAAgB;AACnB,MAAA,cAAA,GAAiB,mDAAA;AAAA,IACnB;AAAA,EACF;AACA,EAAA,OAAO,cAAA;AACT;AAEO,SAAS,oBAAA,CAAqB,YAAoB,QAAA,EAAgC;AACvF,EAAA,UAAA,CAAW,EAAA,CAAG,SAAA,EAAW,OAAO,GAAA,EAAsB,GAAA,KAAwB;AAE5E,IAAA,GAAA,CAAI,SAAA,CAAU,+BAA+B,GAAG,CAAA;AAChD,IAAA,GAAA,CAAI,SAAA,CAAU,gCAAgC,4BAA4B,CAAA;AAC1E,IAAA,GAAA,CAAI,SAAA,CAAU,gCAAgC,cAAc,CAAA;AAE5D,IAAA,IAAI,GAAA,CAAI,WAAW,SAAA,EAAW;AAC5B,MAAA,GAAA,CAAI,UAAU,GAAG,CAAA;AACjB,MAAA,GAAA,CAAI,GAAA,EAAI;AACR,MAAA;AAAA,IACF;AAEA,IAAA,MAAM,GAAA,GAAM,IAAI,GAAA,IAAO,GAAA;AAGvB,IAAA,IAAI,GAAA,CAAI,MAAA,KAAW,KAAA,IAAS,GAAA,KAAQ,GAAA,EAAK;AACvC,MAAA,GAAA,CAAI,SAAA,CAAU,GAAA,EAAK,EAAE,cAAA,EAAgB,0BAAA,EAA4B,kBAAkB,MAAA,CAAO,UAAA,CAAW,aAAa,CAAA,EAAG,CAAA;AACrH,MAAA,GAAA,CAAI,IAAI,aAAa,CAAA;AACrB,MAAA;AAAA,IACF;AAGA,IAAA,IAAI,IAAI,MAAA,KAAW,KAAA,IAAS,GAAA,CAAI,UAAA,CAAW,YAAY,CAAA,EAAG;AACxD,MAAA,MAAM,KAAK,WAAA,EAAY;AACvB,MAAA,GAAA,CAAI,SAAA,CAAU,GAAA,EAAK,EAAE,cAAA,EAAgB,uCAAA,EAAyC,kBAAkB,MAAA,CAAO,UAAA,CAAW,EAAE,CAAA,EAAG,CAAA;AACvH,MAAA,GAAA,CAAI,IAAI,EAAE,CAAA;AACV,MAAA;AAAA,IACF;AAGA,IAAA,IAAI,GAAA,CAAI,MAAA,KAAW,KAAA,IAAS,GAAA,KAAQ,UAAA,EAAY;AAC9C,MAAA,QAAA,CAAS,GAAA,EAAK,KAAK,EAAE,EAAA,EAAI,MAAM,OAAA,EAAS,QAAA,CAAS,aAAa,CAAA;AAC9D,MAAA;AAAA,IACF;AAOA,IAAA,IAAI,GAAA,CAAI,MAAA,KAAW,KAAA,IAAS,GAAA,KAAQ,YAAA,EAAc;AAChD,MAAA,MAAM,IAAA,GAAO,SAAS,cAAA,EAAe;AACrC,MAAA,QAAA,CAAS,GAAA,EAAK,GAAA,EAAK,EAAE,MAAA,EAAQ,IAAA,CAAK,QAAQ,KAAA,EAAO,IAAA,CAAK,MAAA,CAAO,MAAA,EAAQ,CAAA;AACrE,MAAA;AAAA,IACF;AAIA,IAAA,IAAI,GAAA,CAAI,MAAA,KAAW,QAAA,IAAY,GAAA,KAAQ,YAAA,EAAc;AACnD,MAAA,QAAA,CAAS,SAAA,EAAU;AACnB,MAAA,QAAA,CAAS,GAAA,EAAK,GAAA,EAAK,EAAE,OAAA,EAAS,MAAM,CAAA;AACpC,MAAA;AAAA,IACF;AAGA,IAAA,IAAI,GAAA,CAAI,MAAA,KAAW,MAAA,IAAU,GAAA,KAAQ,YAAA,EAAc;AACjD,MAAA,IAAI;AACF,QAAA,MAAM,OAAA,GAAU,MAAM,YAAA,CAAa,GAAG,CAAA;AACtC,QAAA,MAAM,MAAA,GAAS,gBAAgB,OAAO,CAAA;AACtC,QAAA,QAAA,CAAS,UAAU,MAAM,CAAA;AACzB,QAAA,QAAA,CAAS,KAAK,GAAA,EAAK,EAAE,cAAA,EAAgB,MAAA,CAAO,QAAQ,CAAA;AAAA,MACtD,SAAS,CAAA,EAAG;AACV,QAAA,QAAA,CAAS,GAAA,EAAK,GAAA,EAAK,EAAE,KAAA,EAAO,mBAAA,EAAqB,OAAA,EAAS,CAAA,YAAa,KAAA,GAAQ,CAAA,CAAE,OAAA,GAAU,MAAA,CAAO,CAAC,GAAG,CAAA;AAAA,MACxG;AACA,MAAA;AAAA,IACF;AAGA,IAAA,IAAI,GAAA,CAAI,MAAA,KAAW,MAAA,IAAU,GAAA,KAAQ,UAAA,EAAY;AAC/C,MAAA,IAAI;AACF,QAAA,MAAM,OAAA,GAAU,MAAM,YAAA,CAAa,GAAG,CAAA;AACtC,QAAA,MAAM,IAAA,GAAO,cAAc,OAAO,CAAA;AAClC,QAAA,QAAA,CAAS,QAAQ,IAAI,CAAA;AACrB,QAAA,QAAA,CAAS,KAAK,GAAA,EAAK,EAAE,YAAA,EAAc,IAAA,CAAK,QAAQ,CAAA;AAAA,MAClD,SAAS,CAAA,EAAG;AACV,QAAA,QAAA,CAAS,GAAA,EAAK,GAAA,EAAK,EAAE,KAAA,EAAO,mBAAA,EAAqB,OAAA,EAAS,CAAA,YAAa,KAAA,GAAQ,CAAA,CAAE,OAAA,GAAU,MAAA,CAAO,CAAC,GAAG,CAAA;AAAA,MACxG;AACA,MAAA;AAAA,IACF;AAGA,IAAA,IAAI,GAAA,CAAI,MAAA,KAAW,MAAA,IAAU,GAAA,KAAQ,aAAA,EAAe;AAClD,MAAA,IAAI;AACF,QAAA,MAAM,OAAA,GAAU,MAAM,YAAA,CAAa,GAAG,CAAA;AACtC,QAAA,MAAM,KAAA,GAAQ,iBAAiB,OAAO,CAAA;AACtC,QAAA,QAAA,CAAS,GAAA,EAAK,GAAA,EAAK,EAAE,eAAA,EAAiB,OAAO,CAAA;AAAA,MAC/C,SAAS,CAAA,EAAG;AACV,QAAA,QAAA,CAAS,GAAA,EAAK,GAAA,EAAK,EAAE,KAAA,EAAO,mBAAA,EAAqB,OAAA,EAAS,CAAA,YAAa,KAAA,GAAQ,CAAA,CAAE,OAAA,GAAU,MAAA,CAAO,CAAC,GAAG,CAAA;AAAA,MACxG;AACA,MAAA;AAAA,IACF;AAEA,IAAA,QAAA,CAAS,GAAA,EAAK,GAAA,EAAK,EAAE,KAAA,EAAO,aAAa,CAAA;AAAA,EAC3C,CAAC,CAAA;AACH;ACzIA,IAAM,2BAAW,IAAI,GAAA,CAAI,CAAC,WAAA,EAAa,WAAA,EAAa,KAAK,CAAC,CAAA;AAUnD,SAAS,aAAA,CAAc,MAAc,IAAA,EAAsB;AAChE,EAAA,OAAO,IAAA,CAAK,QAAA,CAAS,GAAG,CAAA,GAAI,CAAA,CAAA,EAAI,IAAI,CAAA,EAAA,EAAK,IAAI,CAAA,CAAA,GAAK,CAAA,EAAG,IAAI,CAAA,CAAA,EAAI,IAAI,CAAA,CAAA;AACnE;AAkBO,SAAS,wBAAwB,IAAA,EAKlB;AACpB,EAAA,MAAM,EAAE,OAAA,EAAS,IAAA,EAAM,IAAA,EAAM,iBAAgB,GAAI,IAAA;AACjD,EAAA,IAAI,OAAA;AAEJ,EAAA,MAAM,QAAQ,IAAI,OAAA;AAAA,IAChB,CAACJ,QAAAA,KAAY;AACX,MAAA,MAAM,YAAsB,EAAC;AAC7B,MAAA,MAAM,WAAqB,EAAC;AAG5B,MAAA,MAAM,WAAA,GAAc,IAAA,KAAS,WAAA,GAAc,WAAA,GAAc,IAAA;AAEzD,MAAA,OAAA,CAAQ,MAAA,CAAO,IAAA,EAAM,WAAA,EAAa,MAAM;AACtC,QAAA,MAAM,IAAA,GAAO,QAAQ,OAAA,EAAQ;AAC7B,QAAA,MAAM,eACJ,IAAA,IAAQ,OAAO,IAAA,KAAS,QAAA,GAAW,KAAK,IAAA,GAAO,IAAA;AACjD,QAAA,SAAA,CAAU,IAAA,CAAK,aAAA,CAAc,WAAA,EAAa,YAAY,CAAC,CAAA;AAEvD,QAAA,IAAI,CAAC,QAAA,CAAS,GAAA,CAAI,IAAI,CAAA,EAAG;AACvB,UAAAA,QAAAA,CAAQ,EAAE,SAAA,EAAW,QAAA,EAAU,CAAA;AAC/B,UAAA;AAAA,QACF;AAEA,QAAA,MAAM,WAAA,GAAc,WAAA,KAAgB,KAAA,GAAQ,WAAA,GAAc,KAAA;AAC1D,QAAA,MAAM,IAAIH,iBAAAA,EAAa;AACvB,QAAA,eAAA,CAAgB,CAAC,CAAA;AAEjB,QAAA,MAAM,OAAA,GAAU,CAAC,CAAA,KAAa;AAC5B,UAAA,CAAA,CAAE,KAAA,EAAM;AACR,UAAA,QAAA,CAAS,IAAA;AAAA,YACP,CAAA,oBAAA,EAAuB,aAAA,CAAc,WAAA,EAAa,YAAY,CAAC,CAAA,EAAA,EAAK,CAAA,CAAE,OAAO,CAAA,qBAAA,EACtD,WAAA,KAAgB,KAAA,GAAQ,MAAA,GAAS,MAAM,CAAA,qCAAA;AAAA,WAChE;AACA,UAAAG,QAAAA,CAAQ,EAAE,SAAA,EAAW,QAAA,EAAU,CAAA;AAAA,QACjC,CAAA;AACA,QAAA,CAAA,CAAE,IAAA,CAAK,SAAS,OAAO,CAAA;AACvB,QAAA,CAAA,CAAE,MAAA,CAAO,YAAA,EAAc,WAAA,EAAa,MAAM;AACxC,UAAA,CAAA,CAAE,GAAA,CAAI,SAAS,OAAO,CAAA;AACtB,UAAA,OAAA,GAAU,CAAA;AACV,UAAA,SAAA,CAAU,IAAA,CAAK,aAAA,CAAc,WAAA,EAAa,YAAY,CAAC,CAAA;AACvD,UAAAA,QAAAA,CAAQ,EAAE,SAAA,EAAW,QAAA,EAAU,CAAA;AAAA,QACjC,CAAC,CAAA;AAAA,MACH,CAAC,CAAA;AAAA,IACH;AAAA,GACF;AAEA,EAAA,OAAO;AAAA,IACL,KAAA;AAAA,IACA,YAAA,EAAc,MACZ,IAAI,OAAA,CAAc,CAAC,GAAA,KAAQ;AACzB,MAAA,IAAI,CAAC,OAAA,EAAS,OAAO,GAAA,EAAI;AACzB,MAAA,OAAA,CAAQ,KAAA,CAAM,MAAM,GAAA,EAAK,CAAA;AAAA,IAC3B,CAAC;AAAA,GACL;AACF;;;AC5EA,SAAS,SAAA,GAAkB;AACzB,EAAA,OAAA,CAAQ,MAAA,CAAO,KAAA;AAAA,IACb,CAAA;;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;;AAAA;AAAA;;AAAA;AAAA;;AAAA;AAAA;;AAAA;AAAA,GAoCF;AACF;AAEA,SAAS,YAAA,GAAqB;AAC5B,EAAA,IAAI;AACF,IAAA,MAAM,GAAA,GAAMC,YAAAA,CAAQC,iBAAAA,CAAc,yPAAe,CAAC,CAAA;AAClD,IAAA,MAAM,OAAA,GAAUF,YAAAA,CAAQ,GAAA,EAAK,IAAA,EAAM,cAAc,CAAA;AACjD,IAAA,MAAM,MAAM,IAAA,CAAK,KAAA,CAAMI,eAAAA,CAAa,OAAA,EAAS,MAAM,CAAC,CAAA;AACpD,IAAA,OAAA,CAAQ,MAAA,CAAO,KAAA,CAAM,CAAA,EAAG,GAAA,CAAI,OAAO;AAAA,CAAI,CAAA;AAAA,EACzC,CAAA,CAAA,MAAQ;AACN,IAAA,OAAA,CAAQ,MAAA,CAAO,MAAM,WAAW,CAAA;AAAA,EAClC;AACF;AAEA,SAAS,UAAU,IAAA,EAAmC;AACpD,EAAA,MAAM,OAAA,GAAsB;AAAA,IAC1B,IAAA,EAAM,MAAA,CAAO,OAAA,CAAQ,GAAA,CAAI,yBAAyB,IAAI,CAAA;AAAA,IACtD,IAAA,EAAM,OAAA,CAAQ,GAAA,CAAI,qBAAA,IAAyB,WAAA;AAAA,IAC3C,KAAA,EAAO,QAAQ,GAAA,CAAI;AAAA,GACrB;AAEA,EAAA,KAAA,IAAS,CAAA,GAAI,CAAA,EAAG,CAAA,GAAI,IAAA,CAAK,QAAQ,CAAA,EAAA,EAAK;AACpC,IAAA,MAAM,GAAA,GAAM,KAAK,CAAC,CAAA;AAClB,IAAA,MAAM,IAAA,GAAO,IAAA,CAAK,CAAA,GAAI,CAAC,CAAA;AACvB,IAAA,IAAI,GAAA,KAAQ,QAAA,IAAY,GAAA,KAAQ,IAAA,EAAM;AAAE,MAAA,SAAA,EAAU;AAAG,MAAA,OAAO,IAAA;AAAA,IAAK;AACjE,IAAA,IAAI,GAAA,KAAQ,WAAA,IAAe,GAAA,KAAQ,IAAA,EAAM;AAAE,MAAA,YAAA,EAAa;AAAG,MAAA,OAAO,IAAA;AAAA,IAAK;AACvE,IAAA,IAAA,CAAK,GAAA,KAAQ,QAAA,IAAY,GAAA,KAAQ,IAAA,KAAS,IAAA,EAAM;AAAE,MAAA,OAAA,CAAQ,IAAA,GAAO,OAAO,IAAI,CAAA;AAAG,MAAA,CAAA,EAAA;AAAK,MAAA;AAAA,IAAS;AAC7F,IAAA,IAAA,CAAK,GAAA,KAAQ,QAAA,IAAY,GAAA,KAAQ,IAAA,KAAS,IAAA,EAAM;AAAE,MAAA,OAAA,CAAQ,IAAA,GAAO,IAAA;AAAM,MAAA,CAAA,EAAA;AAAK,MAAA;AAAA,IAAS;AACrF,IAAA,IAAA,CAAK,GAAA,KAAQ,SAAA,IAAa,GAAA,KAAQ,IAAA,KAAS,IAAA,EAAM;AAAE,MAAA,OAAA,CAAQ,KAAA,GAAQ,IAAA;AAAM,MAAA,CAAA,EAAA;AAAK,MAAA;AAAA,IAAS;AAAA,EACzF;AAEA,EAAA,OAAO,OAAA;AACT;AAEA,eAAe,IAAA,GAAsB;AACnC,EAAA,MAAM,UAAU,SAAA,CAAU,OAAA,CAAQ,IAAA,CAAK,KAAA,CAAM,CAAC,CAAC,CAAA;AAC/C,EAAA,IAAI,CAAC,OAAA,EAAS;AAAE,IAAA,OAAA,CAAQ,KAAK,CAAC,CAAA;AAAA,EAAE;AAEhC,EAAA,MAAM,aAAaP,iBAAAA,EAAa;AAChC,EAAA,MAAM,QAAA,GAAW,IAAI,cAAA,CAAe,EAAE,QAAQ,UAAA,EAAY,OAAA,EAAS,MAAM,CAAA;AACzE,EAAA,oBAAA,CAAqB,YAAY,QAAQ,CAAA;AAEzC,EAAA,MAAM,YAAY,uBAAA,CAAwB;AAAA,IACxC,OAAA,EAAS,UAAA;AAAA,IACT,MAAM,OAAA,CAAQ,IAAA;AAAA,IACd,MAAM,OAAA,CAAQ,IAAA;AAAA,IACd,eAAA,EAAiB,CAAC,CAAA,KAAM,oBAAA,CAAqB,GAAG,QAAQ;AAAA,GACzD,CAAA;AAED,EAAA,MAAM,EAAE,SAAA,EAAW,QAAA,EAAS,GAAI,MAAM,SAAA,CAAU,KAAA;AAChD,EAAA,MAAM,MAAA,GAAS,CAAA,OAAA,EAAU,OAAA,CAAQ,IAAA,KAAS,WAAA,GAAc,cAAc,OAAA,CAAQ,IAAI,CAAA,CAAA,EAAI,OAAA,CAAQ,IAAI,CAAA,CAAA;AAClG,EAAA,MAAM,KAAA,GAAQ,QAAQ,KAAA,IAAS,kBAAA;AAC/B,EAAA,OAAA,CAAQ,OAAO,KAAA,CAAM;AAAA,EAAA,EAAO,KAAK;;AAAA,CAAM,CAAA;AACvC,EAAA,OAAA,CAAQ,OAAO,KAAA,CAAM,CAAA,aAAA,EAAgB,SAAA,CAAU,IAAA,CAAK,OAAO,CAAC;AAAA,CAAI,CAAA;AAChE,EAAA,OAAA,CAAQ,MAAA,CAAO,KAAA,CAAM,CAAA,aAAA,EAAgB,MAAM;AAAA,CAAI,CAAA;AAC/C,EAAA,OAAA,CAAQ,MAAA,CAAO,KAAA,CAAM,CAAA,0BAAA,EAA6B,MAAM,CAAA;AAAA,CAAyB,CAAA;AACjF,EAAA,OAAA,CAAQ,OAAO,KAAA,CAAM,CAAA,aAAA,EAAgB,OAAO,OAAA,CAAQ,MAAA,EAAQ,IAAI,CAAC,CAAA;AAAA,CAAO,CAAA;AACxE,EAAA,OAAA,CAAQ,MAAA,CAAO,KAAA,CAAM,CAAA,aAAA,EAAgB,MAAM,CAAA;;AAAA,CAAgB,CAAA;AAC3D,EAAA,OAAA,CAAQ,OAAO,KAAA,CAAM,CAAA;AAAA,CAA+C,CAAA;AACpE,EAAA,OAAA,CAAQ,MAAA,CAAO,KAAA,CAAM,CAAA,kCAAA,EAAqC,MAAM;;AAAA,CAAM,CAAA;AAItE,EAAA,OAAA,CAAQ,MAAA,CAAO,KAAA,CAAM,CAAA,4BAAA,EAA+B,MAAM,CAAA;;AAAA,CAAgB,CAAA;AAC1E,EAAA,KAAA,MAAW,KAAK,QAAA,EAAU;AACxB,IAAA,OAAA,CAAQ,MAAA,CAAO,KAAA,CAAM,CAAA,SAAA,EAAO,CAAC;AAAA,CAAI,CAAA;AAAA,EACnC;AACA,EAAA,IAAI,SAAS,MAAA,GAAS,CAAA,EAAG,OAAA,CAAQ,MAAA,CAAO,MAAM,IAAI,CAAA;AAElD,EAAA,MAAM,WAAW,MAAM;AACrB,IAAA,OAAA,CAAQ,GAAA,CAAI,CAAC,QAAA,CAAS,KAAA,IAAS,SAAA,CAAU,YAAA,EAAc,CAAC,CAAA,CAAE,IAAA;AAAA,MAAK,MAC7D,OAAA,CAAQ,IAAA,CAAK,CAAC;AAAA,KAChB;AAAA,EACF,CAAA;AACA,EAAA,OAAA,CAAQ,EAAA,CAAG,UAAU,QAAQ,CAAA;AAC7B,EAAA,OAAA,CAAQ,EAAA,CAAG,WAAW,QAAQ,CAAA;AAChC;AAEA,IAAA,EAAK,CAAE,KAAA,CAAM,CAAC,KAAA,KAAU;AACtB,EAAA,OAAA,CAAQ,MAAA,CAAO,MAAM,CAAA,oCAAA,EAAuC,KAAA,YAAiB,QAAQ,KAAA,CAAM,OAAA,GAAU,MAAA,CAAO,KAAK,CAAC;AAAA,CAAI,CAAA;AACtH,EAAA,OAAA,CAAQ,KAAK,CAAC,CAAA;AAChB,CAAC,CAAA","file":"cli.cjs","sourcesContent":["/**\n * Error Aggregator\n *\n * Groups similar errors together based on stack trace fingerprinting.\n * Tracks error frequency, first/last occurrence, and affected traces.\n *\n * @example\n * ```typescript\n * const aggregator = new ErrorAggregator({ maxGroups: 100 });\n *\n * // Add errors from spans\n * aggregator.addError({\n * traceId: '123',\n * spanId: '456',\n * spanName: 'api.createUser',\n * service: 'user-service',\n * timestamp: Date.now(),\n * error: {\n * type: 'ValidationError',\n * message: 'Invalid email format',\n * stackTrace: 'Error: Invalid email...'\n * }\n * });\n *\n * // Get aggregated error groups\n * const groups = aggregator.getErrorGroups();\n * ```\n */\n\nimport type { ErrorGroup, ErrorOccurrence, SpanData, TraceData } from './types';\n\nexport interface ErrorAggregatorOptions {\n /**\n * Maximum number of error groups to track (default: 100)\n * Oldest groups are evicted when limit is reached\n */\n maxGroups?: number;\n\n /**\n * Maximum number of affected traces to keep per group (default: 10)\n */\n maxAffectedTraces?: number;\n\n /**\n * Maximum number of affected span names to keep per group (default: 5)\n */\n maxAffectedSpans?: number;\n\n /**\n * Number of stack frames to use for fingerprinting (default: 5)\n */\n stackFramesForFingerprint?: number;\n}\n\nexport class ErrorAggregator {\n private errorGroups: Map<string, ErrorGroup> = new Map();\n private options: Required<ErrorAggregatorOptions>;\n\n constructor(options: ErrorAggregatorOptions = {}) {\n this.options = {\n maxGroups: options.maxGroups ?? 100,\n maxAffectedTraces: options.maxAffectedTraces ?? 10,\n maxAffectedSpans: options.maxAffectedSpans ?? 5,\n stackFramesForFingerprint: options.stackFramesForFingerprint ?? 5,\n };\n }\n\n /**\n * Add an error occurrence to the aggregator\n */\n addError(occurrence: ErrorOccurrence): ErrorGroup {\n const fingerprint = this.generateFingerprint(occurrence);\n const existing = this.errorGroups.get(fingerprint);\n\n if (existing) {\n // Update existing group\n existing.count++;\n existing.lastSeen = occurrence.timestamp;\n\n // Add trace ID if not already present (keep last N)\n if (!existing.affectedTraces.includes(occurrence.traceId)) {\n existing.affectedTraces.push(occurrence.traceId);\n if (existing.affectedTraces.length > this.options.maxAffectedTraces) {\n existing.affectedTraces.shift();\n }\n }\n\n // Add span name if not already present\n if (!existing.affectedSpans.includes(occurrence.spanName)) {\n existing.affectedSpans.push(occurrence.spanName);\n if (existing.affectedSpans.length > this.options.maxAffectedSpans) {\n existing.affectedSpans.shift();\n }\n }\n\n return existing;\n }\n\n // Create new group\n const newGroup: ErrorGroup = {\n fingerprint,\n type: occurrence.error.type,\n message: occurrence.error.message,\n stackTrace: this.normalizeStackTrace(occurrence.error.stackTrace),\n count: 1,\n firstSeen: occurrence.timestamp,\n lastSeen: occurrence.timestamp,\n affectedTraces: [occurrence.traceId],\n affectedSpans: [occurrence.spanName],\n service: occurrence.service,\n attributes: occurrence.attributes,\n };\n\n // Evict oldest group if at capacity\n if (this.errorGroups.size >= this.options.maxGroups) {\n this.evictOldestGroup();\n }\n\n this.errorGroups.set(fingerprint, newGroup);\n return newGroup;\n }\n\n /**\n * Extract errors from a trace and add them to the aggregator\n */\n addErrorsFromTrace(trace: TraceData): ErrorGroup[] {\n const addedGroups: ErrorGroup[] = [];\n\n for (const span of trace.spans) {\n if (span.status.code === 'ERROR') {\n const occurrence = this.extractErrorFromSpan(span, trace);\n if (occurrence) {\n const group = this.addError(occurrence);\n addedGroups.push(group);\n }\n }\n }\n\n return addedGroups;\n }\n\n /**\n * Extract error occurrence from a span\n */\n private extractErrorFromSpan(\n span: SpanData,\n trace: TraceData,\n ): ErrorOccurrence | null {\n // Try to get error info from span attributes or events\n const exceptionEvent = span.events?.find((e) => e.name === 'exception');\n const errorType =\n (span.attributes['exception.type'] as string) ||\n (span.attributes['error.type'] as string) ||\n (exceptionEvent?.attributes?.['exception.type'] as string) ||\n 'Error';\n\n const errorMessage =\n span.status.message ||\n (span.attributes['exception.message'] as string) ||\n (span.attributes['error.message'] as string) ||\n 'Unknown error';\n\n const stackTrace =\n (span.attributes['exception.stacktrace'] as string) ||\n (span.attributes['exception.stack'] as string) ||\n this.extractStackFromEvents(span);\n\n return {\n traceId: trace.traceId,\n spanId: span.spanId,\n spanName: span.name,\n service: trace.service,\n timestamp: span.endTime,\n error: {\n type: errorType,\n message: errorMessage,\n stackTrace,\n },\n attributes: this.extractRelevantAttributes(span.attributes),\n };\n }\n\n /**\n * Extract stack trace from span events (exception events)\n */\n private extractStackFromEvents(span: SpanData): string | undefined {\n if (!span.events) return undefined;\n\n const exceptionEvent = span.events.find((e) => e.name === 'exception');\n if (exceptionEvent?.attributes) {\n return (\n (exceptionEvent.attributes['exception.stacktrace'] as string) ||\n (exceptionEvent.attributes['exception.stack'] as string)\n );\n }\n\n return undefined;\n }\n\n /**\n * Extract relevant attributes for error context\n */\n private extractRelevantAttributes(\n attributes: Record<string, unknown>,\n ): Record<string, unknown> {\n const relevant: Record<string, unknown> = {};\n const keepKeys = [\n 'http.method',\n 'http.url',\n 'http.route',\n 'http.status_code',\n 'db.system',\n 'db.operation',\n 'rpc.method',\n 'rpc.service',\n 'code.function',\n 'code.filepath',\n 'user.id',\n 'operation.name',\n ];\n\n for (const key of keepKeys) {\n if (key in attributes) {\n relevant[key] = attributes[key];\n }\n }\n\n return relevant;\n }\n\n /**\n * Generate a fingerprint for error grouping\n *\n * Uses error type + first N stack frames (normalized)\n */\n private generateFingerprint(occurrence: ErrorOccurrence): string {\n const parts: string[] = [occurrence.error.type];\n\n if (occurrence.error.stackTrace) {\n const frames = this.extractStackFrames(\n occurrence.error.stackTrace,\n this.options.stackFramesForFingerprint,\n );\n parts.push(...frames);\n } else {\n // Fallback to error message if no stack trace\n parts.push(this.normalizeMessage(occurrence.error.message));\n }\n\n // Simple hash function\n return this.simpleHash(parts.join('|'));\n }\n\n /**\n * Extract and normalize stack frames from a stack trace\n */\n private extractStackFrames(stackTrace: string, count: number): string[] {\n const lines = stackTrace.split('\\n');\n const frames: string[] = [];\n\n for (const line of lines) {\n if (frames.length >= count) break;\n\n // Match common stack trace patterns\n const trimmed = line.trim();\n\n // Node.js style: \"at functionName (file:line:col)\"\n const nodeMatch = trimmed.match(/^at\\s+(.+?)\\s+\\((.+?):(\\d+):\\d+\\)$/);\n if (nodeMatch) {\n frames.push(`${nodeMatch[1]}@${this.normalizeFilePath(nodeMatch[2])}`);\n continue;\n }\n\n // Anonymous function style: \"at file:line:col\"\n const anonMatch = trimmed.match(/^at\\s+(.+?):(\\d+):\\d+$/);\n if (anonMatch) {\n frames.push(`anonymous@${this.normalizeFilePath(anonMatch[1])}`);\n continue;\n }\n\n // Browser style: \"functionName@file:line:col\"\n const browserMatch = trimmed.match(/^(.+?)@(.+?):(\\d+):\\d+$/);\n if (browserMatch) {\n frames.push(\n `${browserMatch[1]}@${this.normalizeFilePath(browserMatch[2])}`,\n );\n continue;\n }\n }\n\n return frames;\n }\n\n /**\n * Normalize file path by removing absolute path prefixes and node_modules paths\n */\n private normalizeFilePath(filePath: string): string {\n // Remove node_modules paths (keep package name)\n const nodeModulesMatch = filePath.match(\n /node_modules\\/(@[^/]+\\/[^/]+|[^/]+)/,\n );\n if (nodeModulesMatch) {\n return `[npm]/${nodeModulesMatch[1]}`;\n }\n\n // Remove common absolute path prefixes\n return filePath\n .replace(/^.*?\\/src\\//, 'src/')\n .replace(/^.*?\\/dist\\//, 'dist/')\n .replace(/^.*?\\/lib\\//, 'lib/')\n .replace(/^file:\\/\\//, '');\n }\n\n /**\n * Normalize error message by removing dynamic parts\n */\n private normalizeMessage(message: string): string {\n return (\n message\n // Remove UUIDs\n .replaceAll(\n /[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}/gi,\n '[UUID]',\n )\n // Remove hex IDs\n .replaceAll(/\\b[0-9a-f]{16,}\\b/gi, '[ID]')\n // Remove numbers\n .replaceAll(/\\b\\d+\\b/g, '[N]')\n // Remove quoted strings\n .replaceAll(/\"[^\"]*\"/g, '\"[STR]\"')\n .replaceAll(/'[^']*'/g, \"'[STR]'\")\n // Truncate long messages\n .slice(0, 200)\n );\n }\n\n /**\n * Normalize stack trace for display\n */\n private normalizeStackTrace(stackTrace?: string): string | undefined {\n if (!stackTrace) return undefined;\n\n const lines = stackTrace.split('\\n').slice(0, 10); // Keep first 10 lines\n return lines.join('\\n');\n }\n\n /**\n * Simple hash function for fingerprinting\n */\n private simpleHash(str: string): string {\n let hash = 0;\n for (let i = 0; i < str.length; i++) {\n const char = str.charCodeAt(i);\n hash = (hash << 5) - hash + char;\n hash = hash & hash; // Convert to 32-bit integer\n }\n return Math.abs(hash).toString(16).padStart(8, '0');\n }\n\n /**\n * Evict the oldest error group\n */\n private evictOldestGroup(): void {\n let oldest: { fingerprint: string; lastSeen: number } | null = null;\n\n for (const [fingerprint, group] of this.errorGroups) {\n if (!oldest || group.lastSeen < oldest.lastSeen) {\n oldest = { fingerprint, lastSeen: group.lastSeen };\n }\n }\n\n if (oldest) {\n this.errorGroups.delete(oldest.fingerprint);\n }\n }\n\n /**\n * Get all error groups, sorted by most recent\n */\n getErrorGroups(): ErrorGroup[] {\n return [...this.errorGroups.values()].sort(\n (a, b) => b.lastSeen - a.lastSeen,\n );\n }\n\n /**\n * Get error groups sorted by count (most frequent first)\n */\n getErrorGroupsByFrequency(): ErrorGroup[] {\n return [...this.errorGroups.values()].sort(\n (a, b) => b.count - a.count,\n );\n }\n\n /**\n * Get a specific error group by fingerprint\n */\n getErrorGroup(fingerprint: string): ErrorGroup | undefined {\n return this.errorGroups.get(fingerprint);\n }\n\n /**\n * Get error groups for a specific service\n */\n getErrorGroupsByService(service: string): ErrorGroup[] {\n return this.getErrorGroups().filter((g) => g.service === service);\n }\n\n /**\n * Get total error count across all groups\n */\n getTotalErrorCount(): number {\n let total = 0;\n for (const group of this.errorGroups.values()) {\n total += group.count;\n }\n return total;\n }\n\n /**\n * Get error statistics\n */\n getStats(): {\n totalGroups: number;\n totalErrors: number;\n recentErrors: number;\n topErrorTypes: Array<{ type: string; count: number }>;\n } {\n const now = Date.now();\n const oneHourAgo = now - 60 * 60 * 1000;\n\n let recentErrors = 0;\n const typeCount = new Map<string, number>();\n\n for (const group of this.errorGroups.values()) {\n if (group.lastSeen > oneHourAgo) {\n recentErrors += group.count;\n }\n typeCount.set(group.type, (typeCount.get(group.type) || 0) + group.count);\n }\n\n const topErrorTypes = [...typeCount.entries()]\n .map(([type, count]) => ({ type, count }))\n .sort((a, b) => b.count - a.count)\n .slice(0, 5);\n\n return {\n totalGroups: this.errorGroups.size,\n totalErrors: this.getTotalErrorCount(),\n recentErrors,\n topErrorTypes,\n };\n }\n\n /**\n * Clear all error groups\n */\n clear(): void {\n this.errorGroups.clear();\n }\n\n /**\n * Clear old error groups (not seen in given time window)\n */\n clearOlderThan(maxAgeMs: number): number {\n const cutoff = Date.now() - maxAgeMs;\n let cleared = 0;\n\n for (const [fingerprint, group] of this.errorGroups) {\n if (group.lastSeen < cutoff) {\n this.errorGroups.delete(fingerprint);\n cleared++;\n }\n }\n\n return cleared;\n }\n}\n","import type { DevtoolsData } from './types'\n\nexport interface TelemetryLimits {\n maxTraceCount: number\n maxLogCount: number\n maxMetricCount: number\n}\n\nexport interface ResolveTelemetryLimitsArgs {\n maxHistory?: number\n maxTraceCount?: number\n maxLogCount?: number\n maxMetricCount?: number\n env?: NodeJS.ProcessEnv\n}\n\nconst defaultLimit = 100\n\nfunction parseLimit(value: string | undefined): number | undefined {\n if (!value) return undefined\n const parsed = Number.parseInt(value, 10)\n return Number.isFinite(parsed) && parsed > 0 ? parsed : undefined\n}\n\nexport function resolveTelemetryLimits(\n args: ResolveTelemetryLimitsArgs = {},\n): TelemetryLimits {\n const env = args.env ?? process.env\n const fallback = args.maxHistory ?? defaultLimit\n\n return {\n maxTraceCount:\n args.maxTraceCount ??\n parseLimit(env.AUTOTEL_MAX_TRACE_COUNT) ??\n fallback,\n maxLogCount:\n args.maxLogCount ??\n parseLimit(env.AUTOTEL_MAX_LOG_COUNT) ??\n fallback,\n maxMetricCount:\n args.maxMetricCount ??\n parseLimit(env.AUTOTEL_MAX_METRIC_COUNT) ??\n fallback,\n }\n}\n\nexport function appendWithLimit<T>(items: T[], item: T, limit: number): T[] {\n if (limit <= 0) return []\n const next = [...items, item]\n return next.length > limit ? next.slice(next.length - limit) : next\n}\n\nexport function appendManyWithLimit<T>(\n items: T[],\n incoming: T[],\n limit: number,\n): T[] {\n if (limit <= 0 || incoming.length === 0) return limit <= 0 ? [] : items\n const next = [...items, ...incoming]\n return next.length > limit ? next.slice(next.length - limit) : next\n}\n\nexport function applyTelemetryLimits(\n data: DevtoolsData,\n limits: TelemetryLimits,\n): DevtoolsData {\n return {\n ...data,\n traces: data.traces.slice(-limits.maxTraceCount),\n logs: data.logs.slice(-limits.maxLogCount),\n metrics: data.metrics.slice(-limits.maxMetricCount),\n }\n}\n","// src/server/server.ts\nimport { WebSocketServer, WebSocket } from 'ws'\nimport type { Server as HTTPServer } from 'node:http'\nimport { createServer } from 'node:http'\nimport { ErrorAggregator } from './error-aggregator'\nimport type {\n TraceData,\n LogData,\n MetricData,\n DevtoolsData,\n} from './types'\nimport {\n appendManyWithLimit,\n appendWithLimit,\n resolveTelemetryLimits,\n type TelemetryLimits,\n} from './telemetry-limits'\n\nexport interface DevtoolsServerOptions {\n port?: number\n server?: HTTPServer\n path?: string\n verbose?: boolean\n maxHistory?: number\n maxTraceCount?: number\n maxLogCount?: number\n maxMetricCount?: number\n}\n\nexport class DevtoolsServer {\n private wss: WebSocketServer\n private clients = new Set<WebSocket>()\n private httpServer: HTTPServer\n private traces: TraceData[] = []\n private logs: LogData[] = []\n private metrics: MetricData[] = []\n private errorAggregator = new ErrorAggregator()\n private limits: TelemetryLimits\n private verbose: boolean\n private _port: number\n\n constructor(options: DevtoolsServerOptions = {}) {\n this.limits = resolveTelemetryLimits(options)\n this.verbose = options.verbose ?? false\n this._port = options.port ?? 4318\n\n this.httpServer = options.server ?? createServer()\n this.wss = new WebSocketServer({ server: this.httpServer, path: options.path ?? '/ws' })\n\n this.wss.on('connection', (ws) => {\n this.clients.add(ws)\n this.log(`Client connected (${this.clients.size} total)`)\n\n // Send history to late-connecting clients\n const data = this.getCurrentData()\n if (data.traces.length > 0 || data.logs.length > 0 || data.errors.length > 0) {\n ws.send(JSON.stringify(data))\n }\n\n ws.on('close', () => {\n this.clients.delete(ws)\n this.log(`Client disconnected (${this.clients.size} total)`)\n })\n })\n\n // Only start listening if no external server was provided\n if (!options.server) {\n this.httpServer.listen(this._port, () => {\n const addr = this.httpServer.address()\n if (addr && typeof addr === 'object') this._port = addr.port\n this.log(`WebSocket server listening on port ${this._port}`)\n })\n }\n }\n\n get port(): number {\n const addr = this.httpServer.address()\n if (addr && typeof addr === 'object') return addr.port\n return this._port\n }\n\n get clientCount(): number {\n return this.clients.size\n }\n\n addTrace(trace: TraceData): void {\n // Merge if trace already exists (out-of-order spans)\n const existing = this.traces.find(t => t.traceId === trace.traceId)\n if (existing) {\n const existingSpanIds = new Set(existing.spans.map(s => s.spanId))\n for (const span of trace.spans) {\n if (!existingSpanIds.has(span.spanId)) {\n existing.spans.push(span)\n }\n }\n existing.startTime = Math.min(existing.startTime, trace.startTime)\n existing.endTime = Math.max(existing.endTime, trace.endTime)\n existing.duration = existing.endTime - existing.startTime\n if (trace.status === 'ERROR') existing.status = 'ERROR'\n } else {\n this.traces = appendWithLimit(\n this.traces,\n trace,\n this.limits.maxTraceCount,\n )\n }\n\n this.errorAggregator.addErrorsFromTrace(trace)\n this.broadcast({ traces: [trace], metrics: [], logs: [], errors: this.errorAggregator.getErrorGroups() })\n }\n\n addTraces(traces: TraceData[]): void {\n for (const trace of traces) this.addTrace(trace)\n }\n\n addLog(log: LogData): void {\n this.logs = appendWithLimit(this.logs, log, this.limits.maxLogCount)\n this.broadcast({ traces: [], metrics: [], logs: [log], errors: [] })\n }\n\n addLogs(logs: LogData[]): void {\n this.logs = appendManyWithLimit(this.logs, logs, this.limits.maxLogCount)\n this.broadcast({ traces: [], metrics: [], logs, errors: [] })\n }\n\n addMetric(metric: MetricData): void {\n this.metrics = appendWithLimit(\n this.metrics,\n metric,\n this.limits.maxMetricCount,\n )\n this.broadcast({ traces: [], metrics: [metric], logs: [], errors: [] })\n }\n\n getCurrentData(): DevtoolsData {\n return {\n traces: this.traces,\n metrics: this.metrics,\n logs: this.logs,\n errors: this.errorAggregator.getErrorGroups(),\n }\n }\n\n clearData(): void {\n this.traces = []\n this.logs = []\n this.metrics = []\n this.errorAggregator.clear()\n }\n\n private broadcast(data: DevtoolsData): void {\n const msg = JSON.stringify(data)\n for (const client of this.clients) {\n if (client.readyState === WebSocket.OPEN) {\n client.send(msg)\n }\n }\n }\n\n private log(message: string): void {\n if (this.verbose) console.log(`[autotel-devtools] ${message}`)\n }\n\n async close(): Promise<void> {\n for (const client of this.clients) client.close()\n this.clients.clear()\n this.wss.close()\n await new Promise<void>((resolve) => this.httpServer.close(() => resolve()))\n }\n}\n","export function getResourceName(\n resource: Record<string, unknown> | undefined,\n fallback = 'unknown',\n): string {\n if (!resource) return fallback\n\n const candidates = [\n resource['service.name'],\n resource['service.namespace'],\n resource['deployment.environment.name'],\n resource['host.name'],\n resource['container.name'],\n resource['process.executable.name'],\n ]\n\n for (const candidate of candidates) {\n if (typeof candidate === 'string' && candidate.trim().length > 0) {\n return candidate\n }\n }\n\n return fallback\n}\n","// src/server/otlp.ts\nimport type { IncomingMessage, ServerResponse } from 'node:http'\nimport type { SpanData, TraceData, LogData } from './types'\nimport { getResourceName } from './resource-utils'\n\ntype OtlpAnyValue = {\n stringValue?: string\n boolValue?: boolean\n intValue?: string | number\n doubleValue?: number\n bytesValue?: string\n arrayValue?: { values?: OtlpAnyValue[] }\n kvlistValue?: { values?: OtlpKeyValue[] }\n}\n\ntype OtlpKeyValue = { key: string; value?: OtlpAnyValue }\n\nfunction resolveOtlpValue(v?: OtlpAnyValue): unknown {\n if (!v) return undefined\n if (v.stringValue !== undefined) return v.stringValue\n if (v.boolValue !== undefined) return v.boolValue\n if (v.intValue !== undefined) return typeof v.intValue === 'string' ? Number(v.intValue) : v.intValue\n if (v.doubleValue !== undefined) return v.doubleValue\n if (v.bytesValue !== undefined) return v.bytesValue\n if (v.arrayValue?.values) return v.arrayValue.values.map(resolveOtlpValue)\n if (v.kvlistValue?.values) return flattenAttributes(v.kvlistValue.values)\n return undefined\n}\n\nfunction flattenAttributes(attrs?: OtlpKeyValue[]): Record<string, unknown> {\n const out: Record<string, unknown> = {}\n if (!attrs) return out\n for (const { key, value } of attrs) {\n out[key] = resolveOtlpValue(value)\n }\n return out\n}\n\nfunction nanoToMs(nano?: string): number {\n if (!nano) return 0\n return Number(BigInt(nano) / 1_000_000n)\n}\n\nconst SPAN_KIND_MAP: Record<number | string, SpanData['kind']> = {\n 0: 'INTERNAL', 1: 'INTERNAL', 2: 'SERVER', 3: 'CLIENT', 4: 'PRODUCER', 5: 'CONSUMER',\n SPAN_KIND_INTERNAL: 'INTERNAL', SPAN_KIND_SERVER: 'SERVER',\n SPAN_KIND_CLIENT: 'CLIENT', SPAN_KIND_PRODUCER: 'PRODUCER', SPAN_KIND_CONSUMER: 'CONSUMER',\n}\n\nfunction normalizeHexId(id?: string): string {\n if (!id) return ''\n // Only attempt base64 decode for strings that look like base64-encoded binary IDs\n // (length 24 or 28 chars for 16/32-byte IDs, valid base64 chars, not plain hex)\n const isBase64Like = /^[A-Za-z0-9+/=]+$/.test(id) && !(/^[0-9a-f]+$/i.test(id))\n const isLikelyBase64Id = isBase64Like && (id.length === 24 || id.length === 28 || id.length === 44 || id.length === 48)\n if (isLikelyBase64Id) {\n try {\n const bytes = Buffer.from(id, 'base64')\n return bytes.toString('hex')\n } catch { /* fall through */ }\n }\n return id\n}\n\nexport function parseOtlpTraces(payload: unknown): TraceData[] {\n if (!payload || typeof payload !== 'object') return []\n const { resourceSpans } = payload as any\n if (!Array.isArray(resourceSpans) || resourceSpans.length === 0) return []\n\n const traceMap = new Map<string, { spans: SpanData[]; service: string }>()\n\n for (const rs of resourceSpans) {\n const resourceAttrs = flattenAttributes(rs.resource?.attributes)\n const service = String(resourceAttrs['service.name'] || 'unknown')\n const scopeSpans = rs.scopeSpans || []\n\n for (const ss of scopeSpans) {\n for (const span of ss.spans || []) {\n const traceId = normalizeHexId(span.traceId)\n if (!traceId) continue\n\n const startMs = nanoToMs(span.startTimeUnixNano)\n const endMs = nanoToMs(span.endTimeUnixNano)\n const statusCode = span.status?.code\n let status: SpanData['status']['code'] = 'UNSET'\n if (statusCode === 1 || statusCode === 'STATUS_CODE_OK') status = 'OK'\n if (statusCode === 2 || statusCode === 'STATUS_CODE_ERROR') status = 'ERROR'\n\n const spanData: SpanData = {\n traceId,\n spanId: normalizeHexId(span.spanId),\n parentSpanId: normalizeHexId(span.parentSpanId) || undefined,\n name: span.name || 'unknown',\n kind: SPAN_KIND_MAP[span.kind ?? 0] || 'INTERNAL',\n startTime: startMs,\n endTime: endMs,\n duration: endMs - startMs,\n attributes: { ...resourceAttrs, ...flattenAttributes(span.attributes) } as Record<string, any>,\n status: { code: status, message: span.status?.message },\n events: (span.events || []).map((e: any) => ({\n name: e.name || '',\n timestamp: nanoToMs(e.timeUnixNano),\n attributes: flattenAttributes(e.attributes) as Record<string, any>,\n })),\n }\n\n const existing = traceMap.get(traceId)\n if (existing) {\n existing.spans.push(spanData)\n } else {\n traceMap.set(traceId, { spans: [spanData], service })\n }\n }\n }\n }\n\n const traces: TraceData[] = []\n for (const [traceId, { spans, service }] of traceMap) {\n const sorted = spans.sort((a, b) => a.startTime - b.startTime)\n const rootSpan = sorted.find(s => !s.parentSpanId) || sorted[0]\n const startTime = Math.min(...sorted.map(s => s.startTime))\n const endTime = Math.max(...sorted.map(s => s.endTime))\n const hasError = sorted.some(s => s.status.code === 'ERROR')\n\n traces.push({\n traceId,\n correlationId: traceId.slice(0, 16),\n rootSpan,\n spans: sorted,\n startTime,\n endTime,\n duration: endTime - startTime,\n status: hasError ? 'ERROR' : 'OK',\n service,\n })\n }\n\n return traces\n}\n\nexport function parseOtlpLogs(payload: unknown): LogData[] {\n if (!payload || typeof payload !== 'object') return []\n const { resourceLogs } = payload as any\n if (!Array.isArray(resourceLogs)) return []\n\n const logs: LogData[] = []\n for (const rl of resourceLogs) {\n const resourceAttrs = flattenAttributes(rl.resource?.attributes)\n for (const sl of rl.scopeLogs || []) {\n for (const rec of sl.logRecords || []) {\n const timestamp = nanoToMs(rec.timeUnixNano || rec.observedTimeUnixNano)\n const traceId = normalizeHexId(rec.traceId) || undefined\n const spanId = normalizeHexId(rec.spanId) || undefined\n const body = rec.body ? resolveOtlpValue(rec.body) : ''\n\n logs.push({\n id: `${traceId || 'no-trace'}:${spanId || 'no-span'}:${timestamp}:${rec.severityNumber || 0}`,\n traceId,\n spanId,\n resourceName: getResourceName(resourceAttrs as Record<string, unknown>),\n severityText: rec.severityText,\n severityNumber: rec.severityNumber,\n body: typeof body === 'string' ? body : (body as Record<string, unknown>),\n timestamp,\n attributes: flattenAttributes(rec.attributes) as Record<string, unknown>,\n resource: resourceAttrs as Record<string, unknown>,\n })\n }\n }\n }\n\n return logs\n}\n\nexport function countOtlpMetrics(payload: unknown): number {\n if (!payload || typeof payload !== 'object') return 0\n const { resourceMetrics } = payload as any\n if (!Array.isArray(resourceMetrics)) return 0\n let count = 0\n for (const rm of resourceMetrics) {\n for (const sm of rm.scopeMetrics || []) {\n count += (sm.metrics || []).length\n }\n }\n return count\n}\n\nexport async function readJsonBody(req: IncomingMessage): Promise<unknown> {\n return new Promise((resolve, reject) => {\n const chunks: Buffer[] = []\n req.on('data', (chunk: Buffer) => chunks.push(chunk))\n req.on('end', () => {\n try {\n resolve(JSON.parse(Buffer.concat(chunks).toString()))\n } catch {\n reject(new Error('Invalid JSON'))\n }\n })\n req.on('error', reject)\n })\n}\n\nexport function sendJson(res: ServerResponse, status: number, data: Record<string, unknown>): void {\n const body = JSON.stringify(data)\n res.writeHead(status, { 'Content-Type': 'application/json', 'Content-Length': Buffer.byteLength(body) })\n res.end(body)\n}\n","// src/server/http.ts\nimport { createServer, type IncomingMessage, type ServerResponse, type Server } from 'node:http'\nimport { readFileSync, existsSync } from 'node:fs'\nimport { resolve, dirname } from 'node:path'\nimport { fileURLToPath } from 'node:url'\nimport { parseOtlpTraces, parseOtlpLogs, countOtlpMetrics, readJsonBody, sendJson } from './otlp'\nimport type { DevtoolsServer } from './server'\n\nexport interface HttpServerOptions {\n port?: number\n host?: string\n}\n\nfunction findPackageRoot(): string {\n let dir = dirname(fileURLToPath(import.meta.url))\n for (let i = 0; i < 5; i++) {\n if (existsSync(resolve(dir, 'package.json'))) return dir\n dir = dirname(dir)\n }\n return dir\n}\n\nconst FULLPAGE_HTML = `<!DOCTYPE html><html><head><meta charset=\"UTF-8\"><meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\"><title>autotel-devtools</title><style>*{margin:0;padding:0;box-sizing:border-box}html,body{height:100%;width:100%;overflow:hidden}</style></head><body><script src=\"/widget.js?mode=fullpage\"></script></body></html>`\n\nlet cachedWidgetJs: string | null = null\nfunction getWidgetJs(): string {\n if (!cachedWidgetJs) {\n const pkgRoot = findPackageRoot()\n const candidates = [\n resolve(pkgRoot, 'dist', 'widget.global.js'),\n resolve(pkgRoot, 'widget.global.js'),\n ]\n for (const candidate of candidates) {\n try {\n cachedWidgetJs = readFileSync(candidate, 'utf8')\n break\n } catch { /* try next */ }\n }\n if (!cachedWidgetJs) {\n cachedWidgetJs = '// widget bundle not found - run pnpm build first'\n }\n }\n return cachedWidgetJs\n}\n\nexport function attachDevtoolsRoutes(httpServer: Server, devtools: DevtoolsServer): void {\n httpServer.on('request', async (req: IncomingMessage, res: ServerResponse) => {\n // CORS headers\n res.setHeader('Access-Control-Allow-Origin', '*')\n res.setHeader('Access-Control-Allow-Methods', 'GET, POST, DELETE, OPTIONS')\n res.setHeader('Access-Control-Allow-Headers', 'Content-Type')\n\n if (req.method === 'OPTIONS') {\n res.writeHead(204)\n res.end()\n return\n }\n\n const url = req.url || '/'\n\n // GET / — fullpage HTML\n if (req.method === 'GET' && url === '/') {\n res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8', 'Content-Length': Buffer.byteLength(FULLPAGE_HTML) })\n res.end(FULLPAGE_HTML)\n return\n }\n\n // GET /widget.js — widget bundle\n if (req.method === 'GET' && url.startsWith('/widget.js')) {\n const js = getWidgetJs()\n res.writeHead(200, { 'Content-Type': 'application/javascript; charset=utf-8', 'Content-Length': Buffer.byteLength(js) })\n res.end(js)\n return\n }\n\n // GET /healthz\n if (req.method === 'GET' && url === '/healthz') {\n sendJson(res, 200, { ok: true, clients: devtools.clientCount })\n return\n }\n\n // GET /v1/traces — read back what the collector has actually received.\n // This is the verification primitive for tests: poll the collector over\n // HTTP and assert receipt, instead of only asserting \"the client tried to\n // send\" (which a browser-level route intercept can fake). Bypasses the UI's\n // WebSocket entirely.\n if (req.method === 'GET' && url === '/v1/traces') {\n const data = devtools.getCurrentData()\n sendJson(res, 200, { traces: data.traces, count: data.traces.length })\n return\n }\n\n // DELETE /v1/traces — clear captured telemetry (test isolation / reset).\n // Clears traces, logs, metrics and aggregated errors so each test starts clean.\n if (req.method === 'DELETE' && url === '/v1/traces') {\n devtools.clearData()\n sendJson(res, 200, { cleared: true })\n return\n }\n\n // POST /v1/traces\n if (req.method === 'POST' && url === '/v1/traces') {\n try {\n const payload = await readJsonBody(req)\n const traces = parseOtlpTraces(payload)\n devtools.addTraces(traces)\n sendJson(res, 200, { acceptedTraces: traces.length })\n } catch (e) {\n sendJson(res, 400, { error: 'Invalid OTLP JSON', message: e instanceof Error ? e.message : String(e) })\n }\n return\n }\n\n // POST /v1/logs\n if (req.method === 'POST' && url === '/v1/logs') {\n try {\n const payload = await readJsonBody(req)\n const logs = parseOtlpLogs(payload)\n devtools.addLogs(logs)\n sendJson(res, 200, { acceptedLogs: logs.length })\n } catch (e) {\n sendJson(res, 400, { error: 'Invalid OTLP JSON', message: e instanceof Error ? e.message : String(e) })\n }\n return\n }\n\n // POST /v1/metrics\n if (req.method === 'POST' && url === '/v1/metrics') {\n try {\n const payload = await readJsonBody(req)\n const count = countOtlpMetrics(payload)\n sendJson(res, 200, { acceptedMetrics: count })\n } catch (e) {\n sendJson(res, 400, { error: 'Invalid OTLP JSON', message: e instanceof Error ? e.message : String(e) })\n }\n return\n }\n\n sendJson(res, 404, { error: 'Not found' })\n })\n}\n\nexport function createDevtoolsHttpServer(devtools: DevtoolsServer, _options: HttpServerOptions = {}): Server {\n const server = createServer()\n attachDevtoolsRoutes(server, devtools)\n return server\n}\n","// src/server/listen.ts\nimport { createServer, type Server } from 'node:http'\n\nconst LOOPBACK = new Set(['localhost', '127.0.0.1', '::1'])\n\nexport interface LoopbackListeners {\n /** Resolves once the primary and (attempted) sibling listeners are up. */\n ready: Promise<{ addresses: string[]; warnings: string[] }>\n /** Close the sibling listener (the primary server is owned by the caller). */\n closeSibling: () => Promise<void>\n}\n\n/** Format host:port, bracketing IPv6 literals (e.g. `[::1]:4318`). */\nexport function formatAddress(host: string, port: number): string {\n return host.includes(':') ? `[${host}]:${port}` : `${host}:${port}`\n}\n\n/**\n * Listen on `host:port`, and when `host` is a loopback literal, ALSO listen on\n * the sibling loopback family (IPv4 ⟷ IPv6) so a client reaches the collector\n * whether the OS resolves `localhost` to `127.0.0.1` or `::1`.\n *\n * This kills a notoriously silent footgun: a dev-server proxy targeting\n * `http://localhost:PORT` on macOS resolves `localhost` to `::1`, but a\n * collector bound only to `127.0.0.1` never receives the request — spans\n * vanish with no error. Binding both loopback families makes `localhost` work\n * regardless of resolution order.\n *\n * The sibling listener serves the same HTTP routes (via `attachSecondary`);\n * the WebSocket/UI stays on the primary address. If the sibling cannot bind\n * (e.g. no IPv6, or the port is taken on that family), it is reported as a\n * warning rather than a fatal error.\n */\nexport function listenLoopbackDualStack(args: {\n primary: Server\n port: number\n host: string\n attachSecondary: (server: Server) => void\n}): LoopbackListeners {\n const { primary, port, host, attachSecondary } = args\n let sibling: Server | undefined\n\n const ready = new Promise<{ addresses: string[]; warnings: string[] }>(\n (resolve) => {\n const addresses: string[] = []\n const warnings: string[] = []\n // Normalise `localhost` to an explicit family so the primary bind is\n // deterministic and we know which sibling family to add.\n const primaryHost = host === 'localhost' ? '127.0.0.1' : host\n\n primary.listen(port, primaryHost, () => {\n const addr = primary.address()\n const resolvedPort =\n addr && typeof addr === 'object' ? addr.port : port\n addresses.push(formatAddress(primaryHost, resolvedPort))\n\n if (!LOOPBACK.has(host)) {\n resolve({ addresses, warnings })\n return\n }\n\n const siblingHost = primaryHost === '::1' ? '127.0.0.1' : '::1'\n const s = createServer()\n attachSecondary(s)\n\n const onError = (e: Error) => {\n s.close()\n warnings.push(\n `could not also bind ${formatAddress(siblingHost, resolvedPort)} (${e.message}); ` +\n `clients using the ${siblingHost === '::1' ? 'IPv6' : 'IPv4'} form of \"localhost\" may not connect.`,\n )\n resolve({ addresses, warnings })\n }\n s.once('error', onError)\n s.listen(resolvedPort, siblingHost, () => {\n s.off('error', onError)\n sibling = s\n addresses.push(formatAddress(siblingHost, resolvedPort))\n resolve({ addresses, warnings })\n })\n })\n },\n )\n\n return {\n ready,\n closeSibling: () =>\n new Promise<void>((res) => {\n if (!sibling) return res()\n sibling.close(() => res())\n }),\n }\n}\n","#!/usr/bin/env node\n// src/cli.ts\nimport { createServer } from 'node:http'\nimport { readFileSync } from 'node:fs'\nimport { resolve, dirname } from 'node:path'\nimport { fileURLToPath } from 'node:url'\nimport { DevtoolsServer } from './server/server'\nimport { attachDevtoolsRoutes } from './server/http'\nimport { listenLoopbackDualStack } from './server/listen'\n\ninterface CliOptions {\n port: number\n host: string\n title?: string\n}\n\nfunction printHelp(): void {\n process.stdout.write(\n `autotel-devtools - Standalone OTLP receiver with web devtools UI\n\nUsage: autotel-devtools [options]\n\nOptions:\n -p, --port <port> Port to listen on (default: 4318, env: AUTOTEL_DEVTOOLS_PORT)\n -H, --host <host> Host to bind to (default: 127.0.0.1, env: AUTOTEL_DEVTOOLS_HOST)\n -t, --title <title> Dashboard title (env: AUTOTEL_DEVTOOLS_TITLE)\n Env limits: AUTOTEL_MAX_TRACE_COUNT, AUTOTEL_MAX_LOG_COUNT, AUTOTEL_MAX_METRIC_COUNT\n -h, --help Show this help message\n -v, --version Show version number\n\nEndpoints:\n GET / Web devtools UI (fullpage)\n GET /widget.js Widget bundle (embed in your app)\n POST /v1/traces Receive OTLP JSON trace data\n GET /v1/traces Read back received traces (verify ingestion in tests)\n DELETE /v1/traces Clear captured telemetry (test reset)\n POST /v1/logs Receive OTLP JSON log data\n POST /v1/metrics Receive OTLP JSON metric data\n WS /ws WebSocket stream for real-time updates\n GET /healthz Health check\n\nExamples:\n npx autotel-devtools\n npx autotel-devtools -p 4319\n\nThen point your app:\n OTEL_EXPORTER_OTLP_PROTOCOL=http/json OTEL_EXPORTER_OTLP_ENDPOINT=http://localhost:4318 node app.js\n\nView in browser:\n http://localhost:4318\n\nOr embed widget in your app:\n <script src=\"http://localhost:4318/widget.js\"></script>\n` + '\\n',\n )\n}\n\nfunction printVersion(): void {\n try {\n const dir = dirname(fileURLToPath(import.meta.url))\n const pkgPath = resolve(dir, '..', 'package.json')\n const pkg = JSON.parse(readFileSync(pkgPath, 'utf8'))\n process.stdout.write(`${pkg.version}\\n`)\n } catch {\n process.stdout.write('unknown\\n')\n }\n}\n\nfunction parseArgs(argv: string[]): CliOptions | null {\n const options: CliOptions = {\n port: Number(process.env.AUTOTEL_DEVTOOLS_PORT || 4318),\n host: process.env.AUTOTEL_DEVTOOLS_HOST || '127.0.0.1',\n title: process.env.AUTOTEL_DEVTOOLS_TITLE,\n }\n\n for (let i = 0; i < argv.length; i++) {\n const arg = argv[i]\n const next = argv[i + 1]\n if (arg === '--help' || arg === '-h') { printHelp(); return null }\n if (arg === '--version' || arg === '-v') { printVersion(); return null }\n if ((arg === '--port' || arg === '-p') && next) { options.port = Number(next); i++; continue }\n if ((arg === '--host' || arg === '-H') && next) { options.host = next; i++; continue }\n if ((arg === '--title' || arg === '-t') && next) { options.title = next; i++; continue }\n }\n\n return options\n}\n\nasync function main(): Promise<void> {\n const options = parseArgs(process.argv.slice(2))\n if (!options) { process.exit(0) }\n\n const httpServer = createServer()\n const wsServer = new DevtoolsServer({ server: httpServer, verbose: true })\n attachDevtoolsRoutes(httpServer, wsServer)\n\n const listeners = listenLoopbackDualStack({\n primary: httpServer,\n port: options.port,\n host: options.host,\n attachSecondary: (s) => attachDevtoolsRoutes(s, wsServer),\n })\n\n const { addresses, warnings } = await listeners.ready\n const uiBase = `http://${options.host === 'localhost' ? '127.0.0.1' : options.host}:${options.port}`\n const title = options.title || 'autotel-devtools'\n process.stdout.write(`\\n ${title}\\n\\n`)\n process.stdout.write(` Listening: ${addresses.join(' + ')}\\n`)\n process.stdout.write(` UI: ${uiBase}\\n`)\n process.stdout.write(` Widget: <script src=\"${uiBase}/widget.js\"></script>\\n`)\n process.stdout.write(` WebSocket: ${uiBase.replace('http', 'ws')}/ws\\n`)\n process.stdout.write(` OTLP: ${uiBase}/v1/traces\\n\\n`)\n process.stdout.write(` Set OTEL_EXPORTER_OTLP_PROTOCOL=http/json\\n`)\n process.stdout.write(` Set OTEL_EXPORTER_OTLP_ENDPOINT=${uiBase}\\n\\n`)\n // Self-check: confirm the collector is reachable AND that ingestion works,\n // not just that something is listening. Reading /v1/traces back is the same\n // check a test should make instead of trusting \"the client tried to send\".\n process.stdout.write(` Verify ingestion: curl -s ${uiBase}/v1/traces\\n\\n`)\n for (const w of warnings) {\n process.stdout.write(` ⚠ ${w}\\n`)\n }\n if (warnings.length > 0) process.stdout.write('\\n')\n\n const shutdown = () => {\n Promise.all([wsServer.close(), listeners.closeSibling()]).then(() =>\n process.exit(0),\n )\n }\n process.on('SIGINT', shutdown)\n process.on('SIGTERM', shutdown)\n}\n\nmain().catch((error) => {\n process.stderr.write(`[autotel-devtools] failed to start: ${error instanceof Error ? error.message : String(error)}\\n`)\n process.exit(1)\n})\n"]}
|