Library UniMath.SubstitutionSystems.MonadsMultiSorted_alt

Monads in sort,C
Written by Anders Mörtberg, 2021 (adapted from MonadsMultiSorted.v)
Require Export UniMath.Tactics.EnsureStructuredProofs.
Require Import UniMath.Foundations.PartA.
Require Import UniMath.Foundations.PartD.
Require Import UniMath.Foundations.Sets.

Require Import UniMath.MoreFoundations.Tactics.

Require Import UniMath.Combinatorics.Lists.

Require Import UniMath.CategoryTheory.Core.Categories.
Require Import UniMath.CategoryTheory.Core.NaturalTransformations.
Require Import UniMath.CategoryTheory.Core.Functors.
Require Import UniMath.CategoryTheory.FunctorCategory.
Require Import UniMath.CategoryTheory.whiskering.
Require Import UniMath.CategoryTheory.Limits.Graphs.Colimits.
Require Import UniMath.CategoryTheory.Limits.BinProducts.
Require Import UniMath.CategoryTheory.Limits.Products.
Require Import UniMath.CategoryTheory.Limits.BinCoproducts.
Require Import UniMath.CategoryTheory.Limits.Coproducts.
Require Import UniMath.CategoryTheory.Limits.Terminal.
Require Import UniMath.CategoryTheory.Limits.Initial.
Require Import UniMath.CategoryTheory.FunctorAlgebras.
Require Import UniMath.CategoryTheory.exponentials.
Require Import UniMath.CategoryTheory.Adjunctions.Core.
Require Import UniMath.CategoryTheory.Chains.All.
Require Import UniMath.CategoryTheory.Monads.Monads.

Require Import UniMath.CategoryTheory.Categories.StandardCategories.
Require Import UniMath.CategoryTheory.Groupoids.

Require UniMath.SubstitutionSystems.SortIndexing.

Local Open Scope cat.

Section MonadInSortToC.

Context (sort : hSet) (Hsort : isofhlevel 3 sort) (C : category) (BC : BinCoproducts C) (TC : Terminal C).

Let sortToC : category := SortIndexing.sortToC sort Hsort C.

Local Definition BCsortToC : BinCoproducts sortToC := SortIndexing.BCsortToC sort Hsort _ BC.

Local Definition TsortToC : Terminal sortToC := SortIndexing.TsortToC sort Hsort _ TC.

Local Notation "a ⊕ b" := (BinCoproductObject (BCsortToC a b)).
Local Notation "1" := (TerminalObject TsortToC).

Context {M : Monad sortToC}.

Definition sortToC_fun {X Y : sortToC} (f : t, Cpr1 X t,pr1 Y t) : sortToCX,Y :=
  nat_trans_functor_path_pregroupoid f.

Definition bind_fun {X Y : sortToC} (f : t, Cpr1 X t,pr1 (M Y) t) :
   t, Cpr1 (M X) t,pr1 (M Y) t :=
    λ t, pr1 (bind (sortToC_fun f)) t.

Definition η_fun {X : sortToC} (t : sort) : Cpr1 X t,pr1 (M X) t :=
  pr1 (η M X) t.

Lemma η_bind_fun {X Y : sortToC} (f : t, Cpr1 X t,pr1 (M Y) t) (t : sort) :
  η_fun t · bind_fun f t = f t.
Proof.
exact (nat_trans_eq_pointwise (η_bind (sortToC_fun f)) t).
Qed.

Lemma bind_η_fun {X : sortToC} (t : sort) :
  bind_fun η_fun t = pr1 (identity (M X)) t.
Proof.
etrans; [|apply (nat_trans_eq_pointwise (@bind_η _ M X) t)].
apply cancel_postcomposition.
assert (H : sortToC_fun η_fun = η M X).
{ now apply nat_trans_eq; [apply homset_property|]. }
exact (nat_trans_eq_pointwise (maponpaths (λ a, # M a) H) t).
Qed.

Lemma bind_bind_fun {X Y Z : sortToC}
                    (f : t, Cpr1 X t,pr1 (M Y) t)
                    (g : t, Cpr1 Y t, pr1 (M Z) t )
                    (t : sort) :
  bind_fun f t · bind_fun g t = bind_fun (λ s, f s · bind_fun g s) t.
Proof.
etrans; [apply (nat_trans_eq_pointwise (bind_bind (sortToC_fun f) (sortToC_fun g)) t)|].
apply cancel_postcomposition.
assert (H : sortToC_fun f · bind (sortToC_fun g) = sortToC_fun (λ s, f s · bind_fun g s)).
{ now apply nat_trans_eq; [apply homset_property|]. }
exact (nat_trans_eq_pointwise (maponpaths (λ a, # M a) H) t).
Qed.

Definition monadSubstGen_instantiated {X Y : sortToC} (f : sortToCX,M Y) :
  sortToCM (X Y),M Y := monadSubstGen M BCsortToC Y f.

Definition monadSubstGen_fun {X Y : sortToC} (f : s, Cpr1 X s,pr1 (M Y) s) :
   t, Cpr1 (M (X Y)) t,pr1 (M Y) t :=
    λ t, pr1 (monadSubstGen_instantiated (sortToC_fun f)) t.

Definition monadSubst_instantiated {X : sortToC} (f : sortToC1,M X) :
  sortToCM (1 X),M X := monadSubst M BCsortToC TsortToC X f.

Definition mweak_instantiated {X Y : sortToC} :
  sortToCM Y,M (X Y) := mweak M BCsortToC _ _.

Definition mexch_instantiated {X Y Z : sortToC} :
  sortToCM (X (Y Z)), M (Y (X Z)) :=
    mexch M BCsortToC _ _ _.

Lemma subst_interchange_law_gen_instantiated {X Y Z : sortToC}
      (f : sortToCX,M (Y Z)) (g : sortToCY, M Z) :
  monadSubstGen_instantiated f · monadSubstGen_instantiated g =
  mexch_instantiated · monadSubstGen_instantiated (g · mweak_instantiated)
                     · monadSubstGen_instantiated (f · monadSubstGen_instantiated g).
Proof.
apply subst_interchange_law_gen.
Qed.

Lemma subst_interchange_law_instantiated {X : sortToC}
      (f : sortToC1,M (1 X)) (g : sortToC1,M X) :
  monadSubst_instantiated f · monadSubst_instantiated g =
  mexch_instantiated · monadSubst_instantiated (g · mweak_instantiated)
                     · monadSubst_instantiated (f · monadSubst_instantiated g).
Proof.
apply subst_interchange_law.
Qed.

End MonadInSortToC.