use proc_macro2::Span; use syn::parse::{Parse, ParseStream}; use syn::punctuated::Punctuated; use syn::{Ident, Token, TypePath}; use crate::macro_flag::MacroFlag; use crate::util::error::diagnostic_error_enum; use crate::util::iterator_ext::IteratorExt; pub const INJECTABLE_MACRO_FLAGS: &[&str] = &["no_doc_hidden", "async", "no_declare_concrete_interface"]; pub struct InjectableMacroArgs { pub interface: Option, pub flags: Punctuated, } impl InjectableMacroArgs { pub fn check_flags(&self) -> Result<(), InjectableMacroArgsError> { for flag in &self.flags { if !INJECTABLE_MACRO_FLAGS.contains(&flag.flag.to_string().as_str()) { return Err(InjectableMacroArgsError::UnknownFlag { flag_ident: flag.flag.clone(), }); } } if let Some((dupe_flag_first, dupe_flag_second)) = self.flags.iter().find_duplicate() { return Err(InjectableMacroArgsError::DuplicateFlag { first_flag_ident: dupe_flag_first.flag.clone(), last_flag_span: dupe_flag_second.flag.span(), }); } Ok(()) } } impl Parse for InjectableMacroArgs { fn parse(input: ParseStream) -> Result { let input_fork = input.fork(); let mut interface = None; if input_fork.parse::().is_err() { // Input doesn't begin with flags interface = input.parse::().ok(); if interface.is_some() { let comma_input_lookahead = input.lookahead1(); if !comma_input_lookahead.peek(Token![,]) { return Ok(Self { interface, flags: Punctuated::new(), }); } input.parse::()?; } if input.is_empty() { return Ok(Self { interface, flags: Punctuated::new(), }); } } let flags = Punctuated::::parse_terminated(input)?; Ok(Self { interface, flags }) } } diagnostic_error_enum! { pub enum InjectableMacroArgsError { #[error("Unknown flag '{flag_ident}'"), span = flag_ident.span()] #[ help("Expected one of: {}", INJECTABLE_MACRO_FLAGS.join(", ")), span = flag_ident.span() ] UnknownFlag { flag_ident: Ident }, #[error("Duplicate flag '{first_flag_ident}'"), span = first_flag_ident.span()] #[note("Previously mentioned here"), span = last_flag_span] DuplicateFlag { first_flag_ident: Ident, last_flag_span: Span }, } } #[cfg(test)] mod tests { use std::error::Error; use proc_macro2::Span; use quote::{format_ident, quote}; use syn::{parse2, LitBool}; use super::*; use crate::test_utils; #[test] fn can_parse_with_only_interface() -> Result<(), Box> { let input_args = quote! { IFoo }; let injectable_macro_args = parse2::(input_args)?; assert!(matches!(injectable_macro_args.interface, Some(interface) if interface == TypePath { qself: None, path: test_utils::create_path(&[ test_utils::create_path_segment(format_ident!("IFoo"), &[]) ]) } )); assert!(injectable_macro_args.flags.is_empty()); Ok(()) } #[test] fn can_parse_with_nothing() -> Result<(), Box> { let input_args = quote! {}; let injectable_macro_args = parse2::(input_args)?; assert!(injectable_macro_args.interface.is_none()); assert!(injectable_macro_args.flags.is_empty()); Ok(()) } #[test] fn can_parse_with_interface_and_flags() -> Result<(), Box> { let input_args = quote! { IFoo, no_doc_hidden = true, async = false }; let injectable_macro_args = parse2::(input_args)?; assert!(matches!(injectable_macro_args.interface, Some(interface) if interface == TypePath { qself: None, path: test_utils::create_path(&[ test_utils::create_path_segment(format_ident!("IFoo"), &[]) ]) } )); assert_eq!( injectable_macro_args.flags, Punctuated::from_iter([ MacroFlag { flag: format_ident!("no_doc_hidden"), is_on: LitBool::new(true, Span::call_site()) }, MacroFlag { flag: format_ident!("async"), is_on: LitBool::new(false, Span::call_site()) } ]) ); Ok(()) } #[test] fn can_parse_with_flags_only() -> Result<(), Box> { let input_args = quote! { async = false, no_declare_concrete_interface = false }; let injectable_macro_args = parse2::(input_args)?; assert!(injectable_macro_args.interface.is_none()); assert_eq!( injectable_macro_args.flags, Punctuated::from_iter([ MacroFlag { flag: format_ident!("async"), is_on: LitBool::new(false, Span::call_site()) }, MacroFlag { flag: format_ident!("no_declare_concrete_interface"), is_on: LitBool::new(false, Span::call_site()) } ]) ); Ok(()) } #[test] fn can_parse_with_unknown_flag() { let input_args = quote! { IFoo, haha = true, async = false }; assert!(parse2::(input_args).is_ok()); } #[test] fn can_parse_with_duplicate_flag() { assert!(parse2::(quote! { IFoo, async = false, no_doc_hidden = true, async = false }) .is_ok()); assert!(parse2::(quote! { IFoo, async = true , no_doc_hidden = true, async = false }) .is_ok()); } #[test] fn check_flags_fail_with_unknown_flag() -> Result<(), Box> { let input_args = quote! { IFoo, haha = true, async = false }; let injectable_macro_args = parse2::(input_args)?; assert!(injectable_macro_args.check_flags().is_err()); Ok(()) } #[test] fn check_flags_fail_with_duplicate_flag() -> Result<(), Box> { let macro_args = parse2::(quote! { IFoo, async = false, no_doc_hidden = true, async = false })?; assert!(macro_args.check_flags().is_err()); let macro_args_two = parse2::(quote! { IFoo, async = true , no_doc_hidden = true, async = false })?; assert!(macro_args_two.check_flags().is_err()); Ok(()) } }