diff options
author | HampusM <hampus@hampusmat.com> | 2022-08-02 14:31:31 +0200 |
---|---|---|
committer | HampusM <hampus@hampusmat.com> | 2022-08-02 14:31:31 +0200 |
commit | 826592eac2601e9fcd5aabb17482b4816ed7ab88 (patch) | |
tree | 3c3ba5436ca76e0738e4cc8deefef3025fd5d4bd /macros/src | |
parent | 163cd3cedd398f5676edbcb3249dd958d3e97aca (diff) |
feat: add detection and prevention of circular dependencies
Diffstat (limited to 'macros/src')
-rw-r--r-- | macros/src/injectable_impl.rs | 97 |
1 files changed, 61 insertions, 36 deletions
diff --git a/macros/src/injectable_impl.rs b/macros/src/injectable_impl.rs index 227a8c6..b24749c 100644 --- a/macros/src/injectable_impl.rs +++ b/macros/src/injectable_impl.rs @@ -49,7 +49,14 @@ impl InjectableImpl let di_container_var: Ident = parse_str(DI_CONTAINER_VAR_NAME).unwrap(); - let get_dependencies = Self::_create_get_dependencies(dependency_types); + let get_dep_method_names = Self::_create_get_dep_method_names(dependency_types); + + let get_dependencies = get_dep_method_names.iter().map(|get_dep_method_name| { + parse_str::<ExprMethodCall>( + format!("{}(dependency_history.clone())", get_dep_method_name).as_str(), + ) + .unwrap() + }); let maybe_doc_hidden = if no_doc_hidden { quote! {} @@ -65,20 +72,52 @@ impl InjectableImpl #maybe_doc_hidden impl #generics syrette::interfaces::injectable::Injectable for #self_type { fn resolve( - #di_container_var: &syrette::DIContainer + #di_container_var: &syrette::DIContainer, + mut dependency_history: Vec<&'static str>, ) -> syrette::libs::error_stack::Result< syrette::ptr::TransientPtr<Self>, syrette::errors::injectable::ResolveError> { - use syrette::libs::error_stack::ResultExt; + use syrette::libs::error_stack::{ResultExt, report}; + use syrette::errors::injectable::ResolveError; + + let self_type_name = std::any::type_name::<#self_type>(); + + if dependency_history.contains(&self_type_name) { + dependency_history.push(self_type_name); + + let dependency_trace = dependency_history + .iter() + .map(|dep| { + if dep == &self_type_name { + format!("\x1b[1m{}\x1b[22m", dep) + } else { + dep.to_string() + } + }) + .collect::<Vec<_>>() + .join(" -> "); + + return Err( + report!(ResolveError) + .attach_printable( + format!( + "Detected circular dependencies. {}", + dependency_trace.clone(), + ) + ) + ); + } + + dependency_history.push(self_type_name); return Ok(syrette::ptr::TransientPtr::new(Self::new( #(#get_dependencies - .change_context(syrette::errors::injectable::ResolveError) + .change_context(ResolveError) .attach_printable( format!( "Unable to resolve a dependency of {}", - std::any::type_name::<#self_type>() + self_type_name ) )? ),* @@ -88,48 +127,34 @@ impl InjectableImpl } } - fn _create_get_dependencies( - dependency_types: &[DependencyType], - ) -> Vec<ExprMethodCall> + fn _create_get_dep_method_names(dependency_types: &[DependencyType]) -> Vec<String> { dependency_types .iter() .filter_map(|dep_type| match &dep_type.interface { - Type::TraitObject(dep_type_trait) => Some( - parse_str( - format!( - "{}.get{}::<{}>()", - DI_CONTAINER_VAR_NAME, - if dep_type.ptr == "SingletonPtr" { - "_singleton" - } else { - "" - }, - dep_type_trait.to_token_stream() - ) - .as_str(), - ) - .unwrap(), - ), + Type::TraitObject(dep_type_trait) => Some(format!( + "{}.get_{}::<{}>", + DI_CONTAINER_VAR_NAME, + if dep_type.ptr == "SingletonPtr" { + "singleton_with_history" + } else { + "with_history" + }, + dep_type_trait.to_token_stream() + )), Type::Path(dep_type_path) => { let dep_type_path_str = Self::_path_to_string(&dep_type_path.path); let get_method_name = if dep_type_path_str.ends_with("Factory") { - "get_factory" + "factory" } else { - "get" + "with_history" }; - Some( - parse_str( - format!( - "{}.{}::<{}>()", - DI_CONTAINER_VAR_NAME, get_method_name, dep_type_path_str - ) - .as_str(), - ) - .unwrap(), - ) + Some(format!( + "get_{}.{}::<{}>", + DI_CONTAINER_VAR_NAME, get_method_name, dep_type_path_str + )) } &_ => None, }) |