#![recursion_limit = "128"]
extern crate proc_macro;
extern crate syn;
#[macro_use]
extern crate quote;
use proc_macro::TokenStream;
use syn::{Ident, VariantData};
#[proc_macro_derive(FromHashmap)]
pub fn from_hashmap(input: TokenStream) -> TokenStream {
let source = input.to_string();
let ast = syn::parse_macro_input(&source).unwrap();
let idents: Vec<Ident> = match ast.body {
syn::Body::Struct(vdata) => {
match vdata {
VariantData::Struct(fields) => {
let mut idents = Vec::new();
for ref field in fields.iter() {
match &field.ident {
&Some(ref ident) => idents.push(ident.clone()),
&None => panic!("Your struct is missing a field identity!"),
}
}
idents
},
VariantData::Tuple(_) | VariantData::Unit => {
panic!("You can only derive this for normal structs!");
},
}
},
syn::Body::Enum(_) => panic!("You can only derive this on structs!"),
};
let mut keys = Vec::new();
for ident in idents.iter() {
keys.push(String::from(ident.as_ref()));
}
let name = &ast.ident;
let (impl_generics, ty_generics, where_clause) = ast.generics.split_for_impl();
let tokens = quote! {
fn parse_pair<T>(v: &str) -> T where T : ::std::str::FromStr {
let res = v.parse::<T>();
match res {
Ok(val) => val,
Err(_) => panic!(format!("Unable to convert given input into required type: {}", v)),
}
}
impl #impl_generics FromHashmap<#name> for #name #ty_generics #where_clause {
fn from_hashmap(mut hm: ::std::collections::HashMap<String, String>) -> #name {
let mut settings = #name::default();
#(
match hm.entry(String::from(#keys)) {
::std::collections::hash_map::Entry::Occupied(occ_ent) => {
settings.#idents = parse_pair(occ_ent.get().as_str());
},
::std::collections::hash_map::Entry::Vacant(_) => (),
}
)*
settings
}
}
};
tokens.parse().unwrap()
}