729fb2d42a
- 新增 vitest 测试框架配置 - 添加 54 个测试文件,共 951 个测试用例 - 覆盖核心模块: - Agent: executor, registry, config-loader, permission-merger - Context: manager, compaction, prune, token-counter - Permission: manager, bash/file/git/web checkers, wildcard - Session: manager, storage - Tools: filesystem (12个), git (10个), web, shell, todo, task - LSP: client, server, language - Utils: config, diff - UI: terminal
308 lines
8.7 KiB
TypeScript
308 lines
8.7 KiB
TypeScript
import { describe, it, expect, beforeEach, vi } from 'vitest';
|
|
|
|
// 使用可变的引用对象来绕过 hoisting 问题
|
|
const mockState = {
|
|
execute: vi.fn(),
|
|
};
|
|
|
|
// Mock agent registry 和 AgentExecutor
|
|
vi.mock('../../../../src/agent/index.js', () => {
|
|
// 在 mock 工厂内部定义类
|
|
return {
|
|
agentRegistry: {
|
|
listSubagents: vi.fn(() => [
|
|
{ name: 'explore', description: '代码探索', mode: 'subagent' },
|
|
{ name: 'code-reviewer', description: '代码审查', mode: 'subagent' },
|
|
]),
|
|
get: vi.fn(),
|
|
},
|
|
AgentExecutor: class {
|
|
execute(...args: any[]) {
|
|
return mockState.execute(...args);
|
|
}
|
|
},
|
|
};
|
|
});
|
|
|
|
// Mock tool registry
|
|
vi.mock('../../../../src/tools/registry.js', () => ({
|
|
toolRegistry: {},
|
|
}));
|
|
|
|
// Mock session manager
|
|
vi.mock('../../../../src/session/index.js', () => ({
|
|
SessionManager: vi.fn(),
|
|
}));
|
|
|
|
import { taskTool, initTaskContext, updateTaskDescription } from '../../../../src/tools/task/task.js';
|
|
import { agentRegistry } from '../../../../src/agent/index.js';
|
|
|
|
describe('taskTool - Task 工具', () => {
|
|
beforeEach(() => {
|
|
vi.clearAllMocks();
|
|
// 重置上下文为 null
|
|
initTaskContext(null as any, null as any);
|
|
// 重置 mock
|
|
mockState.execute.mockResolvedValue({
|
|
success: true,
|
|
text: '任务完成',
|
|
steps: 3,
|
|
});
|
|
});
|
|
|
|
describe('工具定义', () => {
|
|
it('有正确的名称', () => {
|
|
expect(taskTool.name).toBe('task');
|
|
});
|
|
|
|
it('有正确的元数据', () => {
|
|
expect(taskTool.metadata.category).toBe('agent');
|
|
expect(taskTool.metadata.keywords).toContain('task');
|
|
expect(taskTool.metadata.keywords).toContain('subagent');
|
|
});
|
|
|
|
it('定义了必需的参数', () => {
|
|
expect(taskTool.parameters.description.required).toBe(true);
|
|
expect(taskTool.parameters.prompt.required).toBe(true);
|
|
expect(taskTool.parameters.subagent_type.required).toBe(true);
|
|
});
|
|
});
|
|
|
|
describe('initTaskContext - 初始化上下文', () => {
|
|
it('设置上下文不报错', () => {
|
|
const mockConfig = { model: 'test' };
|
|
const mockSession = {
|
|
getSessionId: vi.fn(() => 'session-id'),
|
|
createChildSession: vi.fn(() => ({ id: 'child', messages: [] })),
|
|
saveChildSession: vi.fn(),
|
|
};
|
|
|
|
expect(() => initTaskContext(mockConfig as any, mockSession as any)).not.toThrow();
|
|
});
|
|
});
|
|
|
|
describe('updateTaskDescription - 更新描述', () => {
|
|
it('更新工具描述', () => {
|
|
vi.mocked(agentRegistry.listSubagents).mockReturnValue([
|
|
{ name: 'explore', description: '代码探索', mode: 'subagent' },
|
|
{ name: 'code-reviewer', description: '代码审查', mode: 'subagent' },
|
|
]);
|
|
|
|
updateTaskDescription();
|
|
|
|
expect(taskTool.description).toContain('explore');
|
|
expect(taskTool.description).toContain('code-reviewer');
|
|
});
|
|
|
|
it('无子 Agent 时显示提示', () => {
|
|
vi.mocked(agentRegistry.listSubagents).mockReturnValue([]);
|
|
|
|
updateTaskDescription();
|
|
|
|
expect(taskTool.description).toContain('没有可用');
|
|
});
|
|
});
|
|
|
|
describe('execute - 执行', () => {
|
|
it('未初始化上下文时返回错误', async () => {
|
|
// 确保上下文为 null
|
|
initTaskContext(null as any, null as any);
|
|
vi.mocked(agentRegistry.get).mockReturnValue(undefined);
|
|
|
|
const result = await taskTool.execute({
|
|
description: 'test task',
|
|
prompt: 'do something',
|
|
subagent_type: 'explore',
|
|
});
|
|
|
|
expect(result.success).toBe(false);
|
|
// 可能是未初始化或未找到 Agent
|
|
expect(result.error).toBeDefined();
|
|
});
|
|
|
|
it('成功执行子任务', async () => {
|
|
const mockSession = {
|
|
getSessionId: vi.fn(() => 'parent-session'),
|
|
createChildSession: vi.fn(() => ({
|
|
id: 'child-session',
|
|
messages: [],
|
|
})),
|
|
saveChildSession: vi.fn(),
|
|
};
|
|
|
|
initTaskContext({ model: 'test' } as any, mockSession as any);
|
|
|
|
vi.mocked(agentRegistry.get).mockReturnValue({
|
|
name: 'explore',
|
|
description: '探索 Agent',
|
|
mode: 'subagent',
|
|
prompt: '你是探索助手',
|
|
});
|
|
|
|
const result = await taskTool.execute({
|
|
description: 'search code',
|
|
prompt: 'find all API routes',
|
|
subagent_type: 'explore',
|
|
});
|
|
|
|
expect(result.success).toBe(true);
|
|
expect(result.output).toContain('任务完成');
|
|
});
|
|
|
|
it('未找到 Agent 时返回错误', async () => {
|
|
const mockSession = {
|
|
getSessionId: vi.fn(() => 'parent-session'),
|
|
createChildSession: vi.fn(() => ({ id: 'child', messages: [] })),
|
|
saveChildSession: vi.fn(),
|
|
};
|
|
|
|
initTaskContext({ model: 'test' } as any, mockSession as any);
|
|
|
|
vi.mocked(agentRegistry.get).mockReturnValue(undefined);
|
|
vi.mocked(agentRegistry.listSubagents).mockReturnValue([
|
|
{ name: 'explore', description: '探索', mode: 'subagent' },
|
|
]);
|
|
|
|
const result = await taskTool.execute({
|
|
description: 'test',
|
|
prompt: 'test',
|
|
subagent_type: 'nonexistent',
|
|
});
|
|
|
|
expect(result.success).toBe(false);
|
|
expect(result.error).toContain('未找到 Agent');
|
|
});
|
|
|
|
it('primary 模式 Agent 不能作为子任务', async () => {
|
|
const mockSession = {
|
|
getSessionId: vi.fn(() => 'parent-session'),
|
|
createChildSession: vi.fn(() => ({ id: 'child', messages: [] })),
|
|
saveChildSession: vi.fn(),
|
|
};
|
|
|
|
initTaskContext({ model: 'test' } as any, mockSession as any);
|
|
|
|
vi.mocked(agentRegistry.get).mockReturnValue({
|
|
name: 'primary-agent',
|
|
description: '主 Agent',
|
|
mode: 'primary',
|
|
prompt: '你是主助手',
|
|
});
|
|
|
|
const result = await taskTool.execute({
|
|
description: 'test',
|
|
prompt: 'test',
|
|
subagent_type: 'primary-agent',
|
|
});
|
|
|
|
expect(result.success).toBe(false);
|
|
expect(result.error).toContain('primary 模式');
|
|
});
|
|
|
|
it('子任务失败时返回错误', async () => {
|
|
const mockSession = {
|
|
getSessionId: vi.fn(() => 'parent-session'),
|
|
createChildSession: vi.fn(() => ({ id: 'child', messages: [] })),
|
|
saveChildSession: vi.fn(),
|
|
};
|
|
|
|
initTaskContext({ model: 'test' } as any, mockSession as any);
|
|
|
|
vi.mocked(agentRegistry.get).mockReturnValue({
|
|
name: 'explore',
|
|
description: '探索 Agent',
|
|
mode: 'subagent',
|
|
prompt: '你是探索助手',
|
|
});
|
|
|
|
mockState.execute.mockResolvedValue({
|
|
success: false,
|
|
text: '',
|
|
error: '执行失败',
|
|
steps: 1,
|
|
});
|
|
|
|
const result = await taskTool.execute({
|
|
description: 'test',
|
|
prompt: 'test',
|
|
subagent_type: 'explore',
|
|
});
|
|
|
|
expect(result.success).toBe(false);
|
|
expect(result.error).toContain('执行失败');
|
|
});
|
|
|
|
it('返回元数据', async () => {
|
|
const mockSession = {
|
|
getSessionId: vi.fn(() => 'parent-session'),
|
|
createChildSession: vi.fn(() => ({
|
|
id: 'child-session-123',
|
|
messages: [],
|
|
})),
|
|
saveChildSession: vi.fn(),
|
|
};
|
|
|
|
initTaskContext({ model: 'test' } as any, mockSession as any);
|
|
|
|
vi.mocked(agentRegistry.get).mockReturnValue({
|
|
name: 'explore',
|
|
description: '探索 Agent',
|
|
mode: 'subagent',
|
|
prompt: '你是探索助手',
|
|
});
|
|
|
|
mockState.execute.mockResolvedValue({
|
|
success: true,
|
|
text: '完成',
|
|
steps: 5,
|
|
});
|
|
|
|
const result = await taskTool.execute({
|
|
description: 'test',
|
|
prompt: 'test',
|
|
subagent_type: 'explore',
|
|
});
|
|
|
|
expect(result.metadata).toBeDefined();
|
|
expect(result.metadata?.agent).toBe('explore');
|
|
expect(result.metadata?.sessionId).toBe('child-session-123');
|
|
expect(result.metadata?.steps).toBe(5);
|
|
});
|
|
|
|
it('保存子会话', async () => {
|
|
const saveChildSession = vi.fn();
|
|
const mockSession = {
|
|
getSessionId: vi.fn(() => 'parent-session'),
|
|
createChildSession: vi.fn(() => ({
|
|
id: 'child-session',
|
|
messages: [],
|
|
})),
|
|
saveChildSession,
|
|
};
|
|
|
|
initTaskContext({ model: 'test' } as any, mockSession as any);
|
|
|
|
vi.mocked(agentRegistry.get).mockReturnValue({
|
|
name: 'explore',
|
|
description: '探索 Agent',
|
|
mode: 'subagent',
|
|
prompt: '你是探索助手',
|
|
});
|
|
|
|
mockState.execute.mockResolvedValue({
|
|
success: true,
|
|
text: '任务结果',
|
|
steps: 2,
|
|
});
|
|
|
|
await taskTool.execute({
|
|
description: 'test',
|
|
prompt: 'test prompt',
|
|
subagent_type: 'explore',
|
|
});
|
|
|
|
expect(saveChildSession).toHaveBeenCalled();
|
|
});
|
|
});
|
|
});
|