use proc_macro2::Ident;
use syn::{Error, Lit, Meta, NestedMeta, Result};
pub trait FromMeta: Sized {
fn from(meta: Option<Meta>) -> Result<Self>;
}
impl FromMeta for Option<()> {
fn from(meta: Option<Meta>) -> Result<Self> {
meta.map_or(Ok(None), |m| {
if matches!(m, Meta::Path(_)) {
Ok(Some(()))
} else {
Err(Error::new_spanned(m, "unexpected value"))
}
})
}
}
impl FromMeta for Vec<Ident> {
fn from(meta: Option<Meta>) -> Result<Self> {
meta.map_or(Ok(Vec::new()), |m| {
if let Meta::List(list) = m {
list.nested
.iter()
.map(|m| {
if let NestedMeta::Meta(Meta::Path(m)) = m {
m.get_ident()
.cloned()
.ok_or_else(|| Error::new_spanned(m, "expected ident"))
} else {
Err(Error::new_spanned(m, "expected ident"))
}
})
.collect()
} else {
Err(Error::new_spanned(m, "expected idents"))
}
})
}
}
impl FromMeta for Option<Vec<Ident>> {
fn from(meta: Option<Meta>) -> Result<Self> {
meta.map_or(Ok(None), |m| {
if let Meta::List(list) = m {
list.nested
.iter()
.map(|m| {
if let NestedMeta::Meta(Meta::Path(m)) = m {
m.get_ident()
.cloned()
.ok_or_else(|| Error::new_spanned(m, "expected ident"))
} else {
Err(Error::new_spanned(m, "expected ident"))
}
})
.collect::<Result<_>>()
.map(Some)
} else {
Err(Error::new_spanned(m, "expected idents"))
}
})
}
}
pub trait FromLit: Sized {
fn from(lit: Option<Lit>) -> Result<Self>;
}
impl<F: FromLit> FromMeta for F {
fn from(meta: Option<Meta>) -> Result<Self> {
let lit = meta
.map(|m| match m {
Meta::NameValue(mnv) => Ok(mnv.lit),
_ => Err(Error::new_spanned(m, "expected named value")),
})
.transpose()?;
Self::from(lit)
}
}
impl FromLit for Option<Lit> {
fn from(lit: Option<Lit>) -> Result<Self> {
Ok(lit)
}
}
impl FromLit for String {
fn from(lit: Option<Lit>) -> Result<Self> {
if let Some(Lit::Str(v)) = lit {
Ok(v.value())
} else {
Err(Error::new_spanned(lit, "expected string"))
}
}
}
impl FromLit for Vec<u8> {
fn from(lit: Option<Lit>) -> Result<Self> {
if let Some(Lit::ByteStr(v)) = lit {
Ok(v.value())
} else {
Err(Error::new_spanned(lit, "expected bytes"))
}
}
}
impl FromLit for u8 {
fn from(lit: Option<Lit>) -> Result<Self> {
if let Some(Lit::Byte(v)) = lit {
Ok(v.value())
} else {
Err(Error::new_spanned(lit, "expected byte"))
}
}
}
impl FromLit for char {
fn from(lit: Option<Lit>) -> Result<Self> {
if let Some(Lit::Char(v)) = lit {
Ok(v.value())
} else {
Err(Error::new_spanned(lit, "expected char"))
}
}
}
impl FromLit for i32 {
fn from(lit: Option<Lit>) -> Result<Self> {
if let Some(Lit::Int(v)) = &lit {
v.base10_parse()
} else {
Err(Error::new_spanned(lit, "expected i32"))
}
}
}
impl FromLit for f32 {
fn from(lit: Option<Lit>) -> Result<Self> {
if let Some(Lit::Float(v)) = &lit {
v.base10_parse()
} else {
Err(Error::new_spanned(lit, "expected f32"))
}
}
}
impl FromLit for bool {
fn from(lit: Option<Lit>) -> Result<Self> {
if let Some(Lit::Bool(v)) = lit {
Ok(v.value)
} else {
Err(Error::new_spanned(lit, "expected bool"))
}
}
}
impl<V: FromLit> FromLit for Option<V> {
fn from(lit: Option<Lit>) -> Result<Self> {
if lit.is_some() {
Some(V::from(lit)).transpose()
} else {
Ok(None)
}
}
}