Mantenga Type Guards válido después de la refactorización en TypeScript

Dec 04 2022
Los protectores de tipos genéricos ayudan a refactorizar el código con menos errores
TypeScript es un excelente lenguaje para escribir aplicaciones con controles de seguridad de tipo. Ampliar o refactorizar código en TypeScript es mucho más fácil que en JavaScript simple.

TypeScript es un excelente lenguaje para escribir aplicaciones con controles de seguridad de tipo. Ampliar o refactorizar código en TypeScript es mucho más fácil que en JavaScript simple.

TypeScript tiene una buena funcionalidad incorporada para el estrechamiento de la interfaz: Type Guards. Pero no siempre protegen contra errores durante la extensión/refactorización de código, especialmente si tiene un proyecto grande.

Descripción del problema

Por ejemplo, tenemos una tienda de té. Vendemos dos tipos de té: suelto y en bolsa. Mostramos todos los productos en una lista. Cada tipo de producto tiene un enlace humanizado, y en el nombre del producto queremos especificar la cantidad de gramos o la cantidad de bolsitas de té en un paquete.

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");
};

Solución del problema

El problema con nuestro Type Guard es que internamente verificamos la presencia de un campo en el objeto. Esta es una operación de JavaScript y no se escribe de ninguna manera.

Bueno, vamos a manejar eso. Propongo usar una función de fábrica para crear funciones de Type Guard basadas en los campos del modelo de datos.

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")