codama-attributes 0.9.1

Codama attributes definitions and parsers
Documentation
use crate::{utils::SetOnce, Attribute, CodamaAttribute, CodamaDirective};
use codama_errors::CodamaError;
use codama_syn_helpers::{extensions::*, Meta};

#[derive(Debug, PartialEq, Default, Clone)]
pub struct ErrorDirective {
    pub code: Option<usize>,
    pub message: Option<String>,
}

impl ErrorDirective {
    pub fn is_empty(&self) -> bool {
        self.code.is_none() && self.message.is_none()
    }
}

impl ErrorDirective {
    pub fn parse(meta: &Meta) -> syn::Result<Self> {
        let pl = meta.assert_directive("error")?.as_path_list()?;
        let mut code = SetOnce::<usize>::new("code");
        let mut message = SetOnce::<String>::new("message");
        pl.each(|ref meta| match meta.path_str().as_str() {
            "code" => code.set(meta.as_value()?.as_expr()?.as_unsigned_integer()?, meta),
            "message" => message.set(meta.as_value()?.as_expr()?.as_string()?, meta),
            _ => {
                if let Ok(expr) = meta.as_expr() {
                    if let Ok(value) = expr.as_unsigned_integer() {
                        return code.set(value, meta);
                    } else if let Ok(value) = expr.as_string() {
                        return message.set(value, meta);
                    }
                }
                Err(meta.error("unrecognized attribute"))
            }
        })?;
        let directive = Self {
            code: code.option(),
            message: message.option(),
        };
        if directive.is_empty() {
            return Err(pl.error("expected at least one `code` or `message` attribute"));
        }
        Ok(directive)
    }
}

impl<'a> TryFrom<&'a CodamaAttribute<'a>> for &'a ErrorDirective {
    type Error = CodamaError;

    fn try_from(attribute: &'a CodamaAttribute) -> Result<Self, Self::Error> {
        match attribute.directive.as_ref() {
            CodamaDirective::Error(ref a) => Ok(a),
            _ => Err(CodamaError::InvalidCodamaDirective {
                expected: "error".to_string(),
                actual: attribute.directive.name().to_string(),
            }),
        }
    }
}

impl<'a> TryFrom<&'a Attribute<'a>> for &'a ErrorDirective {
    type Error = CodamaError;

    fn try_from(attribute: &'a Attribute) -> Result<Self, Self::Error> {
        <&CodamaAttribute>::try_from(attribute)?.try_into()
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[test]
    fn ok() {
        let meta: Meta = syn::parse_quote! { error(42, "my message") };
        let directive = ErrorDirective::parse(&meta).unwrap();
        assert_eq!(
            directive,
            ErrorDirective {
                code: Some(42),
                message: Some("my message".to_string()),
            }
        );
    }

    #[test]
    fn ok_with_explicit_labels() {
        let meta: Meta = syn::parse_quote! { error(code = 42, message = "my message") };
        let directive = ErrorDirective::parse(&meta).unwrap();
        assert_eq!(
            directive,
            ErrorDirective {
                code: Some(42),
                message: Some("my message".to_string()),
            }
        );
    }

    #[test]
    fn fail_if_nothing_is_provided() {
        let meta: Meta = syn::parse_quote! { error() };
        let error = ErrorDirective::parse(&meta).unwrap_err();
        assert_eq!(
            error.to_string(),
            "expected at least one `code` or `message` attribute"
        );
    }
}