一个功能强大的 Model Context Protocol (MCP) 服务器,基于 TypeScript 和 node-pty 实现持久化终端会话管理。即使客户端断开连接,终端命令也会继续运行,特别适合 Claude、Cursor、Cline 等 AI 助手执行长时间任务。
- 长期运行:创建、复用、管理长期运行的 Shell 会话
- 断线续传:客户端断开后终端继续运行,重连后可继续操作
- 多会话管理:同时管理多个独立的终端会话
- 自动清理:超时会话自动清理,避免资源泄漏
- 循环缓冲区:可配置大小(默认 10,000 行),自动管理内存
- 多种读取模式:
full:完整输出head:只读取开头 N 行tail:只读取末尾 N 行head-tail:同时读取开头和末尾
- 增量读取:使用
since参数只读取新增内容 - Token 估算:自动估算输出的 token 数量,方便 AI 控制上下文
- 自动检测:识别常见的进度动画字符(⠋⠙⠹⠸⠼⠴⠦⠧⠇⠏, ◐◓◑◒ 等)
- 智能节流:减少
npm install、yarn、pnpm等命令的噪音输出 - 保留关键信息:压缩动画的同时保留真实日志
- 灵活配置:可通过环境变量或参数控制开关
- 实时终端:基于 xterm.js 的终端渲染,支持完整 ANSI 颜色
- WebSocket 推送:终端输出实时显示,无需刷新
- 交互操作:直接在浏览器中发送命令、查看输出
- 多实例支持:自动端口分配,支持多个 AI 客户端同时使用
- VS Code 风格:暗色主题,简洁美观的界面设计
- MCP 协议:原生支持 Claude Desktop、Claude Code、Cursor、Cline 等客户端
- REST API:提供 HTTP 接口,方便非 MCP 场景集成
- 严格兼容:完全符合 MCP stdio 协议规范,stdout 纯净无污染
- 输出稳定检测:
wait_for_output工具确保获取完整输出 - 交互式应用支持:完美支持 vim、npm create 等交互式程序
- ANSI 转义序列:正确处理终端控制字符
- 错误恢复:自动重连、异常处理机制
npm install # 安装依赖
npm run build # 编译 TypeScript 到 dist/
npm start # 通过 stdio 启动 MCP 服务器开发阶段可直接运行源代码:
npm run dev # MCP 服务器 (tsx)
npm run dev:rest # REST 服务器 (tsx)启用调试日志(输出到 stderr,不会干扰 MCP 通信):
MCP_DEBUG=true npm startnpm run example:basic # 基础操作:创建 → 写入 → 读取 → 终止
npm run example:smart # 智能读取:head/tail/head-tail 模式演示
npm run example:spinner # Spinner 压缩功能演示
npm run example:webui # Web UI 功能演示
npm run test:tools # 全量验证所有 MCP 工具
npm run test:fixes # 关键修复的回归测试在 MCP 配置文件中添加以下配置:
配置文件位置: ~/Library/Application Support/Claude/claude_desktop_config.json
{
"mcpServers": {
"persistent-terminal": {
"command": "node",
"args": [
"/absolute/path/to/node-pty/dist/index.js"
],
"env": {
"MAX_BUFFER_SIZE": "10000",
"SESSION_TIMEOUT": "86400000",
"COMPACT_ANIMATIONS": "true",
"ANIMATION_THROTTLE_MS": "100"
}
}
}
}重要提示:请将 /absolute/path/to/node-pty 替换为实际的安装目录绝对路径。
使用命令行快速添加 MCP 服务器:
claude mcp add persistent-terminal \
--env MAX_BUFFER_SIZE=10000 \
--env SESSION_TIMEOUT=86400000 \
--env COMPACT_ANIMATIONS=true \
--env ANIMATION_THROTTLE_MS=100 \
-- node /absolute/path/to/node-pty/dist/index.js重要提示:请将 /absolute/path/to/node-pty 替换为实际的安装目录绝对路径。
示例(假设项目在 /Users/admin/Desktop/node-pty):
claude mcp add persistent-terminal \
--env MAX_BUFFER_SIZE=10000 \
--env SESSION_TIMEOUT=86400000 \
--env COMPACT_ANIMATIONS=true \
--env ANIMATION_THROTTLE_MS=100 \
-- node /Users/admin/Desktop/node-pty/dist/index.js配置方式与 Claude Desktop 类似,请参考各客户端的 MCP 配置文档。
对于 Codex,在 .codex/config.toml 文件中添加以下配置:
# MCP Server Configuration (TOML Format)
# 用于配置 persistent-terminal MCP 服务器
[mcp_servers.persistent-terminal]
command = "node"
args = ["/absolute/path/to/node-pty/dist/index.js"]
[mcp_servers.persistent-terminal.env]
MAX_BUFFER_SIZE = "10000"
SESSION_TIMEOUT = "86400000"
COMPACT_ANIMATIONS = "true"
ANIMATION_THROTTLE_MS = "100"重要提示:请将 /absolute/path/to/node-pty 替换为实际的安装目录绝对路径。
| 变量 | 说明 | 默认值 |
|---|---|---|
MAX_BUFFER_SIZE |
缓冲区最大行数 | 10000 |
SESSION_TIMEOUT |
会话超时时间(毫秒) | 86400000 (24小时) |
COMPACT_ANIMATIONS |
是否启用 Spinner 压缩 | true |
ANIMATION_THROTTLE_MS |
动画节流时间(毫秒) | 100 |
MCP_DEBUG |
是否启用调试日志 | false |
| 工具 | 作用 | 主要参数 |
|---|---|---|
create_terminal |
创建持久终端会话 | shell, cwd, env, cols, rows |
create_terminal_basic |
精简版创建入口 | shell, cwd |
write_terminal |
向终端写入命令 | terminalId, input, appendNewline |
read_terminal |
读取缓冲输出 | terminalId, mode, since, stripSpinner |
wait_for_output |
等待输出稳定 | terminalId, timeout, stableTime |
get_terminal_stats |
查看统计信息 | terminalId |
list_terminals |
列出所有活跃终端 | 无 |
kill_terminal |
终止会话 | terminalId, signal |
open_terminal_ui 🆕 |
打开 Web 管理界面 | port, autoOpen |
创建一个新的持久化终端会话。
参数:
shell(可选): Shell 类型,如/bin/bash、/bin/zshcwd(可选): 工作目录env(可选): 环境变量对象cols(可选): 终端列数,默认 80rows(可选): 终端行数,默认 24
返回:
terminalId: 终端 IDstatus: 状态pid: 进程 IDshell: Shell 类型cwd: 工作目录
向终端发送命令或输入。
参数:
terminalId: 终端 IDinput: 要发送的内容appendNewline(可选): 是否自动添加换行符,默认 true
提示:默认会自动添加换行符执行命令,如需发送原始控制字符(如方向键),请设置 appendNewline: false。
读取终端的缓冲输出,支持多种智能截断模式。
参数:
terminalId: 终端 IDmode(可选): 读取模式full: 完整输出(默认)head: 只读取开头tail: 只读取末尾head-tail: 同时读取开头和末尾
since(可选): 从第 N 行开始读取(增量读取)maxLines(可选): 最大行数,默认 1000headLines(可选): head 模式的行数,默认 50tailLines(可选): tail 模式的行数,默认 50stripSpinner(可选): 是否压缩 Spinner 动画
返回:
output: 输出内容totalLines: 总行数lineRange: 实际返回的行范围estimatedTokens: 估算的 token 数量truncated: 是否被截断spinnerCompacted: 是否进行了 Spinner 压缩
等待终端输出稳定后再读取,确保获取完整输出。
参数:
terminalId: 终端 IDtimeout(可选): 最大等待时间(毫秒),默认 5000stableTime(可选): 稳定时间(毫秒),默认 500
使用场景:
- 执行命令后确保获取完整输出
- 等待交互式应用启动完成
- 等待长时间运行的命令完成
启动一个基于浏览器的可视化终端管理界面。
参数:
port(可选): 端口号,默认从 3002 开始自动查找autoOpen(可选): 是否自动打开浏览器,默认 true
返回:
url: Web UI 地址port: 实际使用的端口mode: 启动模式(new/existing)autoOpened: 是否自动打开了浏览器
- 📊 终端列表:查看所有终端的状态、PID、Shell、工作目录等信息
- 🖥️ 实时终端:使用 xterm.js 渲染终端输出,支持 ANSI 颜色
- ⚡ 实时更新:WebSocket 推送,终端输出实时显示
- ⌨️ 交互操作:直接在浏览器中发送命令
- 🎨 VS Code 风格:暗色主题,简洁美观
- 🔄 自动端口:支持多实例,自动避免端口冲突
在 Claude 或其他 MCP 客户端中说:
请打开终端管理界面
或者直接运行测试脚本:
npm run test:webui详细使用说明见 Web UI 使用指南。
如果需要 HTTP 接口,可启动 REST 版本:
npm run start:rest服务器默认监听 3001 端口(可配置),端点与 MCP 工具一一对应:
| 端点 | 方法 | 说明 |
|---|---|---|
/api/terminals |
POST | 创建终端 |
/api/terminals |
GET | 列出所有终端 |
/api/terminals/:id |
GET | 获取终端详情 |
/api/terminals/:id |
DELETE | 终止终端 |
/api/terminals/:id/input |
POST | 发送命令 |
/api/terminals/:id/output |
GET | 读取输出 |
/api/terminals/:id/stats |
GET | 获取统计信息 |
persistent-terminal-mcp/
├── src/ # TypeScript 源码
│ ├── index.ts # MCP 服务器入口
│ ├── mcp-server.ts # MCP 服务器实现
│ ├── terminal-manager.ts # 终端管理器
│ ├── output-buffer.ts # 输出缓冲区
│ ├── web-ui-manager.ts # Web UI 管理器
│ ├── web-ui-server.ts # Web UI 服务器
│ ├── rest-server.ts # REST API 服务器
│ ├── types.ts # 类型定义
│ ├── __tests__/ # 单元测试
│ └── examples/ # 示例脚本
├── dist/ # 编译后的 JavaScript
├── public/ # Web UI 静态文件
├── docs/ # 文档
│ ├── guides/ # 使用指南
│ ├── reference/ # 技术参考
│ ├── clients/ # 客户端配置
│ └── zh/ # 中文文档
├── tests/ # 测试套件
│ └── integration/ # 集成测试
└── scripts/ # 辅助脚本
- 📖 完整文档索引
- 🚨 修复文档索引
- 🧪 集成测试说明
- 🌐 Web UI 使用指南
- 使用指南:使用说明 | 故障排查 | MCP 配置
- 技术参考:技术细节 | 工具总结
- 修复文档:Stdio 修复 | Cursor 修复 | 终端修复
- 客户端配置:Claude Desktop/Code
本 MCP 服务器严格遵循 MCP 协议,确保 stdout 只包含 JSON-RPC 消息,所有日志输出到 stderr。这保证了与 Cursor 等严格客户端的完全兼容。详见 Stdio 修复文档。
完全兼容 Cursor 及其他要求严格 JSON-RPC 通信的 MCP 客户端。快速设置见 快速修复指南。
支持交互式应用(vim、npm create 等),正确处理 ANSI 转义序列。详见 终端修复文档。
使用 wait_for_output 工具确保命令执行后获取完整输出,避免读取不完整的数据。
npm test # 运行所有单元测试
npm run test:integration # 运行所有集成测试
npm run test:all # 运行所有测试npm run test:integration:stdio # Stdio 纯净性测试
npm run test:integration:cursor # Cursor 场景测试
npm run test:integration:terminal # 终端功能测试欢迎提 Issue 或 PR!详细流程与代码规范见 CONTRIBUTING.md。
- Fork 本仓库
- 创建特性分支 (
git checkout -b feature/AmazingFeature) - 提交更改 (
git commit -m 'Add some AmazingFeature') - 推送到分支 (
git push origin feature/AmazingFeature) - 开启 Pull Request
本项目以 MIT 许可证 发布。
- node-pty - 强大的 PTY 库
- Model Context Protocol - MCP 协议规范
- xterm.js - 优秀的终端模拟器
最后更新: 2025-10-08 版本: 1.0.1