module Data.Set.Pushout where
Pushouts of sets🔗
private variable
: Level
ℓ ℓ' ℓ'' : Type ℓ
A B C : C → A
f g
pattern incl x = inc (inl x)
pattern incr x = inc (inr x)
The pushout of two sets can be, as usual, constructed as a coequaliser of maps into a coproduct. However, pushouts in the category have a nice property: they preserve embeddings. This module is dedicated to proving this.
Pushout: ∀ {ℓ ℓ' ℓ''} {A : Type ℓ} {B : Type ℓ'} {C : Type ℓ''}
→ (f : C → A) (g : C → B)
→ Type (ℓ ⊔ ℓ' ⊔ ℓ'')
{A = A} {B} {C} f g = Coeq {B = A ⊎ B} (λ z → inl (f z)) (λ z → inr (g z)) Pushout
: Pushout f g → Pushout g f
swap-pushout (incl x) = incr x
swap-pushout (incr x) = incl x
swap-pushout (glue x i) = glue x (~ i)
swap-pushout (squash x y p q i j) =
swap-pushout let f = swap-pushout in squash (f x) (f y) (λ i → f (p i)) (λ i → f (q i)) i j
: swap-pushout {f = f} {g = g} ∘ swap-pushout ≡ λ x → x
swap-swap = ext λ where
swap-swap (inl x) → refl
(inr x) → refl
: is-equiv (swap-pushout {f = f} {g = g})
swap-pushout-is-equiv = is-iso→is-equiv $
swap-pushout-is-equiv (happly swap-swap) (happly swap-swap) iso swap-pushout
Pushouts of injections🔗
module
_ {ℓ ℓ' ℓ''} {A : Type ℓ} {B : Type ℓ'} {C : Type ℓ''}
: H-Level B 2 ⦄
⦃ bset (f : C → A) (g : C → B) (f-emb : is-embedding f)
where
Let us now get to the proof of our key result. Suppose is an embedding and is some other map. Our objective is to prove that, in the square
the constructor is an embedding.1. As usual when working with higher inductive types, we’ll employ encode-decode, characterising the path spaces based at some by means of a family over We know already we want to be since this is exactly injectivity of But what should the fibre over be?
A possibility comes from generalising the generating equality from identifying to identifying any provided we can cough up with and While this is a trivial rephrasing, it does inspire confidence: can we expect to be given by fibres of over
Well, if this were the case, these fibres would necessarily need to be propositions, so we can start by showing that. The proof turns out very simple: the type is equivalent to and since is an embedding, this is a subtype of a proposition.
private module _ (b : B) where
: ∀ a → is-prop (Σ[ c ∈ C ] (f c ≡ a) × (g c ≡ b))
rem₁ (c , α , β) (c' , α' , β') = ap fst it ,ₚ ap snd it ,ₚ prop! where
rem₁ a = f-emb a (c , α) (c' , α') it
instance
: ∀ {a n} → H-Level (Σ[ c ∈ C ] (f c ≡ a) × (g c ≡ b)) (suc n)
rem₁' {a} = prop-instance (rem₁ a)
rem₁' {-# OVERLAPPING rem₁' #-}
We can then turn back to defining our family Both of the point cases are handled, and since we’re mapping into that also handles the truncation.
: Pushout f g → Prop (ℓ ⊔ ℓ' ⊔ ℓ'')
code (incr b') = el! (Lift (ℓ ⊔ ℓ'') (b ≡ b'))
code (incl a) = el! (Σ[ c ∈ C ] (f c ≡ a) × (g c ≡ b)) code
It remains to handle the paths. Since we’re mapping into a universe, it will suffice to show that the types and are equivalent, and, since we’re mapping into propositions, it will suffice to provide functions in either direction. Going backwards, this is trivial; In the other direction, we show since implies by injectivity of and we have by assumption.
(glue x i) =
code let
: Lift _ (b ≡ g x) → Σ[ c ∈ C ] (f c ≡ f x) × (g c ≡ b)
from (lift p) = x , refl , sym p
from
to : Σ[ c ∈ C ] (f c ≡ f x) × (g c ≡ b) → Lift _ (b ≡ g x)
to (x , α , β) = lift (sym β ∙ ap g (ap fst (f-emb _ (_ , α) (_ , refl))))
in n-ua
{X = el! (Σ[ c ∈ C ] (f c ≡ f x) × (g c ≡ b))}
{Y = el! (Lift (ℓ ⊔ ℓ'') (b ≡ g x))}
(prop-ext! to from) i
(squash x y p q i j) = n-Type-is-hlevel 1
code (code x) (code y) (λ i → code (p i)) (λ i → code (q i)) i j
We then have the two encoding functions. In the case with matched
endpoints, we have exactly our original goal: injectivity of incr
.
: injective incr
encodeᵣ {a} p = transport (λ i → ⌞ code a (p i) ⌟) (lift refl) .lower encodeᵣ
: is-embedding {A = B} {B = Pushout f g} incr
f-inj→incr-inj = injective→is-embedding! encodeᵣ f-inj→incr-inj
In the mismatched case, we have a function turning paths into a fibre of over It’s easy to show that projecting the point is the inverse to considered as a function
encodeₗ: ∀ {a b} (p : incr b ≡ incl a)
→ Σ[ c ∈ C ] (f c ≡ a × g c ≡ b)
{b = b} p = transport (λ i → ⌞ code b (p i) ⌟) (lift refl)
encodeₗ
: ∀ x → fibre f x ≃ fibre {B = Pushout f g} incr (incl x)
pushout-is-pullback' .fst (y , p) = g y , sym (glue y) ∙ ap incl p
pushout-is-pullback' x .snd = is-iso→is-equiv $ iso from ri λ f → f-emb x _ _ where
pushout-is-pullback' x : fibre {B = Pushout f g} incr (incl x) → fibre f x
from (y , p) with (c , α , β) ← encodeₗ p = c , α
from
: is-right-inverse from (pushout-is-pullback' x .fst)
ri (y , p) with (c , α , β) ← encodeₗ p = Σ-prop-path! β ri
Finally, we can extend this to an equivalence between and the pullback of along by general family-fibration considerations.
_ : C ≃ (Σ[ a ∈ A ] Σ[ b ∈ B ] incl a ≡ incr b)
_ = C ≃⟨ Total-equiv f ⟩
Σ[ a ∈ A ] fibre f a ≃⟨ Σ-ap-snd pushout-is-pullback' ⟩(incl a) ≃⟨⟩
Σ[ a ∈ A ] fibre incr (incr b ≡ incl a) ≃⟨ Σ-ap-snd (λ a → Σ-ap-snd λ b → sym-equiv) ⟩
Σ[ a ∈ A ] Σ[ b ∈ B ] (incl a ≡ incr b) ≃∎ Σ[ a ∈ A ] Σ[ b ∈ B ]
-- That equivalence computes like garbage on the third component, and
-- we can do better:
: C ≃ (Σ[ a ∈ A ] Σ[ b ∈ B ] incl a ≡ incr b)
pushout-is-pullback .fst x = f x , g x , glue x
pushout-is-pullback .snd = is-iso→is-equiv (iso from ri λ x → refl) where
pushout-is-pullback module _ (x : _) (let β = x .snd .snd) where
: C
from = encodeₗ (sym β) .fst
from
: (f from , g from , glue from) ≡ (_ , _ , β)
ri = encodeₗ (sym β) .snd .fst ,ₚ encodeₗ (sym β) .snd .snd ,ₚ prop!
ri
module _ ⦃ aset : H-Level A 2 ⦄ (f : C → A) (g : C → B) (g-emb : is-embedding g) where
: is-embedding {A = A} {B = Pushout f g} incl
g-inj→incl-inj = ∙-is-embedding
g-inj→incl-inj (f-inj→incr-inj g f g-emb)
(is-equiv→is-embedding swap-pushout-is-equiv)
In this situation, the map is often referred to as the cobase change of along ↩︎