From 395c5a93435afc8ac76de41f5b2d6bd957f22bbb Mon Sep 17 00:00:00 2001 From: Demur Rumed Date: Mon, 6 Nov 2023 13:43:05 +0000 Subject: [PATCH] derive: syn 2 --- derive/Cargo.toml | 2 +- derive/src/lib.rs | 38 +++++++++++++++++++------------------- 2 files changed, 20 insertions(+), 20 deletions(-) diff --git a/derive/Cargo.toml b/derive/Cargo.toml index 58e1fbf572..e868790a02 100644 --- a/derive/Cargo.toml +++ b/derive/Cargo.toml @@ -18,6 +18,6 @@ edition = "2021" proc-macro = true [dependencies] -syn = "1.0" +syn = { version = "2.0", features = ["full"] } proc-macro2 = "1.0" quote = "1.0" diff --git a/derive/src/lib.rs b/derive/src/lib.rs index fbc3ef2e1c..339f5505d1 100644 --- a/derive/src/lib.rs +++ b/derive/src/lib.rs @@ -2,8 +2,8 @@ use proc_macro2::TokenStream; use quote::{format_ident, quote, quote_spanned, ToTokens}; use syn::spanned::Spanned; use syn::{ - parse_macro_input, parse_quote, Attribute, Data, DeriveInput, Fields, GenericParam, Generics, - Ident, Index, Lit, Meta, MetaNameValue, NestedMeta, + parse_macro_input, parse_quote, Attribute, Data, DeriveInput, Expr, ExprLit, ExprPath, + Fields, GenericParam, Generics, Ident, Index, Lit, Meta, MetaNameValue, }; @@ -81,33 +81,33 @@ struct Attributes { impl Attributes { fn parse(attrs: &[Attribute]) -> Self { let mut out = Self::default(); - for attr in attrs.iter().filter(|a| a.path.is_ident("visit")) { - let meta = attr.parse_meta().expect("visit attribute"); - match meta { - Meta::List(l) => { - for nested in &l.nested { - match nested { - NestedMeta::Meta(Meta::NameValue(v)) => out.parse_name_value(v), - _ => panic!("Expected #[visit(key = \"value\")]"), - } - } + for attr in attrs { + if let Meta::NameValue(ref namevalue) = attr.meta { + if namevalue.path.is_ident("visit") { + out.parse_name_value(namevalue); } - _ => panic!("Expected #[visit(...)]"), } } + if out.with.is_none() { + panic!("Expected #[visit(...)]"); + } out } /// Updates self with a name value attribute fn parse_name_value(&mut self, v: &MetaNameValue) { - if v.path.is_ident("with") { - match &v.lit { - Lit::Str(s) => self.with = Some(format_ident!("{}", s.value(), span = s.span())), - _ => panic!("Expected a string value, got {}", v.lit.to_token_stream()), + if let Expr::Assign(ref assign) = v.value { + if let Expr::Path(ExprPath { ref path, .. }) = *assign.left { + if path.is_ident("with") { + match *assign.right { + Expr::Lit(ExprLit { lit: Lit::Str(ref s), .. }) => self.with = Some(format_ident!("{}", s.value(), span = s.span())), + _ => panic!("Expected a string value, got {}", v.value.to_token_stream()), + } + return; + } } - return; } - panic!("Unrecognised kv attribute {}", v.path.to_token_stream()) + panic!("Unrecognised kv attribute {}", v.to_token_stream()) } /// Returns the pre and post visit token streams