use crate::mcp::support::{is_empty_object, serialize_bool_as_empty_object};
use serde::de::{MapAccess, Visitor};
use serde::ser::SerializeMap;
use serde::{Deserialize, Deserializer, Serialize, Serializer};
use serde_json::Value;
use serde_with::skip_serializing_none; use std::fmt;
#[derive(Debug, Clone, Default, PartialEq)]
pub struct ClientCapabilities {
pub experimental: Option<Value>,
pub roots: Option<ClientRootsCapabilities>,
pub sampling: bool,
}
#[skip_serializing_none]
#[derive(Debug, Clone, Default, PartialEq, Eq, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct ClientRootsCapabilities {
pub list_changed: Option<bool>,
}
impl Serialize for ClientCapabilities {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
let mut map = serializer.serialize_map(None)?;
if let Some(experimental) = &self.experimental {
map.serialize_entry("experimental", experimental)?;
}
if let Some(roots) = &self.roots {
map.serialize_entry("roots", roots)?;
}
serialize_bool_as_empty_object::<S>(&mut map, "sampling", self.sampling)?;
map.end()
}
}
impl<'de> Deserialize<'de> for ClientCapabilities {
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
where
D: Deserializer<'de>,
{
struct ClientCapabilitiesVisitor;
impl<'de> Visitor<'de> for ClientCapabilitiesVisitor {
type Value = ClientCapabilities;
fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str("a map representing ClientCapabilities")
}
fn visit_map<M>(self, mut map: M) -> Result<Self::Value, M::Error>
where
M: MapAccess<'de>,
{
let mut experimental: Option<Value> = None;
let mut roots: Option<ClientRootsCapabilities> = None;
let mut sampling: bool = false;
while let Some(key) = map.next_key::<String>()? {
match key.as_str() {
"experimental" => {
experimental = Some(map.next_value()?);
}
"roots" => {
roots = Some(map.next_value()?);
}
"sampling" => {
let sampling_value: Value = map.next_value()?;
if is_empty_object(&sampling_value) {
sampling = true;
}
}
_ => {
let _ = map.next_value::<Value>()?;
}
}
}
Ok(ClientCapabilities {
experimental,
roots,
sampling,
})
}
}
deserializer.deserialize_map(ClientCapabilitiesVisitor)
}
}