UNPKG

@akala/core

Version:
235 lines (214 loc) 11.5 kB
/** * Async iteration over array-like and object structures * @module eachAsync */ import { isArrayLike } from './each.js'; /** * Type representing an async iteration callback function. * @template TError - Error type * @template T - Argument array type * @template TReturn - Return type */ export type NextFunction<TError = unknown, T extends unknown[] = [], TReturn = void> = (error?: TError, ...args: T) => TReturn; /** * Simplified version of NextFunction for cases without arguments. * @template T - Error type */ export type SimpleNextFunction<T> = NextFunction<T, [], void | Promise<void>>; /** * Represents an error that aggregates multiple errors from asynchronous iterations. * @class * @extends {Error} * @property {Error[]} errors - The array of errors that occurred during iteration. */ export class AggregateErrors extends Error { constructor(public readonly errors: Error[]) { super('One or more errors occurred. Please see errors field for more details'); } } /** * Asynchronously iterates over elements of an array-like structure. * @param {T[] | ArrayLike<T>} array - Array-like structure to iterate over * @param {(element: T, i: number) => Promise<void>} body - Async callback executed for each element * @param {boolean} waitForPrevious - Whether to wait for previous iteration to complete before next * @returns {Promise<void>} Resolves when all iterations complete */ export async function array<T>(array: T[] | ArrayLike<T>, body: (element: T, i: number) => Promise<void>, waitForPrevious: boolean): Promise<void> { if (typeof waitForPrevious == 'undefined') waitForPrevious = true; if (waitForPrevious) for (let index = 0; index < array.length; index++) { const element = array[index]; await body(element, index); } else if (Array.isArray(array)) return Promise.all(array.map(body)).then(() => { }); else return Promise.all(Array.prototype.map.call(array, body)).then(() => { }); } /** * Asynchronously iterates over key-value pairs of an object. * @param {T} o - Object to iterate over * @param {(element: T[keyof T], i: keyof T) => Promise<void>} body - Async callback for each key-value pair * @param {boolean} waitForPrevious - Whether to wait for previous iteration to complete * @returns {Promise<void>} Resolves when all iterations complete */ export function object<T, U extends unknown[] = []>(o: T, body: (element: T[keyof T], i: keyof T) => Promise<void>, waitForPrevious: boolean): Promise<void> { if (typeof waitForPrevious == 'undefined') waitForPrevious = true; return array<keyof T>(Object.keys(o) as (keyof T)[], (key, i) => body(o[key], key), waitForPrevious); } /** * Unified async iteration entry point for arrays/objects. * @function * @param {T[] | ArrayLike<T> | Record<string, unknown>} it - Iterable structure to process * @param {(element: unknown, i: any) => Promise<void>} body - Async iteration callback * @param {boolean} [waitForPrevious=true] - Whether to wait for previous iteration to complete * @returns {Promise<void>} Resolves when all iterations complete */ export function each<T, TError, U extends unknown[]>(array: T[] | ArrayLike<T>, body: (element: T, i?: number) => Promise<void>, waitForPrevious?: boolean): Promise<void> export function each<TError, U extends unknown[], T = Record<string, unknown>>(o: T, body: (element: T[typeof i], i: keyof T,) => Promise<void>, waitForPrevious?: boolean): Promise<void> // eslint-disable-next-line @typescript-eslint/no-explicit-any export function each(it: unknown[] | ArrayLike<unknown> | Record<string, unknown>, body: <TError, U extends unknown[]>(element: unknown, i: any) => Promise<void>, waitForPrevious?: boolean): Promise<void> { if (typeof waitForPrevious === 'undefined') waitForPrevious = true; if (Array.isArray(it) || isArrayLike(it)) return array<unknown>(it, body, waitForPrevious); return object<Record<string, unknown>, []>(it, body, waitForPrevious); } /** * Asynchronously maps elements of an array-like structure to new values. * @param {T[] | ArrayLike<T>} it - Array-like structure to process * @param {(element: T, i: number) => Promise<U>} body - Async transformation function * @param {boolean} waitForPrevious - Whether to wait for previous iteration * @returns {Promise<U[]>} Promise resolving to transformed elements */ export async function mapArray<T, U>(it: T[] | ArrayLike<T>, body: (element: T, i: number) => Promise<U>, waitForPrevious: boolean): Promise<U[]> { const result = []; await array(it, async function (el, i) { result.push(await body(el, i)); }, waitForPrevious); return result; } /** * Asynchronously maps key-value pairs of an object to new values. * @param {TIn} o - Source object to process * @param {(element: TIn[keyof TIn], i: keyof TIn) => TResultValue} body - Transformation function * @param {boolean} asArray - Return results as array instead of object * @param {boolean} waitForPrevious - Whether to wait for previous iteration * @returns {Promise<TResultValue[] | Partial<Proxy<TIn, TResultValue>>>} Transformed results */ export function mapObject<TIn, TResultValue>(o: TIn, body: (element: TIn[keyof TIn], i: keyof TIn) => TResultValue, asArray: true, waitForPrevious: boolean): Promise<TResultValue[]> export function mapObject<TIn, TResultValue>(o: TIn, body: (element: TIn[keyof TIn], i: keyof TIn) => TResultValue, asArray: false, waitForPrevious: boolean): Promise<{ [P in keyof TIn]?: TResultValue }> export function mapObject<TIn, TResultValue>(o: TIn, body: (element: TIn[keyof TIn], i: keyof TIn) => TResultValue, asArray: boolean, waitForPrevious: boolean) // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types export async function mapObject<TIn, TResultValue>(o: TIn, body: (element: TIn[keyof TIn], i: keyof TIn) => TResultValue, asArray: boolean, waitForPrevious: boolean) { const result: Partial<Proxy<TIn, TResultValue>> = {}; const resultArray: TResultValue[] = []; await object(o, async function (el, i) { if (asArray) resultArray.push(await body(el, i)); else result[i] = await body(el, i); }, waitForPrevious); if (asArray) return resultArray; return result; } /** * Unified async mapping function for arrays/objects. * @function * @param {T[] | ArrayLike<T> | Record<string, unknown>} it - Iterable structure to process * @param {(element: any, i: any) => any} body - Async mapping function * @param {boolean} [asArray=false] - Return results as array * @param {boolean} [waitForPrevious=true] - Whether to wait between iterations * @returns {Promise<any>} Transformed results based on input type */ export function map<T, U>(array: T[] | ArrayLike<T>, body: (element: T, i: number) => Promise<U>, asArray: true, waitForPrevious?: boolean): Promise<U[]> export function map<T, U>(array: T[] | ArrayLike<T>, body: (element: T, i: number) => Promise<U>, asArray: boolean, waitForPrevious?: boolean): Promise<U[]> export function map<TIn, TKey extends keyof TIn, TResultValue>(o: TIn, body: (element: TIn[TKey], i: TKey) => Promise<TResultValue>, asArray?: false, waitForPrevious?: boolean): Promise<Proxy<TIn, TResultValue>> export function map<TIn, TKey extends keyof TIn, TResultValue>(o: TIn, body: (element: TIn[TKey], i: TKey) => Promise<TResultValue>, asArray: true, waitForPrevious?: boolean): Promise<TResultValue[]> // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types, @typescript-eslint/no-explicit-any export function map(it: any, body: (element: any, i: any) => any, asArray?: boolean, waitForPrevious?: boolean) { if (isArrayLike(it)) return mapArray(it, body, asArray); return mapObject(it, body, asArray, waitForPrevious); } /** * Asynchronously filters elements of an array-like structure. * @param {T[] | ArrayLike<T>} it - Array-like structure to filter * @param {(element: T, i: number) => Promise<boolean>} body - Async predicate function * @param {boolean} waitForPrevious - Whether to wait for previous iteration * @returns {Promise<T[]>} Filtered elements */ export async function grepArray<T>(it: T[] | ArrayLike<T>, body: (element: T, i: number) => Promise<boolean>, waitForPrevious: boolean): Promise<T[]> { const result = []; await array(it, async function (el, i) { if (await body(el, i)) result.push(el); }, waitForPrevious); return result; } // export type Partial<T> = {[P in keyof T]?: T[P]} export type Proxy<T, U> = { [P in keyof T]: U } /** * Asynchronously filters key-value pairs of an object. * @param {T} o - Object to filter * @param {(element: T[keyof T], i: keyof T) => Promise<boolean>} body - Async predicate function * @param {boolean} asArray - Return results as array * @param {boolean} waitForPrevious - Whether to wait for previous iteration * @returns {Promise<T[keyof T][] | Partial<T>>} Filtered results */ export function grepObject<T>(o: T, body: (element: T[keyof T], i: keyof T) => Promise<boolean>, asArray: true, waitForPrevious: boolean): Promise<T[keyof T][]> export function grepObject<T>(o: T, body: (element: T[keyof T], i: keyof T) => Promise<boolean>, asArray: false, waitForPrevious: boolean): Promise<Partial<T>> export function grepObject<T>(o: T, body: (element: T[keyof T], i: keyof T) => Promise<boolean>, asArray: boolean, waitForPrevious: boolean): Promise<Partial<T> | T[keyof T][]> // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types export async function grepObject<T>(o: T, body: (element: T[keyof T], i: keyof T) => Promise<boolean>, asArray: boolean, waitForPrevious: boolean) { const result: Partial<T> = {}; const resultArray: T[keyof T][] = []; await object(o, async function (el, i) { if (await body(el, i)) if (asArray) resultArray.push(el); else result[i] = el; }, waitForPrevious); if (asArray) return resultArray; return result; } /** * Unified async filter function for arrays/objects. * @function * @param {T[] | ArrayLike<T> | Record<string, unknown>} it - Iterable structure to process * @param {(element: any, i: any) => Promise<boolean>} body - Async predicate function * @param {boolean} [asArray=false] - Return results as array * @param {boolean} [waitForPrevious=true] - Whether to wait between iterations * @returns {Promise<any>} Filtered results based on input type */ export function grep<T>(array: T[] | ArrayLike<T>, body: (element: T, i: number) => Promise<boolean>, waitForPrevious?: boolean): Promise<T[]> export function grep<T>(o: T, body: <U extends keyof T>(element: T[U], i: U) => Promise<boolean>, waitForPrevious?: boolean): Promise<Partial<T>> export function grep<T, U extends keyof T>(o: T, body: (element: T[U], i: U) => Promise<boolean>, asArray: true, waitForPrevious?: boolean): Promise<T[U][]> // eslint-disable-next-line @typescript-eslint/explicit-module-boundary-types, @typescript-eslint/no-explicit-any export function grep(it: any, body: (element: any, i: any) => Promise<boolean>, asArray?: boolean, waitForPrevious?: boolean) { if (isArrayLike(it)) return grepArray(it, body, asArray); return grepObject(it, body, asArray, waitForPrevious); }