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
use std::sync::Arc;
use async_trait::async_trait;
use crate::resolver::ResolverRef;
use crate::error::Error;
use crate::helpers::BoxAny;

pub type ProvideResult<T> = anyhow::Result<T>;

#[async_trait]
pub trait Provider: Send + Sync + 'static {
  type Ref: Resolvable;
  async fn provide(&self, resolver: &ResolverRef) -> ProvideResult<Self::Ref>;
}

#[async_trait]
pub(crate) trait ProviderObject: Send + Sync + 'static {
  async fn provide(&self, resolver: &ResolverRef) -> Result<BoxAny, Error>;
}

#[async_trait]
impl<T> ProviderObject for T
  where T: Provider
{
  async fn provide(&self, resolver: &ResolverRef) -> Result<BoxAny, Error> {
    Provider::provide(self, resolver).await
      .map(|v| Box::new(v) as BoxAny)
      .map_err(|err| {
        match err.downcast::<Error>() {
          Ok(err) => err,
          Err(err) => Error::Service(Arc::new(err))
        }
      })
  }
}

pub struct StaticProvider<T>(T);

impl<T> StaticProvider<T> {
  pub fn new(value: T) -> Self {
    StaticProvider(value)
  }
}

#[async_trait]
impl<T> Provider for StaticProvider<T>
where T: Resolvable
{
  type Ref = T;

  async fn provide(&self, _: &ResolverRef) -> ProvideResult<Self::Ref> {
    Ok(self.0.clone())
  }
}

mod private {
  pub trait Sealed {}

  impl<T> Sealed for T {}
}

pub trait Resolvable: private::Sealed + Clone + Send + Sync + 'static {}

impl<T> Resolvable for T
  where T: Clone + Send + Sync + ?Sized + 'static
{}