sig
  module type Basic =
    sig
      type _ t
      val map : ('-> 'b) -> 'a t -> 'b t
      val pure : '-> 'a t
      val ap : ('-> 'b) t -> (unit -> 'a t) -> 'b t
      val bind : ('-> 'b t) -> 'a t -> 'b t
    end
  module type Basic2 =
    sig
      type (_, _) t
      val map : ('-> 'b) -> ('p, 'a) t -> ('p, 'b) t
      val pure : '-> ('p, 'a) t
      val ap : ('p, '-> 'b) t -> (unit -> ('p, 'a) t) -> ('p, 'b) t
      val bind : ('-> ('p, 'b) t) -> ('p, 'a) t -> ('p, 'b) t
    end
  module type Basic3 =
    sig
      type (_, _, _) t
      val map : ('-> 'b) -> ('p, 'q, 'a) t -> ('p, 'q, 'b) t
      val pure : '-> ('p, 'q, 'a) t
      val ap :
        ('p, 'q, '-> 'b) t -> (unit -> ('p, 'q, 'a) t) -> ('p, 'q, 'b) t
      val bind : ('-> ('p, 'q, 'b) t) -> ('p, 'q, 'a) t -> ('p, 'q, 'b) t
    end
  module type S =
    sig
      type _ t
      val bind : ('-> 'b t) -> 'a t -> 'b t
      val pure : '-> 'a t
      val ap : ('-> 'b) t -> (unit -> 'a t) -> 'b t
      val map : ('-> 'b) -> 'a t -> 'b t
      val ( >|= ) : 'a t -> ('-> 'b) -> 'b t
      val replace : '-> 'b t -> 'a t
      val void : 'a t -> unit t
      val ap' : ('-> 'b) t -> 'a t -> 'b t
      val ( <*> ) : ('-> 'b) t -> 'a t -> 'b t
      val ( <~> ) : ('-> 'b) t -> (unit -> 'a t) -> 'b t
      val discard_left : 'a t -> (unit -> 'b t) -> 'b t
      val discard_right : 'a t -> (unit -> 'b t) -> 'a t
      val repeat : int -> 'a t -> 'a list t
      val repeat_ : int -> 'a t -> unit t
      val forever : 'a t -> 'b t
      val ( >>= ) : 'Monad.S.t -> ('-> 'Monad.S.t) -> 'Monad.S.t
      val join : 'Monad.S.t Monad.S.t -> 'Monad.S.t
      val mcompose :
        ('-> 'Monad.S.t) -> ('-> 'Monad.S.t) -> '-> 'Monad.S.t
    end
  module type S2 =
    sig
      type (_, _) t
      val bind : ('-> ('p, 'b) t) -> ('p, 'a) t -> ('p, 'b) t
      val pure : '-> ('p, 'a) t
      val ap : ('p, '-> 'b) t -> (unit -> ('p, 'a) t) -> ('p, 'b) t
      val map : ('-> 'b) -> ('p, 'a) t -> ('p, 'b) t
      val ( >|= ) : ('p, 'a) t -> ('-> 'b) -> ('p, 'b) t
      val replace : '-> ('p, 'b) t -> ('p, 'a) t
      val void : ('p, 'a) t -> ('p, unit) t
      val ap' : ('p, '-> 'b) t -> ('p, 'a) t -> ('p, 'b) t
      val ( <*> ) : ('p, '-> 'b) t -> ('p, 'a) t -> ('p, 'b) t
      val ( <~> ) : ('p, '-> 'b) t -> (unit -> ('p, 'a) t) -> ('p, 'b) t
      val discard_left : ('p, 'a) t -> (unit -> ('p, 'b) t) -> ('p, 'b) t
      val discard_right : ('p, 'a) t -> (unit -> ('p, 'b) t) -> ('p, 'a) t
      val repeat : int -> ('p, 'a) t -> ('p, 'a list) t
      val repeat_ : int -> ('p, 'a) t -> ('p, unit) t
      val forever : ('p, 'a) t -> ('p, 'b) t
      val ( >>= ) :
        ('p, 'a) Monad.S2.t ->
        ('-> ('p, 'b) Monad.S2.t) -> ('p, 'b) Monad.S2.t
      val join : ('p, ('p, 'a) Monad.S2.t) Monad.S2.t -> ('p, 'a) Monad.S2.t
      val mcompose :
        ('-> ('p, 'c) Monad.S2.t) ->
        ('-> ('p, 'b) Monad.S2.t) -> '-> ('p, 'c) Monad.S2.t
    end
  module type S3 =
    sig
      type (_, _, _) t
      val bind : ('-> ('p, 'q, 'b) t) -> ('p, 'q, 'a) t -> ('p, 'q, 'b) t
      val pure : '-> ('p, 'q, 'a) t
      val ap :
        ('p, 'q, '-> 'b) t -> (unit -> ('p, 'q, 'a) t) -> ('p, 'q, 'b) t
      val map : ('-> 'b) -> ('p, 'q, 'a) t -> ('p, 'q, 'b) t
      val ( >|= ) : ('p, 'q, 'a) t -> ('-> 'b) -> ('p, 'q, 'b) t
      val replace : '-> ('p, 'q, 'b) t -> ('p, 'q, 'a) t
      val void : ('p, 'q, 'a) t -> ('p, 'q, unit) t
      val ap' : ('p, 'q, '-> 'b) t -> ('p, 'q, 'a) t -> ('p, 'q, 'b) t
      val ( <*> ) : ('p, 'q, '-> 'b) t -> ('p, 'q, 'a) t -> ('p, 'q, 'b) t
      val ( <~> ) :
        ('p, 'q, '-> 'b) t -> (unit -> ('p, 'q, 'a) t) -> ('p, 'q, 'b) t
      val discard_left :
        ('p, 'q, 'a) t -> (unit -> ('p, 'q, 'b) t) -> ('p, 'q, 'b) t
      val discard_right :
        ('p, 'q, 'a) t -> (unit -> ('p, 'q, 'b) t) -> ('p, 'q, 'a) t
      val repeat : int -> ('p, 'q, 'a) t -> ('p, 'q, 'a list) t
      val repeat_ : int -> ('p, 'q, 'a) t -> ('p, 'q, unit) t
      val forever : ('p, 'q, 'a) t -> ('p, 'q, 'b) t
      val ( >>= ) :
        ('p, 'q, 'a) Monad.S3.t ->
        ('-> ('p, 'q, 'b) Monad.S3.t) -> ('p, 'q, 'b) Monad.S3.t
      val join :
        ('p, 'q, ('p, 'q, 'a) Monad.S3.t) Monad.S3.t ->
        ('p, 'q, 'a) Monad.S3.t
      val mcompose :
        ('-> ('p, 'q, 'c) Monad.S3.t) ->
        ('-> ('p, 'q, 'b) Monad.S3.t) -> '-> ('p, 'q, 'c) Monad.S3.t
    end
  module Make :
    functor (M : Basic->
      sig
        val bind : ('-> 'M.t) -> 'M.t -> 'M.t
        val pure : '-> 'M.t
        val ap : ('-> 'b) M.t -> (unit -> 'M.t) -> 'M.t
        val map : ('-> 'b) -> 'M.t -> 'M.t
        val ( >|= ) : 'M.t -> ('-> 'b) -> 'M.t
        val replace : '-> 'M.t -> 'M.t
        val void : 'M.t -> unit M.t
        val ap' : ('-> 'b) M.t -> 'M.t -> 'M.t
        val ( <*> ) : ('-> 'b) M.t -> 'M.t -> 'M.t
        val ( <~> ) : ('-> 'b) M.t -> (unit -> 'M.t) -> 'M.t
        val discard_left : 'M.t -> (unit -> 'M.t) -> 'M.t
        val discard_right : 'M.t -> (unit -> 'M.t) -> 'M.t
        val repeat : int -> 'M.t -> 'a list M.t
        val repeat_ : int -> 'M.t -> unit M.t
        val forever : 'M.t -> 'M.t
        val ( >>= ) : 'M.t -> ('-> 'M.t) -> 'M.t
        val join : 'M.t M.t -> 'M.t
        val mcompose : ('-> 'M.t) -> ('-> 'M.t) -> '-> 'M.t
      end
  module Make2 :
    functor (M : Basic2->
      sig
        val bind : ('-> ('p, 'b) M.t) -> ('p, 'a) M.t -> ('p, 'b) M.t
        val pure : '-> ('p, 'a) M.t
        val ap : ('p, '-> 'b) M.t -> (unit -> ('p, 'a) M.t) -> ('p, 'b) M.t
        val map : ('-> 'b) -> ('p, 'a) M.t -> ('p, 'b) M.t
        val ( >|= ) : ('p, 'a) M.t -> ('-> 'b) -> ('p, 'b) M.t
        val replace : '-> ('p, 'b) M.t -> ('p, 'a) M.t
        val void : ('p, 'a) M.t -> ('p, unit) M.t
        val ap' : ('p, '-> 'b) M.t -> ('p, 'a) M.t -> ('p, 'b) M.t
        val ( <*> ) : ('p, '-> 'b) M.t -> ('p, 'a) M.t -> ('p, 'b) M.t
        val ( <~> ) :
          ('p, '-> 'b) M.t -> (unit -> ('p, 'a) M.t) -> ('p, 'b) M.t
        val discard_left :
          ('p, 'a) M.t -> (unit -> ('p, 'b) M.t) -> ('p, 'b) M.t
        val discard_right :
          ('p, 'a) M.t -> (unit -> ('p, 'b) M.t) -> ('p, 'a) M.t
        val repeat : int -> ('p, 'a) M.t -> ('p, 'a list) M.t
        val repeat_ : int -> ('p, 'a) M.t -> ('p, unit) M.t
        val forever : ('p, 'a) M.t -> ('p, 'b) M.t
        val ( >>= ) : ('p, 'a) M.t -> ('-> ('p, 'b) M.t) -> ('p, 'b) M.t
        val join : ('p, ('p, 'a) M.t) M.t -> ('p, 'a) M.t
        val mcompose :
          ('-> ('p, 'c) M.t) -> ('-> ('p, 'b) M.t) -> '-> ('p, 'c) M.t
      end
  module Make3 :
    functor (M : Basic3->
      sig
        val bind :
          ('-> ('p, 'q, 'b) M.t) -> ('p, 'q, 'a) M.t -> ('p, 'q, 'b) M.t
        val pure : '-> ('p, 'q, 'a) M.t
        val ap :
          ('p, 'q, '-> 'b) M.t ->
          (unit -> ('p, 'q, 'a) M.t) -> ('p, 'q, 'b) M.t
        val map : ('-> 'b) -> ('p, 'q, 'a) M.t -> ('p, 'q, 'b) M.t
        val ( >|= ) : ('p, 'q, 'a) M.t -> ('-> 'b) -> ('p, 'q, 'b) M.t
        val replace : '-> ('p, 'q, 'b) M.t -> ('p, 'q, 'a) M.t
        val void : ('p, 'q, 'a) M.t -> ('p, 'q, unit) M.t
        val ap' :
          ('p, 'q, '-> 'b) M.t -> ('p, 'q, 'a) M.t -> ('p, 'q, 'b) M.t
        val ( <*> ) :
          ('p, 'q, '-> 'b) M.t -> ('p, 'q, 'a) M.t -> ('p, 'q, 'b) M.t
        val ( <~> ) :
          ('p, 'q, '-> 'b) M.t ->
          (unit -> ('p, 'q, 'a) M.t) -> ('p, 'q, 'b) M.t
        val discard_left :
          ('p, 'q, 'a) M.t -> (unit -> ('p, 'q, 'b) M.t) -> ('p, 'q, 'b) M.t
        val discard_right :
          ('p, 'q, 'a) M.t -> (unit -> ('p, 'q, 'b) M.t) -> ('p, 'q, 'a) M.t
        val repeat : int -> ('p, 'q, 'a) M.t -> ('p, 'q, 'a list) M.t
        val repeat_ : int -> ('p, 'q, 'a) M.t -> ('p, 'q, unit) M.t
        val forever : ('p, 'q, 'a) M.t -> ('p, 'q, 'b) M.t
        val ( >>= ) :
          ('p, 'q, 'a) M.t -> ('-> ('p, 'q, 'b) M.t) -> ('p, 'q, 'b) M.t
        val join : ('p, 'q, ('p, 'q, 'a) M.t) M.t -> ('p, 'q, 'a) M.t
        val mcompose :
          ('-> ('p, 'q, 'c) M.t) ->
          ('-> ('p, 'q, 'b) M.t) -> '-> ('p, 'q, 'c) M.t
      end
end