优先级:高 | 影响 4 个测试文件 | 预估修改 ~15 处
本计划修复测试中 mock 相关的副作用、状态泄漏和虚假测试。
gitOperationTracking.test.ts — 消除分析副作用当前问题:detectGitOperation 内部调用 logEvent()、getCommitCounter().increment()、getPrCounter().increment(),每次测试运行都触发真实分析代码。
修复步骤:
src/tools/shared/gitOperationTracking.ts,确认 analytics 导入路径在测试文件顶部添加 mock.module:
import { mock } from "bun:test";
mock.module("src/services/analytics/index.ts", () => ({
logEvent: mock(() => {}),
// 按需补充其他导出
}));
如果 getCommitCounter / getPrCounter 来自 src/bootstrap/state.ts:
mock.module("src/bootstrap/state.ts", () => ({
getCommitCounter: mock(() => ({ increment: mock(() => {}) })),
getPrCounter: mock(() => ({ increment: mock(() => {}) })),
// 保留其他被测函数实际需要的导出
}));
使用 await import() 模式加载被测模块
运行测试验证无副作用
风险:mock.module 会替换整个模块。如果 detectGitOperation 还需要其他来自这些模块的导出,需在 mock 工厂中提供。
PermissionMode.test.ts — 修复 isExternalPermissionMode 虚假测试当前问题:isExternalPermissionMode 依赖 process.env.USER_TYPE。非 ant 环境下所有 mode 都返回 true,测试从未覆盖 false 分支。
修复步骤:
使用 beforeEach/afterEach 管理 process.env.USER_TYPE
describe("when USER_TYPE is 'ant'", () => {
const originalUserType = process.env.USER_TYPE;
beforeEach(() => { process.env.USER_TYPE = "ant"; });
afterEach(() => {
if (originalUserType !== undefined) {
process.env.USER_TYPE = originalUserType;
} else {
delete process.env.USER_TYPE;
}
});
test("returns false for 'auto'", () => {
expect(isExternalPermissionMode("auto")).toBe(false);
});
test("returns false for 'bubble'", () => {
expect(isExternalPermissionMode("bubble")).toBe(false);
});
test("returns true for 'plan'", () => {
expect(isExternalPermissionMode("plan")).toBe(true);
});
});
验证新增测试确实执行 false 路径
providers.test.ts — 环境变量快照恢复当前问题:
originalEnv 声明后未使用afterEach 仅删除已知 3 个 key,如果源码新增 env var,测试间状态泄漏修复步骤:
let savedEnv: Record<string, string | undefined>;
beforeEach(() => {
savedEnv = {};
for (const key of Object.keys(process.env)) {
savedEnv[key] = process.env[key];
}
});
afterEach(() => {
// 删除所有当前 env,恢复快照
for (const key of Object.keys(process.env)) {
delete process.env[key];
}
for (const [key, value] of Object.entries(savedEnv)) {
if (value !== undefined) {
process.env[key] = value;
}
}
});
简化方案:只保存/恢复相关 key 列表
["CLAUDE_CODE_USE_BEDROCK", "CLAUDE_CODE_USE_VERTEX", "CLAUDE_CODE_USE_FOUNDRY", "ANTHROPIC_BASE_URL", "USER_TYPE"],但需注释说明新增 env var 时需同步更新。
envUtils.test.ts — 验证环境变量恢复完整性当前状态:已有 afterEach 恢复。需审查:
describe 块中的 afterEach 都完整恢复了修改的 env varprocess.argv 修改也被恢复(getClaudeConfigHomeDir 测试修改了 argv)afterEach 中断言无意外 env 泄漏(可选,CI-only)sleep.test.ts / memoize.test.ts — 时间敏感测试加固当前状态:已有合理 margin。可选加固:
| 文件 | 用例 | 当前 | 加固 |
|---|---|---|---|
sleep.test.ts |
resolves after timeout |
sleep(50), check >= 40ms |
增大 margin:sleep(50), check >= 30ms |
memoize.test.ts |
stale serve & refresh | TTL=1ms, wait 10ms | 增大 margin:TTL=5ms, wait 50ms |
仅在 CI 出现 flaky 时执行此加固。
gitOperationTracking.test.ts 无分析副作用(可通过在 mock 中增加 expect(logEvent).toHaveBeenCalledTimes(N) 验证)PermissionMode.test.ts 的 isExternalPermissionMode 覆盖 true + false 分支providers.test.ts 的 originalEnv 死代码已删除bun test 全部通过