UNPKG

ix

Version:

The Interactive Extensions for JavaScript

64 lines (57 loc) 2 kB
import { OperatorAsyncFunction } from '../../interfaces.js'; import { AsyncIterableX, interval, concat, of } from '../index.js'; import { map } from './map.js'; import { merge } from '../merge.js'; import { wrapWithAbort } from './withabort.js'; const timerEvent = {}; const ended = {}; class BufferCountOrTime<TSource> extends AsyncIterableX<TSource[]> { constructor( private readonly source: AsyncIterable<TSource>, private readonly bufferSize: number, private readonly maxWaitTime: number ) { super(); } async *[Symbol.asyncIterator](signal?: AbortSignal) { const buffer: TSource[] = []; const timer = interval(this.maxWaitTime).pipe(map(() => timerEvent)); const source = concat(this.source, of(ended)); const merged = merge(source, timer); for await (const item of wrapWithAbort(merged, signal)) { if (item === ended) { break; } if (item !== timerEvent) { buffer.push(item as TSource); } if (buffer.length >= this.bufferSize || (buffer.length && item === timerEvent)) { yield buffer.slice(); buffer.length = 0; } } if (buffer.length) { yield buffer; } } } /** * Projects each element of an async-iterable sequence into consecutive buffers * which are emitted when either the threshold count or time is met. * * @template TSource The type of elements in the source sequence. * @param {number} count The size of the buffer. * @param {number} time The threshold number of milliseconds to wait before flushing a non-full buffer * @returns {OperatorAsyncFunction<TSource, TSource[]>} An operator which returns an async-iterable sequence * of buffers */ export function bufferCountOrTime<TSource>( count: number, time: number ): OperatorAsyncFunction<TSource, TSource[]> { return function bufferOperatorFunction( source: AsyncIterable<TSource> ): AsyncIterableX<TSource[]> { return new BufferCountOrTime<TSource>(source, count, time); }; }