modelcontextprotocol/ext-tasks
MCP Tasks 的完整规范和文档。
为什么不直接阻塞?
你可以一直保持连接,直到工作完成。但 Tasks 解决的是阻塞无法解决的问题:- 不需要长期连接。 阻塞会在整个操作期间占用连接。许多 clients 和传输中间层都有超时限制,使得超过几秒的阻塞并不现实。
- 崩溃恢复能力。 Task ID 是持久句柄。如果 client 断开连接或重启,可以使用同一个 ID 继续轮询。
- 进度可见性。 Tasks 携带状态元数据(
working、input_required、completed、failed、cancelled)和可选状态消息,让 clients 可以看到进度。 - 执行中交互。 当 task 需要输入时(例如用于用户确认的 elicitation),它会进入
input_required并暴露请求。Client 通过tasks/update响应,不需要第二条连接,也不需要未经请求的 server-to-client 消息。 - 由 server 决定。 Server 按请求决定是否创建 task。Clients 只需通过扩展能力 opt in 一次,然后处理返回的任意结果形态。不需要按 tool 预热,也不需要按请求设置 flag。
Tasks 如何工作
Tasks 扩展了标准请求流程。当 server 判断某个请求会长时间运行时,它会返回 task handle,而不是最终结果。Client 会轮询直到完成。-
能力协商。 Client 在每个请求的 capabilities 中包含
io.modelcontextprotocol/tasks。Server 在自己的server/discovercapabilities 中声明同一扩展。 -
Task 创建。 响应受支持的请求时,server 返回
CreateTaskResult(通过resultType: "task"标识),其中包含taskId、初始状态、TTL 和建议轮询间隔。Task 会在响应发送前被持久创建。 -
轮询。 Client 使用
taskId调用tasks/get。响应携带当前状态;对于终止状态,还会携带最终结果或错误。 -
执行中输入。 如果 task 进入
input_required,tasks/get响应会包含一个inputRequestsmap,其中带有 elicitations 或其他 server requests。Client 通过tasks/update完成这些请求。 -
完成。 当状态到达
completed时,result字段包含原始请求同步返回时应有的内容。如果状态为failed,error字段包含 JSON-RPC 错误。 -
取消。 Client 可以随时发送
tasks/cancel。取消是协作式的,server 会确认取消意图,但没有义务停止工作。
何时使用 Tasks
当使用场景涉及以下情况时,Tasks 很适合: 长时间运行操作。 需要数分钟或数小时的 CI pipelines、批量数据处理或模型训练任务。 人在环工作流。 审批关卡、评审步骤,或任何会暂停等待用户确认的操作。Task 会进入input_required,client 展示该请求。
外部任务系统。 如果 server 包装的 API 已经使用 job IDs(云部署、异步 API、排队工作),可以在创建 job 时返回 task,并在 job 完成时解析它。
不可靠连接。 移动客户端、间歇性网络,或连接容易断开的环境。Task IDs 可以跨断线保留。
批处理。 处理大量项目(批量导入、大量更新)且部分进度有意义的操作。状态消息可以报告进度。
Task 生命周期
| 状态 | 含义 |
|---|---|
working | 操作正在进行。 |
input_required | Server 需要 client 输入后才能继续。见 inputRequests。 |
completed | 操作已完成。result 字段包含最终输出。 |
failed | 执行期间发生 JSON-RPC 错误。error 字段包含详情。 |
cancelled | 操作已取消(不一定总会被执行)。 |
completed、failed 和 cancelled 是终止状态,一旦达到,task 状态不会再变化。
通知
Servers 可以通过notifications/tasks 推送状态更新。Clients 通过 subscriptions/listen 机制 opt in。每条通知都携带完整 task 状态,因此不需要额外一次 tasks/get 往返。
轮询是默认方式。如果 server 支持通知,clients 可以依赖通知而不是轮询。
实现指南
面向 MCP clients
要消费带 task 的响应,你的 client 必须:轮询完成状态
使用返回的
taskId 调用 tasks/get,并遵守 pollIntervalMs 值。持续轮询,直到 task 达到终止状态(completed、failed 或 cancelled)。面向 MCP servers
要从 server 返回 tasks:检查 client capabilities
返回
CreateTaskResult 前,确认 client 已在每个请求的 capabilities 中包含该扩展。不要向未声明支持的 client 返回 task。返回 CreateTaskResult
当某个请求会长时间运行时,返回
resultType: "task",以及包含唯一 taskId、初始状态、ttlMs 和 pollIntervalMs 的 Task 对象。Task 必须在发送响应前被持久创建。客户端支持
MCP Tasks 是核心 MCP 规范的扩展。不同 client 的 host 支持情况不同。
规范
Tasks 扩展在 experimental-ext-tasks repository 中定义。它使用标准 MCP 扩展协商机制:clients 和 servers 会在初始化期间,在各自 capabilities 的extensions 字段中声明支持。