#![ warn( missing_docs ) ]
#![ warn( missing_debug_implementations ) ]
#[ macro_export ]
macro_rules! tree_print
{
( $src : expr ) =>
{{
let result = $crate::tree_export_str!( $src );
println!( "{}", result );
result
}};
( $( $src : expr ),+ $(,)? ) =>
{{
$( $crate::tree_print!( $src ) );+
}};
}
#[ macro_export ]
macro_rules! tree_export_str
{
( $src : expr ) =>
{{
let src2 = &$src;
format!( "{} : {} :\n{:#?}", stringify!( $src ), quote!{ #src2 }, $src )
}};
}
#[ macro_export ]
macro_rules! syn_err
{
( $msg : expr ) =>
{
syn::Error::new( proc_macro2::Span::call_site(), $msg )
};
( _, $msg : expr ) =>
{
syn::Error::new( proc_macro2::Span::call_site(), $msg )
};
( $span : expr, $msg : expr ) =>
{
syn::Error::new( syn::spanned::Spanned::span( &( $span ) ), $msg )
};
( $span : expr, $msg : expr, $( $arg : expr ),+ ) =>
{
syn::Error::new( syn::spanned::Spanned::span( &( $span ) ), format!( $msg, $( $arg ),+ ) )
};
( _, $msg : expr, $( $arg : expr ),+ ) =>
{
syn::Error::new( proc_macro2::Span::call_site(), format!( $msg, $( $arg ),+ ) )
};
}
#[derive( Debug, PartialEq, Copy, Clone )]
pub enum ContainerKind
{
No,
Vector,
HashMap,
HashSet,
}
pub fn type_container_kind( ty : &syn::Type ) -> ContainerKind
{
if let syn::Type::Path( path ) = ty
{
let last = &path.path.segments.last();
if last.is_none()
{
return ContainerKind::No
}
match last.unwrap().ident.to_string().as_ref()
{
"Vec" => { return ContainerKind::Vector }
"HashMap" => { return ContainerKind::HashMap }
"HashSet" => { return ContainerKind::HashSet }
_ => { return ContainerKind::No }
}
}
ContainerKind::No
}
pub fn type_optional_container_kind( ty : &syn::Type ) -> ( ContainerKind, bool )
{
if type_rightmost( ty ) == Some( "Option".to_string() )
{
let ty2 = type_parameters( ty, 0 ..= 0 ).first().map( | e | *e );
if ty2.is_none()
{
return ( ContainerKind::No, false )
}
let ty2 = ty2.unwrap();
return ( type_container_kind( ty2 ), true );
}
return ( type_container_kind( ty ), false );
}
pub fn type_rightmost( ty : &syn::Type ) -> Option< String >
{
if let syn::Type::Path( path ) = ty
{
let last = &path.path.segments.last();
if last.is_none()
{
return None;
}
return Some( last.unwrap().ident.to_string() );
}
None
}
use winterval::*;
pub fn type_parameters< R >( ty : &syn::Type, range : R ) -> Vec< &syn::Type >
where
R : std::convert::Into< Interval >
{
let range = range.into();
if let syn::Type::Path( syn::TypePath{ path : syn::Path { ref segments, .. }, .. } ) = ty
{
let last = &segments.last();
if last.is_none()
{
return vec![ &ty ]
}
let args = &last.unwrap().arguments;
if let syn::PathArguments::AngleBracketed( ref args2 ) = args
{
let args3 = &args2.args;
let selected : Vec< &syn::Type > = args3
.iter()
.skip_while( | e | if let syn::GenericArgument::Type( _ ) = e { false } else { true } )
.skip( range.first().try_into().unwrap() )
.take( range.len().try_into().unwrap() )
.map( | e | if let syn::GenericArgument::Type( ty ) = e { ty } else { unreachable!( "Expects Type" ) } )
.collect();
return selected;
}
}
vec![ &ty ]
}
pub fn attr_pair_single( attr : &syn::Attribute ) -> Result< ( String, syn::Lit, syn::Meta ), syn::Error >
{
use syn::spanned::Spanned;
let meta = attr.parse_meta()?;
let ( key, val );
match meta
{
syn::Meta::List( ref meta_list ) =>
match meta_list.nested.first()
{
Some( nested_meta ) => match nested_meta
{
syn::NestedMeta::Meta( meta2 ) => match meta2
{
syn::Meta::NameValue( name_value ) => {
if meta_list.nested.len() != 1
{
return Err( syn::Error::new( attr.span(), format!( "Expected single element of the list, but got {}", meta_list.nested.len() ) ) );
}
key = name_value.path.get_ident().unwrap().to_string();
val = name_value.lit.clone();
},
_ => return Err( syn::Error::new( attr.span(), "Unknown format of attribute, expected syn::Meta::NameValue( name_value )" ) ),
},
_ => return Err( syn::Error::new( attr.span(), "Unknown format of attribute, expected syn::NestedMeta::Meta( meta2 )" ) ),
},
_ => return Err( syn::Error::new( attr.span(), "Unknown format of attribute, expected Some( nested_meta )" ) ),
},
_ => return Err( syn::Error::new( attr.span(), "Unknown format of attribute, expected syn::Meta::List( meta_list )" ) ),
};
Ok( ( key, val, meta ) )
}