mirror of
https://github.com/claude-code-best/claude-code.git
synced 2026-06-18 22:35:51 +00:00
feat: 添加 autonomy 自主模式命令系统
- 新增 autonomy CLI handler 和交互式面板 - 新增 autonomyCommandSpec 命令规范定义 - 新增 autonomyAuthority 权限控制 - 新增 autonomyStatus 状态管理 - 注册 CLI 子命令 (claude autonomy status/runs/flows/flow) Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
213
src/cli/handlers/autonomy.ts
Normal file
213
src/cli/handlers/autonomy.ts
Normal file
@@ -0,0 +1,213 @@
|
||||
import {
|
||||
formatAutonomyFlowDetail,
|
||||
formatAutonomyFlowsList,
|
||||
formatAutonomyFlowsStatus,
|
||||
getAutonomyFlowById,
|
||||
listAutonomyFlows,
|
||||
requestManagedAutonomyFlowCancel,
|
||||
} from '../../utils/autonomyFlows.js'
|
||||
import {
|
||||
formatAutonomyRunsList,
|
||||
formatAutonomyRunsStatus,
|
||||
listAutonomyRuns,
|
||||
markAutonomyRunCancelled,
|
||||
resumeManagedAutonomyFlowPrompt,
|
||||
} from '../../utils/autonomyRuns.js'
|
||||
import {
|
||||
formatAutonomyDeepStatus,
|
||||
formatAutonomyDeepStatusSections,
|
||||
type AutonomyDeepStatusSectionId,
|
||||
} from '../../utils/autonomyStatus.js'
|
||||
import {
|
||||
AUTONOMY_USAGE,
|
||||
parseAutonomyArgs,
|
||||
} from '../../utils/autonomyCommandSpec.js'
|
||||
import {
|
||||
enqueuePendingNotification,
|
||||
removeByFilter,
|
||||
} from '../../utils/messageQueueManager.js'
|
||||
|
||||
export function parseAutonomyLimit(raw?: string | number): number {
|
||||
const parsed = typeof raw === 'number' ? raw : Number.parseInt(raw ?? '', 10)
|
||||
if (!Number.isFinite(parsed) || parsed <= 0) {
|
||||
return 10
|
||||
}
|
||||
return Math.min(parsed, 50)
|
||||
}
|
||||
|
||||
export async function getAutonomyStatusText(options?: {
|
||||
deep?: boolean
|
||||
}): Promise<string> {
|
||||
const [runs, flows] = await Promise.all([
|
||||
listAutonomyRuns(),
|
||||
listAutonomyFlows(),
|
||||
])
|
||||
|
||||
if (options?.deep) {
|
||||
return formatAutonomyDeepStatus({ runs, flows })
|
||||
}
|
||||
|
||||
return [
|
||||
formatAutonomyRunsStatus(runs),
|
||||
formatAutonomyFlowsStatus(flows),
|
||||
].join('\n')
|
||||
}
|
||||
|
||||
export async function getAutonomyDeepSectionText(
|
||||
sectionId: AutonomyDeepStatusSectionId,
|
||||
): Promise<string> {
|
||||
const [runs, flows] = await Promise.all([
|
||||
listAutonomyRuns(),
|
||||
listAutonomyFlows(),
|
||||
])
|
||||
const sections = await formatAutonomyDeepStatusSections({ runs, flows })
|
||||
const section = sections.find(item => item.id === sectionId)
|
||||
if (!section) {
|
||||
return `Autonomy deep status section not found: ${sectionId}`
|
||||
}
|
||||
return [`# ${section.title}`, section.content].join('\n')
|
||||
}
|
||||
|
||||
export async function autonomyStatusHandler(options?: {
|
||||
deep?: boolean
|
||||
}): Promise<void> {
|
||||
process.stdout.write(`${await getAutonomyStatusText(options)}\n`)
|
||||
}
|
||||
|
||||
export async function getAutonomyRunsText(
|
||||
limit?: string | number,
|
||||
): Promise<string> {
|
||||
return formatAutonomyRunsList(
|
||||
await listAutonomyRuns(),
|
||||
parseAutonomyLimit(limit),
|
||||
)
|
||||
}
|
||||
|
||||
export async function autonomyRunsHandler(
|
||||
limit?: string | number,
|
||||
): Promise<void> {
|
||||
process.stdout.write(`${await getAutonomyRunsText(limit)}\n`)
|
||||
}
|
||||
|
||||
export async function getAutonomyFlowsText(
|
||||
limit?: string | number,
|
||||
): Promise<string> {
|
||||
return formatAutonomyFlowsList(
|
||||
await listAutonomyFlows(),
|
||||
parseAutonomyLimit(limit),
|
||||
)
|
||||
}
|
||||
|
||||
export async function autonomyFlowsHandler(
|
||||
limit?: string | number,
|
||||
): Promise<void> {
|
||||
process.stdout.write(`${await getAutonomyFlowsText(limit)}\n`)
|
||||
}
|
||||
|
||||
export async function getAutonomyFlowText(flowId: string): Promise<string> {
|
||||
return formatAutonomyFlowDetail(await getAutonomyFlowById(flowId))
|
||||
}
|
||||
|
||||
export async function autonomyFlowHandler(flowId: string): Promise<void> {
|
||||
process.stdout.write(`${await getAutonomyFlowText(flowId)}\n`)
|
||||
}
|
||||
|
||||
export async function cancelAutonomyFlowText(
|
||||
flowId: string,
|
||||
options?: {
|
||||
removeQueuedInMemory?: boolean
|
||||
},
|
||||
): Promise<string> {
|
||||
const cancelled = await requestManagedAutonomyFlowCancel({ flowId })
|
||||
if (!cancelled) {
|
||||
return 'Autonomy flow not found.'
|
||||
}
|
||||
if (!cancelled.accepted) {
|
||||
return `Autonomy flow ${flowId} is already terminal (${cancelled.flow.status}).`
|
||||
}
|
||||
|
||||
let removedCount = 0
|
||||
if (options?.removeQueuedInMemory) {
|
||||
const removed = removeByFilter(cmd => cmd.autonomy?.flowId === flowId)
|
||||
removedCount = removed.length
|
||||
for (const command of removed) {
|
||||
if (command.autonomy?.runId) {
|
||||
await markAutonomyRunCancelled(command.autonomy.runId)
|
||||
}
|
||||
}
|
||||
} else {
|
||||
for (const runId of cancelled.queuedRunIds) {
|
||||
await markAutonomyRunCancelled(runId)
|
||||
}
|
||||
removedCount = cancelled.queuedRunIds.length
|
||||
}
|
||||
|
||||
return cancelled.flow.status === 'running'
|
||||
? `Cancellation requested for flow ${flowId}. The current step is still running, and no new steps will be started.`
|
||||
: `Cancelled flow ${flowId}. Removed ${removedCount} queued step(s).`
|
||||
}
|
||||
|
||||
export async function autonomyFlowCancelHandler(flowId: string): Promise<void> {
|
||||
process.stdout.write(`${await cancelAutonomyFlowText(flowId)}\n`)
|
||||
}
|
||||
|
||||
export async function resumeAutonomyFlowText(
|
||||
flowId: string,
|
||||
options?: {
|
||||
enqueueInMemory?: boolean
|
||||
},
|
||||
): Promise<string> {
|
||||
const command = await resumeManagedAutonomyFlowPrompt({ flowId })
|
||||
if (!command) {
|
||||
return 'Autonomy flow is not waiting or was not found.'
|
||||
}
|
||||
|
||||
if (options?.enqueueInMemory) {
|
||||
enqueuePendingNotification(command)
|
||||
return `Queued the next managed step for flow ${flowId}.`
|
||||
}
|
||||
|
||||
const runId = command.autonomy?.runId ?? 'unknown'
|
||||
return [
|
||||
`Prepared the next managed step for flow ${flowId}.`,
|
||||
`Run ID: ${runId}`,
|
||||
'',
|
||||
'Prompt:',
|
||||
typeof command.value === 'string' ? command.value : String(command.value),
|
||||
].join('\n')
|
||||
}
|
||||
|
||||
export async function autonomyFlowResumeHandler(flowId: string): Promise<void> {
|
||||
process.stdout.write(`${await resumeAutonomyFlowText(flowId)}\n`)
|
||||
}
|
||||
|
||||
export async function getAutonomyCommandText(
|
||||
args: string,
|
||||
options?: {
|
||||
enqueueInMemory?: boolean
|
||||
removeQueuedInMemory?: boolean
|
||||
},
|
||||
): Promise<string> {
|
||||
const parsed = parseAutonomyArgs(args)
|
||||
|
||||
switch (parsed.type) {
|
||||
case 'status':
|
||||
return getAutonomyStatusText({ deep: parsed.deep })
|
||||
case 'runs':
|
||||
return getAutonomyRunsText(parsed.limit)
|
||||
case 'flows':
|
||||
return getAutonomyFlowsText(parsed.limit)
|
||||
case 'flow-detail':
|
||||
return getAutonomyFlowText(parsed.flowId)
|
||||
case 'flow-cancel':
|
||||
return cancelAutonomyFlowText(parsed.flowId, {
|
||||
removeQueuedInMemory: options?.removeQueuedInMemory,
|
||||
})
|
||||
case 'flow-resume':
|
||||
return resumeAutonomyFlowText(parsed.flowId, {
|
||||
enqueueInMemory: options?.enqueueInMemory,
|
||||
})
|
||||
case 'usage':
|
||||
return AUTONOMY_USAGE
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user