type Awaitable = T | Promise; type ForOfAwaitable = Iterable | AsyncIterable; /** * Alternates items from the first and second iterable in the output iterable, until either input runs out of items. */ export function* interleave(xs: Iterable, ys: Iterable): IterableIterator { const itx = xs[Symbol.iterator](); const ity = ys[Symbol.iterator](); while (true) { const rx = itx.next(); if (rx.done) break; else yield rx.value; const ry = ity.next(); if (ry.done) break; else yield ry.value; } } /** * It's like interleave, but will flatten items of the second (async) iterable. */ export async function* aInterleaveFlattenSecond(xs: Iterable, ys: Iterable>): AsyncIterableIterator { const itx = xs[Symbol.iterator](); const ity = ys[Symbol.iterator](); while (true) { const rx = itx.next(); if (rx.done) break; else yield rx.value; const ry = ity.next(); if (ry.done) break; else yield* ry.value; } } export function map(f: (a: A) => B) { return function* (iterable: Iterable): IterableIterator { for (const x of iterable) yield f(x); }; } export function aMap(f: (a: A) => Awaitable) { return async function* (iterable: ForOfAwaitable): AsyncIterableIterator { for await (const x of iterable) yield f(x); }; } export function* join(iterable: Iterable): IterableIterator { yield [...iterable].join(''); } export async function* aJoin(iterable: ForOfAwaitable): AsyncIterableIterator { const chunks = []; for await (const x of iterable) chunks.push(x); yield chunks.join(''); }