mirror of
https://github.com/claude-code-best/claude-code.git
synced 2026-06-17 13:55:50 +00:00
Squash-merge of feat/autofix-pr-test (69 commits) onto upstream/main with -X ours strategy (upstream as authoritative for content conflicts). Key features brought in from fork: - LocalMemoryRecall + VaultHttpFetch tools (end-to-end wired) - /local-memory, /local-vault, /memory-stores, /skill-store interactive panels - /agents-platform, /schedule, /vault command scaffolding - /login: switch / replace / remove of workspace API key - statusline refactor (built-in status row, /statusline as info command) - autofix-pr command + workflow Conflict resolutions (upstream-wins): - 10 .js command stubs kept from upstream (alongside fork's .ts implementations) - src/components/BuiltinStatusLine.tsx accepted upstream's deletion (fork's wire-up references in StatusLine.tsx will be cleaned up next) Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
154 lines
4.9 KiB
TypeScript
154 lines
4.9 KiB
TypeScript
// NOTE: isolation flake, not pollution. The subprocess Bun.spawn'd in
|
|
// runAutonomyCli does not inherit the test runner's tsconfig path-alias
|
|
// resolution, so it reports `Cannot find module 'src/bootstrap/state.js'
|
|
// from src/utils/startupProfiler.ts` even when this file is run alone.
|
|
// Out of scope for the test-flake-fix pass; needs subprocess-launcher rework.
|
|
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)
|
|
})
|