Se você já utilizou TypeScript, provavelmente já ouviu falar da keyword infer. Ela não é muito comum no dia-a-dia, mas a maioria das bibliotecas mais avançadas em algum momento vai utilizar o infer para algum tipo de operação.

Para entendermos completamente o infer, precisamos ter uma noção de como o TypeScript faz a asserção de tipos, e também a hierarquia desses tipos. Eu não vou entrar em detalhes sobre essas informações aqui agora, mas você pode achar vários conteúdos sobre isso na própria documentação do TS.

O infer é uma keyword que complementa o que chamamos de conditional typing, ou tipos condicionais, que é quanto temos uma inferencia de tipos, seguida de uma condição, por exemplo:

type NonNullable<T> = T extends null | undefined ? never : T

No exemplo anterior, estamos pegando um tipo e verificando se ele é uma extensão ou de null ou de undefined, ou seja, tipos que não resolvem para true, e ai estamos fazendo uma type condition para dizer: "Se o tipo for um desses você retorna never, caso contrário retorna o próprio tipo".

O infer permite irmos um pouco mais além do que estamos acostumados nesses modelos. A ideia é que podemos definir uma variável dentro da nossa inferência de tipo que pode ser usada ou retornada, é como se pudéssemos fazer um const tipo = <inferencia>.

Por exemplo, vamos olhar o utilitário nativo do TS chamado ReturnType, que pega uma função passada como parâmetro e retorna qual é o tipo do seu retorno:

type ReturnType<T> = T extends (...args: any[]) => infer R ? R : any

O que está acontecendo aqui é uma inferência condicional, já que o infer não pode ser utilizado fora de condicionais. Primeiro verificamos se o tipo passado estende uma assinatura de função, se sim, vamos jogar o retorno dessa função para uma variável que chamamos de R, e ai retorná-la.

Outro exemplo, é extrair o retorno de uma promise, como eu comentei aqui nesta thread, se formos pensar em como podemos fazer esse tipo, primeiro temos que verificar se o tipo passado é uma extensão do tipo Promise<T>, e depois inferir T para retorná-lo, caso contrário, retornamos never:

type Unpromise<P> = P extends Promise<infer T> ? T : never

Outros casos de uso

Podemos usar o infer em uma série de casos, os mais comuns são:

  • Obter o primeiro parâmetro de uma função:
type FirstArgument<T> = T extends (first: infer F, ...args: any[]) => any ? F : never
  • Obter o tipo de um array
type ArrayType<T> = T extends (infer A)[] ? A : T
  • Recursivamente obter o tipo de uma função até achar seu tipo final
type ExtractType<T> = T extends Promise<infer R>
  ? R
  : T extends (...args: any[]) => any
    	? ExtractType<ReturnType<T>>
		: T