It is already well known in the frontend development community that adopting TypeScript is a good idea for (almost) every project that reaches a certain size. Advantages cited usually evolve around safety, clearly documented interfaces, finding errors before they go to production and being able to refactor safely.
While I totally agree that these are great points in favor of TypeScript, I think there is one advantage that is criminally underrated:
The ability to safely add code
Even though I firmly believe that deleting code is much more fun (and productive) than writing code, what we do most of the time, is adding code.
Adding new features. Adding enhancements to existing features. Making a feature a bit more customizable. After all, it is mostly what customers want.
So how come we never talk about how great TypeScript is, if used correctly, for adding things.
Yes, moving and renaming things and having the compiler tell you where you forgot something is great, but IDEs are catching up and are pretty good at these things for JavaScript files already. But no Editor will tell you that you forgot to handle a new branch in your switch statement when you add a new feature.
This is where exhaustive matching comes into play.
What is exhaustive matching?
Some languages, like OCaml
, F#
or scala
support the concept of pattern matching. It's a little bit like javascript's switch statement on steroids, as it allows matching a value not only against other values, but also against patterns.
Exhaustive matching basically means that the compiler, given that he knows all the possible states, can tell you when you are missing one state in your match. I will use scala
code for the examples since it's the language I am most familiar with.
sealed trait Shape
final case class Circle(radius: Int) extends Shape
final case class Rectangle(width: Int, height: Int) extends Shape
def renderShape(shape: Shape): String = {
shape match {
case _:Rectangle => "I am a Rectangle!"
}
}
Here, the compiler would complain with the following message:
match may not be exhaustive. It would fail on the following input: Circle
Great, so as a JavaScript developer, being aware of the default-case eslint rule, I'll just add a default case here and call it a day:
def renderShape(shape: Shape): String = {
shape match {
case _:Rectangle => "I am a Rectangle!"
case _ => "I'm a Circle"
}
}
The program works, and all the cases in the match are being taken care of, so no one complains. But what happens if we add another shape?
final case class Square(length: Int) extends Shape
def renderShape(shape: Shape): String = {
shape match {
case _:Rectangle => "I am a Rectangle!"
case _ => "I'm a Circle"
}
}
Right. The program will still work, but it will not work correctly. If we pass a Square to the renderShape method, it will identify as a Circle, which is certainly not what we would expect.
Sure, as long as the code is co-located, this might be a non-issue. You will see that you have to adapt the code right below.
But obviously, in a fairly large code base, you will have go to through all the usages, and it's easy to forget one. Being able to utilize compiler driven development (think: Fix everything that's red and then it is guaranteed to work) is of great help.
So here is how the fixed scala code would look:
def renderShape(shape: Shape): String = {
shape match {
case _:Rectangle => "I am a Rectangle!"
case _:Circle => "I'm a Circle"
case _:Square => "I'm a Square"
}
}
Notice how we just got rid of the default case completely. If we add a Triangle now, it will show us an error again.
How can we do this in TypeScript?
This is great and all, but TypeScript doesn't support pattern matching, so how are we supposed to do this in TS?
It turns out that the TypeScript compiler is actually pretty smart when it comes to matching exhaustively on union types.
This is best done with tagged unions, which just means a union where each member define a discriminator of a literal type:
type Circle = {
kind: 'circle'
radius: number
}
type Rectangle = {
kind: 'rectangle'
width: number
height: number
}
type Shape = Circle | Rectangle
const renderShape = (shape: Shape): string => {
switch (shape.kind) {
case 'circle':
return 'I am a circle'
}
}
In this example, the kind field serves as the discriminator: Every shape is uniquely identified by it.
With the above code, you should now see the following error:
Function lacks ending return statement and return type does not include 'undefined'.(2366)
Note that even if you remove the explicit return type, and if you have noImplicitReturns turned on in your tsconfig.json
, you will still get the error:
Not all code paths return a value.(7030)
So the compiler really wants to tell us that we forgot something here, which is great.
Again, we should not fall into the trap of adding a default case here. I would even disable the aforementioned eslint rule for TypeScript files, because I don't think it adds a lot that the compiler won't catch for us anyways.
The compiler will also narrow the type for us in the case block, so we will have access to shape.radius
inside the case 'circle'
, but not outside of it.
A small caveat seems to be that you cannot use object destructuring on the shape param. Even though all members of the union type contain a shape, TypeScript won't accept this:
const renderShape = ({ kind, ...shape }: Shape): string => {
switch (kind) {
case 'circle':
return `I am a circle with ${shape.radius}`
}
}
It is especially important to keep this in mind when working with React components, as their props tend to be destructured a lot.
So with all this in mind, our code would look like this:
const renderShape = (shape: Shape): string => {
switch (shape.kind) {
case 'circle':
return 'I am a circle'
case 'rectangle':
return 'I am a rectangle'
}
}
Typescript is happy with this, and we will get a compile time error when we add a new Shape ๐
Runtime caveats
Types don't exist at runtime - all the safety that we have only exists at compile time. This is no problem as long as we, with our 100% typescript codebase, are the only callers of that function. In the real world, this is sometimes not the case. We might have some untyped JavaScript code that calls our function, or we are not in control at all where our input comes from.
Let's assume for example that we call a rest service that delivers a couple of Shapes that we want to render, and we have established with the backend team that we will focus on Circle and Rectangle first and will add Square later. We will use React
to render our little app:
export const App = () => {
const [shapes, setShapes] = React.useState()
React.useEffect(() => {
getShapes().then(setShapes)
}, [])
if (!shapes) {
return <Loading />
}
return (
<Grid>
{shapes.map((shape) => (
<Shape {...shape} />
))}
</Grid>
)
}
const Shape = (props: Shape): JSX.Element => {
switch (props.kind) {
case 'circle':
return <Circle radius={props.radius} />
case 'rectangle':
return <Rectangle width={props.width} height={props.height} />
}
}
Great, this is future-proof, typescript will tell us what to do as soon as we add another Shape.
Here, you can see the whole App in action:
Undefined strikes back
But then, something else happens: The backend team is faster than expected ๐ฎ. Their sprint is running great, so they decide to implement the Square right away. It's a quick win on their part, and they ship a new minor version of the API.
Guess what happens to our little App?
It will die a horrible runtime death. A minor backend release makes our whole App crash because of this fancy typescript pattern ๐ข. This happens because now, we fall through our switch statement, and because we have no default branch, undefined
is returned. Undefined
is one of the few things that React
cannot render, so we die with the famous error:
Nothing was returned from render. This usually means a return statement is missing. Or, to render nothing, return null.
See it live:
Never to the rescue
In TypeScripts type system, never is it's bottom type. It denotes something that can never happen, for example, a function that always throws an Exception or has an infinite loop will return never.
How is this helpful?
If typescript narrows the type with each case in the switch statement, if all cases are covered, what remains must be of type never. We can assert that with a little helper:
const UnknownShape = ({ shape }: { shape: never }) => <div>Unknown Shape</div>
const Shape = (props: Shape): JSX.Element => {
switch (props.kind) {
case 'circle':
return <Circle radius={props.radius} />
case 'rectangle':
return <Rectangle width={props.width} height={props.height} />
default:
return <UnknownShape shape={props} />
}
}
This approach has two advantages:
- It will not fail at runtime - it will still display all other shapes and display a little not-found helper for the newly added shape
- If we add Square to our Shape type, because we finally catch-up with the backend team and want to implement it as well, we will still get a compile error from TypeScript. You can see that here. This is because now, the type is not narrowed to never (since Square is still left), so the type of props for UnknownShape does not match.
Conclusion
Exhaustive matches are a great tool in any language to make your code safer for additions. When you have full control over the input, omitting the default branch seems like a good choice. If that is not the case, and because TypeScript is in the end just JavaScript, protection at runtime with a never guard is a good alternative.
Top comments (2)
I love exhaustive checking, excellent article! This also helped me have a better mental model of unknown and never types.
Thank you, I'm glad that you liked it :)