Skip to content

Commit 2f643f9

Browse files
committed
fix: use the json rpc error from the initialize response and bubble it up to the client
1 parent eb5a7f7 commit 2f643f9

File tree

2 files changed

+58
-1
lines changed

2 files changed

+58
-1
lines changed

crates/rmcp/src/service/client.rs

Lines changed: 8 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ use crate::{
88
ArgumentInfo, CallToolRequest, CallToolRequestParam, CallToolResult, CancelledNotification,
99
CancelledNotificationParam, ClientInfo, ClientJsonRpcMessage, ClientNotification,
1010
ClientRequest, ClientResult, CompleteRequest, CompleteRequestParam, CompleteResult,
11-
CompletionContext, CompletionInfo, GetPromptRequest, GetPromptRequestParam,
11+
CompletionContext, CompletionInfo, ErrorData, GetPromptRequest, GetPromptRequestParam,
1212
GetPromptResult, InitializeRequest, InitializedNotification, JsonRpcResponse,
1313
ListPromptsRequest, ListPromptsResult, ListResourceTemplatesRequest,
1414
ListResourceTemplatesResult, ListResourcesRequest, ListResourcesResult, ListToolsRequest,
@@ -44,6 +44,9 @@ pub enum ClientInitializeError {
4444
context: Cow<'static, str>,
4545
},
4646

47+
#[error("JSON-RPC error: {0}")]
48+
JsonRpcError(ErrorData),
49+
4750
#[error("Cancelled")]
4851
Cancelled,
4952
}
@@ -92,6 +95,10 @@ where
9295
ServerJsonRpcMessage::Response(JsonRpcResponse { id, result, .. }) => {
9396
break Ok((result, id));
9497
}
98+
// Handle JSON-RPC error responses
99+
ServerJsonRpcMessage::Error(error) => {
100+
break Err(ClientInitializeError::JsonRpcError(error.error));
101+
}
95102
// Server could send logging messages before handshake
96103
ServerJsonRpcMessage::Notification(mut notification) => {
97104
let ServerNotification::LoggingMessageNotification(logging) =
Lines changed: 50 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
// cargo test --features "server client" --package rmcp test_client_initialization
2+
mod common;
3+
4+
use common::handlers::TestClientHandler;
5+
use rmcp::{
6+
ServiceExt,
7+
model::{
8+
ErrorCode, ErrorData, JsonRpcError, JsonRpcVersion2_0, RequestId, ServerJsonRpcMessage,
9+
},
10+
transport::{IntoTransport, Transport},
11+
};
12+
use std::borrow::Cow;
13+
14+
#[tokio::test]
15+
async fn test_client_init_handles_jsonrpc_error() {
16+
let (server_transport, client_transport) = tokio::io::duplex(1024);
17+
let mut server = IntoTransport::<rmcp::RoleServer, _, _>::into_transport(server_transport);
18+
19+
let client_handle = tokio::spawn(async move {
20+
TestClientHandler::new(true, true)
21+
.serve(client_transport)
22+
.await
23+
});
24+
25+
tokio::spawn(async move {
26+
let _init_request = server.receive().await;
27+
28+
let error_msg = ServerJsonRpcMessage::Error(JsonRpcError {
29+
jsonrpc: JsonRpcVersion2_0,
30+
id: RequestId::Number(1),
31+
error: ErrorData {
32+
code: ErrorCode(-32600),
33+
message: Cow::Borrowed("Invalid Request"),
34+
data: None,
35+
},
36+
});
37+
let _: Result<(), _> = server.send(error_msg).await;
38+
});
39+
40+
let result = client_handle.await.unwrap();
41+
42+
assert!(result.is_err());
43+
match result {
44+
Err(rmcp::service::ClientInitializeError::JsonRpcError(error_data)) => {
45+
assert_eq!(error_data.code, ErrorCode(-32600));
46+
assert_eq!(error_data.message, "Invalid Request");
47+
}
48+
_ => panic!("Expected ClientInitializeError::JsonRpcError"),
49+
}
50+
}

0 commit comments

Comments
 (0)