@ai-devkit/agent-manager 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (57) hide show
  1. package/.eslintrc.json +31 -0
  2. package/dist/AgentManager.d.ts +104 -0
  3. package/dist/AgentManager.d.ts.map +1 -0
  4. package/dist/AgentManager.js +185 -0
  5. package/dist/AgentManager.js.map +1 -0
  6. package/dist/adapters/AgentAdapter.d.ts +76 -0
  7. package/dist/adapters/AgentAdapter.d.ts.map +1 -0
  8. package/dist/adapters/AgentAdapter.js +20 -0
  9. package/dist/adapters/AgentAdapter.js.map +1 -0
  10. package/dist/adapters/ClaudeCodeAdapter.d.ts +58 -0
  11. package/dist/adapters/ClaudeCodeAdapter.d.ts.map +1 -0
  12. package/dist/adapters/ClaudeCodeAdapter.js +274 -0
  13. package/dist/adapters/ClaudeCodeAdapter.js.map +1 -0
  14. package/dist/adapters/index.d.ts +4 -0
  15. package/dist/adapters/index.d.ts.map +1 -0
  16. package/dist/adapters/index.js +8 -0
  17. package/dist/adapters/index.js.map +1 -0
  18. package/dist/index.d.ts +10 -0
  19. package/dist/index.d.ts.map +1 -0
  20. package/dist/index.js +23 -0
  21. package/dist/index.js.map +1 -0
  22. package/dist/terminal/TerminalFocusManager.d.ts +22 -0
  23. package/dist/terminal/TerminalFocusManager.d.ts.map +1 -0
  24. package/dist/terminal/TerminalFocusManager.js +196 -0
  25. package/dist/terminal/TerminalFocusManager.js.map +1 -0
  26. package/dist/terminal/index.d.ts +3 -0
  27. package/dist/terminal/index.d.ts.map +1 -0
  28. package/dist/terminal/index.js +6 -0
  29. package/dist/terminal/index.js.map +1 -0
  30. package/dist/utils/file.d.ts +52 -0
  31. package/dist/utils/file.d.ts.map +1 -0
  32. package/dist/utils/file.js +135 -0
  33. package/dist/utils/file.js.map +1 -0
  34. package/dist/utils/index.d.ts +4 -0
  35. package/dist/utils/index.d.ts.map +1 -0
  36. package/dist/utils/index.js +15 -0
  37. package/dist/utils/index.js.map +1 -0
  38. package/dist/utils/process.d.ts +61 -0
  39. package/dist/utils/process.d.ts.map +1 -0
  40. package/dist/utils/process.js +166 -0
  41. package/dist/utils/process.js.map +1 -0
  42. package/jest.config.js +21 -0
  43. package/package.json +42 -0
  44. package/project.json +29 -0
  45. package/src/AgentManager.ts +198 -0
  46. package/src/__tests__/AgentManager.test.ts +308 -0
  47. package/src/__tests__/adapters/ClaudeCodeAdapter.test.ts +286 -0
  48. package/src/adapters/AgentAdapter.ts +94 -0
  49. package/src/adapters/ClaudeCodeAdapter.ts +344 -0
  50. package/src/adapters/index.ts +3 -0
  51. package/src/index.ts +12 -0
  52. package/src/terminal/TerminalFocusManager.ts +206 -0
  53. package/src/terminal/index.ts +2 -0
  54. package/src/utils/file.ts +100 -0
  55. package/src/utils/index.ts +3 -0
  56. package/src/utils/process.ts +184 -0
  57. package/tsconfig.json +17 -0
