Add attributes parsing

This commit is contained in:
vagola 2020-01-03 15:28:45 +03:00
parent a4b7f100a8
commit 6d50407670
3 changed files with 65 additions and 7 deletions

View File

@ -10,7 +10,6 @@ publish = false
proc-macro = true proc-macro = true
[dependencies] [dependencies]
# Versions match serde crate to reduce compile time. proc-macro2 = "1.0"
proc-macro2 = "0.4.30" syn = "1.0"
syn = "0.15.44" quote = "1.0"
quote = "0.6.13"

View File

@ -4,9 +4,10 @@ use proc_macro::TokenStream as TokenStream1;
use proc_macro2::Ident; use proc_macro2::Ident;
use proc_macro2::TokenStream as TokenStream2; use proc_macro2::TokenStream as TokenStream2;
use quote::{quote, TokenStreamExt}; use quote::{quote, TokenStreamExt};
use syn::{parse_macro_input, Data, DeriveInput, Field, Fields}; use std::iter::FromIterator;
use syn::{parse_macro_input, Data, DeriveInput, Field, Fields, Lit, Meta, NestedMeta};
#[proc_macro_derive(Packet)] #[proc_macro_derive(Packet, attributes(packet))]
pub fn derive_packet(input: proc_macro::TokenStream) -> TokenStream1 { pub fn derive_packet(input: proc_macro::TokenStream) -> TokenStream1 {
let input = parse_macro_input!(input as DeriveInput); let input = parse_macro_input!(input as DeriveInput);
let name = &input.ident; let name = &input.ident;
@ -32,8 +33,11 @@ fn impl_encoder_trait(name: &Ident, fields: &Fields) -> TokenStream2 {
let encode = quote_field(fields, |field| { let encode = quote_field(fields, |field| {
let name = &field.ident; let name = &field.ident;
let unparsed_meta = get_packet_field_meta(field);
let parsed_meta = parse_field_meta(&unparsed_meta);
quote! { quote! {
crate::Encoder::encode(&self.#name, writer); crate::Encoder::encode(&self.#name, writer)?;
} }
}); });
@ -66,6 +70,59 @@ fn impl_decoder_trait(name: &Ident, fields: &Fields) -> TokenStream2 {
} }
} }
#[derive(Debug)]
struct PacketFieldMeta {
module: Option<String>,
max_length: Option<u16>,
}
fn parse_field_meta(meta_list: &Vec<NestedMeta>) -> PacketFieldMeta {
let mut module = None;
let mut max_length = None;
for meta in meta_list {
match meta {
NestedMeta::Meta(Meta::NameValue(m)) => match &m.path {
p if p.is_ident("with") => {
if let Lit::Str(lit_str) = &m.lit {
module = Some(lit_str.value());
}
}
p if p.is_ident("max_length") => {
if let Lit::Int(lit_int) = &m.lit {
max_length = Some(
lit_int
.base10_parse::<u16>()
.expect("Failed to parse max length attribute"),
);
}
}
p => panic!(
"Packet derive received unrecognized attribute : \"{}\"",
p.get_ident().unwrap()
),
},
_ => panic!("Packet derive support only named meta values"),
}
}
PacketFieldMeta { module, max_length }
}
fn get_packet_field_meta(field: &Field) -> Vec<NestedMeta> {
field
.attrs
.iter()
.filter(|a| a.path.is_ident("packet"))
.map(|a| a.parse_meta().expect("Failed to parse field attribute"))
.map(|m| match m {
Meta::List(meta_list) => Vec::from_iter(meta_list.nested),
_ => panic!("Packet derive support only list attributes"),
})
.flatten()
.collect()
}
fn quote_field<F: Fn(&Field) -> TokenStream2>(fields: &Fields, func: F) -> TokenStream2 { fn quote_field<F: Fn(&Field) -> TokenStream2>(fields: &Fields, func: F) -> TokenStream2 {
let mut output = quote!(); let mut output = quote!();

View File

@ -170,6 +170,7 @@ impl LoginDisconnect {
#[derive(Packet, Debug)] #[derive(Packet, Debug)]
pub struct EncryptionRequest { pub struct EncryptionRequest {
#[packet(max_length = 20)]
pub server_id: String, pub server_id: String,
pub public_key: Vec<u8>, pub public_key: Vec<u8>,
pub verify_token: Vec<u8>, pub verify_token: Vec<u8>,
@ -207,6 +208,7 @@ impl LoginSuccess {
#[derive(Packet, Debug)] #[derive(Packet, Debug)]
pub struct SetCompression { pub struct SetCompression {
#[packet(with = "varint")]
pub threshold: i32, pub threshold: i32,
} }