use quote::ToTokens;
use std::cell::RefCell;
use std::fmt::Display;
use std::thread;
#[derive(Debug, PartialEq, Eq, Hash, Clone, Copy, Default)]
pub(crate) enum ReflectType {
#[default]
Input,
Output,
}
pub(crate) struct ContextEncounters {
pub fields: Vec<(reflectapi_schema::TypeReference, syn::Type)>,
pub generics: Vec<(reflectapi_schema::TypeParameter, syn::Ident)>,
}
#[derive(Default)]
pub(crate) struct Context {
reflectapi_type: ReflectType,
encountered_type_fields: RefCell<Vec<(reflectapi_schema::TypeReference, syn::Type)>>,
encountered_type_generics: RefCell<Vec<(reflectapi_schema::TypeParameter, syn::Ident)>>,
errors: RefCell<Option<Vec<syn::Error>>>,
}
impl Context {
pub fn new(reflectapi_type: ReflectType) -> Self {
Context {
reflectapi_type,
encountered_type_fields: RefCell::new(Vec::new()),
encountered_type_generics: RefCell::new(Vec::new()),
errors: RefCell::new(Some(Vec::new())),
}
}
pub fn reflectapi_type(&self) -> ReflectType {
self.reflectapi_type
}
pub fn impl_error<A: ToTokens, T: Display>(&self, obj: A, msg: T) {
self.errors
.borrow_mut()
.as_mut()
.unwrap()
.push(syn::Error::new_spanned(obj.into_token_stream(), msg));
}
pub fn syn_error(&self, err: syn::Error) {
self.errors.borrow_mut().as_mut().unwrap().push(err);
}
pub fn check(self) -> syn::Result<ContextEncounters> {
let mut errors = self.errors.borrow_mut().take().unwrap().into_iter();
let mut combined = match errors.next() {
Some(first) => first,
None => {
return Ok(ContextEncounters {
fields: self.encountered_type_fields.replace(Vec::new()),
generics: self.encountered_type_generics.replace(Vec::new()),
})
}
};
for rest in errors {
combined.combine(rest);
}
Err(combined)
}
pub fn encountered_field_type(
&self,
type_ref: reflectapi_schema::TypeReference,
ty: syn::Type,
) {
self.encountered_type_fields
.borrow_mut()
.push((type_ref, ty));
}
pub fn encountered_generic_type(
&self,
type_param: reflectapi_schema::TypeParameter,
ty: syn::Ident,
) {
self.encountered_type_generics
.borrow_mut()
.push((type_param, ty));
}
}
impl Drop for Context {
fn drop(&mut self) {
if !thread::panicking() && self.errors.borrow().is_some() {
panic!("forgot to check for errors");
}
}
}