1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
|
//! Originally from Intertrait by CodeChain
//!
//! <https://github.com/CodeChain-io/intertrait>
//! <https://crates.io/crates/intertrait/0.2.2>
//!
//! Licensed under either of
//!
//! Apache License, Version 2.0 (LICENSE-APACHE or <http://www.apache.org/licenses/LICENSE-2.0>)
//! MIT license (LICENSE-MIT or <http://opensource.org/licenses/MIT>)
//!
//! 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<OtherTrait: ?Sized + 'static>(
self: Arc<Self>,
) -> Result<Arc<OtherTrait>, CastError>;
}
/// A blanket implementation of `CastArc` for traits extending `CastFrom`, `Sync`, and
/// `Send`.
impl<CastFromSelf: ?Sized + CastFromSync> CastArc for CastFromSelf
{
fn cast<OtherTrait: ?Sized + 'static>(
self: Arc<Self>,
) -> Result<Arc<OtherTrait>, CastError>
{
let caster = get_caster::<OtherTrait>((*self).type_id())
.map_err(CastError::GetCasterFailed)?;
let cast_arc = caster
.opt_cast_arc
.ok_or(CastError::NotArcCastable(type_name::<OtherTrait>()))?;
cast_arc(self.arc_any()).map_err(|err| CastError::CastFailed {
source: err,
from: type_name::<Self>(),
to: type_name::<OtherTrait>(),
})
}
}
#[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<dyn subjects::INinja> = concrete_ninja;
let debug_ninja_result = abstract_ninja.cast::<dyn Debug>();
assert!(debug_ninja_result.is_ok());
}
#[test]
fn cannot_cast_arc_wrong()
{
let concrete_ninja = Arc::new(subjects::Ninja);
let abstract_ninja: Arc<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_arc_from_any()
{
let concrete_ninja = Arc::new(subjects::Ninja);
let any_ninja: Arc<dyn Any + Send + Sync> = concrete_ninja;
let debug_ninja_result = any_ninja.cast::<dyn Debug>();
assert!(debug_ninja_result.is_ok());
}
}
|