Files
claude-code/src/commands/__tests__/commit.test.ts
claude-code-best 6a182e45b3 feat: 注册所有新命令到命令系统和工具注册表
- commands.ts: 注册所有新命令(memory-stores、vault、schedule 等),
  移除 require() 动态加载,统一为 ESM import
- tools.ts: 注册 LocalMemoryRecallTool、VaultHttpFetchTool
- 补充命令测试(bridge-kick、commit、commit-push-pr、init-verifiers)
- 补充工具测试(AgentTool、RemoteTrigger、SkillTool、WebFetch、WebSearch)
- 集成测试:autonomy-lifecycle-user-flow 更新
- 探测脚本和功能文档

Co-Authored-By: glm-5-turbo <zai-org@claude-code-best.win>
2026-05-09 23:04:39 +08:00

274 lines
8.0 KiB
TypeScript

import { afterEach, beforeEach, describe, expect, mock, test } from 'bun:test'
import type { Command } from '../../commands.js'
// Mock bun:bundle before any imports that use feature()
mock.module('bun:bundle', () => ({
feature: (_name: string) => false,
}))
// Mock dependencies to avoid side effects
mock.module('src/utils/attribution.ts', () => ({
getAttributionTexts: () => ({ commit: '', pr: '' }),
getEnhancedPRAttribution: async () => undefined,
countUserPromptsInMessages: () => 0,
}))
mock.module('src/utils/undercover.ts', () => ({
isUndercover: () => false,
getUndercoverInstructions: () => '',
shouldShowUndercoverAutoNotice: () => false,
}))
mock.module('src/utils/promptShellExecution.ts', () => ({
executeShellCommandsInPrompt: async (content: string) => content,
}))
let commit: Command
let originalUserType: string | undefined
beforeEach(async () => {
originalUserType = process.env.USER_TYPE
const mod = await import('../commit.js')
commit = mod.default as Command
})
afterEach(() => {
if (originalUserType === undefined) {
delete process.env.USER_TYPE
} else {
process.env.USER_TYPE = originalUserType
}
})
describe('commit command metadata', () => {
test('has correct name', () => {
expect(commit.name).toBe('commit')
})
test('has description', () => {
expect(commit.description).toBeTruthy()
expect(typeof commit.description).toBe('string')
})
test('type is prompt', () => {
expect(commit.type).toBe('prompt')
})
test('has progressMessage', () => {
expect((commit as any).progressMessage).toBeTruthy()
})
test('source is builtin', () => {
expect((commit as any).source).toBe('builtin')
})
test('has allowedTools array', () => {
const tools = (commit as any).allowedTools
expect(Array.isArray(tools)).toBe(true)
expect(tools.length).toBeGreaterThan(0)
})
test('allowedTools includes git add', () => {
const tools = (commit as any).allowedTools as string[]
expect(tools.some(t => t.includes('git add'))).toBe(true)
})
test('allowedTools includes git commit', () => {
const tools = (commit as any).allowedTools as string[]
expect(tools.some(t => t.includes('git commit'))).toBe(true)
})
test('allowedTools includes git status', () => {
const tools = (commit as any).allowedTools as string[]
expect(tools.some(t => t.includes('git status'))).toBe(true)
})
test('contentLength is 0 (dynamic)', () => {
expect((commit as any).contentLength).toBe(0)
})
})
describe('commit command getPromptForCommand', () => {
test('returns array with text type', async () => {
const mockContext = {
getAppState: () => ({
toolPermissionContext: {
alwaysAllowRules: { command: [] },
},
}),
}
const result = await (commit as any).getPromptForCommand('', mockContext)
expect(Array.isArray(result)).toBe(true)
expect(result.length).toBeGreaterThan(0)
expect(result[0].type).toBe('text')
})
test('result text contains git instructions', async () => {
const mockContext = {
getAppState: () => ({
toolPermissionContext: {
alwaysAllowRules: { command: [] },
},
}),
}
const result = await (commit as any).getPromptForCommand('', mockContext)
expect(result[0].text).toContain('git')
})
test('result text contains git status', async () => {
const mockContext = {
getAppState: () => ({
toolPermissionContext: {
alwaysAllowRules: { command: [] },
},
}),
}
const result = await (commit as any).getPromptForCommand('', mockContext)
expect(result[0].text).toContain('git status')
})
test('result text contains commit message instructions', async () => {
const mockContext = {
getAppState: () => ({
toolPermissionContext: {
alwaysAllowRules: { command: [] },
},
}),
}
const result = await (commit as any).getPromptForCommand('', mockContext)
expect(result[0].text).toContain('commit')
})
test('getAppState override preserves alwaysAllowRules', async () => {
let capturedAppState: any
const mockContext = {
getAppState: () => ({
toolPermissionContext: {
alwaysAllowRules: { command: ['existing-rule'] },
otherProp: 'test',
},
otherState: 'value',
}),
}
// Wrap executeShellCommandsInPrompt to capture context
mock.module('src/utils/promptShellExecution.ts', () => ({
executeShellCommandsInPrompt: async (content: string, ctx: any) => {
capturedAppState = ctx.getAppState()
return content
},
}))
const mod = await import('../commit.js')
const freshCommit = mod.default as any
await freshCommit.getPromptForCommand('', mockContext)
// The override should include alwaysAllowRules with command tools
if (capturedAppState) {
expect(
capturedAppState.toolPermissionContext.alwaysAllowRules.command,
).toBeDefined()
}
})
test('getPromptForCommand with non-ant user_type does not include undercover prefix', async () => {
process.env.USER_TYPE = 'external'
const mockContext = {
getAppState: () => ({
toolPermissionContext: {
alwaysAllowRules: { command: [] },
},
}),
}
const result = await (commit as any).getPromptForCommand('', mockContext)
expect(Array.isArray(result)).toBe(true)
})
test('getPromptForCommand with ant user_type and undercover', async () => {
process.env.USER_TYPE = 'ant'
// isUndercover is mocked to return false, so prefix stays empty
const mockContext = {
getAppState: () => ({
toolPermissionContext: {
alwaysAllowRules: { command: [] },
},
}),
}
const result = await (commit as any).getPromptForCommand('', mockContext)
expect(Array.isArray(result)).toBe(true)
expect(result[0].type).toBe('text')
})
test('ant undercover path prepends undercover instructions', async () => {
process.env.USER_TYPE = 'ant'
mock.module('src/utils/undercover.ts', () => ({
isUndercover: () => true,
getUndercoverInstructions: () => 'SECRET_UNDERCOVER_PREFIX',
shouldShowUndercoverAutoNotice: () => false,
}))
mock.module('src/utils/attribution.ts', () => ({
getAttributionTexts: () => ({ commit: 'Co-Authored-By: Claude', pr: '' }),
getEnhancedPRAttribution: async () => undefined,
countUserPromptsInMessages: () => 0,
}))
const { default: freshCommit } = await import('../commit.js')
const mockContext = {
getAppState: () => ({
toolPermissionContext: {
alwaysAllowRules: { command: [] },
},
}),
}
const result = await (freshCommit as any).getPromptForCommand(
'',
mockContext,
)
expect(Array.isArray(result)).toBe(true)
expect(result[0].text).toContain('SECRET_UNDERCOVER_PREFIX')
expect(result[0].text).toContain('Co-Authored-By')
})
test('getAppState override in context passes ALLOWED_TOOLS', async () => {
let capturedCtx: any
mock.module('src/utils/promptShellExecution.ts', () => ({
executeShellCommandsInPrompt: async (content: string, ctx: any) => {
capturedCtx = ctx
return content
},
}))
const { default: freshCommit } = await import('../commit.js')
const baseAppState = {
toolPermissionContext: {
alwaysAllowRules: { command: ['old-rule'] },
otherProp: 'keep-this',
},
globalState: 'preserved',
}
const mockContext = {
getAppState: () => baseAppState,
}
await (freshCommit as any).getPromptForCommand('', mockContext)
expect(capturedCtx).toBeDefined()
const overriddenState = capturedCtx.getAppState()
expect(overriddenState.globalState).toBe('preserved')
expect(
Array.isArray(
overriddenState.toolPermissionContext.alwaysAllowRules.command,
),
).toBe(true)
expect(
overriddenState.toolPermissionContext.alwaysAllowRules.command.some(
(t: string) => t.includes('git add'),
),
).toBe(true)
})
})