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"
);
}
}