@johpaz/hive-sdk 0.0.14 → 0.0.15

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 (199) hide show
  1. package/.github/CODEOWNERS +9 -0
  2. package/.github/workflows/publish.yml +89 -0
  3. package/.github/workflows/version-bump.yml +102 -0
  4. package/CHANGELOG.md +38 -0
  5. package/README.md +158 -0
  6. package/bun.lock +543 -0
  7. package/bunfig.toml +7 -0
  8. package/docs/API-AGENTS.md +316 -0
  9. package/docs/API-CONTEXT-COMPILER.md +252 -0
  10. package/docs/API-DAG-SCHEDULER.md +273 -0
  11. package/docs/API-TOOLS-SKILLS-CHANNELS.md +293 -0
  12. package/docs/API-WORKERS-EVENTS.md +152 -0
  13. package/docs/INDEX.md +141 -0
  14. package/docs/README.md +68 -0
  15. package/package.json +54 -105
  16. package/packages/cli/package.json +17 -0
  17. package/packages/cli/src/commands/init.ts +56 -0
  18. package/packages/cli/src/commands/run.ts +45 -0
  19. package/packages/cli/src/commands/test.ts +42 -0
  20. package/packages/cli/src/commands/trace.ts +55 -0
  21. package/packages/cli/src/index.ts +43 -0
  22. package/packages/core/package.json +58 -0
  23. package/packages/core/src/ace/Curator.ts +158 -0
  24. package/packages/core/src/ace/Reflector.ts +200 -0
  25. package/packages/core/src/ace/Tracer.ts +100 -0
  26. package/packages/core/src/ace/index.ts +4 -0
  27. package/packages/core/src/agent/AgentRunner.ts +699 -0
  28. package/packages/core/src/agent/Compaction.ts +221 -0
  29. package/packages/core/src/agent/ContextCompiler.ts +567 -0
  30. package/packages/core/src/agent/ContextGuard.ts +91 -0
  31. package/packages/core/src/agent/ConversationStore.ts +244 -0
  32. package/packages/core/src/agent/Hooks.ts +166 -0
  33. package/packages/core/src/agent/NativeTools.ts +31 -0
  34. package/packages/core/src/agent/PromptBuilder.ts +169 -0
  35. package/packages/core/src/agent/Service.ts +267 -0
  36. package/packages/core/src/agent/StuckLoop.ts +133 -0
  37. package/packages/core/src/agent/index.ts +12 -0
  38. package/packages/core/src/agent/providers/LLMClient.ts +149 -0
  39. package/packages/core/src/agent/providers/anthropic.ts +212 -0
  40. package/packages/core/src/agent/providers/gemini.ts +215 -0
  41. package/packages/core/src/agent/providers/index.ts +199 -0
  42. package/packages/core/src/agent/providers/interface.ts +195 -0
  43. package/packages/core/src/agent/providers/ollama.ts +175 -0
  44. package/packages/core/src/agent/providers/openai-compat.ts +231 -0
  45. package/packages/core/src/agent/providers.ts +1 -0
  46. package/packages/core/src/agent/selectors/PlaybookSelector.ts +147 -0
  47. package/packages/core/src/agent/selectors/SkillSelector.ts +478 -0
  48. package/packages/core/src/agent/selectors/ToolSelector.ts +577 -0
  49. package/packages/core/src/agent/selectors/index.ts +6 -0
  50. package/packages/core/src/api/createAgent.test.ts +48 -0
  51. package/packages/core/src/api/createAgent.ts +122 -0
  52. package/packages/core/src/api/index.ts +2 -0
  53. package/packages/core/src/canvas/CanvasManager.ts +390 -0
  54. package/packages/core/src/canvas/a2ui-tools.ts +255 -0
  55. package/packages/core/src/canvas/canvas-tools.ts +448 -0
  56. package/packages/core/src/canvas/emitter.ts +149 -0
  57. package/packages/core/src/canvas/index.ts +6 -0
  58. package/packages/core/src/config/index.ts +2 -0
  59. package/packages/core/src/config/loader.ts +554 -0
  60. package/packages/core/src/ethics/EthicsGuard.test.ts +54 -0
  61. package/packages/core/src/ethics/EthicsGuard.ts +66 -0
  62. package/packages/core/src/ethics/index.ts +2 -0
  63. package/packages/core/src/gateway/channel-notify.test.ts +14 -0
  64. package/packages/core/src/gateway/channel-notify.ts +12 -0
  65. package/packages/core/src/gateway/index.ts +1 -0
  66. package/packages/core/src/index.ts +37 -0
  67. package/packages/core/src/mcp/MCPClient.ts +439 -0
  68. package/packages/core/src/mcp/MCPToolAdapter.ts +176 -0
  69. package/packages/core/src/mcp/config.ts +13 -0
  70. package/packages/core/src/mcp/hot-reload.ts +147 -0
  71. package/packages/core/src/mcp/index.ts +11 -0
  72. package/packages/core/src/mcp/logger.ts +42 -0
  73. package/packages/core/src/mcp/singleton.ts +21 -0
  74. package/packages/core/src/mcp/transports/index.ts +67 -0
  75. package/packages/core/src/mcp/transports/sse.ts +241 -0
  76. package/packages/core/src/mcp/transports/websocket.ts +159 -0
  77. package/packages/core/src/memory/Scratchpad.test.ts +47 -0
  78. package/packages/core/src/memory/Scratchpad.ts +37 -0
  79. package/packages/core/src/memory/Storage.ts +6 -0
  80. package/packages/core/src/memory/index.ts +2 -0
  81. package/packages/core/src/multimodal/VisionService.ts +293 -0
  82. package/packages/core/src/multimodal/index.ts +2 -0
  83. package/packages/core/src/multimodal/types.ts +28 -0
  84. package/packages/core/src/security/Pairing.ts +250 -0
  85. package/packages/core/src/security/RateLimit.ts +270 -0
  86. package/packages/core/src/security/index.ts +4 -0
  87. package/packages/core/src/skills/SkillLoader.ts +388 -0
  88. package/packages/core/src/skills/bundled-data.generated.ts +3332 -0
  89. package/packages/core/src/skills/defineSkill.ts +18 -0
  90. package/packages/core/src/skills/index.ts +4 -0
  91. package/packages/core/src/state/index.ts +2 -0
  92. package/packages/core/src/state/store.ts +312 -0
  93. package/packages/core/src/storage/SQLiteStorage.ts +407 -0
  94. package/packages/core/src/storage/crypto.ts +101 -0
  95. package/packages/core/src/storage/index.ts +10 -0
  96. package/packages/core/src/storage/onboarding.ts +1603 -0
  97. package/packages/core/src/storage/schema.ts +689 -0
  98. package/packages/core/src/storage/seed.ts +740 -0
  99. package/packages/core/src/storage/usage.ts +374 -0
  100. package/packages/core/src/swarm/AgentBus.ts +460 -0
  101. package/packages/core/src/swarm/AgentExecutor.ts +53 -0
  102. package/packages/core/src/swarm/Coordinator.ts +251 -0
  103. package/packages/core/src/swarm/EventBridge.ts +122 -0
  104. package/packages/core/src/swarm/EventBus.ts +169 -0
  105. package/packages/core/src/swarm/TaskGraph.ts +192 -0
  106. package/packages/core/src/swarm/TaskNode.ts +97 -0
  107. package/packages/core/src/swarm/TaskResult.ts +22 -0
  108. package/packages/core/src/swarm/WorkerPool.ts +236 -0
  109. package/packages/core/src/swarm/errors.ts +37 -0
  110. package/packages/core/src/swarm/index.ts +30 -0
  111. package/packages/core/src/swarm/presets/HiveLearnPreset.ts +99 -0
  112. package/packages/core/src/swarm/presets/ResearchPreset.ts +97 -0
  113. package/packages/core/src/swarm/presets/index.ts +4 -0
  114. package/packages/core/src/swarm/strategies/ParallelStrategy.ts +21 -0
  115. package/packages/core/src/swarm/strategies/PriorityStrategy.ts +46 -0
  116. package/packages/core/src/swarm/strategies/index.ts +3 -0
  117. package/packages/core/src/swarm/types.ts +164 -0
  118. package/packages/core/src/tools/ToolExecutor.ts +58 -0
  119. package/packages/core/src/tools/ToolRegistry.test.ts +98 -0
  120. package/packages/core/src/tools/ToolRegistry.ts +61 -0
  121. package/packages/core/src/tools/agents/get-available-models.ts +118 -0
  122. package/packages/core/src/tools/agents/index.ts +715 -0
  123. package/packages/core/src/tools/bridge-events.ts +26 -0
  124. package/packages/core/src/tools/canvas/index.ts +375 -0
  125. package/packages/core/src/tools/cli/index.ts +142 -0
  126. package/packages/core/src/tools/codebridge/index.ts +342 -0
  127. package/packages/core/src/tools/core/index.ts +476 -0
  128. package/packages/core/src/tools/cron/index.ts +626 -0
  129. package/packages/core/src/tools/filesystem/fs-delete.ts +78 -0
  130. package/packages/core/src/tools/filesystem/fs-edit.ts +106 -0
  131. package/packages/core/src/tools/filesystem/fs-exists.ts +63 -0
  132. package/packages/core/src/tools/filesystem/fs-glob.ts +108 -0
  133. package/packages/core/src/tools/filesystem/fs-list.ts +129 -0
  134. package/packages/core/src/tools/filesystem/fs-read.ts +72 -0
  135. package/packages/core/src/tools/filesystem/fs-write.ts +67 -0
  136. package/packages/core/src/tools/filesystem/index.ts +34 -0
  137. package/packages/core/src/tools/filesystem/workspace-guard.ts +62 -0
  138. package/packages/core/src/tools/index.ts +231 -0
  139. package/packages/core/src/tools/meeting/index.ts +363 -0
  140. package/packages/core/src/tools/office/index.ts +47 -0
  141. package/packages/core/src/tools/office/office-escribir-docx.ts +192 -0
  142. package/packages/core/src/tools/office/office-escribir-pdf.ts +172 -0
  143. package/packages/core/src/tools/office/office-escribir-pptx.ts +174 -0
  144. package/packages/core/src/tools/office/office-escribir-xlsx.ts +116 -0
  145. package/packages/core/src/tools/office/office-leer-docx.ts +93 -0
  146. package/packages/core/src/tools/office/office-leer-pdf.ts +114 -0
  147. package/packages/core/src/tools/office/office-leer-pptx.ts +136 -0
  148. package/packages/core/src/tools/office/office-leer-xlsx.ts +124 -0
  149. package/packages/core/src/tools/projects/index.ts +37 -0
  150. package/packages/core/src/tools/projects/project-create.ts +94 -0
  151. package/packages/core/src/tools/projects/project-done.ts +66 -0
  152. package/packages/core/src/tools/projects/project-fail.ts +66 -0
  153. package/packages/core/src/tools/projects/project-list.ts +96 -0
  154. package/packages/core/src/tools/projects/project-update.ts +72 -0
  155. package/packages/core/src/tools/projects/task-create.ts +68 -0
  156. package/packages/core/src/tools/projects/task-evaluate.ts +93 -0
  157. package/packages/core/src/tools/projects/task-update.ts +93 -0
  158. package/packages/core/src/tools/types.ts +39 -0
  159. package/packages/core/src/tools/voice/index.ts +104 -0
  160. package/packages/core/src/tools/web/browser-click.ts +78 -0
  161. package/packages/core/src/tools/web/browser-extract.ts +139 -0
  162. package/packages/core/src/tools/web/browser-navigate.ts +106 -0
  163. package/packages/core/src/tools/web/browser-screenshot.ts +87 -0
  164. package/packages/core/src/tools/web/browser-script.ts +88 -0
  165. package/packages/core/src/tools/web/browser-service.ts +554 -0
  166. package/packages/core/src/tools/web/browser-type.ts +101 -0
  167. package/packages/core/src/tools/web/browser-wait.ts +136 -0
  168. package/packages/core/src/tools/web/index.ts +41 -0
  169. package/packages/core/src/tools/web/web-fetch.ts +78 -0
  170. package/packages/core/src/tools/web/web-search.ts +123 -0
  171. package/packages/core/src/utils/benchmark.ts +80 -0
  172. package/packages/core/src/utils/crypto.ts +73 -0
  173. package/packages/core/src/utils/date.ts +42 -0
  174. package/packages/core/src/utils/index.ts +10 -0
  175. package/packages/core/src/utils/logger.ts +389 -0
  176. package/packages/core/src/utils/retry.ts +70 -0
  177. package/packages/core/src/utils/toon.ts +253 -0
  178. package/packages/core/src/voice/index.ts +656 -0
  179. package/test/setup-db.ts +216 -0
  180. package/tsconfig.json +39 -0
  181. package/src/agents.ts +0 -1
  182. package/src/canvas.ts +0 -1
  183. package/src/channels.ts +0 -1
  184. package/src/config.ts +0 -1
  185. package/src/events.ts +0 -1
  186. package/src/gateway.ts +0 -1
  187. package/src/index.ts +0 -304
  188. package/src/mcp.ts +0 -1
  189. package/src/multimodal.ts +0 -1
  190. package/src/scheduler.ts +0 -1
  191. package/src/security.ts +0 -1
  192. package/src/skills.ts +0 -1
  193. package/src/state.ts +0 -1
  194. package/src/storage.ts +0 -1
  195. package/src/tools.ts +0 -1
  196. package/src/tts.ts +0 -1
  197. package/src/types.ts +0 -82
  198. package/src/utils.ts +0 -1
  199. package/src/voice.ts +0 -1
