Всем привет! ?
⚠️ Возрождаю статью, мною написанную в 2021 на Хабре. Из нового - подкорректировал определения на ��усском, добавил куски кода, чтобы было удобно копировать. Английский оригинал тут - https://blog.beraliv.dev/2021-03-26-typed-get. Приятного чтения ?
type Get<O, P> = never; // реализовать type Step1 = Get< { article: { author: "Alexey Berezin"; keywords: ["typescript"] } }, "article.keywords.0" >; type Step2 = Get< { author: "Alexey Berezin"; keywords: ["typescript"] }, "keywords.0" >; type Step3 = Get<["typescript"], "0">; type Step4 = Get<"typescript", "">; type Result = "typescript";
В 2021 году я раскопал на просторах GitHub репозиторий type-challenges. У меня есть целый блог, где я решаю задачки оттуда, но сегодня я попытаюсь показать не только реализацию Get, но и продемонстрирую общие проблемы, покажу улучшения и использование в production.
Если перед началом чтения хочется ознакомиться с понятиями из TypeScript, которые требуются в данной статье, переходите в конец.
1. Базовая реализация
Текущий челлендж располагается в категории "сложное".
У нас есть объект и путь в этом объекте (реализация не предполагает, что пути в массиве и/или кортеже). Ожидается, что тип вернет корректный тип по пути в этом объекте.
Так с чего же начнем?
1.1. Получение ключей
Представим, если бы мы решали эту задачу с помощью JavaScript:
const get = (obj, path) => { const keys = path.split("."); return keys.reduce((currentObj, key) => currentObj[key], obj); };
Перед тем, как вызывать keys.reduce, мы получаем список всех ключей. В JavaScript нам достаточно вызвать метод split. В TypeScript нам тоже надо как-то получить список ключей из строки.
Благодаря TypeScript 4.1, мы можем использовать Template Literal types. С их помощью мы можем удалить точки между ключами. Давайте определим тип Path и попробуем сделать первый подход:
type Path<T> = T extends `${infer Key}.${infer Rest}` ? [Key, ...Path<Rest>] : [];
Выглядит коротко и просто. Однако после покрытия тестами мы поняли, что упустили случай с единственным элементом (без точки). Тесты написаны в Playground. Давайте добавим этот случай:
type Path<T> = T extends `${infer Key}.${infer Rest}` ? [Key, ...Path<Rest>] : T extends `${infer Key}` ? [Key] : [];
Так лучше. Тесты вместе с реализацией доступны в Playground.
1.2. Reducer для объекта
После того, как мы получили ключи, мы наконец-то можем вызватьkeys.reduce. Чтобы это сделать, давайте определим тип GetWithArray , имея уже путь в виде кортежа K
type GetWithArray<O, K> = K extends [infer Key, ...infer Rest] ? Key extends keyof O ? GetWithArray<O[Key], Rest> : never : never;
Немного прокомментирую:
K extends [infer Key, ...infer Rest]проверяет, что у нас есть хотя бы один элемент в кортежеKey extends keyof Oпозволяет использоватьO[Key]и рекурсивно переходит к следующему уровню объекта
Давайте протестируем это решение (ссылка на Playground). Опять мы забыли случай, правда уже когда у нас пустой массив. После добавления код выглядит так:
type GetWithArray<O, K> = K extends [] ? O : K extends [infer Key, ...infer Rest] ? Key extends keyof O ? GetWithArray<O[Key], Rest> : never : never;
Финальная версия с тестами в Playground
1.3. Все вместе
type Get<O, P> = GetWithArray<O, Path<P>>;
Давайте протестируем все вместе и удостоверимся, что тип работает как ожидается: Playground. Отлично, базовую часть мы закончили.
2. Опциональные пути
Когда работаешь с реальными данные в production, тебе иногда данные не приходят или приходят, но не полностью. Поэтому по всему проекту мы используем ? , null илиundefined.
Возьмем такой пример и покроем тестами текущее решение: Playground. Как и ожидалось, TypeScript ругается.
Причина проста. Давайте возьмем какой-нибудь пример и пошагово пройдемся:
type ProductionObject = { description?: string; title: string; date: string; author?: { name: string; location?: { city: string; }; }; }; type Step1 = Get<ProductionObject, "author.name">; type Step2 = Get< | { name: string; location?: | { city: string; } | undefined; } | undefined, "name" >; type Step3 = Get<never, "">; type Result = never; // ожидалось `string | undefined`
Текущее решение не позволяет извлекать ключ из объекта, который может быть undefined or null. Постараемся это решить.
2.1. Удаляем undefined, null или оба типа сразу
Сначала определим 3 вспомогательных типа:
type FilterUndefined<T> = T extends undefined ? never : T; type FilterNull<T> = T extends null ? never : T; type FilterUndefinedAndNull<T> = FilterUndefined<FilterNull<T>>;
Мы проверяем, что undefined и/или null являются частью union type, и если так, удаляем их из него. Это поможет работать с остальной "существенной" частью.
Тесты, как обычно, в Playground
2.2. Редактируем reducer
Давайте обновим вот эту ветку GetWithArray:
type GetWithArray<O, K> = K extends [] ? O : K extends [infer Key, ...infer Rest] ? Key extends keyof O ? GetWithArray<O[Key], Rest> : never // <= добавляем логику здесь ? : never;
Сначала проверим, что ключ существует в объекте с
undefinedи/илиnullВ противном случае, его нет (то есть мы возвращаем
undefined)
type GetWithArray<O, K> = K extends [] ? O : K extends [infer Key, ...infer Rest] ? Key extends keyof O ? GetWithArray<O[Key], Rest> : Key extends keyof FilterUndefinedAndNull<O> ? GetWithArray<FilterUndefinedAndNull<O>[Key], Rest> | undefined : undefined : never;
Добавим здесь тесты и проверим, что тип работает корректно (ссылка на Playground).
3. Получение пути из массива и кортежи
Аналогично берем пример с массивом и пошагово проверяем:
type ProductionObject = { posts: { title: string; description?: string; poster?: string; html: string; }[]; }; type Step1 = Get<ProductionObject, "posts.0">; type Step2 = Get< { title: string; description?: string; poster?: string; html: string; }[], "0" >; type Step3 = Get<undefined, "">; type Result = undefined; // ? ожидалось `string | undefined`
В JavaScript мы бы ходили по индексам:
const get = (arr, path) => { const keys = path.split("."); return keys.reduce((currentArr, key) => currentArr[key], arr); };
Несмотря на то, что ключ может быть string или number, Path оставляем неизменным:
type Path<T> = T extends `${infer Key}.${infer Rest}` ? [Key, ...Path<Rest>] : T extends `${infer Key}` ? [Key] : [];
3.1. Reducer для массива
Как и для объектов, для массивов мы вызываем keys.reduce . Для TypeScript нам надо написать реализацию аналогично GetWithArray . Давайте реализуем это отдельно для массивов, а затем объединим реализации GetWithArray в одно.
Сперва адаптируем тип для массивов и кортежа. Возьмем A вместо O по семантическим причинам:
type GetWithArray<A, K> = K extends [] ? A : K extends [infer Key, ...infer Rest] ? Key extends keyof A ? GetWithArray<A[Key], Rest> : never : never;
После тестирования в Playground, мы столкнулись с несколькими проблемами:
1. Массивы не имеют ключей с типом string :
type Step1 = GetWithArray<string[], "1">; type Step2 = "1" extends keyof string[] ? string[]["1"] : never; type Result = never; // ? ожидалось `string | undefined`
Здесь '1' extends keyof string[] всегда ложно, поэтому возвращает never.
2. Аналогично для массивов с ключевым словом readonly
type Step1 = GetWithArray<readonly string[], "1">; type Step2 = "1" extends keyof (readonly string[]) ? (readonly string[])["1"] : never; type Result = never; // ? ожидалось `string | undefined`
3. Кортежи (например [0, 1, 2]) возвращают never вместо undefined:
type Step1 = GetWithArray<[0, 1, 2], "3">; type Step2 = "3" extends keyof [0, 1, 2] ? [0, 1, 2]["3"] : never; type Result = never; // ? ожидалось `undefined`
Пойдем чинить все пошагово.
3.2. Выводим T | undefined
type GetWithArray<A, K> = K extends [] ? A : K extends [infer Key, ...infer Rest] ? Key extends keyof A ? GetWithArray<A[Key], Rest> : never // <= добавляем логику здесь ? : never;
Для массивов мы хотим получить T | undefined в качестве ответа (так как при извлечении по индексу мы не знаем, есть элемент или нет), в зависимости от значения T:
type GetWithArray<A, K> = K extends [] ? A : K extends [infer Key, ...infer Rest] ? Key extends keyof A ? GetWithArray<A[Key], Rest> : A extends readonly (infer T)[] ? GetWithArray<T | undefined, Rest> : never : never;
Я добавил A extends readonly (infer T)[] , т.к. для всех массивов (в том числе с ключевым слово readonly) это утверждение верно.
После проверки, нам остается починить кортежи. Пример с тестами доступен в Playground.
3.3. Кортежи
Если мы попробуем извлечь значение из несуществующего индекса, мы получим обобщающий тип, как для массивов (ну и еще undefined в придачу)
type Step1 = GetWithArray<[0, 1, 2], "3">; type Step2 = "3" extends keyof [0, 1, 2] // <= false ? [0, 1, 2]["3"] : [0, 1, 2] extends (infer T)[] // <= true ? T | undefined : [0, 1, 2] extends readonly (infer T)[] ? T | undefined : never; type Result = 0 | 1 | 2 | undefined; // ? ожидалось `undefined`
Для того, чтобы справиться с этой проблемой, я предлагаю построить табличку с extends для разных типов (назовем эту табличку ExtendsTable) и будем подбирать правильный условный тип, чтобы разграничить массивы и кортежи:
type ExtendsTableRow<T extends any[], E extends any> = { [K in keyof T]: E extends T[K] ? true : false; }; type ExtendsTable<T extends any[]> = { [K in keyof T]: ExtendsTableRow<T, T[K]>; };
Возьмем 4 разных типа:
[0]number[]readonly number[]any[]
type Result = ExtendsTable<[[0], number[], readonly number[], any[]]>;
Для лучшего отображения нарисую табличку, чтобы было понятно, что происходит:
|
|---|
Если на пересечении знак плюса "+", это значит, что строчка расширяема столбцом. Несколько примеров:
[0] extends [0]number[] extends readonly number[]
Соответственно, если на пересечении знак минуса"-", то значит, что строка не расширяется колонкой. Пару примеров:
number[] extends [0]readonly number[] extends number[]
Возьмем строку с any[]: для колонки [0] мы видим знак минуса "-", когда для остальных типов (столбцов) – это знак плюса "+".
Собственно, мы нашли ответ!
Мы возьмем этот условный тип any[] extends A и применим к GetWithArray:
type GetWithArray<A, K> = K extends [] ? A : K extends [infer Key, ...infer Rest] ? any[] extends A ? // массивы A extends readonly (infer T)[] ? GetWithArray<T | undefined, Rest> : undefined : // кортежи Key extends keyof A ? GetWithArray<A[Key], Rest> : undefined : never;
Мы различаем массив от кортежа с помощью условного типа
any[] extends AДля массивов мы используем
T | undefinedДля кортежей, мы извлекаем значение, если индекс для этого кортежа существует
В противном случае, мы возвращаем
undefined
Если хочется еще раз взглянуть на все текущие изменения, переходите на Playground.
4. Общее решение
На данный момент у нас есть решение для объектов:
type GetWithArray<O, K> = K extends [] ? O : K extends [infer Key, ...infer Rest] ? Key extends keyof O ? GetWithArray<O[Key], Rest> : Key extends keyof FilterUndefinedAndNull<O> ? GetWithArray<FilterUndefinedAndNull<O>[Key], Rest> | undefined : undefined : never;
и для массивов:
type GetWithArray<A, K> = K extends [] ? A : K extends [infer Key, ...infer Rest] ? any[] extends A ? // arrays A extends readonly (infer T)[] ? GetWithArray<T | undefined, Rest> : undefined : // tuples Key extends keyof A ? GetWithArray<A[Key], Rest> : undefined : never;
Определим два вспомогательных типа: ExtractFromObject и ExtractFromArray, где мы будем извлекать значение, зная, с какой структурой в данный момент работаем:
type ExtractFromObject< O extends Record<PropertyKey, unknown>, K > = K extends keyof O ? O[K] : K extends keyof FilterUndefinedAndNull<O> ? FilterUndefinedAndNull<O>[K] | undefined : undefined; type ExtractFromArray<A extends readonly any[], K> = any[] extends A ? A extends readonly (infer T)[] ? T | undefined : undefined : K extends keyof A ? A[K] : undefined;
Здесь пришлось добавлять "ограничения" (Generic Constrains):
Для
ExtractFromObject– этоO extends Record<PropertyKey, unknown>. Это значит, чтоOдолжен быть объектом любого видаДля
ExtractFromArrayаналогично:A extends readonly any[]принимает массив любого типа и кортежи
Добавим соответствующие условные типы в GetWithArray и объединим решения:
type GetWithArray<O, K> = K extends [] ? O : K extends [infer Key, ...infer Rest] ? O extends Record<PropertyKey, unknown> ? GetWithArray<ExtractFromObject<O, Key>, Rest> : O extends readonly any[] ? GetWithArray<ExtractFromArray<O, Key>, Rest> : undefined : never;
Это решение я тоже покрыл тестами. Ссылка на Playground.
5. Связка с JavaScript
Вернемся к решению в JavaScript:
const get = (obj, path) => { const keys = path.split("."); return keys.reduce((currentObj, key) => currentObj[key], obj); };
На данный момент мы используем lodash в нашем проекте, где есть функция get. Если вы выглянете на common/object.d.ts в@types/lodash, то немного огорчитесь. Во многих случаях вызов get возвращает any : typescript-lodash-types
Давайте заменим reduce на цикл с for (например for-of), чтобы была возможность сделать ранний выход из цикла с полученным значением, если оно undefined или null:
export function get = (obj, path) => { const keys = path.split("."); let currentObj = obj; for (const key of keys) { const value = currentObj[key]; if (value === undefined || value === null) { return undefined; } currentObj = value; } return currentObj; }
А теперь покроем эту функцию get типами, которые мы получили на предыдущих шагах. Разделим это на два случая:
Тип
Getможно использовать тогда и только тогда, когда все ограничения применимы и тип корректно выводитсяВ случае какой-то ошибки мы используем вторую сигнатуру (например, мы передали число вместо строки в качестве пути)
Чтобы использовать перегрузку, нам нужно использовать функцию с ключевым слово function, а не стрелочные функции:
export function get(obj: Record<string, unknown>, path: string): unknown { const keys = path.split("."); let currentObj = obj; for (const key of keys) { const value = currentObj[key]; if (value === undefined || value === null) { return undefined; } currentObj = value as Record<string, unknown>; } return currentObj; }
Почти готово. Осталось добавить тип Get :
// strict types ? export function get<O, K extends string>(obj: O, path: K): Get<O, K>; // fallback ? export function get(obj: Record<string, unknown>, path: string): unknown { const keys = path.split("."); let currentObj = obj; for (const key of keys) { const value = currentObj[key]; if (value === undefined || value === null) { return undefined; } currentObj = value as Record<string, unknown>; } return currentObj; }
Все вместе я разместил на Codesandbox:
Терминогия
Для решения задачи использовались следующие термины из TypeScript:
1. Кортежи представлены в TypeScript 1.3, но вариативный вариант (Variadic Tuple Types) был выпущен в версии 4.0, так что теперь можно использовать spread внутри кортежей
// Кортежи type Digits = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]; type Name = | [first: string, last: string] | [first: string, middle: string, last: string]; // Кортежи, называемые "variadic" type AnyArray = readonly any[]; type Merge<T extends AnyArray, U extends AnyArray> = [...T, ...U]; type Result = Merge<[1, 2], [3, 4]>; // ^? [1, 2, 3, 4] ?
2. Условные типы (Conditional types) доступны с версии TypeScript 2.8
type FilterUndefined<T> = T extends undefined ? never : T;
3. Ключевое слово infer в условных типах, которые были представлены в TypeScript 2.8
type AnyFunction = (...args: any) => any; type MyParameters<TFunction extends AnyFunction> = TFunction extends ( ...args: infer TParameters ) => any ? TParameters : never;
4. Рекурсивные условные типы (Recursive conditional types) появились с версии TypeScript 4.1
type ElementType<T> = T extends ReadonlyArray<infer U> ? ElementType<U> : T;
5. Шаблоны для строчных литералов (Template Literal types) также появились с версии TypeScript 4.1
type Fruit = "lemon" | "orange" | "apple"; type Quantity = 1 | 2 | 3; type ShoppingList = `${Quantity} ${Fruit}`; // ^? '1 lemon' | '1 orange' | '1 apple' | ... | '3 lemon' | '3 orange' | '3 apple'
6. Ограничения для дженериков (обобщений?) (Generic Constrains)
type Length<T extends { length: number }> = T["length"];
7. Перегрузка функций (Function Overloads)
function get<O, K extends string>(obj: O, path: K): Get<O, K>; function get(obj: Record<string, unknown>, path: string): unknown { // body }
Всем спасибо за внимание. Если есть пожелания, пишите в комментарии. Всем хорошего вечера и выходных.
