abtars 0.2.2 → 0.2.3-alpha.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/bundle/{_registry.generated-KYX63MGY.js → _registry.generated-KM6LXTNJ.js} +2 -2
- package/bundle/abtars-cli.js +6 -3
- package/bundle/abtars-cli.js.map +2 -2
- package/bundle/abtars.js +29 -28
- package/bundle/abtars.js.map +3 -3
- package/bundle/{agent-registry-PIS5XJHX.js → agent-registry-ABPFQXNL.js} +2 -2
- package/bundle/{chunk-QSC6QZ44.js → chunk-2SFN2VYD.js} +2 -2
- package/bundle/{chunk-BBTQKKDO.js → chunk-2W6JIHZ5.js} +2 -1
- package/bundle/chunk-2W6JIHZ5.js.map +7 -0
- package/bundle/{chunk-3IPMKYYH.js → chunk-6TSCOXF6.js} +56 -25
- package/bundle/chunk-6TSCOXF6.js.map +7 -0
- package/bundle/{chunk-7WFE2JI5.js → chunk-7B3GK5JQ.js} +2 -2
- package/bundle/{chunk-SMZQDMSZ.js → chunk-ENXQMPV3.js} +1 -2
- package/bundle/chunk-ENXQMPV3.js.map +7 -0
- package/bundle/{chunk-N24ROESF.js → chunk-HFPXN6NM.js} +1 -1
- package/bundle/chunk-HFPXN6NM.js.map +7 -0
- package/bundle/{chunk-Y2XBDQP3.js → chunk-SEXVA3GK.js} +144 -37
- package/bundle/chunk-SEXVA3GK.js.map +7 -0
- package/bundle/{commands-LAWVNQTO.js → commands-L6VIMPCR.js} +2 -2
- package/bundle/{direct-api-transport-QIWA5ES2.js → direct-api-transport-BK72AP3I.js} +1 -1
- package/bundle/{direct-api-transport-QIWA5ES2.js.map → direct-api-transport-BK72AP3I.js.map} +2 -2
- package/bundle/{discord-adapter-W6L5KJ6T.js → discord-adapter-DWIQRNDI.js} +3 -3
- package/bundle/{doctor-PIPSGI3H.js → doctor-WHTVSUOF.js} +36 -26
- package/bundle/doctor-WHTVSUOF.js.map +7 -0
- package/bundle/{install-I3CXVW52.js → install-Q4XNCPG7.js} +2 -2
- package/bundle/{message-pipeline-4CTBJ6K2.js → message-pipeline-GCSZCQWO.js} +2 -2
- package/bundle/meta.json +298 -279
- package/bundle/{phase-transport-INFD6ELA.js → phase-transport-F7GQRRYE.js} +3 -3
- package/bundle/{sleep-ENFZFUJJ.js → sleep-MYOZ73IU.js} +2 -2
- package/bundle/{subagent-runtime-5AYOXOU2.js → subagent-runtime-QA4LVU4C.js} +2 -2
- package/bundle/{system-status-7K2QTH3J.js → system-status-KMKPAC5Z.js} +4 -2
- package/bundle/system-status-KMKPAC5Z.js.map +7 -0
- package/bundle/{telegram-adapter-4KI4CJPG.js → telegram-adapter-TRMCC634.js} +7 -4
- package/bundle/telegram-adapter-TRMCC634.js.map +7 -0
- package/config/transport.default.json +2 -1
- package/install-manifest.json +0 -3
- package/package.json +1 -1
- package/scripts/abtars-daemon.service +0 -4
- package/scripts/abtars@.service +0 -4
- package/scripts/build-and-deploy.sh +15 -27
- package/bundle/agent-registry-5VL5KI6U.js +0 -19
- package/bundle/chunk-3IPMKYYH.js.map +0 -7
- package/bundle/chunk-4WKWPU6U.js +0 -1089
- package/bundle/chunk-4WKWPU6U.js.map +0 -7
- package/bundle/chunk-5WFIAUQC.js +0 -672
- package/bundle/chunk-5WFIAUQC.js.map +0 -7
- package/bundle/chunk-B52YRWR6.js +0 -257
- package/bundle/chunk-B52YRWR6.js.map +0 -7
- package/bundle/chunk-BBTQKKDO.js.map +0 -7
- package/bundle/chunk-HAS5NEK7.js +0 -189
- package/bundle/chunk-HAS5NEK7.js.map +0 -7
- package/bundle/chunk-HB54S5OY.js +0 -4036
- package/bundle/chunk-HB54S5OY.js.map +0 -7
- package/bundle/chunk-N24ROESF.js.map +0 -7
- package/bundle/chunk-N7UG4FID.js +0 -4036
- package/bundle/chunk-N7UG4FID.js.map +0 -7
- package/bundle/chunk-PUDGA4RR.js +0 -183
- package/bundle/chunk-QSC6QZ44.js.map +0 -7
- package/bundle/chunk-SMZQDMSZ.js.map +0 -7
- package/bundle/chunk-VY2BUO6L.js +0 -4035
- package/bundle/chunk-VY2BUO6L.js.map +0 -7
- package/bundle/chunk-Y2XBDQP3.js.map +0 -7
- package/bundle/chunk-YMGX6HNP.js +0 -131
- package/bundle/chunk-YMGX6HNP.js.map +0 -7
- package/bundle/commands-IGRSOSK6.js +0 -34
- package/bundle/commands-RBWY7YXB.js +0 -34
- package/bundle/commands-XFZNMZN6.js +0 -34
- package/bundle/direct-api-transport-OZICXTWQ.js +0 -889
- package/bundle/direct-api-transport-OZICXTWQ.js.map +0 -7
- package/bundle/discord-adapter-JFIIVG34.js +0 -589
- package/bundle/discord-adapter-U3FA5OTY.js +0 -589
- package/bundle/discord-adapter-U3FA5OTY.js.map +0 -7
- package/bundle/discord-adapter-W6L5KJ6T.js.map +0 -7
- package/bundle/discord-adapter-WWM6ROTW.js +0 -589
- package/bundle/discord-adapter-WWM6ROTW.js.map +0 -7
- package/bundle/doctor-PIPSGI3H.js.map +0 -7
- package/bundle/kanban-board-6Q5E5GEB.js +0 -31
- package/bundle/kanban-board-6Q5E5GEB.js.map +0 -7
- package/bundle/message-pipeline-4CTBJ6K2.js.map +0 -7
- package/bundle/message-pipeline-4KL7OWUH.js +0 -38
- package/bundle/message-pipeline-4KL7OWUH.js.map +0 -7
- package/bundle/message-pipeline-GFKSHRFU.js +0 -38
- package/bundle/message-pipeline-GFKSHRFU.js.map +0 -7
- package/bundle/message-pipeline-TGI2WJJM.js +0 -38
- package/bundle/message-pipeline-TGI2WJJM.js.map +0 -7
- package/bundle/phase-transport-INFD6ELA.js.map +0 -7
- package/bundle/phase-transport-KXFZ5BVF.js +0 -23
- package/bundle/phase-transport-KXFZ5BVF.js.map +0 -7
- package/bundle/sleep-ENFZFUJJ.js.map +0 -7
- package/bundle/subagent-runtime-5AYOXOU2.js.map +0 -7
- package/bundle/subagent-runtime-VKTX6Q2M.js +0 -13
- package/bundle/subagent-runtime-VKTX6Q2M.js.map +0 -7
- package/bundle/system-status-7K2QTH3J.js.map +0 -7
- package/bundle/telegram-adapter-4KI4CJPG.js.map +0 -7
- package/bundle/telegram-adapter-76B4JRJJ.js +0 -1080
- package/bundle/telegram-adapter-76B4JRJJ.js.map +0 -7
- package/bundle/telegram-adapter-VZA74EMT.js +0 -1080
- package/bundle/telegram-adapter-VZA74EMT.js.map +0 -7
- package/bundle/telegram-adapter-ZO2CLU22.js +0 -1080
- package/bundle/telegram-adapter-ZO2CLU22.js.map +0 -7
- package/bundle/tool-registry-TGNU5AMG.js +0 -43
- package/bundle/tool-registry-TGNU5AMG.js.map +0 -7
- /package/bundle/{_registry.generated-KYX63MGY.js.map → _registry.generated-KM6LXTNJ.js.map} +0 -0
- /package/bundle/{agent-registry-5VL5KI6U.js.map → agent-registry-ABPFQXNL.js.map} +0 -0
- /package/bundle/{chunk-PUDGA4RR.js.map → chunk-2SFN2VYD.js.map} +0 -0
- /package/bundle/{chunk-7WFE2JI5.js.map → chunk-7B3GK5JQ.js.map} +0 -0
- /package/bundle/{agent-registry-PIS5XJHX.js.map → commands-L6VIMPCR.js.map} +0 -0
- /package/bundle/{discord-adapter-JFIIVG34.js.map → discord-adapter-DWIQRNDI.js.map} +0 -0
- /package/bundle/{commands-IGRSOSK6.js.map → install-Q4XNCPG7.js.map} +0 -0
- /package/bundle/{commands-LAWVNQTO.js.map → message-pipeline-GCSZCQWO.js.map} +0 -0
- /package/bundle/{commands-RBWY7YXB.js.map → phase-transport-F7GQRRYE.js.map} +0 -0
- /package/bundle/{commands-XFZNMZN6.js.map → sleep-MYOZ73IU.js.map} +0 -0
- /package/bundle/{install-I3CXVW52.js.map → subagent-runtime-QA4LVU4C.js.map} +0 -0
|
@@ -4,8 +4,8 @@ import {
|
|
|
4
4
|
buildTransport,
|
|
5
5
|
phaseTransport,
|
|
6
6
|
rebuildTransport
|
|
7
|
-
} from "./chunk-
|
|
8
|
-
import "./chunk-
|
|
7
|
+
} from "./chunk-6TSCOXF6.js";
|
|
8
|
+
import "./chunk-7B3GK5JQ.js";
|
|
9
9
|
import "./chunk-PKHYCNTT.js";
|
|
10
10
|
import "./chunk-RTL7HO3N.js";
|
|
11
11
|
import "./chunk-SRFEIZQT.js";
|
|
@@ -20,4 +20,4 @@ export {
|
|
|
20
20
|
phaseTransport,
|
|
21
21
|
rebuildTransport
|
|
22
22
|
};
|
|
23
|
-
//# sourceMappingURL=phase-transport-
|
|
23
|
+
//# sourceMappingURL=phase-transport-F7GQRRYE.js.map
|
|
@@ -3,7 +3,7 @@
|
|
|
3
3
|
import {
|
|
4
4
|
createSleepHandle,
|
|
5
5
|
register
|
|
6
|
-
} from "./chunk-
|
|
6
|
+
} from "./chunk-HFPXN6NM.js";
|
|
7
7
|
import "./chunk-HAF2AFBW.js";
|
|
8
8
|
import "./chunk-JAJ3DUQ2.js";
|
|
9
9
|
import "./chunk-RTL7HO3N.js";
|
|
@@ -17,4 +17,4 @@ export {
|
|
|
17
17
|
createSleepHandle,
|
|
18
18
|
register
|
|
19
19
|
};
|
|
20
|
-
//# sourceMappingURL=sleep-
|
|
20
|
+
//# sourceMappingURL=sleep-MYOZ73IU.js.map
|
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
import { createRequire as __bundleCreateRequire } from 'node:module'; import { fileURLToPath as __bundleFileURLToPath } from 'node:url'; import { dirname as __bundleDirname } from 'node:path'; const require = __bundleCreateRequire(import.meta.url); const __chunk_filename = __bundleFileURLToPath(import.meta.url); const __chunk_dirname = __bundleDirname(__chunk_filename);
|
|
3
3
|
import {
|
|
4
4
|
SubagentRuntime
|
|
5
|
-
} from "./chunk-
|
|
5
|
+
} from "./chunk-2SFN2VYD.js";
|
|
6
6
|
import "./chunk-CYSGXNBY.js";
|
|
7
7
|
import "./chunk-GUTRAMK3.js";
|
|
8
8
|
import "./chunk-WW5F2DCO.js";
|
|
@@ -10,4 +10,4 @@ import "./chunk-7K2YZTLD.js";
|
|
|
10
10
|
export {
|
|
11
11
|
SubagentRuntime
|
|
12
12
|
};
|
|
13
|
-
//# sourceMappingURL=subagent-runtime-
|
|
13
|
+
//# sourceMappingURL=subagent-runtime-QA4LVU4C.js.map
|
|
@@ -135,8 +135,10 @@ function phaseToService(phaseName) {
|
|
|
135
135
|
function renderStatusText(status) {
|
|
136
136
|
const uptime = formatUptime(status.uptimeMs);
|
|
137
137
|
const name = process.env["AGENT_NAME"] ?? process.env["BOT_NAME"] ?? "abtars";
|
|
138
|
+
const failures = status.subsystems.filter((s) => s.status === "failed").length;
|
|
139
|
+
const mood = failures === 0 ? "\u{1F60A}" : failures <= 2 ? "\u{1F610}" : "\u{1F61F}";
|
|
138
140
|
const lines = [
|
|
139
|
-
|
|
141
|
+
`abTARS\u2122 \u2014 online ${mood}`,
|
|
140
142
|
` PID ${process.pid} (up ${uptime})`
|
|
141
143
|
];
|
|
142
144
|
try {
|
|
@@ -194,4 +196,4 @@ export {
|
|
|
194
196
|
getSystemStatus,
|
|
195
197
|
renderStatusText
|
|
196
198
|
};
|
|
197
|
-
//# sourceMappingURL=system-status-
|
|
199
|
+
//# sourceMappingURL=system-status-KMKPAC5Z.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../src/components/system-status.ts"],
|
|
4
|
+
"sourcesContent": ["/**\n * system-status.ts \u2014 Single source of truth for bridge status.\n * One collector, many renderers. Used by /status (text), dashboard (HTML), future API (JSON).\n */\n\nimport { logAndSwallow } from \"./log-and-swallow.js\";\nimport { readFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport { homedir } from \"node:os\";\nimport type { ServiceState } from \"./service-registry.js\";\n\nexport interface SubsystemHealth {\n name: string;\n status: \"ok\" | \"failed\" | \"skipped\" | \"stopped\" | \"retrying\";\n detail?: string;\n}\n\nexport interface SystemStatus {\n version: string;\n commit: string;\n model: string;\n transportType: string;\n transportProvider: string;\n transportReady: boolean;\n uptimeMs: number;\n contextPercent: number;\n sleepStatus: string | null;\n subsystems: SubsystemHealth[];\n tasks: { recurring: number; pending: number; paused: number };\n lastBackup: string | null;\n}\n\nexport interface StatusContext {\n phaseHealth: Map<string, { status: \"ok\" | \"failed\" | \"skipped\"; error?: string }>;\n registry: { getStates(): Record<string, ServiceState> };\n transport: { isReady: boolean } | null;\n startedAt: number;\n bridgeLockPath: string;\n heartbeat: { intervalMs: number } | null;\n}\n\nexport async function getSystemStatus(ctx: StatusContext): Promise<SystemStatus> {\n // Version + commit from manifest.json (single source of truth)\n const { getDeployedVersion } = await import(\"../paths.js\");\n const deployed = getDeployedVersion();\n const version = deployed.version;\n const commit = deployed.commit || \"?\";\n\n // Model + transport\n let model = \"unknown\";\n let transportType = \"unknown\";\n let transportProvider = \"unknown\";\n try {\n const { loadTransport, resolveAgent } = await import(\"./transport-config.js\");\n const tc = loadTransport();\n const prof = tc ? resolveAgent(\"professor\", tc) : null;\n if (prof) {\n model = prof.model;\n transportType = (prof.provider.transport ?? \"acp\").toUpperCase();\n transportProvider = prof.providerName ?? \"unknown\";\n }\n } catch (err) { logAndSwallow(\"system_status\", \"op\", err); }\n\n // Subsystem health: merge phaseHealth + ServiceRegistry live state\n const serviceStates = ctx.registry.getStates();\n const subsystems: SubsystemHealth[] = [];\n\n for (const [name, health] of ctx.phaseHealth) {\n const entry: SubsystemHealth = { name, status: health.status };\n\n // For platform services, override with live ServiceRegistry state\n const svcName = phaseToService(name);\n if (svcName && serviceStates[svcName]) {\n const svc = serviceStates[svcName] as ServiceState;\n if (health.status === \"ok\" && !svc.running) {\n entry.status = svc.retrying ? \"retrying\" : \"stopped\";\n }\n if (svc.retrying) {\n entry.detail = `retry #${svc.retrying.attempt}`;\n }\n }\n\n // Add contextual detail for specific phases\n if (name === \"phaseTransport\" && health.status === \"ok\" && ctx.transport) {\n entry.detail = `${transportType} (${transportProvider}), ${ctx.transport.isReady ? \"ready\" : \"not ready\"}`;\n }\n if (name === \"phaseHeartbeat\" && health.status === \"ok\" && ctx.heartbeat) {\n try {\n const lock = JSON.parse(readFileSync(ctx.bridgeLockPath, \"utf-8\"));\n const ago = Math.round((Date.now() - (lock.lastHeartbeat || 0)) / 60000);\n entry.detail = `${Math.round(ctx.heartbeat.intervalMs / 1000)}s, last tick ${ago}m ago`;\n } catch (err) { logAndSwallow(\"system_status\", \"op\", err); }\n }\n if (name === \"phaseDashboard\" && health.status === \"ok\") {\n entry.detail = `:${process.env[\"WEB_PORT\"] || \"3000\"}`;\n }\n\n subsystems.push(entry);\n }\n\n // Tasks\n let tasks = { recurring: 0, pending: 0, paused: 0 };\n try {\n const { readEntries } = await import(\"./tasks/task-store.js\");\n const entries = readEntries();\n tasks = {\n recurring: entries.filter((e: any) => e.schedule && !e.paused).length,\n pending: entries.filter((e: any) => !e.fired && !e.schedule).length,\n paused: entries.filter((e: any) => e.paused).length,\n };\n } catch (err) { logAndSwallow(\"system_status\", \"op\", err); }\n\n // Last backup\n let lastBackup: string | null = null;\n try {\n const { readdirSync } = await import(\"node:fs\");\n const bd = join(homedir(), \".backup-abtars\");\n const bk = readdirSync(bd).filter((f: string) => f.startsWith(\"abtars-\")).sort();\n if (bk.length > 0) lastBackup = bk[bk.length - 1] ?? null;\n } catch (err) { logAndSwallow(\"system_status\", \"op\", err); }\n\n // Context percent from bridge.lock\n let contextPercent = -1;\n try {\n const lock = JSON.parse(readFileSync(ctx.bridgeLockPath, \"utf-8\"));\n if (typeof lock.contextPercent === \"number\") contextPercent = lock.contextPercent;\n } catch { /* ignore */ }\n\n // Sleep status\n let sleepStatus: string | null = null;\n try {\n const lock = JSON.parse(readFileSync(ctx.bridgeLockPath, \"utf-8\"));\n sleepStatus = lock.sleepStatus ?? null;\n } catch { /* ignore */ }\n\n return {\n version,\n commit,\n model,\n transportType,\n transportProvider,\n transportReady: ctx.transport?.isReady ?? false,\n uptimeMs: Date.now() - ctx.startedAt,\n contextPercent,\n sleepStatus,\n subsystems,\n tasks,\n lastBackup,\n };\n}\n\n/** Map phase names to ServiceRegistry service names (for live state merge). */\nfunction phaseToService(phaseName: string): string | null {\n switch (phaseName) {\n case \"phasePlatforms\": return null; // platforms registers telegram + discord separately\n case \"phaseDashboard\": return null;\n case \"phaseAgentApi\": return \"agent-api\";\n default: return null;\n }\n}\n\n/** Render SystemStatus as plain text for Telegram/Discord /status command. */\nexport function renderStatusText(status: SystemStatus): string {\n const uptime = formatUptime(status.uptimeMs);\n const name = process.env[\"AGENT_NAME\"] ?? process.env[\"BOT_NAME\"] ?? \"abtars\";\n const failures = status.subsystems.filter(s => s.status === \"failed\").length;\n const mood = failures === 0 ? \"\uD83D\uDE0A\" : failures <= 2 ? \"\uD83D\uDE10\" : \"\uD83D\uDE1F\";\n const lines: string[] = [\n `abTARS\u2122 \u2014 online ${mood}`,\n ` PID ${process.pid} (up ${uptime})`,\n ];\n\n // Watchdog\n try {\n const { execFileSync } = require(\"node:child_process\") as typeof import(\"node:child_process\");\n const wdPid = execFileSync(\"pgrep\", [\"-f\", \"watchdog.sh\"], { encoding: \"utf-8\", timeout: 2000 }).trim().split(\"\\n\")[0];\n if (wdPid) lines.push(` Watchdog: PID ${wdPid} (bash)`);\n else lines.push(\" Watchdog: not running\");\n } catch { lines.push(\" Watchdog: not detected\"); }\n\n // Platforms\n const platforms = status.subsystems.find(s => s.name === \"phasePlatforms\");\n if (platforms?.detail) lines.push(` Platforms: ${platforms.detail}`);\n\n // Model + context\n lines.push(` Model: ${status.model.split(\"/\").pop() ?? status.model}`);\n if (status.contextPercent >= 0) lines.push(` Context: ${status.contextPercent}%`);\n\n // Last prompt\n try {\n const lock = JSON.parse(readFileSync(join(homedir(), \".abtars\", \"bridge.lock\"), \"utf-8\"));\n if (lock.lastPromptAt) {\n const ago = Math.round((Date.now() - lock.lastPromptAt) / 1000);\n lines.push(` Last prompt: ${ago < 60 ? `${ago}s ago` : `${Math.round(ago / 60)}m ago`}`);\n }\n } catch { /* no lock */ }\n\n // Sleep\n if (status.sleepStatus) lines.push(` Sleep: ${status.sleepStatus}`);\n\n // Skills\n try {\n const { getSkillCache } = require(\"../capabilities/hotskills/index.js\") as typeof import(\"../capabilities/hotskills/index.js\");\n const skills = getSkillCache();\n if (skills.length > 0) {\n const active = skills.filter(s => !s.skipped).length;\n lines.push(` Skills: ${active} active`);\n }\n } catch { /* hotskills not loaded */ }\n\n lines.push(\"\", \"\uD83C\uDFE5 Subsystems:\");\n\n for (const s of status.subsystems) {\n const icon = s.status === \"ok\" ? \"\u2713\" : s.status === \"skipped\" ? \"\u25CB\" : \"\u2717\";\n const label = s.name.replace(\"phase\", \"\").replace(/([A-Z])/g, \" $1\").trim().toLowerCase();\n const detail = s.detail ? ` \u2014 ${s.detail}` : \"\";\n lines.push(` ${icon} ${label}${detail}`);\n }\n\n lines.push(\"\");\n lines.push(`\u23F0 Tasks: ${status.tasks.recurring} recurring, ${status.tasks.pending} pending${status.tasks.paused ? `, ${status.tasks.paused} paused` : \"\"}`);\n if (status.lastBackup) lines.push(`\uD83D\uDCBE Last backup: ${status.lastBackup}`);\n\n return lines.join(\"\\n\");\n}\n\nfunction formatUptime(ms: number): string {\n const s = Math.floor(ms / 1000);\n const m = Math.floor(s / 60) % 60;\n const h = Math.floor(s / 3600) % 24;\n const d = Math.floor(s / 86400);\n if (d > 0) return `${d}d ${h}h ${m}m`;\n if (h > 0) return `${h}h ${m}m`;\n return `${m}m`;\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|
|
@@ -6,14 +6,14 @@ import {
|
|
|
6
6
|
SecurityGate,
|
|
7
7
|
emojiToScore,
|
|
8
8
|
emojiToTag
|
|
9
|
-
} from "./chunk-
|
|
9
|
+
} from "./chunk-2W6JIHZ5.js";
|
|
10
10
|
import {
|
|
11
11
|
cleanResponse,
|
|
12
12
|
formatReactionSignal,
|
|
13
13
|
handleInboundMessage,
|
|
14
14
|
resetAndPrepare,
|
|
15
15
|
routeReaction
|
|
16
|
-
} from "./chunk-
|
|
16
|
+
} from "./chunk-SEXVA3GK.js";
|
|
17
17
|
import "./chunk-NIRYBWUW.js";
|
|
18
18
|
import "./chunk-L33WNMCP.js";
|
|
19
19
|
import "./chunk-SA6YEFNG.js";
|
|
@@ -257,6 +257,7 @@ var TelegramPoller = class {
|
|
|
257
257
|
onUpdate;
|
|
258
258
|
offsetStore;
|
|
259
259
|
offset = 0;
|
|
260
|
+
lastProcessedId = 0;
|
|
260
261
|
running = false;
|
|
261
262
|
abortController = null;
|
|
262
263
|
/** Timestamp of last successful poll cycle. */
|
|
@@ -302,6 +303,7 @@ var TelegramPoller = class {
|
|
|
302
303
|
if (updates.length > 0) {
|
|
303
304
|
const sorted = [...updates].sort((a, b) => a.update_id - b.update_id);
|
|
304
305
|
for (const update of sorted) {
|
|
306
|
+
if (update.update_id <= this.lastProcessedId) continue;
|
|
305
307
|
try {
|
|
306
308
|
const result = this.onUpdate(update);
|
|
307
309
|
if (result instanceof Promise) {
|
|
@@ -311,6 +313,7 @@ var TelegramPoller = class {
|
|
|
311
313
|
logError("poller", "Error in update handler (sync)", err);
|
|
312
314
|
}
|
|
313
315
|
}
|
|
316
|
+
this.lastProcessedId = sorted[sorted.length - 1].update_id;
|
|
314
317
|
this.offset = sorted[sorted.length - 1].update_id + 1;
|
|
315
318
|
await this.offsetStore.write(this.offset);
|
|
316
319
|
}
|
|
@@ -1021,7 +1024,7 @@ ${text}`;
|
|
|
1021
1024
|
const tag = emojiToTag(emojis[0]);
|
|
1022
1025
|
const updated = this.deps.memory.updateEmotionByPlatformId(loadUsers().byPlatformId.get(`telegram:${chatId}`)?.userId ?? "master", reaction.message_id, score, tag);
|
|
1023
1026
|
if (updated) logDebug(TAG4, `Emotion score ${score} set on platform msg ${reaction.message_id}`);
|
|
1024
|
-
const { getRecalledIdsForMessage } = await import("./message-pipeline-
|
|
1027
|
+
const { getRecalledIdsForMessage } = await import("./message-pipeline-GCSZCQWO.js");
|
|
1025
1028
|
const recalledIds = getRecalledIdsForMessage(reaction.message_id);
|
|
1026
1029
|
if (recalledIds && recalledIds.length > 0) {
|
|
1027
1030
|
if (score < 0) {
|
|
@@ -1077,4 +1080,4 @@ export {
|
|
|
1077
1080
|
TELEGRAM_CAPABILITIES,
|
|
1078
1081
|
TelegramAdapter
|
|
1079
1082
|
};
|
|
1080
|
-
//# sourceMappingURL=telegram-adapter-
|
|
1083
|
+
//# sourceMappingURL=telegram-adapter-TRMCC634.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../src/platforms/telegram/telegram-adapter.ts", "../src/platforms/telegram/telegram-api.ts", "../src/platforms/telegram/telegram-poller.ts", "../src/platforms/telegram/offset-store.ts", "../src/platforms/telegram/telegram-model-picker.ts"],
|
|
4
|
+
"sourcesContent": ["import { BOT_COMMANDS } from \"../../components/command-registry.js\";\nimport { abtarsHome } from \"../../paths.js\";\n/**\n * Telegram platform adapter \u2014 wraps TelegramApi, TelegramPoller, SecurityGate.\n * Handles Telegram-specific pre-processing (voice, reactions, groups, mentions)\n * then delegates to the shared message pipeline.\n */\n\nimport { TelegramApi } from \"./telegram-api.js\";\nimport { TelegramPoller } from \"./telegram-poller.js\";\nimport { createFileOffsetStore } from \"./offset-store.js\";\nimport { SecurityGate } from \"../../components/security-gate.js\";\nimport { ResponseFormatter } from \"../../components/response-formatter.js\";\nimport { formatReactionSignal, routeReaction } from \"../../components/reactions.js\";\nimport { cleanResponse } from \"../../components/clean-response.js\";\n\nexport const TELEGRAM_CAPABILITIES: PlatformCapabilities = { voice: true, reactions: true, typing: true, threads: true };\nimport { emojiToScore, emojiToTag } from \"../../utils/emoji-score.js\";\nimport { logInfo, logWarn, logError, logDebug } from \"../../components/logger.js\";\nimport { logAndSwallow } from \"../../components/log-and-swallow.js\";\nimport { handleInboundMessage, resetAndPrepare, type PipelineDeps } from \"../../components/message-pipeline.js\";\nimport type { PlatformAdapter, PlatformCapabilities, InboundMessage, SendOpts } from \"../../types/platform.js\";\nimport type { TelegramUpdate } from \"../../types/index.js\";\nimport type { ConversationBuffer } from \"../../components/conversation-buffer.js\";\nimport type { IKiroTransport } from \"../../components/transport/kiro-transport.js\";\nimport type { IMemorySystem } from \"abmind\";\nimport { loadUsers } from \"../../components/user-registry.js\";\nimport { isModelPickerCallback, handleModelPickerCallback } from \"./telegram-model-picker.js\";\n\nconst TAG = \"telegram\";\n\nexport interface TelegramAdapterConfig {\n botToken: string;\n allowedUserIds: Set<number>;\n pollTimeoutS: number;\n}\n\nexport interface TelegramAdapterDeps {\n pipeline: PipelineDeps;\n conversationBuffer: ConversationBuffer;\n transport: IKiroTransport;\n memory: IMemorySystem | null;\n sessionManager: { getActiveSessionId(userId: string, platform: string): string };\n actionGate?: { handleCallback(data: string): boolean } | null;\n}\n\nexport class TelegramAdapter implements PlatformAdapter {\n readonly name = \"telegram\" as const;\n readonly capabilities: PlatformCapabilities = TELEGRAM_CAPABILITIES;\n\n private readonly api: TelegramApi;\n private readonly securityGate: SecurityGate;\n private readonly formatter = new ResponseFormatter();\n private readonly config: TelegramAdapterConfig;\n private readonly deps: TelegramAdapterDeps;\n private poller: TelegramPoller | null = null;\n private botUsername = \"\";\n _pendingSlot: string | undefined;\n _modelPickerCache: string[] = [];\n\n constructor(config: TelegramAdapterConfig, deps: TelegramAdapterDeps) {\n this.api = new TelegramApi(config.botToken);\n this.securityGate = new SecurityGate(loadUsers());\n this.config = config;\n this.deps = deps;\n }\n\n /** Send a system notification to a chat (fire-and-forget). */\n sendNotification(chatId: string, text: string): void {\n this.api.sendMessage(parseInt(chatId, 10), text).catch(err => logAndSwallow(TAG, \"sendNotification\", err));\n }\n\n /** Reset session after model switch \u2014 saves idle state, clears buffer, marks pendingStart. */\n private async resetSessionForModelSwitch(chatId: number, reason = \"model-switch\"): Promise<void> {\n await this.api.sendMessage(chatId, \"Resetting session\u2026\");\n const p = this.deps.pipeline;\n const sessionKey = `telegram:${chatId}`;\n const bufKey = `telegram:${chatId}`;\n await p.idleSave.save(sessionKey, chatId);\n await resetAndPrepare({\n transport: this.deps.transport, sessionKey, reason,\n sessions: p.sessions, conversationBuffer: this.deps.conversationBuffer, bufKey,\n });\n if (p.memoryConfig.memoryEnabled) {\n const reg = loadUsers();\n const user = reg.byPlatformId.get(String(chatId));\n if (user) p.updateCtxStart(p.memoryConfig.memoryDir, user.userId);\n }\n }\n\n async start(): Promise<void> {\n const botInfo = await this.api.getMe();\n this.botUsername = botInfo.username?.toLowerCase() ?? \"\";\n logInfo(TAG, `\uD83E\uDD16 Bot: @${botInfo.username}`);\n\n await this.api.setMyCommands(BOT_COMMANDS.map(c => ({ command: c.name, description: c.description })))\n .catch((err) => logWarn(TAG, `setMyCommands failed: ${err instanceof Error ? err.message : String(err)}`));\n\n const home = abtarsHome();\n const offsetStore = createFileOffsetStore(`${home}/state/telegram-offset`);\n this.poller = new TelegramPoller(this.api, this.config.pollTimeoutS, (u) => this.handleUpdate(u), offsetStore);\n await this.poller.start();\n }\n\n stop(): void {\n this.poller?.stop();\n this.poller = null;\n }\n\n authorize(msg: InboundMessage): boolean {\n return this.securityGate.authorizeById(msg.senderId);\n }\n\n async sendMessage(channelId: string, text: string, opts?: SendOpts): Promise<number | undefined> {\n const chatId = parseInt(channelId, 10);\n const sendOpts: Record<string, unknown> = {};\n if (opts?.threadId) sendOpts.message_thread_id = parseInt(opts.threadId, 10);\n if (opts?.parseMode) sendOpts.parse_mode = opts.parseMode;\n if (opts?.reply_markup) sendOpts.reply_markup = opts.reply_markup;\n return this.api.sendMessage(chatId, text, sendOpts);\n }\n\n async editMessage(channelId: string, messageId: number, text: string): Promise<void> {\n const chatId = parseInt(channelId, 10);\n await this.api.editMessageText(chatId, messageId, text);\n }\n\n chunkResponse(text: string): string[] {\n return this.formatter.chunkText(text);\n }\n\n async sendTyping(channelId: string, threadId?: string): Promise<void> {\n await this.api.sendChatAction(parseInt(channelId, 10), \"typing\", threadId ? parseInt(threadId, 10) : undefined);\n }\n\n async setReaction(channelId: string, messageId: number, emoji: string): Promise<void> {\n if (messageId <= 0) return;\n try {\n const reaction = emoji ? [{ type: \"emoji\" as const, emoji }] : [];\n await this.api.setMessageReaction(parseInt(channelId, 10), messageId, reaction);\n } catch (err) {\n logDebug(TAG, `React failed (${emoji || \"remove\"}): ${err instanceof Error ? err.message : String(err)}`);\n }\n }\n\n async downloadVoice(fileId: string): Promise<Buffer> {\n const fileInfo = await this.api.getFile(fileId);\n if (!fileInfo.file_path) throw new Error(\"No file_path returned\");\n return this.api.downloadFile(fileInfo.file_path);\n }\n\n async sendVoice(channelId: string, audio: Buffer, opts?: SendOpts): Promise<void> {\n const sendOpts: Record<string, unknown> = {};\n if (opts?.threadId) sendOpts.message_thread_id = parseInt(opts.threadId, 10);\n await this.api.sendVoice(parseInt(channelId, 10), audio, sendOpts);\n }\n\n /** Send a file from disk as a Telegram document. */\n async sendDocument(channelId: string, filePath: string, caption?: string, opts?: SendOpts): Promise<number> {\n const sendOpts: { message_thread_id?: number } = {};\n if (opts?.threadId) sendOpts.message_thread_id = parseInt(opts.threadId, 10);\n return this.api.sendDocument(parseInt(channelId, 10), filePath, caption, sendOpts);\n }\n\n injectMessage(msg: InboundMessage): void {\n if (!this.poller) return;\n this.poller.injectUpdate({\n update_id: 0,\n message: {\n message_id: 0,\n from: { id: parseInt(msg.channelId, 10), is_bot: false, first_name: \"queued\" },\n chat: { id: parseInt(msg.channelId, 10), type: \"private\" },\n date: Math.floor(Date.now() / 1000),\n text: msg.text,\n ...(msg.threadId ? { message_thread_id: parseInt(msg.threadId, 10) } : {}),\n },\n });\n }\n\n // --- Internal: Telegram update handler ---\n\n private async handleUpdate(update: TelegramUpdate): Promise<void> {\n logDebug(TAG, `Update: ${JSON.stringify(update).slice(0, 200)}`);\n\n if (update.callback_query) {\n const data = update.callback_query.data ?? \"\";\n const chatId = update.callback_query.message?.chat?.id;\n this.api.answerCallbackQuery(update.callback_query.id).catch(err => logAndSwallow(TAG, \"answerCallbackQuery\", err));\n if (!chatId) return;\n\n try {\n if (isModelPickerCallback(data)) {\n await handleModelPickerCallback(data, chatId, this.api, this, {\n transport: this.deps.transport as any,\n pipeline: this.deps.pipeline as any,\n resetSessionForModelSwitch: (cid, reason) => this.resetSessionForModelSwitch(cid, reason),\n });\n } else if (data.startsWith(\"model:\")) {\n // Legacy callback \u2014 direct model switch\n const newModel = data.slice(6);\n const transport = this.deps.transport;\n if (\"setModel\" in transport && typeof (transport as { setModel: unknown }).setModel === \"function\") {\n try {\n await (transport as { setModel: (m: string) => Promise<void> | void }).setModel(newModel);\n await this.resetSessionForModelSwitch(chatId);\n if (chatId) await this.api.sendMessage(chatId, `\uD83E\uDD16 Model switched \u2192 ${newModel}`);\n } catch (err) {\n if (chatId) await this.api.sendMessage(chatId, `\u274C Model switch failed: ${err instanceof Error ? err.message : String(err)}`);\n }\n }\n } else if (data.startsWith(\"auth:\")) {\n // ActionGate authorization callback\n if (this.deps.actionGate?.handleCallback(data)) {\n await this.api.sendMessage(chatId, \"\u2713 Responded.\");\n }\n }\n } catch (err) {\n logWarn(TAG, `Callback handler error: ${err instanceof Error ? err.message : String(err)}`);\n this.api.sendMessage(chatId, `\u26A0\uFE0F Action failed \u2014 try again.`).catch(err2 => logAndSwallow(TAG, \"sendMessage action failed\", err2));\n }\n return;\n }\n\n if (update.message_reaction) {\n await this.handleReaction(update);\n return;\n }\n\n const message = update.message;\n if (!message?.from) return;\n\n const hasText = Boolean(message.text);\n const hasVoice = Boolean(message.voice || message.audio);\n const hasPhoto = Boolean(message.photo?.length);\n const hasDocument = Boolean(message.document);\n if (!hasText && !hasVoice && !hasPhoto && !hasDocument) return;\n\n const chatId = message.chat.id;\n const isGroup = message.chat.type === \"group\" || message.chat.type === \"supergroup\";\n const threadId = isGroup ? message.message_thread_id : undefined;\n const messageId = message.message_id;\n const senderName = message.from.first_name || message.from.username || `id:${message.from.id}`;\n const bufKey = threadId != null ? `tg:${chatId}:${threadId}` : `tg:${chatId}`;\n\n let text = message.text ?? \"\";\n let isVoiceNote = false;\n let voiceFileId: string | undefined;\n\n // --- Voice note pre-processing ---\n if (hasVoice && !hasText) {\n if (!this.deps.pipeline.sttConfig) {\n if (isGroup) {\n this.deps.conversationBuffer.push(bufKey, senderName, \"[voice note - STT disabled]\");\n } else if (this.securityGate.authorizeById(String(message.from?.id))) {\n await this.api.sendMessage(chatId, \"\uD83C\uDFA4 Voice notes require STT (set GROQ_API_KEY).\", { message_thread_id: threadId });\n }\n return;\n }\n\n if (!this.securityGate.authorizeById(String(message.from?.id))) {\n if (isGroup) this.deps.conversationBuffer.push(bufKey, senderName, \"[voice note]\");\n return;\n }\n\n // For groups: we need to transcribe first to check for bot mention\n if (isGroup) {\n try {\n await this.setReaction(String(chatId), messageId, \"\uD83D\uDC40\");\n const voiceFile = message.voice || message.audio;\n const audioBuffer = await this.downloadVoice(voiceFile!.file_id);\n const { transcribeAudio } = await import(\"../../components/stt.js\");\n const result = await transcribeAudio(audioBuffer, \"voice.ogg\", this.deps.pipeline.sttConfig!);\n const transcript = result.text;\n\n if (!transcript) {\n await this.setReaction(String(chatId), messageId, \"\");\n this.deps.conversationBuffer.push(bufKey, senderName, \"[voice note - empty]\");\n return;\n }\n\n const mentionRe = new RegExp(`@?${this.botUsername}\\\\b`, \"i\");\n if (!mentionRe.test(transcript) && !transcript.startsWith(\"/\")) {\n await this.setReaction(String(chatId), messageId, \"\");\n this.deps.conversationBuffer.push(bufKey, senderName, `[voice] ${transcript}`);\n logDebug(TAG, `Buffered voice transcript: \"${transcript.slice(0, 60)}\"`);\n return;\n }\n text = transcript.replace(mentionRe, \"\").trim();\n isVoiceNote = true;\n if (!text) { await this.setReaction(String(chatId), messageId, \"\"); return; }\n } catch (err) {\n logError(TAG, \"Voice transcription failed\", err);\n await this.setReaction(String(chatId), messageId, \"\");\n return;\n }\n } else {\n // DM voice: let pipeline handle STT\n isVoiceNote = true;\n voiceFileId = (message.voice || message.audio)!.file_id;\n }\n }\n\n // --- Group text filtering ---\n if (!isVoiceNote && isGroup) {\n const mentionRe = new RegExp(`@${this.botUsername}\\\\b`, \"i\");\n const isMention = mentionRe.test(text);\n const isCommand = text.startsWith(\"/\");\n\n if (!isMention && !isCommand) {\n this.deps.conversationBuffer.push(bufKey, senderName, text);\n logDebug(TAG, `Buffered group msg from ${senderName}: \"${text.slice(0, 60)}\"`);\n return;\n }\n\n if (isMention) {\n text = text.replace(mentionRe, \"\").trim();\n if (!text) return;\n }\n }\n\n // --- Security ---\n if (!isVoiceNote && !this.securityGate.authorizeById(String(message.from?.id))) {\n if (isGroup) this.deps.conversationBuffer.push(bufKey, senderName, text);\n logWarn(TAG, `Unauthorized user ${message.from.id}`);\n return;\n }\n\n // --- Photo/document handling ---\n let mediaPath: string | undefined;\n\n if ((hasPhoto || hasDocument) && this.securityGate.authorizeById(String(message.from?.id))) {\n try {\n const { saveInboundMedia } = await import(\"../../components/media-utils.js\");\n let fileId: string;\n let extHint: string | undefined;\n let claimedMime: string | undefined;\n\n if (hasPhoto) {\n const photo = message.photo![message.photo!.length - 1]!;\n fileId = photo.file_id;\n extHint = \".jpg\";\n } else {\n fileId = message.document!.file_id;\n extHint = message.document!.file_name ? \".\" + (message.document!.file_name.split(\".\").pop() ?? \"\") : undefined;\n claimedMime = message.document!.mime_type;\n }\n\n const buf = await this.downloadVoice(fileId); // reuse file download\n const saved = await saveInboundMedia(buf, chatId, { extHint, claimedMime });\n if (saved) {\n mediaPath = saved.path;\n if (!text) text = message.caption ?? `User sent a ${saved.isImage ? \"photo\" : \"file\"}.`;\n } else {\n if (!text) text = \"\u26A0\uFE0F File too large (max 16MB).\";\n }\n } catch (err) {\n logWarn(TAG, `Media download failed: ${err instanceof Error ? err.message : String(err)}`);\n if (!text) text = \"\u26A0\uFE0F Failed to download media.\";\n }\n }\n\n // --- Reply context ---\n const reply = message.reply_to_message;\n if (reply?.text) {\n const replyFrom = reply.from?.first_name ?? \"someone\";\n text = `[Replying to ${replyFrom}: \"${reply.text.slice(0, 500)}\"]\\n${text}`;\n }\n\n // --- Reply-to-photo: re-download and inject (#668) ---\n if (reply?.photo?.length && !mediaPath && this.securityGate.authorizeById(String(message.from?.id))) {\n try {\n const { saveInboundMedia } = await import(\"../../components/media-utils.js\");\n const photo = reply.photo[reply.photo.length - 1]!;\n const buf = await this.downloadVoice(photo.file_id);\n const saved = await saveInboundMedia(buf, chatId, { extHint: \".jpg\" });\n if (saved) mediaPath = saved.path;\n } catch (err) {\n logWarn(TAG, `Reply-to-photo download failed: ${err instanceof Error ? err.message : String(err)}`);\n }\n }\n\n // --- Dispatch to pipeline ---\n const resolvedUser = loadUsers().byPlatformId.get(\"telegram:\" + message.from.id)?.userId ?? \"unknown\";\n const inbound: InboundMessage = {\n platform: \"telegram\",\n channelId: String(chatId),\n userId: resolvedUser,\n senderId: String(message.from.id),\n senderName,\n text,\n timestamp: message.date * 1000,\n threadId: threadId != null ? String(threadId) : undefined,\n messageId,\n isGroup,\n isVoice: isVoiceNote,\n voiceFileId,\n mediaPath,\n rawPlatformData: message,\n };\n\n // /stop, /ctrlc, /restart now handled in commandMiddleware (platform-agnostic)\n\n // #512: commands bypass the sequential await \u2014 execute immediately even if agent is mid-stream\n if (text.startsWith(\"/\") && !text.startsWith(\"//\")) {\n handleInboundMessage(inbound, this, this.deps.pipeline).catch((err) => {\n logError(TAG, `Command dispatch error: ${err instanceof Error ? err.message : String(err)}`);\n });\n return;\n }\n\n await handleInboundMessage(inbound, this, this.deps.pipeline);\n }\n\n private async handleReaction(update: TelegramUpdate): Promise<void> {\n const reaction = update.message_reaction!;\n const user = reaction.user;\n if (!user) { logDebug(TAG, \"Reaction update missing user field\"); return; }\n if (user.is_bot) return;\n\n const oldEmojis = new Set(reaction.old_reaction.map((r) => r.emoji));\n const added = reaction.new_reaction.filter((r) => !oldEmojis.has(r.emoji));\n if (added.length === 0) return;\n\n const senderName = user.first_name || user.username || `id:${user.id}`;\n const emojis = added.map((r) => r.emoji);\n logInfo(TAG, `Reaction ${emojis.join(\"\")} from ${senderName} on msg ${reaction.message_id}`);\n\n const isAuthorized = this.securityGate.authorizeById(String(user.id));\n const signal = formatReactionSignal(senderName, emojis);\n const chatId = reaction.chat.id;\n const route = routeReaction(isAuthorized, reaction.chat.type);\n\n if (isAuthorized && this.deps.memory) {\n const score = emojiToScore(emojis[0]!);\n const tag = emojiToTag(emojis[0]!);\n const updated = this.deps.memory.updateEmotionByPlatformId(loadUsers().byPlatformId.get(`telegram:${chatId}`)?.userId ?? \"master\", reaction.message_id, score, tag);\n if (updated) logDebug(TAG, `Emotion score ${score} set on platform msg ${reaction.message_id}`);\n\n // #824: Emoji reaction as recall quality feedback\n const { getRecalledIdsForMessage } = await import(\"../../components/message-pipeline.js\");\n const recalledIds = getRecalledIdsForMessage(reaction.message_id);\n if (recalledIds && recalledIds.length > 0) {\n if (score < 0) {\n this.deps.memory.bumpRejectedCount(recalledIds);\n logDebug(TAG, `Recall rejection: ${recalledIds.length} memories penalized (emoji ${emojis[0]})`);\n } else if (score > 0) {\n this.deps.memory.bumpCitedCount(recalledIds);\n logDebug(TAG, `Recall confirmed: ${recalledIds.length} memories boosted (emoji ${emojis[0]})`);\n }\n }\n }\n\n if (route === \"discard\") {\n logDebug(TAG, `Unauthorized reaction from user ${user.id}, discarding`);\n return;\n }\n\n if (route === \"buffer\") {\n const bufKey = `tg:${chatId}`;\n this.deps.conversationBuffer.push(bufKey, senderName, signal);\n logDebug(TAG, `Buffered reaction signal for group ${chatId}`);\n } else {\n const reactionUser = loadUsers().byPlatformId.get(\"telegram:\" + user.id)?.userId ?? \"unknown\";\n const activeId = this.deps.sessionManager.getActiveSessionId(reactionUser, \"telegram\");\n const { sessions } = this.deps.pipeline;\n const entry = sessions.getOrCreate(activeId);\n if (entry.busy) {\n entry.queue.push({ msg: { userId: reactionUser, channelId: String(chatId), senderName, senderId: String(user.id), text: signal, messageId: reaction.message_id, platform: \"telegram\", timestamp: Date.now(), isGroup: false, isVoice: false }, adapter: this });\n logDebug(TAG, `Queued reaction signal for busy ${activeId} (${entry.queue.length} pending)`);\n } else {\n try {\n const response = await this.deps.transport.sendPrompt(activeId, signal, undefined, reactionUser);\n logDebug(TAG, `Sent reaction signal to transport for chat ${chatId}`);\n if (response) {\n const { text, reactionEmoji, noReply } = cleanResponse(response);\n if (noReply) { /* silent */ }\n else if (reactionEmoji) {\n try { await this.api.setMessageReaction(chatId, reaction.message_id, [{ type: \"emoji\", emoji: reactionEmoji }]); }\n catch { await this.sendMessage(String(chatId), reactionEmoji); }\n } else if (text.trim()) {\n await this.sendMessage(String(chatId), text.trim());\n }\n }\n } catch (err) {\n logError(TAG, `Failed to send reaction signal for chat ${chatId}`, err);\n }\n }\n }\n }\n}\n", "import { getEnv } from \"../../components/env-schema.js\";\nimport { logDebug } from \"../../components/logger.js\";\nimport { logAndSwallow } from \"../../components/log-and-swallow.js\";\nimport type {\n TelegramUpdate,\n TelegramInlineKeyboardMarkup,\n} from \"../../types/index.js\";\nimport { readFileSync } from \"node:fs\";\nimport { basename } from \"node:path\";\n\ntype SendMessageOptions = {\n parse_mode?: \"MarkdownV2\" | \"HTML\";\n reply_markup?: TelegramInlineKeyboardMarkup;\n message_thread_id?: number;\n};\n\nconst TAG = \"telegram-api\";\n\n/** Options for fetchWithRetry. */\ninterface FetchRetryOpts {\n method: string;\n timeoutMs?: number;\n outerSignal?: AbortSignal;\n /** Retry on 4xx responses. Default false \u2014 4xx usually permanent (404 expired, 403 auth). */\n retryable4xx?: boolean;\n}\n\n/**\n * Thin wrapper around the Telegram Bot API using native fetch.\n * All methods throw on HTTP errors.\n */\nexport class TelegramApi {\n private readonly baseUrl: string;\n private readonly token: string;\n\n constructor(botToken: string) {\n this.token = botToken;\n this.baseUrl = `https://api.telegram.org/bot${botToken}`;\n }\n\n /** Get bot info. */\n async getMe(): Promise<{ id: number; username?: string }> {\n const res = await this.call(\"getMe\", {});\n return res as { id: number; username?: string };\n }\n\n async getChat(chatId: number): Promise<{ id: number; first_name?: string; username?: string }> {\n const res = await this.call(\"getChat\", { chat_id: chatId });\n return res as { id: number; first_name?: string; username?: string };\n }\n\n /** Get file path for downloading. */\n async getFile(fileId: string): Promise<{ file_id: string; file_path?: string; file_size?: number }> {\n const res = await this.call(\"getFile\", { file_id: fileId });\n return res as { file_id: string; file_path?: string; file_size?: number };\n }\n\n /** Download a file by its file_path (from getFile). Returns the raw buffer. */\n async downloadFile(filePath: string): Promise<Buffer> {\n const url = `https://api.telegram.org/file/bot${this.token}/${filePath}`;\n // 4xx = expired file / bad path / wrong token \u2192 permanent, don't retry\n const response = await this.fetchWithRetry(\n (signal) => fetch(url, { signal }),\n { method: \"downloadFile\", timeoutMs: getEnv().telegramFileTimeoutMs, retryable4xx: false },\n );\n const arrayBuffer = await response.arrayBuffer();\n return Buffer.from(arrayBuffer);\n }\n\n /** Send a text message. Returns the sent message_id. */\n async sendMessage(\n chatId: number,\n text: string,\n options?: SendMessageOptions,\n ): Promise<number> {\n const body: Record<string, unknown> = { chat_id: chatId, text, ...options };\n const res = await this.call(\"sendMessage\", body);\n return (res as { message_id: number }).message_id;\n }\n\n /** Edit an existing message's text. */\n async editMessageText(\n chatId: number,\n messageId: number,\n text: string,\n options?: { parse_mode?: string },\n ): Promise<void> {\n await this.call(\"editMessageText\", { chat_id: chatId, message_id: messageId, text, ...options });\n }\n\n /** Acknowledge a callback query (inline keyboard press). */\n async answerCallbackQuery(callbackQueryId: string): Promise<void> {\n await this.call(\"answerCallbackQuery\", { callback_query_id: callbackQueryId });\n }\n\n /** Send a chat action (typing indicator, etc.). */\n async sendChatAction(chatId: number, action: string = \"typing\", messageThreadId?: number): Promise<void> {\n const body: Record<string, unknown> = { chat_id: chatId, action };\n if (messageThreadId) body.message_thread_id = messageThreadId;\n await this.call(\"sendChatAction\", body);\n }\n\n /** Set an emoji reaction on a message. Pass empty array to remove. */\n async setMessageReaction(\n chatId: number,\n messageId: number,\n reaction: Array<{ type: \"emoji\"; emoji: string }>,\n ): Promise<void> {\n await this.call(\"setMessageReaction\", {\n chat_id: chatId,\n message_id: messageId,\n reaction,\n });\n }\n\n /** Send a voice note (OGG Opus). Returns the sent message_id. */\n async sendVoice(\n chatId: number,\n audioBuffer: Buffer,\n options?: { message_thread_id?: number },\n ): Promise<number> {\n // Factory rebuilds FormData per attempt \u2014 FormData is single-use in some runtimes.\n const makeForm = (): FormData => {\n const form = new FormData();\n form.append(\"chat_id\", String(chatId));\n const blob = new Blob([audioBuffer], { type: \"audio/webm\" });\n form.append(\"voice\", blob, \"voice.webm\");\n if (options?.message_thread_id) form.append(\"message_thread_id\", String(options.message_thread_id));\n return form;\n };\n const response = await this.fetchWithRetry(\n (signal) => fetch(`${this.baseUrl}/sendVoice`, { method: \"POST\", body: makeForm(), signal }),\n { method: \"sendVoice\" },\n );\n const json = (await response.json()) as { ok: boolean; result: { message_id: number } };\n if (!json.ok) throw new Error(\"Telegram API sendVoice returned ok=false\");\n return json.result.message_id;\n }\n\n /** Send a file from disk as a Telegram document. Returns the sent message_id. */\n async sendDocument(\n chatId: number,\n filePath: string,\n caption?: string,\n options?: { message_thread_id?: number },\n ): Promise<number> {\n const buf = readFileSync(filePath);\n const fileName = basename(filePath);\n // Factory rebuilds FormData per attempt \u2014 FormData is single-use in some runtimes.\n const makeForm = (): FormData => {\n const form = new FormData();\n form.append(\"chat_id\", String(chatId));\n const blob = new Blob([buf], { type: \"text/markdown\" });\n form.append(\"document\", blob, fileName);\n if (caption) form.append(\"caption\", caption.slice(0, 1024));\n if (options?.message_thread_id) form.append(\"message_thread_id\", String(options.message_thread_id));\n return form;\n };\n const response = await this.fetchWithRetry(\n (signal) => fetch(`${this.baseUrl}/sendDocument`, { method: \"POST\", body: makeForm(), signal }),\n { method: \"sendDocument\", timeoutMs: getEnv().telegramFileTimeoutMs },\n );\n const json = (await response.json()) as { ok: boolean; result: { message_id: number } };\n if (!json.ok) throw new Error(\"Telegram API sendDocument returned ok=false\");\n return json.result.message_id;\n }\n\n /** Register bot command menu (shown when user types /). */\n async setMyCommands(commands: Array<{ command: string; description: string }>): Promise<void> {\n await this.call(\"setMyCommands\", { commands });\n }\n\n /** Long-poll for updates. */\n async getUpdates(\n offset: number,\n timeout: number,\n signal?: AbortSignal,\n ): Promise<TelegramUpdate[]> {\n const body = {\n offset,\n timeout,\n allowed_updates: [\n \"message\",\n \"callback_query\",\n \"message_reaction\",\n ],\n };\n return (await this.call(\"getUpdates\", body, signal, (timeout + 10) * 1000)) as TelegramUpdate[];\n }\n\n private static readonly MAX_ATTEMPTS = 3;\n private static readonly BACKOFF = [1000, 3000, 9000];\n\n private static readonly PERMANENT_ERRORS = [\n /chat not found/i, /bot was blocked/i, /user is deactivated/i,\n /chat_id is empty/i, /forbidden/i, /not enough rights/i,\n /CHAT_WRITE_FORBIDDEN/i, /have no rights/i,\n /wrong file_path/i, /file is temporarily unavailable/i,\n ];\n\n private static isPermanent(msg: string): boolean {\n return this.PERMANENT_ERRORS.some(re => re.test(msg));\n }\n\n private static isTransient(msg: string): boolean {\n return /429|5\\d\\d|bad gateway|service unavailable|timeout|timed out|ETIMEDOUT|ECONNRESET|ECONNREFUSED|socket hang up|network/i.test(msg);\n }\n\n /**\n * Retry + timeout wrapper for any Telegram fetch. Returns the Response on 2xx,\n * throws on non-OK or transient-exhaustion. Factory MUST build a fresh request\n * per attempt (required for FormData; single-use in some runtimes).\n */\n private async fetchWithRetry(\n requestFn: (signal: AbortSignal) => Promise<Response>,\n opts: FetchRetryOpts,\n ): Promise<Response> {\n const timeoutMs = opts.timeoutMs ?? getEnv().telegramTimeoutMs;\n for (let attempt = 0; attempt < TelegramApi.MAX_ATTEMPTS; attempt++) {\n const ctrl = new AbortController();\n const timer = setTimeout(() => ctrl.abort(new Error(\"telegram timeout\")), timeoutMs);\n const composed = opts.outerSignal\n ? AbortSignal.any([opts.outerSignal, ctrl.signal])\n : ctrl.signal;\n\n try {\n const response = await requestFn(composed);\n if (!response.ok) {\n const text = await response.text().catch(err2 => { logAndSwallow(TAG, \"read TG API error body\", err2); return \"\"; });\n const err = new Error(`Telegram API ${opts.method} failed (${response.status}): ${text}`);\n // 4xx treated as permanent unless caller opts in\n if (response.status >= 400 && response.status < 500 && !opts.retryable4xx) throw err;\n throw err;\n }\n return response;\n } catch (err) {\n const msg = err instanceof Error ? err.message : String(err);\n if (TelegramApi.isPermanent(msg) || opts.outerSignal?.aborted) throw err;\n // Non-retryable 4xx (when retryable4xx is false) \u2014 already thrown above with status code\n if (!opts.retryable4xx && /failed \\([4]\\d\\d\\)/.test(msg)) throw err;\n if (attempt < TelegramApi.MAX_ATTEMPTS - 1 && TelegramApi.isTransient(msg)) {\n logDebug(TAG, `${opts.method} attempt ${attempt + 1}/${TelegramApi.MAX_ATTEMPTS} \u2014 ${msg.slice(0, 80)}`);\n await new Promise(r => setTimeout(r, TelegramApi.BACKOFF[attempt]!));\n continue;\n }\n throw err;\n } finally {\n clearTimeout(timer);\n }\n }\n throw new Error(`Telegram API ${opts.method}: unreachable`);\n }\n\n private async call(\n method: string,\n body: Record<string, unknown>,\n signal?: AbortSignal,\n timeoutMs?: number,\n ): Promise<unknown> {\n const response = await this.fetchWithRetry(\n (composed) => fetch(`${this.baseUrl}/${method}`, {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify(body),\n signal: composed,\n }),\n { method, timeoutMs, outerSignal: signal },\n );\n const json = (await response.json()) as { ok: boolean; result: unknown };\n if (!json.ok) throw new Error(`Telegram API ${method} returned ok=false`);\n return json.result;\n }\n}\n", "import type { TelegramUpdate } from \"../../types/index.js\";\nimport type { TelegramApi } from \"./telegram-api.js\";\nimport type { OffsetStore } from \"./offset-store.js\";\nimport { logError, logWarn } from \"../../components/logger.js\";\n\n/**\n * Long-polls the Telegram Bot API for updates. Never self-terminates \u2014\n * uses exponential backoff with jitter on transient errors.\n *\n * Offset is advanced per-update AFTER the handler settles (success only).\n * Handlers run fire-and-forget to keep /stop responsive; the disk offset\n * only advances once the contiguous prefix of settled handlers is known.\n */\nexport class TelegramPoller {\n private readonly api: TelegramApi;\n private readonly pollTimeout: number;\n private readonly onUpdate: (update: TelegramUpdate) => void | Promise<void>;\n private readonly offsetStore: OffsetStore;\n private offset = 0;\n private lastProcessedId = 0;\n private running = false;\n private abortController: AbortController | null = null;\n\n /** Timestamp of last successful poll cycle. */\n lastPollAt = Date.now();\n\n constructor(\n api: TelegramApi,\n pollTimeoutS: number,\n onUpdate: (update: TelegramUpdate) => void | Promise<void>,\n offsetStore?: OffsetStore,\n ) {\n this.api = api;\n this.pollTimeout = pollTimeoutS;\n this.onUpdate = onUpdate;\n this.offsetStore = offsetStore ?? { read: async () => 0, write: async () => {} };\n }\n\n /** Start the long-poll loop. Non-blocking \u2014 runs in background. */\n async start(): Promise<void> {\n if (this.running) return;\n this.offset = await this.offsetStore.read();\n this.running = true;\n this.poll().catch((err) => {\n logError(\"poller\", \"Fatal error in poll loop\", err);\n });\n }\n\n /** Stop polling. Cancels any in-flight request. */\n stop(): void {\n this.running = false;\n this.abortController?.abort();\n this.abortController = null;\n }\n\n /** Inject a synthetic update into the handler (used for queued messages after sleep). */\n injectUpdate(update: TelegramUpdate): void {\n void this.onUpdate(update);\n }\n\n private async poll(): Promise<void> {\n let failures = 0;\n\n while (this.running) {\n try {\n this.abortController = new AbortController();\n const updates = await this.api.getUpdates(\n this.offset,\n this.pollTimeout,\n this.abortController.signal,\n );\n\n failures = 0;\n this.lastPollAt = Date.now();\n\n // Process each update: dispatch handler, advance offset only on success.\n // Handlers run in background (fire-and-forget) so /stop stays responsive.\n // We track settlements and advance the disk offset for the contiguous\n // prefix of successfully-handled updates.\n if (updates.length > 0) {\n const sorted = [...updates].sort((a, b) => a.update_id - b.update_id);\n\n for (const update of sorted) {\n if (update.update_id <= this.lastProcessedId) continue;\n try {\n const result = this.onUpdate(update);\n if (result instanceof Promise) {\n result.catch((err: unknown) => logError(\"poller\", \"Error in update handler\", err));\n }\n } catch (err) {\n logError(\"poller\", \"Error in update handler (sync)\", err);\n }\n }\n\n this.lastProcessedId = sorted[sorted.length - 1]!.update_id;\n this.offset = sorted[sorted.length - 1]!.update_id + 1;\n await this.offsetStore.write(this.offset);\n }\n } catch (err) {\n if (!this.running) break;\n\n failures++;\n const baseDelay = Math.min(2 ** failures * 1000, 60_000);\n const jitter = Math.random() * baseDelay;\n const delay = baseDelay + jitter;\n const is409 = String(err).includes(\"409\") || String(err).includes(\"terminated by other getUpdates\");\n const log = is409 ? (failures >= 50 ? logError : logWarn) : (failures < 3 ? logWarn : logError);\n if (is409 && failures < 50) {\n log(\"poller\", `409 conflict (attempt ${failures}) \u2014 another instance running, backing off`);\n } else {\n log(\"poller\", `Error (attempt ${failures}), retrying in ${Math.round(delay)}ms`, err);\n }\n await sleep(delay);\n }\n }\n }\n}\n\nfunction sleep(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}\n", "/**\n * Atomic offset store for Telegram poller.\n * Persists the last-acked update_id to disk so crash recovery resumes\n * from the correct position. Writes atomically (temp + rename).\n */\n\nimport { readFile, writeFile, rename, mkdir } from \"node:fs/promises\";\nimport { dirname } from \"node:path\";\nimport { logAndSwallow } from \"../../components/log-and-swallow.js\";\n\nconst TAG = \"offset_store\";\n\nexport interface OffsetStore {\n read(): Promise<number>;\n write(offset: number): Promise<void>;\n}\n\n/** File-backed offset store with atomic writes. */\nexport function createFileOffsetStore(filePath: string): OffsetStore {\n let pending: Promise<void> = Promise.resolve();\n\n return {\n async read(): Promise<number> {\n try {\n const raw = await readFile(filePath, \"utf-8\");\n const val = parseInt(raw.trim(), 10);\n return Number.isFinite(val) ? val : 0;\n } catch (err) {\n logAndSwallow(TAG, \"read offset file\", err);\n return 0;\n }\n },\n write(offset: number): Promise<void> {\n // Serialize writes via promise chain (async mutex).\n pending = pending.then(async () => {\n const tmp = filePath + \".tmp\";\n await mkdir(dirname(filePath), { recursive: true });\n await writeFile(tmp, String(offset), \"utf-8\");\n await rename(tmp, filePath);\n }).catch(err => logAndSwallow(TAG, \"write offset file\", err));\n return pending;\n },\n };\n}\n\n/** In-memory offset store for tests. */\nexport function createMemoryOffsetStore(initial = 0): OffsetStore & { value: number } {\n const store = {\n value: initial,\n async read(): Promise<number> { return store.value; },\n async write(offset: number): Promise<void> { store.value = offset; },\n };\n return store;\n}\n", "/**\n * telegram-model-picker.ts \u2014 Extracted model picker callback handlers.\n * Handles: mb:, mslot:, mprov:, mpos:, mprov2:, mset:, model: prefixes.\n */\nimport type { TelegramApi } from \"./telegram-api.js\";\nimport { getEnv } from \"../../components/env-schema.js\";\nimport { logAndSwallow } from \"../../components/log-and-swallow.js\";\n\nconst TAG = \"telegram\";\n\nexport interface PickerState {\n _pendingSlot: string | undefined;\n _modelPickerCache: string[];\n}\n\nexport interface PickerDeps {\n transport: { setModel?: (m: string) => Promise<void> | void; switchProvider?: (o: unknown) => void; [k: string]: unknown };\n pipeline: { rebuildTransport?: () => Promise<void>; [k: string]: unknown };\n resetSessionForModelSwitch: (chatId: number, reason?: string) => Promise<void>;\n}\n\nconst MODEL_PREFIXES = [\"mb:\", \"mslot:\", \"mprov:\", \"mpos:\", \"mprov2:\", \"mset:\", \"model:\"];\n\nexport function isModelPickerCallback(data: string): boolean {\n return MODEL_PREFIXES.some(p => data.startsWith(p));\n}\n\nexport async function handleModelPickerCallback(\n data: string, chatId: number, api: TelegramApi, state: PickerState, deps: PickerDeps,\n): Promise<void> {\n\n if (data.startsWith(\"mb:\")) {\n const target = data.slice(3);\n if (!target) { await api.sendMessage(chatId, \"\uD83D\uDC4C Cancelled.\"); return; }\n if (target === \"a\") {\n const AGENT_LABELS = [\n { key: \"professor\", label: \"Professor\" },\n { key: \"dreamy\", label: \"Dreamy (sleep)\" },\n { key: \"browsie\", label: \"Browsie (browse)\" },\n { key: \"coding\", label: \"Cody (coding)\" },\n ];\n const buttons = AGENT_LABELS.map(a => [{ text: a.label, callback_data: `mslot:${a.key}` }]);\n buttons.push([{ text: \"\u2190 Cancel\", callback_data: \"mb:\" }]);\n await api.sendMessage(chatId, \"\uD83E\uDD16 Which agent to change?\", { reply_markup: { inline_keyboard: buttons } });\n return;\n }\n if (target.startsWith(\"p:\")) {\n const agent = target.slice(2);\n const { loadTransport, resolveAgent, getAvailableProviders, getModelsForProvider } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n if (!tc) { await api.sendMessage(chatId, \"\u274C transport.json not loaded\"); return; }\n let providers = getAvailableProviders(tc).filter(p => p.config.transport !== \"tmux\");\n if (providers.length === 0) { await api.sendMessage(chatId, \"\u274C No compatible providers\"); return; }\n const currentProvider = resolveAgent(agent, tc)?.providerName;\n const prefix = agent === \"professor\" ? \"mprov2:professor\" : `mprov:${agent}`;\n const buttons = providers.map(p => {\n const count = getModelsForProvider(p.name).length;\n const label = p.name === currentProvider ? `\u2705 ${p.name} (${count})` : `${p.name} (${count})`;\n return [{ text: label, callback_data: `${prefix}:${p.name}` }];\n });\n buttons.push([{ text: \"\u2190 Back\", callback_data: \"mb:a\" }]);\n await api.sendMessage(chatId, `\uD83D\uDD0C Pick provider:`, { reply_markup: { inline_keyboard: buttons } });\n return;\n }\n if (target.startsWith(\"s:\")) {\n const slot = target.slice(2);\n state._pendingSlot = slot;\n const { loadTransport, resolveAgent, getAvailableProviders, getModelsForProvider } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n if (!tc) { await api.sendMessage(chatId, \"\u274C transport.json not loaded\"); return; }\n let providers = getAvailableProviders(tc).filter(p => p.config.transport !== \"tmux\");\n const currentProvider = resolveAgent(\"professor\", tc)?.providerName;\n const slotLabel = slot === \"professor\" ? \"Main\" : slot.replace(\"professor_fb\", \"Fb\");\n const buttons = providers.map(p => {\n const count = getModelsForProvider(p.name).length;\n const label = p.name === currentProvider ? `\u2705 ${p.name} (${count})` : `${p.name} (${count})`;\n return [{ text: label, callback_data: `mprov2:${slot}:${p.name}` }];\n });\n buttons.push([{ text: \"\u2190 Back\", callback_data: \"mslot:professor\" }]);\n await api.sendMessage(chatId, `\uD83D\uDD0C Provider for ${slotLabel}:`, { reply_markup: { inline_keyboard: buttons } });\n return;\n }\n } else if (data.startsWith(\"mslot:\")) {\n const agent = data.slice(6);\n const { loadTransport, resolveAgent, getAvailableProviders, getModelsForProvider } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n if (!tc) { await api.sendMessage(chatId, \"\u274C transport.json not loaded\"); return; }\n\n if (agent === \"professor\") {\n const profResolved = resolveAgent(\"professor\", tc);\n const fallbacks = tc.agents[\"professor\"]?.fallbacks ?? [];\n const slots: Array<{ label: string; key: string }> = [\n { label: `\u2605 Main: ${profResolved?.model ?? \"?\"}`, key: `mpos:professor::professor` },\n ];\n for (let i = 0; i < fallbacks.length; i++) {\n slots.push({ label: `\u21B3 Fb${i + 1}: ${fallbacks[i]!.model}`, key: `mpos:professor::professor_fb${i + 1}` });\n }\n if (fallbacks.length < 3) {\n slots.push({ label: `\u21B3 Fb${fallbacks.length + 1}: (add)`, key: `mpos:professor::professor_fb${fallbacks.length + 1}` });\n }\n const buttons = slots.map(s => [{ text: s.label, callback_data: s.key }]);\n buttons.push([{ text: \"\u2190 Back\", callback_data: \"mb:a\" }]);\n await api.sendMessage(chatId, `\uD83C\uDFAF Which slot?`, { reply_markup: { inline_keyboard: buttons } });\n return;\n }\n\n let providers = getAvailableProviders(tc).filter(p => p.config.transport !== \"tmux\");\n if (providers.length === 0) { await api.sendMessage(chatId, \"\u274C No compatible providers\"); return; }\n const currentProvider = resolveAgent(agent, tc)?.providerName;\n const buttons = providers.map(p => {\n const count = getModelsForProvider(p.name).length;\n const label = p.name === currentProvider ? `\u2705 ${p.name} (${count})` : `${p.name} (${count})`;\n return [{ text: label, callback_data: `mprov:${agent}:${p.name}` }];\n });\n buttons.push([{ text: \"\u2190 Back\", callback_data: \"mb:a\" }]);\n await api.sendMessage(chatId, `\uD83D\uDD0C Pick provider:`, { reply_markup: { inline_keyboard: buttons } });\n\n } else if (data.startsWith(\"mprov:\")) {\n const [, agent, providerName] = data.split(\":\");\n const { loadTransport, getModelsForProvider, formatRank, formatCost } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n let models = getModelsForProvider(providerName!);\n const providerConfig = tc?.providers[providerName!];\n if (providerConfig?.transport === \"api\") models = models.filter(m => !m.entry.status || m.entry.status === \"alive\");\n if (models.length === 0) { await api.sendMessage(chatId, `\u274C No alive models for ${providerName}`); return; }\n state._pendingSlot = agent;\n state._modelPickerCache = models.map(m => m.id);\n const buttons = models.map((m, i) => [{ text: `${m.id} (${formatRank(m.entry.rank)}, ${formatCost(m.entry.cost)})`, callback_data: `mset:${providerName}:${i}` }]);\n buttons.push([{ text: \"\u2190 Back\", callback_data: `mb:p:${agent}` }]);\n await api.sendMessage(chatId, `\uD83D\uDCCB Models on ${providerName}:`, { reply_markup: { inline_keyboard: buttons } });\n\n } else if (data.startsWith(\"mpos:\")) {\n const [, , , slot] = data.split(\":\");\n const { loadTransport, resolveAgent, getAvailableProviders, getModelsForProvider } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n if (!tc) { await api.sendMessage(chatId, \"\u274C transport.json not loaded\"); return; }\n let providers = getAvailableProviders(tc).filter(p => p.config.transport !== \"tmux\");\n const profResolved = resolveAgent(\"professor\", tc);\n const mainTransport = profResolved?.provider.transport;\n if (slot && slot.startsWith(\"professor_fb\") && mainTransport) {\n if (mainTransport === \"api\") { providers = providers.filter(p => p.config.transport === \"api\"); }\n else { providers = providers.filter(p => p.name === profResolved!.providerName); }\n }\n if (providers.length === 0) { await api.sendMessage(chatId, \"\u274C No compatible providers for this slot\"); return; }\n const currentProvider = profResolved?.providerName;\n const slotLabel = slot === \"professor\" ? \"Main\" : slot!.replace(\"professor_fb\", \"Fb\");\n const buttons = providers.map(p => {\n const count = getModelsForProvider(p.name).length;\n const label = p.name === currentProvider ? `\u2705 ${p.name} (${count})` : `${p.name} (${count})`;\n return [{ text: label, callback_data: `mprov2:${slot}:${p.name}` }];\n });\n buttons.push([{ text: \"\u2190 Back\", callback_data: \"mslot:professor\" }]);\n await api.sendMessage(chatId, `\uD83D\uDD0C Provider for ${slotLabel}:`, { reply_markup: { inline_keyboard: buttons } });\n\n } else if (data.startsWith(\"mprov2:\")) {\n const [, slot, providerName] = data.split(\":\");\n const { getModelsForProvider, formatRank, formatCost, loadTransport } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n const providerConfig = tc?.providers[providerName!];\n let models = getModelsForProvider(providerName!);\n if (providerConfig?.transport === \"api\") models = models.filter(m => !m.entry.status || m.entry.status === \"alive\");\n if (models.length === 0) { await api.sendMessage(chatId, `\u274C No alive models for ${providerName}`); return; }\n state._pendingSlot = slot;\n const slotLabel = slot === \"professor\" ? \"Main\" : slot!.replace(\"professor_fb\", \"Fb\");\n state._modelPickerCache = models.map(m => m.id);\n const buttons = models.map((m, i) => [{ text: `${m.id} (${formatRank(m.entry.rank)}, ${formatCost(m.entry.cost)})`, callback_data: `mset:${providerName}:${i}` }]);\n buttons.push([{ text: \"\u2190 Back\", callback_data: `mb:s:${slot}` }]);\n await api.sendMessage(chatId, `\uD83D\uDCCB Pick model for ${slotLabel}:`, { reply_markup: { inline_keyboard: buttons } });\n\n } else if (data.startsWith(\"mset:\")) {\n const parts = data.split(\":\");\n const providerName = parts[1]!;\n const modelIdx = parseInt(parts[2]!, 10);\n const model = Number.isFinite(modelIdx) && state._modelPickerCache[modelIdx]\n ? state._modelPickerCache[modelIdx]!\n : parts.slice(2).join(\":\");\n const slot = state._pendingSlot ?? \"professor\";\n state._pendingSlot = undefined;\n state._modelPickerCache = [];\n const { loadTransport, writeTransportConfig, resolveAgent, getModelsForProvider, validateProviderReady, formatValidationError } = await import(\"../../components/transport-config.js\");\n const tc = loadTransport();\n if (!tc) { await api.sendMessage(chatId, \"\u274C transport.json not loaded\"); return; }\n\n const validModels = getModelsForProvider(providerName);\n if (!validModels.some(m => m.id === model)) { await api.sendMessage(chatId, `\u274C ${model} is not available on ${providerName}. Pick another.`); return; }\n\n const providerConfig = tc.providers[providerName];\n if (!providerConfig) { await api.sendMessage(chatId, `\u274C Provider ${providerName} not found`); return; }\n const validation = validateProviderReady(providerName, providerConfig, getEnv());\n if (!validation.ok) { await api.sendMessage(chatId, formatValidationError(providerName, validation)); return; }\n\n if (providerConfig?.transport === \"api\") {\n try {\n const endpoint = providerConfig.endpoint ?? \"\";\n const apiKey = getEnv().getApiKey(providerConfig.apiKeyEnv ?? \"API_KEY\");\n const headers: Record<string, string> = {};\n if (apiKey) headers[\"Authorization\"] = `Bearer ${apiKey}`;\n const res = await fetch(`${endpoint}/models`, { headers, signal: AbortSignal.timeout(5000) });\n if (!res.ok) { await api.sendMessage(chatId, `\u26A0\uFE0F ${providerName} unreachable (${res.status}). Try another?`); return; }\n } catch (err) { logAndSwallow(TAG, \"provider reachability check\", err); await api.sendMessage(chatId, `\u26A0\uFE0F ${providerName} unreachable. Try another?`); return; }\n }\n\n const agentKey = slot.startsWith(\"professor_fb\") ? \"professor\" : slot;\n const fbIndex = slot === \"professor_fb1\" ? 0 : slot === \"professor_fb2\" ? 1 : slot === \"professor_fb3\" ? 2 : -1;\n\n if (fbIndex >= 0) {\n if (!tc.agents[\"professor\"]) tc.agents[\"professor\"] = { model: \"\", provider: \"\" };\n if (!tc.agents[\"professor\"]!.fallbacks) tc.agents[\"professor\"]!.fallbacks = [];\n tc.agents[\"professor\"]!.fallbacks[fbIndex] = { model, provider: providerName };\n const { cleanDemotedModels } = await import(\"../../components/transport-config.js\");\n cleanDemotedModels(tc, model);\n writeTransportConfig(tc, `professor fallback ${fbIndex + 1} \u2192 ${model} (${providerName})`);\n await api.sendMessage(chatId, `\u2705 Fallback ${fbIndex + 1} \u2192 ${model} (${providerName})`);\n } else {\n const oldProvider = tc.agents[agentKey]?.provider;\n tc.agents[agentKey] = { ...tc.agents[agentKey]!, model, provider: providerName };\n const { cleanDemotedModels } = await import(\"../../components/transport-config.js\");\n cleanDemotedModels(tc, model);\n writeTransportConfig(tc, `${agentKey} \u2192 ${model} (${providerName})`);\n\n const providerChanged = oldProvider !== providerName;\n const isProfessor = agentKey === \"professor\";\n\n let oldType: string | undefined;\n let newType: string | undefined;\n let newResolved: ReturnType<typeof resolveAgent> | undefined;\n if (isProfessor && providerChanged) {\n const oldResolved = resolveAgent(\"_old\", { ...tc, agents: { ...tc.agents, _old: { model: \"\", provider: oldProvider! } } });\n newResolved = resolveAgent(\"_new\", { ...tc, agents: { ...tc.agents, _new: { model, provider: providerName } } });\n oldType = oldResolved?.provider.transport ?? \"api\";\n newType = newResolved?.provider.transport ?? \"api\";\n if (oldType !== newType) {\n const resetAgents: string[] = [];\n for (const [a, assignment] of Object.entries(tc.agents)) {\n if (a === \"professor\") continue;\n const ap = tc.providers[assignment.provider];\n if (ap && ap.transport !== newType) { tc.agents[a] = { model, provider: providerName }; resetAgents.push(a); }\n }\n if (resetAgents.length > 0) writeTransportConfig(tc, `cascade: ${resetAgents.join(\", \")} \u2192 ${providerName}`);\n }\n }\n\n if (isProfessor && !providerChanged && \"setModel\" in deps.transport) {\n await (deps.transport as unknown as { setModel: (m: string) => Promise<void> }).setModel(model);\n await api.sendMessage(chatId, `\u2705 Switched to ${model}`);\n } else if (isProfessor && providerChanged && oldType === newType && \"switchProvider\" in deps.transport) {\n try {\n const { FallbackPolicy } = await import(\"../../components/transport/fallback-policy.js\");\n const { ModelHealthRegistry } = await import(\"../../components/transport/model-health-registry.js\");\n const apiKey = getEnv().getApiKey(newResolved?.provider.apiKeyEnv ?? \"API_KEY\");\n const candidates = [{ endpoint: newResolved!.provider.endpoint!, apiKey, model, maxContext: newResolved!.contextWindow }];\n for (const fb of (tc.agents[\"professor\"]?.fallbacks ?? [])) {\n const fbRes = resolveAgent(\"_fb\", { ...tc, agents: { ...tc.agents, _fb: { model: fb.model, provider: fb.provider } } });\n if (fbRes) candidates.push({ endpoint: fbRes.provider.endpoint!, apiKey: fbRes.provider.apiKeyEnv ? getEnv().getApiKey(fbRes.provider.apiKeyEnv) : apiKey, model: fb.model, maxContext: fbRes.contextWindow });\n }\n const registry = (deps.transport as unknown as { policy?: { registry: InstanceType<typeof ModelHealthRegistry> } }).policy?.registry ?? new ModelHealthRegistry();\n const policy = new FallbackPolicy(candidates, registry);\n (deps.transport as unknown as { switchProvider: (o: unknown) => void }).switchProvider({ endpoint: newResolved!.provider.endpoint!, apiKey, model, maxContext: newResolved!.contextWindow, policy });\n } catch (err) {\n await api.sendMessage(chatId, `\u26A0\uFE0F Hot swap failed: ${err instanceof Error ? err.message : String(err)}. Use /reset to apply.`);\n return;\n }\n try { await api.sendMessage(chatId, `\u2705 Switched to ${model} (${providerName})`); } catch (err) { logAndSwallow(TAG, \"sendMessage model switch confirm\", err); }\n } else if (isProfessor && providerChanged) {\n const cascadeNote = oldType !== newType ? \" Subagents also reset.\" : \"\";\n try {\n if (deps.pipeline.rebuildTransport) await deps.pipeline.rebuildTransport();\n await deps.resetSessionForModelSwitch(chatId, \"cross-transport-switch\");\n await api.sendMessage(chatId, `\uD83D\uDD04 Switched to ${model} (${providerName}). Transport rebuilt.${cascadeNote}`);\n } catch (err) {\n await api.sendMessage(chatId, `\u26A0\uFE0F Transport rebuild failed: ${err instanceof Error ? err.message : String(err)}. Try /reset manually.`);\n }\n } else {\n await api.sendMessage(chatId, `\u2705 ${agentKey} \u2192 ${model} (${providerName})`);\n }\n }\n } else if (data.startsWith(\"model:\")) {\n const newModel = data.slice(6);\n if (\"setModel\" in deps.transport && typeof deps.transport.setModel === \"function\") {\n try {\n await deps.transport.setModel(newModel);\n await deps.resetSessionForModelSwitch(chatId);\n await api.sendMessage(chatId, `\uD83E\uDD16 Model switched \u2192 ${newModel}`);\n } catch (err) {\n await api.sendMessage(chatId, `\u274C Model switch failed: ${err instanceof Error ? err.message : String(err)}`);\n }\n }\n }\n}\n"],
|
|
5
|
+
"mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AACA;;;ACDA;AACA;AACA;AAKA,SAAS,oBAAoB;AAC7B,SAAS,gBAAgB;AAQzB,IAAM,MAAM;AAeL,IAAM,cAAN,MAAM,aAAY;AAAA,EACN;AAAA,EACA;AAAA,EAEjB,YAAY,UAAkB;AAC5B,SAAK,QAAQ;AACb,SAAK,UAAU,+BAA+B,QAAQ;AAAA,EACxD;AAAA;AAAA,EAGA,MAAM,QAAoD;AACxD,UAAM,MAAM,MAAM,KAAK,KAAK,SAAS,CAAC,CAAC;AACvC,WAAO;AAAA,EACT;AAAA,EAEA,MAAM,QAAQ,QAAiF;AAC7F,UAAM,MAAM,MAAM,KAAK,KAAK,WAAW,EAAE,SAAS,OAAO,CAAC;AAC1D,WAAO;AAAA,EACT;AAAA;AAAA,EAGA,MAAM,QAAQ,QAAsF;AAClG,UAAM,MAAM,MAAM,KAAK,KAAK,WAAW,EAAE,SAAS,OAAO,CAAC;AAC1D,WAAO;AAAA,EACT;AAAA;AAAA,EAGA,MAAM,aAAa,UAAmC;AACpD,UAAM,MAAM,oCAAoC,KAAK,KAAK,IAAI,QAAQ;AAEtE,UAAM,WAAW,MAAM,KAAK;AAAA,MAC1B,CAAC,WAAW,MAAM,KAAK,EAAE,OAAO,CAAC;AAAA,MACjC,EAAE,QAAQ,gBAAgB,WAAW,OAAO,EAAE,uBAAuB,cAAc,MAAM;AAAA,IAC3F;AACA,UAAM,cAAc,MAAM,SAAS,YAAY;AAC/C,WAAO,OAAO,KAAK,WAAW;AAAA,EAChC;AAAA;AAAA,EAGA,MAAM,YACJ,QACA,MACA,SACiB;AACjB,UAAM,OAAgC,EAAE,SAAS,QAAQ,MAAM,GAAG,QAAQ;AAC1E,UAAM,MAAM,MAAM,KAAK,KAAK,eAAe,IAAI;AAC/C,WAAQ,IAA+B;AAAA,EACzC;AAAA;AAAA,EAGA,MAAM,gBACJ,QACA,WACA,MACA,SACe;AACf,UAAM,KAAK,KAAK,mBAAmB,EAAE,SAAS,QAAQ,YAAY,WAAW,MAAM,GAAG,QAAQ,CAAC;AAAA,EACjG;AAAA;AAAA,EAGA,MAAM,oBAAoB,iBAAwC;AAChE,UAAM,KAAK,KAAK,uBAAuB,EAAE,mBAAmB,gBAAgB,CAAC;AAAA,EAC/E;AAAA;AAAA,EAGA,MAAM,eAAe,QAAgB,SAAiB,UAAU,iBAAyC;AACvG,UAAM,OAAgC,EAAE,SAAS,QAAQ,OAAO;AAChE,QAAI,gBAAiB,MAAK,oBAAoB;AAC9C,UAAM,KAAK,KAAK,kBAAkB,IAAI;AAAA,EACxC;AAAA;AAAA,EAGA,MAAM,mBACJ,QACA,WACA,UACe;AACf,UAAM,KAAK,KAAK,sBAAsB;AAAA,MACpC,SAAS;AAAA,MACT,YAAY;AAAA,MACZ;AAAA,IACF,CAAC;AAAA,EACH;AAAA;AAAA,EAGA,MAAM,UACJ,QACA,aACA,SACiB;AAEjB,UAAM,WAAW,MAAgB;AAC/B,YAAM,OAAO,IAAI,SAAS;AAC1B,WAAK,OAAO,WAAW,OAAO,MAAM,CAAC;AACrC,YAAM,OAAO,IAAI,KAAK,CAAC,WAAW,GAAG,EAAE,MAAM,aAAa,CAAC;AAC3D,WAAK,OAAO,SAAS,MAAM,YAAY;AACvC,UAAI,SAAS,kBAAmB,MAAK,OAAO,qBAAqB,OAAO,QAAQ,iBAAiB,CAAC;AAClG,aAAO;AAAA,IACT;AACA,UAAM,WAAW,MAAM,KAAK;AAAA,MAC1B,CAAC,WAAW,MAAM,GAAG,KAAK,OAAO,cAAc,EAAE,QAAQ,QAAQ,MAAM,SAAS,GAAG,OAAO,CAAC;AAAA,MAC3F,EAAE,QAAQ,YAAY;AAAA,IACxB;AACA,UAAM,OAAQ,MAAM,SAAS,KAAK;AAClC,QAAI,CAAC,KAAK,GAAI,OAAM,IAAI,MAAM,0CAA0C;AACxE,WAAO,KAAK,OAAO;AAAA,EACrB;AAAA;AAAA,EAGA,MAAM,aACJ,QACA,UACA,SACA,SACiB;AACjB,UAAM,MAAM,aAAa,QAAQ;AACjC,UAAM,WAAW,SAAS,QAAQ;AAElC,UAAM,WAAW,MAAgB;AAC/B,YAAM,OAAO,IAAI,SAAS;AAC1B,WAAK,OAAO,WAAW,OAAO,MAAM,CAAC;AACrC,YAAM,OAAO,IAAI,KAAK,CAAC,GAAG,GAAG,EAAE,MAAM,gBAAgB,CAAC;AACtD,WAAK,OAAO,YAAY,MAAM,QAAQ;AACtC,UAAI,QAAS,MAAK,OAAO,WAAW,QAAQ,MAAM,GAAG,IAAI,CAAC;AAC1D,UAAI,SAAS,kBAAmB,MAAK,OAAO,qBAAqB,OAAO,QAAQ,iBAAiB,CAAC;AAClG,aAAO;AAAA,IACT;AACA,UAAM,WAAW,MAAM,KAAK;AAAA,MAC1B,CAAC,WAAW,MAAM,GAAG,KAAK,OAAO,iBAAiB,EAAE,QAAQ,QAAQ,MAAM,SAAS,GAAG,OAAO,CAAC;AAAA,MAC9F,EAAE,QAAQ,gBAAgB,WAAW,OAAO,EAAE,sBAAsB;AAAA,IACtE;AACA,UAAM,OAAQ,MAAM,SAAS,KAAK;AAClC,QAAI,CAAC,KAAK,GAAI,OAAM,IAAI,MAAM,6CAA6C;AAC3E,WAAO,KAAK,OAAO;AAAA,EACrB;AAAA;AAAA,EAGA,MAAM,cAAc,UAA0E;AAC5F,UAAM,KAAK,KAAK,iBAAiB,EAAE,SAAS,CAAC;AAAA,EAC/C;AAAA;AAAA,EAGA,MAAM,WACJ,QACA,SACA,QAC2B;AAC3B,UAAM,OAAO;AAAA,MACX;AAAA,MACA;AAAA,MACA,iBAAiB;AAAA,QACf;AAAA,QACA;AAAA,QACA;AAAA,MACF;AAAA,IACF;AACA,WAAQ,MAAM,KAAK,KAAK,cAAc,MAAM,SAAS,UAAU,MAAM,GAAI;AAAA,EAC3E;AAAA,EAEA,OAAwB,eAAe;AAAA,EACvC,OAAwB,UAAU,CAAC,KAAM,KAAM,GAAI;AAAA,EAEnD,OAAwB,mBAAmB;AAAA,IACzC;AAAA,IAAmB;AAAA,IAAoB;AAAA,IACvC;AAAA,IAAqB;AAAA,IAAc;AAAA,IACnC;AAAA,IAAyB;AAAA,IACzB;AAAA,IAAoB;AAAA,EACtB;AAAA,EAEA,OAAe,YAAY,KAAsB;AAC/C,WAAO,KAAK,iBAAiB,KAAK,QAAM,GAAG,KAAK,GAAG,CAAC;AAAA,EACtD;AAAA,EAEA,OAAe,YAAY,KAAsB;AAC/C,WAAO,wHAAwH,KAAK,GAAG;AAAA,EACzI;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAc,eACZ,WACA,MACmB;AACnB,UAAM,YAAY,KAAK,aAAa,OAAO,EAAE;AAC7C,aAAS,UAAU,GAAG,UAAU,aAAY,cAAc,WAAW;AACnE,YAAM,OAAO,IAAI,gBAAgB;AACjC,YAAM,QAAQ,WAAW,MAAM,KAAK,MAAM,IAAI,MAAM,kBAAkB,CAAC,GAAG,SAAS;AACnF,YAAM,WAAW,KAAK,cAClB,YAAY,IAAI,CAAC,KAAK,aAAa,KAAK,MAAM,CAAC,IAC/C,KAAK;AAET,UAAI;AACF,cAAM,WAAW,MAAM,UAAU,QAAQ;AACzC,YAAI,CAAC,SAAS,IAAI;AAChB,gBAAM,OAAO,MAAM,SAAS,KAAK,EAAE,MAAM,UAAQ;AAAE,0BAAc,KAAK,0BAA0B,IAAI;AAAG,mBAAO;AAAA,UAAI,CAAC;AACnH,gBAAM,MAAM,IAAI,MAAM,gBAAgB,KAAK,MAAM,YAAY,SAAS,MAAM,MAAM,IAAI,EAAE;AAExF,cAAI,SAAS,UAAU,OAAO,SAAS,SAAS,OAAO,CAAC,KAAK,aAAc,OAAM;AACjF,gBAAM;AAAA,QACR;AACA,eAAO;AAAA,MACT,SAAS,KAAK;AACZ,cAAM,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG;AAC3D,YAAI,aAAY,YAAY,GAAG,KAAK,KAAK,aAAa,QAAS,OAAM;AAErE,YAAI,CAAC,KAAK,gBAAgB,qBAAqB,KAAK,GAAG,EAAG,OAAM;AAChE,YAAI,UAAU,aAAY,eAAe,KAAK,aAAY,YAAY,GAAG,GAAG;AAC1E,mBAAS,KAAK,GAAG,KAAK,MAAM,YAAY,UAAU,CAAC,IAAI,aAAY,YAAY,WAAM,IAAI,MAAM,GAAG,EAAE,CAAC,EAAE;AACvG,gBAAM,IAAI,QAAQ,OAAK,WAAW,GAAG,aAAY,QAAQ,OAAO,CAAE,CAAC;AACnE;AAAA,QACF;AACA,cAAM;AAAA,MACR,UAAE;AACA,qBAAa,KAAK;AAAA,MACpB;AAAA,IACF;AACA,UAAM,IAAI,MAAM,gBAAgB,KAAK,MAAM,eAAe;AAAA,EAC5D;AAAA,EAEA,MAAc,KACZ,QACA,MACA,QACA,WACkB;AAClB,UAAM,WAAW,MAAM,KAAK;AAAA,MAC1B,CAAC,aAAa,MAAM,GAAG,KAAK,OAAO,IAAI,MAAM,IAAI;AAAA,QAC/C,QAAQ;AAAA,QACR,SAAS,EAAE,gBAAgB,mBAAmB;AAAA,QAC9C,MAAM,KAAK,UAAU,IAAI;AAAA,QACzB,QAAQ;AAAA,MACV,CAAC;AAAA,MACD,EAAE,QAAQ,WAAW,aAAa,OAAO;AAAA,IAC3C;AACA,UAAM,OAAQ,MAAM,SAAS,KAAK;AAClC,QAAI,CAAC,KAAK,GAAI,OAAM,IAAI,MAAM,gBAAgB,MAAM,oBAAoB;AACxE,WAAO,KAAK;AAAA,EACd;AACF;;;AC7QA;AAUO,IAAM,iBAAN,MAAqB;AAAA,EACT;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACT,SAAS;AAAA,EACT,kBAAkB;AAAA,EAClB,UAAU;AAAA,EACV,kBAA0C;AAAA;AAAA,EAGlD,aAAa,KAAK,IAAI;AAAA,EAEtB,YACE,KACA,cACA,UACA,aACA;AACA,SAAK,MAAM;AACX,SAAK,cAAc;AACnB,SAAK,WAAW;AAChB,SAAK,cAAc,eAAe,EAAE,MAAM,YAAY,GAAG,OAAO,YAAY;AAAA,IAAC,EAAE;AAAA,EACjF;AAAA;AAAA,EAGA,MAAM,QAAuB;AAC3B,QAAI,KAAK,QAAS;AAClB,SAAK,SAAS,MAAM,KAAK,YAAY,KAAK;AAC1C,SAAK,UAAU;AACf,SAAK,KAAK,EAAE,MAAM,CAAC,QAAQ;AACzB,eAAS,UAAU,4BAA4B,GAAG;AAAA,IACpD,CAAC;AAAA,EACH;AAAA;AAAA,EAGA,OAAa;AACX,SAAK,UAAU;AACf,SAAK,iBAAiB,MAAM;AAC5B,SAAK,kBAAkB;AAAA,EACzB;AAAA;AAAA,EAGA,aAAa,QAA8B;AACzC,SAAK,KAAK,SAAS,MAAM;AAAA,EAC3B;AAAA,EAEA,MAAc,OAAsB;AAClC,QAAI,WAAW;AAEf,WAAO,KAAK,SAAS;AACnB,UAAI;AACF,aAAK,kBAAkB,IAAI,gBAAgB;AAC3C,cAAM,UAAU,MAAM,KAAK,IAAI;AAAA,UAC7B,KAAK;AAAA,UACL,KAAK;AAAA,UACL,KAAK,gBAAgB;AAAA,QACvB;AAEA,mBAAW;AACX,aAAK,aAAa,KAAK,IAAI;AAM3B,YAAI,QAAQ,SAAS,GAAG;AACtB,gBAAM,SAAS,CAAC,GAAG,OAAO,EAAE,KAAK,CAAC,GAAG,MAAM,EAAE,YAAY,EAAE,SAAS;AAEpE,qBAAW,UAAU,QAAQ;AAC3B,gBAAI,OAAO,aAAa,KAAK,gBAAiB;AAC9C,gBAAI;AACF,oBAAM,SAAS,KAAK,SAAS,MAAM;AACnC,kBAAI,kBAAkB,SAAS;AAC7B,uBAAO,MAAM,CAAC,QAAiB,SAAS,UAAU,2BAA2B,GAAG,CAAC;AAAA,cACnF;AAAA,YACF,SAAS,KAAK;AACZ,uBAAS,UAAU,kCAAkC,GAAG;AAAA,YAC1D;AAAA,UACF;AAEA,eAAK,kBAAkB,OAAO,OAAO,SAAS,CAAC,EAAG;AAClD,eAAK,SAAS,OAAO,OAAO,SAAS,CAAC,EAAG,YAAY;AACrD,gBAAM,KAAK,YAAY,MAAM,KAAK,MAAM;AAAA,QAC1C;AAAA,MACF,SAAS,KAAK;AACZ,YAAI,CAAC,KAAK,QAAS;AAEnB;AACA,cAAM,YAAY,KAAK,IAAI,KAAK,WAAW,KAAM,GAAM;AACvD,cAAM,SAAS,KAAK,OAAO,IAAI;AAC/B,cAAM,QAAQ,YAAY;AAC1B,cAAM,QAAQ,OAAO,GAAG,EAAE,SAAS,KAAK,KAAK,OAAO,GAAG,EAAE,SAAS,gCAAgC;AAClG,cAAM,MAAM,QAAS,YAAY,KAAK,WAAW,UAAY,WAAW,IAAI,UAAU;AACtF,YAAI,SAAS,WAAW,IAAI;AAC1B,cAAI,UAAU,yBAAyB,QAAQ,gDAA2C;AAAA,QAC5F,OAAO;AACL,cAAI,UAAU,kBAAkB,QAAQ,kBAAkB,KAAK,MAAM,KAAK,CAAC,MAAM,GAAG;AAAA,QACtF;AACA,cAAM,MAAM,KAAK;AAAA,MACnB;AAAA,IACF;AAAA,EACF;AACF;AAEA,SAAS,MAAM,IAA2B;AACxC,SAAO,IAAI,QAAQ,CAAC,YAAY,WAAW,SAAS,EAAE,CAAC;AACzD;;;AChHA;AAFA,SAAS,UAAU,WAAW,QAAQ,aAAa;AACnD,SAAS,eAAe;AAGxB,IAAMA,OAAM;AAQL,SAAS,sBAAsB,UAA+B;AACnE,MAAI,UAAyB,QAAQ,QAAQ;AAE7C,SAAO;AAAA,IACL,MAAM,OAAwB;AAC5B,UAAI;AACF,cAAM,MAAM,MAAM,SAAS,UAAU,OAAO;AAC5C,cAAM,MAAM,SAAS,IAAI,KAAK,GAAG,EAAE;AACnC,eAAO,OAAO,SAAS,GAAG,IAAI,MAAM;AAAA,MACtC,SAAS,KAAK;AACZ,sBAAcA,MAAK,oBAAoB,GAAG;AAC1C,eAAO;AAAA,MACT;AAAA,IACF;AAAA,IACA,MAAM,QAA+B;AAEnC,gBAAU,QAAQ,KAAK,YAAY;AACjC,cAAM,MAAM,WAAW;AACvB,cAAM,MAAM,QAAQ,QAAQ,GAAG,EAAE,WAAW,KAAK,CAAC;AAClD,cAAM,UAAU,KAAK,OAAO,MAAM,GAAG,OAAO;AAC5C,cAAM,OAAO,KAAK,QAAQ;AAAA,MAC5B,CAAC,EAAE,MAAM,SAAO,cAAcA,MAAK,qBAAqB,GAAG,CAAC;AAC5D,aAAO;AAAA,IACT;AAAA,EACF;AACF;;;AHzBA;AACA;;;AIdA;AACA;AAEA,IAAMC,OAAM;AAaZ,IAAM,iBAAiB,CAAC,OAAO,UAAU,UAAU,SAAS,WAAW,SAAS,QAAQ;AAEjF,SAAS,sBAAsB,MAAuB;AAC3D,SAAO,eAAe,KAAK,OAAK,KAAK,WAAW,CAAC,CAAC;AACpD;AAEA,eAAsB,0BACpB,MAAc,QAAgB,KAAkB,OAAoB,MACrD;AAEf,MAAI,KAAK,WAAW,KAAK,GAAG;AAC1B,UAAM,SAAS,KAAK,MAAM,CAAC;AAC3B,QAAI,CAAC,QAAQ;AAAE,YAAM,IAAI,YAAY,QAAQ,sBAAe;AAAG;AAAA,IAAQ;AACvE,QAAI,WAAW,KAAK;AAClB,YAAM,eAAe;AAAA,QACnB,EAAE,KAAK,aAAa,OAAO,YAAY;AAAA,QACvC,EAAE,KAAK,UAAU,OAAO,iBAAiB;AAAA,QACzC,EAAE,KAAK,WAAW,OAAO,mBAAmB;AAAA,QAC5C,EAAE,KAAK,UAAU,OAAO,gBAAgB;AAAA,MAC1C;AACA,YAAM,UAAU,aAAa,IAAI,OAAK,CAAC,EAAE,MAAM,EAAE,OAAO,eAAe,SAAS,EAAE,GAAG,GAAG,CAAC,CAAC;AAC1F,cAAQ,KAAK,CAAC,EAAE,MAAM,iBAAY,eAAe,MAAM,CAAC,CAAC;AACzD,YAAM,IAAI,YAAY,QAAQ,oCAA6B,EAAE,cAAc,EAAE,iBAAiB,QAAQ,EAAE,CAAC;AACzG;AAAA,IACF;AACA,QAAI,OAAO,WAAW,IAAI,GAAG;AAC3B,YAAM,QAAQ,OAAO,MAAM,CAAC;AAC5B,YAAM,EAAE,eAAe,cAAc,uBAAuB,qBAAqB,IAAI,MAAM,OAAO,gCAAsC;AACxI,YAAM,KAAK,cAAc;AACzB,UAAI,CAAC,IAAI;AAAE,cAAM,IAAI,YAAY,QAAQ,kCAA6B;AAAG;AAAA,MAAQ;AACjF,UAAI,YAAY,sBAAsB,EAAE,EAAE,OAAO,OAAK,EAAE,OAAO,cAAc,MAAM;AACnF,UAAI,UAAU,WAAW,GAAG;AAAE,cAAM,IAAI,YAAY,QAAQ,gCAA2B;AAAG;AAAA,MAAQ;AAClG,YAAM,kBAAkB,aAAa,OAAO,EAAE,GAAG;AACjD,YAAM,SAAS,UAAU,cAAc,qBAAqB,SAAS,KAAK;AAC1E,YAAM,UAAU,UAAU,IAAI,OAAK;AACjC,cAAM,QAAQ,qBAAqB,EAAE,IAAI,EAAE;AAC3C,cAAM,QAAQ,EAAE,SAAS,kBAAkB,UAAK,EAAE,IAAI,KAAK,KAAK,MAAM,GAAG,EAAE,IAAI,KAAK,KAAK;AACzF,eAAO,CAAC,EAAE,MAAM,OAAO,eAAe,GAAG,MAAM,IAAI,EAAE,IAAI,GAAG,CAAC;AAAA,MAC/D,CAAC;AACD,cAAQ,KAAK,CAAC,EAAE,MAAM,eAAU,eAAe,OAAO,CAAC,CAAC;AACxD,YAAM,IAAI,YAAY,QAAQ,4BAAqB,EAAE,cAAc,EAAE,iBAAiB,QAAQ,EAAE,CAAC;AACjG;AAAA,IACF;AACA,QAAI,OAAO,WAAW,IAAI,GAAG;AAC3B,YAAM,OAAO,OAAO,MAAM,CAAC;AAC3B,YAAM,eAAe;AACrB,YAAM,EAAE,eAAe,cAAc,uBAAuB,qBAAqB,IAAI,MAAM,OAAO,gCAAsC;AACxI,YAAM,KAAK,cAAc;AACzB,UAAI,CAAC,IAAI;AAAE,cAAM,IAAI,YAAY,QAAQ,kCAA6B;AAAG;AAAA,MAAQ;AACjF,UAAI,YAAY,sBAAsB,EAAE,EAAE,OAAO,OAAK,EAAE,OAAO,cAAc,MAAM;AACnF,YAAM,kBAAkB,aAAa,aAAa,EAAE,GAAG;AACvD,YAAM,YAAY,SAAS,cAAc,SAAS,KAAK,QAAQ,gBAAgB,IAAI;AACnF,YAAM,UAAU,UAAU,IAAI,OAAK;AACjC,cAAM,QAAQ,qBAAqB,EAAE,IAAI,EAAE;AAC3C,cAAM,QAAQ,EAAE,SAAS,kBAAkB,UAAK,EAAE,IAAI,KAAK,KAAK,MAAM,GAAG,EAAE,IAAI,KAAK,KAAK;AACzF,eAAO,CAAC,EAAE,MAAM,OAAO,eAAe,UAAU,IAAI,IAAI,EAAE,IAAI,GAAG,CAAC;AAAA,MACpE,CAAC;AACD,cAAQ,KAAK,CAAC,EAAE,MAAM,eAAU,eAAe,kBAAkB,CAAC,CAAC;AACnE,YAAM,IAAI,YAAY,QAAQ,0BAAmB,SAAS,KAAK,EAAE,cAAc,EAAE,iBAAiB,QAAQ,EAAE,CAAC;AAC7G;AAAA,IACF;AAAA,EACF,WAAW,KAAK,WAAW,QAAQ,GAAG;AACpC,UAAM,QAAQ,KAAK,MAAM,CAAC;AAC1B,UAAM,EAAE,eAAe,cAAc,uBAAuB,qBAAqB,IAAI,MAAM,OAAO,gCAAsC;AACxI,UAAM,KAAK,cAAc;AACzB,QAAI,CAAC,IAAI;AAAE,YAAM,IAAI,YAAY,QAAQ,kCAA6B;AAAG;AAAA,IAAQ;AAEjF,QAAI,UAAU,aAAa;AACzB,YAAM,eAAe,aAAa,aAAa,EAAE;AACjD,YAAM,YAAY,GAAG,OAAO,WAAW,GAAG,aAAa,CAAC;AACxD,YAAM,QAA+C;AAAA,QACnD,EAAE,OAAO,gBAAW,cAAc,SAAS,GAAG,IAAI,KAAK,4BAA4B;AAAA,MACrF;AACA,eAAS,IAAI,GAAG,IAAI,UAAU,QAAQ,KAAK;AACzC,cAAM,KAAK,EAAE,OAAO,YAAO,IAAI,CAAC,KAAK,UAAU,CAAC,EAAG,KAAK,IAAI,KAAK,+BAA+B,IAAI,CAAC,GAAG,CAAC;AAAA,MAC3G;AACA,UAAI,UAAU,SAAS,GAAG;AACxB,cAAM,KAAK,EAAE,OAAO,YAAO,UAAU,SAAS,CAAC,WAAW,KAAK,+BAA+B,UAAU,SAAS,CAAC,GAAG,CAAC;AAAA,MACxH;AACA,YAAMC,WAAU,MAAM,IAAI,OAAK,CAAC,EAAE,MAAM,EAAE,OAAO,eAAe,EAAE,IAAI,CAAC,CAAC;AACxE,MAAAA,SAAQ,KAAK,CAAC,EAAE,MAAM,eAAU,eAAe,OAAO,CAAC,CAAC;AACxD,YAAM,IAAI,YAAY,QAAQ,yBAAkB,EAAE,cAAc,EAAE,iBAAiBA,SAAQ,EAAE,CAAC;AAC9F;AAAA,IACF;AAEA,QAAI,YAAY,sBAAsB,EAAE,EAAE,OAAO,OAAK,EAAE,OAAO,cAAc,MAAM;AACnF,QAAI,UAAU,WAAW,GAAG;AAAE,YAAM,IAAI,YAAY,QAAQ,gCAA2B;AAAG;AAAA,IAAQ;AAClG,UAAM,kBAAkB,aAAa,OAAO,EAAE,GAAG;AACjD,UAAM,UAAU,UAAU,IAAI,OAAK;AACjC,YAAM,QAAQ,qBAAqB,EAAE,IAAI,EAAE;AAC3C,YAAM,QAAQ,EAAE,SAAS,kBAAkB,UAAK,EAAE,IAAI,KAAK,KAAK,MAAM,GAAG,EAAE,IAAI,KAAK,KAAK;AACzF,aAAO,CAAC,EAAE,MAAM,OAAO,eAAe,SAAS,KAAK,IAAI,EAAE,IAAI,GAAG,CAAC;AAAA,IACpE,CAAC;AACD,YAAQ,KAAK,CAAC,EAAE,MAAM,eAAU,eAAe,OAAO,CAAC,CAAC;AACxD,UAAM,IAAI,YAAY,QAAQ,4BAAqB,EAAE,cAAc,EAAE,iBAAiB,QAAQ,EAAE,CAAC;AAAA,EAEnG,WAAW,KAAK,WAAW,QAAQ,GAAG;AACpC,UAAM,CAAC,EAAE,OAAO,YAAY,IAAI,KAAK,MAAM,GAAG;AAC9C,UAAM,EAAE,eAAe,sBAAsB,YAAY,WAAW,IAAI,MAAM,OAAO,gCAAsC;AAC3H,UAAM,KAAK,cAAc;AACzB,QAAI,SAAS,qBAAqB,YAAa;AAC/C,UAAM,iBAAiB,IAAI,UAAU,YAAa;AAClD,QAAI,gBAAgB,cAAc,MAAO,UAAS,OAAO,OAAO,OAAK,CAAC,EAAE,MAAM,UAAU,EAAE,MAAM,WAAW,OAAO;AAClH,QAAI,OAAO,WAAW,GAAG;AAAE,YAAM,IAAI,YAAY,QAAQ,8BAAyB,YAAY,EAAE;AAAG;AAAA,IAAQ;AAC3G,UAAM,eAAe;AACrB,UAAM,oBAAoB,OAAO,IAAI,OAAK,EAAE,EAAE;AAC9C,UAAM,UAAU,OAAO,IAAI,CAAC,GAAG,MAAM,CAAC,EAAE,MAAM,GAAG,EAAE,EAAE,KAAK,WAAW,EAAE,MAAM,IAAI,CAAC,KAAK,WAAW,EAAE,MAAM,IAAI,CAAC,KAAK,eAAe,QAAQ,YAAY,IAAI,CAAC,GAAG,CAAC,CAAC;AACjK,YAAQ,KAAK,CAAC,EAAE,MAAM,eAAU,eAAe,QAAQ,KAAK,GAAG,CAAC,CAAC;AACjE,UAAM,IAAI,YAAY,QAAQ,uBAAgB,YAAY,KAAK,EAAE,cAAc,EAAE,iBAAiB,QAAQ,EAAE,CAAC;AAAA,EAE/G,WAAW,KAAK,WAAW,OAAO,GAAG;AACnC,UAAM,CAAC,EAAE,EAAE,EAAE,IAAI,IAAI,KAAK,MAAM,GAAG;AACnC,UAAM,EAAE,eAAe,cAAc,uBAAuB,qBAAqB,IAAI,MAAM,OAAO,gCAAsC;AACxI,UAAM,KAAK,cAAc;AACzB,QAAI,CAAC,IAAI;AAAE,YAAM,IAAI,YAAY,QAAQ,kCAA6B;AAAG;AAAA,IAAQ;AACjF,QAAI,YAAY,sBAAsB,EAAE,EAAE,OAAO,OAAK,EAAE,OAAO,cAAc,MAAM;AACnF,UAAM,eAAe,aAAa,aAAa,EAAE;AACjD,UAAM,gBAAgB,cAAc,SAAS;AAC7C,QAAI,QAAQ,KAAK,WAAW,cAAc,KAAK,eAAe;AAC5D,UAAI,kBAAkB,OAAO;AAAE,oBAAY,UAAU,OAAO,OAAK,EAAE,OAAO,cAAc,KAAK;AAAA,MAAG,OAC3F;AAAE,oBAAY,UAAU,OAAO,OAAK,EAAE,SAAS,aAAc,YAAY;AAAA,MAAG;AAAA,IACnF;AACA,QAAI,UAAU,WAAW,GAAG;AAAE,YAAM,IAAI,YAAY,QAAQ,8CAAyC;AAAG;AAAA,IAAQ;AAChH,UAAM,kBAAkB,cAAc;AACtC,UAAM,YAAY,SAAS,cAAc,SAAS,KAAM,QAAQ,gBAAgB,IAAI;AACpF,UAAM,UAAU,UAAU,IAAI,OAAK;AACjC,YAAM,QAAQ,qBAAqB,EAAE,IAAI,EAAE;AAC3C,YAAM,QAAQ,EAAE,SAAS,kBAAkB,UAAK,EAAE,IAAI,KAAK,KAAK,MAAM,GAAG,EAAE,IAAI,KAAK,KAAK;AACzF,aAAO,CAAC,EAAE,MAAM,OAAO,eAAe,UAAU,IAAI,IAAI,EAAE,IAAI,GAAG,CAAC;AAAA,IACpE,CAAC;AACD,YAAQ,KAAK,CAAC,EAAE,MAAM,eAAU,eAAe,kBAAkB,CAAC,CAAC;AACnE,UAAM,IAAI,YAAY,QAAQ,0BAAmB,SAAS,KAAK,EAAE,cAAc,EAAE,iBAAiB,QAAQ,EAAE,CAAC;AAAA,EAE/G,WAAW,KAAK,WAAW,SAAS,GAAG;AACrC,UAAM,CAAC,EAAE,MAAM,YAAY,IAAI,KAAK,MAAM,GAAG;AAC7C,UAAM,EAAE,sBAAsB,YAAY,YAAY,cAAc,IAAI,MAAM,OAAO,gCAAsC;AAC3H,UAAM,KAAK,cAAc;AACzB,UAAM,iBAAiB,IAAI,UAAU,YAAa;AAClD,QAAI,SAAS,qBAAqB,YAAa;AAC/C,QAAI,gBAAgB,cAAc,MAAO,UAAS,OAAO,OAAO,OAAK,CAAC,EAAE,MAAM,UAAU,EAAE,MAAM,WAAW,OAAO;AAClH,QAAI,OAAO,WAAW,GAAG;AAAE,YAAM,IAAI,YAAY,QAAQ,8BAAyB,YAAY,EAAE;AAAG;AAAA,IAAQ;AAC3G,UAAM,eAAe;AACrB,UAAM,YAAY,SAAS,cAAc,SAAS,KAAM,QAAQ,gBAAgB,IAAI;AACpF,UAAM,oBAAoB,OAAO,IAAI,OAAK,EAAE,EAAE;AAC9C,UAAM,UAAU,OAAO,IAAI,CAAC,GAAG,MAAM,CAAC,EAAE,MAAM,GAAG,EAAE,EAAE,KAAK,WAAW,EAAE,MAAM,IAAI,CAAC,KAAK,WAAW,EAAE,MAAM,IAAI,CAAC,KAAK,eAAe,QAAQ,YAAY,IAAI,CAAC,GAAG,CAAC,CAAC;AACjK,YAAQ,KAAK,CAAC,EAAE,MAAM,eAAU,eAAe,QAAQ,IAAI,GAAG,CAAC,CAAC;AAChE,UAAM,IAAI,YAAY,QAAQ,4BAAqB,SAAS,KAAK,EAAE,cAAc,EAAE,iBAAiB,QAAQ,EAAE,CAAC;AAAA,EAEjH,WAAW,KAAK,WAAW,OAAO,GAAG;AACnC,UAAM,QAAQ,KAAK,MAAM,GAAG;AAC5B,UAAM,eAAe,MAAM,CAAC;AAC5B,UAAM,WAAW,SAAS,MAAM,CAAC,GAAI,EAAE;AACvC,UAAM,QAAQ,OAAO,SAAS,QAAQ,KAAK,MAAM,kBAAkB,QAAQ,IACvE,MAAM,kBAAkB,QAAQ,IAChC,MAAM,MAAM,CAAC,EAAE,KAAK,GAAG;AAC3B,UAAM,OAAO,MAAM,gBAAgB;AACnC,UAAM,eAAe;AACrB,UAAM,oBAAoB,CAAC;AAC3B,UAAM,EAAE,eAAe,sBAAsB,cAAc,sBAAsB,uBAAuB,sBAAsB,IAAI,MAAM,OAAO,gCAAsC;AACrL,UAAM,KAAK,cAAc;AACzB,QAAI,CAAC,IAAI;AAAE,YAAM,IAAI,YAAY,QAAQ,kCAA6B;AAAG;AAAA,IAAQ;AAEjF,UAAM,cAAc,qBAAqB,YAAY;AACrD,QAAI,CAAC,YAAY,KAAK,OAAK,EAAE,OAAO,KAAK,GAAG;AAAE,YAAM,IAAI,YAAY,QAAQ,UAAK,KAAK,wBAAwB,YAAY,iBAAiB;AAAG;AAAA,IAAQ;AAEtJ,UAAM,iBAAiB,GAAG,UAAU,YAAY;AAChD,QAAI,CAAC,gBAAgB;AAAE,YAAM,IAAI,YAAY,QAAQ,mBAAc,YAAY,YAAY;AAAG;AAAA,IAAQ;AACtG,UAAM,aAAa,sBAAsB,cAAc,gBAAgB,OAAO,CAAC;AAC/E,QAAI,CAAC,WAAW,IAAI;AAAE,YAAM,IAAI,YAAY,QAAQ,sBAAsB,cAAc,UAAU,CAAC;AAAG;AAAA,IAAQ;AAE9G,QAAI,gBAAgB,cAAc,OAAO;AACvC,UAAI;AACF,cAAM,WAAW,eAAe,YAAY;AAC5C,cAAM,SAAS,OAAO,EAAE,UAAU,eAAe,aAAa,SAAS;AACvE,cAAM,UAAkC,CAAC;AACzC,YAAI,OAAQ,SAAQ,eAAe,IAAI,UAAU,MAAM;AACvD,cAAM,MAAM,MAAM,MAAM,GAAG,QAAQ,WAAW,EAAE,SAAS,QAAQ,YAAY,QAAQ,GAAI,EAAE,CAAC;AAC5F,YAAI,CAAC,IAAI,IAAI;AAAE,gBAAM,IAAI,YAAY,QAAQ,gBAAM,YAAY,iBAAiB,IAAI,MAAM,iBAAiB;AAAG;AAAA,QAAQ;AAAA,MACxH,SAAS,KAAK;AAAE,sBAAcD,MAAK,+BAA+B,GAAG;AAAG,cAAM,IAAI,YAAY,QAAQ,gBAAM,YAAY,4BAA4B;AAAG;AAAA,MAAQ;AAAA,IACjK;AAEA,UAAM,WAAW,KAAK,WAAW,cAAc,IAAI,cAAc;AACjE,UAAM,UAAU,SAAS,kBAAkB,IAAI,SAAS,kBAAkB,IAAI,SAAS,kBAAkB,IAAI;AAE7G,QAAI,WAAW,GAAG;AAChB,UAAI,CAAC,GAAG,OAAO,WAAW,EAAG,IAAG,OAAO,WAAW,IAAI,EAAE,OAAO,IAAI,UAAU,GAAG;AAChF,UAAI,CAAC,GAAG,OAAO,WAAW,EAAG,UAAW,IAAG,OAAO,WAAW,EAAG,YAAY,CAAC;AAC7E,SAAG,OAAO,WAAW,EAAG,UAAU,OAAO,IAAI,EAAE,OAAO,UAAU,aAAa;AAC7E,YAAM,EAAE,mBAAmB,IAAI,MAAM,OAAO,gCAAsC;AAClF,yBAAmB,IAAI,KAAK;AAC5B,2BAAqB,IAAI,sBAAsB,UAAU,CAAC,WAAM,KAAK,KAAK,YAAY,GAAG;AACzF,YAAM,IAAI,YAAY,QAAQ,mBAAc,UAAU,CAAC,WAAM,KAAK,KAAK,YAAY,GAAG;AAAA,IACxF,OAAO;AACL,YAAM,cAAc,GAAG,OAAO,QAAQ,GAAG;AACzC,SAAG,OAAO,QAAQ,IAAI,EAAE,GAAG,GAAG,OAAO,QAAQ,GAAI,OAAO,UAAU,aAAa;AAC/E,YAAM,EAAE,mBAAmB,IAAI,MAAM,OAAO,gCAAsC;AAClF,yBAAmB,IAAI,KAAK;AAC5B,2BAAqB,IAAI,GAAG,QAAQ,WAAM,KAAK,KAAK,YAAY,GAAG;AAEnE,YAAM,kBAAkB,gBAAgB;AACxC,YAAM,cAAc,aAAa;AAEjC,UAAI;AACJ,UAAI;AACJ,UAAI;AACJ,UAAI,eAAe,iBAAiB;AAClC,cAAM,cAAc,aAAa,QAAQ,EAAE,GAAG,IAAI,QAAQ,EAAE,GAAG,GAAG,QAAQ,MAAM,EAAE,OAAO,IAAI,UAAU,YAAa,EAAE,EAAE,CAAC;AACzH,sBAAc,aAAa,QAAQ,EAAE,GAAG,IAAI,QAAQ,EAAE,GAAG,GAAG,QAAQ,MAAM,EAAE,OAAO,UAAU,aAAa,EAAE,EAAE,CAAC;AAC/G,kBAAU,aAAa,SAAS,aAAa;AAC7C,kBAAU,aAAa,SAAS,aAAa;AAC7C,YAAI,YAAY,SAAS;AACvB,gBAAM,cAAwB,CAAC;AAC/B,qBAAW,CAAC,GAAG,UAAU,KAAK,OAAO,QAAQ,GAAG,MAAM,GAAG;AACvD,gBAAI,MAAM,YAAa;AACvB,kBAAM,KAAK,GAAG,UAAU,WAAW,QAAQ;AAC3C,gBAAI,MAAM,GAAG,cAAc,SAAS;AAAE,iBAAG,OAAO,CAAC,IAAI,EAAE,OAAO,UAAU,aAAa;AAAG,0BAAY,KAAK,CAAC;AAAA,YAAG;AAAA,UAC/G;AACA,cAAI,YAAY,SAAS,EAAG,sBAAqB,IAAI,YAAY,YAAY,KAAK,IAAI,CAAC,WAAM,YAAY,EAAE;AAAA,QAC7G;AAAA,MACF;AAEA,UAAI,eAAe,CAAC,mBAAmB,cAAc,KAAK,WAAW;AACnE,cAAO,KAAK,UAAoE,SAAS,KAAK;AAC9F,cAAM,IAAI,YAAY,QAAQ,sBAAiB,KAAK,EAAE;AAAA,MACxD,WAAW,eAAe,mBAAmB,YAAY,WAAW,oBAAoB,KAAK,WAAW;AACtG,YAAI;AACF,gBAAM,EAAE,eAAe,IAAI,MAAM,OAAO,+BAA+C;AACvF,gBAAM,EAAE,oBAAoB,IAAI,MAAM,OAAO,qCAAqD;AAClG,gBAAM,SAAS,OAAO,EAAE,UAAU,aAAa,SAAS,aAAa,SAAS;AAC9E,gBAAM,aAAa,CAAC,EAAE,UAAU,YAAa,SAAS,UAAW,QAAQ,OAAO,YAAY,YAAa,cAAc,CAAC;AACxH,qBAAW,MAAO,GAAG,OAAO,WAAW,GAAG,aAAa,CAAC,GAAI;AAC1D,kBAAM,QAAQ,aAAa,OAAO,EAAE,GAAG,IAAI,QAAQ,EAAE,GAAG,GAAG,QAAQ,KAAK,EAAE,OAAO,GAAG,OAAO,UAAU,GAAG,SAAS,EAAE,EAAE,CAAC;AACtH,gBAAI,MAAO,YAAW,KAAK,EAAE,UAAU,MAAM,SAAS,UAAW,QAAQ,MAAM,SAAS,YAAY,OAAO,EAAE,UAAU,MAAM,SAAS,SAAS,IAAI,QAAQ,OAAO,GAAG,OAAO,YAAY,MAAM,cAAc,CAAC;AAAA,UAC/M;AACA,gBAAM,WAAY,KAAK,UAA6F,QAAQ,YAAY,IAAI,oBAAoB;AAChK,gBAAM,SAAS,IAAI,eAAe,YAAY,QAAQ;AACtD,UAAC,KAAK,UAAkE,eAAe,EAAE,UAAU,YAAa,SAAS,UAAW,QAAQ,OAAO,YAAY,YAAa,eAAe,OAAO,CAAC;AAAA,QACrM,SAAS,KAAK;AACZ,gBAAM,IAAI,YAAY,QAAQ,iCAAuB,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC,wBAAwB;AAC7H;AAAA,QACF;AACA,YAAI;AAAE,gBAAM,IAAI,YAAY,QAAQ,sBAAiB,KAAK,KAAK,YAAY,GAAG;AAAA,QAAG,SAAS,KAAK;AAAE,wBAAcA,MAAK,oCAAoC,GAAG;AAAA,QAAG;AAAA,MAChK,WAAW,eAAe,iBAAiB;AACzC,cAAM,cAAc,YAAY,UAAU,2BAA2B;AACrE,YAAI;AACF,cAAI,KAAK,SAAS,iBAAkB,OAAM,KAAK,SAAS,iBAAiB;AACzE,gBAAM,KAAK,2BAA2B,QAAQ,wBAAwB;AACtE,gBAAM,IAAI,YAAY,QAAQ,yBAAkB,KAAK,KAAK,YAAY,wBAAwB,WAAW,EAAE;AAAA,QAC7G,SAAS,KAAK;AACZ,gBAAM,IAAI,YAAY,QAAQ,0CAAgC,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC,wBAAwB;AAAA,QACxI;AAAA,MACF,OAAO;AACL,cAAM,IAAI,YAAY,QAAQ,UAAK,QAAQ,WAAM,KAAK,KAAK,YAAY,GAAG;AAAA,MAC5E;AAAA,IACF;AAAA,EACF,WAAW,KAAK,WAAW,QAAQ,GAAG;AACpC,UAAM,WAAW,KAAK,MAAM,CAAC;AAC7B,QAAI,cAAc,KAAK,aAAa,OAAO,KAAK,UAAU,aAAa,YAAY;AACjF,UAAI;AACF,cAAM,KAAK,UAAU,SAAS,QAAQ;AACtC,cAAM,KAAK,2BAA2B,MAAM;AAC5C,cAAM,IAAI,YAAY,QAAQ,mCAAuB,QAAQ,EAAE;AAAA,MACjE,SAAS,KAAK;AACZ,cAAM,IAAI,YAAY,QAAQ,+BAA0B,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC,EAAE;AAAA,MAC5G;AAAA,IACF;AAAA,EACF;AACF;;;AJhRO,IAAM,wBAA8C,EAAE,OAAO,MAAM,WAAW,MAAM,QAAQ,MAAM,SAAS,KAAK;AAavH,IAAME,OAAM;AAiBL,IAAM,kBAAN,MAAiD;AAAA,EAC7C,OAAO;AAAA,EACP,eAAqC;AAAA,EAE7B;AAAA,EACA;AAAA,EACA,YAAY,IAAI,kBAAkB;AAAA,EAClC;AAAA,EACA;AAAA,EACT,SAAgC;AAAA,EAChC,cAAc;AAAA,EACtB;AAAA,EACA,oBAA8B,CAAC;AAAA,EAE/B,YAAY,QAA+B,MAA2B;AACpE,SAAK,MAAM,IAAI,YAAY,OAAO,QAAQ;AAC1C,SAAK,eAAe,IAAI,aAAa,UAAU,CAAC;AAChD,SAAK,SAAS;AACd,SAAK,OAAO;AAAA,EACd;AAAA;AAAA,EAGA,iBAAiB,QAAgB,MAAoB;AACnD,SAAK,IAAI,YAAY,SAAS,QAAQ,EAAE,GAAG,IAAI,EAAE,MAAM,SAAO,cAAcA,MAAK,oBAAoB,GAAG,CAAC;AAAA,EAC3G;AAAA;AAAA,EAGA,MAAc,2BAA2B,QAAgB,SAAS,gBAA+B;AAC/F,UAAM,KAAK,IAAI,YAAY,QAAQ,yBAAoB;AACvD,UAAM,IAAI,KAAK,KAAK;AACpB,UAAM,aAAa,YAAY,MAAM;AACrC,UAAM,SAAS,YAAY,MAAM;AACjC,UAAM,EAAE,SAAS,KAAK,YAAY,MAAM;AACxC,UAAM,gBAAgB;AAAA,MACpB,WAAW,KAAK,KAAK;AAAA,MAAW;AAAA,MAAY;AAAA,MAC5C,UAAU,EAAE;AAAA,MAAU,oBAAoB,KAAK,KAAK;AAAA,MAAoB;AAAA,IAC1E,CAAC;AACD,QAAI,EAAE,aAAa,eAAe;AAChC,YAAM,MAAM,UAAU;AACtB,YAAM,OAAO,IAAI,aAAa,IAAI,OAAO,MAAM,CAAC;AAChD,UAAI,KAAM,GAAE,eAAe,EAAE,aAAa,WAAW,KAAK,MAAM;AAAA,IAClE;AAAA,EACF;AAAA,EAEA,MAAM,QAAuB;AAC3B,UAAM,UAAU,MAAM,KAAK,IAAI,MAAM;AACrC,SAAK,cAAc,QAAQ,UAAU,YAAY,KAAK;AACtD,YAAQA,MAAK,mBAAY,QAAQ,QAAQ,EAAE;AAE3C,UAAM,KAAK,IAAI,cAAc,aAAa,IAAI,QAAM,EAAE,SAAS,EAAE,MAAM,aAAa,EAAE,YAAY,EAAE,CAAC,EAClG,MAAM,CAAC,QAAQ,QAAQA,MAAK,yBAAyB,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC,EAAE,CAAC;AAE3G,UAAM,OAAO,WAAW;AACxB,UAAM,cAAc,sBAAsB,GAAG,IAAI,wBAAwB;AACzE,SAAK,SAAS,IAAI,eAAe,KAAK,KAAK,KAAK,OAAO,cAAc,CAAC,MAAM,KAAK,aAAa,CAAC,GAAG,WAAW;AAC7G,UAAM,KAAK,OAAO,MAAM;AAAA,EAC1B;AAAA,EAEA,OAAa;AACX,SAAK,QAAQ,KAAK;AAClB,SAAK,SAAS;AAAA,EAChB;AAAA,EAEA,UAAU,KAA8B;AACtC,WAAO,KAAK,aAAa,cAAc,IAAI,QAAQ;AAAA,EACrD;AAAA,EAEA,MAAM,YAAY,WAAmB,MAAc,MAA8C;AAC/F,UAAM,SAAS,SAAS,WAAW,EAAE;AACrC,UAAM,WAAoC,CAAC;AAC3C,QAAI,MAAM,SAAU,UAAS,oBAAoB,SAAS,KAAK,UAAU,EAAE;AAC3E,QAAI,MAAM,UAAW,UAAS,aAAa,KAAK;AAChD,QAAI,MAAM,aAAc,UAAS,eAAe,KAAK;AACrD,WAAO,KAAK,IAAI,YAAY,QAAQ,MAAM,QAAQ;AAAA,EACpD;AAAA,EAEA,MAAM,YAAY,WAAmB,WAAmB,MAA6B;AACnF,UAAM,SAAS,SAAS,WAAW,EAAE;AACrC,UAAM,KAAK,IAAI,gBAAgB,QAAQ,WAAW,IAAI;AAAA,EACxD;AAAA,EAEA,cAAc,MAAwB;AACpC,WAAO,KAAK,UAAU,UAAU,IAAI;AAAA,EACtC;AAAA,EAEA,MAAM,WAAW,WAAmB,UAAkC;AACpE,UAAM,KAAK,IAAI,eAAe,SAAS,WAAW,EAAE,GAAG,UAAU,WAAW,SAAS,UAAU,EAAE,IAAI,MAAS;AAAA,EAChH;AAAA,EAEA,MAAM,YAAY,WAAmB,WAAmB,OAA8B;AACpF,QAAI,aAAa,EAAG;AACpB,QAAI;AACF,YAAM,WAAW,QAAQ,CAAC,EAAE,MAAM,SAAkB,MAAM,CAAC,IAAI,CAAC;AAChE,YAAM,KAAK,IAAI,mBAAmB,SAAS,WAAW,EAAE,GAAG,WAAW,QAAQ;AAAA,IAChF,SAAS,KAAK;AACZ,eAASA,MAAK,iBAAiB,SAAS,QAAQ,MAAM,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC,EAAE;AAAA,IAC1G;AAAA,EACF;AAAA,EAEA,MAAM,cAAc,QAAiC;AACnD,UAAM,WAAW,MAAM,KAAK,IAAI,QAAQ,MAAM;AAC9C,QAAI,CAAC,SAAS,UAAW,OAAM,IAAI,MAAM,uBAAuB;AAChE,WAAO,KAAK,IAAI,aAAa,SAAS,SAAS;AAAA,EACjD;AAAA,EAEA,MAAM,UAAU,WAAmB,OAAe,MAAgC;AAChF,UAAM,WAAoC,CAAC;AAC3C,QAAI,MAAM,SAAU,UAAS,oBAAoB,SAAS,KAAK,UAAU,EAAE;AAC3E,UAAM,KAAK,IAAI,UAAU,SAAS,WAAW,EAAE,GAAG,OAAO,QAAQ;AAAA,EACnE;AAAA;AAAA,EAGA,MAAM,aAAa,WAAmB,UAAkB,SAAkB,MAAkC;AAC1G,UAAM,WAA2C,CAAC;AAClD,QAAI,MAAM,SAAU,UAAS,oBAAoB,SAAS,KAAK,UAAU,EAAE;AAC3E,WAAO,KAAK,IAAI,aAAa,SAAS,WAAW,EAAE,GAAG,UAAU,SAAS,QAAQ;AAAA,EACnF;AAAA,EAEA,cAAc,KAA2B;AACvC,QAAI,CAAC,KAAK,OAAQ;AAClB,SAAK,OAAO,aAAa;AAAA,MACvB,WAAW;AAAA,MACX,SAAS;AAAA,QACP,YAAY;AAAA,QACZ,MAAM,EAAE,IAAI,SAAS,IAAI,WAAW,EAAE,GAAG,QAAQ,OAAO,YAAY,SAAS;AAAA,QAC7E,MAAM,EAAE,IAAI,SAAS,IAAI,WAAW,EAAE,GAAG,MAAM,UAAU;AAAA,QACzD,MAAM,KAAK,MAAM,KAAK,IAAI,IAAI,GAAI;AAAA,QAClC,MAAM,IAAI;AAAA,QACV,GAAI,IAAI,WAAW,EAAE,mBAAmB,SAAS,IAAI,UAAU,EAAE,EAAE,IAAI,CAAC;AAAA,MAC1E;AAAA,IACF,CAAC;AAAA,EACH;AAAA;AAAA,EAIA,MAAc,aAAa,QAAuC;AAChE,aAASA,MAAK,WAAW,KAAK,UAAU,MAAM,EAAE,MAAM,GAAG,GAAG,CAAC,EAAE;AAE/D,QAAI,OAAO,gBAAgB;AACzB,YAAM,OAAO,OAAO,eAAe,QAAQ;AAC3C,YAAMC,UAAS,OAAO,eAAe,SAAS,MAAM;AACpD,WAAK,IAAI,oBAAoB,OAAO,eAAe,EAAE,EAAE,MAAM,SAAO,cAAcD,MAAK,uBAAuB,GAAG,CAAC;AAClH,UAAI,CAACC,QAAQ;AAEb,UAAI;AACJ,YAAI,sBAAsB,IAAI,GAAG;AAC/B,gBAAM,0BAA0B,MAAMA,SAAQ,KAAK,KAAK,MAAM;AAAA,YAC5D,WAAW,KAAK,KAAK;AAAA,YACrB,UAAU,KAAK,KAAK;AAAA,YACpB,4BAA4B,CAAC,KAAK,WAAW,KAAK,2BAA2B,KAAK,MAAM;AAAA,UAC1F,CAAC;AAAA,QACH,WAAW,KAAK,WAAW,QAAQ,GAAG;AAEpC,gBAAM,WAAW,KAAK,MAAM,CAAC;AAC7B,gBAAM,YAAY,KAAK,KAAK;AAC5B,cAAI,cAAc,aAAa,OAAQ,UAAoC,aAAa,YAAY;AAClG,gBAAI;AACF,oBAAO,UAAgE,SAAS,QAAQ;AACxF,oBAAM,KAAK,2BAA2BA,OAAM;AAC5C,kBAAIA,QAAQ,OAAM,KAAK,IAAI,YAAYA,SAAQ,mCAAuB,QAAQ,EAAE;AAAA,YAClF,SAAS,KAAK;AACZ,kBAAIA,QAAQ,OAAM,KAAK,IAAI,YAAYA,SAAQ,+BAA0B,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC,EAAE;AAAA,YAC7H;AAAA,UACF;AAAA,QACF,WAAW,KAAK,WAAW,OAAO,GAAG;AAEnC,cAAI,KAAK,KAAK,YAAY,eAAe,IAAI,GAAG;AAC9C,kBAAM,KAAK,IAAI,YAAYA,SAAQ,mBAAc;AAAA,UACnD;AAAA,QACF;AAAA,MACA,SAAS,KAAK;AACZ,gBAAQD,MAAK,2BAA2B,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC,EAAE;AAC1F,aAAK,IAAI,YAAYC,SAAQ,8CAA+B,EAAE,MAAM,UAAQ,cAAcD,MAAK,6BAA6B,IAAI,CAAC;AAAA,MACnI;AACA;AAAA,IACF;AAEA,QAAI,OAAO,kBAAkB;AAC3B,YAAM,KAAK,eAAe,MAAM;AAChC;AAAA,IACF;AAEA,UAAM,UAAU,OAAO;AACvB,QAAI,CAAC,SAAS,KAAM;AAEpB,UAAM,UAAU,QAAQ,QAAQ,IAAI;AACpC,UAAM,WAAW,QAAQ,QAAQ,SAAS,QAAQ,KAAK;AACvD,UAAM,WAAW,QAAQ,QAAQ,OAAO,MAAM;AAC9C,UAAM,cAAc,QAAQ,QAAQ,QAAQ;AAC5C,QAAI,CAAC,WAAW,CAAC,YAAY,CAAC,YAAY,CAAC,YAAa;AAExD,UAAM,SAAS,QAAQ,KAAK;AAC5B,UAAM,UAAU,QAAQ,KAAK,SAAS,WAAW,QAAQ,KAAK,SAAS;AACvE,UAAM,WAAW,UAAU,QAAQ,oBAAoB;AACvD,UAAM,YAAY,QAAQ;AAC1B,UAAM,aAAa,QAAQ,KAAK,cAAc,QAAQ,KAAK,YAAY,MAAM,QAAQ,KAAK,EAAE;AAC5F,UAAM,SAAS,YAAY,OAAO,MAAM,MAAM,IAAI,QAAQ,KAAK,MAAM,MAAM;AAE3E,QAAI,OAAO,QAAQ,QAAQ;AAC3B,QAAI,cAAc;AAClB,QAAI;AAGJ,QAAI,YAAY,CAAC,SAAS;AACxB,UAAI,CAAC,KAAK,KAAK,SAAS,WAAW;AACjC,YAAI,SAAS;AACX,eAAK,KAAK,mBAAmB,KAAK,QAAQ,YAAY,6BAA6B;AAAA,QACrF,WAAW,KAAK,aAAa,cAAc,OAAO,QAAQ,MAAM,EAAE,CAAC,GAAG;AACpE,gBAAM,KAAK,IAAI,YAAY,QAAQ,yDAAkD,EAAE,mBAAmB,SAAS,CAAC;AAAA,QACtH;AACA;AAAA,MACF;AAEA,UAAI,CAAC,KAAK,aAAa,cAAc,OAAO,QAAQ,MAAM,EAAE,CAAC,GAAG;AAC9D,YAAI,QAAS,MAAK,KAAK,mBAAmB,KAAK,QAAQ,YAAY,cAAc;AACjF;AAAA,MACF;AAGA,UAAI,SAAS;AACX,YAAI;AACF,gBAAM,KAAK,YAAY,OAAO,MAAM,GAAG,WAAW,WAAI;AACtD,gBAAM,YAAY,QAAQ,SAAS,QAAQ;AAC3C,gBAAM,cAAc,MAAM,KAAK,cAAc,UAAW,OAAO;AAC/D,gBAAM,EAAE,gBAAgB,IAAI,MAAM,OAAO,mBAAyB;AAClE,gBAAM,SAAS,MAAM,gBAAgB,aAAa,aAAa,KAAK,KAAK,SAAS,SAAU;AAC5F,gBAAM,aAAa,OAAO;AAE1B,cAAI,CAAC,YAAY;AACf,kBAAM,KAAK,YAAY,OAAO,MAAM,GAAG,WAAW,EAAE;AACpD,iBAAK,KAAK,mBAAmB,KAAK,QAAQ,YAAY,sBAAsB;AAC5E;AAAA,UACF;AAEA,gBAAM,YAAY,IAAI,OAAO,KAAK,KAAK,WAAW,OAAO,GAAG;AAC5D,cAAI,CAAC,UAAU,KAAK,UAAU,KAAK,CAAC,WAAW,WAAW,GAAG,GAAG;AAC9D,kBAAM,KAAK,YAAY,OAAO,MAAM,GAAG,WAAW,EAAE;AACpD,iBAAK,KAAK,mBAAmB,KAAK,QAAQ,YAAY,WAAW,UAAU,EAAE;AAC7E,qBAASA,MAAK,+BAA+B,WAAW,MAAM,GAAG,EAAE,CAAC,GAAG;AACvE;AAAA,UACF;AACA,iBAAO,WAAW,QAAQ,WAAW,EAAE,EAAE,KAAK;AAC9C,wBAAc;AACd,cAAI,CAAC,MAAM;AAAE,kBAAM,KAAK,YAAY,OAAO,MAAM,GAAG,WAAW,EAAE;AAAG;AAAA,UAAQ;AAAA,QAC9E,SAAS,KAAK;AACZ,mBAASA,MAAK,8BAA8B,GAAG;AAC/C,gBAAM,KAAK,YAAY,OAAO,MAAM,GAAG,WAAW,EAAE;AACpD;AAAA,QACF;AAAA,MACF,OAAO;AAEL,sBAAc;AACd,uBAAe,QAAQ,SAAS,QAAQ,OAAQ;AAAA,MAClD;AAAA,IACF;AAGA,QAAI,CAAC,eAAe,SAAS;AAC3B,YAAM,YAAY,IAAI,OAAO,IAAI,KAAK,WAAW,OAAO,GAAG;AAC3D,YAAM,YAAY,UAAU,KAAK,IAAI;AACrC,YAAM,YAAY,KAAK,WAAW,GAAG;AAErC,UAAI,CAAC,aAAa,CAAC,WAAW;AAC5B,aAAK,KAAK,mBAAmB,KAAK,QAAQ,YAAY,IAAI;AAC1D,iBAASA,MAAK,2BAA2B,UAAU,MAAM,KAAK,MAAM,GAAG,EAAE,CAAC,GAAG;AAC7E;AAAA,MACF;AAEA,UAAI,WAAW;AACb,eAAO,KAAK,QAAQ,WAAW,EAAE,EAAE,KAAK;AACxC,YAAI,CAAC,KAAM;AAAA,MACb;AAAA,IACF;AAGA,QAAI,CAAC,eAAe,CAAC,KAAK,aAAa,cAAc,OAAO,QAAQ,MAAM,EAAE,CAAC,GAAG;AAC9E,UAAI,QAAS,MAAK,KAAK,mBAAmB,KAAK,QAAQ,YAAY,IAAI;AACvE,cAAQA,MAAK,qBAAqB,QAAQ,KAAK,EAAE,EAAE;AACnD;AAAA,IACF;AAGA,QAAI;AAEJ,SAAK,YAAY,gBAAgB,KAAK,aAAa,cAAc,OAAO,QAAQ,MAAM,EAAE,CAAC,GAAG;AAC1F,UAAI;AACF,cAAM,EAAE,iBAAiB,IAAI,MAAM,OAAO,2BAAiC;AAC3E,YAAI;AACJ,YAAI;AACJ,YAAI;AAEJ,YAAI,UAAU;AACZ,gBAAM,QAAQ,QAAQ,MAAO,QAAQ,MAAO,SAAS,CAAC;AACtD,mBAAS,MAAM;AACf,oBAAU;AAAA,QACZ,OAAO;AACL,mBAAS,QAAQ,SAAU;AAC3B,oBAAU,QAAQ,SAAU,YAAY,OAAO,QAAQ,SAAU,UAAU,MAAM,GAAG,EAAE,IAAI,KAAK,MAAM;AACrG,wBAAc,QAAQ,SAAU;AAAA,QAClC;AAEA,cAAM,MAAM,MAAM,KAAK,cAAc,MAAM;AAC3C,cAAM,QAAQ,MAAM,iBAAiB,KAAK,QAAQ,EAAE,SAAS,YAAY,CAAC;AAC1E,YAAI,OAAO;AACT,sBAAY,MAAM;AAClB,cAAI,CAAC,KAAM,QAAO,QAAQ,WAAW,eAAe,MAAM,UAAU,UAAU,MAAM;AAAA,QACtF,OAAO;AACL,cAAI,CAAC,KAAM,QAAO;AAAA,QACpB;AAAA,MACF,SAAS,KAAK;AACZ,gBAAQA,MAAK,0BAA0B,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC,EAAE;AACzF,YAAI,CAAC,KAAM,QAAO;AAAA,MACpB;AAAA,IACF;AAGA,UAAM,QAAQ,QAAQ;AACtB,QAAI,OAAO,MAAM;AACf,YAAM,YAAY,MAAM,MAAM,cAAc;AAC5C,aAAO,gBAAgB,SAAS,MAAM,MAAM,KAAK,MAAM,GAAG,GAAG,CAAC;AAAA,EAAO,IAAI;AAAA,IAC3E;AAGA,QAAI,OAAO,OAAO,UAAU,CAAC,aAAa,KAAK,aAAa,cAAc,OAAO,QAAQ,MAAM,EAAE,CAAC,GAAG;AACnG,UAAI;AACF,cAAM,EAAE,iBAAiB,IAAI,MAAM,OAAO,2BAAiC;AAC3E,cAAM,QAAQ,MAAM,MAAM,MAAM,MAAM,SAAS,CAAC;AAChD,cAAM,MAAM,MAAM,KAAK,cAAc,MAAM,OAAO;AAClD,cAAM,QAAQ,MAAM,iBAAiB,KAAK,QAAQ,EAAE,SAAS,OAAO,CAAC;AACrE,YAAI,MAAO,aAAY,MAAM;AAAA,MAC/B,SAAS,KAAK;AACZ,gBAAQA,MAAK,mCAAmC,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC,EAAE;AAAA,MACpG;AAAA,IACF;AAGA,UAAM,eAAe,UAAU,EAAE,aAAa,IAAI,cAAc,QAAQ,KAAK,EAAE,GAAG,UAAU;AAC5F,UAAM,UAA0B;AAAA,MAC9B,UAAU;AAAA,MACV,WAAW,OAAO,MAAM;AAAA,MACxB,QAAQ;AAAA,MACR,UAAU,OAAO,QAAQ,KAAK,EAAE;AAAA,MAChC;AAAA,MACA;AAAA,MACA,WAAW,QAAQ,OAAO;AAAA,MAC1B,UAAU,YAAY,OAAO,OAAO,QAAQ,IAAI;AAAA,MAChD;AAAA,MACA;AAAA,MACA,SAAS;AAAA,MACT;AAAA,MACA;AAAA,MACA,iBAAiB;AAAA,IACnB;AAKA,QAAI,KAAK,WAAW,GAAG,KAAK,CAAC,KAAK,WAAW,IAAI,GAAG;AAClD,2BAAqB,SAAS,MAAM,KAAK,KAAK,QAAQ,EAAE,MAAM,CAAC,QAAQ;AACrE,iBAASA,MAAK,2BAA2B,eAAe,QAAQ,IAAI,UAAU,OAAO,GAAG,CAAC,EAAE;AAAA,MAC7F,CAAC;AACD;AAAA,IACF;AAEA,UAAM,qBAAqB,SAAS,MAAM,KAAK,KAAK,QAAQ;AAAA,EAC9D;AAAA,EAEA,MAAc,eAAe,QAAuC;AAClE,UAAM,WAAW,OAAO;AACxB,UAAM,OAAO,SAAS;AACtB,QAAI,CAAC,MAAM;AAAE,eAASA,MAAK,oCAAoC;AAAG;AAAA,IAAQ;AAC1E,QAAI,KAAK,OAAQ;AAEjB,UAAM,YAAY,IAAI,IAAI,SAAS,aAAa,IAAI,CAAC,MAAM,EAAE,KAAK,CAAC;AACnE,UAAM,QAAQ,SAAS,aAAa,OAAO,CAAC,MAAM,CAAC,UAAU,IAAI,EAAE,KAAK,CAAC;AACzE,QAAI,MAAM,WAAW,EAAG;AAExB,UAAM,aAAa,KAAK,cAAc,KAAK,YAAY,MAAM,KAAK,EAAE;AACpE,UAAM,SAAS,MAAM,IAAI,CAAC,MAAM,EAAE,KAAK;AACvC,YAAQA,MAAK,YAAY,OAAO,KAAK,EAAE,CAAC,SAAS,UAAU,WAAW,SAAS,UAAU,EAAE;AAE3F,UAAM,eAAe,KAAK,aAAa,cAAc,OAAO,KAAK,EAAE,CAAC;AACpE,UAAM,SAAS,qBAAqB,YAAY,MAAM;AACtD,UAAM,SAAS,SAAS,KAAK;AAC7B,UAAM,QAAQ,cAAc,cAAc,SAAS,KAAK,IAAI;AAE5D,QAAI,gBAAgB,KAAK,KAAK,QAAQ;AACpC,YAAM,QAAQ,aAAa,OAAO,CAAC,CAAE;AACrC,YAAM,MAAM,WAAW,OAAO,CAAC,CAAE;AACjC,YAAM,UAAU,KAAK,KAAK,OAAO,0BAA0B,UAAU,EAAE,aAAa,IAAI,YAAY,MAAM,EAAE,GAAG,UAAU,UAAU,SAAS,YAAY,OAAO,GAAG;AAClK,UAAI,QAAS,UAASA,MAAK,iBAAiB,KAAK,wBAAwB,SAAS,UAAU,EAAE;AAG9F,YAAM,EAAE,yBAAyB,IAAI,MAAM,OAAO,gCAAsC;AACxF,YAAM,cAAc,yBAAyB,SAAS,UAAU;AAChE,UAAI,eAAe,YAAY,SAAS,GAAG;AACzC,YAAI,QAAQ,GAAG;AACb,eAAK,KAAK,OAAO,kBAAkB,WAAW;AAC9C,mBAASA,MAAK,qBAAqB,YAAY,MAAM,8BAA8B,OAAO,CAAC,CAAC,GAAG;AAAA,QACjG,WAAW,QAAQ,GAAG;AACpB,eAAK,KAAK,OAAO,eAAe,WAAW;AAC3C,mBAASA,MAAK,qBAAqB,YAAY,MAAM,4BAA4B,OAAO,CAAC,CAAC,GAAG;AAAA,QAC/F;AAAA,MACF;AAAA,IACF;AAEA,QAAI,UAAU,WAAW;AACvB,eAASA,MAAK,mCAAmC,KAAK,EAAE,cAAc;AACtE;AAAA,IACF;AAEA,QAAI,UAAU,UAAU;AACtB,YAAM,SAAS,MAAM,MAAM;AAC3B,WAAK,KAAK,mBAAmB,KAAK,QAAQ,YAAY,MAAM;AAC5D,eAASA,MAAK,sCAAsC,MAAM,EAAE;AAAA,IAC9D,OAAO;AACL,YAAM,eAAe,UAAU,EAAE,aAAa,IAAI,cAAc,KAAK,EAAE,GAAG,UAAU;AACpF,YAAM,WAAW,KAAK,KAAK,eAAe,mBAAmB,cAAc,UAAU;AACrF,YAAM,EAAE,SAAS,IAAI,KAAK,KAAK;AAC/B,YAAM,QAAQ,SAAS,YAAY,QAAQ;AAC3C,UAAI,MAAM,MAAM;AACd,cAAM,MAAM,KAAK,EAAE,KAAK,EAAE,QAAQ,cAAc,WAAW,OAAO,MAAM,GAAG,YAAY,UAAU,OAAO,KAAK,EAAE,GAAG,MAAM,QAAQ,WAAW,SAAS,YAAY,UAAU,YAAY,WAAW,KAAK,IAAI,GAAG,SAAS,OAAO,SAAS,MAAM,GAAG,SAAS,KAAK,CAAC;AAC9P,iBAASA,MAAK,mCAAmC,QAAQ,KAAK,MAAM,MAAM,MAAM,WAAW;AAAA,MAC7F,OAAO;AACL,YAAI;AACF,gBAAM,WAAW,MAAM,KAAK,KAAK,UAAU,WAAW,UAAU,QAAQ,QAAW,YAAY;AAC/F,mBAASA,MAAK,8CAA8C,MAAM,EAAE;AACpE,cAAI,UAAU;AACZ,kBAAM,EAAE,MAAM,eAAe,QAAQ,IAAI,cAAc,QAAQ;AAC/D,gBAAI,SAAS;AAAA,YAAe,WACnB,eAAe;AACtB,kBAAI;AAAE,sBAAM,KAAK,IAAI,mBAAmB,QAAQ,SAAS,YAAY,CAAC,EAAE,MAAM,SAAS,OAAO,cAAc,CAAC,CAAC;AAAA,cAAG,QAC3G;AAAE,sBAAM,KAAK,YAAY,OAAO,MAAM,GAAG,aAAa;AAAA,cAAG;AAAA,YACjE,WAAW,KAAK,KAAK,GAAG;AACtB,oBAAM,KAAK,YAAY,OAAO,MAAM,GAAG,KAAK,KAAK,CAAC;AAAA,YACpD;AAAA,UACF;AAAA,QACF,SAAS,KAAK;AACZ,mBAASA,MAAK,2CAA2C,MAAM,IAAI,GAAG;AAAA,QACxE;AAAA,MACF;AAAA,IACF;AAAA,EACF;AACF;",
|
|
6
|
+
"names": ["TAG", "TAG", "buttons", "TAG", "chatId"]
|
|
7
|
+
}
|
|
@@ -9,7 +9,8 @@
|
|
|
9
9
|
"kiro": { "transport": "acp", "cli": "kiro-cli" },
|
|
10
10
|
"gemini": { "transport": "acp", "cli": "gemini" },
|
|
11
11
|
"ollama": { "transport": "api", "endpoint": "http://localhost:11434/v1" },
|
|
12
|
-
"openrouter": { "transport": "api", "endpoint": "https://openrouter.ai/api/v1", "apiKeyEnv": "OPENROUTER_API_KEY" }
|
|
12
|
+
"openrouter": { "transport": "api", "endpoint": "https://openrouter.ai/api/v1", "apiKeyEnv": "OPENROUTER_API_KEY" },
|
|
13
|
+
"nvidia": { "transport": "api", "endpoint": "https://integrate.api.nvidia.com/v1", "apiKeyEnv": "NVIDIA_API_KEY" }
|
|
13
14
|
},
|
|
14
15
|
"transportDefaults": {
|
|
15
16
|
"tmux": { "session": "kiro-bridge", "captureDelaySec": 3, "maxWaitSec": 300 },
|
package/install-manifest.json
CHANGED
package/package.json
CHANGED
|
@@ -11,10 +11,6 @@ WorkingDirectory=/home/{{USER}}/.abtars
|
|
|
11
11
|
ExecStart=/home/{{USER}}/.abtars/scripts/watchdog.sh
|
|
12
12
|
Restart=always
|
|
13
13
|
RestartSec=5
|
|
14
|
-
<<<<<<< HEAD
|
|
15
|
-
Environment=NODE_ENV=production
|
|
16
|
-
=======
|
|
17
|
-
>>>>>>> dev
|
|
18
14
|
Environment=PATH={{PATH}}
|
|
19
15
|
Environment=ABTARS_HOME=/home/{{USER}}/.abtars
|
|
20
16
|
Environment=ABMIND_HOME=/home/{{USER}}/.abmind
|
package/scripts/abtars@.service
CHANGED
|
@@ -1,45 +1,30 @@
|
|
|
1
1
|
#!/bin/bash
|
|
2
2
|
# build-and-deploy.sh — Non-blocking build + deploy (#871)
|
|
3
3
|
# Spawned detached by /update deploy. Bridge stays responsive.
|
|
4
|
-
<<<<<<< HEAD
|
|
5
|
-
set -uo pipefail
|
|
6
|
-
|
|
7
|
-
ABTARS_SRC="${1:?Usage: build-and-deploy.sh <abtars-src> [abmind-src]}"
|
|
8
|
-
ABMIND_SRC="${2:-$(dirname "$ABTARS_SRC")/abmind}"
|
|
9
|
-
|
|
10
|
-
# Build abmind (if repo exists)
|
|
11
|
-
if [ -d "$ABMIND_SRC/.git" ]; then
|
|
12
|
-
cd "$ABMIND_SRC" || exit 1
|
|
13
|
-
if ! npm ci 2>/dev/null; then
|
|
14
|
-
rm -rf node_modules
|
|
15
|
-
exit 1
|
|
16
|
-
fi
|
|
17
|
-
npm run build --silent || exit 1
|
|
18
|
-
fi
|
|
19
|
-
|
|
20
|
-
# Build abtars
|
|
21
|
-
cd "$ABTARS_SRC" || exit 1
|
|
22
|
-
if ! npm ci 2>/dev/null; then
|
|
23
|
-
rm -rf node_modules
|
|
24
|
-
exit 1
|
|
25
|
-
fi
|
|
26
|
-
node esbuild.config.js || exit 1
|
|
27
|
-
|
|
28
|
-
# Deploy (health-verified, auto-rollback)
|
|
29
|
-
=======
|
|
30
4
|
set -euo pipefail
|
|
31
5
|
|
|
32
6
|
ABTARS_SRC="${1:?Usage: build-and-deploy.sh <abtars-src> [abmind-src]}"
|
|
33
7
|
ABMIND_SRC="${2:-$(dirname "$ABTARS_SRC")/abmind}"
|
|
34
8
|
LOG="$HOME/.abtars/logs/deploy-$(date +%F_%H%M%S).log"
|
|
9
|
+
STATE_FILE="$HOME/.abtars/deploy.state"
|
|
10
|
+
PHASE="init"
|
|
35
11
|
|
|
36
12
|
# Force development mode — npm ci skips devDeps under NODE_ENV=production
|
|
37
13
|
export NODE_ENV=development
|
|
38
14
|
|
|
15
|
+
cleanup() {
|
|
16
|
+
local exit_code=$?
|
|
17
|
+
if [ $exit_code -ne 0 ]; then
|
|
18
|
+
echo "{\"status\":\"failed\",\"completedAt\":\"$(date -u +%FT%TZ)\",\"error\":\"$PHASE failed (exit $exit_code)\",\"logFile\":\"$(basename "$LOG")\"}" > "$STATE_FILE"
|
|
19
|
+
fi
|
|
20
|
+
}
|
|
21
|
+
trap cleanup EXIT
|
|
22
|
+
|
|
39
23
|
exec > "$LOG" 2>&1
|
|
40
24
|
|
|
41
25
|
# Build abmind (if repo exists)
|
|
42
26
|
if [ -d "$ABMIND_SRC/.git" ]; then
|
|
27
|
+
PHASE="npm-ci-abmind"
|
|
43
28
|
echo "=== abmind: npm ci ==="
|
|
44
29
|
cd "$ABMIND_SRC"
|
|
45
30
|
if ! npm ci; then
|
|
@@ -47,11 +32,13 @@ if [ -d "$ABMIND_SRC/.git" ]; then
|
|
|
47
32
|
rm -rf node_modules
|
|
48
33
|
exit 1
|
|
49
34
|
fi
|
|
35
|
+
PHASE="build-abmind"
|
|
50
36
|
echo "=== abmind: build ==="
|
|
51
37
|
npm run build || { echo "FAILED: abmind build"; exit 1; }
|
|
52
38
|
fi
|
|
53
39
|
|
|
54
40
|
# Build abtars
|
|
41
|
+
PHASE="npm-ci-abtars"
|
|
55
42
|
echo "=== abtars: npm ci ==="
|
|
56
43
|
cd "$ABTARS_SRC"
|
|
57
44
|
if ! npm ci; then
|
|
@@ -59,10 +46,11 @@ if ! npm ci; then
|
|
|
59
46
|
rm -rf node_modules
|
|
60
47
|
exit 1
|
|
61
48
|
fi
|
|
49
|
+
PHASE="esbuild"
|
|
62
50
|
echo "=== abtars: esbuild ==="
|
|
63
51
|
node esbuild.config.js || { echo "FAILED: esbuild"; exit 1; }
|
|
64
52
|
|
|
65
53
|
# Deploy (health-verified, auto-rollback)
|
|
54
|
+
PHASE="deploy"
|
|
66
55
|
echo "=== deploying ==="
|
|
67
|
-
>>>>>>> dev
|
|
68
56
|
exec node bundle/abtars-cli.js update --from-local
|
|
@@ -1,19 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
import { createRequire as __bundleCreateRequire } from 'node:module'; import { fileURLToPath as __bundleFileURLToPath } from 'node:url'; import { dirname as __bundleDirname } from 'node:path'; const require = __bundleCreateRequire(import.meta.url); const __chunk_filename = __bundleFileURLToPath(import.meta.url); const __chunk_dirname = __bundleDirname(__chunk_filename);
|
|
3
|
-
import {
|
|
4
|
-
createAgentTransport,
|
|
5
|
-
createSubagentTransport
|
|
6
|
-
} from "./chunk-HAS5NEK7.js";
|
|
7
|
-
import "./chunk-PKHYCNTT.js";
|
|
8
|
-
import "./chunk-RTL7HO3N.js";
|
|
9
|
-
import "./chunk-MZWMYN4O.js";
|
|
10
|
-
import "./chunk-3OXQWII3.js";
|
|
11
|
-
import "./chunk-CYSGXNBY.js";
|
|
12
|
-
import "./chunk-GUTRAMK3.js";
|
|
13
|
-
import "./chunk-WW5F2DCO.js";
|
|
14
|
-
import "./chunk-7K2YZTLD.js";
|
|
15
|
-
export {
|
|
16
|
-
createAgentTransport,
|
|
17
|
-
createSubagentTransport
|
|
18
|
-
};
|
|
19
|
-
//# sourceMappingURL=agent-registry-5VL5KI6U.js.map
|