assert_type_match/
flags.rs1use syn::parse::{Parse, ParseStream};
2use syn::spanned::Spanned;
3use syn::token::Token;
4use syn::{LitBool, Token};
5
6pub(crate) type Flag = Option<LitBool>;
8
9pub(crate) trait ParseFlag {
11 fn parse_flag<T: Parse + Spanned>(&self) -> syn::Result<Flag>;
20}
21
22impl<'a> ParseFlag for ParseStream<'a> {
23 fn parse_flag<T: Parse + Spanned>(&self) -> syn::Result<Flag> {
24 let keyword = self.parse::<T>()?;
25 if self.peek(Token![=]) {
26 self.parse::<Token![=]>()?;
27 Ok(Some(self.parse::<LitBool>()?))
28 } else {
29 Ok(Some(LitBool::new(true, keyword.span())))
30 }
31 }
32}
33
34pub(crate) fn flag_to_bool(flag: &Flag) -> bool {
36 flag.as_ref().map(LitBool::value).unwrap_or_default()
37}
38
39pub(crate) fn merge_flags<T: Token>(base: Flag, other: Flag) -> syn::Result<Flag> {
41 match (base, other) {
42 (Some(a), Some(b)) if a.value == b.value => Ok(Some(a)),
43 (Some(a), None) => Ok(Some(a)),
44 (None, Some(b)) => Ok(Some(b)),
45 (None, None) => Ok(None),
46 (Some(_), Some(b)) => Err(syn::Error::new_spanned(
47 b,
48 format_args!("conflicting `{}` arguments", T::display()),
49 )),
50 }
51}