MCP Callback Protocol extension for enabling asynchronous tool callbacks and streaming updates from MCP clients
Project description
MCPC - Model Context Protocol Callback
MCPC is an extension to MCP (Model-Context-Protocol) that solves a critical limitation in LLM tool interactions: enabling continued conversations while running tools background tasks. It facilitates asynchronous two-way communication between LLMs and tools through the already existing MCP transport - no additional transport layer needed, while maintaining full backward compatibility.
What is MCPC?
MCPC is an extension to the MCP protocol, not a replacement. It builds upon the existing MCP infrastructure to add real-time two-way communication capabilities while maintaining full compatibility with standard MCP implementations.
MCPC solves a critical limitation in LLM tool interactions: enabling continuous two-way communication while running background tasks:
- Bidirectional communication between LLMs and tools using the same MCP transport
- Continuous conversation with LLMs during tool execution
- Real-time updates from background processes
- Asynchronous notifications when operations complete
- Support for indefinitely running tasks with streaming updates
Compatibility Matrix
Features
| Feature | Status | Notes |
|---|---|---|
| STDIO Transport | ✅ Implemented | Full support for standard input/output transport |
| SSE Transport | ⚠️ Limited Support | Standard MCP operations only, MCPC features pending |
| MCPC Client → Standard MCP Server | ✅ Implemented | Full backward compatibility |
| Standard MCP Client → MCPC Server | ✅ Implemented | Automatic fallback to synchronous results |
Frameworks
| Framework | Status | Notes |
|---|---|---|
| FastMCP 😎 | ✅ Implemented | Recommended |
| Standard MCP SDK Server | ✅ Implemented | Works (Use FastMCP if you can) |
Quick Installation
# With UV (recommended)
uv add mcpc
# With pip
pip install mcpc
Basic Client Usage
# Initialize the MCPC handler
mcpc_handler = MCPCHandler("my-provider")
# Define your event listener function
async def my_mcpc_listener(mcpc_message: MCPCMessage) -> None:
print(f"Received MCPC message: {mcpc_message}")
# Handle the message based on status
if mcpc_message.type == "task" and mcpc_message.event == "complete":
print(f"Task {mcpc_message.task_id} completed with result: {mcpc_message.result}")
# Add your event listener for MCPC Message
mcpc_handler.add_event_listener(my_mcpc_listener)
# Wrap the transport with MCPC event listeners
wrapped_transport = await mcpc_handler.wrap_streams(*transport)
# Create a ClientSession with the wrapped transport
session = await ClientSession(*wrapped_transport)
# Initialize MCPC features by checking for MCPC support
mcpc_supported = await mcpc_handler.init_mcpc(session)
if mcpc_supported:
print(f"MCPC protocol v{mcpc_handler.protocol_version} supported")
Basic Server Usage - FastMCP
# Initialize MCPC helper
server = FastMCP("my-provider")
mcpc = MCPCHelper(server)
@server.tool()
async def process_data(url: str) -> dict:
async def process_data_task():
yield mcpc.create_task_event(
event="update",
tool_name="process_data",
session_id=session_id,
task_id=task_id,
result=f"Processing {url}..."
))
await asyncio.sleep(3) # Simulate processing time
yield mcpc.create_task_event(
event="complete",
tool_name="process_data",
session_id=session_id,
task_id=task_id,
result={
YOUR_DATA_OBJECT
}
)
# Start a background task - or run synchronous if no MCPC support
collected_messages = await mcpc.start_task(task_id, process_data_task)
# For standard MCP clients, return collected complete/failed messages
if collected_messages:
# Keep in mind that FastMCP differs from Standard MCP
# by expecting direct result rather than TextContent
# It has implications in that it returns
# all messages as "one" item, rather than multiple
return collected_messages
# For MCPC clients, return immediate acknowledgment
# See comment above on why this is not [TextContent]
return mcpc.create_task_event(
event="created",
tool_name="process_data",
session_id=session_id,
task_id=task_id,
result=f"Started processing data_id={data_id}. Updates will stream in real-time."
)
if __name__ == "__main__":
asyncio.run(server.run())
Documentation
For detailed documentation, please see:
- Non-FastMCP Server Guide - basic usage of standard MCP SDK Server
- API Reference - Detailed API documentation
- Protocol Details - Message structure and protocol information
- Use Cases - Example scenarios and use cases
License
MIT
Project details
Release history Release notifications | RSS feed
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 mcpc-0.2.6.tar.gz.
File metadata
- Download URL: mcpc-0.2.6.tar.gz
- Upload date:
- Size: 115.3 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.12.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a970c1d23f60c52f13d480f8a1a641ea2893d48905c0dad49536003db425a18e
|
|
| MD5 |
0e369f0b6128dcc864d5a3275db08260
|
|
| BLAKE2b-256 |
8e4ede9c3ca175b11b5d7e971e144573dcdff40001b4f456251f760289d9a3fd
|
Provenance
The following attestation bundles were made for mcpc-0.2.6.tar.gz:
Publisher:
python-publish.yml on OlaHulleberg/mcpc
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
mcpc-0.2.6.tar.gz -
Subject digest:
a970c1d23f60c52f13d480f8a1a641ea2893d48905c0dad49536003db425a18e - Sigstore transparency entry: 192151658
- Sigstore integration time:
-
Permalink:
OlaHulleberg/mcpc@84a83498d5361b4ead0ec58346e0626ea12633af -
Branch / Tag:
refs/tags/0.2.6 - Owner: https://github.com/OlaHulleberg
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@84a83498d5361b4ead0ec58346e0626ea12633af -
Trigger Event:
release
-
Statement type:
File details
Details for the file mcpc-0.2.6-py3-none-any.whl.
File metadata
- Download URL: mcpc-0.2.6-py3-none-any.whl
- Upload date:
- Size: 11.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.12.9
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
8b150e4baee08c5f6f28b192e613e62b221f1e4c2416a5e309622f8ead1a134b
|
|
| MD5 |
b138fb27471be0dde5c982442e55dfe4
|
|
| BLAKE2b-256 |
93f7063a65f3d8608e29e23beb71d0cc35ea3462962a20e71ed4b0519f51ab91
|
Provenance
The following attestation bundles were made for mcpc-0.2.6-py3-none-any.whl:
Publisher:
python-publish.yml on OlaHulleberg/mcpc
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
mcpc-0.2.6-py3-none-any.whl -
Subject digest:
8b150e4baee08c5f6f28b192e613e62b221f1e4c2416a5e309622f8ead1a134b - Sigstore transparency entry: 192151659
- Sigstore integration time:
-
Permalink:
OlaHulleberg/mcpc@84a83498d5361b4ead0ec58346e0626ea12633af -
Branch / Tag:
refs/tags/0.2.6 - Owner: https://github.com/OlaHulleberg
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
python-publish.yml@84a83498d5361b4ead0ec58346e0626ea12633af -
Trigger Event:
release
-
Statement type: