import { AsyncIterableX } from '../asynciterablex.js'; import { OperatorAsyncFunction } from '../../interfaces.js'; import { wrapWithAbort } from './withabort.js'; import { throwIfAborted } from '../../aborterror.js'; interface AsyncResolver { resolve: (value: T | PromiseLike) => void; reject: (reason?: any) => void; } const WAITING_TYPE = 'waiting'; const BATCHING_TYPE = 'batching'; interface WaitingState { type: 'waiting'; resolver: AsyncResolver>; } interface BatchingState { type: 'batching'; values: T[]; } type State = WaitingState | BatchingState; function assertNever(value: never): never { throw new Error(`Unhandled discriminated union member ${value}`); } /** @ignore */ export class BatchAsyncIterable extends AsyncIterableX { private _source: AsyncIterable; constructor(source: AsyncIterable) { super(); this._source = source; } [Symbol.asyncIterator](signal?: AbortSignal) { throwIfAborted(signal); const it = wrapWithAbort(this._source, signal)[Symbol.asyncIterator](); let state: State = { type: BATCHING_TYPE, values: [] }; let ended: null | Promise> = null; let error: any = null; function consumeNext() { it.next().then( (res) => { if (res.done) { ended = Promise.resolve({ done: true } as IteratorResult); if (state.type === WAITING_TYPE) { state.resolver.resolve(ended); } } else { if (state.type === WAITING_TYPE) { const { resolve } = state.resolver; state = { type: BATCHING_TYPE, values: [] }; resolve({ done: res.done, value: [res.value] }); } else if (state.type === BATCHING_TYPE) { state.values.push(res.value); } else { assertNever(state); } consumeNext(); } }, (err) => { error = err; if (state.type === WAITING_TYPE) { state.resolver.reject(err); } } ); } consumeNext(); return { next() { if (error) { return Promise.reject(error); } if (state.type === BATCHING_TYPE && state.values.length > 0) { const { values } = state; state.values = []; return Promise.resolve({ done: false, value: values }); } if (ended) { return ended; } if (state.type === WAITING_TYPE) { throw new Error('Previous `next()` is still in progress'); } return new Promise>((resolve, reject) => { state = { type: WAITING_TYPE, resolver: { resolve, reject }, }; }); }, return(value: any) { return it.return ? it.return(value).then(() => ({ done: true } as IteratorResult)) : Promise.resolve({ done: true } as IteratorResult); }, }; } } /** Returns an async iterable sequence of batches that are collected from the source sequence between * subsequent `next()` calls. * * @template TSource The type of elements in the source sequence. * @returns {OperatorAsyncFunction} An operator returning an async-iterable of batches that are collection from the * source sequence between subsequent `next()` calls. */ export function batch(): OperatorAsyncFunction { return function batchOperator(source: AsyncIterable): AsyncIterableX { return new BatchAsyncIterable(source); }; }