Files
ai-terminal-assistant/tests/unit/tools/shell/bash.test.ts
T
kurihada 729fb2d42a feat: 添加完整的单元测试套件
- 新增 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
2025-12-11 14:45:24 +08:00

184 lines
5.4 KiB
TypeScript

import { describe, it, expect, beforeEach, vi } from 'vitest';
// 定义一个可控的 mock
let mockExecAsyncResult: { stdout: string; stderr: string } | Error = {
stdout: 'command output',
stderr: '',
};
// Mock child_process
vi.mock('child_process', () => ({
exec: vi.fn(),
}));
// Mock util - 返回一个函数,该函数使用外部变量
vi.mock('util', () => ({
promisify: vi.fn(() => vi.fn(async () => {
if (mockExecAsyncResult instanceof Error) {
throw mockExecAsyncResult;
}
return mockExecAsyncResult;
})),
}));
// Mock permission manager
vi.mock('../../../../src/permission/index.js', () => ({
getPermissionManager: vi.fn(() => ({
checkBashPermission: vi.fn().mockResolvedValue({
allowed: true,
action: 'allow',
}),
})),
}));
// Mock loadDescription
vi.mock('../../../../src/tools/load_description.js', () => ({
loadDescription: vi.fn(() => '执行 shell 命令'),
}));
import { bashTool } from '../../../../src/tools/shell/bash.js';
import { getPermissionManager } from '../../../../src/permission/index.js';
describe('bashTool - Bash 命令工具', () => {
beforeEach(() => {
vi.clearAllMocks();
mockExecAsyncResult = {
stdout: 'command output',
stderr: '',
};
});
describe('工具定义', () => {
it('有正确的名称', () => {
expect(bashTool.name).toBe('bash');
});
it('有正确的元数据', () => {
expect(bashTool.metadata.category).toBe('shell');
expect(bashTool.metadata.keywords).toContain('bash');
expect(bashTool.metadata.keywords).toContain('command');
expect(bashTool.metadata.keywords).toContain('terminal');
});
it('定义了必需的 command 参数', () => {
expect(bashTool.parameters.command).toBeDefined();
expect(bashTool.parameters.command.required).toBe(true);
});
it('定义了可选的 cwd 参数', () => {
expect(bashTool.parameters.cwd).toBeDefined();
expect(bashTool.parameters.cwd.required).toBe(false);
});
});
describe('execute - 执行', () => {
it('成功执行命令', async () => {
const result = await bashTool.execute({ command: 'ls -la' });
expect(result.success).toBe(true);
expect(result.output).toContain('command output');
});
it('包含 stderr 输出', async () => {
mockExecAsyncResult = {
stdout: 'output',
stderr: 'warning message',
};
const result = await bashTool.execute({ command: 'some_cmd' });
expect(result.success).toBe(true);
expect(result.output).toContain('output');
expect(result.output).toContain('STDERR');
expect(result.output).toContain('warning message');
});
it('权限被拒绝时返回错误', async () => {
vi.mocked(getPermissionManager).mockReturnValue({
checkBashPermission: vi.fn().mockResolvedValue({
allowed: false,
action: 'deny',
reason: '命令不被允许执行',
}),
} as any);
const result = await bashTool.execute({ command: 'rm -rf /' });
expect(result.success).toBe(false);
expect(result.error).toContain('权限被拒绝');
});
it('需要确认时返回提示', async () => {
vi.mocked(getPermissionManager).mockReturnValue({
checkBashPermission: vi.fn().mockResolvedValue({
allowed: false,
action: 'ask',
needsConfirmation: true,
reason: '需要确认',
patterns: ['rm *'],
}),
} as any);
const result = await bashTool.execute({ command: 'rm file.txt' });
expect(result.success).toBe(false);
expect(result.error).toContain('需要用户确认');
expect(result.error).toContain('rm file.txt');
});
it('命令执行失败时返回错误', async () => {
vi.mocked(getPermissionManager).mockReturnValue({
checkBashPermission: vi.fn().mockResolvedValue({
allowed: true,
action: 'allow',
}),
} as any);
mockExecAsyncResult = Object.assign(
new Error('Command failed'),
{ stdout: '', stderr: 'command not found', message: 'Command failed' }
);
const result = await bashTool.execute({ command: 'nonexistent_cmd' });
expect(result.success).toBe(false);
expect(result.error).toContain('command not found');
});
it('保留失败命令的 stdout', async () => {
vi.mocked(getPermissionManager).mockReturnValue({
checkBashPermission: vi.fn().mockResolvedValue({
allowed: true,
action: 'allow',
}),
} as any);
mockExecAsyncResult = Object.assign(
new Error('Command failed'),
{ stdout: 'partial output', stderr: 'error occurred', message: 'Command failed' }
);
const result = await bashTool.execute({ command: 'failing_cmd' });
expect(result.success).toBe(false);
expect(result.output).toBe('partial output');
expect(result.error).toContain('error occurred');
});
it('传递正确的参数给权限检查', async () => {
const mockCheck = vi.fn().mockResolvedValue({
allowed: true,
action: 'allow',
});
vi.mocked(getPermissionManager).mockReturnValue({
checkBashPermission: mockCheck,
} as any);
await bashTool.execute({ command: 'ls -la', cwd: '/home/user' });
expect(mockCheck).toHaveBeenCalledWith({
command: 'ls -la',
workdir: '/home/user',
});
});
});
});