diff --git a/async/chain.ts b/async/chain.ts index 362e477..cf9e8ad 100644 --- a/async/chain.ts +++ b/async/chain.ts @@ -12,13 +12,33 @@ * const iter = chain([1, 2], [3, 4]); * console.log(await toArray(iter)); // [1, 2, 3, 4] * ``` + * + * It supports chaining malformed iterables. + * + * @example + * ```ts + * import { toArray } from "@core/iterutil/async/to-array"; + * import { chain } from "@core/iterutil/async/chain"; + * + * const iter = chain([1, 2], ["a", "b"], [true]); + * console.log(await toArray(iter)); // [1, 2, "a", "b", true] + * ``` */ -export async function* chain( - ...iterables: (Iterable | AsyncIterable)[] -): AsyncIterable { +export async function* chain< + T extends (Iterable | AsyncIterable)[], +>( + ...iterables: T +): AsyncIterable> { for await (const iterable of iterables) { for await (const value of iterable) { - yield value; + yield value as Chain; } } } + +export type Chain = T extends readonly [] ? never + : T extends readonly [Iterable] ? U + : T extends readonly [AsyncIterable] ? U + : T extends readonly [Iterable, ...infer R] ? U | Chain + : T extends readonly [AsyncIterable, ...infer R] ? U | Chain + : never; diff --git a/async/chain_test.ts b/async/chain_test.ts index 898baa7..516ec54 100644 --- a/async/chain_test.ts +++ b/async/chain_test.ts @@ -33,4 +33,17 @@ Deno.test("chain", async (t) => { assertEquals(await toArray(result), expected); assertType>>(true); }); + + await t.step("with malform iterable", async () => { + const result = chain( + toAsyncIterable([1, 2]), + ["a", "b"], + toAsyncIterable([true]), + ); + const expected = [1, 2, "a", "b", true]; + assertEquals(await toArray(result), expected); + assertType< + IsExact> + >(true); + }); }); diff --git a/chain.ts b/chain.ts index af73bac..2b601a1 100644 --- a/chain.ts +++ b/chain.ts @@ -11,9 +11,26 @@ * const iter = chain([1, 2], [3, 4]); * console.log([...iter]); // [1, 2, 3, 4] * ``` + * + * It supports chaining malformed iterables. + * + * @example + * ```ts + * import { chain } from "@core/iterutil/chain"; + * + * const iter = chain([1, 2], ["a", "b"], [true]); + * console.log([...iter]); // [1, 2, "a", "b", true] + * ``` */ -export function* chain(...iterables: Iterable[]): Iterable { +export function* chain[]>( + ...iterables: T +): Iterable> { for (const iterable of iterables) { - yield* iterable; + yield* iterable as Iterable>; } } + +export type Chain = T extends readonly [] ? never + : T extends readonly [Iterable] ? U + : T extends readonly [Iterable, ...infer R] ? U | Chain + : never; diff --git a/chain_test.ts b/chain_test.ts index 03df554..ed74e5b 100644 --- a/chain_test.ts +++ b/chain_test.ts @@ -2,9 +2,20 @@ import { assertEquals } from "@std/assert"; import { assertType, type IsExact } from "@std/testing/types"; import { chain } from "./chain.ts"; -Deno.test("chain", () => { - const result = chain([1, 2], [3, 4], [5]); - const expected = [1, 2, 3, 4, 5]; - assertEquals([...result], expected); - assertType>>(true); +Deno.test("chain", async (t) => { + await t.step("uniform iterables", () => { + const result = chain([1, 2], [3, 4], [5]); + const expected = [1, 2, 3, 4, 5]; + assertEquals([...result], expected); + assertType>>(true); + }); + + await t.step("malform iterables", () => { + const result = chain([1, 2], ["a", "b"], [true]); + const expected = [1, 2, "a", "b", true]; + assertEquals([...result], expected); + assertType>>( + true, + ); + }); });