module Cat.Instances.StrictCat.Cohesive where
Strict-cats is “cohesive”🔗
We prove that the category admits an adjoint quadruple
where the “central” adjoint is the functor which sends a strict category to its underlying set of objects. This lets us treat categories as giving a kind of “spatial” structure over The left- and right- adjoints to equip sets with the “discrete” and “codiscrete” spatial structures, where nothing is stuck together, or everything is stuck together.
The extra right adjoint to assigns a category to its set of connected components, which can be thought of as the “pieces” of the category. Two objects land in the same connected component if there is a path of morphisms connecting them, hence the name.
Generally, the term “cohesive” is applied to Grothendieck topoi,
which Strict-cats
is very
far from being. We’re using it here by analogy: There’s an adjoint
quadruple, where the functor
sends each category to its set of points: see the last section. Strictly
speaking, the left
adjoint to
isn’t defined by tensoring with Sets
, but it does have the
effect of sending
to the coproduct of
copies of the point category.
Disc ⊣ Γ🔗
We begin by defining the object set functor.
: Functor (Strict-cats o h) (Sets o)
Γ .F₀ (C , obset) = el (Ob C) obset
Γ .F₁ = F₀
Γ .F-id = refl
Γ .F-∘ _ _ = refl Γ
We must then prove that the assignment Disc'
extends to a functor from Sets
, and prove that it’s left adjoint to
the functor Γ
we defined above.
Then we define the adjunction Disc⊣Γ
.
: Functor (Sets ℓ) (Strict-cats ℓ ℓ)
Disc .F₀ S = Disc' S , S .is-tr
Disc .F₁ = lift-disc
Disc .F-id = Functor-path (λ x → refl) λ f → refl
Disc .F-∘ _ _ = Functor-path (λ x → refl) λ f → refl
Disc
: Disc {ℓ} ⊣ Γ
Disc⊣Γ = adj where Disc⊣Γ
For the adjunction unit
, we’re
asked to provide a natural transformation from the identity functor to
Since the object set of
is simply
the identity map suffices:
: Disc {ℓ} ⊣ Γ
adj .unit = NT (λ _ x → x) λ x y f i o → f o adj
The adjunction counit is slightly more complicated, as we have to
give a functor
naturally in
Since morphisms in discrete categories are paths, for a map
(in {- 1 -}
), it
suffices to assume
really is
and so the identity map suffices.
.counit = NT (λ x → F x) nat where
adj : (x : ⌞ Strict-cats ℓ ℓ ⌟)
F → Functor (Disc' (el _ (x .snd))) _
.F₀ x = x
F X .F₁ p = subst (X .fst .Hom _) p (X .fst .id) {- 1 -}
F X .F-id = transport-refl _
F X .F-∘ = lemma {A = X} F X
Fortunately the triangle identities are straightforwardly checked.
.zig {x} = Functor-path (λ x i → x) λ f → x .is-tr _ _ _ _
adj .zag = refl adj
Γ ⊣ Codisc🔗
The codiscrete category on a set is the strict category with object space and all hom-spaces contractible. The assignment of codiscrete categories extends to a functor where we lift functions to act on object parts and the action on morphisms is trivial.
: Functor (Sets ℓ) (Strict-cats ℓ ℓ)
Codisc .F₀ S = Codisc' ∣ S ∣ , S .is-tr
Codisc
.F₁ f .F₀ = f
Codisc .F₁ f .F₁ = λ _ → lift tt
Codisc .F₁ f .F-id = refl
Codisc .F₁ f .F-∘ = λ _ _ → refl
Codisc
.F-id = Functor-path (λ x → refl) λ f → refl
Codisc .F-∘ _ _ = Functor-path (λ x → refl) λ f → refl Codisc
The codiscrete category functor is right adjoint to the object set functor The construction of the adjunction is now simple in both directions:
: Γ ⊣ Codisc {ℓ}
Γ⊣Codisc = adj where
Γ⊣Codisc : _ ⊣ _
adj .unit =
adj (λ x → record { F₀ = λ x → x ; F₁ = λ _ → lift tt
NT ; F-id = refl ; F-∘ = λ _ _ → refl })
λ x y f → Functor-path (λ _ → refl) λ _ → refl
.counit = NT (λ _ x → x) λ x y f i o → f o
adj .zig = refl
adj .zag = Functor-path (λ _ → refl) λ _ → refl adj
Object set vs global sections🔗
Above, we defined the functor by directly projecting the underlying set of each category. Normally in the definition of a cohesion structure, we use the global sections functor which maps (where is the terminal object). Here we prove that these functors are naturally isomorphic, so our abbreviation above is harmless.
Below, we represent the terminal category as the codiscrete category on the terminal set. Using the codiscrete category here is equivalent to using the discrete category, but it is more convenient since the are definitionally contractible.
module _ {ℓ} where
import Cat.Morphism Cat[ Strict-cats ℓ ℓ , Sets ℓ ] as Nt
: Functor (Strict-cats ℓ ℓ) (Sets ℓ)
GlobalSections .F₀ C =
GlobalSections (Functor (Codisc' (Lift _ ⊤)) (C .fst)) (Functor-is-set (C .snd))
el .F₁ G F = G F∘ F
GlobalSections .F-id = funext λ _ → Functor-path (λ _ → refl) λ _ → refl
GlobalSections .F-∘ f g = funext λ _ → Functor-path (λ _ → refl) λ _ → refl GlobalSections
Since GlobalSections
is a
section of the
functor, it acts on maps by composition. The functor identities hold
definitionally.
: Γ {ℓ} Nt.≅ GlobalSections
GlobalSections≅Γ = Nt.make-iso f g f∘g g∘f where
GlobalSections≅Γ open Precategory
We define a natural isomorphism from Γ
to the GlobalSections
functor by sending each
object to the constant functor on that object. This assignment is
natural because it is essentially independent of the coordinate.
: Γ => GlobalSections
f .η x ob = record
f { F₀ = λ _ → ob
; F₁ = λ _ → x .fst .id
; F-id = refl
; F-∘ = λ _ _ → sym (x .fst .idl _)
}
.is-natural x y f = funext λ _ → Functor-path (λ _ → refl) λ _ → sym (f .F-id) f
In the opposite direction, the natural transformation is defined by
evaluating at the point. These natural transformations compose to the
identity almost definitionally, but Agda does need some convincing,
using our path helpers, Functor-path
and trivial!
.
: GlobalSections => Γ
g .η x f = f # lift tt
g .is-natural x y f = refl
g
: f ∘nt g ≡ idnt
f∘g = ext λ c x → Functor-path (λ x → refl) λ f → sym (x .F-id)
f∘g
: g ∘nt f ≡ idnt
g∘f = trivial! g∘f
Connected components🔗
The set of connected components of a category is the quotient of the
object set by the “relation” generated by the Hom
sets. This is not a relation because
Hom
takes values in sets, not
propositions; Thus the quotient forgets precisely how objects
are connected. This is intentional!
: Precategory o h → Set (o ⊔ h)
π₀ = el (Ob C / Hom C) squash π₀ C
The π₀
construction extends to a
functor Π₀
(capital pi for
Pieces) from Strict-cats
back to Sets
. We send a functor
to its object part, but postcomposing with the map inc
which sends an object of
to the connected component it inhabits.
: Functor (Strict-cats o h) (Sets (o ⊔ h))
Π₀ .F₀ (C , _) = π₀ C
Π₀ .F₁ F =
Π₀ (λ _ → squash) (λ x → inc (F .F₀ x))
Quot-elim λ x y r → glue (_ , _ , F .F₁ r)
We must prove that this assignment respects the quotient, which is where the morphism part of the functor comes in: Two objects are in the same connected component if there is a map To show that and are also in the same connected component, we must give a map but this can be canonically chosen to be
.F-id = funext (Coeq-elim-prop (λ _ → squash _ _) λ x → refl)
Π₀ .F-∘ f g = funext (Coeq-elim-prop (λ _ → squash _ _) λ x → refl) Π₀
The adjunction unit
is a natural
assignment of functors
We send
to its connected component, and we must send a map
to an equality between the connected components of
and
But we get this from the quotient.
: Π₀ ⊣ Disc {ℓ}
Π₀⊣Disc = adj where
Π₀⊣Disc : _ ⊣ _
adj .unit .η x = Disc-into _ inc quot
adj .unit .is-natural x y f = Functor-path (λ x → refl) λ _ → squash _ _ _ _ adj
The adjunction counit
is an
assignment of functions
This is essentially a natural isomorphism: the set of connected
components of a discrete category is the same set we started with.
.counit .η X = Quot-elim (λ _ → X .is-tr) (λ x → x) λ x y r → r
adj .counit .is-natural x y f = trivial! adj
The triangle identities are again straightforwardly checked.
.zig {x} = trivial!
adj .zag = Functor-path (λ x → refl) λ f → refl adj
Furthermore, we can prove that the connected components of a product category are product sets of connected components.
Π₀-preserve-prods: ∀ {C D : Precategory o h} → π₀ ʻ (C ×ᶜ D) ≡ π₀ ʻ C × π₀ ʻ D
{C = C} {D = D} = Iso→Path (f , isom) where
Π₀-preserve-prods open is-iso
We have a map splitting of the product category onto of each factor. This maps respect the quotient because we can also split the morphisms.
: π₀ ʻ (C ×ᶜ D) → π₀ ʻ C × π₀ ʻ D
f = Quot-elim
f (λ _ → hlevel 2)
(λ (a , b) → inc a , inc b)
λ (x , x') (y , y') (f , g) i →
(x , y , f) i , glue (x' , y' , g) i glue
This map has an inverse given by joining up the pairs:
: is-iso f
isom .inv (a , b) = Coeq-rec₂ squash (λ x y → inc (x , y))
isom (λ a (x , y , r) i → glue ((x , a) , (y , a) , r , Precategory.id D) i)
(λ a (x , y , r) i → glue ((a , x) , (a , y) , Precategory.id C , r) i)
a b
.rinv = elim! λ x y → refl
isom .linv = elim! λ x y → refl isom
Pieces have points🔗
An important property of the cohesive quadruple defined above is that the canonically-defined natural morphism is surjective, i.e. each piece has at least one point.
: Γ {ℓ} {ℓ} => Π₀
Points→Pieces .η _ x = inc x
Points→Pieces .is-natural x y f i o = inc (f .F₀ o)
Points→Pieces
: ∀ {x} → is-surjective (Points→Pieces {ℓ} .η x)
pieces-have-points = elim! λ x → inc (x , refl) pieces-have-points