Sum types, also called discriminated unions, are a powerful concept in TypeScript that allow you to model data with a finite set of possible shapes.
You can make sum types more concise and maintainable by using a a utility type.
If you're already using discriminated unions in TypeScript, you might be adding
a type
property to each variant. While this works, it can become repetitive
and error-prone as your codebase grows. A helper type like SumType
can
automate this process, making your code more concise and maintainable.
Introducing SumType
The SumType
utility type transforms a record of object types into a
discriminated union. Here's the implementation:
export const type = Symbol("type");
export type SumType<T extends Record<string | symbol, object>> = Simplify<
{
[K in keyof T]: { [type]: K } & T[K];
}[keyof T]
>;
type Simplify<T> = { [KeyType in keyof T]: T[KeyType] } & {};
This utility automatically adds a unique type
property to each object in the
record, eliminating the need to define it manually.
Why use a symbol for the type property?
Use a unique symbol for the type
property to avoid potential conflicts with
other properties. This approach ensures that the type
property is distinct and
doesn't interfere with a type property that might already exist on the object.
You can use ["type"]
in place of [type]
if you prefer using a string, but I
recommended to use a symbol.
Example: Refactoring Shapes
Imagine you have some shapes defined as a discriminated union, you would typically write it like this:
type Shape =
| { type: "rectangle"; width: number; length: number }
| { type: "circle"; radius: number };
Using SumType
, you can refactor this to the following:
type Shape = SumType<{
rectangle: { width: number; length: number };
circle: { radius: number };
}>;
- Adding a new shape is as simple as adding a new property to the record.
- Missing the
type
property will result in a compile-time error. - Duplicate
type
properties will also be caught.
You can also use unique symbols for each type to avoid conflicts between different sum types.
const rectangle = Symbol("rectangle");
const circle = Symbol("circle");
type Shape = SumType<{
[rectangle]: { width: number; length: number };
[circle]: { radius: number };
}>;
Exhaustive Checks Made Easy
With SumType
, exhaustive checks in switch
statements remain straightforward.
Here's an example:
class ExhaustiveError extends Error {
constructor(value: never) {
super(`Exhaustive check failed for value: ${value}`);
this.name = "ExhaustiveError";
}
}
function area(shape: Shape): number {
switch (shape[type]) {
case "rectangle":
return shape.width * shape.length;
case "circle":
return Math.PI * shape.radius ** 2;
default:
throw new ExhaustiveError(shape[type]);
}
}
The type
property added by SumType
ensures that TypeScript enforces
exhaustive checks, helping you catch missing cases at compile time. Or you can
also use your favourite pattern matching library such as
ts-pattern or
arktype
ExhaustiveError
takes a type never
as an argument, which ensures that the
error is thrown only when all cases are exhausted. This approach helps you catch
missing cases early and avoid runtime errors while keeping your.
Thanks for reading, and consider refactoring your code to use this pattern for a
cleaner and more scalable implementation of sum types in TypeScript. You are
probaly already using sum types in your codebase, so why not make it more
concise and maintainable with SumType
?