use serde::{
de::{self, Visitor}, Deserializer, Serialize,
Serializer,
};
use std::fmt;
pub fn serialize<S>(level: &Option<Vec<String>>, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
match level {
Some(vec) if vec.len() == 1 => serializer.serialize_str(&vec[0]),
Some(vec) if !vec.is_empty() => vec.serialize(serializer),
_ => serializer.serialize_none(),
}
}
pub fn deserialize<'de, D>(deserializer: D) -> Result<Option<Vec<String>>, D::Error>
where
D: Deserializer<'de>,
{
deserializer.deserialize_option(OptionVisitor)
}
struct OptionVisitor;
impl<'de> Visitor<'de> for OptionVisitor {
type Value = Option<Vec<String>>;
fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str("a string, comma-separated string, or array of strings")
}
fn visit_none<E>(self) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(None)
}
fn visit_some<D>(self, deserializer: D) -> Result<Self::Value, D::Error>
where
D: Deserializer<'de>,
{
deserializer.deserialize_any(StringOrVecOptionVisitor)
}
}
struct StringOrVecOptionVisitor;
impl<'de> Visitor<'de> for StringOrVecOptionVisitor {
type Value = Option<Vec<String>>;
fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str("a string, comma-separated string, or array of strings")
}
fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
where
E: de::Error,
{
self.visit_string(v.to_string())
}
fn visit_string<E>(self, v: String) -> Result<Self::Value, E>
where
E: de::Error,
{
let vec: Vec<_> = v
.split(',')
.map(|s| s.trim().to_string())
.filter(|s| !s.is_empty())
.collect();
Ok(Some(vec))
}
fn visit_seq<A>(self, mut seq: A) -> Result<Self::Value, A::Error>
where
A: de::SeqAccess<'de>,
{
let mut vec = Vec::new();
while let Some(element) = seq.next_element::<String>()? {
let element = element.trim().to_string();
if element.is_empty() {
continue;
}
vec.push(element);
}
Ok(Some(vec))
}
}