Expand description
§RMCP
A better and clean rust Model Context Protocol SDK implementation with tokio async runtime.
§Comparing to official SDK
The Official SDK has too much limit and it was originally built for goose rather than general using purpose.
All the features listed on specification would be implemented in this crate. And the first and most important thing is, this crate has the correct and intact data types. See it yourself.
§Usage
§Import
rmcp = { version = "0.1", features = ["server"] }
§Quick start
Start a client in one line:
let client = ().serve(
TokioChildProcess::new(Command::new("npx").arg("-y").arg("@modelcontextprotocol/server-everything"))?
).await?;
Start a client in one line:
let client = ().serve(
TokioChildProcess::new(Command::new("npx").arg("-y").arg("@modelcontextprotocol/server-everything"))?
).await?;
§1. Build a transport
The transport type must implemented IntoTransport
trait, which allow split into a sink and a stream.
For client, the sink item is ClientJsonRpcMessage
and stream item is ServerJsonRpcMessage
For server, the sink item is ServerJsonRpcMessage
and stream item is ClientJsonRpcMessage
§These types is automatically implemented IntoTransport
trait
- For type that already implement both
Sink
andStream
trait, they are automatically implementedIntoTransport
trait - For tuple of sink
Tx
and streamRx
, type(Tx, Rx)
are automatically implementedIntoTransport
trait - For type that implement both
tokio::io::AsyncRead
andtokio::io::AsyncWrite
trait, they are automatically implementedIntoTransport
trait - For tuple of
tokio::io::AsyncRead
R
andtokio::io::AsyncWrite
W
, type(R, W)
are automatically implementedIntoTransport
trait
use tokio::io::{stdin, stdout};
let transport = (stdin(), stdout());
§2. Build a service
You can easily build a service by using ServerHandler
or ClientHandler
.
let service = common::counter::Counter::new();
Or if you want to use tower
, you can [TowerHandler
] as a adapter.
You can reference the server examples.
§3. Serve them together
// this call will finish the initialization process
let server = service.serve(transport).await?;
§4. Interact with the server
Once the server is initialized, you can send requests or notifications:
// request
let roots = server.list_roots().await?;
// or send notification
server.notify_cancelled(...).await?;
§5. Waiting for service shutdown
let quit_reason = server.waiting().await?;
// or cancel it
let quit_reason = server.cancel().await?;
§Use marcos to declaring tool
Use toolbox
and tool
macros to create tool quickly.
Check this file.
use rmcp::{ServerHandler, model::ServerInfo, schemars, tool};
use super::counter::Counter;
#[derive(Debug, serde::Deserialize, schemars::JsonSchema)]
pub struct SumRequest {
#[schemars(description = "the left hand side number")]
pub a: i32,
#[schemars(description = "the right hand side number")]
pub b: i32,
}
#[derive(Debug, Clone)]
pub struct Calculator;
// create a static toolbox to store the tool attributes
#[tool(tool_box)]
impl Calculator {
// async function
#[tool(description = "Calculate the sum of two numbers")]
async fn sum(&self, #[tool(aggr)] SumRequest { a, b }: SumRequest) -> String {
(a + b).to_string()
}
// sync function
#[tool(description = "Calculate the sum of two numbers")]
fn sub(
&self,
#[tool(param)]
// this macro will transfer the schemars and serde's attributes
#[schemars(description = "the left hand side number")]
a: i32,
#[tool(param)]
#[schemars(description = "the right hand side number")]
b: i32,
) -> String {
(a - b).to_string()
}
}
// impl call_tool and list_tool by querying static toolbox
#[tool(tool_box)]
impl ServerHandler for Calculator {
fn get_info(&self) -> ServerInfo {
ServerInfo {
instructions: Some("A simple calculator".into()),
..Default::default()
}
}
}
The only thing you should do is to make the function’s return type implement IntoCallToolResult
.
And you can just implement IntoContents
, and the return value will be marked as success automatically.
If you return a type of Result<T, E>
where T
and E
both implemented IntoContents
, it’s also OK.
§Manage Multi Services
For many cases you need to manage several service in a collection, you can call into_dyn
to convert services into the same type.
let service = service.into_dyn();
§Examples
See examples
§Features
client
: use client side sdkserver
: use server side sdkmacros
: macros default
§Transports
transport-io
: Server stdio transporttransport-sse-server
: Server SSE transporttransport-child-process
: Client stdio transporttransport-sse
: Client sse transport
§Related Resources
Re-exports§
pub use service::Peer;
pub use service::Service;
pub use service::ServiceError;
pub use service::ServiceExt;
pub use service::RoleClient;
pub use service::serve_client;
pub use service::RoleServer;
pub use service::serve_server;
pub use handler::client::ClientHandler;
pub use handler::server::ServerHandler;
pub use schemars;
pub use serde;
pub use serde_json;