Matt PocockMatt is a well-regarded TypeScript expert known for his ability to demystify complex TypeScript concepts.
any is an extremely powerful type in TypeScript. It lets you treat a value as if you were in JavaScript, not TypeScript. This means that it disables all of TypeScript's features - type checking, autocomplete, and safety.
constmyFunction = (input: any) => {input.someMethod();};myFunction("abc"); // This will fail at runtime!
Using any is rightly considered harmful by most of the community. There are ESLint rules to prevent its use. This can turn developers off using any entirely.
However, there are a few advanced cases where any is always the right choice. Here are some of them:
Let's imagine we wanted to implement the ReturnType utility in TypeScript. This utility takes a function type and returns the type of its return value.
We need to create a generic type which takes a function type as a type argument. If we restricted ourselves to not use any, we might use unknown:
typeReturnType<T extends (...args: unknown[]) => unknown> = // Not important for our explanation:T extends (...args: unknown[]) => inferR ? R : never;
It's not important to understand all of this code, only the constraint - T extends (...args: unknown[]) => unknown. What we're saying here is that only functions which accept an arguments array of unknown[] and return unknown are allowed.
It seems to work fine for functions which have no arguments:
Type '(input: string) => void' does not satisfy the constraint '(...args: unknown[]) => unknown'.
Types of parameters 'input' and 'args' are incompatible.
Type 'unknown' is not assignable to type 'string'.2344
Type '(input: string) => void' does not satisfy the constraint '(...args: unknown[]) => unknown'.
Types of parameters 'input' and 'args' are incompatible.
Type 'unknown' is not assignable to type 'string'.
In fact, it only works if we change the parameter of our function to input: unknown:
So accidentally, we've created a ReturnType function that only works on functions which accept unknown as an argument. This is not what we wanted. We wanted it to work on any function.
The solution is to use any[] as the type argument constraint:
Now it works as expected. We're declaring that we don't care what types the function accepts - it could be anything.
The reason this is safe is because we're deliberately declaring a wide type. We're saying "I don't care what the function accepts, as long as it's a function". This is a safe use of any.
Returning Conditional Types From Generic Functions
In some places, TypeScript's narrowing abilites are not as good as we'd like them to be. Let's say we want to create a function which returns different types based on a condition:
This function isn't really doing what we want it to. We want it to return the type "goodbye" when we pass in "hello". But currently, result is typed as "hello" | "goodbye".
We've added a conditional type to the return type of the function which mirrors our runtime logic. If TInput, inferred from the runtime argument input, is "hello", we return "goodbye". Otherwise, we return "hello".
But there's a problem. I've deliberately disabled the errors in the snippet above. Let's see what happens when we enable them:
Yes, this does make our function less type-safe. We could accidentally return "bonsoir" from the function instead.
But in these situations, it's often better to use as any and add a unit test for this function's behavior. Because of TypeScript's limitations in checking this stuff, this is often as close as you'll get to type safety.
There are several other use cases like this, where inside generic functions you need to use any to get around TypeScript's limitations. To me, this is fine.
A question remains: should you ban any from your codebase? I think, on balance, the answer should be yes. You should turn on the ESLint rule which prevents its use, and you should avoid it wherever possible.
However, there are cases where any is needed. They're worth using eslint-disable to get around them. So, bookmark this article, and attach it to your PR's when you feel the need to use it.
Have you spotted any other legitimate use cases for any? Let me know!