use quote::format_ident;
use std::borrow::Cow;
use std::ops::Deref;
use syn::parse::{Parse, ParseStream};
use syn::punctuated::Punctuated;
use syn::{AttrStyle, Attribute, Ident, Path, Result, Token};
fn is_nuts_bytes_attr(attr: &Attribute) -> bool {
attr.style == AttrStyle::Outer && attr.path().is_ident("nuts_bytes")
}
struct AttributeList<T>(Punctuated<T, Token![,]>);
impl<T> Deref for AttributeList<T> {
type Target = Punctuated<T, Token![,]>;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl<T: Parse> Parse for AttributeList<T> {
fn parse(input: ParseStream) -> Result<Self> {
let attrs = input.call(Punctuated::parse_separated_nonempty)?;
Ok(AttributeList(attrs))
}
}
#[derive(Clone)]
enum FieldAttribute {
Map(Path),
MapFromBytes(Path),
MapToBytes(Path),
Skip,
Default(Path),
}
impl FieldAttribute {
fn as_map(&self) -> Option<&Path> {
match self {
Self::Map(path) => Some(path),
_ => None,
}
}
fn as_map_from_bytes(&self) -> Option<&Path> {
match self {
Self::MapFromBytes(path) => Some(path),
_ => None,
}
}
fn as_map_to_bytes(&self) -> Option<&Path> {
match self {
Self::MapToBytes(path) => Some(path),
_ => None,
}
}
fn is_skip(&self) -> bool {
matches!(self, Self::Skip)
}
fn as_default(&self) -> Option<&Path> {
match self {
Self::Default(path) => Some(path),
_ => None,
}
}
}
impl Parse for FieldAttribute {
fn parse(input: ParseStream) -> Result<Self> {
let key: Ident = input.parse()?;
if key == "map" {
let _: Token![=] = input.parse()?;
Ok(Self::Map(input.parse()?))
} else if key == "map_from_bytes" {
let _: Token![=] = input.parse()?;
Ok(Self::MapFromBytes(input.parse()?))
} else if key == "map_to_bytes" {
let _: Token![=] = input.parse()?;
Ok(Self::MapToBytes(input.parse()?))
} else if key == "skip" {
Ok(Self::Skip)
} else if key == "default" {
let _: Token![=] = input.parse()?;
Ok(Self::Default(input.parse()?))
} else {
Err(syn::Error::new_spanned(
key,
"unsupported attribute for nuts_bytes",
))
}
}
}
pub struct FieldAttributes(Vec<FieldAttribute>);
impl FieldAttributes {
pub fn parse(attrs: &[Attribute]) -> Result<FieldAttributes> {
let mut attr_vec = vec![];
let filtered = attrs.iter().filter(|attr| is_nuts_bytes_attr(attr));
for attr in filtered {
let list = attr.parse_args::<AttributeList<FieldAttribute>>()?;
attr_vec.extend(list.iter().map(Clone::clone));
}
Ok(FieldAttributes(attr_vec))
}
pub fn map_from_bytes(&self) -> Option<Cow<Path>> {
if let Some(path) = self.0.iter().find_map(|attr| attr.as_map_from_bytes()) {
Some(Cow::Borrowed(path))
} else {
self.0
.iter()
.find_map(|attr| attr.as_map().cloned())
.map(|mut path| {
path.segments.push(format_ident!("from_bytes").into());
Cow::Owned(path)
})
}
}
pub fn map_to_bytes(&self) -> Option<Cow<Path>> {
if let Some(path) = self.0.iter().find_map(|attr| attr.as_map_to_bytes()) {
Some(Cow::Borrowed(path))
} else {
self.0
.iter()
.find_map(|attr| attr.as_map().cloned())
.map(|mut path| {
path.segments.push(format_ident!("to_bytes").into());
Cow::Owned(path)
})
}
}
pub fn is_skip(&self) -> bool {
self.0.iter().any(|attr| attr.is_skip())
}
pub fn default(&self) -> Option<&Path> {
self.0
.iter()
.find(|attr| attr.as_default().is_some())
.and_then(|attr| attr.as_default())
}
}
macro_rules! parse_field_attributes {
($input:expr) => {
match crate::attr::FieldAttributes::parse($input) {
Ok(attrs) => attrs,
Err(err) => return err.into_compile_error().into(),
}
};
}
pub(crate) use parse_field_attributes;