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::util::prelude::*;
use syn::visit::Visit;

impl super::FnInputCtx<'_> {
    pub(super) fn validate(&self) -> Result {
        if self.impl_ctx.is_none() {
            let explanation = "\
                which likely means the builder attribute was used inside of an \
                impl block; the impl block needs to be annotated with the #[bon] \
                attribute and the builder attribute must be spelled as #[builder] \
                without any additional path prefix, since it's used as a simple \
                inert config attribute for #[bon] in impl blocks; more info on \
                inert vs active attributes: \
                https://doc.rust-lang.org/reference/attributes.html#active-and-inert-attributes";

            if let Some(receiver) = &self.fn_item.orig.sig.receiver() {
                bail!(
                    &receiver.self_token,
                    "this function contains a `self` parameter {explanation}"
                );
            }

            let mut ctx = FindSelfReference::default();
            ctx.visit_item_fn(&self.fn_item.orig);
            if let Some(self_span) = ctx.self_span {
                bail!(
                    &self_span,
                    "this function contains a `Self` type reference {explanation}"
                );
            }
        }

        if let Some(const_) = &self.config.const_ {
            if self.fn_item.orig.sig.constness.is_none() {
                bail!(
                    &const_,
                    "#[builder(const)] requires the underlying function to be \
                    marked as `const fn`"
                );
            }
        }

        Ok(())
    }

    pub(crate) fn warnings(&self) -> TokenStream {
        // We used to emit some warnings here previously, but then that logic
        // was removed. The code for the `warnings()` method was preserved just
        // in case if we need to issue some new warnings again. However, it's not
        // critical. Feel free to eliminate this method if you feel like it.
        let _ = self;

        TokenStream::new()
    }
}

#[derive(Default)]
struct FindSelfReference {
    self_span: Option<Span>,
}

impl Visit<'_> for FindSelfReference {
    fn visit_item(&mut self, _: &syn::Item) {
        // Don't recurse into nested items. We are interested in the reference
        // to `Self` on the current item level
    }

    fn visit_path(&mut self, path: &syn::Path) {
        if self.self_span.is_some() {
            return;
        }
        syn::visit::visit_path(self, path);

        let first_segment = match path.segments.first() {
            Some(first_segment) => first_segment,
            _ => return,
        };

        if first_segment.ident == "Self" {
            self.self_span = Some(first_segment.ident.span());
        }
    }
}