@tyvm/knowhow 0.0.82 → 0.0.84

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 (210) hide show
  1. package/package.json +4 -2
  2. package/src/agents/base/base.ts +72 -62
  3. package/src/agents/index.ts +30 -14
  4. package/src/agents/tools/startAgentTask.ts +3 -1
  5. package/src/chat/CliChatService.ts +20 -4
  6. package/src/chat/modules/AgentModule.ts +399 -357
  7. package/src/chat/modules/CustomCommandsModule.ts +0 -1
  8. package/src/chat/modules/InternalChatModule.ts +18 -2
  9. package/src/chat/modules/RendererModule.ts +109 -0
  10. package/src/chat/modules/SessionsModule.ts +854 -0
  11. package/src/chat/modules/SetupModule.ts +6 -8
  12. package/src/chat/modules/index.ts +1 -0
  13. package/src/chat/renderer/CompactRenderer.ts +209 -0
  14. package/src/chat/renderer/ConsoleRenderer.ts +141 -0
  15. package/src/chat/renderer/FancyRenderer.ts +421 -0
  16. package/src/chat/renderer/index.ts +5 -0
  17. package/src/chat/renderer/loadRenderer.ts +314 -0
  18. package/src/chat/renderer/messagesToRenderEvents.ts +96 -0
  19. package/src/chat/renderer/types.ts +88 -0
  20. package/src/chat/types.ts +5 -0
  21. package/src/chat.ts +69 -5
  22. package/src/cli.ts +24 -5
  23. package/src/config.ts +15 -0
  24. package/src/plugins/AgentsMdPlugin.ts +1 -1
  25. package/src/plugins/GitPlugin.ts +20 -20
  26. package/src/plugins/PluginBase.ts +11 -0
  27. package/src/plugins/SkillsPlugin.ts +150 -0
  28. package/src/plugins/asana.ts +4 -4
  29. package/src/plugins/embedding.ts +3 -5
  30. package/src/plugins/exec.ts +3 -3
  31. package/src/plugins/figma.ts +3 -7
  32. package/src/plugins/github.ts +18 -29
  33. package/src/plugins/jira.ts +2 -2
  34. package/src/plugins/language.ts +4 -4
  35. package/src/plugins/linear.ts +4 -4
  36. package/src/plugins/notion.ts +6 -8
  37. package/src/plugins/plugins.ts +30 -4
  38. package/src/plugins/url.ts +2 -2
  39. package/src/plugins/vim.ts +4 -3
  40. package/src/services/AgentService.ts +17 -0
  41. package/src/services/AgentSyncFs.ts +3 -0
  42. package/src/services/EventService.ts +168 -27
  43. package/src/services/KnowhowClient.ts +1 -0
  44. package/src/services/SessionManager.ts +51 -1
  45. package/src/services/SyncedAgentWatcher.ts +397 -0
  46. package/src/services/SyncerService.ts +147 -0
  47. package/src/services/index.ts +2 -0
  48. package/src/services/modules/index.ts +14 -3
  49. package/src/types.ts +25 -0
  50. package/src/worker.ts +80 -2
  51. package/src/workers/auth/PasskeySetup.ts +185 -0
  52. package/src/workers/auth/WorkerPasskeyAuth.ts +190 -0
  53. package/src/workers/auth/types.ts +58 -0
  54. package/src/workers/tools/getChallenge.ts +33 -0
  55. package/src/workers/tools/index.ts +8 -0
  56. package/src/workers/tools/lock.ts +31 -0
  57. package/src/workers/tools/unlock.ts +116 -0
  58. package/tests/unit/modules/moduleLoading.test.ts +226 -0
  59. package/tests/unit/plugins/pluginLoading.test.ts +151 -0
  60. package/ts_build/package.json +4 -2
  61. package/ts_build/src/agents/base/base.d.ts +4 -3
  62. package/ts_build/src/agents/base/base.js +54 -30
  63. package/ts_build/src/agents/base/base.js.map +1 -1
  64. package/ts_build/src/agents/index.d.ts +3 -0
  65. package/ts_build/src/agents/index.js +21 -11
  66. package/ts_build/src/agents/index.js.map +1 -1
  67. package/ts_build/src/agents/tools/startAgentTask.js +2 -1
  68. package/ts_build/src/agents/tools/startAgentTask.js.map +1 -1
  69. package/ts_build/src/chat/CliChatService.js +16 -5
  70. package/ts_build/src/chat/CliChatService.js.map +1 -1
  71. package/ts_build/src/chat/modules/AgentModule.d.ts +34 -17
  72. package/ts_build/src/chat/modules/AgentModule.js +248 -258
  73. package/ts_build/src/chat/modules/AgentModule.js.map +1 -1
  74. package/ts_build/src/chat/modules/CustomCommandsModule.js.map +1 -1
  75. package/ts_build/src/chat/modules/InternalChatModule.d.ts +3 -0
  76. package/ts_build/src/chat/modules/InternalChatModule.js +16 -1
  77. package/ts_build/src/chat/modules/InternalChatModule.js.map +1 -1
  78. package/ts_build/src/chat/modules/RendererModule.d.ts +16 -0
  79. package/ts_build/src/chat/modules/RendererModule.js +76 -0
  80. package/ts_build/src/chat/modules/RendererModule.js.map +1 -0
  81. package/ts_build/src/chat/modules/SessionsModule.d.ts +33 -0
  82. package/ts_build/src/chat/modules/SessionsModule.js +582 -0
  83. package/ts_build/src/chat/modules/SessionsModule.js.map +1 -0
  84. package/ts_build/src/chat/modules/SetupModule.d.ts +3 -3
  85. package/ts_build/src/chat/modules/SetupModule.js +4 -6
  86. package/ts_build/src/chat/modules/SetupModule.js.map +1 -1
  87. package/ts_build/src/chat/modules/index.d.ts +1 -0
  88. package/ts_build/src/chat/modules/index.js +3 -1
  89. package/ts_build/src/chat/modules/index.js.map +1 -1
  90. package/ts_build/src/chat/renderer/CompactRenderer.d.ts +23 -0
  91. package/ts_build/src/chat/renderer/CompactRenderer.js +167 -0
  92. package/ts_build/src/chat/renderer/CompactRenderer.js.map +1 -0
  93. package/ts_build/src/chat/renderer/ConsoleRenderer.d.ts +22 -0
  94. package/ts_build/src/chat/renderer/ConsoleRenderer.js +110 -0
  95. package/ts_build/src/chat/renderer/ConsoleRenderer.js.map +1 -0
  96. package/ts_build/src/chat/renderer/FancyRenderer.d.ts +23 -0
  97. package/ts_build/src/chat/renderer/FancyRenderer.js +328 -0
  98. package/ts_build/src/chat/renderer/FancyRenderer.js.map +1 -0
  99. package/ts_build/src/chat/renderer/index.d.ts +5 -0
  100. package/ts_build/src/chat/renderer/index.js +29 -0
  101. package/ts_build/src/chat/renderer/index.js.map +1 -0
  102. package/ts_build/src/chat/renderer/loadRenderer.d.ts +4 -0
  103. package/ts_build/src/chat/renderer/loadRenderer.js +246 -0
  104. package/ts_build/src/chat/renderer/loadRenderer.js.map +1 -0
  105. package/ts_build/src/chat/renderer/messagesToRenderEvents.d.ts +15 -0
  106. package/ts_build/src/chat/renderer/messagesToRenderEvents.js +72 -0
  107. package/ts_build/src/chat/renderer/messagesToRenderEvents.js.map +1 -0
  108. package/ts_build/src/chat/renderer/types.d.ts +75 -0
  109. package/ts_build/src/chat/renderer/types.js +3 -0
  110. package/ts_build/src/chat/renderer/types.js.map +1 -0
  111. package/ts_build/src/chat/types.d.ts +5 -0
  112. package/ts_build/src/chat.js +46 -4
  113. package/ts_build/src/chat.js.map +1 -1
  114. package/ts_build/src/cli.js +18 -5
  115. package/ts_build/src/cli.js.map +1 -1
  116. package/ts_build/src/config.d.ts +1 -0
  117. package/ts_build/src/config.js +17 -1
  118. package/ts_build/src/config.js.map +1 -1
  119. package/ts_build/src/plugins/AgentsMdPlugin.js +1 -1
  120. package/ts_build/src/plugins/AgentsMdPlugin.js.map +1 -1
  121. package/ts_build/src/plugins/GitPlugin.js +20 -20
  122. package/ts_build/src/plugins/GitPlugin.js.map +1 -1
  123. package/ts_build/src/plugins/PluginBase.d.ts +1 -0
  124. package/ts_build/src/plugins/PluginBase.js +13 -0
  125. package/ts_build/src/plugins/PluginBase.js.map +1 -1
  126. package/ts_build/src/plugins/SkillsPlugin.d.ts +13 -0
  127. package/ts_build/src/plugins/SkillsPlugin.js +149 -0
  128. package/ts_build/src/plugins/SkillsPlugin.js.map +1 -0
  129. package/ts_build/src/plugins/asana.js +4 -4
  130. package/ts_build/src/plugins/asana.js.map +1 -1
  131. package/ts_build/src/plugins/embedding.js +3 -3
  132. package/ts_build/src/plugins/embedding.js.map +1 -1
  133. package/ts_build/src/plugins/exec.js +3 -3
  134. package/ts_build/src/plugins/exec.js.map +1 -1
  135. package/ts_build/src/plugins/figma.js +3 -3
  136. package/ts_build/src/plugins/figma.js.map +1 -1
  137. package/ts_build/src/plugins/github.js +18 -18
  138. package/ts_build/src/plugins/github.js.map +1 -1
  139. package/ts_build/src/plugins/jira.js +2 -2
  140. package/ts_build/src/plugins/jira.js.map +1 -1
  141. package/ts_build/src/plugins/language.js +4 -4
  142. package/ts_build/src/plugins/language.js.map +1 -1
  143. package/ts_build/src/plugins/linear.js +4 -4
  144. package/ts_build/src/plugins/linear.js.map +1 -1
  145. package/ts_build/src/plugins/notion.js +6 -6
  146. package/ts_build/src/plugins/notion.js.map +1 -1
  147. package/ts_build/src/plugins/plugins.d.ts +3 -0
  148. package/ts_build/src/plugins/plugins.js +19 -4
  149. package/ts_build/src/plugins/plugins.js.map +1 -1
  150. package/ts_build/src/plugins/url.js +2 -2
  151. package/ts_build/src/plugins/url.js.map +1 -1
  152. package/ts_build/src/plugins/vim.js +2 -2
  153. package/ts_build/src/plugins/vim.js.map +1 -1
  154. package/ts_build/src/services/AgentService.d.ts +3 -0
  155. package/ts_build/src/services/AgentService.js +7 -0
  156. package/ts_build/src/services/AgentService.js.map +1 -1
  157. package/ts_build/src/services/AgentSyncFs.d.ts +1 -0
  158. package/ts_build/src/services/AgentSyncFs.js +2 -0
  159. package/ts_build/src/services/AgentSyncFs.js.map +1 -1
  160. package/ts_build/src/services/EventService.d.ts +25 -2
  161. package/ts_build/src/services/EventService.js +92 -14
  162. package/ts_build/src/services/EventService.js.map +1 -1
  163. package/ts_build/src/services/KnowhowClient.d.ts +1 -0
  164. package/ts_build/src/services/KnowhowClient.js.map +1 -1
  165. package/ts_build/src/services/SessionManager.d.ts +6 -0
  166. package/ts_build/src/services/SessionManager.js +39 -1
  167. package/ts_build/src/services/SessionManager.js.map +1 -1
  168. package/ts_build/src/services/SyncedAgentWatcher.d.ts +101 -0
  169. package/ts_build/src/services/SyncedAgentWatcher.js +312 -0
  170. package/ts_build/src/services/SyncedAgentWatcher.js.map +1 -0
  171. package/ts_build/src/services/SyncerService.d.ts +30 -0
  172. package/ts_build/src/services/SyncerService.js +72 -0
  173. package/ts_build/src/services/SyncerService.js.map +1 -0
  174. package/ts_build/src/services/index.d.ts +2 -0
  175. package/ts_build/src/services/index.js +2 -0
  176. package/ts_build/src/services/index.js.map +1 -1
  177. package/ts_build/src/services/modules/index.js +10 -2
  178. package/ts_build/src/services/modules/index.js.map +1 -1
  179. package/ts_build/src/types.d.ts +19 -0
  180. package/ts_build/src/types.js.map +1 -1
  181. package/ts_build/src/worker.d.ts +2 -0
  182. package/ts_build/src/worker.js +59 -4
  183. package/ts_build/src/worker.js.map +1 -1
  184. package/ts_build/src/workers/auth/PasskeySetup.d.ts +10 -0
  185. package/ts_build/src/workers/auth/PasskeySetup.js +131 -0
  186. package/ts_build/src/workers/auth/PasskeySetup.js.map +1 -0
  187. package/ts_build/src/workers/auth/WorkerPasskeyAuth.d.ts +35 -0
  188. package/ts_build/src/workers/auth/WorkerPasskeyAuth.js +129 -0
  189. package/ts_build/src/workers/auth/WorkerPasskeyAuth.js.map +1 -0
  190. package/ts_build/src/workers/auth/types.d.ts +36 -0
  191. package/ts_build/src/workers/auth/types.js +3 -0
  192. package/ts_build/src/workers/auth/types.js.map +1 -0
  193. package/ts_build/src/workers/tools/getChallenge.d.ts +9 -0
  194. package/ts_build/src/workers/tools/getChallenge.js +27 -0
  195. package/ts_build/src/workers/tools/getChallenge.js.map +1 -0
  196. package/ts_build/src/workers/tools/index.d.ts +6 -0
  197. package/ts_build/src/workers/tools/index.js +10 -0
  198. package/ts_build/src/workers/tools/index.js.map +1 -1
  199. package/ts_build/src/workers/tools/lock.d.ts +9 -0
  200. package/ts_build/src/workers/tools/lock.js +27 -0
  201. package/ts_build/src/workers/tools/lock.js.map +1 -0
  202. package/ts_build/src/workers/tools/unlock.d.ts +18 -0
  203. package/ts_build/src/workers/tools/unlock.js +78 -0
  204. package/ts_build/src/workers/tools/unlock.js.map +1 -0
  205. package/ts_build/tests/unit/modules/moduleLoading.test.d.ts +1 -0
  206. package/ts_build/tests/unit/modules/moduleLoading.test.js +187 -0
  207. package/ts_build/tests/unit/modules/moduleLoading.test.js.map +1 -0
  208. package/ts_build/tests/unit/plugins/pluginLoading.test.d.ts +1 -0
  209. package/ts_build/tests/unit/plugins/pluginLoading.test.js +123 -0
  210. package/ts_build/tests/unit/plugins/pluginLoading.test.js.map +1 -0
