use syn::Token;
use syn::punctuated::Punctuated;
use crate::ctxt::Ctxt;
use crate::{attr, check};
pub struct Container<'a> {
pub ident: syn::Ident,
pub attrs: attr::Container,
pub data: Struct<'a>,
pub generics: &'a syn::Generics,
pub original: &'a syn::DeriveInput,
}
pub type Struct<'a> = Vec<Field<'a>>;
pub struct Field<'a> {
pub member: syn::Member,
pub attrs: attr::Field,
pub ty: &'a syn::Type,
pub original: &'a syn::Field,
}
impl<'a> Container<'a> {
pub fn from_ast(cx: &Ctxt, item: &'a syn::DeriveInput) -> Option<Container<'a>> {
let attrs = attr::Container::from_ast(cx, item);
let mut data = match &item.data {
syn::Data::Struct(data) => struct_from_ast(cx, &data.fields, attrs.default()),
syn::Data::Union(_) => {
cx.error_spanned_by(item, "ClickHouse Native Row does not support unions");
return None;
}
syn::Data::Enum(_) => {
cx.error_spanned_by(item, "ClickHouse Native Row does not support enums");
return None;
}
};
for field in &mut data {
field.attrs.rename_by_rules(attrs.rename_all_rule());
}
let mut item = Container {
ident: item.ident.clone(),
attrs,
data,
generics: &item.generics,
original: item,
};
check::check(cx, &mut item);
Some(item)
}
}
fn struct_from_ast<'a>(
cx: &Ctxt,
fields: &'a syn::Fields,
container_default: &attr::Default,
) -> Vec<Field<'a>> {
match fields {
syn::Fields::Named(fields) => fields_from_ast(cx, &fields.named, container_default),
syn::Fields::Unnamed(fields) => {
cx.error_spanned_by(fields, "ClickHouse Native Row does not support tuple structs");
vec![]
}
syn::Fields::Unit => {
cx.error_spanned_by(fields, "ClickHouse Native Row does not support unit structs");
vec![]
}
}
}
fn fields_from_ast<'a>(
cx: &Ctxt,
fields: &'a Punctuated<syn::Field, Token![,]>,
container_default: &attr::Default,
) -> Vec<Field<'a>> {
fields
.iter()
.enumerate()
.map(|(i, field)| Field {
member: match &field.ident {
Some(ident) => syn::Member::Named(ident.clone()),
None => syn::Member::Unnamed(i.into()),
},
attrs: attr::Field::from_ast(cx, i, field, container_default),
ty: &field.ty,
original: field,
})
.collect()
}