use crate::{ attr::{fold_attrs, Attr}, error::compile_error_at, fields_parse::ParserType, rename_rules::RenameRule, Result, }; use proc_macro::TokenStream; use proc_macro2::Span; use syn::{ parse::{ParseStream, Peek}, Attribute, Token, }; /// All attributes that can be used for `derive(BotCommands)` pub(crate) struct CommandAttrs { pub prefix: Option<(String, Span)>, /// The bool is true if the description contains a doc comment pub description: Option<(String, bool, Span)>, pub rename_rule: Option<(RenameRule, Span)>, pub rename: Option<(String, Span)>, pub parser: Option<(ParserType, Span)>, pub separator: Option<(String, Span)>, pub command_separator: Option<(String, Span)>, pub hide: Option<((), Span)>, } /// A single k/v attribute for `BotCommands` derive macro. /// /// For example: /// ```text /// #[command(prefix = "!", rename_rule = "snake_case")] /// /^^^^^^^^^^^^ ^^^^^^^^^^^^^^^^^^^^^^^^^^---- CommandAttr { kind: RenameRule(SnakeCase) } /// | /// CommandAttr { kind: Prefix("!") } /// ``` struct CommandAttr { kind: CommandAttrKind, sp: Span, } /// Kind of [`CommandAttr`]. enum CommandAttrKind { Prefix(String), /// Description of the command. and if its doc comment or not Description(String, bool), RenameRule(RenameRule), Rename(String), ParseWith(ParserType), Separator(String), CommandSeparator(String), Hide, } impl CommandAttrs { pub fn from_attributes(attributes: &[Attribute]) -> Result { use CommandAttrKind::*; fold_attrs( attributes, |attr| is_command_attribute(attr) || is_doc_comment(attr), CommandAttr::parse, Self { prefix: None, description: None, rename_rule: None, rename: None, parser: None, separator: None, command_separator: None, hide: None, }, |mut this, attr| { fn insert(opt: &mut Option<(T, Span)>, x: T, sp: Span) -> Result<()> { match opt { slot @ None => { *slot = Some((x, sp)); Ok(()) } Some(_) => Err(compile_error_at("duplicate attribute", sp)), } } fn join_string(opt: &mut Option<(String, bool, Span)>, new_str: &str, sp: Span) { match opt { slot @ None => { *slot = Some((new_str.to_owned(), false, sp)); } Some((old_str, ..)) => { *old_str = format!("{old_str}\n{new_str}"); } } } match attr.kind { Prefix(p) => insert(&mut this.prefix, p, attr.sp), Description(d, is_doc) => { join_string( &mut this.description, // Sometimes doc comments include a space before them, this removes it d.strip_prefix(' ').unwrap_or(&d), attr.sp, ); if is_doc { if let Some((_, is_doc, _)) = &mut this.description { *is_doc = true; } } Ok(()) } RenameRule(r) => insert(&mut this.rename_rule, r, attr.sp), Rename(r) => insert(&mut this.rename, r, attr.sp), ParseWith(p) => insert(&mut this.parser, p, attr.sp), Separator(s) => insert(&mut this.separator, s, attr.sp), CommandSeparator(s) => insert(&mut this.command_separator, s, attr.sp), Hide => insert(&mut this.hide, (), attr.sp), }?; Ok(this) }, ) } } impl CommandAttr { fn parse(attr: Attr) -> Result { use CommandAttrKind::*; let sp = attr.span(); let Attr { mut key, value } = attr; let outermost_key = key.pop().unwrap(); // `Attr`'s invariants ensure `key.len() > 0` let kind = match &*outermost_key.to_string() { "doc" => { if let Some(unexpected_key) = key.last() { return Err(compile_error_at( "`doc` can't have nested attributes", unexpected_key.span(), )); } Description(value.expect_string()?, true) } "command" => { let Some(attr) = key.pop() else { return Err(compile_error_at( "expected an attribute name", outermost_key.span(), )) }; if let Some(unexpected_key) = key.last() { return Err(compile_error_at( &format!("{attr} can't have nested attributes"), unexpected_key.span(), )); } match &*attr.to_string() { "prefix" => Prefix(value.expect_string()?), "description" => Description(value.expect_string()?, false), "rename_rule" => { RenameRule(value.expect_string().and_then(|r| self::RenameRule::parse(&r))?) } "rename" => Rename(value.expect_string()?), "parse_with" => ParseWith(ParserType::parse(value)?), "separator" => Separator(value.expect_string()?), "command_separator" => CommandSeparator(value.expect_string()?), "hide" => value.expect_none("hide").map(|_| Hide)?, _ => { return Err(compile_error_at( "unexpected attribute name (expected one of `prefix`, `description`, \ `rename`, `parse_with`, `separator` and `hide`", attr.span(), )) } } } _ => { return Err(compile_error_at( "unexpected attribute (expected `command` or `doc`)", outermost_key.span(), )) } }; Ok(Self { kind, sp }) } } fn is_command_attribute(a: &Attribute) -> bool { matches!(a.path.get_ident(), Some(ident) if ident == "command") } fn is_doc_comment(a: &Attribute) -> bool { matches!(a.path.get_ident(), Some(ident) if ident == "doc" && peek_at_token_stream(a.tokens.clone().into(), Token![=])) } fn peek_at_token_stream(s: TokenStream, p: impl Peek) -> bool { // syn be fr challenge 2023 (impossible) use syn::parse::Parser; (|input: ParseStream<'_>| { let r = input.peek(p); _ = input.step(|_| Ok(((), syn::buffer::Cursor::empty()))); Ok(r) }) .parse(s) .unwrap() }