@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,58 @@
1
+ {
2
+ "name": "@hive/core",
3
+ "version": "0.0.15",
4
+ "description": "Hive Core — Agentes AI con Context Engineering, FTS5, ACE, Swarm",
5
+ "license": "MIT",
6
+ "type": "module",
7
+ "main": "./src/index.ts",
8
+ "types": "./src/index.ts",
9
+ "exports": {
10
+ ".": "./src/index.ts",
11
+ "./agent": "./src/agent/index.ts",
12
+ "./agent/providers": "./src/agent/providers/index.ts",
13
+ "./agent/selectors": "./src/agent/selectors/index.ts",
14
+ "./tools": "./src/tools/index.ts",
15
+ "./skills": "./src/skills/index.ts",
16
+ "./storage": "./src/storage/index.ts",
17
+ "./swarm": "./src/swarm/index.ts",
18
+ "./swarm/strategies": "./src/swarm/strategies/index.ts",
19
+ "./swarm/presets": "./src/swarm/presets/index.ts",
20
+ "./ace": "./src/ace/index.ts",
21
+ "./ethics": "./src/ethics/index.ts",
22
+ "./canvas": "./src/canvas/index.ts",
23
+ "./config": "./src/config/index.ts",
24
+ "./mcp": "./src/mcp/index.ts",
25
+ "./mcp/transports": "./src/mcp/transports/index.ts",
26
+ "./memory": "./src/memory/index.ts",
27
+ "./multimodal": "./src/multimodal/index.ts",
28
+ "./security": "./src/security/index.ts",
29
+ "./state": "./src/state/index.ts",
30
+ "./utils": "./src/utils/index.ts",
31
+ "./gateway": "./src/gateway/index.ts",
32
+ "./api": "./src/api/index.ts"
33
+ },
34
+ "dependencies": {
35
+ "@anthropic-ai/sdk": "^0.74.0",
36
+ "@google/genai": "^1.43.0",
37
+ "@modelcontextprotocol/sdk": "latest",
38
+ "@sapphire/snowflake": "latest",
39
+ "async-mutex": "^0.5.0",
40
+ "cron-parser": "^5.5.0",
41
+ "croner": "^10.0.1",
42
+ "docx": "^9.6.1",
43
+ "groq-sdk": "^0.37.0",
44
+ "jszip": "^3.10.1",
45
+ "pdf-lib": "^1.17.1",
46
+ "mammoth": "^1.12.0",
47
+ "ollama": "^0.6.3",
48
+ "openai": "^6.18.0",
49
+ "pdfjs-dist": "^5.6.205",
50
+ "pptxgenjs": "^4.0.1",
51
+ "toon-format-parser": "^1.1.0",
52
+ "xlsx": "^0.18.5",
53
+ "zod": "latest"
54
+ },
55
+ "peerDependencies": {
56
+ "typescript": "^5.0.0"
57
+ }
58
+ }
@@ -0,0 +1,158 @@
1
+ /**
2
+ * ACE Curator — converts reflections into playbook rules.
3
+ *
4
+ * Runs after the Reflector. Performs incremental edits to the playbook:
5
+ * - New insights → new rules
6
+ * - Repeated patterns → increment helpful_count
7
+ * - Contradicted rules → increment harmful_count or deactivate
8
+ * - Deactivate rules where harmful_count > helpful_count
9
+ * - Archive unused workers
10
+ *
11
+ * Never rewrites the whole playbook — only incremental edits.
12
+ */
13
+
14
+ import { logger } from "../utils/logger.ts"
15
+
16
+ const log = logger.child("curator")
17
+
18
+ const DAYS_BEFORE_ARCHIVE = 14 // archive workers not used in N days
19
+ const MAX_HARMFUL_BEFORE_PRUNE = 3
20
+
21
+ /** Entry point — called by reflector.ts after it inserts new reflections */
22
+ export async function runCurator(): Promise<void> {
23
+ try {
24
+ const { getDb } = await import("../storage/SQLiteStorage.ts")
25
+ const db = getDb()
26
+
27
+ // Process unprocessed reflections (those newer than last run)
28
+ const lastProcessed = (db.query<any, []>(
29
+ "SELECT COALESCE(MAX(source_reflection_id), 0) as mid FROM playbook"
30
+ ).get() as any)?.mid ?? 0
31
+
32
+ const reflections = (db.query as any)(
33
+ "SELECT * FROM reflections WHERE id > ? ORDER BY id ASC"
34
+ ).all(lastProcessed)
35
+
36
+ if (reflections.length === 0) {
37
+ log.debug("[curator] No new reflections to process")
38
+ } else {
39
+ log.info(`[curator] Processing ${reflections.length} new reflections`)
40
+ for (const reflection of reflections) {
41
+ processReflection(db, reflection)
42
+ }
43
+ }
44
+
45
+ // Prune rules where harmful > helpful (consistently bad rules)
46
+ db.query(`
47
+ UPDATE playbook
48
+ SET active = 0, updated_at = unixepoch()
49
+ WHERE active = 1
50
+ AND harmful_count > helpful_count
51
+ AND harmful_count >= ?
52
+ `).run(MAX_HARMFUL_BEFORE_PRUNE)
53
+
54
+ // Archive unused workers
55
+ const cutoff = Math.floor(Date.now() / 1000) - (DAYS_BEFORE_ARCHIVE * 86400)
56
+ const staleworkers = (db.query as any)(`
57
+ SELECT a.id, a.name
58
+ FROM agents a
59
+ WHERE a.role = 'worker'
60
+ AND a.status != 'archived'
61
+ AND a.enabled = 1
62
+ AND (
63
+ SELECT MAX(t.created_at) FROM traces t WHERE t.agent_id = a.id
64
+ ) < ?
65
+ `).all(cutoff)
66
+
67
+ for (const worker of staleworkers) {
68
+ db.query(
69
+ "UPDATE agents SET status = 'archived', updated_at = unixepoch() WHERE id = ?"
70
+ ).run(worker.id)
71
+
72
+ // Add playbook note about archival
73
+ addOrUpdateRule(db, {
74
+ rule: `Worker '${worker.name}' was archived due to inactivity (>${DAYS_BEFORE_ARCHIVE} days unused).`,
75
+ category: "agent_creation",
76
+ applicable_to: null,
77
+ sourceReflectionId: null,
78
+ })
79
+
80
+ log.info(`[curator] Archived inactive worker: ${worker.name} (${worker.id})`)
81
+ }
82
+
83
+ log.info("[curator] Playbook updated")
84
+ } catch (err) {
85
+ log.warn("[curator] Error:", err)
86
+ }
87
+ }
88
+
89
+ // ─── Process a single reflection ─────────────────────────────────────────────
90
+
91
+ function processReflection(db: any, reflection: any): void {
92
+ const category = mapInsightTypeToCategory(reflection.insight_type)
93
+
94
+ // Check if a similar rule already exists (fuzzy check by first 60 chars)
95
+ const prefix = reflection.description.substring(0, 60)
96
+ const existing = (db.query as any)(
97
+ "SELECT id, helpful_count FROM playbook WHERE rule LIKE ? AND active = 1 LIMIT 1"
98
+ ).get(`${prefix}%`)
99
+
100
+ if (existing) {
101
+ // Reinforce existing rule
102
+ db.query(
103
+ "UPDATE playbook SET helpful_count = helpful_count + 1, updated_at = unixepoch() WHERE id = ?"
104
+ ).run(existing.id)
105
+ return
106
+ }
107
+
108
+ // Insert new rule
109
+ db.query(`
110
+ INSERT INTO playbook (rule, category, applicable_to, helpful_count, source_reflection_id)
111
+ VALUES (?, ?, ?, 1, ?)
112
+ `).run(
113
+ reflection.description,
114
+ category,
115
+ reflection.affected_tools
116
+ ? JSON.stringify(JSON.parse(reflection.affected_tools))
117
+ : null,
118
+ reflection.id,
119
+ )
120
+ }
121
+
122
+ function mapInsightTypeToCategory(
123
+ type: string
124
+ ): "tool_selection" | "response_quality" | "error_avoidance" | "optimization" | "agent_creation" {
125
+ const map: Record<string, any> = {
126
+ success_pattern: "tool_selection",
127
+ failure_pattern: "error_avoidance",
128
+ optimization: "optimization",
129
+ ethics_violation: "error_avoidance",
130
+ }
131
+ return map[type] ?? "optimization"
132
+ }
133
+
134
+ function addOrUpdateRule(
135
+ db: any,
136
+ opts: {
137
+ rule: string
138
+ category: string
139
+ applicable_to: string | null
140
+ sourceReflectionId: number | null
141
+ }
142
+ ): void {
143
+ const prefix = opts.rule.substring(0, 60)
144
+ const existing = (db.query as any)(
145
+ "SELECT id FROM playbook WHERE rule LIKE ? LIMIT 1"
146
+ ).get(`${prefix}%`)
147
+
148
+ if (existing) {
149
+ db.query(
150
+ "UPDATE playbook SET helpful_count = helpful_count + 1, updated_at = unixepoch() WHERE id = ?"
151
+ ).run(existing.id)
152
+ } else {
153
+ db.query(`
154
+ INSERT INTO playbook (rule, category, applicable_to, helpful_count, source_reflection_id)
155
+ VALUES (?, ?, ?, 1, ?)
156
+ `).run(opts.rule, opts.category, opts.applicable_to, opts.sourceReflectionId)
157
+ }
158
+ }
@@ -0,0 +1,200 @@
1
+ /**
2
+ * ACE Reflector — analyzes recent traces and produces insights.
3
+ *
4
+ * Runs in the background (never blocks the main agent loop).
5
+ * Triggered by the tracer after N new traces.
6
+ *
7
+ * Output → `reflections` table → picked up by Curator.
8
+ */
9
+
10
+ import { logger } from "../utils/logger.ts"
11
+
12
+ const log = logger.child("reflector")
13
+
14
+ const MAX_TRACES_TO_ANALYZE = 30
15
+ const MIN_TRACES_TO_RUN = 10
16
+
17
+ /** Main entry point — called from tracer.ts */
18
+ export async function runReflector(): Promise<void> {
19
+ try {
20
+ const { getDb } = await import("../storage/SQLiteStorage.ts")
21
+ const db = getDb()
22
+
23
+ log.info(`[reflector] Starting reflection cycle...`)
24
+
25
+ // Fetch traces not yet covered by any reflection
26
+ log.debug(`[reflector] Querying last reflection ID...`)
27
+ const lastReflectionId = (db.query<any, []>(
28
+ "SELECT MAX(id) as mid FROM reflections"
29
+ ).get() as any)?.mid ?? 0
30
+ log.debug(`[reflector] Last reflection ID: ${lastReflectionId}`)
31
+
32
+ // Get last processed trace ID from reflections
33
+ log.debug(`[reflector] Querying last processed trace ID with json_each...`)
34
+ let lastProcessedTrace = 0
35
+ try {
36
+ const result = (db.query<any, []>(
37
+ `SELECT MAX(CAST(json_each.value AS INTEGER)) as max_id
38
+ FROM reflections, json_each(reflections.trace_ids)
39
+ WHERE reflections.id = (SELECT MAX(id) FROM reflections)`
40
+ ).get() as any)?.max_id ?? 0
41
+ lastProcessedTrace = result
42
+ log.debug(`[reflector] Last processed trace ID: ${lastProcessedTrace}`)
43
+ } catch (jsonErr) {
44
+ log.error(`[reflector] json_each query failed:`, jsonErr)
45
+ log.error(`[reflector] Full error details:`, {
46
+ message: (jsonErr as Error).message,
47
+ stack: (jsonErr as Error).stack,
48
+ errno: (jsonErr as any).errno,
49
+ byteOffset: (jsonErr as any).byteOffset,
50
+ })
51
+ throw jsonErr // Re-throw to see full stack trace in main catch
52
+ }
53
+
54
+ log.debug(`[reflector] Fetching traces from DB, lastProcessedTrace=${lastProcessedTrace}, limit=${MAX_TRACES_TO_ANALYZE}`)
55
+ const traces = db.query<any, [number, number]>(`
56
+ SELECT id, agent_id, agent_name, tool_used, input_summary,
57
+ output_summary, success, error_message, duration_ms, tokens_used, created_at
58
+ FROM traces
59
+ WHERE id > ?
60
+ ORDER BY id ASC
61
+ LIMIT ?
62
+ `).all(lastProcessedTrace, MAX_TRACES_TO_ANALYZE)
63
+
64
+ log.debug(`[reflector] Fetched ${traces.length} traces from DB`)
65
+
66
+ if (traces.length < MIN_TRACES_TO_RUN) {
67
+ log.debug(`[reflector] Not enough traces (${traces.length}/${MIN_TRACES_TO_RUN}), skipping`)
68
+ return
69
+ }
70
+
71
+ log.info(`[reflector] Analyzing ${traces.length} traces...`)
72
+
73
+ const insights = analyzeTracesLocally(traces)
74
+
75
+ if (insights.length === 0) {
76
+ log.debug("[reflector] No insights generated")
77
+ return
78
+ }
79
+
80
+ const traceIds = JSON.stringify(traces.map((t: any) => t.id))
81
+
82
+ for (const insight of insights) {
83
+ db.query(`
84
+ INSERT INTO reflections (trace_ids, insight_type, description, affected_tools, affected_agents, confidence)
85
+ VALUES (?, ?, ?, ?, ?, ?)
86
+ `).run(
87
+ traceIds,
88
+ insight.type,
89
+ insight.description,
90
+ insight.affectedTools ? JSON.stringify(insight.affectedTools) : null,
91
+ insight.affectedAgents ? JSON.stringify(insight.affectedAgents) : null,
92
+ insight.confidence,
93
+ )
94
+ }
95
+
96
+ log.info(`[reflector] Generated ${insights.length} insights`)
97
+
98
+ // Trigger curator
99
+ const { runCurator } = await import("./Curator.ts")
100
+ await runCurator()
101
+
102
+ log.info(`[reflector] Reflection cycle completed successfully`)
103
+ } catch (err) {
104
+ log.error(`[reflector] Error during reflection:`, {
105
+ message: (err as Error).message,
106
+ stack: (err as Error).stack,
107
+ errno: (err as any).errno,
108
+ byteOffset: (err as any).byteOffset,
109
+ code: (err as any).code,
110
+ })
111
+ }
112
+ }
113
+
114
+ // ─── Local analysis (heuristic, no LLM call needed for basic patterns) ────────
115
+
116
+ interface Insight {
117
+ type: "success_pattern" | "failure_pattern" | "optimization" | "ethics_violation"
118
+ description: string
119
+ affectedTools?: string[]
120
+ affectedAgents?: string[]
121
+ confidence: number
122
+ }
123
+
124
+ function analyzeTracesLocally(traces: any[]): Insight[] {
125
+ const insights: Insight[] = []
126
+
127
+ // ── Failure patterns ─────────────────────────────────────────────────────
128
+ const failures = traces.filter((t: any) => !t.success)
129
+ if (failures.length > 3) {
130
+ // Group by tool
131
+ const toolFailures: Record<string, number> = {}
132
+ for (const f of failures) {
133
+ if (f.tool_used) {
134
+ toolFailures[f.tool_used] = (toolFailures[f.tool_used] || 0) + 1
135
+ }
136
+ }
137
+ for (const [tool, count] of Object.entries(toolFailures)) {
138
+ if (count >= 3) {
139
+ insights.push({
140
+ type: "failure_pattern",
141
+ description: `Tool '${tool}' failed ${count} times recently. Consider verifying its configuration or avoiding it for this type of task.`,
142
+ affectedTools: [tool],
143
+ confidence: Math.min(0.9, count / 10),
144
+ })
145
+ }
146
+ }
147
+ }
148
+
149
+ // ── Slow tools ───────────────────────────────────────────────────────────
150
+ const slowThresholdMs = 5000
151
+ const slowTools: Record<string, number[]> = {}
152
+ for (const t of traces) {
153
+ if (t.tool_used && t.duration_ms > slowThresholdMs) {
154
+ if (!slowTools[t.tool_used]) slowTools[t.tool_used] = []
155
+ slowTools[t.tool_used].push(t.duration_ms)
156
+ }
157
+ }
158
+ for (const [tool, durations] of Object.entries(slowTools)) {
159
+ if (durations.length >= 3) {
160
+ const avg = Math.round(durations.reduce((a, b) => a + b, 0) / durations.length)
161
+ insights.push({
162
+ type: "optimization",
163
+ description: `Tool '${tool}' is consistently slow (avg ${avg}ms). Cache results when possible or use a faster alternative.`,
164
+ affectedTools: [tool],
165
+ confidence: 0.6,
166
+ })
167
+ }
168
+ }
169
+
170
+ // ── High success pattern ─────────────────────────────────────────────────
171
+ const successByTool: Record<string, { ok: number; total: number }> = {}
172
+ for (const t of traces) {
173
+ if (!t.tool_used) continue
174
+ if (!successByTool[t.tool_used]) successByTool[t.tool_used] = { ok: 0, total: 0 }
175
+ successByTool[t.tool_used].total++
176
+ if (t.success) successByTool[t.tool_used].ok++
177
+ }
178
+ for (const [tool, stats] of Object.entries(successByTool)) {
179
+ if (stats.total >= 5 && stats.ok / stats.total >= 0.9) {
180
+ insights.push({
181
+ type: "success_pattern",
182
+ description: `Tool '${tool}' has a high success rate (${stats.ok}/${stats.total}). Prefer it for related tasks.`,
183
+ affectedTools: [tool],
184
+ confidence: stats.ok / stats.total,
185
+ })
186
+ }
187
+ }
188
+
189
+ // ── High token usage ─────────────────────────────────────────────────────
190
+ const highTokenTraces = traces.filter((t: any) => t.tokens_used > 4000)
191
+ if (highTokenTraces.length > 3) {
192
+ insights.push({
193
+ type: "optimization",
194
+ description: `${highTokenTraces.length} recent calls used >4000 tokens. Be more concise and use tool results as summaries, not raw dumps.`,
195
+ confidence: 0.7,
196
+ })
197
+ }
198
+
199
+ return insights
200
+ }
@@ -0,0 +1,100 @@
1
+ /**
2
+ * Tracer — ACE Generator output.
3
+ *
4
+ * Records every agent execution to the `traces` table.
5
+ * Fire-and-forget (non-blocking). Also updates playbook helpful/harmful counts
6
+ * based on execution outcome.
7
+ */
8
+
9
+ import { logger } from "../utils/logger.ts"
10
+
11
+ const log = logger.child("tracer")
12
+
13
+ export interface TraceInput {
14
+ threadId: string
15
+ agentId: string
16
+ agentName: string
17
+ toolUsed?: string | null
18
+ inputSummary: string
19
+ outputSummary: string
20
+ success: boolean
21
+ errorMessage?: string | null
22
+ durationMs?: number
23
+ tokensUsed?: number
24
+ }
25
+
26
+ /**
27
+ * Save a trace record. Non-blocking — errors are swallowed so they never
28
+ * affect the main agent loop.
29
+ */
30
+ export function saveTrace(trace: TraceInput): void {
31
+ // Run asynchronously so it never blocks the caller
32
+ Promise.resolve().then(async () => {
33
+ try {
34
+ const { getDb } = await import("../storage/SQLiteStorage.ts")
35
+ const db = getDb()
36
+
37
+ db.query(`
38
+ INSERT INTO traces
39
+ (thread_id, agent_id, agent_name, tool_used, input_summary,
40
+ output_summary, success, error_message, duration_ms, tokens_used)
41
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
42
+ `).run(
43
+ trace.threadId,
44
+ trace.agentId,
45
+ trace.agentName,
46
+ trace.toolUsed ?? null,
47
+ trace.inputSummary.substring(0, 500),
48
+ trace.outputSummary.substring(0, 500),
49
+ trace.success ? 1 : 0,
50
+ trace.errorMessage ?? null,
51
+ trace.durationMs ?? null,
52
+ trace.tokensUsed ?? null,
53
+ )
54
+
55
+ // Trigger reflector check in background
56
+ checkReflectorTrigger().catch(() => { /* ignore */ })
57
+ } catch (err) {
58
+ log.warn("[tracer] Failed to save trace:", err)
59
+ }
60
+ })
61
+ }
62
+
63
+ // ─── Reflector trigger ────────────────────────────────────────────────────────
64
+
65
+ const REFLECTOR_TRACE_THRESHOLD = 20 // run reflector after N new traces
66
+
67
+ let _tracesSinceLastReflection = 0
68
+
69
+ async function checkReflectorTrigger(): Promise<void> {
70
+ _tracesSinceLastReflection++
71
+ if (_tracesSinceLastReflection < REFLECTOR_TRACE_THRESHOLD) return
72
+ _tracesSinceLastReflection = 0
73
+
74
+ // Lazy import to avoid circular deps
75
+ const { runReflector } = await import("./Reflector.ts")
76
+ runReflector().catch((err) => {
77
+ log.warn("[tracer] Reflector run failed:", err)
78
+ })
79
+ }
80
+
81
+ // ─── Usage recording ──────────────────────────────────────────────────────────
82
+
83
+ export function recordLLMUsage(opts: {
84
+ provider: string
85
+ model: string
86
+ inputTokens: number
87
+ outputTokens: number
88
+ }): void {
89
+ Promise.resolve().then(async () => {
90
+ try {
91
+ const { recordUsage } = await import("../storage/usage.ts")
92
+ recordUsage({
93
+ provider: opts.provider,
94
+ model: opts.model,
95
+ inputTokens: opts.inputTokens,
96
+ outputTokens: opts.outputTokens,
97
+ })
98
+ } catch { /* ignore */ }
99
+ })
100
+ }
@@ -0,0 +1,4 @@
1
+ export { runCurator } from "./Curator.ts";
2
+ export { runReflector } from "./Reflector.ts";
3
+ export type { TraceInput } from "./Tracer.ts";
4
+ export { saveTrace } from "./Tracer.ts";