@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.
- package/.eslintrc.json +31 -0
- package/dist/AgentManager.d.ts +104 -0
- package/dist/AgentManager.d.ts.map +1 -0
- package/dist/AgentManager.js +185 -0
- package/dist/AgentManager.js.map +1 -0
- package/dist/adapters/AgentAdapter.d.ts +76 -0
- package/dist/adapters/AgentAdapter.d.ts.map +1 -0
- package/dist/adapters/AgentAdapter.js +20 -0
- package/dist/adapters/AgentAdapter.js.map +1 -0
- package/dist/adapters/ClaudeCodeAdapter.d.ts +58 -0
- package/dist/adapters/ClaudeCodeAdapter.d.ts.map +1 -0
- package/dist/adapters/ClaudeCodeAdapter.js +274 -0
- package/dist/adapters/ClaudeCodeAdapter.js.map +1 -0
- package/dist/adapters/index.d.ts +4 -0
- package/dist/adapters/index.d.ts.map +1 -0
- package/dist/adapters/index.js +8 -0
- package/dist/adapters/index.js.map +1 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +23 -0
- package/dist/index.js.map +1 -0
- package/dist/terminal/TerminalFocusManager.d.ts +22 -0
- package/dist/terminal/TerminalFocusManager.d.ts.map +1 -0
- package/dist/terminal/TerminalFocusManager.js +196 -0
- package/dist/terminal/TerminalFocusManager.js.map +1 -0
- package/dist/terminal/index.d.ts +3 -0
- package/dist/terminal/index.d.ts.map +1 -0
- package/dist/terminal/index.js +6 -0
- package/dist/terminal/index.js.map +1 -0
- package/dist/utils/file.d.ts +52 -0
- package/dist/utils/file.d.ts.map +1 -0
- package/dist/utils/file.js +135 -0
- package/dist/utils/file.js.map +1 -0
- package/dist/utils/index.d.ts +4 -0
- package/dist/utils/index.d.ts.map +1 -0
- package/dist/utils/index.js +15 -0
- package/dist/utils/index.js.map +1 -0
- package/dist/utils/process.d.ts +61 -0
- package/dist/utils/process.d.ts.map +1 -0
- package/dist/utils/process.js +166 -0
- package/dist/utils/process.js.map +1 -0
- package/jest.config.js +21 -0
- package/package.json +42 -0
- package/project.json +29 -0
- package/src/AgentManager.ts +198 -0
- package/src/__tests__/AgentManager.test.ts +308 -0
- package/src/__tests__/adapters/ClaudeCodeAdapter.test.ts +286 -0
- package/src/adapters/AgentAdapter.ts +94 -0
- package/src/adapters/ClaudeCodeAdapter.ts +344 -0
- package/src/adapters/index.ts +3 -0
- package/src/index.ts +12 -0
- package/src/terminal/TerminalFocusManager.ts +206 -0
- package/src/terminal/index.ts +2 -0
- package/src/utils/file.ts +100 -0
- package/src/utils/index.ts +3 -0
- package/src/utils/process.ts +184 -0
- 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
|
+
});
|