//! Originally from Intertrait by CodeChain //! //! //! //! //! Licensed under either of //! //! Apache License, Version 2.0 (LICENSE-APACHE or ) //! MIT license (LICENSE-MIT or ) //! //! at your option. use std::any::type_name; use std::sync::Arc; use crate::libs::intertrait::cast::error::CastError; use crate::libs::intertrait::{get_caster, CastFromSync}; pub trait CastArc { /// Casts an `Arc` for this trait into that for type `OtherTrait`. fn cast( self: Arc, ) -> Result, CastError>; } /// A blanket implementation of `CastArc` for traits extending `CastFrom`, `Sync`, and /// `Send`. impl CastArc for CastFromSelf { fn cast( self: Arc, ) -> Result, CastError> { let caster = get_caster::((*self).type_id()) .map_err(CastError::GetCasterFailed)?; let cast_arc = caster .opt_cast_arc .ok_or(CastError::NotArcCastable(type_name::()))?; cast_arc(self.arc_any()).map_err(|err| CastError::CastFailed { source: err, from: type_name::(), to: type_name::(), }) } } #[cfg(test)] mod tests { use std::any::Any; use std::fmt::{Debug, Display}; use std::sync::Arc; use super::*; use crate::test_utils::subjects; #[test] fn can_cast_arc() { let concrete_ninja = Arc::new(subjects::Ninja); let abstract_ninja: Arc = concrete_ninja; let debug_ninja_result = abstract_ninja.cast::(); assert!(debug_ninja_result.is_ok()); } #[test] fn cannot_cast_arc_wrong() { let concrete_ninja = Arc::new(subjects::Ninja); let abstract_ninja: Arc = concrete_ninja; let display_ninja_result = abstract_ninja.cast::(); assert!(matches!( display_ninja_result, Err(CastError::GetCasterFailed(_)) )); } #[test] fn can_cast_arc_from_any() { let concrete_ninja = Arc::new(subjects::Ninja); let any_ninja: Arc = concrete_ninja; let debug_ninja_result = any_ninja.cast::(); assert!(debug_ninja_result.is_ok()); } }