use std::any::type_name;
use std::rc::Rc;
use crate::private::cast::error::CastError;
use crate::private::cast::{get_caster, CastFrom};
pub trait CastRc
{
fn cast<Dest: ?Sized + 'static>(self: Rc<Self>) -> Result<Rc<Dest>, CastError>;
}
impl<CastFromSelf: ?Sized + CastFrom> CastRc for CastFromSelf
{
fn cast<Dest: ?Sized + 'static>(self: Rc<Self>) -> Result<Rc<Dest>, CastError>
{
let caster =
get_caster::<Dest>((*self).type_id()).map_err(CastError::GetCasterFailed)?;
(caster.cast_rc)(self.rc_any()).map_err(|err| CastError::CastFailed {
source: err,
from: type_name::<Self>(),
to: type_name::<Dest>(),
})
}
}
#[cfg(test)]
mod tests
{
use std::any::Any;
use std::fmt::{Debug, Display};
use super::*;
use crate::test_utils::subjects;
#[test]
fn can_cast_rc()
{
let concrete_ninja = Rc::new(subjects::Ninja);
let abstract_ninja: Rc<dyn subjects::INinja> = concrete_ninja;
let debug_ninja_result = abstract_ninja.cast::<dyn Debug>();
assert!(debug_ninja_result.is_ok());
}
#[test]
fn cannot_cast_rc_wrong()
{
let concrete_ninja = Rc::new(subjects::Ninja);
let abstract_ninja: Rc<dyn subjects::INinja> = concrete_ninja;
let display_ninja_result = abstract_ninja.cast::<dyn Display>();
assert!(matches!(
display_ninja_result,
Err(CastError::GetCasterFailed(_))
));
}
#[test]
fn can_cast_rc_from_any()
{
let concrete_ninja = Rc::new(subjects::Ninja);
let any_ninja: Rc<dyn Any> = concrete_ninja;
let debug_ninja_result = any_ninja.cast::<dyn Debug>();
assert!(debug_ninja_result.is_ok());
}
}