mirror of
https://github.com/claude-code-best/claude-code.git
synced 2026-06-15 21:05:51 +00:00
- src/cli/print.ts: cron onFire 改用 createAutonomyQueuedPromptIfNoActiveSource 并以 prompt 文本作为 sourceId,避免同一定时提示在前一次 run 仍活跃时被重复 入队叠加;顺手移除 4 个已没人引用的 dead import (commitAutonomyQueuedPrompt / prepareAutonomyTurnPrompt / markAutonomyRunCancelled / createAutonomyQueuedPrompt) - src/services/compact/postCompactCleanup.ts: 在 void import().then() 处加 注释,明确 sweepFileContentCache 是有意的 fire-and-forget,函数对外保持 同步签名是设计而非疏忽 - src/utils/autonomyFlows.ts: 给 selectPersistedAutonomyFlows 的两阶段排序 加文档注释(先按 active+updatedAt 选 top-N,再统一按 updatedAt 重排) - tests/integration/autonomy-lifecycle-user-flow.test.ts: stderr 断言失败时 把实际 stderr 内容写进 message,方便 CI 失败时定位
149 lines
4.5 KiB
TypeScript
149 lines
4.5 KiB
TypeScript
import { afterEach, beforeEach, describe, expect, test } from 'bun:test'
|
|
import { existsSync, mkdtempSync, rmSync } from 'node:fs'
|
|
import { tmpdir } from 'node:os'
|
|
import { join, resolve } from 'node:path'
|
|
import {
|
|
resetStateForTests,
|
|
setOriginalCwd,
|
|
setProjectRoot,
|
|
} from '../../src/bootstrap/state'
|
|
import {
|
|
listAutonomyRuns,
|
|
startManagedAutonomyFlowFromHeartbeatTask,
|
|
} from '../../src/utils/autonomyRuns'
|
|
import { listAutonomyFlows } from '../../src/utils/autonomyFlows'
|
|
|
|
const CLI_ENTRYPOINT = resolve(import.meta.dir, '../../src/entrypoints/cli.tsx')
|
|
|
|
let tempDir = ''
|
|
let configDir = ''
|
|
let previousConfigDir: string | undefined
|
|
|
|
async function runAutonomyCli(args: string[]): Promise<string> {
|
|
const proc = Bun.spawn({
|
|
cmd: [process.execPath, CLI_ENTRYPOINT, 'autonomy', ...args],
|
|
cwd: tempDir,
|
|
env: {
|
|
...process.env,
|
|
CLAUDE_CONFIG_DIR: configDir,
|
|
CI: 'true',
|
|
GITHUB_ACTIONS: 'true',
|
|
NODE_ENV: 'development',
|
|
NO_COLOR: '1',
|
|
},
|
|
stdin: 'ignore',
|
|
stdout: 'pipe',
|
|
stderr: 'pipe',
|
|
})
|
|
|
|
const [stdout, stderr, exitCode] = await Promise.all([
|
|
new Response(proc.stdout).text(),
|
|
new Response(proc.stderr).text(),
|
|
proc.exited,
|
|
])
|
|
|
|
expect(stderr, `unexpected stderr output:\n${stderr}`).toBe('')
|
|
expect(exitCode, `non-zero exit ${exitCode}; stderr:\n${stderr}`).toBe(0)
|
|
return stdout
|
|
}
|
|
|
|
beforeEach(() => {
|
|
tempDir = mkdtempSync(join(tmpdir(), 'autonomy-user-flow-'))
|
|
configDir = join(tempDir, 'config')
|
|
previousConfigDir = process.env.CLAUDE_CONFIG_DIR
|
|
process.env.CLAUDE_CONFIG_DIR = configDir
|
|
resetStateForTests()
|
|
setOriginalCwd(tempDir)
|
|
setProjectRoot(tempDir)
|
|
})
|
|
|
|
afterEach(() => {
|
|
resetStateForTests()
|
|
if (previousConfigDir === undefined) {
|
|
delete process.env.CLAUDE_CONFIG_DIR
|
|
} else {
|
|
process.env.CLAUDE_CONFIG_DIR = previousConfigDir
|
|
}
|
|
if (tempDir) {
|
|
rmSync(tempDir, { recursive: true, force: true })
|
|
}
|
|
})
|
|
|
|
describe('autonomy lifecycle user-equivalent CLI flow', () => {
|
|
test('status --deep works from a clean project without creating autonomy state', async () => {
|
|
const output = await runAutonomyCli(['status', '--deep'])
|
|
|
|
expect(output).toContain('# Autonomy Deep Status')
|
|
expect(output).toContain('Autonomy runs: 0')
|
|
expect(output).toContain('Autonomy flows: 0')
|
|
expect(existsSync(join(tempDir, '.claude', 'autonomy', 'runs.json'))).toBe(
|
|
false,
|
|
)
|
|
expect(existsSync(join(tempDir, '.claude', 'autonomy', 'flows.json'))).toBe(
|
|
false,
|
|
)
|
|
})
|
|
|
|
test('real CLI can inspect, resume, and cancel a persisted managed flow', async () => {
|
|
await startManagedAutonomyFlowFromHeartbeatTask({
|
|
rootDir: tempDir,
|
|
currentDir: tempDir,
|
|
task: {
|
|
name: 'manual-user-flow',
|
|
interval: '1h',
|
|
prompt: 'Manual lifecycle acceptance',
|
|
steps: [
|
|
{
|
|
name: 'approve',
|
|
prompt: 'Wait for manual approval',
|
|
waitFor: 'manual',
|
|
},
|
|
{
|
|
name: 'execute',
|
|
prompt: 'Execute approved work',
|
|
},
|
|
],
|
|
},
|
|
})
|
|
const [waitingFlow] = await listAutonomyFlows(tempDir)
|
|
expect(waitingFlow?.status).toBe('waiting')
|
|
|
|
const status = await runAutonomyCli(['status', '--deep'])
|
|
expect(status).toContain('Autonomy flows: 1')
|
|
expect(status).toContain('Waiting: 1')
|
|
|
|
const flows = await runAutonomyCli(['flows', '5'])
|
|
expect(flows).toContain(waitingFlow!.flowId)
|
|
expect(flows).toContain('waiting')
|
|
|
|
const detailBefore = await runAutonomyCli(['flow', waitingFlow!.flowId])
|
|
expect(detailBefore).toContain('Status: waiting')
|
|
expect(detailBefore).toContain('Current step: approve')
|
|
|
|
const resume = await runAutonomyCli(['flow', 'resume', waitingFlow!.flowId])
|
|
expect(resume).toContain('Prepared the next managed step')
|
|
expect(resume).toContain('Prompt:')
|
|
|
|
const detailAfterResume = await runAutonomyCli([
|
|
'flow',
|
|
waitingFlow!.flowId,
|
|
])
|
|
expect(detailAfterResume).toContain('Status: queued')
|
|
expect(detailAfterResume).toContain('Latest run:')
|
|
|
|
const cancel = await runAutonomyCli(['flow', 'cancel', waitingFlow!.flowId])
|
|
expect(cancel).toContain('Cancelled flow')
|
|
|
|
const [cancelledRun] = await listAutonomyRuns(tempDir)
|
|
const [cancelledFlow] = await listAutonomyFlows(tempDir)
|
|
expect(cancelledRun?.status).toBe('cancelled')
|
|
expect(cancelledFlow?.status).toBe('cancelled')
|
|
|
|
const detailAfterCancel = await runAutonomyCli([
|
|
'flow',
|
|
waitingFlow!.flowId,
|
|
])
|
|
expect(detailAfterCancel).toContain('Status: cancelled')
|
|
}, 30000)
|
|
})
|