- 新增 terminal-guide skill: 持久化终端使用指南 - 新增 sub-agent-guide skill: 子智能体使用指南 - 优化终端工具定义和执行逻辑 - 更新系统提示词以引用新 skills - 添加 utils/__init__.py 模块初始化文件 Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
13 KiB
13 KiB
| name | description |
|---|---|
| terminal-guide | 持久化终端使用指南。使用 terminal_session、terminal_input、terminal_snapshot 工具前必须阅读此技能。用于理解持久化终端与 run_command 的区别、output_wait 参数的正确含义、以及如何处理长时间运行的任务。 |
持久化终端使用指南
核心原则
持久化终端 = 长期运行的命令行会话
Terminal 工具提供了一个可以持续存在的终端会话,命令在其中运行不会因为你去做其他事情而中断。
与 run_command 的本质区别
| 特性 | terminal 系列 | run_command |
|---|---|---|
| 会话类型 | 持久会话,可以一直存在 | 一次性执行 |
| 超时行为 | output_wait 到期后命令继续运行 | timeout 到期强制终止命令 |
| 适用场景 | 长时间任务、多步操作、后台服务 | 快速查看信息、一次性命令 |
| 状态检查 | 可随时用 snapshot 查看 | 执行完就结束,无法再查看 |
| 典型用途 | pip install、git clone、npm run dev | ls、file、grep -n |
选择标准:
- 需要运行超过 30 秒?→ terminal
- 需要启动后继续做其他事?→ terminal
- 需要多次检查进度?→ terminal
- 只是快速查看信息?→ run_command
何时使用
适合 Terminal 的场景
长时间运行的任务:
- 下载大文件(wget、curl、git clone)
- 安装依赖(pip install、npm install、apt install)
- 编译构建(make、cargo build、npm run build)
- 数据处理(批量转换、压缩、解压)
需要保持运行的服务:
- 开发服务器(npm run dev、python -m http.server)
- 数据库服务(redis-server、mongod)
- 监控工具(tail -f、watch)
多步交互操作:
- cd 到某个目录后继续执行多个命令
- 激活虚拟环境后运行命令
- 设置环境变量后执行任务
需要监控进度:
- 运行测试套件并查看进度
- 批量处理文件并检查完成情况
- 长时间脚本执行并定期查看输出
不适合 Terminal 的场景
快速信息查看(用 run_command):
- 查看文件信息(file、stat、ls -la)
- 检查编码(iconv -l、file -i)
- 简单的 grep 定位
交互式程序(禁止使用):
- Python/Node REPL
- vim、nano、emacs
- top、htop
- 任何需要完整 TTY 的程序
一次性简单命令(用 run_command):
- 创建目录(mkdir)
- 移动文件(mv、cp)
- 查看环境变量(echo $PATH)
三步工作流
使用 terminal 工具的标准流程:
1. 打开会话
terminal_session(
action="open",
session_name="main", # 给会话起个名字
working_dir="project" # 可选:指定工作目录
)
会话命名建议:
main- 主要工作终端server- 运行开发服务器build- 执行构建任务test- 运行测试
2. 发送命令
terminal_input(
command="pip install -r requirements.txt",
session_name="main",
output_wait=30 # 收集输出的窗口期(秒)
)
3. 检查状态
terminal_snapshot(
session_name="main",
lines=50, # 可选:返回最近 50 行
max_chars=5000 # 可选:限制字符数
)
output_wait 的艺术
output_wait 不是命令超时! 这是最容易误解的地方。
它是什么
output_wait 是"收集输出的窗口期":
- 在这个时间内,工具会等待并收集命令的输出
- 窗口期结束后,命令继续在后台运行,只是不再等待输出
- 如果命令在窗口期内完成,会提前返回结果
它不是什么
- ❌ 不是命令超时(命令不会被终止)
- ❌ 不是最大执行时间(命令可以运行更久)
- ❌ 不是强制等待时间(命令完成会提前返回)
设置策略
场景 1:快速命令,只需确认启动
# 例如:启动开发服务器
terminal_input(
command="npm run dev",
output_wait=10 # 10秒足够看到启动信息
)
# 服务器会继续运行,你可以去做其他事情
场景 2:长任务,只需确认开始
# 例如:下载大文件
terminal_input(
command="wget https://example.com/large-file.zip",
output_wait=30 # 30秒确认下载开始
)
# 下载继续进行,稍后用 snapshot 检查进度
场景 3:必须等待完成的任务
# 例如:运行测试套件(预计 2 分钟完成)
terminal_input(
command="pytest tests/",
output_wait=150 # 设置足够长,确保完成
)
场景 4:可能长时间无输出的任务
# 例如:git clone 大仓库(可能长时间无输出)
# 技巧:在命令后加 ; echo "__DONE__" 作为完成标记
terminal_input(
command="git clone https://github.com/large/repo.git ; echo '__DONE__'",
output_wait=30 # 30秒确认开始
)
# 稍后用 snapshot 检查,看到 __DONE__ 就知道完成了
# 例如:批量处理文件
terminal_input(
command="for f in *.mp4; do ffmpeg -i \"$f\" \"${f%.mp4}.mp3\"; done ; echo '__DONE__'",
output_wait=60 # 给足时间,或稍后用 snapshot 检查 __DONE__
)
经验值参考
- 确认启动:5-10 秒
- 确认开始:20-30 秒
- 短任务完成:30-60 秒
- 中等任务完成:60-120 秒
- 长任务完成:120-300 秒
原则:宁可设长,不要设短。 设短了可能看不到关键输出,设长了最多就是等一会儿。
常见场景示例
场景 1:启动开发服务器(启动后不需要等待)
# 1. 打开终端
terminal_session(action="open", session_name="server")
# 2. 启动服务器
terminal_input(
command="npm run dev",
session_name="server",
output_wait=15 # 15秒看到启动信息即可
)
# 输出:Server running on http://localhost:3000
# 3. 服务器在后台运行,你可以继续做其他事情
# 需要时可以随时查看日志
terminal_snapshot(session_name="server", lines=30)
场景 2:下载大文件(启动后定期检查)
# 1. 打开终端
terminal_session(action="open", session_name="download")
# 2. 启动下载(加完成标记)
terminal_input(
command="wget https://example.com/dataset.tar.gz ; echo '__DONE__'",
session_name="download",
output_wait=20 # 20秒确认下载开始
)
# 3. 去做其他事情...
# 4. 稍后检查进度
terminal_snapshot(session_name="download")
# 输出:50% [======> ] 512MB 10.2MB/s eta 45s
# 5. 再次检查
terminal_snapshot(session_name="download")
# 输出:100% [=============] 1024MB 10.5MB/s in 98s
# __DONE__ ← 看到这个就知道完成了
场景 3:运行测试套件(需要等待完成)
# 1. 打开终端
terminal_session(action="open", session_name="test", working_dir="backend")
# 2. 运行测试(预计 2 分钟)
terminal_input(
command="pytest tests/ -v",
session_name="test",
output_wait=150 # 设置足够长,等待完成
)
# 输出:完整的测试结果
# 3. 如果不确定是否完成,再检查一次
terminal_snapshot(session_name="test", lines=20)
场景 4:批量处理文件(需要等待完成)
# 1. 打开终端
terminal_session(action="open", session_name="process")
# 2. 批量转换(预计 5 分钟)
terminal_input(
command="for img in *.png; do convert \"$img\" -resize 800x600 \"thumb_$img\"; done",
session_name="process",
output_wait=300 # 5分钟
)
# 3. 检查是否完成
terminal_snapshot(session_name="process")
场景 5:多步操作(cd + 虚拟环境 + 命令)
# 1. 打开终端
terminal_session(action="open", session_name="main")
# 2. 切换目录
terminal_input(
command="cd backend && source venv/bin/activate",
session_name="main",
output_wait=5
)
# 3. 在虚拟环境中安装依赖
terminal_input(
command="pip install -r requirements.txt",
session_name="main",
output_wait=60
)
# 4. 运行应用
terminal_input(
command="python app.py",
session_name="main",
output_wait=10
)
常见陷阱
1. 把 output_wait 当作命令超时
❌ 错误理解:
# "我想让命令最多运行 30 秒,超时就终止"
terminal_input(command="long_task.sh", output_wait=30)
✅ 正确做法:
# 如果需要强制超时终止,使用 run_command
run_command(command="long_task.sh", timeout=30)
# 如果任务可以持续运行,使用 terminal
terminal_input(command="long_task.sh", output_wait=30)
# 30秒后命令继续运行,你可以稍后用 snapshot 检查
2. 不检查命令是否完成就继续输入
❌ 错误:
terminal_input(command="npm install", output_wait=10)
# 10秒后立即执行下一个命令,但 npm install 可能还在运行
terminal_input(command="npm run build", output_wait=10)
# 可能会冲突或失败
✅ 正确:
terminal_input(command="npm install", output_wait=10)
# 检查是否完成
terminal_snapshot(session_name="main")
# 如果看到还在运行,等待或增加 output_wait
# 确认完成后再继续
terminal_input(command="npm run build", output_wait=60)
3. 在终端中启动交互式程序
❌ 错误:
terminal_input(command="python", output_wait=5) # 启动 Python REPL
terminal_input(command="print('hello')", output_wait=5) # 不会工作
❌ 错误:
terminal_input(command="vim file.txt", output_wait=5) # vim 需要完整 TTY
✅ 正确:
# 使用 run_python 执行 Python 代码
run_python(code="print('hello')", timeout=5)
# 使用 edit_file 修改文件
edit_file(file_path="file.txt", old_string="...", new_string="...")
4. 对快速命令使用 terminal
❌ 低效:
terminal_session(action="open", session_name="check")
terminal_input(command="ls -la", session_name="check", output_wait=5)
terminal_session(action="close", session_name="check")
✅ 高效:
run_command(command="ls -la", timeout=5)
5. 忘记关闭不再使用的终端
# 任务完成后,记得关闭终端释放资源
terminal_session(action="close", session_name="build")
或者重置终端(清空输出但保持会话):
terminal_session(action="reset", session_name="main")
6. output_wait 设置过短导致看不到关键输出
❌ 问题:
# pip install 可能需要 30-60 秒
terminal_input(command="pip install tensorflow", output_wait=5)
# 5秒后返回,只看到 "Collecting tensorflow...",看不到安装结果
✅ 改进:
# 方案 1:设置足够长的 output_wait
terminal_input(command="pip install tensorflow", output_wait=90)
# 方案 2:短 output_wait 确认开始,稍后用 snapshot 检查
terminal_input(command="pip install tensorflow", output_wait=10)
# ... 做其他事情 ...
terminal_snapshot(session_name="main") # 检查是否完成
检查命令是否完成
使用 terminal_snapshot 判断命令状态的技巧:
技巧 1:使用完成标记(推荐)
对于可能长时间无输出的任务,在命令后加 ; echo "__DONE__":
# git clone 大仓库
terminal_input(
command="git clone https://github.com/large/repo.git ; echo '__DONE__'",
output_wait=30
)
# 稍后检查
terminal_snapshot(session_name="main")
# 看到 __DONE__ 就知道完成了
为什么有用:
- git clone、wget、scp 等命令可能长时间无输出
- 完成后也可能没有明显的提示信息
__DONE__是明确的完成信号,不会误判
技巧 2:看提示符
# 命令还在运行(没有提示符)
Downloading... 45%
# 命令已完成(出现提示符)
Downloading... 100%
user@host:~/project$
技巧 3:看输出内容
# 安装完成的标志
Successfully installed package-1.0.0
# 测试完成的标志
===== 10 passed in 2.5s =====
# 构建完成的标志
Build completed successfully
# 下载完成的标志
100% [=============] 1024MB
技巧 4:看进程状态
# 如果不确定,可以检查进程
terminal_input(command="ps aux | grep your_process", output_wait=5)
总结
Terminal 工具的本质: 持久化会话,让命令可以长期运行,随时查看状态。
核心工作流:
terminal_session(action="open")- 打开会话terminal_input(command="...", output_wait=N)- 发送命令terminal_snapshot()- 检查状态
关键理解:
output_wait是收集输出的窗口期,不是命令超时- 命令会在后台持续运行,即使 output_wait 结束
- 随时可以用
snapshot查看最新状态
选择标准:
- 长时间任务、后台服务、多步操作 → terminal
- 快速查看信息、一次性命令 → run_command
- 交互式程序 → 禁止使用,改用专用工具
最佳实践:
- output_wait 宁长勿短
- 不确定是否完成时,先用 snapshot 检查
- 任务完成后关闭或重置终端
- 给会话起有意义的名字