Use MCP tools¶
For tools that live in a separate process (or to share a tool catalogue across many agents), wrap a FastMCP 2.0 server with MCPClient, then call .as_tools() to add its tools to your model client's tools list.
Connect to a tool server¶
from aimu.tools import MCPClient
import aimu
mcp_client = MCPClient({
"mcpServers": {
"mytools": {"command": "python", "args": ["tools.py"]},
},
})
mcp_client.ping() # raises MCPConnectionError if dead
client = aimu.client("ollama:qwen3.5:9b")
client.tools = mcp_client.as_tools() # MCP tools become @tool-style callables
client.chat("Use the mytools to do something.")
as_tools() does one list_tools() round-trip and returns a callable per server tool, each carrying its __tool_spec__. They dispatch through the same path as in-process @tool functions, so MCP and Python tools are interchangeable from the client's point of view. Keep the MCPClient reference (or the callables, which hold one) alive for the connection's lifetime; call as_tools() again to refresh after the server's tool set changes.
MCPClient requires exactly one of:
config={...}: FastMCP server config dict (the form above)server=...: an in-processFastMCPinstancefile="path/to/server.py": a local server script
Call a tool directly¶
You can use MCPClient standalone, without an agent:
result = mcp_client.call_tool("mytool", {"input": "hello world!"})
# result.content[0].text → tool's text response
Use the built-in MCP server¶
AIMU ships a FastMCP server that exposes the entire builtin toolkit:
Connect to it from another process the same way:
mcp_client = MCPClient({
"mcpServers": {
"aimu-builtin": {"command": "python", "args": ["-m", "aimu.tools.mcp"]},
},
})
Combine with in-process tools¶
Both routes produce callables for the one tools list; concatenate them. On a name collision the last entry wins, so append a local override after the MCP tools to shadow one:
client.tools = mcp_client.as_tools() + [my_local_tool] # my_local_tool shadows a same-named MCP tool
Loud failures¶
Connection problems raise MCPConnectionError with the original cause chained:
try:
mcp_client = MCPClient({"mcpServers": {"bad": {"command": "nonexistent"}}})
except MCPConnectionError as exc:
print(exc)
# MCPConnectionError: failed to connect MCP transport ...: ...
A subsequent .ping() or .call_tool() failure also raises MCPConnectionError, so you can re-establish the connection without silently broken state.
See also¶
- Add a custom tool: the in-process
@toolroute - Explanation: tool integration: when to pick which route
aimu.tools.MCPClient: API reference