Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 0 additions & 5 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -107,8 +107,6 @@ let quit_reason = server.cancel().await?;
</details>




## Examples

See [examples](examples/README.md)
Expand All @@ -118,9 +116,6 @@ See [examples](examples/README.md)
See [oauth_support](docs/OAUTH_SUPPORT.md) for details.





## Related Resources

- [MCP Specification](https://spec.modelcontextprotocol.io/specification/2024-11-05/)
Expand Down
5 changes: 1 addition & 4 deletions examples/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -55,10 +55,7 @@

# Client Examples

- [Client SSE](clients/src/sse.rs), using reqwest and eventsource-client.
- [Client stdio](clients/src/std_io.rs), using tokio to spawn child process.
- [Everything](clients/src/everything_stdio.rs), test with `@modelcontextprotocol/server-everything`
- [Collection](clients/src/collection.rs), How to transpose service into dynamic object, so they will have a same type.
see [clients/README.md](clients/README.md)

# Server Examples

Expand Down
8 changes: 4 additions & 4 deletions examples/clients/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -34,8 +34,8 @@ name = "clients_sse"
path = "src/sse.rs"

[[example]]
name = "clients_std_io"
path = "src/std_io.rs"
name = "clients_git_stdio"
path = "src/git_stdio.rs"

[[example]]
name = "clients_streamable_http"
Expand All @@ -50,5 +50,5 @@ name = "clients_collection"
path = "src/collection.rs"

[[example]]
name = "oauth_client"
path = "src/oauth_client.rs"
name = "clients_oauth_client"
path = "src/auth/oauth_client.rs"
94 changes: 94 additions & 0 deletions examples/clients/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,94 @@
# MCP Client Examples

This directory contains Model Context Protocol (MCP) client examples implemented in Rust. These examples demonstrate how to communicate with MCP servers using different transport methods and how to use various client APIs.

## Example List

### SSE Client (`sse.rs`)

A client that communicates with an MCP server using Server-Sent Events (SSE) transport.

- Connects to an MCP server running at `http://localhost:8000/sse`
- Retrieves server information and list of available tools
- Calls a tool named "increment"

### Git Standard I/O Client (`git_stdio.rs`)

A client that communicates with a Git-related MCP server using standard input/output.

- Launches the `uvx mcp-server-git` command as a child process
- Retrieves server information and list of available tools
- Calls the `git_status` tool to check the Git status of the current directory

### Streamable HTTP Client (`streamable_http.rs`)

A client that communicates with an MCP server using HTTP streaming transport.
- Connects to an MCP server running at `http://localhost:8000`
- Retrieves server information and list of available tools
- Calls a tool named "increment"

### Full-Featured Standard I/O Client (`everything_stdio.rs`)

An example demonstrating all MCP client capabilities.

- Launches `npx -y @modelcontextprotocol/server-everything` as a child process
- Retrieves server information and list of available tools
- Calls various tools, including "echo" and "longRunningOperation"
- Lists and reads available resources
- Lists and retrieves simple and complex prompts
- Lists available resource templates

### Client Collection (`collection.rs`)

An example showing how to manage multiple MCP clients.

- Creates 10 clients connected to Git servers
- Stores these clients in a HashMap
- Performs the same sequence of operations on each client
- Uses `into_dyn()` to convert services to dynamic services

### OAuth Client (`auth/oauth_client.rs`)

A client demonstrating how to authenticate with an MCP server using OAuth.

- Starts a local HTTP server to handle OAuth callbacks
- Initializes the OAuth state machine and begins the authorization flow
- Displays the authorization URL and waits for user authorization
- Establishes an authorized connection to the MCP server using the acquired access token
- Demonstrates how to use the authorized connection to retrieve available tools and prompts

## How to Run

Each example can be run using Cargo:

```bash
# Run the SSE client example
cargo run --example clients_sse

# Run the Git standard I/O client example
cargo run --example clients_git_stdio

# Run the streamable HTTP client example
cargo run --example clients_streamable_http

# Run the full-featured standard I/O client example
cargo run --example clients_everything_stdio

# Run the client collection example
cargo run --example clients_collection

# Run the OAuth client example
cargo run --example clients_oauth_client
```

## Dependencies

These examples use the following main dependencies:

- `rmcp`: Rust implementation of the MCP client library
- `tokio`: Asynchronous runtime
- `serde` and `serde_json`: For JSON serialization and deserialization
- `tracing` and `tracing-subscriber`: For logging, not must, only for logging
- `anyhow`: Error handling, not must, only for error handling
- `axum`: For the OAuth callback HTTP server (used only in the OAuth example)
- `reqwest`: HTTP client library (used for OAuth and streamable HTTP transport)
12 changes: 8 additions & 4 deletions examples/clients/src/collection.rs
Original file line number Diff line number Diff line change
@@ -1,3 +1,7 @@
/// This example show how to store multiple clients in a map and call tools on them.
/// into_dyn() is used to convert the service to a dynamic service.
/// For example, you can use this to call tools on a service that is running in a different process.
/// or a service that is running in a different machine.
use std::collections::HashMap;

use anyhow::Result;
Expand All @@ -20,7 +24,7 @@ async fn main() -> Result<()> {
.with(tracing_subscriber::fmt::layer())
.init();

let mut client_list = HashMap::new();
let mut clients_map = HashMap::new();
for idx in 0..10 {
let service = ()
.into_dyn()
Expand All @@ -30,10 +34,10 @@ async fn main() -> Result<()> {
},
))?)
.await?;
client_list.insert(idx, service);
clients_map.insert(idx, service);
}

for (_, service) in client_list.iter() {
for (_, service) in clients_map.iter() {
// Initialize
let _server_info = service.peer_info();

Expand All @@ -48,7 +52,7 @@ async fn main() -> Result<()> {
})
.await?;
}
for (_, service) in client_list {
for (_, service) in clients_map {
service.cancel().await?;
}
Ok(())
Expand Down
File renamed without changes.
2 changes: 2 additions & 0 deletions examples/servers/src/common/generic_service.rs
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@ pub struct GenericService<DS: DataService> {

#[tool(tool_box)]
impl<DS: DataService> GenericService<DS> {
#[allow(dead_code)]
pub fn new(data_service: DS) -> Self {
Self {
data_service: Arc::new(data_service),
Expand All @@ -62,6 +63,7 @@ impl<DS: DataService> GenericService<DS> {
}
}

#[tool(tool_box)]
impl<DS: DataService> ServerHandler for GenericService<DS> {
fn get_info(&self) -> ServerInfo {
ServerInfo {
Expand Down