openhands/sdk/mcp/
Core Responsibilities
The MCP Integration system has four primary responsibilities:- MCP Client Management - Connect to and communicate with MCP servers
- Tool Discovery - Enumerate available tools from MCP servers
- Schema Adaptation - Convert MCP tool schemas to SDK tool definitions
- Execution Bridge - Execute MCP tool calls from agent actions
Architecture
Key Components
| Component | Purpose | Design |
|---|---|---|
MCPClient | Client wrapper | Extends FastMCP with sync/async bridge |
MCPToolDefinition | Tool metadata | Converts MCP schemas to SDK format |
MCPToolExecutor | Execution handler | Bridges agent actions to MCP calls |
MCPToolAction | Dynamic action model | Runtime-generated Pydantic model |
MCPToolObservation | Result wrapper | Wraps MCP tool results |
MCP Client
Sync/Async Bridge
The SDK’sMCPClient extends FastMCP’s async client with synchronous wrappers:
Bridge Pattern:
- Problem: MCP protocol is async, but agent tools run synchronously
- Solution: Background event loop that executes async code from sync contexts
- Benefit: Agents use MCP tools without async/await in tool definitions
- Lifecycle Management:
__enter__/__exit__for context manager - Timeout Support: Configurable timeouts for MCP operations
- Error Handling: Wraps MCP errors in observations
- Connection Pooling: Reuses connections across tool calls
MCP Server Configuration
MCP servers are configured using the FastMCP format:- command: Executable to spawn (e.g.,
uvx,npx,node) - args: Arguments to pass to command
- env: Environment variables (optional)
Tool Discovery and Conversion
Discovery Flow
Discovery Steps:- Spawn Server: Launch MCP server via stdio
- List Tools: Call
tools/listMCP endpoint - Parse Schemas: Extract tool names, descriptions, parameters
- Generate Models: Dynamically create Pydantic models for actions
- Create Definitions: Wrap in
ToolDefinitionobjects - Register: Add to agent’s tool registry
Schema Conversion
MCP tool schemas are converted to SDK tool definitions: Conversion Rules:| MCP Schema | SDK Action Model |
|---|---|
| name | Class name (camelCase) |
| description | Docstring |
| inputSchema | Pydantic fields |
| required | Field(required=True) |
| type | Python type hints |
Tool Execution
Execution Flow
Execution Steps:- Action Creation: LLM generates tool call, parsed into
MCPToolAction - Executor Lookup: Find
MCPToolExecutorfor tool name - Format Conversion: Convert action fields to MCP arguments
- MCP Call: Execute
call_toolvia MCP client - Result Parsing: Parse MCP result (text, images, resources)
- Observation Creation: Wrap in
MCPToolObservation - Error Handling: Catch exceptions, return error observations
MCPToolExecutor
Executors bridge SDK actions to MCP calls: Executor Responsibilities:- Client Management: Hold reference to MCP client
- Tool Identification: Know which MCP tool to call
- Argument Conversion: Transform action fields to MCP format
- Result Handling: Parse MCP responses
- Error Recovery: Handle connection errors, timeouts, server failures
MCP Tool Lifecycle
From Configuration to Execution
Lifecycle Phases:| Phase | Operations | Components |
|---|---|---|
| Initialization | Spawn servers, discover tools | MCPClient, ToolRegistry |
| Registration | Create definitions, executors | MCPToolDefinition, MCPToolExecutor |
| Execution | Handle tool calls | Agent, MCPToolAction |
| Cleanup | Close connections, shutdown servers | MCPClient.sync_close() |
MCP Annotations
MCP tools can include metadata hints for agents: Annotation Types:| Annotation | Meaning | Use Case |
|---|---|---|
| readOnlyHint | Tool doesn’t modify state | Lower security risk |
| destructiveHint | Tool modifies/deletes data | Require confirmation |
| progressEnabled | Tool reports progress | Show progress UI |
Component Relationships
How MCP Integrates
Relationship Characteristics:- Skills → MCP: Repository skills can embed MCP configurations
- MCP → Tools: MCP tools registered alongside native tools
- Agent → Tools: Agents use MCP tools like any other tool
- MCP → Security: Annotations inform security risk assessment
- Transparent Integration: Agent doesn’t distinguish MCP from native tools
Design Rationale
Async Bridge Pattern: MCP protocol requires async, but synchronous tool execution simplifies agent implementation. Background event loop bridges the gap without exposing async complexity to tool users. Dynamic Model Generation: Creating Pydantic models at runtime from MCP schemas enables type-safe tool calls without manual model definitions. This supports arbitrary MCP servers without SDK code changes. Unified Tool Interface: Wrapping MCP tools inToolDefinition makes them indistinguishable from native tools. Agents use the same interface regardless of tool source.
FastMCP Foundation: Building on FastMCP (MCP SDK for Python) provides battle-tested client implementation, protocol compliance, and ongoing updates as MCP evolves.
Annotation Support: Exposing MCP hints (readOnly, destructive) enables intelligent security analysis and user confirmation flows based on tool characteristics.
Lifecycle Management: Automatic spawn/cleanup of MCP servers in conversation lifecycle ensures resources are properly managed without manual bookkeeping.
See Also
- Tool System - How MCP tools integrate with tool framework
- Skill Architecture - Embedding MCP configs in repository skills
- Security - How MCP annotations inform risk assessment
- MCP Guide - Using MCP tools in applications
- FastMCP Documentation - Underlying MCP client library

