| Crates.io | turul-mcp-client |
| lib.rs | turul-mcp-client |
| version | 0.2.1 |
| created_at | 2025-09-02 23:37:20.144653+00 |
| updated_at | 2025-10-20 20:57:16.331646+00 |
| description | Comprehensive MCP client library with multi-transport support |
| homepage | https://github.com/aussierobots/turul-mcp-framework |
| repository | https://github.com/aussierobots/turul-mcp-framework |
| max_upload_size | |
| id | 1821802 |
| size | 220,607 |
MCP client library with multi-transport support and full MCP 2025-06-18 protocol compliance.
turul-mcp-client provides a complete client implementation for the Model Context Protocol (MCP), supporting multiple transport layers and offering both high-level and low-level APIs for interacting with MCP servers.
Add this to your Cargo.toml:
[dependencies]
turul-mcp-client = "0.2.0"
tokio = { version = "1.0", features = ["full"] }
use turul_mcp_client::{McpClient, McpClientBuilder};
use turul_mcp_client::transport::HttpTransport;
#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
// Create HTTP transport
let transport = HttpTransport::new("http://localhost:8080/mcp")?;
// Build client
let client = McpClientBuilder::new()
.with_transport(Box::new(transport))
.build();
// Connect and initialize
client.connect().await?;
// List available tools
let tools = client.list_tools().await?;
println!("Available tools: {}", tools.len());
// Call a tool
let result = client.call_tool("calculator", serde_json::json!({
"operation": "add",
"a": 5,
"b": 3
})).await?;
println!("Tool result: {:?}", result);
Ok(())
}
For modern MCP servers supporting MCP 2025-06-18:
use turul_mcp_client::transport::HttpTransport;
let transport = HttpTransport::new("http://localhost:8080/mcp")?;
let client = McpClientBuilder::new()
.with_transport(Box::new(transport))
.build();
For servers supporting server-sent events:
use turul_mcp_client::transport::SseTransport;
let transport = SseTransport::new("http://localhost:8080/mcp")?;
let client = McpClientBuilder::new()
.with_transport(Box::new(transport))
.build();
Additional transport implementations (WebSocket, stdio) are planned for future releases.
use turul_mcp_client::{McpClientBuilder, ClientConfig, RetryConfig, TimeoutConfig};
use std::time::Duration;
let config = ClientConfig {
client_info: ClientInfo {
name: "My MCP Client".to_string(),
version: "1.0.0".to_string(),
description: Some("Custom MCP client".to_string()),
vendor: None,
metadata: None,
},
timeouts: TimeoutConfig {
connect: Duration::from_secs(10),
request: Duration::from_secs(30),
long_operation: Duration::from_secs(120),
initialization: Duration::from_secs(15),
heartbeat: Duration::from_secs(30),
},
retry: RetryConfig {
max_attempts: 3,
initial_delay: Duration::from_millis(100),
max_delay: Duration::from_secs(5),
backoff_multiplier: 2.0,
jitter: 0.1,
exponential_backoff: true,
},
connection: ConnectionConfig::default(),
logging: LoggingConfig::default(),
request_timeout: Duration::from_secs(30),
},
};
let client = McpClientBuilder::new()
.with_config(config)
.with_url("http://localhost:8080/mcp")?
.build();
let client = McpClientBuilder::new()
.with_url("http://localhost:8080/mcp")? // Automatically detects transport type
.build();
use turul_mcp_client::session::SessionState;
// Check connection and session status
let status = client.connection_status().await;
println!("Transport connected: {}", status.transport_connected);
println!("Session state: {:?}", status.session_state);
println!("Transport type: {}", status.transport_type);
if let Some(session_id) = status.session_id {
println!("Session ID: {}", session_id);
}
// Get detailed session information
let session_info = client.session_info().await;
println!("Session ID: {:?}", session_info.session_id);
println!("Created: {:?}", session_info.created_at);
println!("State: {:?}", session_info.state);
// Check if client is ready
if !client.is_ready().await {
client.connect().await?;
}
// Disconnect and cleanup
client.disconnect().await?;
use turul_mcp_client::{McpClientError, McpClientResult};
async fn robust_operation(client: &McpClient) -> McpClientResult<()> {
match client.call_tool("my_tool", serde_json::json!({"param": "value"})).await {
Ok(result) => {
println!("Success: {:?}", result);
Ok(())
}
Err(McpClientError::Transport(e)) => {
tracing::warn!("Transport error, attempting reconnect: {}", e);
client.disconnect().await?;
client.connect().await?;
Err(e.into())
}
Err(McpClientError::Session(e)) => {
tracing::error!("Session error: {}", e);
Err(e.into())
}
Err(McpClientError::Protocol(e)) => {
tracing::error!("Protocol error: {}", e);
Err(e.into())
}
Err(e) => Err(e),
}
}
// List available tools
let tools = client.list_tools().await?;
for tool in &tools {
println!("Tool: {}", tool.name);
if let Some(description) = &tool.description {
println!(" Description: {}", description);
}
}
// Call a tool
let result = client.call_tool("calculator", serde_json::json!({
"operation": "multiply",
"a": 7,
"b": 6
})).await?;
println!("Tool result: {:?}", result.content);
// List available resources
let resources = client.list_resources().await?;
for resource in &resources {
println!("Resource: {}", resource.uri);
if let Some(description) = &resource.description {
println!(" Description: {}", description);
}
}
// Read a resource
let content = client.read_resource("file:///path/to/file.txt").await?;
println!("Resource content: {:?}", content);
// List available prompts
let prompts = client.list_prompts().await?;
for prompt in &prompts {
println!("Prompt: {}", prompt.name);
if let Some(description) = &prompt.description {
println!(" Description: {}", description);
}
}
// Get a prompt with arguments
let prompt_result = client.get_prompt("greeting", Some(serde_json::json!({
"name": "Alice"
}))).await?;
println!("Prompt messages: {:?}", prompt_result.messages);
// Access the stream handler for server events
let stream_handler = client.stream_handler().await;
// The stream handler processes server-sent events automatically
// This is primarily for internal use and advanced scenarios
The client automatically sends the appropriate protocol version header:
// Client automatically sends: MCP-Protocol-Version: 2025-06-18
// Server responds with: mcp-session-id: <session-uuid>
// Access session ID from connection status
let status = client.connection_status().await;
if let Some(session_id) = status.session_id {
println!("Session ID from server: {}", session_id);
}
// Ping the server to check connectivity
match client.ping().await {
Ok(_) => println!("Server is responsive"),
Err(e) => println!("Server ping failed: {}", e),
}
// Get transport layer statistics
let stats = client.transport_stats().await;
println!("Requests sent: {}", stats.requests_sent);
println!("Responses received: {}", stats.responses_received);
println!("Average response time: {:.2}ms", stats.avg_response_time_ms);
use turul_mcp_client::transport::{TransportFactory, detect_transport_type};
// Detect transport type from URL
let transport_type = detect_transport_type("http://localhost:8080/mcp")?;
println!("Detected transport: {}", transport_type);
// Create transport automatically
let transport = TransportFactory::from_url("http://localhost:8080/mcp")?;
// List available transports
let available = TransportFactory::available_transports();
println!("Available transports: {:?}", available);
use turul_mcp_client::{McpClient, McpClientBuilder, ClientConfig};
use turul_mcp_client::transport::HttpTransport;
use tracing::{info, error};
use std::time::Duration;
#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
// Initialize logging
tracing_subscriber::fmt::init();
// Create client with configuration
let mut config = ClientConfig::default();
config.timeouts.connect = Duration::from_secs(10);
config.timeouts.request = Duration::from_secs(30);
let transport = HttpTransport::new("http://localhost:8080/mcp")?;
let client = McpClientBuilder::new()
.with_transport(Box::new(transport))
.with_config(config)
.build();
// Connect
info!("Connecting to MCP server...");
client.connect().await?;
info!("Connected successfully!");
// Check if ready
if !client.is_ready().await {
error!("Client not ready after connect");
return Ok(());
}
// Discover capabilities
let tools = client.list_tools().await?;
info!("Server provides {} tools", tools.len());
for tool in &tools {
info!(" - {}: {}", tool.name,
tool.description.as_deref().unwrap_or("No description"));
}
// Use first available tool
if !tools.is_empty() {
let tool_name = &tools[0].name;
info!("Calling tool: {}", tool_name);
match client.call_tool(tool_name, serde_json::json!({})).await {
Ok(result) => {
info!("Tool result: {:?}", result.content);
}
Err(e) => {
error!("Tool call failed: {}", e);
}
}
}
// Get connection status
let status = client.connection_status().await;
info!("Connection status: transport_connected={}, session_state={:?}",
status.transport_connected, status.session_state);
// Graceful cleanup
info!("Disconnecting...");
client.disconnect().await?;
Ok(())
}
use turul_mcp_client::transport::{HttpTransport, SseTransport, TransportCapabilities};
// Compare transport capabilities
fn compare_transports() -> Result<(), Box<dyn std::error::Error>> {
let http_transport = HttpTransport::new("http://localhost:8080/mcp")?;
let sse_transport = SseTransport::new("http://localhost:8080/mcp")?;
let http_caps = http_transport.capabilities();
let sse_caps = sse_transport.capabilities();
println!("HTTP - Streaming: {}, Server Events: {}",
http_caps.streaming, http_caps.server_events);
println!("SSE - Streaming: {}, Server Events: {}",
sse_caps.streaming, sse_caps.server_events);
Ok(())
}
[dependencies]
turul-mcp-client = { version = "0.2.0", features = ["sse"] }
Available features:
default = ["http", "sse"] - HTTP and SSE transporthttp - HTTP transport support (included by default)sse - Server-Sent Events transport (included by default)websocket - (Planned) WebSocket transport supportstdio - (Planned) Standard I/O transport for executable serversuse turul_mcp_client::McpClientError;
match error {
McpClientError::Transport(e) => {
// Network/transport related errors
eprintln!("Transport error: {}", e);
}
McpClientError::Protocol(e) => {
// MCP protocol violations or incompatibilities
eprintln!("Protocol error: {}", e);
}
McpClientError::Session(e) => {
// Session management errors
eprintln!("Session error: {}", e);
}
McpClientError::Timeout => {
// Request timeout
eprintln!("Request timed out");
}
McpClientError::NotConnected => {
// Client not connected
eprintln!("Client not connected");
}
McpClientError::InvalidResponse(msg) => {
// Invalid response from server
eprintln!("Invalid response: {}", msg);
}
}
The client automatically adapts to server capabilities:
Licensed under the MIT License. See LICENSE for details.