module Data.Fin.Finite where

Finite types🔗

This module pieces together a couple of pre-existing constructions: In terms of the standard finite sets (which are defined for natural numbers and deloopings of automorphism groups, we construct the type of finite types. By univalence, the space of finite types classifies maps with finite fibres.

But what does it mean for a type to be finite? A naïve first approach is to define “ is finite” to mean “ is equipped with and ” but this turns out to be too strong: This doesn’t just equip the type with a cardinality, but also with a choice of total order. Additionally, defined like this, the type of finite types is a set!

naïve-fin-is-set : is-set (Σ[ X  Type ] Σ[ n  Nat ] Fin n  X)
naïve-fin-is-set = Equiv→is-hlevel 2 Σ-swap₂ $
  Σ-is-hlevel 2 (hlevel 2) λ x  is-prop→is-hlevel-suc {n = 1} $
    is-contr→is-prop $ Equiv-is-contr (Fin x)

That’s because, as the proof above shows, it’s equivalent to the type of natural numbers: The type

is equivalent to the type

and univalence says (rather directly) that the sum of as ranges over a universe is contractible, so we’re left with the type of natural numbers.

This simply won’t do: we want the type of finite sets to be equivalent to the (core of the) category of finite sets, where the automorphism group of has elements, not exactly one element. What we do is appeal to a basic intuition: A groupoid is the sum over its connected components, and we have representatives for every connected component (given by the standard finite sets):

Fin-type : Type (lsuc lzero)
Fin-type = Σ[ n  Nat ] BAut (Fin n)

Fin-type-is-groupoid : is-hlevel Fin-type 3
Fin-type-is-groupoid = Σ-is-hlevel 3 (hlevel 3) λ _ 
  BAut-is-hlevel (Fin _) 2 (hlevel 2)

Informed by this, we now express the correct definition of “being finite”, namely, being merely equivalent to some standard finite set. Rather than using Σ types for this, we can set up typeclass machinery for automatically deriving boring instances of finiteness, i.e. those that follow directly from the closure properties.

record Finite {} (T : Type ) : Type  where
  constructor fin
  field
    {cardinality} : Nat
    enumeration   :  T  Fin cardinality 
  Finite→is-set : is-set T
  Finite→is-set =
    ∥-∥-rec (is-hlevel-is-prop 2)  e  Equiv→is-hlevel 2 e (hlevel 2)) enumeration

  instance
    Finite→H-Level : H-Level T 2
    Finite→H-Level = basic-instance 2 Finite→is-set

open Finite  ...  using (cardinality; enumeration) public
open Finite using (Finite→is-set) public

instance opaque
  H-Level-Finite :  {} {A : Type } {n : Nat}  H-Level (Finite A) (suc n)
  H-Level-Finite = prop-instance {T = Finite _} λ where
    x y i .Finite.cardinality  ∥-∥-out!
       Fin-injective (⦇  x .enumeration e⁻¹  ∙e y .enumeration ⦈) 
      i
    x y i .Finite.enumeration  is-prop→pathp
      {B = λ i   _  Fin (∥-∥-out!  Fin-injective (⦇  x .enumeration e⁻¹  ∙e y .enumeration ⦈)  i) }
       _  squash)
      (x .enumeration) (y .enumeration) i

Finite→Discrete :  {} {A : Type }   Finite A   Discrete A
Finite→Discrete {A = A}  f  {x} {y} = rec! go (f .enumeration) where
  open Finite f using (Finite→H-Level)
  go : A  Fin (f .cardinality)  Dec (x  y)
  go e with Equiv.to e x ≡? Equiv.to e y
  ... | yes p = yes (Equiv.injective e p)
  ... | no ¬p = no λ p  ¬p (ap (e .fst) p)

Dec→Finite :  {} {A : Type }  is-prop A  Dec A  Finite A
Dec→Finite ap d = fin (inc (Dec→Fin ap d .snd e⁻¹))

Discrete→Finite≡ :  {} {A : Type }  Discrete A  {x y : A}  Finite (x  y)
Discrete→Finite≡ d = Dec→Finite (Discrete→is-set d _ _) d

Finite-choice
  :  { ℓ'} {A : Type } {B : A  Type ℓ'}
    Finite A 
   (∀ x   B x )   (∀ x  B x) 
Finite-choice {B = B}  fin {sz} e  k = do
  e  e
  choose  finite-choice sz λ x  k (equiv→inverse (e .snd) x)
  pure $ λ x  subst B (equiv→unit (e .snd) x) (choose (e .fst x))

Finite-≃ :  { ℓ'} {A : Type } {B : Type ℓ'}   Finite A   A  B  Finite B
Finite-≃  fin {n} e  e' = fin (∥-∥-map (e' e⁻¹ ∙e_) e)

equiv→same-cardinality
  :  { ℓ'} {A : Type } {B : Type ℓ'}  fa : Finite A   fb : Finite B 
    A  B   fa .Finite.cardinality  fb .Finite.cardinality
equiv→same-cardinality  fa   fb  e = ∥-∥-out! do
  e  e
  ea  fa .Finite.enumeration
  eb  fb .Finite.enumeration
  pure (Fin-injective (ea e⁻¹ ∙e e ∙e eb))

same-cardinality→equiv
  :  { ℓ'} {A : Type } {B : Type ℓ'}  fa : Finite A   fb : Finite B 
   fa .Finite.cardinality  fb .Finite.cardinality   A  B 
