新增 docs/agentic-design/ 教育文档,全中文+英文技术术语,覆盖: - README.md: 总索引,整体架构图,阅读指南 - 00-codebase-tour.md: 代码库全景,递归覆盖所有重要目录 - 01-agent-loop.md: Query loop 状态机,token budget,compaction - 02-tool-system.md: Tool interface,权限检查,并发执行,sibling abort - 03-multi-agent-coordination.md: Agent 类型,coordinator 4 阶段,XML 通信 - 04-permission-system.md: Permission mode,rule system,YOLO 分类器,denial tracking - 05-context-and-memory.md: System prompt 分层,compaction,auto-dream 后台巩固 - 06-feature-gating.md: Compile-time feature,runtime flags,Bun dead-code elimination 总计 ~2.5 小时阅读,每篇含 Design Decision 专栏和常见误解纠正 https://claude.ai/code/session_017Vdqo9B8eTXiEDcqnv9DxM
8 KiB
核心 Query Loop:Agent 的心脏
为什么需要 Loop?
想象一个简单的"一问一答"的 AI:用户问 → API 回复 → 完成。但 agent 不一样:
- 用户问:"帮我修复 bug"
- Agent 读取文件 → 分析问题 → 尝试修复 → 运行测试 → 测试失败 → 再次修改 → 再次测试 → 成功
这个过程需要多轮往返,每一轮中 Agent 决定"下一步应该执行哪个 tool"。这就是 Query Loop。
核心概念:状态机
Query loop 本质上是一个状态机,在以下几个状态间循环:
START
↓
QUERY (提交 prompt 给 Claude API)
↓
TOOL_USE (API 返回 tool_use block,包含 tool 名 + 输入)
↓
执行 Tool (BashTool, FileEditTool, 等等)
↓
RESULT (Tool 返回结果)
↓
提交结果给 Claude (作为 user 消息)
↓
QUERY (继续循环)
↓
END (API 返回 stop_reason="end_turn" 或其他终止条件)
代码位置
关键文件:query.ts (~1700 行)
这个文件包含:
- Loop 的完整实现 (使用
AsyncGenerator) - 状态跟踪
- Tool 调用的编排
- Streaming 支持
- Compaction 触发
- 错误处理和恢复
你应该从 query.ts 的函数签名开始:
export function* query(...): AsyncGenerator<QueryMessage | QueryEvent, ...>
它返回一个 AsyncGenerator,每次 yield 都是一个"事件"(API 消息、tool 调用、错误等)。
Loop 的生命周期
阶段 1: 初始化
当你调用 query() 时:
messages数组初始化(包含 system prompt + 历史消息)token_budget初始化(计算剩余 token)- Compaction 检查(如果历史太长,自动压缩)
- Abort controller 初始化(支持中途停止)
阶段 2: 发送 Prompt
// 构建消息数组
let messages = [system_prompt, ...history];
// 调用 Claude API(使用缓存)
let response = await claude.messages.create({
messages,
system: system_prompt,
max_tokens: remaining_budget,
stream: true, // 重要:流式输出
...config
});
Streaming 很关键——API 会逐步返回:
- First chunk: 可能包含
message.start - Content blocks:
text_delta,tool_useblock 开始 - Stop reason: 表示本轮结束
阶段 3: 处理 Streaming
当 API 返回 tool_use block 时,loop 收集完整的参数,然后:
接收 tool_use block
↓
验证 tool 名称是否存在
↓
检查权限 (useCanUseTool hook)
↓
执行 tool (StreamingToolExecutor)
↓
Yield tool 结果给消费者
↓
添加到 messages 作为 user 消息
阶段 4: 工具执行
Tool 执行由 services/tools/StreamingToolExecutor.ts 处理:
- 并发:最多 5 个 tool 同时运行
- 顺序输出:结果按调用顺序返回(即使执行顺序不同)
- Sibling abort:一个失败会 cancel 兄弟 tool
阶段 5: 检查终止条件
每一轮后,loop 检查是否应该继续:
stop_reason == "end_turn"? → 正常结束
stop_reason == "max_tokens"? → Token 超限,触发 compaction
consecutive_errors >= 3? → 连续错误,放弃
max_turns_reached? → 超过最大轮数
abort_signal? → 用户中断
Token Budget:Context Window 的有限性
Context window 是有限的(比如 200K tokens)。这里的逻辑:
available_tokens = context_window_size - system_prompt_tokens - reserved_buffer
每一轮后:
usage = response.usage (API 返回的实际使用)
remaining = available_tokens - usage.input_tokens - usage.output_tokens
if remaining < threshold:
触发 compaction (压缩历史)
if remaining < min_required:
拒绝继续,返回错误
Threshold 通常设得比较激进,比如还剩 20% 时就开始压缩,以避免在真的耗尽时措手不及。
相关文件:query/tokenBudget.ts
Compaction:自动历史压缩
当 token 预算吃紧时,loop 会自动压缩历史:
- Analyzer phase: 扫描历史消息,找出"长输出消息"(比如代码块)
- Prioritize: 对话越早,压缩优先级越高
- Compress: 使用 Claude API 总结这段历史,生成 200 行以内的 summary
- Replace: 把原始的 20 条消息替换为 1 条 summary 消息
这保证了 loop 可以持续运行,即使初始历史很长。
Streaming 与 Tool Call 的交织
这是一个复杂的地方。API 可能返回:
message {
content: [
{ type: "text", text: "让我先读文件..." },
{ type: "tool_use", id: "...", name: "FileReadTool", ... },
{ type: "text", text: "现在我看到问题了..." },
{ type: "tool_use", id: "...", name: "FileEditTool", ... },
]
}
所以一条消息中可能混合了文本和 tool call。Loop 需要:
- 收集每个 tool_use block 的完整输入参数(可能分段到达)
- 在 tool_use 结束时立即执行(不等待消息全部返回)
- 继续接收后续的文本或 tool call
- 当消息完全接收后,将所有 tool 结果作为一条 user 消息发回
这使得 interleaved thinking 成为可能:模型可以一边思考一边调用工具。
Design Decision 专栏
为什么用 AsyncGenerator 而不是 callback?
不好的设计(callback):
query(messages, {
onMessage: (msg) => {...},
onToolCall: (tool, input) => {...},
onError: (err) => {...},
onProgress: (progress) => {...},
})
问题:
- 4 种不同的 callback,难以理解执行顺序
- 消费者需要管理状态机来追踪"当前在哪个阶段"
- 测试困难
更好的设计(Generator):
for await (const event of query(messages, config)) {
if (event.type === 'message') {...}
else if (event.type === 'tool_use') {...}
else if (event.type === 'error') {...}
}
优点:
- 单一的
for await...of循环,清晰的顺序 - 每个 event 都是一个联合类型,编译器可以帮你检查
- 可以在 loop 中添加复杂的条件逻辑(abort, timeout 等)
- 易于测试(可以模拟一个生成事件的 generator)
Generator 让"一系列事件"变得一级公民。
为什么需要 Compaction?
不做 compaction:
- 100 轮 agent 工作后,messages 数组有几千条消息
- 提交给 API 时,prompt cache hit 率下降(因为前缀不再是"静态部分")
- Token 浪费在重复的对话历史上
做 compaction:
- 定期将"已经达成共识的历史部分"总结成一条消息
- 保持 prompt cache 的"静态前缀"有效
- 节省 token,延长 agent 能工作的轮数
常见误解
误解 1:"Query loop 就是不断问 Claude?"
实际:Loop 实现的是一个两层状态机:
- 外层:Claude API 的请求/响应循环
- 内层:Tool 的执行和结果收集
tool 执行完全不涉及 Claude API 的额外调用(除非 tool 本身调用 API)。
误解 2:"Compaction 会丢失信息?"
实际:Compaction 使用 Claude 自己来总结历史,所以关键信息被保留。丢失的只是"冗余的解释"或"中间尝试",这些对后续工作没用。
误解 3:"Token budget 追踪是精确的?"
实际:API 返回的 usage 是经过舍入的(某些模型),所以我们的估计可能有 ±5% 的误差。这就是为什么我们用激进的阈值(20% 时就开始压缩)而不是等到 100% 才反应。
关键要点
- Query loop 是异步生成器,yield 事件流而不是回调
- 循环的关键状态转移:QUERY → TOOL_USE → RESULT → QUERY
- Token budget 必须主动管理,compaction 是自动压缩的关键
- Streaming 支持 interleaved thinking,一条消息中可以混合文本和 tool call
- Compaction 会定期触发,保持 history 可控,并维护 prompt cache 的有效性
深入阅读
query.ts:完整 loop 实现(必读)query/tokenBudget.ts:Token 追踪逻辑services/tools/StreamingToolExecutor.ts:Tool 执行编排query/stopHooks.ts:终止条件检查
下一步:去了解 Tool 系统是如何设计的,使得这个 loop 可以灵活地执行任意 tool。