@@ -0,0 +1,198 @@
1
+ /**
2
+ * Agent Manager
3
+ *
4
+ * Orchestrates agent detection across multiple adapter types.
5
+ * Manages adapter registration and aggregates results from all adapters.
6
+ */
7
+
8
+ import type { AgentAdapter, AgentInfo } from './adapters/AgentAdapter';
9
+ import { AgentStatus } from './adapters/AgentAdapter';
10
+
11
+ /**
12
+ * Agent Manager Class
13
+ *
14
+ * Central manager for detecting AI agents across different types.
15
+ * Supports multiple adapters (Claude Code, Gemini CLI, etc.)
16
+ *
17
+ * @example
18
+ * ```typescript
19
+ * const manager = new AgentManager();
20
+ * manager.registerAdapter(new ClaudeCodeAdapter());
21
+ *
22
+ * const agents = await manager.listAgents();
23
+ * console.log(`Found ${agents.length} agents`);
24
+ * ```
25
+ */
26
+ export class AgentManager {
27
+ private adapters: Map<string, AgentAdapter> = new Map();
28
+
29
+ /**
30
+ * Register an adapter for a specific agent type
31
+ *
32
+ * @param adapter Agent adapter to register
33
+ * @throws Error if an adapter for this type is already registered
34
+ *
35
+ * @example
36
+ * ```typescript
37
+ * manager.registerAdapter(new ClaudeCodeAdapter());
38
+ * ```
39
+ */
40
+ registerAdapter(adapter: AgentAdapter): void {
41
+ const adapterKey = adapter.type;
42
+
43
+ if (this.adapters.has(adapterKey)) {
44
+ throw new Error(`Adapter for type "${adapterKey}" is already registered`);
45
+ }
46
+
47
+ this.adapters.set(adapterKey, adapter);
48
+ }
49
+
50
+ /**
51
+ * Unregister an adapter by type
52
+ *
53
+ * @param type Agent type to unregister
54
+ * @returns True if adapter was removed, false if not found
55
+ */
56
+ unregisterAdapter(type: string): boolean {
57
+ return this.adapters.delete(type);
58
+ }
59
+
60
+ /**
61
+ * Get all registered adapters
62
+ *
63
+ * @returns Array of registered adapters
64
+ */
65
+ getAdapters(): AgentAdapter[] {
66
+ return Array.from(this.adapters.values());
67
+ }
68
+
69
+ /**
70
+ * Check if an adapter is registered for a specific type
71
+ *
72
+ * @param type Agent type to check
73
+ * @returns True if adapter is registered
74
+ */
75
+ hasAdapter(type: string): boolean {
76
+ return this.adapters.has(type);
77
+ }
78
+
79
+ /**
80
+ * List all running AI agents detected by registered adapters
81
+ *
82
+ * Queries all registered adapters and aggregates results.
83
+ * Handles errors gracefully - if one adapter fails, others still run.
84
+ *
85
+ * @returns Array of detected agents from all adapters
86
+ *
87
+ * @example
88
+ * ```typescript
89
+ * const agents = await manager.listAgents();
90
+ *
91
+ * agents.forEach(agent => {
92
+ * console.log(`${agent.name}: ${agent.status}`);
93
+ * });
94
+ * ```
95
+ */
96
+ async listAgents(): Promise<AgentInfo[]> {
97
+ const allAgents: AgentInfo[] = [];
98
+ const errors: Array<{ type: string; error: Error }> = [];
99
+
100
+ // Query all adapters in parallel
101
+ const adapterPromises = Array.from(this.adapters.values()).map(async (adapter) => {
102
+ try {
103
+ const agents = await adapter.detectAgents();
104
+ return { type: adapter.type, agents, error: null };
105
+ } catch (error) {
106
+ // Capture error but don't throw - allow other adapters to continue
107
+ const err = error instanceof Error ? error : new Error(String(error));
108
+ errors.push({ type: adapter.type, error: err });
109
+ return { type: adapter.type, agents: [], error: err };
110
+ }
111
+ });
112
+
113
+ const results = await Promise.all(adapterPromises);
114
+
115
+ // Aggregate all successful results
116
+ for (const result of results) {
117
+ if (result.error === null) {
118
+ allAgents.push(...result.agents);
119
+ }
120
+ }
121
+
122
+ // Log errors if any (but don't throw - partial results are useful)
123
+ if (errors.length > 0) {
124
+ console.error(`Warning: ${errors.length} adapter(s) failed:`);
125
+ errors.forEach(({ type, error }) => {
126
+ console.error(` - ${type}: ${error.message}`);
127
+ });
128
+ }
129
+
130
+ // Sort by status priority (waiting first, then running, then idle)
131
+ return this.sortAgentsByStatus(allAgents);
132
+ }
133
+
134
+ /**
135
+ * Sort agents by status priority
136
+ *
137
+ * Priority order: waiting > running > idle > unknown
138
+ * This ensures agents that need attention appear first.
139
+ *
140
+ * @param agents Array of agents to sort
141
+ * @returns Sorted array of agents
142
+ */
143
+ private sortAgentsByStatus(agents: AgentInfo[]): AgentInfo[] {
144
+ const statusPriority: Record<AgentStatus, number> = {
145
+ [AgentStatus.WAITING]: 0,
146
+ [AgentStatus.RUNNING]: 1,
147
+ [AgentStatus.IDLE]: 2,
148
+ [AgentStatus.UNKNOWN]: 3,
149
+ };
150
+
151
+ return agents.sort((a, b) => {
152
+ const priorityA = statusPriority[a.status] ?? 999;
153
+ const priorityB = statusPriority[b.status] ?? 999;
154
+ return priorityA - priorityB;
155
+ });
156
+ }
157
+
158
+ /**
159
+ * Get count of registered adapters
160
+ *
161
+ * @returns Number of registered adapters
162
+ */
163
+ getAdapterCount(): number {
164
+ return this.adapters.size;
165
+ }
166
+
167
+ /**
168
+ * Clear all registered adapters
169
+ */
170
+ clear(): void {
171
+ this.adapters.clear();
172
+ }
173
+
174
+ /**
175
+ * Resolve an agent by name (exact or partial match)
176
+ *
177
+ * @param input Name to search for
178
+ * @param agents List of agents to search within
179
+ * @returns Matched agent (unique), array of agents (ambiguous), or null (none)
180
+ */
181
+ resolveAgent(input: string, agents: AgentInfo[]): AgentInfo | AgentInfo[] | null {
182
+ if (!input || agents.length === 0) return null;
183
+
184
+ const lowerInput = input.toLowerCase();
185
+
186
+ // 1. Exact match (case-insensitive)
187
+ const exactMatch = agents.find(a => a.name.toLowerCase() === lowerInput);
188
+ if (exactMatch) return exactMatch;
189
+
190
+ // 2. Partial match (prefix or contains)
191
+ const matches = agents.filter(a => a.name.toLowerCase().includes(lowerInput));
192
+
193
+ if (matches.length === 1) return matches[0];
194
+ if (matches.length > 1) return matches;
195
+
196
+ return null;
197
+ }
198
+ }
@@ -0,0 +1,308 @@
1
+ /**
2
+ * Tests for AgentManager
3
+ */
4
+
5
+ import { describe, it, expect, beforeEach } from '@jest/globals';
6
+ import { AgentManager } from '../AgentManager';
7
+ import type { AgentAdapter, AgentInfo, AgentType } from '../adapters/AgentAdapter';
8
+ import { AgentStatus } from '../adapters/AgentAdapter';
9
+
10
+ // Mock adapter for testing
11
+ class MockAdapter implements AgentAdapter {
12
+ constructor(
13
+ public readonly type: AgentType,
14
+ private mockAgents: AgentInfo[] = [],
15
+ private shouldFail: boolean = false
16
+ ) { }
17
+
18
+ async detectAgents(): Promise<AgentInfo[]> {
19
+ if (this.shouldFail) {
20
+ throw new Error(`Mock adapter ${this.type} failed`);
21
+ }
22
+ return this.mockAgents;
23
+ }
24
+
25
+ canHandle(): boolean {
26
+ return true;
27
+ }
28
+
29
+ setAgents(agents: AgentInfo[]): void {
30
+ this.mockAgents = agents;
31
+ }
32
+
33
+ setFail(shouldFail: boolean): void {
34
+ this.shouldFail = shouldFail;
35
+ }
36
+ }
37
+
38
+ // Helper to create mock agent
39
+ function createMockAgent(overrides: Partial<AgentInfo> = {}): AgentInfo {
40
+ return {
41
+ name: 'test-agent',
42
+ type: 'claude',
43
+ status: AgentStatus.RUNNING,
44
+ summary: 'Test summary',
45
+ pid: 12345,
46
+ projectPath: '/test/path',
47
+ sessionId: 'test-session-id',
48
+ slug: 'test-slug',
49
+ lastActive: new Date(),
50
+ ...overrides,
51
+ };
52
+ }
53
+
54
+ describe('AgentManager', () => {
55
+ let manager: AgentManager;
56
+
57
+ beforeEach(() => {
58
+ manager = new AgentManager();
59
+ });
60
+
61
+ describe('registerAdapter', () => {
62
+ it('should register a new adapter', () => {
63
+ const adapter = new MockAdapter('claude');
64
+
65
+ manager.registerAdapter(adapter);
66
+
67
+ expect(manager.hasAdapter('claude')).toBe(true);
68
+ expect(manager.getAdapterCount()).toBe(1);
69
+ });
70
+
71
+ it('should throw error when registering duplicate adapter type', () => {
72
+ const adapter1 = new MockAdapter('claude');
73
+ const adapter2 = new MockAdapter('claude');
74
+
75
+ manager.registerAdapter(adapter1);
76
+
77
+ expect(() => manager.registerAdapter(adapter2)).toThrow(
78
+ 'Adapter for type "claude" is already registered'
79
+ );
80
+ });
81
+
82
+ it('should allow registering multiple different adapter types', () => {
83
+ const adapter1 = new MockAdapter('claude');
84
+ const adapter2 = new MockAdapter('gemini_cli');
85
+
86
+ manager.registerAdapter(adapter1);
87
+ manager.registerAdapter(adapter2);
88
+
89
+ expect(manager.getAdapterCount()).toBe(2);
90
+ expect(manager.hasAdapter('claude')).toBe(true);
91
+ expect(manager.hasAdapter('gemini_cli')).toBe(true);
92
+ });
93
+ });
94
+
95
+ describe('unregisterAdapter', () => {
96
+ it('should unregister an existing adapter', () => {
97
+ const adapter = new MockAdapter('claude');
98
+ manager.registerAdapter(adapter);
99
+
100
+ const removed = manager.unregisterAdapter('claude');
101
+
102
+ expect(removed).toBe(true);
103
+ expect(manager.hasAdapter('claude')).toBe(false);
104
+ expect(manager.getAdapterCount()).toBe(0);
105
+ });
106
+
107
+ it('should return false when unregistering non-existent adapter', () => {
108
+ const removed = manager.unregisterAdapter('NonExistent');
109
+ expect(removed).toBe(false);
110
+ });
111
+ });
112
+
113
+ describe('getAdapters', () => {
114
+ it('should return empty array when no adapters registered', () => {
115
+ const adapters = manager.getAdapters();
116
+ expect(adapters).toEqual([]);
117
+ });
118
+
119
+ it('should return all registered adapters', () => {
120
+ const adapter1 = new MockAdapter('claude');
121
+ const adapter2 = new MockAdapter('gemini_cli');
122
+
123
+ manager.registerAdapter(adapter1);
124
+ manager.registerAdapter(adapter2);
125
+
126
+ const adapters = manager.getAdapters();
127
+ expect(adapters).toHaveLength(2);
128
+ expect(adapters).toContain(adapter1);
129
+ expect(adapters).toContain(adapter2);
130
+ });
131
+ });
132
+
133
+ describe('hasAdapter', () => {
134
+ it('should return true for registered adapter', () => {
135
+ manager.registerAdapter(new MockAdapter('claude'));
136
+ expect(manager.hasAdapter('claude')).toBe(true);
137
+ });
138
+
139
+ it('should return false for non-registered adapter', () => {
140
+ expect(manager.hasAdapter('claude')).toBe(false);
141
+ });
142
+ });
143
+
144
+ describe('listAgents', () => {
145
+ it('should return empty array when no adapters registered', async () => {
146
+ const agents = await manager.listAgents();
147
+ expect(agents).toEqual([]);
148
+ });
149
+
150
+ it('should return agents from single adapter', async () => {
151
+ const mockAgents = [
152
+ createMockAgent({ name: 'agent1' }),
153
+ createMockAgent({ name: 'agent2' }),
154
+ ];
155
+ const adapter = new MockAdapter('claude', mockAgents);
156
+
157
+ manager.registerAdapter(adapter);
158
+ const agents = await manager.listAgents();
159
+
160
+ expect(agents).toHaveLength(2);
161
+ expect(agents[0].name).toBe('agent1');
162
+ expect(agents[1].name).toBe('agent2');
163
+ });
164
+
165
+ it('should aggregate agents from multiple adapters', async () => {
166
+ const claudeAgents = [createMockAgent({ name: 'claude-agent', type: 'claude' })];
167
+ const geminiAgents = [createMockAgent({ name: 'gemini-agent', type: 'gemini_cli' })];
168
+
169
+ manager.registerAdapter(new MockAdapter('claude', claudeAgents));
170
+ manager.registerAdapter(new MockAdapter('gemini_cli', geminiAgents));
171
+
172
+ const agents = await manager.listAgents();
173
+
174
+ expect(agents).toHaveLength(2);
175
+ expect(agents.find(a => a.name === 'claude-agent')).toBeDefined();
176
+ expect(agents.find(a => a.name === 'gemini-agent')).toBeDefined();
177
+ });
178
+
179
+ it('should sort agents by status priority (waiting first)', async () => {
180
+ const mockAgents = [
181
+ createMockAgent({ name: 'idle-agent', status: AgentStatus.IDLE }),
182
+ createMockAgent({ name: 'waiting-agent', status: AgentStatus.WAITING }),
183
+ createMockAgent({ name: 'running-agent', status: AgentStatus.RUNNING }),
184
+ createMockAgent({ name: 'unknown-agent', status: AgentStatus.UNKNOWN }),
185
+ ];
186
+ const adapter = new MockAdapter('claude', mockAgents);
187
+
188
+ manager.registerAdapter(adapter);
189
+ const agents = await manager.listAgents();
190
+
191
+ expect(agents[0].name).toBe('waiting-agent');
192
+ expect(agents[1].name).toBe('running-agent');
193
+ expect(agents[2].name).toBe('idle-agent');
194
+ expect(agents[3].name).toBe('unknown-agent');
195
+ });
196
+
197
+ it('should handle adapter errors gracefully', async () => {
198
+ const goodAdapter = new MockAdapter('claude', [
199
+ createMockAgent({ name: 'good-agent' }),
200
+ ]);
201
+ const badAdapter = new MockAdapter('gemini_cli', [], true); // Will fail
202
+
203
+ manager.registerAdapter(goodAdapter);
204
+ manager.registerAdapter(badAdapter);
205
+
206
+ // Should not throw, should return results from working adapter
207
+ const agents = await manager.listAgents();
208
+
209
+ expect(agents).toHaveLength(1);
210
+ expect(agents[0].name).toBe('good-agent');
211
+ });
212
+
213
+ it('should return empty array when all adapters fail', async () => {
214
+ const adapter1 = new MockAdapter('claude', [], true);
215
+ const adapter2 = new MockAdapter('gemini_cli', [], true);
216
+
217
+ manager.registerAdapter(adapter1);
218
+ manager.registerAdapter(adapter2);
219
+
220
+ const agents = await manager.listAgents();
221
+ expect(agents).toEqual([]);
222
+ });
223
+ });
224
+
225
+ describe('getAdapterCount', () => {
226
+ it('should return 0 when no adapters registered', () => {
227
+ expect(manager.getAdapterCount()).toBe(0);
228
+ });
229
+
230
+ it('should return correct count', () => {
231
+ manager.registerAdapter(new MockAdapter('claude'));
232
+ expect(manager.getAdapterCount()).toBe(1);
233
+
234
+ manager.registerAdapter(new MockAdapter('gemini_cli'));
235
+ expect(manager.getAdapterCount()).toBe(2);
236
+ });
237
+ });
238
+
239
+ describe('clear', () => {
240
+ it('should remove all adapters', () => {
241
+ manager.registerAdapter(new MockAdapter('claude'));
242
+ manager.registerAdapter(new MockAdapter('gemini_cli'));
243
+
244
+ manager.clear();
245
+
246
+ expect(manager.getAdapterCount()).toBe(0);
247
+ expect(manager.getAdapters()).toEqual([]);
248
+ });
249
+ });
250
+
251
+ describe('resolveAgent', () => {
252
+ it('should return null for empty input or empty agents list', () => {
253
+ const agent = createMockAgent({ name: 'test-agent' });
254
+ expect(manager.resolveAgent('', [agent])).toBeNull();
255
+ expect(manager.resolveAgent('test', [])).toBeNull();
256
+ });
257
+
258
+ it('should resolve exact match (case-insensitive)', () => {
259
+ const agent = createMockAgent({ name: 'My-Agent' });
260
+ const agents = [agent, createMockAgent({ name: 'Other' })];
261
+
262
+ // Exact match
263
+ expect(manager.resolveAgent('My-Agent', agents)).toBe(agent);
264
+ // Case-insensitive
265
+ expect(manager.resolveAgent('my-agent', agents)).toBe(agent);
266
+ });
267
+
268
+ it('should resolve unique partial match', () => {
269
+ const agent = createMockAgent({ name: 'ai-devkit' });
270
+ const agents = [
271
+ agent,
272
+ createMockAgent({ name: 'other-project' })
273
+ ];
274
+
275
+ const result = manager.resolveAgent('dev', agents);
276
+ expect(result).toBe(agent);
277
+ });
278
+
279
+ it('should return array for ambiguous partial match', () => {
280
+ const agent1 = createMockAgent({ name: 'my-website' });
281
+ const agent2 = createMockAgent({ name: 'my-app' });
282
+ const agents = [agent1, agent2, createMockAgent({ name: 'other' })];
283
+
284
+ const result = manager.resolveAgent('my', agents);
285
+
286
+ expect(Array.isArray(result)).toBe(true);
287
+ const matches = result as AgentInfo[];
288
+ expect(matches).toHaveLength(2);
289
+ expect(matches).toContain(agent1);
290
+ expect(matches).toContain(agent2);
291
+ });
292
+
293
+ it('should return null for no match', () => {
294
+ const agents = [createMockAgent({ name: 'ai-devkit' })];
295
+ expect(manager.resolveAgent('xyz', agents)).toBeNull();
296
+ });
297
+
298
+ it('should prefer exact match over partial matches', () => {
299
+ // Edge case: "test" matches "test" (exact) and "testing" (partial)
300
+ // Should return exact "test"
301
+ const exact = createMockAgent({ name: 'test' });
302
+ const partial = createMockAgent({ name: 'testing' });
303
+ const agents = [exact, partial];
304
+
305
+ expect(manager.resolveAgent('test', agents)).toBe(exact);
306
+ });
307
+ });
308
+ });