@johpaz/hive 1.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 (156) hide show
  1. package/CONTRIBUTING.md +44 -0
  2. package/README.md +310 -0
  3. package/package.json +96 -0
  4. package/packages/cli/package.json +28 -0
  5. package/packages/cli/src/commands/agent-run.ts +168 -0
  6. package/packages/cli/src/commands/agents.ts +398 -0
  7. package/packages/cli/src/commands/chat.ts +142 -0
  8. package/packages/cli/src/commands/config.ts +50 -0
  9. package/packages/cli/src/commands/cron.ts +161 -0
  10. package/packages/cli/src/commands/dev.ts +95 -0
  11. package/packages/cli/src/commands/doctor.ts +133 -0
  12. package/packages/cli/src/commands/gateway.ts +443 -0
  13. package/packages/cli/src/commands/logs.ts +57 -0
  14. package/packages/cli/src/commands/mcp.ts +175 -0
  15. package/packages/cli/src/commands/message.ts +77 -0
  16. package/packages/cli/src/commands/onboard.ts +1868 -0
  17. package/packages/cli/src/commands/security.ts +144 -0
  18. package/packages/cli/src/commands/service.ts +50 -0
  19. package/packages/cli/src/commands/sessions.ts +116 -0
  20. package/packages/cli/src/commands/skills.ts +187 -0
  21. package/packages/cli/src/commands/update.ts +25 -0
  22. package/packages/cli/src/index.ts +185 -0
  23. package/packages/cli/src/utils/token.ts +6 -0
  24. package/packages/code-bridge/README.md +78 -0
  25. package/packages/code-bridge/package.json +18 -0
  26. package/packages/code-bridge/src/index.ts +95 -0
  27. package/packages/code-bridge/src/process-manager.ts +212 -0
  28. package/packages/code-bridge/src/schemas.ts +133 -0
  29. package/packages/core/package.json +46 -0
  30. package/packages/core/src/agent/agent-loop.ts +369 -0
  31. package/packages/core/src/agent/compaction.ts +140 -0
  32. package/packages/core/src/agent/context-compiler.ts +378 -0
  33. package/packages/core/src/agent/context-guard.ts +91 -0
  34. package/packages/core/src/agent/context.ts +138 -0
  35. package/packages/core/src/agent/conversation-store.ts +198 -0
  36. package/packages/core/src/agent/curator.ts +158 -0
  37. package/packages/core/src/agent/hooks.ts +166 -0
  38. package/packages/core/src/agent/index.ts +116 -0
  39. package/packages/core/src/agent/llm-client.ts +503 -0
  40. package/packages/core/src/agent/native-tools.ts +505 -0
  41. package/packages/core/src/agent/prompt-builder.ts +532 -0
  42. package/packages/core/src/agent/providers/index.ts +167 -0
  43. package/packages/core/src/agent/providers.ts +1 -0
  44. package/packages/core/src/agent/reflector.ts +170 -0
  45. package/packages/core/src/agent/service.ts +64 -0
  46. package/packages/core/src/agent/stuck-loop.ts +133 -0
  47. package/packages/core/src/agent/supervisor.ts +39 -0
  48. package/packages/core/src/agent/tracer.ts +102 -0
  49. package/packages/core/src/agent/workspace.ts +110 -0
  50. package/packages/core/src/canvas/canvas-manager.test.ts +161 -0
  51. package/packages/core/src/canvas/canvas-manager.ts +319 -0
  52. package/packages/core/src/canvas/canvas-tools.ts +420 -0
  53. package/packages/core/src/canvas/emitter.ts +115 -0
  54. package/packages/core/src/canvas/index.ts +2 -0
  55. package/packages/core/src/channels/base.ts +138 -0
  56. package/packages/core/src/channels/discord.ts +260 -0
  57. package/packages/core/src/channels/index.ts +7 -0
  58. package/packages/core/src/channels/manager.ts +383 -0
  59. package/packages/core/src/channels/slack.ts +287 -0
  60. package/packages/core/src/channels/telegram.ts +502 -0
  61. package/packages/core/src/channels/webchat.ts +128 -0
  62. package/packages/core/src/channels/whatsapp.ts +375 -0
  63. package/packages/core/src/config/index.ts +12 -0
  64. package/packages/core/src/config/loader.ts +529 -0
  65. package/packages/core/src/events/event-bus.ts +169 -0
  66. package/packages/core/src/gateway/index.ts +5 -0
  67. package/packages/core/src/gateway/initializer.ts +290 -0
  68. package/packages/core/src/gateway/lane-queue.ts +169 -0
  69. package/packages/core/src/gateway/resolver.ts +108 -0
  70. package/packages/core/src/gateway/router.ts +124 -0
  71. package/packages/core/src/gateway/server.ts +3317 -0
  72. package/packages/core/src/gateway/session.ts +95 -0
  73. package/packages/core/src/gateway/slash-commands.ts +192 -0
  74. package/packages/core/src/heartbeat/index.ts +157 -0
  75. package/packages/core/src/index.ts +19 -0
  76. package/packages/core/src/integrations/catalog.ts +286 -0
  77. package/packages/core/src/integrations/env.ts +64 -0
  78. package/packages/core/src/integrations/index.ts +2 -0
  79. package/packages/core/src/memory/index.ts +1 -0
  80. package/packages/core/src/memory/notes.ts +68 -0
  81. package/packages/core/src/plugins/api.ts +128 -0
  82. package/packages/core/src/plugins/index.ts +2 -0
  83. package/packages/core/src/plugins/loader.ts +365 -0
  84. package/packages/core/src/resilience/circuit-breaker.ts +225 -0
  85. package/packages/core/src/security/google-chat.ts +269 -0
  86. package/packages/core/src/security/index.ts +192 -0
  87. package/packages/core/src/security/pairing.ts +250 -0
  88. package/packages/core/src/security/rate-limit.ts +270 -0
  89. package/packages/core/src/security/signal.ts +321 -0
  90. package/packages/core/src/state/store.ts +312 -0
  91. package/packages/core/src/storage/bun-sqlite-store.ts +188 -0
  92. package/packages/core/src/storage/crypto.ts +101 -0
  93. package/packages/core/src/storage/db-context.ts +333 -0
  94. package/packages/core/src/storage/onboarding.ts +1087 -0
  95. package/packages/core/src/storage/schema.ts +541 -0
  96. package/packages/core/src/storage/seed.ts +571 -0
  97. package/packages/core/src/storage/sqlite.ts +387 -0
  98. package/packages/core/src/storage/usage.ts +212 -0
  99. package/packages/core/src/tools/bridge-events.ts +74 -0
  100. package/packages/core/src/tools/browser.ts +275 -0
  101. package/packages/core/src/tools/codebridge.ts +421 -0
  102. package/packages/core/src/tools/coordinator-tools.ts +179 -0
  103. package/packages/core/src/tools/cron.ts +611 -0
  104. package/packages/core/src/tools/exec.ts +140 -0
  105. package/packages/core/src/tools/fs.ts +364 -0
  106. package/packages/core/src/tools/index.ts +12 -0
  107. package/packages/core/src/tools/memory.ts +176 -0
  108. package/packages/core/src/tools/notify.ts +113 -0
  109. package/packages/core/src/tools/project-management.ts +376 -0
  110. package/packages/core/src/tools/project.ts +375 -0
  111. package/packages/core/src/tools/read.ts +158 -0
  112. package/packages/core/src/tools/web.ts +436 -0
  113. package/packages/core/src/tools/workspace.ts +171 -0
  114. package/packages/core/src/utils/benchmark.ts +80 -0
  115. package/packages/core/src/utils/crypto.ts +73 -0
  116. package/packages/core/src/utils/date.ts +42 -0
  117. package/packages/core/src/utils/index.ts +4 -0
  118. package/packages/core/src/utils/logger.ts +388 -0
  119. package/packages/core/src/utils/retry.ts +70 -0
  120. package/packages/core/src/voice/index.ts +583 -0
  121. package/packages/core/tsconfig.json +9 -0
  122. package/packages/mcp/package.json +26 -0
  123. package/packages/mcp/src/config.ts +13 -0
  124. package/packages/mcp/src/index.ts +1 -0
  125. package/packages/mcp/src/logger.ts +42 -0
  126. package/packages/mcp/src/manager.ts +434 -0
  127. package/packages/mcp/src/transports/index.ts +67 -0
  128. package/packages/mcp/src/transports/sse.ts +241 -0
  129. package/packages/mcp/src/transports/websocket.ts +159 -0
  130. package/packages/skills/package.json +21 -0
  131. package/packages/skills/src/bundled/agent_management/SKILL.md +24 -0
  132. package/packages/skills/src/bundled/browser_automation/SKILL.md +30 -0
  133. package/packages/skills/src/bundled/context_compact/SKILL.md +35 -0
  134. package/packages/skills/src/bundled/cron_manager/SKILL.md +52 -0
  135. package/packages/skills/src/bundled/file_manager/SKILL.md +76 -0
  136. package/packages/skills/src/bundled/http_client/SKILL.md +24 -0
  137. package/packages/skills/src/bundled/memory/SKILL.md +42 -0
  138. package/packages/skills/src/bundled/project_management/SKILL.md +26 -0
  139. package/packages/skills/src/bundled/shell/SKILL.md +43 -0
  140. package/packages/skills/src/bundled/system_notify/SKILL.md +52 -0
  141. package/packages/skills/src/bundled/voice/SKILL.md +25 -0
  142. package/packages/skills/src/bundled/web_search/SKILL.md +29 -0
  143. package/packages/skills/src/index.ts +1 -0
  144. package/packages/skills/src/loader.ts +282 -0
  145. package/packages/tools/package.json +43 -0
  146. package/packages/tools/src/browser/browser.test.ts +111 -0
  147. package/packages/tools/src/browser/index.ts +272 -0
  148. package/packages/tools/src/canvas/index.ts +220 -0
  149. package/packages/tools/src/cron/cron.test.ts +164 -0
  150. package/packages/tools/src/cron/index.ts +304 -0
  151. package/packages/tools/src/filesystem/filesystem.test.ts +240 -0
  152. package/packages/tools/src/filesystem/index.ts +379 -0
  153. package/packages/tools/src/git/index.ts +239 -0
  154. package/packages/tools/src/index.ts +4 -0
  155. package/packages/tools/src/shell/detect-env.ts +70 -0
  156. package/packages/tools/tsconfig.json +9 -0
