bon-macros 3.9.2

This is a proc-macro crate that is supposed to be a private implementation detail of the `bon` crate
Documentation
use crate::parsing::{ItemSigConfig, ItemSigConfigParsing};
use crate::util::prelude::*;
use darling::FromMeta;

#[derive(Debug, Clone, Default)]
pub(crate) struct GenericsConfig {
    pub(crate) setters: Option<ItemSigConfig<String>>,
}

impl FromMeta for GenericsConfig {
    fn from_meta(meta: &syn::Meta) -> Result<Self> {
        meta.require_list()?.require_parens_delim()?;

        #[derive(FromMeta)]
        struct Parsed {
            #[darling(default, with = parse_setters_config, map = Some)]
            setters: Option<ItemSigConfig<String>>,
        }

        let parsed = Parsed::from_meta(meta)?;

        Ok(Self {
            setters: parsed.setters,
        })
    }
}

fn parse_setters_config(meta: &syn::Meta) -> Result<ItemSigConfig<String>> {
    if !cfg!(feature = "experimental-generics-setters") {
        bail!(
            meta,
            "🔬 `generics(setters(...))` attribute is experimental and requires \
             \"experimental-generics-setters\" cargo feature to be enabled",
        );
    }

    const DOCS_CONTEXT: &str = "builder struct's impl block";
    let config: ItemSigConfig<String> =
        ItemSigConfigParsing::new(meta, Some(DOCS_CONTEXT)).parse()?;

    // Validate that name is provided and contains the placeholder
    let name_pattern = config.name.as_ref().ok_or_else(|| {
        err!(
            meta,
            "`name` parameter is required for `generics(setters(...))`; \
             specify a pattern like `name = \"conv_{{}}\"` where `{{}}` will be \
             replaced with the snake_case name of each generic parameter"
        )
    })?;

    if !name_pattern.value.contains("{}") {
        bail!(
            &name_pattern.key,
            "the `name` pattern must contain the `{{}}` placeholder, \
             which will be replaced with the snake_case name of each generic parameter"
        );
    }

    Ok(config)
}