@@ -9,7 +9,12 @@ import { BaseChatModule } from "./BaseChatModule";
9
9
  export class SetupModule extends BaseChatModule {
10
10
  name = "setup";
11
11
  description = "Setup commands";
12
- private agentModule?: AgentModule;
12
+ private agentModule: AgentModule;
13
+
14
+ constructor(agentModule: AgentModule) {
15
+ super();
16
+ this.agentModule = agentModule;
17
+ }
13
18
 
14
19
  getCommands(): ChatCommand[] {
15
20
  return [
@@ -25,13 +30,6 @@ export class SetupModule extends BaseChatModule {
25
30
  return [];
26
31
  }
27
32
 
28
- async initialize(chatService: CliChatService): Promise<void> {
29
- await super.initialize(chatService);
30
- // We'll need access to AgentModule for launching Setup agent
31
- this.agentModule = new AgentModule();
32
- await this.agentModule.initialize(chatService);
33
- }
34
-
35
33
  /**
36
34
  * Check environment variables and show setup status
37
35
  */
@@ -4,3 +4,4 @@ export { AgentModule } from './AgentModule';
4
4
  export { VoiceModule } from './VoiceModule';
5
5
  export { InternalChatModule } from './InternalChatModule';
6
6
  export { ShellCommandModule } from './ShellCommandModule';
7
+ export { RendererModule } from './RendererModule';
@@ -0,0 +1,209 @@
1
+ /**
2
+ * CompactRenderer - A minimal, dense renderer for knowhow chat.
3
+ * Great for smaller terminals or when you want less visual noise.
4
+ *
5
+ * Activate via config:
6
+ * { "chat": { "renderer": "compact" } }
7
+ */
8
+
9
+ import {
10
+ AgentRenderer,
11
+ LogEvent,
12
+ AgentStatusEvent,
13
+ ToolCallEvent,
14
+ ToolResultEvent,
15
+ AgentMessageEvent,
16
+ AgentDoneEvent,
17
+ RenderEvent,
18
+ } from "./types";
19
+ import { EventEmitter } from "events";
20
+
21
+ const c = {
22
+ reset: "\x1b[0m",
23
+ bold: "\x1b[1m",
24
+ dim: "\x1b[2m",
25
+ red: "\x1b[31m",
26
+ green: "\x1b[32m",
27
+ yellow: "\x1b[33m",
28
+ cyan: "\x1b[36m",
29
+ gray: "\x1b[90m",
30
+ brightGreen: "\x1b[92m",
31
+ brightYellow: "\x1b[93m",
32
+ brightCyan: "\x1b[96m",
33
+ };
34
+
35
+ function truncate(s: string, n: number) {
36
+ return s.length > n ? s.slice(0, n - 1) + "…" : s;
37
+ }
38
+
39
+ function formatArgs(raw: string): string {
40
+ try {
41
+ const obj = JSON.parse(raw || "{}");
42
+ const entries = Object.entries(obj);
43
+ if (!entries.length) return "{}";
44
+ if (entries.length === 1) {
45
+ const [k, v] = entries[0];
46
+ return `${k}=${truncate(JSON.stringify(v), 60)}`;
47
+ }
48
+ return entries
49
+ .map(([k, v]) => `${k}=${truncate(JSON.stringify(v), 40)}`)
50
+ .join(", ");
51
+ } catch {
52
+ return truncate(raw, 80);
53
+ }
54
+ }
55
+
56
+ const timers = new Map<string, number>();
57
+ const agentPalette = [
58
+ "\x1b[96m",
59
+ "\x1b[95m",
60
+ "\x1b[92m",
61
+ "\x1b[93m",
62
+ "\x1b[94m",
63
+ ];
64
+ const agentColors = new Map<string, string>();
65
+ let pi = 0;
66
+ function colorFor(n: string) {
67
+ if (!agentColors.has(n))
68
+ agentColors.set(n, agentPalette[pi++ % agentPalette.length]);
69
+ return agentColors.get(n)!;
70
+ }
71
+
72
+ export class CompactRenderer implements AgentRenderer {
73
+ private activeTaskId: string | undefined;
74
+ private emitter = new EventEmitter();
75
+
76
+ setActiveTaskId(id: string | undefined): void {
77
+ this.activeTaskId = id;
78
+ }
79
+
80
+ getActiveTaskId(): string | undefined {
81
+ return this.activeTaskId;
82
+ }
83
+
84
+ private isActiveTask(taskId: string): boolean {
85
+ if (!this.activeTaskId) return true;
86
+ return taskId === this.activeTaskId;
87
+ }
88
+
89
+ onLog(handler: (event: LogEvent) => void): void {
90
+ this.emitter.on("log", handler);
91
+ }
92
+
93
+ onAgentStatus(handler: (event: AgentStatusEvent) => void): void {
94
+ this.emitter.on("agentStatus", handler);
95
+ }
96
+
97
+ onToolCall(handler: (event: ToolCallEvent) => void): void {
98
+ this.emitter.on("toolCall", handler);
99
+ }
100
+
101
+ onToolResult(handler: (event: ToolResultEvent) => void): void {
102
+ this.emitter.on("toolResult", handler);
103
+ }
104
+
105
+ onAgentMessage(handler: (event: AgentMessageEvent) => void): void {
106
+ this.emitter.on("agentMessage", handler);
107
+ }
108
+
109
+ onAgentDone(handler: (event: AgentDoneEvent) => void): void {
110
+ this.emitter.on("agentDone", handler);
111
+ }
112
+
113
+ render(event: RenderEvent): void {
114
+ if (!this.isActiveTask(event.taskId)) return;
115
+ this.emitter.emit(event.type, event);
116
+ switch (event.type) {
117
+ case "log":
118
+ return this._log(event as LogEvent);
119
+ case "agentStatus":
120
+ return this._status(event as AgentStatusEvent);
121
+ case "toolCall":
122
+ return this._toolCall(event as ToolCallEvent);
123
+ case "toolResult":
124
+ return this._toolResult(event as ToolResultEvent);
125
+ case "agentMessage":
126
+ return this._message(event as AgentMessageEvent);
127
+ case "agentDone":
128
+ return this._done(event as AgentDoneEvent);
129
+ }
130
+ }
131
+
132
+ private _log(e: LogEvent): void {
133
+ const tag =
134
+ { info: c.cyan + "i", warn: c.yellow + "!", error: c.red + "āœ—" }[
135
+ e.level
136
+ ] ?? c.gray + "Ā·";
137
+ process.stdout.write(`${tag}${c.reset} ${c.dim}${e.message}${c.reset}\n`);
138
+ }
139
+
140
+ private _status(e: AgentStatusEvent): void {
141
+ const col = colorFor(e.agentName);
142
+ const cost = `${c.brightGreen}$${e.details.totalCostUsd.toFixed(3)}${c.reset}`;
143
+ const elapsed = `${Math.floor(e.details.elapsedMs / 1000)}s`;
144
+ process.stdout.write(
145
+ `${c.dim}ā—${c.reset} ${col}${e.agentName}${c.reset} ${cost} ${elapsed}\n`
146
+ );
147
+ }
148
+
149
+ private _toolCall(e: ToolCallEvent): void {
150
+ timers.set(`${e.taskId}:${e.toolCall.function.name}`, Date.now());
151
+ const col = colorFor(e.agentName);
152
+ const args = formatArgs(e.toolCall.function.arguments);
153
+ process.stdout.write(
154
+ `${c.dim}ā–¶${c.reset} ${col}${c.bold}${e.toolCall.function.name}${c.reset} ${c.dim}(${args})${c.reset}\n`
155
+ );
156
+ }
157
+
158
+ private _toolResult(e: ToolResultEvent): void {
159
+ const name = e.toolCall.function.name;
160
+ const start = timers.get(`${e.taskId}:${name}`);
161
+ const ms = start ? `${Date.now() - start}ms` : "";
162
+ timers.delete(`${e.taskId}:${name}`);
163
+
164
+ let preview: string;
165
+ try {
166
+ const s =
167
+ typeof e.result === "string" ? e.result : JSON.stringify(e.result);
168
+ preview = truncate(s.replace(/\s+/g, " "), 120);
169
+ } catch {
170
+ preview = String(e.result).slice(0, 120);
171
+ }
172
+
173
+ process.stdout.write(
174
+ ` ${c.brightGreen}āœ“${c.reset} ${c.dim}${ms}${c.reset} ${c.dim}${preview}${c.reset}\n`
175
+ );
176
+ }
177
+
178
+ private _message(e: AgentMessageEvent): void {
179
+ if (e.role !== "assistant") return;
180
+ const col = colorFor(e.agentName);
181
+ const lines = e.message.split("\n");
182
+ const first = lines[0] ?? "";
183
+ process.stdout.write(`\n${col}${c.bold}${e.agentName}${c.reset} ${first}\n`);
184
+ for (const l of lines.slice(1)) {
185
+ if (l.trim()) process.stdout.write(` ${l}\n`);
186
+ }
187
+ }
188
+
189
+ private _done(e: AgentDoneEvent): void {
190
+ const col = colorFor(e.agentName);
191
+ const cost = e.totalCost
192
+ ? ` ${c.brightGreen}$${e.totalCost.toFixed(4)}${c.reset}`
193
+ : "";
194
+ process.stdout.write(`\n${col}āœ” ${e.agentName}${c.reset}${cost}\n\n`);
195
+ }
196
+
197
+ logMessages(events: RenderEvent[], count: number = 10): void {
198
+ const recent = events.slice(-count);
199
+ process.stdout.write(`\n[logs] Last ${recent.length} messages:\n`);
200
+ process.stdout.write("-".repeat(60) + "\n");
201
+ for (const event of recent) {
202
+ this.render(event);
203
+ }
204
+ process.stdout.write("-".repeat(60) + "\n");
205
+ }
206
+
207
+ }
208
+
209
+ export default CompactRenderer;
@@ -0,0 +1,141 @@
1
+ /**
2
+ * ConsoleRenderer - Default renderer that outputs to console
3
+ * Implements AgentRenderer interface with standard console.log behavior
4
+ */
5
+ import {
6
+ AgentRenderer,
7
+ LogEvent,
8
+ AgentStatusEvent,
9
+ ToolCallEvent,
10
+ ToolResultEvent,
11
+ AgentMessageEvent,
12
+ AgentDoneEvent,
13
+ RenderEvent,
14
+ } from "./types";
15
+ import { EventEmitter } from "events";
16
+
17
+ export class ConsoleRenderer implements AgentRenderer {
18
+ private activeTaskId: string | undefined;
19
+ private emitter = new EventEmitter();
20
+
21
+ setActiveTaskId(taskId: string | undefined): void {
22
+ this.activeTaskId = taskId;
23
+ }
24
+
25
+ getActiveTaskId(): string | undefined {
26
+ return this.activeTaskId;
27
+ }
28
+
29
+ private isActiveTask(taskId: string): boolean {
30
+ // If no active task is set, show all events
31
+ if (!this.activeTaskId) return true;
32
+ return taskId === this.activeTaskId;
33
+ }
34
+
35
+ onLog(handler: (event: LogEvent) => void): void {
36
+ this.emitter.on("log", handler);
37
+ }
38
+
39
+ onAgentStatus(handler: (event: AgentStatusEvent) => void): void {
40
+ this.emitter.on("agentStatus", handler);
41
+ }
42
+
43
+ onToolCall(handler: (event: ToolCallEvent) => void): void {
44
+ this.emitter.on("toolCall", handler);
45
+ }
46
+
47
+ onToolResult(handler: (event: ToolResultEvent) => void): void {
48
+ this.emitter.on("toolResult", handler);
49
+ }
50
+
51
+ onAgentMessage(handler: (event: AgentMessageEvent) => void): void {
52
+ this.emitter.on("agentMessage", handler);
53
+ }
54
+
55
+ onAgentDone(handler: (event: AgentDoneEvent) => void): void {
56
+ this.emitter.on("agentDone", handler);
57
+ }
58
+
59
+ render(event: RenderEvent): void {
60
+ if (!this.isActiveTask(event.taskId)) return;
61
+
62
+ switch (event.type) {
63
+ case "log":
64
+ this.emitter.emit("log", event);
65
+ this.renderLog(event);
66
+ break;
67
+ case "agentStatus":
68
+ this.emitter.emit("agentStatus", event);
69
+ this.renderAgentStatus(event);
70
+ break;
71
+ case "toolCall":
72
+ this.emitter.emit("toolCall", event);
73
+ this.renderToolCall(event);
74
+ break;
75
+ case "toolResult":
76
+ this.emitter.emit("toolResult", event);
77
+ this.renderToolResult(event);
78
+ break;
79
+ case "agentMessage":
80
+ this.emitter.emit("agentMessage", event);
81
+ this.renderAgentMessage(event);
82
+ break;
83
+ case "agentDone":
84
+ this.emitter.emit("agentDone", event);
85
+ this.renderAgentDone(event);
86
+ break;
87
+ }
88
+ }
89
+
90
+ logMessages(events: RenderEvent[], count: number = 20): void {
91
+ const recent = events.slice(-count);
92
+ console.log(`\nšŸ“œ Last ${recent.length} messages:`);
93
+ console.log("─".repeat(60));
94
+ for (const event of recent) {
95
+ this.render(event);
96
+ }
97
+ console.log("─".repeat(60));
98
+ }
99
+
100
+ private renderLog(event: LogEvent): void {
101
+ switch (event.level) {
102
+ case "warn":
103
+ console.warn(event.message);
104
+ break;
105
+ case "error":
106
+ console.error(event.message);
107
+ break;
108
+ default:
109
+ console.log(event.message);
110
+ }
111
+ }
112
+
113
+ private renderAgentStatus(event: AgentStatusEvent): void {
114
+ console.log(`\nā— ${event.agentName} status: ${event.statusMessage}`);
115
+ }
116
+
117
+ private renderToolCall(event: ToolCallEvent): void {
118
+ console.time(JSON.stringify(event.toolCall.function.name));
119
+ console.log(
120
+ ` šŸ”Ø Tool: ${event.toolCall.function.name}\n Args: ${event.toolCall.function.arguments}\n`
121
+ );
122
+ }
123
+
124
+ private renderToolResult(event: ToolResultEvent): void {
125
+ console.timeEnd(JSON.stringify(event.toolCall.function.name));
126
+ console.log(
127
+ ` šŸ”Ø Tool Response:\n ${JSON.stringify(event.result, null, 2)}`
128
+ );
129
+ }
130
+
131
+ private renderAgentMessage(event: AgentMessageEvent): void {
132
+ console.log(`[${event.agentName}]: ${event.message}`);
133
+ }
134
+
135
+ private renderAgentDone(event: AgentDoneEvent): void {
136
+ console.log(`šŸŽÆ Agent Done: ${event.agentName}`);
137
+ if (event.totalCost) {
138
+ console.log(`šŸ’° Total cost: $${event.totalCost.toFixed(4)}`);
139
+ }
140
+ }
141
+ }