Built with Alectryon, running Coq+SerAPI v8.19.0+0.19.3. Bubbles () indicate interactive fragments: hover for details, tap to reveal contents. Use Ctrl+↑ Ctrl+↓ to navigate, Ctrl+🖱️ to focus. On Mac, use instead of Ctrl.
[Loading ML file number_string_notation_plugin.cmxs (using legacy method) ... done]
[Loading ML file tauto_plugin.cmxs (using legacy method) ... done]
[Loading ML file cc_plugin.cmxs (using legacy method) ... done]
[Loading ML file firstorder_plugin.cmxs (using legacy method) ... done]
Require Import Basics.Overture Basics.Tactics Basics.Equivalences. Require Import WildCat.Core WildCat.Equiv WildCat.EquivGpd WildCat.PointedCat WildCat.Yoneda WildCat.Graph WildCat.ZeroGroupoid WildCat.Pullbacks WildCat.AbEnriched WildCat.FunctorCat. (** * Epi-stable categories *) (** Epi-stable categories are those in which pullbacks exist and epimorphisms are stable under pullback. This is somewhat similar to a regular category, but differs in a couple of ways. First, we use the ordinary epimorphisms rather than the effective epimorphisms, mostly because they are easier to formalize. Second, we don't assume that kernel pairs have co-equalizers. *) (** ** Definition *) Class IsEpiStable (A : Type) `{Is1Cat A} := { haspullbacks :: HasPullbacks A; stable_epic :: forall {a b c} (f : a $-> c) (g : b $-> c) {ep : Epic f}, Epic (cat_pb_pr2 (CatPullback:=haspullbacks a b c f g)); }. (** ** Diagram chasing in an epi-stable category *) (** One can do a certain amount of diagram chasing in an epi-stable category. We'll see below that more can be done with an enrichment over abelian groups. *) Section EpiStable. Context {A : Type} `{IsEpiStable A}. (** A generalized element of [B] with domain [P]. *) Definition elt (P B : A) := P $-> B. (** A generalized lift of a generalized element. *) Definition Lift {B C P : A} (c : elt P C) (f : B $-> C) := { P' : A & { e : P' $->> P & { b : elt P' B & f $o b $== c $o e }}}. (** Epimorphisms are characterized by the existence of generalized lifts. *)
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C: A
f: B $-> C
lift: forall (P : A) (c : elt P C), Lift c f

Epic f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C: A
f: B $-> C
lift: forall (P : A) (c : elt P C), Lift c f

Epic f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C: A
f: B $-> C
lift: Lift (Id C) f

Epic f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C: A
f: B $-> C
P': A
e: P' $-> C
ep: Epic e
b: elt P' B
h: f $o b $== Id C $o (e; ep)

Epic f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C: A
f: B $-> C
P': A
e: P' $-> C
ep: Epic e
b: elt P' B
h: f $o b $== Id C $o e

Epic f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C: A
f: B $-> C
P': A
e: P' $-> C
ep: Epic e
b: elt P' B
h': f $o b $== e

Epic f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C: A
f: B $-> C
P': A
e: P' $-> C
ep: Epic e
b: elt P' B
h': f $o b $== e

Epic (f $o b)
apply (epic_homotopic _ _ h'^$). Defined. (** For the converse, we need the pullback epi of an epi. *)
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
a, b, c: A
f: a $->> c
g: b $-> c

cat_pb f g $->> b
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
a, b, c: A
f: a $->> c
g: b $-> c

cat_pb f g $->> b
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
a, b, c: A
f: a $->> c
g: b $-> c

Epic cat_pb_pr2
exact _. Defined.
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C: A
f: B $-> C
ef: Epic f
P: A
c: elt P C

Lift c f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C: A
f: B $-> C
ef: Epic f
P: A
c: elt P C

Lift c f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C: A
f: B $-> C
ef: Epic f
P: A
c: elt P C

{e : cat_pb f c $->> P & {b : elt (cat_pb f c) B & f $o b $== c $o e}}
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C: A
f: B $-> C
ef: Epic f
P: A
c: elt P C

{b : elt (cat_pb f c) B & f $o b $== c $o cat_pb_pr2_epi (f; ef) c}
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C: A
f: B $-> C
ef: Epic f
P: A
c: elt P C

f $o cat_pb_pr1 $== c $o cat_pb_pr2_epi (f; ef) c
apply cat_pb_glue. Defined. (** The definition of [Monic] already implicitly involves generalized elements. So this result isn't really needed, but might become more useful if we introduce a separate notation for applying a function to an [elt]. *) Definition monic_via_elt {B C : A} (f : B $-> C) (m : forall P (b b' : elt P B), f $o b $== f $o b' -> b $== b') : Monic f := m. (** To lift [b] through [f], it's enough to lift a pullback of [b] along an epi through [f]. *)
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C, P, P': A
b: elt P C
f: B $-> C
e: P' $->> P
l: Lift (b $o e) f

Lift b f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C, P, P': A
b: elt P C
f: B $-> C
e: P' $->> P
l: Lift (b $o e) f

Lift b f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C, P, P': A
b: elt P C
f: B $-> C
e: P' $->> P
P'': A
e': P'' $-> P'
ep: Epic e'
b': elt P'' B
h: f $o b' $== b $o e $o (e'; ep)

Lift b f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C, P, P': A
b: elt P C
f: B $-> C
e: P' $->> P
P'': A
e': P'' $-> P'
ep: Epic e'
b': elt P'' B
h: f $o b' $== b $o e $o (e'; ep)

{e : P'' $->> P & {b0 : elt P'' B & f $o b0 $== b $o e}}
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C, P, P': A
b: elt P C
f: B $-> C
e: P' $->> P
P'': A
e': P'' $-> P'
ep: Epic e'
b': elt P'' B
h: f $o b' $== b $o e $o (e'; ep)

{b0 : elt P'' B & f $o b0 $== b $o (e $o e'; epic_compose e' e)}
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C, P, P': A
b: elt P C
f: B $-> C
e: P' $->> P
P'': A
e': P'' $-> P'
ep: Epic e'
b': elt P'' B
h: f $o b' $== b $o e $o (e'; ep)

f $o b' $== b $o (e $o e'; epic_compose e' e)
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsEpiStable A
B, C, P, P': A
b: elt P C
f: B $-> C
e: P' $->> P
P'': A
e': P'' $-> P'
ep: Epic e'
b': elt P'' B
h: f $o b' $== b $o e $o (e'; ep)

f $o b' $== b $o e $o e'
exact h. Defined. End EpiStable. (** Many proofs using diagram chasing end by supplying an element of [Lift] with [e] being the identity map. This helps with this common case. See below for an example. *) Tactic Notation "provide_lift" uconstr(a) := refine (_; id_epi _; a; _); try rhs' napply cat_idr. (** ** Epi-stable categories enriched in abelian groups *) Class IsAbEpiStable (A : Type) `{Is1Cat A} := { isepistable_abepistable :: IsEpiStable A; isabenriched_abepistable :: IsAbEnriched A; }. Section AbEpiStable. Context {A : Type} `{IsAbEpiStable A}. Open Scope mc_add_scope. (** We *define* exactness using generalized elements, which avoids needing to assume the existence of kernels, cokernels or images (or even define them). In a category with kernels and images, you can show that this implies that the natural map from the image to the kernel is monic and epic, so in a nice enough category it is an isomorphism. *) Class CatIsAbExact {B C D : A} (f : B $-> C) (g : C $-> D) := { isabcomplex : g $o f $== 0; isabexact : forall P (b : elt P C), g $o b $== 0 -> Lift b f; }. (** If a sequence [B $-> C] and [0 : C $-> D] is exact, then [B $-> C] is epi. *)
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: B $-> C
CatIsAbExact0: CatIsAbExact f (0 : C $-> D)

Epic f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: B $-> C
CatIsAbExact0: CatIsAbExact f (0 : C $-> D)

Epic f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: B $-> C
CatIsAbExact0: CatIsAbExact f (0 : C $-> D)

forall (P : A) (c : elt P C), Lift c f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: B $-> C
CatIsAbExact0: CatIsAbExact f (0 : C $-> D)
P: A
c: elt P C

Lift c f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: B $-> C
CatIsAbExact0: CatIsAbExact f (0 : C $-> D)
P: A
c: elt P C

0 $o c $== 0
apply precomp_zero. Defined. (** If [B $-> C] is epi, then the sequence [B $-> C] and [0 : C$-> D] is exact. *)
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: B $->> C

CatIsAbExact f (0 : C $-> D)
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: B $->> C

CatIsAbExact f (0 : C $-> D)
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: B $->> C

forall (P : A) (b : elt P C), 0 $o b $== 0 -> Lift b f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: B $->> C
P: A
c: elt P C
h: 0 $o c $== 0

Lift c f
rapply elt_lift_epic. Defined. (** This is a variant of [ismonic], where the RHS is [0]. *)
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C: A
f: B $-> C
Monic0: Monic f
P: A
b: elt P B
h: f $o b $== 0

b $== 0
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C: A
f: B $-> C
Monic0: Monic f
P: A
b: elt P B
h: f $o b $== 0

b $== 0
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C: A
f: B $-> C
Monic0: Monic f
P: A
b: elt P B
h: f $o b $== 0

f $o b $== f $o 0
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C: A
f: B $-> C
Monic0: Monic f
P: A
b: elt P B
h: f $o b $== 0

0 $== f $o 0
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C: A
f: B $-> C
Monic0: Monic f
P: A
b: elt P B
h: f $o b $== 0

f $o 0 $== 0
apply postcomp_zero. Defined. (** A map is monic if only the zero map is mapped to zero by post-composition. *)
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C: A
f: B $-> C
h: forall (P : A) (b : elt P B), f $o b $== 0 -> b $== 0

Monic f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C: A
f: B $-> C
h: forall (P : A) (b : elt P B), f $o b $== 0 -> b $== 0

Monic f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C: A
f: B $-> C
h: forall (P : A) (b : elt P B), f $o b $== 0 -> b $== 0
P: A
b', b'': elt P B
h': f $o b' $== f $o b''

b' $== b''
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C: A
f: B $-> C
h: forall (P : A) (b : elt P B), f $o b $== 0 -> b $== 0
P: A
b', b'': elt P B
h': f $o b' - f $o b'' $== 0

b' $== b''
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C: A
f: B $-> C
h: forall (P : A) (b : elt P B), f $o b $== 0 -> b $== 0
P: A
b', b'': elt P B
h': f $o (b' - b'') $== 0

b' $== b''
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C: A
f: B $-> C
h: forall (P : A) (b : elt P B), f $o b $== 0 -> b $== 0
P: A
b', b'': elt P B
h': f $o (b' - b'') $== 0

b' - b'' $== 0
exact (h P (b' - b'') h'). Defined. (** If [C $-> D] is monic, then the sequence [0 : B $-> C] and [C $-> D] is exact. *)
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $>-> D

CatIsAbExact (0 : B $-> C) f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $>-> D

CatIsAbExact (0 : B $-> C) f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $>-> D

forall (P : A) (b : elt P C), f $o b $== 0 -> Lift b 0
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $>-> D
P: A
b: elt P C
h: f $o b $== 0

Lift b 0
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $>-> D
P: A
b: elt P C
h: f $o b $== 0

0 $o 0 $== b
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $>-> D
P: A
b: elt P C
h: f $o b $== 0

0 $== b
exact (ismonic' f b h)^$. Defined. (** If the sequence [0 : B $-> C] and [C $-> D] is exact, then [C $-> D] is a mono. *)
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $-> D
CatIsAbExact0: CatIsAbExact (0 : B $-> C) f

Monic f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $-> D
CatIsAbExact0: CatIsAbExact (0 : B $-> C) f

Monic f
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $-> D
CatIsAbExact0: CatIsAbExact (0 : B $-> C) f

forall (P : A) (b b' : elt P C), f $o b $== f $o b' -> b $== b'
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $-> D
CatIsAbExact0: CatIsAbExact (0 : B $-> C) f
P: A
b, b': elt P C
h: f $o b $== f $o b'

b $== b'
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $-> D
CatIsAbExact0: CatIsAbExact (0 : B $-> C) f
P: A
b, b': elt P C
h: f $o b - f $o b' $== 0

b $== b'
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $-> D
CatIsAbExact0: CatIsAbExact (0 : B $-> C) f
P: A
b, b': elt P C
h: f $o (b - b') $== 0

b $== b'
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $-> D
CatIsAbExact0: CatIsAbExact (0 : B $-> C) f
P: A
b, b': elt P C
h: Lift (b - b') 0

b $== b'
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $-> D
CatIsAbExact0: CatIsAbExact (0 : B $-> C) f
P: A
b, b': elt P C
P': A
e: P' $->> P
b'': elt P' B
h: 0 $o b'' $== (b - b') $o e

b $== b'
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $-> D
CatIsAbExact0: CatIsAbExact (0 : B $-> C) f
P: A
b, b': elt P C
P': A
e: P' $->> P
b'': elt P' B
h: 0 $o b'' $== (b - b') $o e

b - b' $== 0
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $-> D
CatIsAbExact0: CatIsAbExact (0 : B $-> C) f
P: A
b, b': elt P C
P': A
e: P' $->> P
b'': elt P' B
h: 0 $o b'' $== (b - b') $o e

(b - b') $o e $== 0 $o e
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $-> D
CatIsAbExact0: CatIsAbExact (0 : B $-> C) f
P: A
b, b': elt P C
P': A
e: P' $->> P
b'': elt P' B
h: 0 $o b'' $== (b - b') $o e

(b - b') $o e $== 0
A: Type
IsGraph0: IsGraph A
Is2Graph0: Is2Graph A
Is01Cat0: Is01Cat A
H: Is1Cat A
H0: IsAbEpiStable A
B, C, D: A
f: C $-> D
CatIsAbExact0: CatIsAbExact (0 : B $-> C) f
P: A
b, b': elt P C
P': A
e: P' $->> P
b'': elt P' B
h: 0 $o b'' $== (b - b') $o e

(b - b') $o e $== 0 $o b''
exact h^$. Defined. End AbEpiStable. (** ** Tactics *) (** The [fix_left] tactic is the key to smooth diagram chasing in an [IsAbEpiStable] category. Given [lift : Lift ? ?]; we extract the lifted element using the provided name [d] and the proof it is a lift using the name [l]. Then we update all other generalized elements to have the same domain as [d]. We could also have a limited version of this tactic for an [IsEpiStable] category. *) Ltac fix_lift lift d l := let P2 := fresh "P" in let e := fresh "e" in let ee := fresh "ee" in destruct lift as [P2 [[e ee] [d l]]]; unfold hom_epi, ".1" in l; match type of e with | P2 $-> ?P => (* Adjust everything involving the domain [P] to have domain [P2]: *) repeat match goal with | [ |- @Lift _ _ _ _ _ _ _ P ?c ?f ] => apply (lift_helper (e; ee)); unfold hom_epi, ".1" | [ |- @GpdHom (@Hom _ _ P _) _ _ _ _ _ ] => apply ee | [ |- @GpdHom (@elt _ _ P _) _ _ _ _ _ ] => apply ee | [ h : @GpdHom (@Hom _ _ P _) _ _ _ _ _ |- _ ] => apply (fun w => cat_prewhisker w e) in h end; clear ee; (* Reassociate all homotopies so that [e] is right against the map to its left: *) rewrite ? (cat_assoc e); repeat match goal with (* Replace [(g $o f) $o e] with [g $o (f $o e)]: *) | [ h : @GpdHom (@Hom _ _ P2 _) _ _ _ _ _ |- _ ] => rewrite ! (cat_assoc e) in h (* The above sometimes doesn't do all of the reassociating it could. In the notation of the next line, we know that [B] and [D] are definitionally equal, but they might not be syntactically equal, and this can cause the rewrite to fail. So we coerce it to be of the right type. The previous line is thus redundant, but having it causes no slowdown, and it makes it clear what we are doing here. Similar issues may arise with the four other rewrites in this tactic, as well as the [cat_prewhisker]. An alternative to fixing them all would be to "sanitize" all [cat_comp] occurences to have the canonical objects as implicit arguments. *) | [ h : context [ @cat_comp ?A _ _ P2 P ?B (@cat_comp ?A _ _ P ?C ?D ?g ?f) e ] |- _ ] => rewrite (cat_assoc e f g : @cat_comp A _ _ P2 P B (@cat_comp A _ _ P C D g f) e $== _) in h (* Replace [(f - g) $o e] with [f $o e - (g $o e)]: *) | [ h : @GpdHom (@Hom _ _ P2 _) _ _ _ _ _ |- _ ] => rewrite ! (precomp_op_inv e) in h (* Replace [(f + g) $o e] with [f $o e + g $o e]: *) | [ h : @GpdHom (@Hom _ _ P2 _) _ _ _ _ _ |- _ ] => rewrite ! (precomp_op e) in h (* Replace [(-f) $o e] with [-(f $o e)]: *) | [ h : @GpdHom (@Hom _ _ P2 _) _ _ _ _ _ |- _ ] => rewrite ! (precomp_inv e) in h end; (* At this point, all generalized elements [c] with domain [P] should only appear in the form [c $o e], so by generalizing [c $o e] we can replace them with elements with domain [P2]. *) repeat match goal with | [ c : elt P ?C |- _ ] => let tmp := fresh in set (tmp := c $o e : elt P2 _) in *; (* Unfortunately, [set] doesn't always find all occurrences, sometimes because the implicit arguments giving the domains/codomains of [c] and [e] are syntactically different. The next match handles this. *) repeat match goal with | [ h : context [ ?f $o e ] |- _ ] => change (f $o e) with tmp in h end; clearbody tmp; clear c; rename tmp into c end; (* We may also have expressions of the form [0 $o e], so we replace those with [0]. *) rewrite ? (precomp_zero e) in *; (* Now we can get rid of [e] and [P]. Add [try] before these two lines when debugging. *) clear e P; rename P2 into P end. (** Ideas for making the above tactic faster: - In localized tests, defining and using things like [pose (cp := fun c f g w => cat_prewhisker (c:=c) (f:=f) (g:=g) w e).] made things faster, maybe because typeclass inference of the wild category structure didn't need to be repeated. But when done here, the overall speed does not improve. - In the last clause of the first [repeat match], we could add "revert h". Then instead of the second [repeat match] we would just need to do rewriting in the goal, without scanning the context for appropriate terms, using something like: [repeat progress rewrite ? (cat_assoc e), ? (precomp_op_inv e), ? (precomp_op e), ? (precomp_inv e).] For this to work, we need to extend setoid rewriting to handle [->]. - Alternatively, in the second [repeat match], we could make the matches more specific to ensure that the lemmas we try apply to the situation. *) Ltac elt_lift_epic f a b l := fix_lift (elt_lift_epic f a) b l. Tactic Notation "elt_lift_exact" constr(f) constr(g) constr(a) ident(b) ident(l) := let lift := fresh in unshelve epose proof (lift := isabexact (f:=f) (g:=g) _ a _); only 2: fix_lift lift b l. (** This allows us to insert a tactic [tac] before [fix_lift] is called. Typically this is used to clear unneeded hypotheses from the context, which speeds up [fix_lift]. *) Tactic Notation "elt_lift_exact" constr(f) constr(g) constr(a) ident(b) ident(l) "using" tactic3(tac) := let lift := fresh in unshelve epose proof (lift := isabexact (f:=f) (g:=g) _ a _); only 2: (tac; fix_lift lift b l). (** Other tactics that are convenient when diagram chasing. *) (** Given a homotopy [h : a $o b = a' $o b'], use associativity to change [a $o (b $o c)] to [a' $o (b' $o c)]. *) Ltac rewrite_with_assoc h := rewrite (cat_assoc_opp _ _ _ $@ cat_prewhisker h _ $@ cat_assoc _ _ _). (** Similar, with the other parenthesization. *) Ltac rewrite_with_assoc_opp h := rewrite (cat_assoc _ _ _ $@ cat_postwhisker _ h $@ cat_assoc_opp _ _ _).