module Cat.Instances.MarkedGraphs where
Marked graphs🔗
Recall that every graph gives rise to a free category Though useful, this construction is rather limiting; if we want to construct (strict) categories via combinatorial methods, then we really do need a way to add non-trivial equalities. In other words, we need to be able to construct categories freely from a set of generators and relations! This leads us to the following series of definitions:
A marked graph consists of a graph along with a set of pairs of distinguished paths in which we will call marked pairs.
record Marked-graph (o ℓ : Level) : Type (lsuc o ⊔ lsuc ℓ) where
no-eta-equality
field
: Graph o ℓ
graph
open Graph graph public
field
: ∀ {x y} → Path-in graph x y → Path-in graph x y → Ω Marked
private variable
: Level
o ℓ o' ℓ' : Marked-graph o ℓ
G H K
open Marked-graph
open hlevel-projection
instance
: Underlying (Marked-graph o ℓ)
Underlying-Marked-graph = record { ⌞_⌟ = Vertex } Underlying-Marked-graph
A marked graph homomorphism is a graph homomorphism that takes a marked pair in to a marked pair in
record Marked-graph-hom (G : Marked-graph o ℓ) (H : Marked-graph o' ℓ') : Type (o ⊔ ℓ ⊔ o' ⊔ ℓ') where
no-eta-equality
field
: Graph-hom (G .graph) (H .graph)
hom
pres-marked: ∀ {x y} {p q : Path-in (G .graph) x y}
→ ∣ G .Marked p q ∣
→ ∣ H .Marked (path-map hom p) (path-map hom q) ∣
open Graph-hom hom public
open Marked-graph-hom
unquoteDecl H-Level-Marked-graph-hom = declare-record-hlevel 2 H-Level-Marked-graph-hom (quote Marked-graph-hom)
instance
: Funlike (Marked-graph-hom G H) ⌞ G ⌟ λ _ → ⌞ H ⌟
Funlike-Marked-graph-hom .Funlike._#_ = vertex
Funlike-Marked-graph-hom
Marked-graph-hom-pathp: {G : I → Marked-graph o ℓ} {H : I → Marked-graph o' ℓ'}
→ {f : Marked-graph-hom (G i0) (H i0)} {g : Marked-graph-hom (G i1) (H i1)}
→ (p0 : ∀ (x : ∀ i → G i .Vertex)
→ PathP (λ i → H i .Vertex)
(f .vertex (x i0)) (g .vertex (x i1)))
→ (p1 : ∀ {x y : ∀ i → G i .Vertex}
→ (e : ∀ i → G i .Edge (x i) (y i))
→ PathP (λ i → H i .Edge (p0 x i) (p0 y i))
(f .edge (e i0)) (g .edge (e i1)))
→ PathP (λ i → Marked-graph-hom (G i) (H i)) f g
{G = G} {H = H} {f = f} {g = g} p0 p1 = comm-path where
Marked-graph-hom-pathp : PathP (λ i → Graph-hom (G i .graph) (H i .graph)) (f .hom) (g .hom)
hom-path =
hom-path {G = λ i → G i .graph} {H = λ i → H i .graph}
Graph-hom-pathp {f = f .hom} {g .hom}
p0 p1
: PathP (λ i → Marked-graph-hom (G i) (H i)) f g
comm-path .hom = hom-path i
comm-path i .pres-marked {x} {y} {p} {q} =
comm-path i (λ i →
is-prop→pathp {A = G i .Vertex} 1 λ x →
Π-is-hlevel {A = G i .Vertex} 1 λ y →
Π-is-hlevel {A = Path-in (G i .graph) x y} 1 λ p →
Π-is-hlevel {A = Path-in (G i .graph) x y} 1 λ q →
Π-is-hlevel {A = ∣ G i .Marked p q ∣} 1 λ _ →
Π-is-hlevel .Marked (path-map (hom-path i) p) (path-map (hom-path i) q) .is-tr)
H i (λ _ _ _ _ → f .pres-marked)
(λ _ _ _ _ → g .pres-marked)
i x y p q
Marked-graph-hom-path: {f g : Marked-graph-hom G H}
→ (p0 : ∀ x → f .vertex x ≡ g .vertex x)
→ (p1 : ∀ {x y} → (e : G .Edge x y) → PathP (λ i → H .Edge (p0 x i) (p0 y i)) (f .edge e) (g .edge e))
→ f ≡ g
{G = G} {H = H} p0 p1 =
Marked-graph-hom-path {G = λ _ → G} {H = λ _ → H}
Marked-graph-hom-pathp (λ x i → p0 (x i) i)
(λ e i → p1 (e i) i)
We can organize marked graphs and their homomorphisms into a category
: ∀ o ℓ → Precategory (lsuc o ⊔ lsuc ℓ) (o ⊔ ℓ)
Marked-graphs .Precategory.Ob = Marked-graph o ℓ
Marked-graphs o ℓ .Precategory.Hom = Marked-graph-hom Marked-graphs o ℓ
Composition, identities, and their corresponding equations are a bit tedious, so we omit the details.
.Precategory.Hom-set _ _ = hlevel 2
Marked-graphs o ℓ .Precategory.id {G} .hom = Graphs.id
Marked-graphs o ℓ .Precategory.id {G} .pres-marked p~q =
Marked-graphs o ℓ (λ p q → ∣ G .Marked p q ∣)
subst₂ (sym (path-map-id _))
(sym (path-map-id _))
p~q.Precategory._∘_ f g .hom = (f .hom) Graphs.∘ (g .hom)
Marked-graphs o ℓ .Precategory._∘_ {G} {H} {K} f g .pres-marked p~q =
Marked-graphs o ℓ (λ p q → ∣ K .Marked p q ∣)
subst₂ (sym (path-map-∘ _))
(sym (path-map-∘ _))
(f .pres-marked (g .pres-marked p~q))
.Precategory.idr _ =
Marked-graphs o ℓ (λ _ → refl) (λ _ → refl)
Marked-graph-hom-path .Precategory.idl _ =
Marked-graphs o ℓ (λ _ → refl) (λ _ → refl)
Marked-graph-hom-path .Precategory.assoc _ _ _ =
Marked-graphs o ℓ (λ _ → refl) (λ _ → refl) Marked-graph-hom-path
open Functor
Moreover, there is a faithful functor from the category of strict categories to the category of marked graphs.
: Functor (Strict-cats o ℓ) (Marked-graphs o ℓ) Strict-cats↪Marked-graphs
The action on objects takes a strict category to its underlying graph, and adds a marked pair between whenever and are equal as morphisms in
.F₀ C .graph =
Strict-cats↪Marked-graphs .F₀ C
Strict-cats↪Graphs .F₀ C .Marked p q =
Strict-cats↪Marked-graphs (path-reduce C p ≡ path-reduce C q) elΩ
Functoriality follows from the fact that functors take equal paths in to equal paths in
.F₁ F .hom =
Strict-cats↪Marked-graphs .F₁ F
Strict-cats↪Graphs .F₁ F .pres-marked {p = p} {q = q} =
Strict-cats↪Marked-graphs λ p~q → path-reduce-natural F p ∙ ap (F .F₁) p~q ∙ sym (path-reduce-natural F q)
□-map .F-id =
Strict-cats↪Marked-graphs (λ _ → refl) (λ _ → refl)
Marked-graph-hom-path .F-∘ F G =
Strict-cats↪Marked-graphs (λ _ → refl) (λ _ → refl) Marked-graph-hom-path
This functor is clearly faithful; the proof is essentially just relabeling data.
Strict-cats↪Marked-graphs-faithful: is-faithful (Strict-cats↪Marked-graphs {o} {ℓ})
=
Strict-cats↪Marked-graphs-faithful p (λ x i → p i .vertex x) (λ f i → p i .edge f) Functor-path
More surprisingly this functor is also full!
Strict-cats↪Marked-graphs-full: is-full (Strict-cats↪Marked-graphs {o} {ℓ})
To see why, suppose that is a marked graph homomorphism between strict categories. By definition, already contains the data of a functor; the tricky part is showing that this data is functorial.
{x = C} {y = D} f =
Strict-cats↪Marked-graphs-full (functor , Marked-graph-hom-path (λ _ → refl) (λ _ → refl))
pure where
module C = Precategory (C .fst)
module D = Precategory (D .fst)
: Functor (C .fst) (D .fst)
functor .F₀ = f .vertex
functor .F₁ = f .edge functor
The trick is that marked graph homomorphisms between categories cannot observe the intensional structure of paths, as they must preserve all marked pairs. In particular, will preserve the makred pair consisting of and the empty path; so
.F-id =
functor .edge C.id ≡˘⟨ D.idl _ ⟩
f .id D.∘ f .edge C.id ≡⟨ □-out! (f .pres-marked {p = cons C.id nil} {q = nil} (inc (C.idl _))) ⟩
D.id ∎ D
Additionally, will preserve the marked pair consisting of the singleton path and the path so
.F-∘ g h =
functor .edge (g C.∘ h) ≡˘⟨ D.idl _ ⟩
f .id D.∘ f .edge (g C.∘ h) ≡˘⟨ □-out! (f .pres-marked {p = cons h (cons g nil)} {q = cons (g C.∘ h) nil} (pure (sym (C.assoc _ _ _)))) ⟩
D(D.id D.∘ f .edge g) D.∘ f .edge h ≡⟨ ap₂ D._∘_ (D.idl _) refl ⟩
.edge g D.∘ f .edge h ∎ f
Categories from generators and relations🔗
module _ (G : Marked-graph o ℓ) where
private module G = Marked-graph G
In this section, we will detail how to freely construct a category from a marked graph Intuitvely, this works by freely generating a category from and then identifying all marked pairs. However, there is a slight subtlety here: the marked pairs of may not respect path concatenation, and may not even form an equivalence relation!
Luckily, there is an easy (though tedious) solution to this problem: we can instead quotient by the reflexive-transitive-congruent closure instead, which we encode in Agda via the following higher-inductive type.
data Marking
: ∀ {x y} → Path-in G.graph x y → Path-in G.graph x y
→ Type (o ⊔ ℓ)
where
marked: ∀ {x y} {p q : Path-in G.graph x y}
→ ∣ G.Marked p q ∣
→ Marking p q
reflexive: ∀ {x y} {p : Path-in G.graph x y}
→ Marking p p
symmetric: ∀ {x y} {p q : Path-in G.graph x y}
→ Marking q p
→ Marking p q
transitive: ∀ {x y} {p q r : Path-in G.graph x y}
→ Marking p q → Marking q r
→ Marking p r
congruent: ∀ {x y z} {p q : Path-in G.graph x y} {r s : Path-in G.graph y z}
→ Marking p q → Marking r s
→ Marking (p ++ r) (q ++ s)
: ∀ {x y} {p q : Path-in G.graph x y} → is-prop (Marking p q) trunc
unquoteDecl Marking-elim-prop = make-elim-n 1 Marking-elim-prop (quote Marking)
The resulting category is not too difficult to construct: the objects are the vertices of and the morphisms are paths in quotiented by the aforementioned closure.
: Precategory o (o ⊔ ℓ)
Marked-path-category .Precategory.Ob = G.Vertex
Marked-path-category .Precategory.Hom x y =
Marked-path-category .graph x y / Marking
Path-in G.Precategory.Hom-set _ _ = hlevel 2
Marked-path-category .Precategory.id = inc nil
Marked-path-category .Precategory._∘_ =
Marked-path-category
Quot-op₂(λ _ → reflexive) (λ _ → reflexive)
(flip _++_)
(λ p q r s p~q r~s → congruent r~s p~q)
.Precategory.idr =
Marked-path-category λ p → refl
elim! .Precategory.idl =
Marked-path-category λ p → ap inc (++-idr p)
elim! .Precategory.assoc =
Marked-path-category λ p q r → ap inc (++-assoc r q p) elim!
Proving that this construction is free involves a bit more work.
We start with a useful lemma. Let be an arbitrary strict category, a marked graph, and a marked graph homomorphism. If are related by the marked closure of then folding over and results in the same morphism.
module _ (C : Σ (Precategory o ℓ) is-strict) where
private
module C = Cat.Reasoning (C .fst)
: Marked-graph o ℓ
∣C∣ = Strict-cats↪Marked-graphs .F₀ C
∣C∣
path-fold-marking: (f : Marked-graph-hom G ∣C∣)
→ ∀ {x y} (p q : Path-in (G .graph) x y)
→ Marking G p q
→ path-fold C (f .hom) p ≡ path-fold C (f .hom) q
The proof is an easy but tedious application of the elimination
principle of Marking
.
{G = G} f p q =
path-fold-marking
Marking-elim-prop G{P = λ {x} {y} {p} {q} _ →
(f .hom) p ≡ path-fold C (f .hom) q}
path-fold C (λ _ → hlevel 1)
(λ {_} {_} {p} {q} p~q →
(path-reduce-map p)
sym (f .pres-marked p~q)
∙ □-out! )
∙ path-reduce-map q(λ _ → sym) (λ _ p=q _ q=r → p=q ∙ q=r)
refl (λ {_} {_} {_} {p} {q} {r} {s} _ p=q _ r=s →
path-fold-++ C p r._∘_ r=s p=q
∙ ap₂ C(path-fold-++ C q s)) ∙ sym
This lemma lets us extend folds over paths to folds over quotiented paths.
comm-path-fold: (f : Marked-graph-hom G ∣C∣)
→ ∀ {x y} → Path-in (G .graph) x y / Marking G
→ C.Hom (f .vertex x) (f .vertex y)
=
comm-path-fold f (λ _ → hlevel 2)
Quot-elim (path-fold C (f .hom))
(path-fold-marking f)
Moreover, this extends to a functor from the free category over to
MarkedF: Marked-graph-hom G ∣C∣
→ Functor (Marked-path-category G) (C .fst)
.F₀ = f .vertex
MarkedF f .F₁ = comm-path-fold f
MarkedF f .F-id = refl
MarkedF f .F-∘ = elim! λ p q → path-fold-++ C q p MarkedF f
Like path categories, functors out of marked path categories are purely characterized by their behaviour on edges.
module _ {C : Precategory o ℓ} where
private module C = Cat.Reasoning C
Marked-path-category-functor-path: {F F' : Functor (Marked-path-category G) C}
→ (p0 : ∀ x → F .F₀ x ≡ F' .F₀ x)
→ (p1 : ∀ {x y} (e : G .Edge x y)
→ PathP (λ i → C.Hom (p0 x i) (p0 y i))
(F .F₁ (inc (cons e nil)))
(F' .F₁ (inc (cons e nil))))
→ F ≡ F'
Like the analogous result for path categories, this proof involves some cubical yoga which we will hide from the innocent reader.
{G = G} {F = F} {F' = F'} p0 p1 =
Marked-path-category-functor-path (elim! p1-paths)
Functor-path p0 where
p1-paths: ∀ {x y}
→ (p : Path-in (G .graph) x y)
→ PathP (λ i → C.Hom (p0 x i) (p0 y i)) (F .F₁ (inc p)) (F' .F₁ (inc p))
{x = x} nil i =
p1-paths (∂ i) λ where
hcomp (i = i0) → F .F-id (~ j)
j (i = i1) → F' .F-id (~ j)
j (j = i0) → C.id
j (cons e p) i =
p1-paths (∂ i) λ where
hcomp (i = i0) → F .F-∘ (inc p) (inc (cons e nil)) (~ j)
j (i = i1) → F' .F-∘ (inc p) (inc (cons e nil)) (~ j)
j (j = i0) → p1-paths p i C.∘ p1 e i j
With these results in hand, we can return to our original goal of showing that the marked path category on is a free object relative to the underlying marked graph functor.
Marked-free-category: ∀ (G : Marked-graph ℓ ℓ)
→ Free-object Strict-cats↪Marked-graphs G
.Free-object.free = Marked-path-category G , hlevel 2 Marked-free-category G
The unit of the free object takes vertices to themselves, and edges to singleton paths. We need to do a bit of work to show that this construction preserves markings but it’s not too difficult.
.Free-object.unit = unit-comm where
Marked-free-category G : Σ (Precategory _ _) is-strict
G* = Marked-path-category G , hlevel 2
G*
: Graph _ _
∣G*∣ = Strict-cats↪Graphs .F₀ G*
∣G*∣
: Graph-hom (G .graph) ∣G*∣
unit .Graph-hom.vertex x = x
unit .Graph-hom.edge e = inc (cons e nil)
unit
: Marked-graph-hom G (Strict-cats↪Marked-graphs .F₀ G*)
unit-comm .hom = unit
unit-comm .pres-marked {p = p} {q = q} p~q =
unit-comm
pure $(path-map unit p) ≡⟨ path-reduce-map p ⟩
path-reduce G* (λ _ _ → refl) p ⟩
path-fold G* unit p ≡˘⟨ path-fold-unique G* inc refl (marked p~q) ⟩
inc p ≡⟨ quot (λ _ _ → refl) q ⟩
inc q ≡⟨ path-fold-unique G* inc refl
path-fold G* unit q ≡˘⟨ path-reduce-map q ⟩(path-map unit q) ∎ path-reduce G*
We’ve already put in the work for the universal property: all that we need to do is assemble previous results!
.Free-object.fold {Y = C} f = MarkedF C f
Marked-free-category G .Free-object.commute {Y = C} =
Marked-free-category G (λ _ → refl) (λ _ → idl _)
Marked-graph-hom-path where open Precategory (C .fst)
.Free-object.unique {Y = C} F p =
Marked-free-category G
Marked-path-category-functor-path(λ x i → p i .vertex x)
(λ e → to-pathp (from-pathp (λ i → p i .edge e) ∙ sym (idl _)))
where open Precategory (C .fst)
: Functor (Marked-graphs ℓ ℓ) (Strict-cats ℓ ℓ)
Marked-free-categories =
Marked-free-categories
free-objects→functor Marked-free-category
Marked-free-categories⊣Underlying-marked-graph: Marked-free-categories {ℓ} ⊣ Strict-cats↪Marked-graphs
=
Marked-free-categories⊣Underlying-marked-graph free-objects→left-adjoint Marked-free-category
This means that the category of strict categories is a reflective subcategory of the category of marked graphs. In more intuitive terms, that we can think about strict categories as algebraic structure placed atop a marked graph, as inclusions from reflective subcategories are monadic!
Marked-free-categories⊣Underlying-marked-graph-reflective: is-reflective (Marked-free-categories⊣Underlying-marked-graph {ℓ})
=
Marked-free-categories⊣Underlying-marked-graph-reflective
full+faithful→ff Strict-cats↪Marked-graphs
Strict-cats↪Marked-graphs-full Strict-cats↪Marked-graphs-faithful