@@ -0,0 +1,448 @@
1
+ import type { Tool } from "../agent/NativeTools.ts";
2
+ import type { Config } from "../config/loader.ts";
3
+ import { canvasManager } from "./CanvasManager.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
+ span: {
36
+ type: "string",
37
+ enum: ["full", "half"],
38
+ description: "Width span: 'full' for full-width component, 'half' for half width. Default: single column",
39
+ },
40
+ },
41
+ required: ["id", "type", "props"],
42
+ },
43
+ },
44
+ required: ["component"],
45
+ },
46
+ execute: async (params: Record<string, unknown>, config?: any) => {
47
+ const userId = config?.configurable?.user_id;
48
+ const rawSessionId = params.sessionId as string;
49
+ const sessionId = rawSessionId
50
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
51
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
52
+ const component = params.component as {
53
+ id: string;
54
+ type: "button" | "form" | "chart" | "table" | "markdown" | "text" | "image" | "card" | "progress" | "list" | "confirm";
55
+ props: Record<string, unknown>;
56
+ span?: "full" | "half";
57
+ };
58
+
59
+ log.debug(`Rendering component ${component.id} to session ${sessionId}`);
60
+
61
+ // Check if session is connected, if not try to render to any available session
62
+ if (!canvasManager.isSessionConnected(sessionId)) {
63
+ const connectedSessions = canvasManager.getConnectedSessions();
64
+ if (connectedSessions.length > 0) {
65
+ log.warn(`Session ${sessionId} not connected, using first available: ${connectedSessions[0]}`);
66
+ } else {
67
+ log.warn(`No canvas sessions connected. Rendering to ${sessionId} anyway.`);
68
+ }
69
+ }
70
+
71
+ await canvasManager.render(sessionId, {
72
+ id: component.id,
73
+ type: component.type as any,
74
+ props: component.props,
75
+ span: component.span,
76
+ });
77
+
78
+ return {
79
+ success: true,
80
+ componentId: component.id,
81
+ sessionId,
82
+ };
83
+ },
84
+ };
85
+ }
86
+
87
+ export function createCanvasAskTool(_config: Config): Tool {
88
+ const log = logger.child("canvas-ask");
89
+
90
+ return {
91
+ name: "canvas_ask",
92
+ description: "Display a form and wait for user response",
93
+ parameters: {
94
+ type: "object",
95
+ properties: {
96
+ sessionId: {
97
+ type: "string",
98
+ description: "Session ID",
99
+ },
100
+ title: {
101
+ type: "string",
102
+ description: "Form title",
103
+ },
104
+ fields: {
105
+ type: "array",
106
+ items: {
107
+ type: "object",
108
+ properties: {
109
+ name: { type: "string" },
110
+ label: { type: "string" },
111
+ type: { type: "string", enum: ["text", "email", "textarea", "select"] },
112
+ required: { type: "boolean" },
113
+ options: {
114
+ type: "array",
115
+ items: {
116
+ type: "object",
117
+ properties: {
118
+ label: { type: "string" },
119
+ value: { type: "string" },
120
+ },
121
+ },
122
+ },
123
+ },
124
+ required: ["name", "label", "type"],
125
+ },
126
+ description: "Form fields",
127
+ },
128
+ timeout: {
129
+ type: "number",
130
+ description: "Timeout in milliseconds (default: 300000)",
131
+ },
132
+ },
133
+ required: ["fields"],
134
+ },
135
+ execute: async (params: Record<string, unknown>, config?: any) => {
136
+ const userId = config?.configurable?.user_id;
137
+ const rawSessionId = params.sessionId as string;
138
+ const sessionId = rawSessionId
139
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
140
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
141
+ const title = (params.title as string) ?? "Form";
142
+ const fields = params.fields as Array<{
143
+ name: string;
144
+ label: string;
145
+ type: string;
146
+ required?: boolean;
147
+ options?: Array<{ label: string; value: string }>;
148
+ }>;
149
+ const timeout = (params.timeout as number) ?? 300000;
150
+
151
+ const formId = `form-${Date.now()}`;
152
+
153
+ log.debug(`Asking user via form ${formId}`);
154
+
155
+ await canvasManager.render(sessionId, {
156
+ id: formId,
157
+ type: "form",
158
+ props: { title, fields },
159
+ });
160
+
161
+ try {
162
+ const response = await canvasManager.waitForInteraction(sessionId, formId, timeout);
163
+
164
+ return {
165
+ success: true,
166
+ formId,
167
+ data: response,
168
+ };
169
+ } catch (error) {
170
+ return {
171
+ success: false,
172
+ formId,
173
+ error: (error as Error).message,
174
+ };
175
+ }
176
+ },
177
+ };
178
+ }
179
+
180
+ export function createCanvasClearTool(_config: Config): Tool {
181
+ const log = logger.child("canvas-clear");
182
+
183
+ return {
184
+ name: "canvas_clear",
185
+ description: "Clear the canvas for a session",
186
+ parameters: {
187
+ type: "object",
188
+ properties: {
189
+ sessionId: {
190
+ type: "string",
191
+ description: "Session ID to clear",
192
+ },
193
+ },
194
+ required: [],
195
+ },
196
+ execute: async (params: Record<string, unknown>, config?: any) => {
197
+ const userId = config?.configurable?.user_id;
198
+ const rawSessionId = params.sessionId as string;
199
+ const sessionId = rawSessionId
200
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
201
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
202
+
203
+ log.debug(`Clearing canvas for session ${sessionId}`);
204
+
205
+ await canvasManager.clear(sessionId);
206
+
207
+ return { success: true, sessionId };
208
+ },
209
+ };
210
+ }
211
+
212
+ export function createCanvasTools(config: Config): Tool[] {
213
+ return [
214
+ createCanvasRenderTool(config),
215
+ createCanvasAskTool(config),
216
+ createCanvasClearTool(config),
217
+ createCanvasCardTool(config),
218
+ createCanvasProgressTool(config),
219
+ createCanvasListTool(config),
220
+ createCanvasConfirmTool(config),
221
+ ];
222
+ }
223
+
224
+ // ═══════════════════════════════════════════════════════════════════════════
225
+ // Extended Canvas Tools for A2UI
226
+ // ═══════════════════════════════════════════════════════════════════════════
227
+
228
+ export function createCanvasCardTool(_config: Config): Tool {
229
+ const log = logger.child("canvas-card");
230
+
231
+ return {
232
+ name: "canvas_show_card",
233
+ description: "Display a card with labeled items (useful for showing status, summaries)",
234
+ parameters: {
235
+ type: "object",
236
+ properties: {
237
+ sessionId: { type: "string", description: "Session ID" },
238
+ title: { type: "string", description: "Card title" },
239
+ items: {
240
+ type: "array",
241
+ items: {
242
+ type: "object",
243
+ properties: {
244
+ label: { type: "string" },
245
+ value: { type: "string" },
246
+ variant: { type: "string", enum: ["default", "success", "warning", "danger"] },
247
+ },
248
+ },
249
+ },
250
+ actions: {
251
+ type: "array",
252
+ items: {
253
+ type: "object",
254
+ properties: {
255
+ id: { type: "string" },
256
+ label: { type: "string" },
257
+ variant: { type: "string", enum: ["primary", "secondary", "danger", "success"] },
258
+ },
259
+ },
260
+ },
261
+ span: {
262
+ type: "string",
263
+ enum: ["full", "half"],
264
+ description: "Width span: 'full' for full-width card, 'half' for half width. Default: single column",
265
+ },
266
+ },
267
+ required: ["items"],
268
+ },
269
+ execute: async (params: Record<string, unknown>, config?: any) => {
270
+ const userId = config?.configurable?.user_id;
271
+ const rawSessionId = params.sessionId as string;
272
+ const sessionId = rawSessionId
273
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
274
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
275
+ const title = (params.title as string) ?? "Information";
276
+ const items = (params.items as Array<{ label: string; value: string; variant?: string }>) ?? [];
277
+ const actions = (params.actions as Array<{ id: string; label: string; variant?: string }>) ?? [];
278
+ const span = params.span as "full" | "half" | undefined;
279
+
280
+ const cardId = `card-${Date.now()}`;
281
+
282
+ await canvasManager.render(sessionId, {
283
+ id: cardId,
284
+ type: "card",
285
+ props: { title, items, actions },
286
+ span,
287
+ });
288
+
289
+ return { success: true, cardId, sessionId };
290
+ },
291
+ };
292
+ }
293
+
294
+ export function createCanvasProgressTool(_config: Config): Tool {
295
+ const log = logger.child("canvas-progress");
296
+
297
+ return {
298
+ name: "canvas_show_progress",
299
+ description: "Display progress bars for tasks (useful for multi-step operations)",
300
+ parameters: {
301
+ type: "object",
302
+ properties: {
303
+ sessionId: { type: "string", description: "Session ID" },
304
+ tasks: {
305
+ type: "array",
306
+ items: {
307
+ type: "object",
308
+ properties: {
309
+ id: { type: "string" },
310
+ name: { type: "string" },
311
+ progress: { type: "number" },
312
+ status: { type: "string", enum: ["pending", "running", "completed", "error"] },
313
+ },
314
+ },
315
+ },
316
+ span: {
317
+ type: "string",
318
+ enum: ["full", "half"],
319
+ description: "Width span: 'full' for full-width, 'half' for half width. Default: single column",
320
+ },
321
+ },
322
+ required: ["tasks"],
323
+ },
324
+ execute: async (params: Record<string, unknown>, config?: any) => {
325
+ const userId = config?.configurable?.user_id;
326
+ const rawSessionId = params.sessionId as string;
327
+ const sessionId = rawSessionId
328
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
329
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
330
+ const tasks = (params.tasks as Array<{ id: string; name: string; progress: number; status?: string }>) ?? [];
331
+ const span = params.span as "full" | "half" | undefined;
332
+
333
+ const progressId = `progress-${Date.now()}`;
334
+
335
+ await canvasManager.render(sessionId, {
336
+ id: progressId,
337
+ type: "progress",
338
+ props: { tasks },
339
+ span,
340
+ });
341
+
342
+ return { success: true, progressId, sessionId };
343
+ },
344
+ };
345
+ }
346
+
347
+ export function createCanvasListTool(_config: Config): Tool {
348
+ const log = logger.child("canvas-list");
349
+
350
+ return {
351
+ name: "canvas_show_list",
352
+ description: "Display a list of key-value pairs (useful for configuration display)",
353
+ parameters: {
354
+ type: "object",
355
+ properties: {
356
+ sessionId: { type: "string", description: "Session ID" },
357
+ title: { type: "string", description: "List title" },
358
+ items: {
359
+ type: "array",
360
+ items: {
361
+ type: "object",
362
+ properties: {
363
+ key: { type: "string" },
364
+ value: { type: "string" },
365
+ },
366
+ },
367
+ },
368
+ span: {
369
+ type: "string",
370
+ enum: ["full", "half"],
371
+ description: "Width span: 'full' for full-width, 'half' for half width. Default: single column",
372
+ },
373
+ },
374
+ required: ["items"],
375
+ },
376
+ execute: async (params: Record<string, unknown>, config?: any) => {
377
+ const userId = config?.configurable?.user_id;
378
+ const rawSessionId = params.sessionId as string;
379
+ const sessionId = rawSessionId
380
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
381
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
382
+ const title = (params.title as string) ?? "Details";
383
+ const items = (params.items as Array<{ key: string; value: string }>) ?? [];
384
+ const span = params.span as "full" | "half" | undefined;
385
+
386
+ const listId = `list-${Date.now()}`;
387
+
388
+ await canvasManager.render(sessionId, {
389
+ id: listId,
390
+ type: "list",
391
+ props: { title, items },
392
+ span,
393
+ });
394
+
395
+ return { success: true, listId, sessionId };
396
+ },
397
+ };
398
+ }
399
+
400
+ export function createCanvasConfirmTool(_config: Config): Tool {
401
+ const log = logger.child("canvas-confirm");
402
+
403
+ return {
404
+ name: "canvas_confirm",
405
+ description: "Show a confirmation dialog and wait for user response",
406
+ parameters: {
407
+ type: "object",
408
+ properties: {
409
+ sessionId: { type: "string", description: "Session ID" },
410
+ title: { type: "string", description: "Dialog title" },
411
+ message: { type: "string", description: "Confirmation message" },
412
+ confirmLabel: { type: "string", description: "Confirm button label" },
413
+ cancelLabel: { type: "string", description: "Cancel button label" },
414
+ danger: { type: "boolean", description: "Show as dangerous action" },
415
+ timeout: { type: "number", description: "Timeout in ms (default: 60000)" },
416
+ },
417
+ required: ["message"],
418
+ },
419
+ execute: async (params: Record<string, unknown>, config?: any) => {
420
+ const userId = config?.configurable?.user_id;
421
+ const rawSessionId = params.sessionId as string;
422
+ const sessionId = rawSessionId
423
+ ? (rawSessionId.startsWith("canvas:") ? rawSessionId : `canvas:${rawSessionId}`)
424
+ : (userId ? `canvas:${userId}` : (() => { throw new Error("No session or user ID provided"); })());
425
+ const title = (params.title as string) ?? "Confirm";
426
+ const message = params.message as string;
427
+ const confirmLabel = (params.confirmLabel as string) ?? "Confirm";
428
+ const cancelLabel = (params.cancelLabel as string) ?? "Cancel";
429
+ const danger = (params.danger as boolean) ?? false;
430
+ const timeout = (params.timeout as number) ?? 60000;
431
+
432
+ const confirmId = `confirm-${Date.now()}`;
433
+
434
+ await canvasManager.render(sessionId, {
435
+ id: confirmId,
436
+ type: "confirm",
437
+ props: { title, message, confirmLabel, cancelLabel, danger },
438
+ });
439
+
440
+ try {
441
+ const response = await canvasManager.waitForInteraction(sessionId, confirmId, timeout);
442
+ return { success: true, confirmed: response === true, confirmId, sessionId };
443
+ } catch (error) {
444
+ return { success: false, confirmed: false, confirmId, error: (error as Error).message, sessionId };
445
+ }
446
+ },
447
+ };
448
+ }
@@ -0,0 +1,149 @@
1
+ import { getDb } from "../storage/SQLiteStorage.ts"
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
+ | "canvas:render"
17
+ | "canvas:ask"
18
+ | "canvas:confirm"
19
+ | "canvas:clear"
20
+ | "ag-ui:event"
21
+
22
+ const subscribers = new Set<{ send: (data: string) => void }>()
23
+
24
+ interface AgentLiveState { status: string; currentTool: string | null }
25
+ const agentLiveState = new Map<string, AgentLiveState>()
26
+ const canvasComponents = new Map<string, unknown>()
27
+
28
+ export function subscribeCanvas(ws: { send: (data: string) => void }) {
29
+ subscribers.add(ws)
30
+ }
31
+
32
+ export function unsubscribeCanvas(ws: { send: (data: string) => void }) {
33
+ subscribers.delete(ws)
34
+ }
35
+
36
+ export function emitCanvas(type: CanvasEventType, data: any) {
37
+ // Track canvas components for new subscribers
38
+ if (type === "canvas:render" && data?.component?.id) {
39
+ canvasComponents.set(data.component.id, data.component)
40
+ }
41
+ if (type === "canvas:clear") {
42
+ canvasComponents.clear()
43
+ }
44
+
45
+ // Track live agent state for new subscribers
46
+ if (type === "canvas:node_update" && data?.nodeId && data?.changes) {
47
+ const prev = agentLiveState.get(data.nodeId) ?? { status: "idle", currentTool: null }
48
+ agentLiveState.set(data.nodeId, {
49
+ status: data.changes.status ?? prev.status,
50
+ currentTool: "currentTool" in data.changes ? data.changes.currentTool : prev.currentTool,
51
+ })
52
+ }
53
+
54
+ const event: CanvasEvent = { type, data, timestamp: Date.now() }
55
+ const payload = JSON.stringify(event)
56
+ for (const ws of subscribers) {
57
+ try {
58
+ ws.send(payload)
59
+ } catch {
60
+ subscribers.delete(ws)
61
+ }
62
+ }
63
+ }
64
+
65
+ export function removeCanvasComponent(id: string) {
66
+ canvasComponents.delete(id);
67
+ }
68
+
69
+ export function getCanvasSnapshot() {
70
+ const db = getDb()
71
+
72
+ const agentNodes = db
73
+ .query<any, []>("SELECT id, name, description, role, status FROM agents")
74
+ .all()
75
+ .map((a: any) => {
76
+ const live = agentLiveState.get(a.id)
77
+ return {
78
+ id: a.id,
79
+ name: a.name,
80
+ description: a.description,
81
+ status: live?.status ?? a.status,
82
+ type: "agent",
83
+ data: { role: a.role, currentTool: live?.currentTool ?? null },
84
+ }
85
+ })
86
+
87
+ const mcpNodes = db
88
+ .query<any, []>("SELECT id, name, status FROM mcp_servers WHERE enabled = 1")
89
+ .all()
90
+ .map((m: any) => ({
91
+ id: `mcp:${m.id}`,
92
+ name: m.name,
93
+ status: m.status,
94
+ type: "mcp",
95
+ }))
96
+
97
+ // Proyectos activos
98
+ const projectNodes = db
99
+ .query<any, []>("SELECT id, name, type, status, progress, agent_id FROM projects WHERE status IN ('active','pending','paused')")
100
+ .all()
101
+ .map((p: any) => ({
102
+ id: `project_${p.id}`,
103
+ name: p.name,
104
+ status: p.status,
105
+ type: "project",
106
+ data: { progress: p.progress, projectType: p.type, agentId: p.agent_id },
107
+ }))
108
+
109
+ // Tareas de proyectos activos
110
+ const taskNodes = db
111
+ .query<any, []>(`
112
+ SELECT t.id, t.name, t.status, t.progress, t.agent_id, t.project_id
113
+ FROM tasks t
114
+ INNER JOIN projects p ON t.project_id = p.id
115
+ WHERE p.status IN ('active','pending','paused')
116
+ `)
117
+ .all()
118
+ .map((t: any) => ({
119
+ id: `task_${t.id}`,
120
+ name: t.name,
121
+ status: t.status,
122
+ type: "task",
123
+ data: { progress: t.progress, agentId: t.agent_id, projectId: t.project_id },
124
+ }))
125
+
126
+ // Edges: proyecto → tarea
127
+ const projectTaskEdges = taskNodes.map((t: any) => ({
128
+ id: `edge_proj_task_${t.id.replace("task_", "")}`,
129
+ source: `project_${t.data.projectId}`,
130
+ target: t.id,
131
+ edgeType: "contains",
132
+ }))
133
+
134
+ // Edges: tarea → agente asignado
135
+ const taskAgentEdges = taskNodes
136
+ .filter((t: any) => t.data.agentId)
137
+ .map((t: any) => ({
138
+ id: `edge_task_agent_${t.id.replace("task_", "")}`,
139
+ source: t.id,
140
+ target: t.data.agentId,
141
+ edgeType: "assigned_to",
142
+ }))
143
+
144
+ return {
145
+ nodes: [...agentNodes, ...mcpNodes, ...projectNodes, ...taskNodes],
146
+ edges: [...projectTaskEdges, ...taskAgentEdges],
147
+ components: Array.from(canvasComponents.values()),
148
+ }
149
+ }
@@ -0,0 +1,6 @@
1
+ export { createCanvasRenderTool, createCanvasAskTool, createCanvasClearTool, createCanvasTools, createCanvasCardTool, createCanvasProgressTool, createCanvasListTool, createCanvasConfirmTool } from "./canvas-tools.ts";
2
+ export { createA2UISurfaceTool, createA2UIUpdateComponentsTool, createA2UIUpdateDataModelTool, createA2UIDeleteSurfaceTool } from "./a2ui-tools.ts";
3
+ export type { CanvasEvent, CanvasEventType } from "./emitter.ts";
4
+ export { subscribeCanvas, unsubscribeCanvas, emitCanvas, getCanvasSnapshot, removeCanvasComponent } from "./emitter.ts";
5
+ export type { WebSocketLike, CanvasComponent, CanvasMessage, InteractionEvent } from "./CanvasManager.ts";
6
+ export { WebSocketState, CanvasManager, canvasManager } from "./CanvasManager.ts";
@@ -0,0 +1,2 @@
1
+ export { loadEnv, getHiveDir, loadConfig } from "./loader.ts";
2
+ export type { Config } from "./loader.ts";