Mantenha os Type Guards válidos após a refatoração no TypeScript

Dec 04 2022
Protetores de tipo genérico ajudam a refatorar o código com menos bugs
TypeScript é uma ótima linguagem para escrever aplicativos com verificações de segurança de tipo. Estender ou refatorar o código em TypeScript é muito mais fácil do que em JavaScript simples.

TypeScript é uma ótima linguagem para escrever aplicativos com verificações de segurança de tipo. Estender ou refatorar o código em TypeScript é muito mais fácil do que em JavaScript simples.

O TypeScript tem uma ótima funcionalidade integrada para estreitamento de interface — Type Guards. Mas eles nem sempre protegem contra erros durante a extensão / refatoração do código, especialmente se você tiver um projeto grande.

Descrição do Problema

Por exemplo, temos uma loja de chá. Vendemos dois tipos de chá: a granel e ensacado. Mostramos todos os produtos em uma lista. Cada tipo de produto tem um link humanizado, e no nome do produto queremos especificar o número de gramas ou o número de saquinhos de chá em uma embalagem.

import React from "react";

type Tea = { id: number; name: string; price: number };

type LooseTea = Tea & { weight: number };
type BaggedTea = Tea & { size: number };

const isLooseTea = (x: Tea): x is LooseTea => "weight" in x;
const isBaggedTea = (x: Tea): x is BaggedTea => "size" in x;

const getLink = (x: Tea): string => {
  if (isLooseTea(x)) return `/loose-tea/${x.id}`;
  if (isBaggedTea(x)) return `/bagged-tea/${x.id}`;
  throw new Error("Unknown tea");
};

const getTitle = (x: Tea): string => {
  if (isLooseTea(x)) return `${x.name} / ${x.weight}g`;
  if (isBaggedTea(x)) return `${x.name} / ${x.size} teabags`;
  throw new Error("Unknown tea");
};

const TeaItems: React.FC<{ items: Tea[] }> = ({ items }) => {
  return (
    <ul>
      {items.map((x) => (
        <li key={x.id}>
          <a href={getLink(x)}>{getTitle(x)}</a>
        </li>
      ))}
    </ul>
  );
};

type BaggedTea = Tea & { bags: number; weightPerBag: number };

// before:
const getTitle = (x: Tea): string => {
  if (isLooseTea(x)) return `${x.name} / ${x.weight}g`;
  // Property 'size' does not exist on type 'BaggedTea'. ts(2339)
  if (isBaggedTea(x)) return `${x.name} / ${x.size} teabags`;
  throw new Error("Unknown tea");
};

// after:
const getTitle = (x: Tea): string => {
  if (isLooseTea(x)) return `${x.name} / ${x.weight}g`;
  if (isBaggedTea(x)) return `${x.name} / ${x.bags} teabags ~ ${x.weightPerBag}`;
  throw new Error("Unknown tea");
};

Solução do problema

O problema com nosso Type Guard é que internamente verificamos a presença de um campo no objeto. Esta é uma operação JavaScript e não é digitada de forma alguma.

Bem, vamos lidar com isso. Proponho usar uma função de fábrica para criar funções Type Guard com base nos campos do modelo de dados.

type AnyObject = Record<string, any>;

// https://stackoverflow.com/a/52991061
type RequiredKeys<T> = {
  [K in keyof T]-?: AnyObject extends Pick<T, K> ? never : K
}[keyof T];

const createShapeGuard = <T extends AnyObject>(...keys: RequiredKeys<T>[]) => {
  return (obj: unknown): obj is T => {
    if (typeof obj !== "object" || obj === null) return false;

    for (const key of (keys as string[])) {
      if (!(key in obj)) return false;
    }

    return true;
  };
};

const isLooseTea = createShapeGuard<LooseTea>("weight")
const isBaggedTea = createShapeGuard<BaggedTea>("bags", "weightPerBag")

type BaggedTea = Tea & { bags: number; bagWeight: number };

// Argument of type '"weightPerBag"' is not assignable to parameter of type 'RequiredKeys<BaggedTea>'.ts(2345)
const isBaggedTea = createShapeGuard<BaggedTea>("bags", "weightPerBag")