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

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