copilot-hub 0.1.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.
Files changed (128) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +215 -0
  3. package/apps/agent-engine/.env.example +41 -0
  4. package/apps/agent-engine/LICENSE +21 -0
  5. package/apps/agent-engine/README.md +57 -0
  6. package/apps/agent-engine/bot-registry.example.json +28 -0
  7. package/apps/agent-engine/capabilities/example/index.js +3 -0
  8. package/apps/agent-engine/capabilities/example/manifest.json +14 -0
  9. package/apps/agent-engine/dist/agent-worker.js +241 -0
  10. package/apps/agent-engine/dist/config.js +225 -0
  11. package/apps/agent-engine/dist/index.js +352 -0
  12. package/apps/agent-engine/dist/test/project-fingerprint.test.js +40 -0
  13. package/apps/agent-engine/dist/test/thread-id.test.js +12 -0
  14. package/apps/agent-engine/package.json +28 -0
  15. package/apps/control-plane/.env.example +25 -0
  16. package/apps/control-plane/README.md +35 -0
  17. package/apps/control-plane/bot-registry.example.json +40 -0
  18. package/apps/control-plane/capabilities/example/index.js +3 -0
  19. package/apps/control-plane/capabilities/example/manifest.json +14 -0
  20. package/apps/control-plane/dist/agent-worker.js +243 -0
  21. package/apps/control-plane/dist/channels/channel-factory.js +21 -0
  22. package/apps/control-plane/dist/channels/hub-ops-commands.js +752 -0
  23. package/apps/control-plane/dist/channels/telegram-channel.js +743 -0
  24. package/apps/control-plane/dist/channels/whatsapp-channel.js +35 -0
  25. package/apps/control-plane/dist/config.js +230 -0
  26. package/apps/control-plane/dist/copilot-hub.js +138 -0
  27. package/apps/control-plane/dist/index.js +349 -0
  28. package/apps/control-plane/dist/kernel/admin-contract.js +51 -0
  29. package/apps/control-plane/dist/test/project-fingerprint.test.js +40 -0
  30. package/apps/control-plane/dist/test/thread-id.test.js +12 -0
  31. package/apps/control-plane/package.json +27 -0
  32. package/package.json +89 -0
  33. package/packages/contracts/README.md +10 -0
  34. package/packages/contracts/dist/control-plane.d.ts +24 -0
  35. package/packages/contracts/dist/control-plane.js +37 -0
  36. package/packages/contracts/dist/control-plane.js.map +1 -0
  37. package/packages/contracts/dist/index.d.ts +1 -0
  38. package/packages/contracts/dist/index.js +2 -0
  39. package/packages/contracts/dist/index.js.map +1 -0
  40. package/packages/contracts/package.json +27 -0
  41. package/packages/core/README.md +33 -0
  42. package/packages/core/dist/agent-supervisor.d.ts +39 -0
  43. package/packages/core/dist/agent-supervisor.js +552 -0
  44. package/packages/core/dist/agent-supervisor.js.map +1 -0
  45. package/packages/core/dist/bot-manager.d.ts +66 -0
  46. package/packages/core/dist/bot-manager.js +333 -0
  47. package/packages/core/dist/bot-manager.js.map +1 -0
  48. package/packages/core/dist/bot-registry.d.ts +60 -0
  49. package/packages/core/dist/bot-registry.js +381 -0
  50. package/packages/core/dist/bot-registry.js.map +1 -0
  51. package/packages/core/dist/bot-runtime.d.ts +135 -0
  52. package/packages/core/dist/bot-runtime.js +349 -0
  53. package/packages/core/dist/bot-runtime.js.map +1 -0
  54. package/packages/core/dist/bridge-service.d.ts +39 -0
  55. package/packages/core/dist/bridge-service.js +272 -0
  56. package/packages/core/dist/bridge-service.js.map +1 -0
  57. package/packages/core/dist/capability-manager.d.ts +18 -0
  58. package/packages/core/dist/capability-manager.js +335 -0
  59. package/packages/core/dist/capability-manager.js.map +1 -0
  60. package/packages/core/dist/capability-scaffold.d.ts +26 -0
  61. package/packages/core/dist/capability-scaffold.js +118 -0
  62. package/packages/core/dist/capability-scaffold.js.map +1 -0
  63. package/packages/core/dist/channel-factory.d.ts +6 -0
  64. package/packages/core/dist/channel-factory.js +22 -0
  65. package/packages/core/dist/channel-factory.js.map +1 -0
  66. package/packages/core/dist/codex-app-client.d.ts +56 -0
  67. package/packages/core/dist/codex-app-client.js +762 -0
  68. package/packages/core/dist/codex-app-client.js.map +1 -0
  69. package/packages/core/dist/codex-provider.d.ts +31 -0
  70. package/packages/core/dist/codex-provider.js +64 -0
  71. package/packages/core/dist/codex-provider.js.map +1 -0
  72. package/packages/core/dist/control-permission.d.ts +19 -0
  73. package/packages/core/dist/control-permission.js +106 -0
  74. package/packages/core/dist/control-permission.js.map +1 -0
  75. package/packages/core/dist/control-plane-actions.d.ts +1 -0
  76. package/packages/core/dist/control-plane-actions.js +2 -0
  77. package/packages/core/dist/control-plane-actions.js.map +1 -0
  78. package/packages/core/dist/example-capability.d.ts +17 -0
  79. package/packages/core/dist/example-capability.js +22 -0
  80. package/packages/core/dist/example-capability.js.map +1 -0
  81. package/packages/core/dist/extension-contract.d.ts +22 -0
  82. package/packages/core/dist/extension-contract.js +28 -0
  83. package/packages/core/dist/extension-contract.js.map +1 -0
  84. package/packages/core/dist/index.d.ts +26 -0
  85. package/packages/core/dist/index.js +27 -0
  86. package/packages/core/dist/index.js.map +1 -0
  87. package/packages/core/dist/instance-lock.d.ts +9 -0
  88. package/packages/core/dist/instance-lock.js +74 -0
  89. package/packages/core/dist/instance-lock.js.map +1 -0
  90. package/packages/core/dist/kernel-control-plane.d.ts +16 -0
  91. package/packages/core/dist/kernel-control-plane.js +500 -0
  92. package/packages/core/dist/kernel-control-plane.js.map +1 -0
  93. package/packages/core/dist/kernel-version.d.ts +1 -0
  94. package/packages/core/dist/kernel-version.js +2 -0
  95. package/packages/core/dist/kernel-version.js.map +1 -0
  96. package/packages/core/dist/project-fingerprint.d.ts +11 -0
  97. package/packages/core/dist/project-fingerprint.js +33 -0
  98. package/packages/core/dist/project-fingerprint.js.map +1 -0
  99. package/packages/core/dist/provider-factory.d.ts +7 -0
  100. package/packages/core/dist/provider-factory.js +21 -0
  101. package/packages/core/dist/provider-factory.js.map +1 -0
  102. package/packages/core/dist/secret-store.d.ts +18 -0
  103. package/packages/core/dist/secret-store.js +110 -0
  104. package/packages/core/dist/secret-store.js.map +1 -0
  105. package/packages/core/dist/state-store.d.ts +50 -0
  106. package/packages/core/dist/state-store.js +324 -0
  107. package/packages/core/dist/state-store.js.map +1 -0
  108. package/packages/core/dist/telegram-channel.d.ts +27 -0
  109. package/packages/core/dist/telegram-channel.js +951 -0
  110. package/packages/core/dist/telegram-channel.js.map +1 -0
  111. package/packages/core/dist/thread-id.d.ts +1 -0
  112. package/packages/core/dist/thread-id.js +12 -0
  113. package/packages/core/dist/thread-id.js.map +1 -0
  114. package/packages/core/dist/whatsapp-channel.d.ts +26 -0
  115. package/packages/core/dist/whatsapp-channel.js +36 -0
  116. package/packages/core/dist/whatsapp-channel.js.map +1 -0
  117. package/packages/core/dist/workspace-paths.d.ts +5 -0
  118. package/packages/core/dist/workspace-paths.js +77 -0
  119. package/packages/core/dist/workspace-paths.js.map +1 -0
  120. package/packages/core/dist/workspace-policy.d.ts +30 -0
  121. package/packages/core/dist/workspace-policy.js +104 -0
  122. package/packages/core/dist/workspace-policy.js.map +1 -0
  123. package/packages/core/package.json +126 -0
  124. package/scripts/cli.mjs +537 -0
  125. package/scripts/configure.mjs +254 -0
  126. package/scripts/ensure-shared-build.mjs +96 -0
  127. package/scripts/run-node-tests.mjs +52 -0
  128. package/scripts/supervisor.mjs +332 -0
