haspar.us

Refinement Types

Oct 07, 2019 · ☕ 5 min read

What are they?

TLDR: type + predicate = refinement type

We have a type called T and a predicate on T ( (x: T) => boolean ) called P.
We can say that P refines T into a type PT, where PT is a subtype of T containing all values x of type T for which P(x) === true.

Sources

  • Wikipedia

    In type theory, a refinement type is a type endowed with a predicate which is assumed to hold for any element of the refined type.

  • Refinement Types for TypeScript

    A basic refinement type is a basic type, e.g. number, refined with a logical formula from an SMT decidable logic.
    For example, the types:

    type nat = { v: number | 0 ≤ v }
    type pos = { v: number | 0 < v }
    type natN<n> = { v: nat | v = n }
    type idx<a> = { v: nat | v < len(a) }

    describe (the set of values corresponding to) non-negative numbers, positive numbers, numbers equal to some value n, and valid indexes for an array a, respectively…

Refinement types are not supported on language level by TypeScript, but they’re a concept from type theory — They’re just math and because of it we can use them and reap the benefits with just a little hack effort.

How can I use them?
What will I gain?

I’ve refined the number into Even during my Wrocław TypeScript talk.
Having a type for even numbers doesn’t sound really useful but the idea stays the same for real world types with complex names copied from Jira.

We can call the predicate once and if it’s true, we remember this fact in the type system to avoid accidental errors.

Example time

Almost real example

Assume we’re building a form in which users can enter a list of emails into a textarea to send activation links to give their friends write permissions for the app (let’s say it’s a CMS).

We validate these emails once, and if they’re all valid, we save them, and later, few steps further in the form, we send our activation links. Few days later, we get a new requirement — the user should be able to edit the data at the last step of the form. Cool, the form is pretty long, we understand that it can be useful. Several lines of code and we’re done. We go home happy about the good code we pushed. Next morning we see a new issue in the tracker — “Validate emails after the user does final edits” — We totally forgot about the validation.

How could we save the day with a refinement type?

  • Create a subtype of string called ValidEmail, such that string is not assignable to ValidEmail.
  • Hold already validated emails in a list of type ValidEmails.
  • Now you can’t push a string to a list of already validated emails ✨
  • Change the type of sendEmail function from (email: string) => void to (email: ValidEmail) => void*.
    It doesn’t make sense to send an email to "🦄🐵💣" which is a perfectly valid string.

( * ) or IO, Result, choose your favorite.

Yeah right, but how can I create this “ValidEmail” type?

However you want! It’s just an idea from type theory and you can implement it in your favorite way. Few ideas:

  • you can go full OOP and extend a String,
  • use nominal typing and leave no runtime trail (my favorite option),
  • put the string into a value object, because ValidEmail doesn’t even have to be a subtype of string.
    The key is that string is not assignable to ValidEmail, because we want to ensure validation.

User defined type guards

We can use TypeScript’s user defined type guards to tell the compiler that our predicate checks the type and this is exactly what we’re interested in when we talk about refinements.

Let’s empower our isValidEmail predicate by changing its signature from (s: string) => boolean to (s: string) => s is ValidEmail.

Takeaways

  • Refinements are not implemented in TypeScript, but you can make them in userspace.
  • You can use nominal typing to make sure your refinements have no runtime trail (except predicate checks).
  • You can use them to encode facts about your data in the type system.

Further reading

This is mostly a reading list for future me, but I hope you can also find it interesting.


Edited 11 times between Sep 30, 2019 and Nov 03, 2020.