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
    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
    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
    end
  module type S =
    sig
      type _ 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) Applicative.S.t ->
        'Applicative.S.t -> 'Applicative.S.t
      val ( <*> ) :
        ('-> 'b) Applicative.S.t ->
        'Applicative.S.t -> 'Applicative.S.t
      val ( <~> ) :
        ('-> 'b) Applicative.S.t ->
        (unit -> 'Applicative.S.t) -> 'Applicative.S.t
      val discard_left :
        'Applicative.S.t ->
        (unit -> 'Applicative.S.t) -> 'Applicative.S.t
      val discard_right :
        'Applicative.S.t ->
        (unit -> 'Applicative.S.t) -> 'Applicative.S.t
      val repeat : int -> 'Applicative.S.t -> 'a list Applicative.S.t
      val repeat_ : int -> 'Applicative.S.t -> unit Applicative.S.t
      val forever : 'Applicative.S.t -> 'Applicative.S.t
    end
  module type S2 =
    sig
      type (_, _) 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) Applicative.S2.t ->
        ('p, 'a) Applicative.S2.t -> ('p, 'b) Applicative.S2.t
      val ( <*> ) :
        ('p, '-> 'b) Applicative.S2.t ->
        ('p, 'a) Applicative.S2.t -> ('p, 'b) Applicative.S2.t
      val ( <~> ) :
        ('p, '-> 'b) Applicative.S2.t ->
        (unit -> ('p, 'a) Applicative.S2.t) -> ('p, 'b) Applicative.S2.t
      val discard_left :
        ('p, 'a) Applicative.S2.t ->
        (unit -> ('p, 'b) Applicative.S2.t) -> ('p, 'b) Applicative.S2.t
      val discard_right :
        ('p, 'a) Applicative.S2.t ->
        (unit -> ('p, 'b) Applicative.S2.t) -> ('p, 'a) Applicative.S2.t
      val repeat :
        int -> ('p, 'a) Applicative.S2.t -> ('p, 'a list) Applicative.S2.t
      val repeat_ :
        int -> ('p, 'a) Applicative.S2.t -> ('p, unit) Applicative.S2.t
      val forever : ('p, 'a) Applicative.S2.t -> ('p, 'b) Applicative.S2.t
    end
  module type S3 =
    sig
      type (_, _, _) 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) Applicative.S3.t ->
        ('p, 'q, 'a) Applicative.S3.t -> ('p, 'q, 'b) Applicative.S3.t
      val ( <*> ) :
        ('p, 'q, '-> 'b) Applicative.S3.t ->
        ('p, 'q, 'a) Applicative.S3.t -> ('p, 'q, 'b) Applicative.S3.t
      val ( <~> ) :
        ('p, 'q, '-> 'b) Applicative.S3.t ->
        (unit -> ('p, 'q, 'a) Applicative.S3.t) ->
        ('p, 'q, 'b) Applicative.S3.t
      val discard_left :
        ('p, 'q, 'a) Applicative.S3.t ->
        (unit -> ('p, 'q, 'b) Applicative.S3.t) ->
        ('p, 'q, 'b) Applicative.S3.t
      val discard_right :
        ('p, 'q, 'a) Applicative.S3.t ->
        (unit -> ('p, 'q, 'b) Applicative.S3.t) ->
        ('p, 'q, 'a) Applicative.S3.t
      val repeat :
        int ->
        ('p, 'q, 'a) Applicative.S3.t -> ('p, 'q, 'a list) Applicative.S3.t
      val repeat_ :
        int ->
        ('p, 'q, 'a) Applicative.S3.t -> ('p, 'q, unit) Applicative.S3.t
      val forever :
        ('p, 'q, 'a) Applicative.S3.t -> ('p, 'q, 'b) Applicative.S3.t
    end
  module Make :
    functor (A : Basic->
      sig
        val pure : '-> 'A.t
        val ap : ('-> 'b) A.t -> (unit -> 'A.t) -> 'A.t
        val map : ('-> 'b) -> 'A.t -> 'A.t
        val ( >|= ) : 'A.t -> ('-> 'b) -> 'A.t
        val replace : '-> 'A.t -> 'A.t
        val void : 'A.t -> unit A.t
        val ap' : ('-> 'b) A.t -> 'A.t -> 'A.t
        val ( <*> ) : ('-> 'b) A.t -> 'A.t -> 'A.t
        val ( <~> ) : ('-> 'b) A.t -> (unit -> 'A.t) -> 'A.t
        val discard_left : 'A.t -> (unit -> 'A.t) -> 'A.t
        val discard_right : 'A.t -> (unit -> 'A.t) -> 'A.t
        val repeat : int -> 'A.t -> 'a list A.t
        val repeat_ : int -> 'A.t -> unit A.t
        val forever : 'A.t -> 'A.t
      end
  module Make2 :
    functor (A : Basic2->
      sig
        val pure : '-> ('p, 'a) A.t
        val ap : ('p, '-> 'b) A.t -> (unit -> ('p, 'a) A.t) -> ('p, 'b) A.t
        val map : ('-> 'b) -> ('p, 'a) A.t -> ('p, 'b) A.t
        val ( >|= ) : ('p, 'a) A.t -> ('-> 'b) -> ('p, 'b) A.t
        val replace : '-> ('p, 'b) A.t -> ('p, 'a) A.t
        val void : ('p, 'a) A.t -> ('p, unit) A.t
        val ap' : ('p, '-> 'b) A.t -> ('p, 'a) A.t -> ('p, 'b) A.t
        val ( <*> ) : ('p, '-> 'b) A.t -> ('p, 'a) A.t -> ('p, 'b) A.t
        val ( <~> ) :
          ('p, '-> 'b) A.t -> (unit -> ('p, 'a) A.t) -> ('p, 'b) A.t
        val discard_left :
          ('p, 'a) A.t -> (unit -> ('p, 'b) A.t) -> ('p, 'b) A.t
        val discard_right :
          ('p, 'a) A.t -> (unit -> ('p, 'b) A.t) -> ('p, 'a) A.t
        val repeat : int -> ('p, 'a) A.t -> ('p, 'a list) A.t
        val repeat_ : int -> ('p, 'a) A.t -> ('p, unit) A.t
        val forever : ('p, 'a) A.t -> ('p, 'b) A.t
      end
  module Make3 :
    functor (A : Basic3->
      sig
        val pure : '-> ('p, 'q, 'a) A.t
        val ap :
          ('p, 'q, '-> 'b) A.t ->
          (unit -> ('p, 'q, 'a) A.t) -> ('p, 'q, 'b) A.t
        val map : ('-> 'b) -> ('p, 'q, 'a) A.t -> ('p, 'q, 'b) A.t
        val ( >|= ) : ('p, 'q, 'a) A.t -> ('-> 'b) -> ('p, 'q, 'b) A.t
        val replace : '-> ('p, 'q, 'b) A.t -> ('p, 'q, 'a) A.t
        val void : ('p, 'q, 'a) A.t -> ('p, 'q, unit) A.t
        val ap' :
          ('p, 'q, '-> 'b) A.t -> ('p, 'q, 'a) A.t -> ('p, 'q, 'b) A.t
        val ( <*> ) :
          ('p, 'q, '-> 'b) A.t -> ('p, 'q, 'a) A.t -> ('p, 'q, 'b) A.t
        val ( <~> ) :
          ('p, 'q, '-> 'b) A.t ->
          (unit -> ('p, 'q, 'a) A.t) -> ('p, 'q, 'b) A.t
        val discard_left :
          ('p, 'q, 'a) A.t -> (unit -> ('p, 'q, 'b) A.t) -> ('p, 'q, 'b) A.t
        val discard_right :
          ('p, 'q, 'a) A.t -> (unit -> ('p, 'q, 'b) A.t) -> ('p, 'q, 'a) A.t
        val repeat : int -> ('p, 'q, 'a) A.t -> ('p, 'q, 'a list) A.t
        val repeat_ : int -> ('p, 'q, 'a) A.t -> ('p, 'q, unit) A.t
        val forever : ('p, 'q, 'a) A.t -> ('p, 'q, 'b) A.t
      end
end