Software Development

The Hidden TypeScript Hack You Need to Know

How to Loosely Union Literal Strings

The Hidden TypeScript Hack You Need to Know
This image has been generated by AI. <source>

Hey there TypeScript enthusiasts.

I will not waste your time with a long introduction today, let's get straight to it!

1. The Trick

If you have been using TypeScript for some time now, you will have likely come across union types.

Union types take this form:

type Fruit = "apple" | "orange"

Now if you annotate a variable with the Fruit type, you will only be able to assign it either apple or orange.

type Fruit = "apple" | "orange"

// Can only be "apple" or "orange"
const fruit: Fruit = "apple"

This is cool, but there are other fruits out there like bananas, berries, and tomatoes.

And please don't debate this in the comments, a tomato is a fruit not a vegetable.

Obviously, extending the the Fruit type for every new fruit is not feasible, and therefore we need another solution.


You may be tempted to just union the Fruit type with string as follows:

type Fruit = "apple" | "orange" | string

And you may expect this type to include apple, orange, and every other fruit.

Wrong.

This actually evaluates the type of Fruit as string, and therefore you will lose your typesafety for any existing fruits!

// Not what we want, but this is what happens.
type Fruit = string

So how do we solve this problem? Let's find out.

2. Using The Trick

To make this trick as clear as possible, let's use a contextual example.

In this example, we are defining a function that takes in a string argument representing an authentication provider you can use to sign into an application.

E.g. "google" represents Google sign in.

type Provider = "google" | "github"

async function authenticate(provider: Provider) {
  await signIn(provider)
}

// Only "google" or "github" is accepted.
await authenticate("google")

As the code says, we can only input "google" or "github", but what if we need to extend this function to accept google and github with typesafety, but also accept additional providers like resend or facebook.

Well, we have already discussed that adding | string is not the right approach, so what is the right approach?

It may seem awkward, but we will go through it in detail.

Here is the code to solve this problem:

type Provider = "google" | "github" | (string & {})

Instead of adding union string, we add union string & {}. But why?

Well typescript's type checker is bad at distinguishing between the string type and literal string types (e.g. "apple").

On the other hand, string & {} is a type that represents all possible strings except null or undefined values, hence the intersection with the {} type.

In other words, string & {} is a more stricter version of string, and therefore will preserve the typesafety for any existing literal strings, but still accept additional strings, similar to a "catch-all" type.

Here is the updated authenticate function demonstrating the application of this trick:

// The new "(string & {}) type acts like a catch-all"
// The typesafety is preserved!
type Provider = "google" | "github" | (string & {})

async function authenticate(provider: Provider) {
  await signIn(provider)
}

// Typesafety for "google" and "github", but any other string is also welcome.
// E.g. Entering "resend" here works and doesn't throw an error!
await authenticate("resend")

Conclusion

This trick is very useful if you are incrementally adopting new authentication providers into your web application, and you don't want to update the type definition each time.

Of course, the most maintainable approach is to keep the type as exact as possible, therefore you shouldn't use this TypeScript trick religiously, since you will be slowly descending back into JavaScript territory.

But in certain cases, like the authentication provider example, this could be very useful for developer experience, and therefore has it's place in TypeScript.


If you enjoyed this article, please make sure to Subscribe, Clap, Comment and Connect with me today! 🌐
Want to ship code like a hacker? Visit Next Inject today!
2