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
use super::type_app::*;

pub trait Functor : TypeAppGeneric
{
  fn fmap < A, B >
    ( fa: Applied < Self, A >,
      mapper: impl Fn (A) -> B,
    ) ->
      Applied < Self, B >
  where
    A: Send + 'static,
    B: Send + 'static,
  ;
}

pub trait Applicative
  : Functor
{
  fn apply < A, B, Func >
    ( fab : Applied < Self, Func >,
      fa : Applied < Self, A >
    ) ->
      Applied < Self, B >
  where
    Func : Fn (A) -> B,
    A: Send + 'static,
    B: Send + 'static,
  ;
}

pub trait Monad
  : Applicative
{
  fn bind < A, B >
    ( fa : Applied < Self, A >,
      cont : impl Fn (A) -> Applied < Self, B >
    ) ->
      Applied < Self, B >
  where
    A: Send + 'static,
    B: Send + 'static,
  ;
}

pub trait NaturalTransformation < F1, F2 >
where
  F1: TyCon,
  F2: TyCon,
{
  fn lift < A >
    ( fa: Applied < F1, A > )
    -> Applied < F2, A >
  where
    A: Send + 'static,
  ;
}