same-cardinality→equiv  fa   fb  p = do
  ea  fa .Finite.enumeration
  eb  fb .Finite.enumeration
  pure (ea ∙e path→equiv (ap Fin p) ∙e eb e⁻¹)

module _ { ℓ'} {A : Type } {B : Type ℓ'}  fb : Finite B 
  (e :  A  B ) (f : A  B) where

  Finite-injection→equiv : injective f  is-equiv f
  Finite-injection→equiv inj = ∥-∥-out! do
    e  e
    eb  fb .Finite.enumeration
    pure
      $ equiv-cancell (eb .snd)
      $ equiv-cancelr ((eb e⁻¹ ∙e e e⁻¹) .snd)
      $ Fin-injection→equiv _
      $ Equiv.injective (eb e⁻¹ ∙e e e⁻¹)  inj  Equiv.injective eb

  Finite-surjection→equiv : is-surjective f  is-equiv f
  Finite-surjection→equiv surj = ∥-∥-out! do
    e  e
    eb  fb .Finite.enumeration
    pure
      $ equiv-cancell (eb .snd)
      $ equiv-cancelr ((eb e⁻¹ ∙e e e⁻¹) .snd)
      $ Fin-surjection→equiv _
      $ ∘-is-surjective (is-equiv→is-surjective (eb .snd))
      $ ∘-is-surjective surj
      $ is-equiv→is-surjective ((eb e⁻¹ ∙e e e⁻¹) .snd)

private variable
   : Level
  A B : Type 
  P Q : A  Type 
instance
  Finite-Fin :  {n}  Finite (Fin n)
  Finite-⊎ :  Finite A    Finite B   Finite (A  B)
  Finite-Maybe :  fa : Finite A   Finite (Maybe A)

  Finite-Σ
    : {P : A  Type }   Finite A     {x}  Finite (P x)   Finite (Σ A P)
  Finite-Π
    : {P : A  Type }   Finite A     {x}  Finite (P x)   Finite (∀ x  P x)

  Finite-⊥ : Finite 
  Finite-⊤ : Finite 
  Finite-Bool : Finite Bool

  Finite-PathP
    :  {A : I  Type }  s : Finite (A i1)  {x y}
     Finite (PathP A x y)

  Finite-Lift :  {}   Finite A   Finite (Lift  A)
Finite-Fin = fin (inc (_ , id-equiv))

Finite-⊎ {A = A} {B = B} = fin $ do
  aeq  enumeration {T = A}
  beq  enumeration {T = B}
  pure (⊎-ap aeq beq ∙e Finite-coproduct)

Finite-Maybe {A = A} = fin do
  an  enumeration {T =   A}
  pure (Maybe-is-sum ∙e an)

Finite-Π {A = A} {P = P}  afin   pfin  = ∥-∥-out! do
  aeq  afin .Finite.enumeration
  let
    module aeq = Equiv aeq
    bc : Fin (afin .Finite.cardinality)  Nat
    bc x = pfin {aeq.from x} .Finite.cardinality
  pure $ fin do
    t  Finite-choice λ x  pfin {x} .Finite.enumeration
    pure (Π-cod≃ t ∙e Π-dom≃ aeq.inverse ∙e Finite-product bc)

Finite-Σ {A = A} {P = P}  afin   pfin  = ∥-∥-out! do
  aeq  afin .Finite.enumeration
  let
    module aeq = Equiv aeq
    bc : Fin (afin .Finite.cardinality)  Nat
    bc x = pfin {aeq.from x} .Finite.cardinality
  pure $ fin do
    t  Finite-choice λ x  pfin {x} .Finite.enumeration
    pure (Σ-ap-snd t ∙e Σ-ap-fst aeq.inverse e⁻¹ ∙e Finite-sum bc)

Finite-⊥ = fin (inc (Finite-zero-is-initial e⁻¹))
Finite-⊤ = fin (inc (is-contr→≃⊤ Finite-one-is-contr e⁻¹))

Bool≃Fin2 : Bool  Fin 2
Bool≃Fin2 = Iso→Equiv enum where
  enum : Iso Bool (Fin 2)
  enum .fst false = 0
  enum .fst true = 1
  enum .snd .is-iso.inv i with fin-view i
  enum .snd .is-iso.inv _ | zero  = false
  enum .snd .is-iso.inv _ | suc _ = true
  enum .snd .is-iso.rinv i with fin-view i
  enum .snd .is-iso.rinv _ | suc fzero = refl
  enum .snd .is-iso.rinv _ | suc (fin (suc n)  forget p ⦄) = absurd (¬suc≤0 (≤-peel p))
  enum .snd .is-iso.rinv _ | zero  = refl
  enum .snd .is-iso.linv true = refl
  enum .snd .is-iso.linv false = refl

Finite-Bool = fin (inc Bool≃Fin2)

Finite-PathP = subst Finite (sym (PathP≡Path _ _ _)) (Discrete→Finite≡ Finite→Discrete)

Finite-Lift = Finite-≃ (Lift-≃ e⁻¹)
abstract instance
  Finite-Coeq :  { ℓ'} {A : Type } {B : Type ℓ'} {f g : A  B}  _ : Finite A   _ : Finite B   Finite (Coeq f g)