#![cfg(not(feature = "local"))]
mod common;
use common::handlers::{TestClientHandler, TestServer};
use rmcp::{
ServiceExt,
model::*,
service::{RequestContext, Service},
};
#[tokio::test]
async fn test_message_roles() {
let messages = vec![
SamplingMessage::user_text("user message"),
SamplingMessage::assistant_text("assistant message"),
];
let json = serde_json::to_string(&messages).unwrap();
let deserialized: Vec<SamplingMessage> = serde_json::from_str(&json).unwrap();
assert_eq!(messages, deserialized);
}
#[tokio::test]
async fn test_context_inclusion_integration() -> anyhow::Result<()> {
let (server_transport, client_transport) = tokio::io::duplex(4096);
let server_handle = tokio::spawn(async move {
let server = TestServer::new().serve(server_transport).await?;
server.waiting().await?;
anyhow::Ok(())
});
let handler = TestClientHandler::new(true, true);
let client = handler.clone().serve(client_transport).await?;
let request = ServerRequest::CreateMessageRequest(CreateMessageRequest::new(
CreateMessageRequestParams::new(vec![SamplingMessage::user_text("test message")], 100)
.with_include_context(ContextInclusion::ThisServer),
));
let result = handler
.handle_request(
request.clone(),
RequestContext::new(NumberOrString::Number(1), client.peer().clone()),
)
.await?;
if let ClientResult::CreateMessageResult(result) = result {
let text = result
.message
.content
.first()
.unwrap()
.as_text()
.unwrap()
.text
.as_str();
assert!(
text.contains("test context"),
"Response should include context for ThisServer"
);
} else {
panic!("Expected CreateMessageResult");
}
let request = ServerRequest::CreateMessageRequest(CreateMessageRequest::new(
CreateMessageRequestParams::new(vec![SamplingMessage::user_text("test message")], 100)
.with_include_context(ContextInclusion::AllServers),
));
let result = handler
.handle_request(
request.clone(),
RequestContext::new(NumberOrString::Number(2), client.peer().clone()),
)
.await?;
if let ClientResult::CreateMessageResult(result) = result {
let text = result
.message
.content
.first()
.unwrap()
.as_text()
.unwrap()
.text
.as_str();
assert!(
text.contains("test context"),
"Response should include context for AllServers"
);
} else {
panic!("Expected CreateMessageResult");
}
let request = ServerRequest::CreateMessageRequest(CreateMessageRequest::new(
CreateMessageRequestParams::new(vec![SamplingMessage::user_text("test message")], 100)
.with_include_context(ContextInclusion::None),
));
let result = handler
.handle_request(
request.clone(),
RequestContext::new(NumberOrString::Number(3), client.peer().clone()),
)
.await?;
if let ClientResult::CreateMessageResult(result) = result {
let text = result
.message
.content
.first()
.unwrap()
.as_text()
.unwrap()
.text
.as_str();
assert!(
!text.contains("test context"),
"Response should not include context for None"
);
} else {
panic!("Expected CreateMessageResult");
}
client.cancel().await?;
server_handle.await??;
Ok(())
}
#[tokio::test]
async fn test_context_inclusion_ignored_integration() -> anyhow::Result<()> {
let (server_transport, client_transport) = tokio::io::duplex(4096);
let server_handle = tokio::spawn(async move {
let server = TestServer::new().serve(server_transport).await?;
server.waiting().await?;
anyhow::Ok(())
});
let handler = TestClientHandler::new(false, false);
let client = handler.clone().serve(client_transport).await?;
let request = ServerRequest::CreateMessageRequest(CreateMessageRequest::new(
CreateMessageRequestParams::new(vec![SamplingMessage::user_text("test message")], 100)
.with_include_context(ContextInclusion::ThisServer),
));
let result = handler
.handle_request(
request.clone(),
RequestContext::new(NumberOrString::Number(1), client.peer().clone()),
)
.await?;
if let ClientResult::CreateMessageResult(result) = result {
let text = result
.message
.content
.first()
.unwrap()
.as_text()
.unwrap()
.text
.as_str();
assert!(
!text.contains("test context"),
"Context should be ignored when client chooses not to honor requests"
);
} else {
panic!("Expected CreateMessageResult");
}
client.cancel().await?;
server_handle.await??;
Ok(())
}
#[tokio::test]
async fn test_message_sequence_integration() -> anyhow::Result<()> {
let (server_transport, client_transport) = tokio::io::duplex(4096);
let server_handle = tokio::spawn(async move {
let server = TestServer::new().serve(server_transport).await?;
server.waiting().await?;
anyhow::Ok(())
});
let handler = TestClientHandler::new(true, true);
let client = handler.clone().serve(client_transport).await?;
let request = ServerRequest::CreateMessageRequest(CreateMessageRequest::new(
CreateMessageRequestParams::new(
vec![
SamplingMessage::user_text("first message"),
SamplingMessage::assistant_text("second message"),
],
100,
)
.with_include_context(ContextInclusion::ThisServer),
));
let result = handler
.handle_request(
request.clone(),
RequestContext::new(NumberOrString::Number(1), client.peer().clone()),
)
.await?;
if let ClientResult::CreateMessageResult(result) = result {
let text = result
.message
.content
.first()
.unwrap()
.as_text()
.unwrap()
.text
.as_str();
assert!(
text.contains("test context"),
"Response should include context when ThisServer is specified"
);
assert_eq!(result.model, "test-model");
assert_eq!(
result.stop_reason,
Some(CreateMessageResult::STOP_REASON_END_TURN.to_string())
);
} else {
panic!("Expected CreateMessageResult");
}
client.cancel().await?;
server_handle.await??;
Ok(())
}
#[tokio::test]
async fn test_message_sequence_validation_integration() -> anyhow::Result<()> {
let (server_transport, client_transport) = tokio::io::duplex(4096);
let server_handle = tokio::spawn(async move {
let server = TestServer::new().serve(server_transport).await?;
server.waiting().await?;
anyhow::Ok(())
});
let handler = TestClientHandler::new(true, true);
let client = handler.clone().serve(client_transport).await?;
let request = ServerRequest::CreateMessageRequest(CreateMessageRequest::new(
CreateMessageRequestParams::new(
vec![
SamplingMessage::user_text("first user message"),
SamplingMessage::assistant_text("first assistant response"),
SamplingMessage::user_text("second user message"),
],
100,
),
));
let result = handler
.handle_request(
request.clone(),
RequestContext::new(NumberOrString::Number(1), client.peer().clone()),
)
.await?;
assert!(matches!(result, ClientResult::CreateMessageResult(_)));
let request = ServerRequest::CreateMessageRequest(CreateMessageRequest::new(
CreateMessageRequestParams::new(
vec![SamplingMessage::assistant_text("assistant message")],
100,
),
));
let result = handler
.handle_request(
request.clone(),
RequestContext::new(NumberOrString::Number(2), client.peer().clone()),
)
.await;
assert!(result.is_err());
client.cancel().await?;
server_handle.await??;
Ok(())
}
#[tokio::test]
async fn test_selective_context_handling_integration() -> anyhow::Result<()> {
let (server_transport, client_transport) = tokio::io::duplex(4096);
let server_handle = tokio::spawn(async move {
let server = TestServer::new().serve(server_transport).await?;
server.waiting().await?;
anyhow::Ok(())
});
let handler = TestClientHandler::new(true, false);
let client = handler.clone().serve(client_transport).await?;
let request = ServerRequest::CreateMessageRequest(CreateMessageRequest::new(
CreateMessageRequestParams::new(vec![SamplingMessage::user_text("test message")], 100)
.with_include_context(ContextInclusion::ThisServer),
));
let result = handler
.handle_request(
request.clone(),
RequestContext::new(NumberOrString::Number(1), client.peer().clone()),
)
.await?;
if let ClientResult::CreateMessageResult(result) = result {
let text = result
.message
.content
.first()
.unwrap()
.as_text()
.unwrap()
.text
.as_str();
assert!(
text.contains("test context"),
"ThisServer context request should be honored"
);
}
let request = ServerRequest::CreateMessageRequest(CreateMessageRequest::new(
CreateMessageRequestParams::new(vec![SamplingMessage::user_text("test message")], 100)
.with_include_context(ContextInclusion::AllServers),
));
let result = handler
.handle_request(
request.clone(),
RequestContext::new(NumberOrString::Number(2), client.peer().clone()),
)
.await?;
if let ClientResult::CreateMessageResult(result) = result {
let text = result
.message
.content
.first()
.unwrap()
.as_text()
.unwrap()
.text
.as_str();
assert!(
!text.contains("test context"),
"AllServers context request should be ignored"
);
}
client.cancel().await?;
server_handle.await??;
Ok(())
}
#[tokio::test]
async fn test_context_inclusion() -> anyhow::Result<()> {
let (server_transport, client_transport) = tokio::io::duplex(4096);
let server_handle = tokio::spawn(async move {
let server = TestServer::new().serve(server_transport).await?;
server.waiting().await?;
anyhow::Ok(())
});
let handler = TestClientHandler::new(true, true);
let client = handler.clone().serve(client_transport).await?;
let request = ServerRequest::CreateMessageRequest(CreateMessageRequest::new(
CreateMessageRequestParams::new(vec![SamplingMessage::user_text("test")], 100)
.with_include_context(ContextInclusion::ThisServer),
));
let result = handler
.handle_request(
request.clone(),
RequestContext::new(NumberOrString::Number(1), client.peer().clone()),
)
.await?;
if let ClientResult::CreateMessageResult(result) = result {
let text = result
.message
.content
.first()
.unwrap()
.as_text()
.unwrap()
.text
.as_str();
assert!(text.contains("test context"));
}
client.cancel().await?;
server_handle.await??;
Ok(())
}