UNPKG

ix

Version:

The Interactive Extensions for JavaScript

86 lines (81 loc) 3.74 kB
import { AsyncIterableX } from '../asynciterablex.js'; import { createGrouping } from './_grouping.js'; import { identity } from '../../util/identity.js'; import { OperatorAsyncFunction } from '../../interfaces.js'; import { wrapWithAbort } from './withabort.js'; import { throwIfAborted } from '../../aborterror.js'; /** @ignore */ export class JoinAsyncIterable<TOuter, TInner, TKey, TResult> extends AsyncIterableX<TResult> { private _outer: AsyncIterable<TOuter>; private _inner: AsyncIterable<TInner>; private _outerSelector: (value: TOuter, signal?: AbortSignal) => TKey | Promise<TKey>; private _innerSelector: (value: TInner, signal?: AbortSignal) => TKey | Promise<TKey>; private _resultSelector: ( outer: TOuter, inner: TInner, signal?: AbortSignal ) => TResult | Promise<TResult>; constructor( outer: AsyncIterable<TOuter>, inner: AsyncIterable<TInner>, outerSelector: (value: TOuter, signal?: AbortSignal) => TKey | Promise<TKey>, innerSelector: (value: TInner, signal?: AbortSignal) => TKey | Promise<TKey>, resultSelector: ( outer: TOuter, inner: TInner, signal?: AbortSignal ) => TResult | Promise<TResult> ) { super(); this._outer = outer; this._inner = inner; this._outerSelector = outerSelector; this._innerSelector = innerSelector; this._resultSelector = resultSelector; } async *[Symbol.asyncIterator](signal?: AbortSignal) { throwIfAborted(signal); const map = await createGrouping(this._inner, this._innerSelector, identity, signal); for await (const outerElement of wrapWithAbort(this._outer, signal)) { const outerKey = await this._outerSelector(outerElement, signal); if (map.has(outerKey)) { for (const innerElement of map.get(outerKey)!) { yield await this._resultSelector(outerElement, innerElement, signal); } } } } } /** * Correlates the elements of two sequences based on matching keys. * * @template TOuter The type of the elements of the first async-iterable sequence. * @template TInner The type of the elements of the second async-iterable sequence. * @template TKey The type of the keys returned by the key selector functions. * @template TResult The type of the result elements. * @param {AsyncIterable<TInner>} inner The async-enumerable sequence to join to the first sequence. * @param {((value: TOuter, signal?: AbortSignal) => TKey | Promise<TKey>)} outerSelector A function to extract the join key from each element * of the first sequence. * @param {((value: TInner, signal?: AbortSignal) => TKey | Promise<TKey>)} innerSelector A function to extract the join key from each element * of the second sequence. * @param {((outer: TOuter, inner: TInner, signal?: AbortSignal) => TResult | Promise<TResult>)} resultSelector A function to create a result element * from two matching elements. * @returns {OperatorAsyncFunction<TOuter, TResult>} An async-iterable sequence that has elements that are obtained by performing an inner join * on two sequences. */ export function innerJoin<TOuter, TInner, TKey, TResult>( inner: AsyncIterable<TInner>, outerSelector: (value: TOuter, signal?: AbortSignal) => TKey | Promise<TKey>, innerSelector: (value: TInner, signal?: AbortSignal) => TKey | Promise<TKey>, resultSelector: (outer: TOuter, inner: TInner, signal?: AbortSignal) => TResult | Promise<TResult> ): OperatorAsyncFunction<TOuter, TResult> { return function innerJoinOperatorFunction(outer: AsyncIterable<TOuter>): AsyncIterableX<TResult> { return new JoinAsyncIterable<TOuter, TInner, TKey, TResult>( outer, inner, outerSelector, innerSelector, resultSelector ); }; }