@@ -0,0 +1,420 @@
1
+ import type { Tool } from "../agent/native-tools.ts";
2
+ import type { Config } from "../config/loader.ts";
3
+ import { canvasManager } from "./canvas-manager.ts";
4
+ import { logger } from "../utils/logger.ts";
5
+
6
+ export function createCanvasRenderTool(_config: Config): Tool {
7
+ const log = logger.child("canvas-render");
8
+
9
+ return {
10
+ name: "canvas_render",
11
+ description: "Render a component on the user's canvas",
12
+ parameters: {
13
+ type: "object",
14
+ properties: {
15
+ sessionId: {
16
+ type: "string",
17
+ description: "Session ID to render to (auto-resolved from user context if omitted)",
18
+ },
19
+ component: {
20
+ type: "object",
21
+ properties: {
22
+ id: {
23
+ type: "string",
24
+ description: "Unique component ID",
25
+ },
26
+ type: {
27
+ type: "string",
28
+ enum: ["button", "form", "chart", "table", "markdown", "text", "image"],
29
+ description: "Component type",
30
+ },
31
+ props: {
32
+ type: "object",
33
+ description: "Component properties",
34
+ },
35
+ },
36
+ required: ["id", "type", "props"],
37
+ },
38
+ },
39
+ required: ["component"],
40
+ },
41
+ execute: async (params: Record<string, unknown>, config?: any) => {
42
+ const userId = config?.configurable?.user_id;
43
+ const rawSessionId = params.sessionId as string;
44
+ const sessionId = rawSessionId
45
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
46
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
47
+ const component = params.component as {
48
+ id: string;
49
+ type: "button" | "form" | "chart" | "table" | "markdown" | "text" | "image" | "card" | "progress" | "list" | "confirm";
50
+ props: Record<string, unknown>;
51
+ };
52
+
53
+ log.debug(`Rendering component ${component.id} to session ${sessionId}`);
54
+
55
+ // Check if session is connected, if not try to render to any available session
56
+ if (!canvasManager.isSessionConnected(sessionId)) {
57
+ const connectedSessions = canvasManager.getConnectedSessions();
58
+ if (connectedSessions.length > 0) {
59
+ log.warn(`Session ${sessionId} not connected, using first available: ${connectedSessions[0]}`);
60
+ } else {
61
+ log.warn(`No canvas sessions connected. Rendering to ${sessionId} anyway.`);
62
+ }
63
+ }
64
+
65
+ await canvasManager.render(sessionId, {
66
+ id: component.id,
67
+ type: component.type as any,
68
+ props: component.props,
69
+ });
70
+
71
+ return {
72
+ success: true,
73
+ componentId: component.id,
74
+ sessionId,
75
+ };
76
+ },
77
+ };
78
+ }
79
+
80
+ export function createCanvasAskTool(_config: Config): Tool {
81
+ const log = logger.child("canvas-ask");
82
+
83
+ return {
84
+ name: "canvas_ask",
85
+ description: "Display a form and wait for user response",
86
+ parameters: {
87
+ type: "object",
88
+ properties: {
89
+ sessionId: {
90
+ type: "string",
91
+ description: "Session ID",
92
+ },
93
+ title: {
94
+ type: "string",
95
+ description: "Form title",
96
+ },
97
+ fields: {
98
+ type: "array",
99
+ items: {
100
+ type: "object",
101
+ properties: {
102
+ name: { type: "string" },
103
+ label: { type: "string" },
104
+ type: { type: "string", enum: ["text", "email", "textarea", "select"] },
105
+ required: { type: "boolean" },
106
+ options: {
107
+ type: "array",
108
+ items: {
109
+ type: "object",
110
+ properties: {
111
+ label: { type: "string" },
112
+ value: { type: "string" },
113
+ },
114
+ },
115
+ },
116
+ },
117
+ required: ["name", "label", "type"],
118
+ },
119
+ description: "Form fields",
120
+ },
121
+ timeout: {
122
+ type: "number",
123
+ description: "Timeout in milliseconds (default: 300000)",
124
+ },
125
+ },
126
+ required: ["fields"],
127
+ },
128
+ execute: async (params: Record<string, unknown>, config?: any) => {
129
+ const userId = config?.configurable?.user_id;
130
+ const rawSessionId = params.sessionId as string;
131
+ const sessionId = rawSessionId
132
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
133
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
134
+ const title = (params.title as string) ?? "Form";
135
+ const fields = params.fields as Array<{
136
+ name: string;
137
+ label: string;
138
+ type: string;
139
+ required?: boolean;
140
+ options?: Array<{ label: string; value: string }>;
141
+ }>;
142
+ const timeout = (params.timeout as number) ?? 300000;
143
+
144
+ const formId = `form-${Date.now()}`;
145
+
146
+ log.debug(`Asking user via form ${formId}`);
147
+
148
+ await canvasManager.render(sessionId, {
149
+ id: formId,
150
+ type: "form",
151
+ props: { title, fields },
152
+ });
153
+
154
+ try {
155
+ const response = await canvasManager.waitForInteraction(sessionId, formId, timeout);
156
+
157
+ return {
158
+ success: true,
159
+ formId,
160
+ data: response,
161
+ };
162
+ } catch (error) {
163
+ return {
164
+ success: false,
165
+ formId,
166
+ error: (error as Error).message,
167
+ };
168
+ }
169
+ },
170
+ };
171
+ }
172
+
173
+ export function createCanvasClearTool(_config: Config): Tool {
174
+ const log = logger.child("canvas-clear");
175
+
176
+ return {
177
+ name: "canvas_clear",
178
+ description: "Clear the canvas for a session",
179
+ parameters: {
180
+ type: "object",
181
+ properties: {
182
+ sessionId: {
183
+ type: "string",
184
+ description: "Session ID to clear",
185
+ },
186
+ },
187
+ required: [],
188
+ },
189
+ execute: async (params: Record<string, unknown>, config?: any) => {
190
+ const userId = config?.configurable?.user_id;
191
+ const rawSessionId = params.sessionId as string;
192
+ const sessionId = rawSessionId
193
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
194
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
195
+
196
+ log.debug(`Clearing canvas for session ${sessionId}`);
197
+
198
+ await canvasManager.clear(sessionId);
199
+
200
+ return { success: true, sessionId };
201
+ },
202
+ };
203
+ }
204
+
205
+ export function createCanvasTools(config: Config): Tool[] {
206
+ return [
207
+ createCanvasRenderTool(config),
208
+ createCanvasAskTool(config),
209
+ createCanvasClearTool(config),
210
+ createCanvasCardTool(config),
211
+ createCanvasProgressTool(config),
212
+ createCanvasListTool(config),
213
+ createCanvasConfirmTool(config),
214
+ ];
215
+ }
216
+
217
+ // ═══════════════════════════════════════════════════════════════════════════
218
+ // Extended Canvas Tools for A2UI
219
+ // ═══════════════════════════════════════════════════════════════════════════
220
+
221
+ export function createCanvasCardTool(_config: Config): Tool {
222
+ const log = logger.child("canvas-card");
223
+
224
+ return {
225
+ name: "canvas_show_card",
226
+ description: "Display a card with labeled items (useful for showing status, summaries)",
227
+ parameters: {
228
+ type: "object",
229
+ properties: {
230
+ sessionId: { type: "string", description: "Session ID" },
231
+ title: { type: "string", description: "Card title" },
232
+ items: {
233
+ type: "array",
234
+ items: {
235
+ type: "object",
236
+ properties: {
237
+ label: { type: "string" },
238
+ value: { type: "string" },
239
+ variant: { type: "string", enum: ["default", "success", "warning", "danger"] },
240
+ },
241
+ },
242
+ },
243
+ actions: {
244
+ type: "array",
245
+ items: {
246
+ type: "object",
247
+ properties: {
248
+ id: { type: "string" },
249
+ label: { type: "string" },
250
+ variant: { type: "string", enum: ["primary", "secondary", "danger", "success"] },
251
+ },
252
+ },
253
+ },
254
+ },
255
+ required: ["items"],
256
+ },
257
+ execute: async (params: Record<string, unknown>, config?: any) => {
258
+ const userId = config?.configurable?.user_id;
259
+ const rawSessionId = params.sessionId as string;
260
+ const sessionId = rawSessionId
261
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
262
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
263
+ const title = (params.title as string) ?? "Information";
264
+ const items = (params.items as Array<{ label: string; value: string; variant?: string }>) ?? [];
265
+ const actions = (params.actions as Array<{ id: string; label: string; variant?: string }>) ?? [];
266
+
267
+ const cardId = `card-${Date.now()}`;
268
+
269
+ await canvasManager.render(sessionId, {
270
+ id: cardId,
271
+ type: "card",
272
+ props: { title, items, actions },
273
+ });
274
+
275
+ return { success: true, cardId, sessionId };
276
+ },
277
+ };
278
+ }
279
+
280
+ export function createCanvasProgressTool(_config: Config): Tool {
281
+ const log = logger.child("canvas-progress");
282
+
283
+ return {
284
+ name: "canvas_show_progress",
285
+ description: "Display progress bars for tasks (useful for multi-step operations)",
286
+ parameters: {
287
+ type: "object",
288
+ properties: {
289
+ sessionId: { type: "string", description: "Session ID" },
290
+ tasks: {
291
+ type: "array",
292
+ items: {
293
+ type: "object",
294
+ properties: {
295
+ id: { type: "string" },
296
+ name: { type: "string" },
297
+ progress: { type: "number" },
298
+ status: { type: "string", enum: ["pending", "running", "completed", "error"] },
299
+ },
300
+ },
301
+ },
302
+ },
303
+ required: ["tasks"],
304
+ },
305
+ execute: async (params: Record<string, unknown>, config?: any) => {
306
+ const userId = config?.configurable?.user_id;
307
+ const rawSessionId = params.sessionId as string;
308
+ const sessionId = rawSessionId
309
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
310
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
311
+ const tasks = (params.tasks as Array<{ id: string; name: string; progress: number; status?: string }>) ?? [];
312
+
313
+ const progressId = `progress-${Date.now()}`;
314
+
315
+ await canvasManager.render(sessionId, {
316
+ id: progressId,
317
+ type: "progress",
318
+ props: { tasks },
319
+ });
320
+
321
+ return { success: true, progressId, sessionId };
322
+ },
323
+ };
324
+ }
325
+
326
+ export function createCanvasListTool(_config: Config): Tool {
327
+ const log = logger.child("canvas-list");
328
+
329
+ return {
330
+ name: "canvas_show_list",
331
+ description: "Display a list of key-value pairs (useful for configuration display)",
332
+ parameters: {
333
+ type: "object",
334
+ properties: {
335
+ sessionId: { type: "string", description: "Session ID" },
336
+ title: { type: "string", description: "List title" },
337
+ items: {
338
+ type: "array",
339
+ items: {
340
+ type: "object",
341
+ properties: {
342
+ key: { type: "string" },
343
+ value: { type: "string" },
344
+ },
345
+ },
346
+ },
347
+ },
348
+ required: ["items"],
349
+ },
350
+ execute: async (params: Record<string, unknown>, config?: any) => {
351
+ const userId = config?.configurable?.user_id;
352
+ const rawSessionId = params.sessionId as string;
353
+ const sessionId = rawSessionId
354
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
355
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
356
+ const title = (params.title as string) ?? "Details";
357
+ const items = (params.items as Array<{ key: string; value: string }>) ?? [];
358
+
359
+ const listId = `list-${Date.now()}`;
360
+
361
+ await canvasManager.render(sessionId, {
362
+ id: listId,
363
+ type: "list",
364
+ props: { title, items },
365
+ });
366
+
367
+ return { success: true, listId, sessionId };
368
+ },
369
+ };
370
+ }
371
+
372
+ export function createCanvasConfirmTool(_config: Config): Tool {
373
+ const log = logger.child("canvas-confirm");
374
+
375
+ return {
376
+ name: "canvas_confirm",
377
+ description: "Show a confirmation dialog and wait for user response",
378
+ parameters: {
379
+ type: "object",
380
+ properties: {
381
+ sessionId: { type: "string", description: "Session ID" },
382
+ title: { type: "string", description: "Dialog title" },
383
+ message: { type: "string", description: "Confirmation message" },
384
+ confirmLabel: { type: "string", description: "Confirm button label" },
385
+ cancelLabel: { type: "string", description: "Cancel button label" },
386
+ danger: { type: "boolean", description: "Show as dangerous action" },
387
+ timeout: { type: "number", description: "Timeout in ms (default: 60000)" },
388
+ },
389
+ required: ["message"],
390
+ },
391
+ execute: async (params: Record<string, unknown>, config?: any) => {
392
+ const userId = config?.configurable?.user_id;
393
+ const rawSessionId = params.sessionId as string;
394
+ const sessionId = rawSessionId
395
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
396
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
397
+ const title = (params.title as string) ?? "Confirm";
398
+ const message = params.message as string;
399
+ const confirmLabel = (params.confirmLabel as string) ?? "Confirm";
400
+ const cancelLabel = (params.cancelLabel as string) ?? "Cancel";
401
+ const danger = (params.danger as boolean) ?? false;
402
+ const timeout = (params.timeout as number) ?? 60000;
403
+
404
+ const confirmId = `confirm-${Date.now()}`;
405
+
406
+ await canvasManager.render(sessionId, {
407
+ id: confirmId,
408
+ type: "confirm",
409
+ props: { title, message, confirmLabel, cancelLabel, danger },
410
+ });
411
+
412
+ try {
413
+ const response = await canvasManager.waitForInteraction(sessionId, confirmId, timeout);
414
+ return { success: true, confirmed: response === true, confirmId, sessionId };
415
+ } catch (error) {
416
+ return { success: false, confirmed: false, confirmId, error: (error as Error).message, sessionId };
417
+ }
418
+ },
419
+ };
420
+ }
@@ -0,0 +1,115 @@
1
+ import { getDb } from "../storage/sqlite"
2
+
3
+ export interface CanvasEvent {
4
+ type: CanvasEventType
5
+ data: any
6
+ timestamp: number
7
+ }
8
+
9
+ export type CanvasEventType =
10
+ | "canvas:snapshot"
11
+ | "canvas:node_add"
12
+ | "canvas:node_update"
13
+ | "canvas:node_remove"
14
+ | "canvas:edge_add"
15
+ | "canvas:edge_remove"
16
+ | "ag-ui:event"
17
+
18
+ const subscribers = new Set<{ send: (data: string) => void }>()
19
+
20
+ export function subscribeCanvas(ws: { send: (data: string) => void }) {
21
+ subscribers.add(ws)
22
+ }
23
+
24
+ export function unsubscribeCanvas(ws: { send: (data: string) => void }) {
25
+ subscribers.delete(ws)
26
+ }
27
+
28
+ export function emitCanvas(type: CanvasEventType, data: any) {
29
+ const event: CanvasEvent = { type, data, timestamp: Date.now() }
30
+ const payload = JSON.stringify(event)
31
+ for (const ws of subscribers) {
32
+ try {
33
+ ws.send(payload)
34
+ } catch {
35
+ subscribers.delete(ws)
36
+ }
37
+ }
38
+ }
39
+
40
+ export function getCanvasSnapshot() {
41
+ const db = getDb()
42
+
43
+ const agentNodes = db
44
+ .query<any, []>("SELECT id, name, description, status FROM agents")
45
+ .all()
46
+ .map((a: any) => ({
47
+ id: a.id,
48
+ name: a.name,
49
+ description: a.description,
50
+ status: a.status,
51
+ type: "agent",
52
+ }))
53
+
54
+ const mcpNodes = db
55
+ .query<any, []>("SELECT id, name, status FROM mcp_servers WHERE enabled = 1")
56
+ .all()
57
+ .map((m: any) => ({
58
+ id: `mcp:${m.id}`,
59
+ name: m.name,
60
+ status: m.status,
61
+ type: "mcp",
62
+ }))
63
+
64
+ // Proyectos activos
65
+ const projectNodes = db
66
+ .query<any, []>("SELECT id, name, type, status, progress, agent_id FROM projects WHERE status IN ('active','pending','paused')")
67
+ .all()
68
+ .map((p: any) => ({
69
+ id: `project_${p.id}`,
70
+ name: p.name,
71
+ status: p.status,
72
+ type: "project",
73
+ data: { progress: p.progress, projectType: p.type, agentId: p.agent_id },
74
+ }))
75
+
76
+ // Tareas de proyectos activos
77
+ const taskNodes = db
78
+ .query<any, []>(`
79
+ SELECT t.id, t.name, t.status, t.progress, t.agent_id, t.project_id
80
+ FROM tasks t
81
+ INNER JOIN projects p ON t.project_id = p.id
82
+ WHERE p.status IN ('active','pending','paused')
83
+ `)
84
+ .all()
85
+ .map((t: any) => ({
86
+ id: `task_${t.id}`,
87
+ name: t.name,
88
+ status: t.status,
89
+ type: "task",
90
+ data: { progress: t.progress, agentId: t.agent_id, projectId: t.project_id },
91
+ }))
92
+
93
+ // Edges: proyecto → tarea
94
+ const projectTaskEdges = taskNodes.map((t: any) => ({
95
+ id: `edge_proj_task_${t.id.replace("task_", "")}`,
96
+ source: `project_${t.data.projectId}`,
97
+ target: t.id,
98
+ edgeType: "contains",
99
+ }))
100
+
101
+ // Edges: tarea → agente asignado
102
+ const taskAgentEdges = taskNodes
103
+ .filter((t: any) => t.data.agentId)
104
+ .map((t: any) => ({
105
+ id: `edge_task_agent_${t.id.replace("task_", "")}`,
106
+ source: t.id,
107
+ target: t.data.agentId,
108
+ edgeType: "assigned_to",
109
+ }))
110
+
111
+ return {
112
+ nodes: [...agentNodes, ...mcpNodes, ...projectNodes, ...taskNodes],
113
+ edges: [...projectTaskEdges, ...taskAgentEdges],
114
+ }
115
+ }
@@ -0,0 +1,2 @@
1
+ export * from "./canvas-manager.ts";
2
+ export * from "./canvas-tools.ts";
@@ -0,0 +1,138 @@
1
+ export interface OutboundMessage {
2
+ type: "message" | "stream" | "status" | "error" | "pong" | "command_result" | "log" | "typing" | "audio";
3
+ sessionId: string;
4
+ content?: string;
5
+ chunk?: string;
6
+ isLast?: boolean;
7
+ isStep?: boolean;
8
+ stepType?: "plan" | "tool_call" | "tool_result" | "text";
9
+ audio?: {
10
+ buffer?: Buffer;
11
+ base64?: string;
12
+ mimeType?: string;
13
+ };
14
+ status?: {
15
+ state: string;
16
+ model?: string;
17
+ tokens?: number;
18
+ };
19
+ error?: string;
20
+ result?: unknown;
21
+ logEntry?: {
22
+ timestamp: string;
23
+ level: string;
24
+ source: string;
25
+ message: string;
26
+ meta?: Record<string, unknown>;
27
+ };
28
+ }
29
+
30
+ export interface IncomingMessage {
31
+ sessionId: string;
32
+ channel: string;
33
+ accountId: string;
34
+ peerId: string;
35
+ peerKind: "direct" | "group";
36
+ content: string;
37
+ audio?: {
38
+ buffer?: Buffer;
39
+ url?: string;
40
+ base64?: string;
41
+ mimeType?: string;
42
+ };
43
+ metadata?: Record<string, unknown>;
44
+ replyToId?: string;
45
+ }
46
+
47
+ export interface ChannelConfig {
48
+ enabled: boolean;
49
+ dmPolicy: "open" | "pairing" | "allowlist";
50
+ allowFrom: string[];
51
+ }
52
+
53
+ export interface IChannel {
54
+ name: string;
55
+ accountId: string;
56
+ config: ChannelConfig;
57
+ start(): Promise<void>;
58
+ stop(): Promise<void>;
59
+ send(sessionId: string, message: OutboundMessage): Promise<void>;
60
+ sendAudio?(sessionId: string, audio: Buffer, mimeType: string): Promise<void>;
61
+ onMessage(handler: MessageHandler): void;
62
+ isRunning(): boolean;
63
+ startTyping?(sessionId: string): Promise<void>;
64
+ stopTyping?(sessionId: string): Promise<void>;
65
+ markAsRead?(sessionId: string, messageId?: string): Promise<void>;
66
+ }
67
+
68
+ export type MessageHandler = (message: IncomingMessage) => Promise<void>;
69
+
70
+ export abstract class BaseChannel implements IChannel {
71
+ abstract name: string;
72
+ abstract accountId: string;
73
+ abstract config: ChannelConfig;
74
+
75
+ protected messageHandler?: MessageHandler;
76
+ protected running = false;
77
+ protected typingIntervals: Map<string, Timer> = new Map();
78
+
79
+ abstract start(): Promise<void>;
80
+ abstract stop(): Promise<void>;
81
+ abstract send(sessionId: string, message: OutboundMessage): Promise<void>;
82
+
83
+ onMessage(handler: MessageHandler): void {
84
+ this.messageHandler = handler;
85
+ }
86
+
87
+ isRunning(): boolean {
88
+ return this.running;
89
+ }
90
+
91
+ async startTyping(_sessionId: string): Promise<void> {
92
+ // Default: no-op, override in subclasses
93
+ }
94
+
95
+ async stopTyping(sessionId: string): Promise<void> {
96
+ const interval = this.typingIntervals.get(sessionId);
97
+ if (interval) {
98
+ clearInterval(interval);
99
+ this.typingIntervals.delete(sessionId);
100
+ }
101
+ }
102
+
103
+ async markAsRead(_sessionId: string, _messageId?: string): Promise<void> {
104
+ // Default: no-op, override in subclasses
105
+ }
106
+
107
+ protected async handleMessage(message: IncomingMessage): Promise<void> {
108
+ if (this.messageHandler) {
109
+ await this.messageHandler(message);
110
+ }
111
+ }
112
+
113
+ protected isUserAllowed(peerId: string): boolean {
114
+ if (this.config.dmPolicy === "open") {
115
+ return true;
116
+ }
117
+
118
+ const normalizedPeerId = `tg:${peerId}`;
119
+
120
+ if (this.config.dmPolicy === "allowlist") {
121
+ return this.config.allowFrom.some(
122
+ (allowed) => allowed === peerId || allowed === normalizedPeerId
123
+ );
124
+ }
125
+
126
+ if (this.config.dmPolicy === "pairing") {
127
+ return this.config.allowFrom.some(
128
+ (allowed) => allowed === peerId || allowed === normalizedPeerId
129
+ );
130
+ }
131
+
132
+ return false;
133
+ }
134
+
135
+ protected formatSessionId(peerId: string, _kind: "direct" | "group"): string {
136
+ return peerId;
137
+ }
138
+ }