module Data.Partial.Base where

The partiality monad🔗

The partiality monad is an approach to modelling partial computations in constructive type theory — computations which may fail. The meaning of failure depends on the specific computation: for example, it might be non-terminating, or undefined on a given input. The partiality monad abstracts away from these details to arrive at the general notion of partial elements.

A partial element of a type consists of a predicate and a function We refer to as the extent of definition of the partial element, since its function is to record precisely where a given element is defined. The function contained within produces an element on that extent.

record  {} (A : Type ) : Type  where
  no-eta-equality
  field
    def : Ω
    elt :  def   A
\ Warning

This notion of partial element is not to be confused with the partial elements which feature in the syntax of cubical type theory!

We note that, while each element of stores a proposition — and, therefore, a type — the type of partial lives in the same universe level as itself. This is because of our standing assumption of propositional resizing. Were it not for this assumption, we would additionally need to quantify over a level bounding the size of the extents, generating an type of elements.

Partial elements enjoy the following extensionality principle: whenever they are defined on equivalent extents, and they agree whenever defined.

part-ext
  : {x y :  A} (to :  x    y ) (of :  y    x )
   (∀ xd yd  x .elt xd  y .elt yd)
   x  y
part-ext to from p = Part-pathp refl
  (Ω-ua (biimp to from)) (funext-dep λ _  p _ _)

To close the initial definitions, if we have a partial element and a total then we write for the relation is defined and :

_↓_ :  A  A  Type _
x  y = Σ[ d  x ] (x .elt d  y)

The information ordering🔗

The information ordering on partial elements embodies the idea that is a computation valued in If we have two such computations there is an emergent notion of one having made more progress than the other — for example, if we were to model them as Turing machines, one might have literally taken more steps than the other.

We write the information ordering as it holds whenever is more defined than and they agree on this common extent.

record _⊑_ {} {A : Type } (x y :  A) : Type  where
  no-eta-equality
  field
    implies :  x    y 
    refines :  xd  y .elt (implies xd)  x .elt xd

At the bottom element in the information order, we have the computation which never succeeds.

never :  A
never .def = ⊥Ω

Monadic structure🔗

We claimed above that the type of partial elements is a monad: equipped with the information order, it’s actually the free pointed directed-continuous partial order on a set. However, it’s also a monad in the looser sense of supporting do notation.

We can define the necessary components here, without getting out the entire adjoint functor machinery. First, we note that function composition lets us extend to

part-map : (A  B)   A   B
part-map f x .def = x .def
part-map f x .elt = f  x .elt

Then, we can define the transformation and the “extension” operator part-bind. This latter definition highlights an important feature of it is closed under dependent sums. A type of propositions closed under dependent sums is called a dominance.

always : A   A
always a .def   = ⊤Ω
always a .elt _ = a

part-bind :  A  (A   B)   B
part-bind x f .def .∣_∣   = Σ[ px  x ] (f ʻ x .elt px)
part-bind x f .def .is-tr = hlevel 1
part-bind x f .elt (px , pfx) = f (x .elt px) .elt pfx

We note that we can also lift the operation of function application to the application of partial functions to partial arguments: we get a result whenever they are both defined.

part-ap :  (A  B)   A   B
part-ap f x .def .∣_∣      =  f  ×  x 
part-ap f x .def .is-tr    = hlevel 1
part-ap f x .elt (pf , px) = f .elt pf (x .elt px)

What about Maybe?🔗

In functional programming circles, it’s common to define a partial function to be a total function The Maybe type is certainly simpler than our : so why not use that?

The answer lies, as it so often does, in constructivism. The functions are precisely those for which it is decidable whether or not is defined, for each But if we do not accept the law of excluded middle, then using the proper partiality monad gains us actual generality.

We can prove that the type Maybe A is equivalent to the subtype of ↯ A with decidable extent, by writing simple functions back and forth:

↯-Maybe :  {}  Type   Type 
↯-Maybe A = Σ[ p   A ] Dec  p 

maybe→↯-maybe : Maybe A  ↯-Maybe A
maybe→↯-maybe nothing  = never , auto
maybe→↯-maybe (just x) = always x , auto

↯-maybe→maybe : ↯-Maybe A  Maybe A
↯-maybe→maybe (x , yes x-def) = just (x .elt x-def)
↯-maybe→maybe (x , no ¬x-def) = nothing

maybe→↯→maybe : (x : Maybe A)  ↯-maybe→maybe (maybe→↯-maybe x)  x
maybe→↯→maybe nothing  = refl
maybe→↯→maybe (just x) = refl

↯→maybe→↯ : (x : ↯-Maybe A)  maybe→↯-maybe (↯-maybe→maybe x)  x
↯→maybe→↯ (x , yes a) = Σ-prop-path! $ part-ext  _  a)  _  tt)
   _ _  ↯-indep x)
↯→maybe→↯ (x , no ¬a) = Σ-prop-path! $ part-ext  ()) ¬a λ ()

However, we can not prove that this is a genuine generalisation. That is because constructive type theory is compatible with the law of excluded middle, but if we could internally demonstrate an undecidable proposition, it would contradict the law of excluded middle.