use proc_macro2::TokenStream;
use quote::quote;
use crate::context::{BaseModule, Context};
use crate::objects;
use crate::types::objects::{ErrorDefinition, ObjectDefinition};
pub fn error_object_definition(def: &ErrorDefinition) -> ObjectDefinition {
ObjectDefinition::builder()
.type_name(def.error_name().clone())
.fields(def.safe_args().iter().chain(def.unsafe_args()).cloned())
.docs(def.docs().cloned())
.build()
}
pub fn generate(ctx: &Context, def: &ErrorDefinition) -> TokenStream {
let object = error_object_definition(def);
let object_def = objects::generate(ctx, BaseModule::Errors, &object);
let error_type = generate_error_type(ctx, def);
quote! {
#object_def
#error_type
}
}
fn generate_error_type(ctx: &Context, def: &ErrorDefinition) -> TokenStream {
let type_name = ctx.type_name(def.error_name().name());
let code = ctx.type_name(def.code().as_str());
let name = format!("{}:{}", def.namespace(), def.error_name().name());
let mut safe_args = def
.safe_args()
.iter()
.map(|f| &f.field_name().0)
.collect::<Vec<_>>();
safe_args.sort();
quote! {
impl conjure_error::ErrorType for #type_name {
#[inline]
fn code() -> conjure_error::ErrorCode {
conjure_error::ErrorCode::#code
}
#[inline]
fn name() -> &'static str {
#name
}
#[inline]
fn safe_args() -> &'static [&'static str] {
&[#(#safe_args,)*]
}
}
}
}