A lightweight, pure Python implementation of the Model Context Protocol client
Project description
Pure MCP Client
A lightweight, pure Python implementation of the Model Context Protocol (MCP) client that enables communication with MCP servers without external SDK dependencies.
Why Pure MCP?
While the official MCP SDK is comprehensive, it comes with heavy dependencies and complex abstractions that can be overkill for many use cases. Pure MCP addresses common pain points:
- 🪶 Lightweight Deployments: Minimal dependencies mean faster Docker builds, smaller Lambda packages, and easier integration into existing projects
- 🔒 Dependency Control: No surprise transitive dependencies or version conflicts with your existing stack
- 🚀 Cloud-Friendly: Perfect for serverless, containers, or any environment where package size matters
- ⚡ Simple Integration: Drop into any Python project without restructuring your codebase
- 🎯 Just What You Need: Full MCP protocol support without the bloat
Choose Pure MCP when you want MCP functionality without the overhead.
Features
- Minimal Dependencies: Core functionality using only Python standard library
- Full Protocol Support: Complete implementation of MCP including tools, resources, prompts, and completions
- Type-Safe: Comprehensive Pydantic models for all protocol types
- Async-First: Built on Python's asyncio for efficient concurrent operations
- Multiple Transports: Support for Server-Sent Events (SSE) and streamable HTTP
- Progress Tracking: Monitor long-running operations with callbacks
- Error Handling: Robust error handling with typed exceptions
Installation
# Install
pip install pure-mcp
# Or install from source
git clone https://github.com/John-Rood/pure_mcp.git
cd pure-mcp
pip install -e .
# Or install with development dependencies
pip install -e ".[dev]"
Quick Start
Basic Usage
import asyncio
from pure_mcp import ClientSession, sse_client
async def main():
# Connect to MCP server
async with sse_client("http://localhost:8080/sse") as (read_stream, write_stream):
async with ClientSession(read_stream, write_stream) as session:
# Initialize connection
result = await session.initialize()
print(f"Connected to: {result.serverInfo.name}")
# List and call tools
tools = await session.list_tools()
for tool in tools.tools:
print(f"Tool: {tool.name} - {tool.description}")
asyncio.run(main())
Working with Tools
async def use_tools():
async with sse_client("http://localhost:8080/sse") as (read_stream, write_stream):
async with ClientSession(read_stream, write_stream) as session:
await session.initialize()
# Call a tool
result = await session.call_tool(
name="search",
arguments={"query": "Python MCP"}
)
# Process results
for content in result.content:
if hasattr(content, 'text'):
print(content.text)
Resource Management
async def manage_resources():
async with sse_client("http://localhost:8080/sse") as (read_stream, write_stream):
async with ClientSession(read_stream, write_stream) as session:
await session.initialize()
# List resources
resources = await session.list_resources()
# Read a resource
if resources.resources:
content = await session.read_resource(resources.resources[0].uri)
for item in content.contents:
if hasattr(item, 'text'):
print(item.text)
Using Prompts
async def use_prompts():
async with sse_client("http://localhost:8080/sse") as (read_stream, write_stream):
async with ClientSession(read_stream, write_stream) as session:
await session.initialize()
# Get a prompt
prompt_result = await session.get_prompt(
name="code_review",
arguments={"language": "python"}
)
for message in prompt_result.messages:
print(f"{message.role}: {message.content.text}")
Advanced Features
Progress Tracking
Monitor progress for long-running operations:
async def track_progress():
async def progress_callback(progress: float, total: float | None, message: str | None):
percent = (progress / total * 100) if total else 0
print(f"Progress: {percent:.1f}% - {message or 'Processing...'}")
async with sse_client("http://localhost:8080/sse") as (read_stream, write_stream):
async with ClientSession(read_stream, write_stream) as session:
await session.initialize()
result = await session.call_tool(
name="long_task",
arguments={"data": "large_input"},
progress_callback=progress_callback
)
Custom Timeouts
Configure timeouts for different scenarios:
from datetime import timedelta
# Session-level timeout
session = ClientSession(
read_stream,
write_stream,
read_timeout_seconds=timedelta(seconds=60)
)
# Per-request timeout
result = await session.call_tool(
name="slow_tool",
arguments={},
read_timeout_seconds=timedelta(seconds=120)
)
Error Handling
Handle MCP-specific errors:
from pure_mcp import McpError
try:
result = await session.call_tool("unknown_tool")
except McpError as e:
print(f"Error {e.error.code}: {e.error.message}")
if e.error.data:
print(f"Details: {e.error.data}")
Project Structure
pure_mcp/
├── __init__.py # Package exports
├── core/ # Core functionality
│ ├── client.py # ClientSession implementation
│ ├── session.py # Base session protocol
│ ├── context.py # Request context management
│ └── message.py # Message types and metadata
├── transport/ # Transport implementations
│ ├── sse.py # Server-Sent Events transport
│ ├── streams.py # Memory stream implementation
│ └── http_utils.py # HTTP client utilities
└── types/ # Type definitions
├── protocol.py # MCP protocol types
├── exceptions.py # Error types
└── version.py # Protocol version support
API Reference
ClientSession
The main interface for MCP communication.
Methods
initialize()- Initialize connection and exchange capabilitieslist_tools()- Get available tools from the servercall_tool(name, arguments)- Execute a tool with argumentslist_resources()- Get available resourcesread_resource(uri)- Read resource contentlist_prompts()- Get available prompt templatesget_prompt(name, arguments)- Get prompt with argumentsset_logging_level(level)- Configure server loggingsend_ping()- Send keepalive ping
Transport Configuration
sse_client
Connect using Server-Sent Events:
async with sse_client(
url="http://localhost:8080/sse",
headers={"Authorization": "Bearer token"},
timeout=30,
sse_read_timeout=300
) as (read_stream, write_stream):
# Use streams with ClientSession
Supported Types
The package exports commonly used types:
- Core Types:
Tool,Resource,Prompt - Result Types:
InitializeResult,ListToolsResult,CallToolResult, etc. - Content Types:
TextContent,ImageContent - Other Types:
LoggingLevel,McpError
Protocol Versions
Supports MCP protocol versions:
- 2024-11-05
- 2025-03-26
- 2025-06-18 (latest)
Requirements
- Python 3.8+
- anyio >= 3.0.0
- httpx >= 0.25.0
- httpx-sse >= 0.3.0
- pydantic >= 2.0.0
- jsonschema >= 4.0.0
- typing-extensions >= 4.0.0
Examples
See example.py for complete working examples demonstrating:
- Basic connection and initialization
- Tool discovery and execution
- Resource listing and reading
- Prompt management
- Progress tracking
- Error handling
Contributing
Contributions are welcome! Please ensure:
- Type hints for all functions
- Docstrings for public APIs
- Async-first implementations
- Tests for new features
License
MIT License - see LICENSE file for details.
Acknowledgments
This implementation provides a lightweight alternative to the official MCP SDK, focusing on minimal dependencies and cloud-friendly deployment.
Project details
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file pure_mcp-0.1.4.tar.gz.
File metadata
- Download URL: pure_mcp-0.1.4.tar.gz
- Upload date:
- Size: 31.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.10.16
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e807490eb2c8ea031960f90847ff454c24e14d619e91b8775e4ba1bb6e32dc08
|
|
| MD5 |
4b2983b071823f347b1f279bc878a0fd
|
|
| BLAKE2b-256 |
9aba62e949d08050a7880d929cdcf630fab7c80221321d8ead45f524ef001ccf
|
File details
Details for the file pure_mcp-0.1.4-py3-none-any.whl.
File metadata
- Download URL: pure_mcp-0.1.4-py3-none-any.whl
- Upload date:
- Size: 31.9 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.10.16
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
65052f09f71bc30d53a5a7fd1cfc30d961d6e3f57fa45731f3f6a72d6b40178d
|
|
| MD5 |
82a5b234ff4bded6a877a3ee82053c43
|
|
| BLAKE2b-256 |
662d87b320e51ae728d33161c751ef9c904e4f2b6c1b66737ef8cb253c0bd892
|