@@ -0,0 +1,349 @@
1
+ // @ts-nocheck
2
+ import path from "node:path";
3
+ import { ConversationEngine } from "./bridge-service.js";
4
+ import { createChannelAdapter } from "./channel-factory.js";
5
+ import { CapabilityManager } from "./capability-manager.js";
6
+ import { KERNEL_VERSION } from "./kernel-version.js";
7
+ import { createProjectFingerprint } from "./project-fingerprint.js";
8
+ import { JsonStateStore } from "./state-store.js";
9
+ import { createAssistantProvider } from "./provider-factory.js";
10
+ const DEFAULT_WEB_THREAD_SUFFIX = "web-main";
11
+ export class BotRuntime {
12
+ constructor({ botConfig, providerDefaults, turnActivityTimeoutMs, maxMessages, kernelControl = null, channelAdapterFactory = null, }) {
13
+ this.config = {
14
+ ...botConfig,
15
+ channels: Array.isArray(botConfig.channels) ? botConfig.channels : [],
16
+ capabilities: Array.isArray(botConfig.capabilities) ? botConfig.capabilities : [],
17
+ kernelAccess: botConfig.kernelAccess && typeof botConfig.kernelAccess === "object"
18
+ ? {
19
+ enabled: botConfig.kernelAccess.enabled === true,
20
+ allowedActions: Array.isArray(botConfig.kernelAccess.allowedActions)
21
+ ? [...botConfig.kernelAccess.allowedActions]
22
+ : [],
23
+ allowedChatIds: Array.isArray(botConfig.kernelAccess.allowedChatIds)
24
+ ? [...botConfig.kernelAccess.allowedChatIds]
25
+ : [],
26
+ }
27
+ : {
28
+ enabled: false,
29
+ allowedActions: [],
30
+ allowedChatIds: [],
31
+ },
32
+ };
33
+ this.providerDefaults = {
34
+ defaultKind: "codex",
35
+ ...(providerDefaults ?? {}),
36
+ };
37
+ this.kernelControl =
38
+ kernelControl && typeof kernelControl.request === "function" ? kernelControl : null;
39
+ this.channelAdapterFactory =
40
+ typeof channelAdapterFactory === "function" ? channelAdapterFactory : createChannelAdapter;
41
+ this.turnActivityTimeoutMs = turnActivityTimeoutMs;
42
+ this.maxMessages = maxMessages;
43
+ this.projectRoot = path.resolve(String(this.config.workspaceRoot));
44
+ this.config.workspaceRoot = this.projectRoot;
45
+ this.projectFingerprint = createProjectFingerprint({
46
+ runtimeId: this.config.id,
47
+ workspaceRoot: this.projectRoot,
48
+ providerKind: this.config.provider?.kind,
49
+ channels: this.config.channels,
50
+ });
51
+ this.store = null;
52
+ this.engine = null;
53
+ this.provider = null;
54
+ this.capabilityManager = null;
55
+ this.channels = [];
56
+ this.telegramRunning = false;
57
+ this.telegramError = null;
58
+ this.webPublicBaseUrl = "http://127.0.0.1:8787";
59
+ this.initPromise = null;
60
+ }
61
+ get id() {
62
+ return this.config.id;
63
+ }
64
+ get name() {
65
+ return this.config.name;
66
+ }
67
+ get providerKind() {
68
+ return String(this.config.provider?.kind ?? this.providerDefaults?.defaultKind ?? "codex");
69
+ }
70
+ get webThreadId() {
71
+ if (this.config.threadMode === "single") {
72
+ return this.config.sharedThreadId;
73
+ }
74
+ return `web:${this.config.id}:${DEFAULT_WEB_THREAD_SUFFIX}`;
75
+ }
76
+ setWebPublicBaseUrl(value) {
77
+ this.webPublicBaseUrl = String(value ?? "").trim() || this.webPublicBaseUrl;
78
+ }
79
+ async ensureInitialized() {
80
+ if (this.initPromise) {
81
+ await this.initPromise;
82
+ return;
83
+ }
84
+ this.initPromise = (async () => {
85
+ const stateFilePath = path.join(this.config.dataDir, "sessions.json");
86
+ this.store = new JsonStateStore(stateFilePath);
87
+ await this.store.init();
88
+ await this.#ensureStoreFingerprint();
89
+ this.provider = createAssistantProvider({
90
+ providerConfig: this.config.provider,
91
+ providerDefaults: this.providerDefaults,
92
+ workspaceRoot: this.projectRoot,
93
+ turnActivityTimeoutMs: this.turnActivityTimeoutMs,
94
+ });
95
+ this.engine = new ConversationEngine({
96
+ store: this.store,
97
+ assistantProvider: this.provider,
98
+ projectRoot: this.projectRoot,
99
+ turnActivityTimeoutMs: this.turnActivityTimeoutMs,
100
+ maxMessages: this.maxMessages,
101
+ onApprovalRequested: (approval) => this.onApprovalRequested(approval),
102
+ });
103
+ this.capabilityManager = new CapabilityManager({
104
+ runtimeId: this.id,
105
+ kernelVersion: KERNEL_VERSION,
106
+ workspaceRoot: this.projectRoot,
107
+ capabilityDefinitions: this.config.capabilities,
108
+ });
109
+ await this.capabilityManager.initialize();
110
+ await this.store.ensureThread(this.webThreadId);
111
+ const createAdapter = this.channelAdapterFactory;
112
+ this.channels = this.config.channels.map((channelConfig) => createAdapter({ channelConfig, runtime: this.#buildChannelRuntime() }));
113
+ this.#syncTelegramStatus();
114
+ })();
115
+ await this.initPromise;
116
+ }
117
+ async startChannels() {
118
+ await this.ensureInitialized();
119
+ for (const channel of this.channels) {
120
+ await channel.start();
121
+ }
122
+ this.#syncTelegramStatus();
123
+ return this.getStatus();
124
+ }
125
+ async stopChannels() {
126
+ await this.ensureInitialized();
127
+ for (const channel of this.channels) {
128
+ await channel.stop();
129
+ }
130
+ this.#syncTelegramStatus();
131
+ return this.getStatus();
132
+ }
133
+ async shutdown() {
134
+ if (this.channels.length > 0) {
135
+ for (const channel of this.channels) {
136
+ await channel.shutdown();
137
+ }
138
+ }
139
+ if (this.engine) {
140
+ await this.engine.shutdown();
141
+ }
142
+ if (this.capabilityManager) {
143
+ await this.capabilityManager.shutdown();
144
+ }
145
+ this.channels = [];
146
+ this.engine = null;
147
+ this.provider = null;
148
+ this.capabilityManager = null;
149
+ this.store = null;
150
+ this.initPromise = null;
151
+ this.telegramRunning = false;
152
+ this.telegramError = null;
153
+ }
154
+ async setProjectRoot(projectRoot) {
155
+ const nextRoot = path.resolve(String(projectRoot));
156
+ if (nextRoot === this.projectRoot) {
157
+ return;
158
+ }
159
+ const wasRunning = this.channels.some((channel) => channel.getStatus().running);
160
+ await this.shutdown();
161
+ this.projectRoot = nextRoot;
162
+ this.config.workspaceRoot = nextRoot;
163
+ this.projectFingerprint = createProjectFingerprint({
164
+ runtimeId: this.id,
165
+ workspaceRoot: this.projectRoot,
166
+ providerKind: this.providerKind,
167
+ channels: this.config.channels,
168
+ });
169
+ await this.ensureInitialized();
170
+ if (wasRunning) {
171
+ await this.startChannels();
172
+ }
173
+ }
174
+ async resetWebThread() {
175
+ await this.ensureInitialized();
176
+ return this.engine.resetThread(this.webThreadId);
177
+ }
178
+ async listPendingApprovals(threadId) {
179
+ await this.ensureInitialized();
180
+ return this.engine.listPendingApprovals(threadId);
181
+ }
182
+ async resolvePendingApproval({ threadId, approvalId, decision }) {
183
+ await this.ensureInitialized();
184
+ return this.engine.resolvePendingApproval({ threadId, approvalId, decision });
185
+ }
186
+ async interruptThread(threadId) {
187
+ await this.ensureInitialized();
188
+ return this.engine.interruptThread(threadId);
189
+ }
190
+ async resolveThreadIdForChannel({ channelKind, channelId, externalUserId }) {
191
+ await this.ensureInitialized();
192
+ if (this.config.threadMode === "single") {
193
+ const threadId = this.config.sharedThreadId;
194
+ await this.store.ensureThread(threadId);
195
+ return threadId;
196
+ }
197
+ const key = `${String(channelKind)}:${String(channelId)}:${String(externalUserId)}`;
198
+ return this.store.getOrCreateThreadIdForChannelUser(key);
199
+ }
200
+ async getThread(threadId) {
201
+ await this.ensureInitialized();
202
+ return this.engine.getThread(threadId);
203
+ }
204
+ async resetThread(threadId) {
205
+ await this.ensureInitialized();
206
+ return this.engine.resetThread(threadId);
207
+ }
208
+ async sendTurn(payload) {
209
+ await this.ensureInitialized();
210
+ const input = await this.capabilityManager.transformTurnInput({
211
+ ...payload,
212
+ metadata: payload?.metadata && typeof payload.metadata === "object" ? payload.metadata : {},
213
+ });
214
+ if (!String(input.prompt ?? "").trim()) {
215
+ throw new Error("Capability pipeline produced an empty prompt.");
216
+ }
217
+ const result = await this.engine.sendTurn(input);
218
+ await this.capabilityManager.runHook("onTurnResult", {
219
+ threadId: input.threadId,
220
+ prompt: input.prompt,
221
+ source: input.source,
222
+ metadata: input.metadata,
223
+ result,
224
+ });
225
+ return result;
226
+ }
227
+ getStatus() {
228
+ this.#syncTelegramStatus();
229
+ return {
230
+ id: this.config.id,
231
+ name: this.config.name,
232
+ enabled: this.config.enabled,
233
+ autoStart: this.config.autoStart,
234
+ threadMode: this.config.threadMode,
235
+ sharedThreadId: this.config.sharedThreadId,
236
+ providerKind: this.providerKind,
237
+ kernelVersion: KERNEL_VERSION,
238
+ webThreadId: this.webThreadId,
239
+ running: this.channels.some((channel) => channel.getStatus().running),
240
+ telegramRunning: this.telegramRunning,
241
+ telegramError: this.telegramError,
242
+ workspaceRoot: this.projectRoot,
243
+ dataDir: this.config.dataDir,
244
+ kernelAccess: {
245
+ enabled: this.config.kernelAccess?.enabled === true,
246
+ allowedActions: Array.isArray(this.config.kernelAccess?.allowedActions)
247
+ ? [...this.config.kernelAccess.allowedActions]
248
+ : [],
249
+ allowedChatIds: Array.isArray(this.config.kernelAccess?.allowedChatIds)
250
+ ? [...this.config.kernelAccess.allowedChatIds]
251
+ : [],
252
+ },
253
+ capabilities: this.capabilityManager ? this.capabilityManager.getStatus() : [],
254
+ channels: this.channels.map((channel) => channel.getStatus()),
255
+ };
256
+ }
257
+ async reloadCapabilities(nextDefinitions = null) {
258
+ await this.ensureInitialized();
259
+ if (Array.isArray(nextDefinitions)) {
260
+ this.config.capabilities = nextDefinitions;
261
+ }
262
+ await this.capabilityManager.reload(this.config.capabilities);
263
+ return this.getStatus();
264
+ }
265
+ async onApprovalRequested(approval) {
266
+ if (this.capabilityManager) {
267
+ await this.capabilityManager.notifyApprovalRequested(approval);
268
+ }
269
+ const source = String(approval?.source ?? "");
270
+ if (source !== "telegram") {
271
+ return;
272
+ }
273
+ const requestedChannelId = String(approval?.metadata?.channelId ?? "").trim();
274
+ const telegramChannels = this.channels.filter((channel) => channel.kind === "telegram");
275
+ const targets = requestedChannelId
276
+ ? telegramChannels.filter((channel) => channel.id === requestedChannelId)
277
+ : telegramChannels;
278
+ if (targets.length === 0) {
279
+ return;
280
+ }
281
+ for (const channel of targets) {
282
+ await channel.notifyApproval(approval);
283
+ }
284
+ }
285
+ buildWebBotUrl() {
286
+ try {
287
+ const url = new URL(this.webPublicBaseUrl);
288
+ url.searchParams.set("bot", this.id);
289
+ return url.toString();
290
+ }
291
+ catch {
292
+ const base = this.webPublicBaseUrl.replace(/\/+$/, "");
293
+ return `${base}/?bot=${encodeURIComponent(this.id)}`;
294
+ }
295
+ }
296
+ async #ensureStoreFingerprint() {
297
+ if (!this.store) {
298
+ return;
299
+ }
300
+ const result = await this.store.ensureFingerprint(this.projectFingerprint);
301
+ if (result.reset) {
302
+ console.log(`[${this.id}] Session store reset due project fingerprint change (${result.previousFingerprint} -> ${this.projectFingerprint}).`);
303
+ }
304
+ }
305
+ #buildChannelRuntime() {
306
+ return {
307
+ runtimeId: this.id,
308
+ runtimeName: this.name,
309
+ getWorkspaceRoot: () => this.projectRoot,
310
+ buildWebBotUrl: () => this.buildWebBotUrl(),
311
+ isKernelControlEnabled: () => this.isKernelControlEnabled(),
312
+ executeKernelAction: (payload) => this.executeKernelAction(payload),
313
+ resolveThreadIdForChannel: (payload) => this.resolveThreadIdForChannel(payload),
314
+ getThread: (threadId) => this.getThread(threadId),
315
+ resetThread: (threadId) => this.resetThread(threadId),
316
+ interruptThread: (threadId) => this.interruptThread(threadId),
317
+ listPendingApprovals: (threadId) => this.listPendingApprovals(threadId),
318
+ resolvePendingApproval: (payload) => this.resolvePendingApproval(payload),
319
+ sendTurn: (payload) => this.sendTurn(payload),
320
+ };
321
+ }
322
+ isKernelControlEnabled() {
323
+ return this.config.kernelAccess?.enabled === true && this.kernelControl !== null;
324
+ }
325
+ async executeKernelAction({ action, payload, context }) {
326
+ if (!this.isKernelControlEnabled()) {
327
+ throw new Error(`Kernel control is disabled for '${this.id}'.`);
328
+ }
329
+ return this.kernelControl.request({
330
+ action,
331
+ payload: payload ?? {},
332
+ context: context ?? { source: "internal", metadata: {} },
333
+ });
334
+ }
335
+ #syncTelegramStatus() {
336
+ if (!this.channels || this.channels.length === 0) {
337
+ this.telegramRunning = false;
338
+ this.telegramError = null;
339
+ return;
340
+ }
341
+ const telegram = this.channels
342
+ .filter((channel) => channel.kind === "telegram")
343
+ .map((channel) => channel.getStatus());
344
+ this.telegramRunning = telegram.some((entry) => entry.running);
345
+ const errors = telegram.map((entry) => entry.error).filter(Boolean);
346
+ this.telegramError = errors.length > 0 ? errors[0] : null;
347
+ }
348
+ }
349
+ //# sourceMappingURL=bot-runtime.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"bot-runtime.js","sourceRoot":"","sources":["../src/bot-runtime.ts"],"names":[],"mappings":"AAAA,cAAc;AACd,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,EAAE,kBAAkB,EAAE,MAAM,qBAAqB,CAAC;AACzD,OAAO,EAAE,oBAAoB,EAAE,MAAM,sBAAsB,CAAC;AAC5D,OAAO,EAAE,iBAAiB,EAAE,MAAM,yBAAyB,CAAC;AAC5D,OAAO,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC;AACrD,OAAO,EAAE,wBAAwB,EAAE,MAAM,0BAA0B,CAAC;AACpE,OAAO,EAAE,cAAc,EAAE,MAAM,kBAAkB,CAAC;AAClD,OAAO,EAAE,uBAAuB,EAAE,MAAM,uBAAuB,CAAC;AAEhE,MAAM,yBAAyB,GAAG,UAAU,CAAC;AAE7C,MAAM,OAAO,UAAU;IACrB,YAAY,EACV,SAAS,EACT,gBAAgB,EAChB,qBAAqB,EACrB,WAAW,EACX,aAAa,GAAG,IAAI,EACpB,qBAAqB,GAAG,IAAI,GAC7B;QACC,IAAI,CAAC,MAAM,GAAG;YACZ,GAAG,SAAS;YACZ,QAAQ,EAAE,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE;YACrE,YAAY,EAAE,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC,CAAC,EAAE;YACjF,YAAY,EACV,SAAS,CAAC,YAAY,IAAI,OAAO,SAAS,CAAC,YAAY,KAAK,QAAQ;gBAClE,CAAC,CAAC;oBACE,OAAO,EAAE,SAAS,CAAC,YAAY,CAAC,OAAO,KAAK,IAAI;oBAChD,cAAc,EAAE,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,YAAY,CAAC,cAAc,CAAC;wBAClE,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,YAAY,CAAC,cAAc,CAAC;wBAC5C,CAAC,CAAC,EAAE;oBACN,cAAc,EAAE,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,YAAY,CAAC,cAAc,CAAC;wBAClE,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,YAAY,CAAC,cAAc,CAAC;wBAC5C,CAAC,CAAC,EAAE;iBACP;gBACH,CAAC,CAAC;oBACE,OAAO,EAAE,KAAK;oBACd,cAAc,EAAE,EAAE;oBAClB,cAAc,EAAE,EAAE;iBACnB;SACR,CAAC;QACF,IAAI,CAAC,gBAAgB,GAAG;YACtB,WAAW,EAAE,OAAO;YACpB,GAAG,CAAC,gBAAgB,IAAI,EAAE,CAAC;SAC5B,CAAC;QACF,IAAI,CAAC,aAAa;YAChB,aAAa,IAAI,OAAO,aAAa,CAAC,OAAO,KAAK,UAAU,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC;QACtF,IAAI,CAAC,qBAAqB;YACxB,OAAO,qBAAqB,KAAK,UAAU,CAAC,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,oBAAoB,CAAC;QAC7F,IAAI,CAAC,qBAAqB,GAAG,qBAAqB,CAAC;QACnD,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC;QAC/B,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC,CAAC;QACnE,IAAI,CAAC,MAAM,CAAC,aAAa,GAAG,IAAI,CAAC,WAAW,CAAC;QAC7C,IAAI,CAAC,kBAAkB,GAAG,wBAAwB,CAAC;YACjD,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,EAAE;YACzB,aAAa,EAAE,IAAI,CAAC,WAAW;YAC/B,YAAY,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,IAAI;YACxC,QAAQ,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ;SAC/B,CAAC,CAAC;QAEH,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC;QAClB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;QACnB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;QACrB,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;QAC9B,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;QACnB,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;QAC7B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;QAC1B,IAAI,CAAC,gBAAgB,GAAG,uBAAuB,CAAC;QAChD,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;IAC1B,CAAC;IAED,IAAI,EAAE;QACJ,OAAO,IAAI,CAAC,MAAM,CAAC,EAAE,CAAC;IACxB,CAAC;IAED,IAAI,IAAI;QACN,OAAO,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC;IAC1B,CAAC;IAED,IAAI,YAAY;QACd,OAAO,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,IAAI,IAAI,IAAI,CAAC,gBAAgB,EAAE,WAAW,IAAI,OAAO,CAAC,CAAC;IAC7F,CAAC;IAED,IAAI,WAAW;QACb,IAAI,IAAI,CAAC,MAAM,CAAC,UAAU,KAAK,QAAQ,EAAE,CAAC;YACxC,OAAO,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC;QACpC,CAAC;QACD,OAAO,OAAO,IAAI,CAAC,MAAM,CAAC,EAAE,IAAI,yBAAyB,EAAE,CAAC;IAC9D,CAAC;IAED,mBAAmB,CAAC,KAAK;QACvB,IAAI,CAAC,gBAAgB,GAAG,MAAM,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,IAAI,IAAI,CAAC,gBAAgB,CAAC;IAC9E,CAAC;IAED,KAAK,CAAC,iBAAiB;QACrB,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;YACrB,MAAM,IAAI,CAAC,WAAW,CAAC;YACvB,OAAO;QACT,CAAC;QAED,IAAI,CAAC,WAAW,GAAG,CAAC,KAAK,IAAI,EAAE;YAC7B,MAAM,aAAa,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,eAAe,CAAC,CAAC;YACtE,IAAI,CAAC,KAAK,GAAG,IAAI,cAAc,CAAC,aAAa,CAAC,CAAC;YAC/C,MAAM,IAAI,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;YACxB,MAAM,IAAI,CAAC,uBAAuB,EAAE,CAAC;YAErC,IAAI,CAAC,QAAQ,GAAG,uBAAuB,CAAC;gBACtC,cAAc,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ;gBACpC,gBAAgB,EAAE,IAAI,CAAC,gBAAgB;gBACvC,aAAa,EAAE,IAAI,CAAC,WAAW;gBAC/B,qBAAqB,EAAE,IAAI,CAAC,qBAAqB;aAClD,CAAC,CAAC;YAEH,IAAI,CAAC,MAAM,GAAG,IAAI,kBAAkB,CAAC;gBACnC,KAAK,EAAE,IAAI,CAAC,KAAK;gBACjB,iBAAiB,EAAE,IAAI,CAAC,QAAQ;gBAChC,WAAW,EAAE,IAAI,CAAC,WAAW;gBAC7B,qBAAqB,EAAE,IAAI,CAAC,qBAAqB;gBACjD,WAAW,EAAE,IAAI,CAAC,WAAW;gBAC7B,mBAAmB,EAAE,CAAC,QAAQ,EAAE,EAAE,CAAC,IAAI,CAAC,mBAAmB,CAAC,QAAQ,CAAC;aACtE,CAAC,CAAC;YAEH,IAAI,CAAC,iBAAiB,GAAG,IAAI,iBAAiB,CAAC;gBAC7C,SAAS,EAAE,IAAI,CAAC,EAAE;gBAClB,aAAa,EAAE,cAAc;gBAC7B,aAAa,EAAE,IAAI,CAAC,WAAW;gBAC/B,qBAAqB,EAAE,IAAI,CAAC,MAAM,CAAC,YAAY;aAChD,CAAC,CAAC;YACH,MAAM,IAAI,CAAC,iBAAiB,CAAC,UAAU,EAAE,CAAC;YAE1C,MAAM,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YAChD,MAAM,aAAa,GAAG,IAAI,CAAC,qBAAqB,CAAC;YACjD,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,aAAa,EAAE,EAAE,CACzD,aAAa,CAAC,EAAE,aAAa,EAAE,OAAO,EAAE,IAAI,CAAC,oBAAoB,EAAE,EAAE,CAAC,CACvE,CAAC;YACF,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC7B,CAAC,CAAC,EAAE,CAAC;QAEL,MAAM,IAAI,CAAC,WAAW,CAAC;IACzB,CAAC;IAED,KAAK,CAAC,aAAa;QACjB,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,KAAK,MAAM,OAAO,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YACpC,MAAM,OAAO,CAAC,KAAK,EAAE,CAAC;QACxB,CAAC;QACD,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC3B,OAAO,IAAI,CAAC,SAAS,EAAE,CAAC;IAC1B,CAAC;IAED,KAAK,CAAC,YAAY;QAChB,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,KAAK,MAAM,OAAO,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YACpC,MAAM,OAAO,CAAC,IAAI,EAAE,CAAC;QACvB,CAAC;QACD,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC3B,OAAO,IAAI,CAAC,SAAS,EAAE,CAAC;IAC1B,CAAC;IAED,KAAK,CAAC,QAAQ;QACZ,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC7B,KAAK,MAAM,OAAO,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;gBACpC,MAAM,OAAO,CAAC,QAAQ,EAAE,CAAC;YAC3B,CAAC;QACH,CAAC;QACD,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,MAAM,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC;QAC/B,CAAC;QACD,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,MAAM,IAAI,CAAC,iBAAiB,CAAC,QAAQ,EAAE,CAAC;QAC1C,CAAC;QACD,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;QACnB,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC;QACnB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;QACrB,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC;QAC9B,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC;QAClB,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;QACxB,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;QAC7B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;IAC5B,CAAC;IAED,KAAK,CAAC,cAAc,CAAC,WAAW;QAC9B,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC;QACnD,IAAI,QAAQ,KAAK,IAAI,CAAC,WAAW,EAAE,CAAC;YAClC,OAAO;QACT,CAAC;QAED,MAAM,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,SAAS,EAAE,CAAC,OAAO,CAAC,CAAC;QAChF,MAAM,IAAI,CAAC,QAAQ,EAAE,CAAC;QACtB,IAAI,CAAC,WAAW,GAAG,QAAQ,CAAC;QAC5B,IAAI,CAAC,MAAM,CAAC,aAAa,GAAG,QAAQ,CAAC;QACrC,IAAI,CAAC,kBAAkB,GAAG,wBAAwB,CAAC;YACjD,SAAS,EAAE,IAAI,CAAC,EAAE;YAClB,aAAa,EAAE,IAAI,CAAC,WAAW;YAC/B,YAAY,EAAE,IAAI,CAAC,YAAY;YAC/B,QAAQ,EAAE,IAAI,CAAC,MAAM,CAAC,QAAQ;SAC/B,CAAC,CAAC;QAEH,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,IAAI,UAAU,EAAE,CAAC;YACf,MAAM,IAAI,CAAC,aAAa,EAAE,CAAC;QAC7B,CAAC;IACH,CAAC;IAED,KAAK,CAAC,cAAc;QAClB,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,OAAO,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IACnD,CAAC;IAED,KAAK,CAAC,oBAAoB,CAAC,QAAQ;QACjC,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,OAAO,IAAI,CAAC,MAAM,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC;IACpD,CAAC;IAED,KAAK,CAAC,sBAAsB,CAAC,EAAE,QAAQ,EAAE,UAAU,EAAE,QAAQ,EAAE;QAC7D,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,OAAO,IAAI,CAAC,MAAM,CAAC,sBAAsB,CAAC,EAAE,QAAQ,EAAE,UAAU,EAAE,QAAQ,EAAE,CAAC,CAAC;IAChF,CAAC;IAED,KAAK,CAAC,eAAe,CAAC,QAAQ;QAC5B,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,OAAO,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,QAAQ,CAAC,CAAC;IAC/C,CAAC;IAED,KAAK,CAAC,yBAAyB,CAAC,EAAE,WAAW,EAAE,SAAS,EAAE,cAAc,EAAE;QACxE,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,IAAI,IAAI,CAAC,MAAM,CAAC,UAAU,KAAK,QAAQ,EAAE,CAAC;YACxC,MAAM,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC;YAC5C,MAAM,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC;YACxC,OAAO,QAAQ,CAAC;QAClB,CAAC;QAED,MAAM,GAAG,GAAG,GAAG,MAAM,CAAC,WAAW,CAAC,IAAI,MAAM,CAAC,SAAS,CAAC,IAAI,MAAM,CAAC,cAAc,CAAC,EAAE,CAAC;QACpF,OAAO,IAAI,CAAC,KAAK,CAAC,iCAAiC,CAAC,GAAG,CAAC,CAAC;IAC3D,CAAC;IAED,KAAK,CAAC,SAAS,CAAC,QAAQ;QACtB,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,OAAO,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;IACzC,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,QAAQ;QACxB,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,OAAO,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;IAC3C,CAAC;IAED,KAAK,CAAC,QAAQ,CAAC,OAAO;QACpB,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,kBAAkB,CAAC;YAC5D,GAAG,OAAO;YACV,QAAQ,EAAE,OAAO,EAAE,QAAQ,IAAI,OAAO,OAAO,CAAC,QAAQ,KAAK,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE;SAC5F,CAAC,CAAC;QAEH,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC;YACvC,MAAM,IAAI,KAAK,CAAC,+CAA+C,CAAC,CAAC;QACnE,CAAC;QAED,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QACjD,MAAM,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,cAAc,EAAE;YACnD,QAAQ,EAAE,KAAK,CAAC,QAAQ;YACxB,MAAM,EAAE,KAAK,CAAC,MAAM;YACpB,MAAM,EAAE,KAAK,CAAC,MAAM;YACpB,QAAQ,EAAE,KAAK,CAAC,QAAQ;YACxB,MAAM;SACP,CAAC,CAAC;QACH,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,SAAS;QACP,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC3B,OAAO;YACL,EAAE,EAAE,IAAI,CAAC,MAAM,CAAC,EAAE;YAClB,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,IAAI;YACtB,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO;YAC5B,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,SAAS;YAChC,UAAU,EAAE,IAAI,CAAC,MAAM,CAAC,UAAU;YAClC,cAAc,EAAE,IAAI,CAAC,MAAM,CAAC,cAAc;YAC1C,YAAY,EAAE,IAAI,CAAC,YAAY;YAC/B,aAAa,EAAE,cAAc;YAC7B,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,OAAO,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,SAAS,EAAE,CAAC,OAAO,CAAC;YACrE,eAAe,EAAE,IAAI,CAAC,eAAe;YACrC,aAAa,EAAE,IAAI,CAAC,aAAa;YACjC,aAAa,EAAE,IAAI,CAAC,WAAW;YAC/B,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO;YAC5B,YAAY,EAAE;gBACZ,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE,OAAO,KAAK,IAAI;gBACnD,cAAc,EAAE,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE,cAAc,CAAC;oBACrE,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,cAAc,CAAC;oBAC9C,CAAC,CAAC,EAAE;gBACN,cAAc,EAAE,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE,cAAc,CAAC;oBACrE,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,cAAc,CAAC;oBAC9C,CAAC,CAAC,EAAE;aACP;YACD,YAAY,EAAE,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC,EAAE;YAC9E,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,SAAS,EAAE,CAAC;SAC9D,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,kBAAkB,CAAC,eAAe,GAAG,IAAI;QAC7C,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC/B,IAAI,KAAK,CAAC,OAAO,CAAC,eAAe,CAAC,EAAE,CAAC;YACnC,IAAI,CAAC,MAAM,CAAC,YAAY,GAAG,eAAe,CAAC;QAC7C,CAAC;QACD,MAAM,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;QAC9D,OAAO,IAAI,CAAC,SAAS,EAAE,CAAC;IAC1B,CAAC;IAED,KAAK,CAAC,mBAAmB,CAAC,QAAQ;QAChC,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,MAAM,IAAI,CAAC,iBAAiB,CAAC,uBAAuB,CAAC,QAAQ,CAAC,CAAC;QACjE,CAAC;QAED,MAAM,MAAM,GAAG,MAAM,CAAC,QAAQ,EAAE,MAAM,IAAI,EAAE,CAAC,CAAC;QAC9C,IAAI,MAAM,KAAK,UAAU,EAAE,CAAC;YAC1B,OAAO;QACT,CAAC;QAED,MAAM,kBAAkB,GAAG,MAAM,CAAC,QAAQ,EAAE,QAAQ,EAAE,SAAS,IAAI,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;QAC9E,MAAM,gBAAgB,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,KAAK,UAAU,CAAC,CAAC;QACxF,MAAM,OAAO,GAAG,kBAAkB;YAChC,CAAC,CAAC,gBAAgB,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,EAAE,KAAK,kBAAkB,CAAC;YACzE,CAAC,CAAC,gBAAgB,CAAC;QAErB,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACzB,OAAO;QACT,CAAC;QAED,KAAK,MAAM,OAAO,IAAI,OAAO,EAAE,CAAC;YAC9B,MAAM,OAAO,CAAC,cAAc,CAAC,QAAQ,CAAC,CAAC;QACzC,CAAC;IACH,CAAC;IAED,cAAc;QACZ,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;YAC3C,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,KAAK,EAAE,IAAI,CAAC,EAAE,CAAC,CAAC;YACrC,OAAO,GAAG,CAAC,QAAQ,EAAE,CAAC;QACxB,CAAC;QAAC,MAAM,CAAC;YACP,MAAM,IAAI,GAAG,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;YACvD,OAAO,GAAG,IAAI,SAAS,kBAAkB,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,CAAC;QACvD,CAAC;IACH,CAAC;IAED,KAAK,CAAC,uBAAuB;QAC3B,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC;YAChB,OAAO;QACT,CAAC;QAED,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC,IAAI,CAAC,kBAAkB,CAAC,CAAC;QAC3E,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;YACjB,OAAO,CAAC,GAAG,CACT,IAAI,IAAI,CAAC,EAAE,yDAAyD,MAAM,CAAC,mBAAmB,OAAO,IAAI,CAAC,kBAAkB,IAAI,CACjI,CAAC;QACJ,CAAC;IACH,CAAC;IAED,oBAAoB;QAClB,OAAO;YACL,SAAS,EAAE,IAAI,CAAC,EAAE;YAClB,WAAW,EAAE,IAAI,CAAC,IAAI;YACtB,gBAAgB,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW;YACxC,cAAc,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,cAAc,EAAE;YAC3C,sBAAsB,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,sBAAsB,EAAE;YAC3D,mBAAmB,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,IAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC;YACnE,yBAAyB,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,IAAI,CAAC,yBAAyB,CAAC,OAAO,CAAC;YAC/E,SAAS,EAAE,CAAC,QAAQ,EAAE,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC;YACjD,WAAW,EAAE,CAAC,QAAQ,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC;YACrD,eAAe,EAAE,CAAC,QAAQ,EAAE,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,QAAQ,CAAC;YAC7D,oBAAoB,EAAE,CAAC,QAAQ,EAAE,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,QAAQ,CAAC;YACvE,sBAAsB,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,IAAI,CAAC,sBAAsB,CAAC,OAAO,CAAC;YACzE,QAAQ,EAAE,CAAC,OAAO,EAAE,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC;SAC9C,CAAC;IACJ,CAAC;IAED,sBAAsB;QACpB,OAAO,IAAI,CAAC,MAAM,CAAC,YAAY,EAAE,OAAO,KAAK,IAAI,IAAI,IAAI,CAAC,aAAa,KAAK,IAAI,CAAC;IACnF,CAAC;IAED,KAAK,CAAC,mBAAmB,CAAC,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE;QACpD,IAAI,CAAC,IAAI,CAAC,sBAAsB,EAAE,EAAE,CAAC;YACnC,MAAM,IAAI,KAAK,CAAC,mCAAmC,IAAI,CAAC,EAAE,IAAI,CAAC,CAAC;QAClE,CAAC;QACD,OAAO,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC;YAChC,MAAM;YACN,OAAO,EAAE,OAAO,IAAI,EAAE;YACtB,OAAO,EAAE,OAAO,IAAI,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,EAAE,EAAE;SACzD,CAAC,CAAC;IACL,CAAC;IAED,mBAAmB;QACjB,IAAI,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACjD,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;YAC7B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC;YAC1B,OAAO;QACT,CAAC;QAED,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ;aAC3B,MAAM,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,KAAK,UAAU,CAAC;aAChD,GAAG,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,OAAO,CAAC,SAAS,EAAE,CAAC,CAAC;QACzC,IAAI,CAAC,eAAe,GAAG,QAAQ,CAAC,IAAI,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QAC/D,MAAM,MAAM,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;QACpE,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IAC5D,CAAC;CACF"}
@@ -0,0 +1,39 @@
1
+ export declare class ConversationEngine {
2
+ #private;
3
+ constructor({ store, assistantProvider, projectRoot, turnActivityTimeoutMs, maxMessages, onApprovalRequested, }: {
4
+ store: any;
5
+ assistantProvider: any;
6
+ projectRoot: any;
7
+ turnActivityTimeoutMs: any;
8
+ maxMessages?: number | undefined;
9
+ onApprovalRequested?: null | undefined;
10
+ });
11
+ setProjectRoot(projectRoot: any): void;
12
+ sendTurn({ threadId, prompt, source, metadata, inputItems }: {
13
+ threadId: any;
14
+ prompt: any;
15
+ source: any;
16
+ metadata?: {} | undefined;
17
+ inputItems?: never[] | undefined;
18
+ }): Promise<any>;
19
+ getThread(threadId: any): Promise<{
20
+ threadId: string;
21
+ thread: any;
22
+ }>;
23
+ getMessages(threadId: any, limit: any): Promise<{
24
+ threadId: string;
25
+ messages: any;
26
+ }>;
27
+ resetThread(threadId: any): Promise<{
28
+ threadId: string;
29
+ thread: any;
30
+ }>;
31
+ listPendingApprovals(threadId?: null): any[];
32
+ resolvePendingApproval({ threadId, approvalId, decision }: {
33
+ threadId: any;
34
+ approvalId: any;
35
+ decision: any;
36
+ }): Promise<any>;
37
+ interruptThread(threadId: any): Promise<any>;
38
+ shutdown(): Promise<void>;
39
+ }
@@ -0,0 +1,272 @@
1
+ // @ts-nocheck
2
+ import { normalizeThreadId } from "./thread-id.js";
3
+ const DEFAULT_MAX_MESSAGES = 200;
4
+ export class ConversationEngine {
5
+ constructor({ store, assistantProvider, projectRoot, turnActivityTimeoutMs, maxMessages = DEFAULT_MAX_MESSAGES, onApprovalRequested = null, }) {
6
+ this.store = store;
7
+ this.assistantProvider = assistantProvider;
8
+ this.projectRoot = projectRoot;
9
+ this.turnActivityTimeoutMs = Number.parseInt(String(turnActivityTimeoutMs), 10) || 3600000;
10
+ this.maxMessages = maxMessages;
11
+ this.queueByThread = new Map();
12
+ this.threadByProviderSession = new Map();
13
+ this.turnContextByThread = new Map();
14
+ this.pendingApprovalsByThread = new Map();
15
+ this.onApprovalRequested =
16
+ typeof onApprovalRequested === "function" ? onApprovalRequested : null;
17
+ this.assistantProvider.on("approvalRequested", (approval) => {
18
+ void this.#handleApprovalRequested(approval).catch((error) => {
19
+ console.error(`Approval callback failed: ${sanitizeError(error)}`);
20
+ });
21
+ });
22
+ }
23
+ setProjectRoot(projectRoot) {
24
+ this.projectRoot = projectRoot;
25
+ this.assistantProvider.setWorkspaceRoot(projectRoot);
26
+ }
27
+ async sendTurn({ threadId, prompt, source, metadata = {}, inputItems = [] }) {
28
+ const normalizedThreadId = normalizeThreadId(threadId);
29
+ const text = String(prompt ?? "").trim();
30
+ const normalizedInputItems = Array.isArray(inputItems) ? [...inputItems] : [];
31
+ const normalizedSource = normalizeSource(source);
32
+ if (!text && normalizedInputItems.length === 0) {
33
+ throw new Error("Message cannot be empty.");
34
+ }
35
+ return this.#queue(normalizedThreadId, async () => {
36
+ const previous = (await this.store.getThread(normalizedThreadId)) ??
37
+ (await this.store.ensureThread(normalizedThreadId));
38
+ const previousProviderSessionId = String(previous.sessionId ?? "").trim();
39
+ if (previousProviderSessionId) {
40
+ this.threadByProviderSession.set(previousProviderSessionId, normalizedThreadId);
41
+ }
42
+ await this.store.appendMessage(normalizedThreadId, {
43
+ role: "user",
44
+ source: normalizedSource,
45
+ text: text || "[non-text message]",
46
+ }, this.maxMessages);
47
+ this.turnContextByThread.set(normalizedThreadId, {
48
+ source: normalizedSource,
49
+ metadata: { ...metadata },
50
+ });
51
+ const result = await this.assistantProvider.sendTurn({
52
+ sessionId: previous.sessionId,
53
+ prompt: text,
54
+ inputItems: normalizedInputItems,
55
+ turnActivityTimeoutMs: this.turnActivityTimeoutMs,
56
+ onSessionReady: async (providerSessionId) => {
57
+ this.threadByProviderSession.set(String(providerSessionId), normalizedThreadId);
58
+ },
59
+ });
60
+ this.threadByProviderSession.set(result.sessionId, normalizedThreadId);
61
+ const assistantText = (result.assistantText || "Assistant provider returned no text output.").trim();
62
+ const thread = await this.store.upsertThread(normalizedThreadId, (current) => ({
63
+ ...current,
64
+ sessionId: result.sessionId ?? current.sessionId ?? null,
65
+ turnCount: (current.turnCount ?? 0) + 1,
66
+ lastMode: this.assistantProvider.kind,
67
+ messages: trimMessages([
68
+ ...(current.messages ?? []),
69
+ {
70
+ role: "assistant",
71
+ source: normalizedSource,
72
+ text: assistantText,
73
+ },
74
+ ], this.maxMessages),
75
+ }));
76
+ return {
77
+ threadId: normalizedThreadId,
78
+ thread,
79
+ assistantText,
80
+ mode: this.assistantProvider.kind,
81
+ };
82
+ });
83
+ }
84
+ async getThread(threadId) {
85
+ const normalizedThreadId = normalizeThreadId(threadId);
86
+ const thread = (await this.store.getThread(normalizedThreadId)) ??
87
+ (await this.store.ensureThread(normalizedThreadId));
88
+ return {
89
+ threadId: normalizedThreadId,
90
+ thread,
91
+ };
92
+ }
93
+ async getMessages(threadId, limit) {
94
+ const normalizedThreadId = normalizeThreadId(threadId);
95
+ const messages = await this.store.listMessages(normalizedThreadId, limit);
96
+ return {
97
+ threadId: normalizedThreadId,
98
+ messages,
99
+ };
100
+ }
101
+ async resetThread(threadId) {
102
+ const normalizedThreadId = normalizeThreadId(threadId);
103
+ const previous = await this.store.getThread(normalizedThreadId);
104
+ this.pendingApprovalsByThread.delete(normalizedThreadId);
105
+ const thread = await this.store.resetThread(normalizedThreadId);
106
+ if (previous?.sessionId) {
107
+ this.threadByProviderSession.delete(previous.sessionId);
108
+ }
109
+ return {
110
+ threadId: normalizedThreadId,
111
+ thread,
112
+ };
113
+ }
114
+ listPendingApprovals(threadId = null) {
115
+ if (threadId) {
116
+ const normalizedThreadId = normalizeThreadId(threadId);
117
+ return [...(this.pendingApprovalsByThread.get(normalizedThreadId) ?? [])].map((entry) => ({
118
+ ...entry,
119
+ }));
120
+ }
121
+ const all = [];
122
+ for (const entries of this.pendingApprovalsByThread.values()) {
123
+ all.push(...entries);
124
+ }
125
+ all.sort((a, b) => a.createdAt.localeCompare(b.createdAt));
126
+ return all.map((entry) => ({ ...entry }));
127
+ }
128
+ async resolvePendingApproval({ threadId, approvalId, decision }) {
129
+ const normalizedThreadId = normalizeThreadId(threadId);
130
+ const targetApprovalId = String(approvalId ?? "").trim();
131
+ if (!targetApprovalId) {
132
+ throw new Error("approvalId is required.");
133
+ }
134
+ const approvals = this.pendingApprovalsByThread.get(normalizedThreadId) ?? [];
135
+ const target = approvals.find((entry) => entry.id === targetApprovalId);
136
+ if (!target) {
137
+ throw new Error(`Unknown approval '${targetApprovalId}' for thread '${normalizedThreadId}'.`);
138
+ }
139
+ const resolved = await this.assistantProvider.resolveApproval({
140
+ approvalId: target.id,
141
+ decision,
142
+ });
143
+ this.pendingApprovalsByThread.set(normalizedThreadId, approvals.filter((entry) => entry.id !== target.id));
144
+ return {
145
+ ...target,
146
+ decision: resolved.decision,
147
+ };
148
+ }
149
+ async interruptThread(threadId) {
150
+ const normalizedThreadId = normalizeThreadId(threadId);
151
+ const thread = (await this.store.getThread(normalizedThreadId)) ??
152
+ (await this.store.ensureThread(normalizedThreadId));
153
+ const sessionId = String(thread?.sessionId ?? "").trim();
154
+ if (!sessionId) {
155
+ return {
156
+ threadId: normalizedThreadId,
157
+ interrupted: false,
158
+ reason: "no_active_session",
159
+ };
160
+ }
161
+ if (typeof this.assistantProvider?.interruptTurn !== "function") {
162
+ return {
163
+ threadId: normalizedThreadId,
164
+ sessionId,
165
+ interrupted: false,
166
+ reason: "not_supported",
167
+ };
168
+ }
169
+ try {
170
+ const result = await this.assistantProvider.interruptTurn({
171
+ sessionId,
172
+ });
173
+ return {
174
+ threadId: normalizedThreadId,
175
+ sessionId,
176
+ ...(result && typeof result === "object" ? result : {}),
177
+ };
178
+ }
179
+ catch (error) {
180
+ return {
181
+ threadId: normalizedThreadId,
182
+ sessionId,
183
+ interrupted: false,
184
+ reason: "error",
185
+ error: sanitizeError(error),
186
+ };
187
+ }
188
+ }
189
+ async shutdown() {
190
+ await this.assistantProvider.shutdown();
191
+ }
192
+ #queue(threadId, work) {
193
+ const previous = this.queueByThread.get(threadId) ?? Promise.resolve();
194
+ const next = previous.then(work, work);
195
+ const tracked = next.finally(() => {
196
+ if (this.queueByThread.get(threadId) === tracked) {
197
+ this.queueByThread.delete(threadId);
198
+ }
199
+ });
200
+ this.queueByThread.set(threadId, tracked);
201
+ return tracked;
202
+ }
203
+ async #handleApprovalRequested(approval) {
204
+ const providerSessionId = String(approval.sessionId ?? approval.threadId ?? "").trim();
205
+ if (!providerSessionId) {
206
+ return;
207
+ }
208
+ let bridgeThreadId = this.threadByProviderSession.get(providerSessionId);
209
+ if (!bridgeThreadId) {
210
+ bridgeThreadId = await this.store.findThreadIdBySessionId(providerSessionId);
211
+ if (bridgeThreadId) {
212
+ this.threadByProviderSession.set(providerSessionId, bridgeThreadId);
213
+ }
214
+ }
215
+ if (!bridgeThreadId) {
216
+ return;
217
+ }
218
+ const entry = {
219
+ id: approval.id,
220
+ kind: approval.kind,
221
+ threadId: bridgeThreadId,
222
+ sessionId: providerSessionId,
223
+ turnId: approval.turnId,
224
+ itemId: approval.itemId,
225
+ command: approval.command,
226
+ cwd: approval.cwd,
227
+ reason: approval.reason,
228
+ commandActions: approval.commandActions,
229
+ createdAt: approval.createdAt,
230
+ };
231
+ const previousEntries = this.pendingApprovalsByThread.get(bridgeThreadId) ?? [];
232
+ this.pendingApprovalsByThread.set(bridgeThreadId, [...previousEntries, entry]);
233
+ if (!this.onApprovalRequested) {
234
+ return;
235
+ }
236
+ const context = this.turnContextByThread.get(bridgeThreadId) ?? {
237
+ source: "internal",
238
+ metadata: {},
239
+ };
240
+ try {
241
+ await this.onApprovalRequested({
242
+ ...entry,
243
+ source: context.source,
244
+ metadata: context.metadata,
245
+ });
246
+ }
247
+ catch (error) {
248
+ console.error(`onApprovalRequested failed: ${sanitizeError(error)}`);
249
+ }
250
+ }
251
+ }
252
+ function trimMessages(messages, maxMessages) {
253
+ const safeMax = Number.isFinite(maxMessages) && maxMessages > 0
254
+ ? Math.min(maxMessages, 1000)
255
+ : DEFAULT_MAX_MESSAGES;
256
+ if (messages.length <= safeMax) {
257
+ return messages;
258
+ }
259
+ return messages.slice(messages.length - safeMax);
260
+ }
261
+ function normalizeSource(value) {
262
+ const source = String(value ?? "").toLowerCase();
263
+ if (source === "web" || source === "telegram" || source === "internal") {
264
+ return source;
265
+ }
266
+ return "internal";
267
+ }
268
+ function sanitizeError(error) {
269
+ const raw = error instanceof Error ? error.message : String(error);
270
+ return raw.split(/\r?\n/).slice(0, 12).join("\n");
271
+ }
272
+ //# sourceMappingURL=bridge-service.js.map