aboutsummaryrefslogtreecommitdiff
path: root/src/libs/intertrait/cast/arc.rs
blob: 33d84d2bcf0242bc699630319be8c21b28defa77 (plain)
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
//! 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_or_else(
            || {
                Err(CastError::CastFailed {
                    from: type_name::<CastFromSelf>(),
                    to: type_name::<OtherTrait>(),
                })
            },
            Ok,
        )?;

        match caster.opt_cast_arc {
            Some(cast_arc) => Ok(cast_arc(self.arc_any())),
            None => Err(CastError::NotArcCastable(type_name::<OtherTrait>())),
        }
    }
}