aboutsummaryrefslogtreecommitdiff
path: root/derive/src/from/mod.rs
blob: 818623d829f580b47d5943649ef0117a337bfecf (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
use quote::quote;
use syn::parse_macro_input;

use darling::{FromDeriveInput, FromVariant};

#[derive(Debug, FromDeriveInput)]
#[darling(attributes(from), supports(enum_any))]
struct Enum {
    ident: syn::Ident,
}

#[derive(Debug, FromVariant)]
#[darling(attributes(from))]
struct EnumField {
    ident: syn::Ident,
    fields: darling::ast::Fields<syn::Type>,

    #[darling(default)]
    ignore: bool,
}

pub fn from(input: proc_macro::TokenStream) -> proc_macro::TokenStream {
    let input = parse_macro_input!(input as syn::DeriveInput);
    // println!("{:#?}", input);

    let network = Enum::from_derive_input(&input).unwrap();
    // println!("{:#?}", network);

    let enum_name = network.ident;

    let fields: Vec<EnumField> = match &input.data {
        syn::Data::Enum(data) => data
            .variants
            .iter()
            .map(|field| EnumField::from_variant(field).expect("Could not parse field"))
            .collect(),
        _ => panic!("from: not an enum"),
    };

    let derives = fields
        .iter()
        .filter(|field| !field.fields.fields.is_empty() && !field.ignore)
        .map(|field| {
            let variant = &field.ident;
            let inner_type = &field.fields.fields[0];

            quote! {
                impl From<#inner_type> for #enum_name {
                    fn from(input: #inner_type) -> Self {
                        Self::#variant(input)
                    }
                }

                impl std::convert::TryFrom<#enum_name> for #inner_type {
                    type Error = crate::error::ProtocolError;

                    fn try_from(input: #enum_name) -> Result<Self, Self::Error> {
                        match input {
                            #enum_name::#variant(input) => Ok(input),
                            _ => Err(crate::error::ProtocolError::WrongVariant),
                        }
                    }
                }

                impl std::convert::TryFrom<&#enum_name> for #inner_type {
                    type Error = crate::error::ProtocolError;

                    fn try_from(input: &#enum_name) -> Result<Self, Self::Error> {
                        match input {
                            #enum_name::#variant(input) => Ok(input.clone()),
                            _ => Err(crate::error::ProtocolError::WrongVariant),
                        }
                    }
                }
            }
        });

    // println!("{:#?}", fields);

    let gen = quote! {
        #(#derives)*
    };

    // println!("{}", gen);

    gen.into()
}