123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239 |
- // Licensed to the .NET Foundation under one or more agreements.
- // The .NET Foundation licenses this file to you under the MIT License.
- // See the LICENSE file in the project root for more information.
- using System.Collections.Generic;
- using System.Reactive.Disposables;
- using System.Reactive.Subjects;
- using System.Threading;
- using System.Threading.Tasks;
- namespace System.Reactive.Linq
- {
- partial class AsyncObservable
- {
- public static IAsyncObservable<TResult> GroupJoin<TLeft, TRight, TLeftDuration, TRightDuration, TResult>(this IAsyncObservable<TLeft> left, IAsyncObservable<TRight> right, Func<TLeft, IAsyncObservable<TLeftDuration>> leftDurationSelector, Func<TRight, IAsyncObservable<TRightDuration>> rightDurationSelector, Func<TLeft, IAsyncObservable<TRight>, TResult> resultSelector)
- {
- if (left == null)
- throw new ArgumentNullException(nameof(left));
- if (right == null)
- throw new ArgumentNullException(nameof(right));
- if (leftDurationSelector == null)
- throw new ArgumentNullException(nameof(leftDurationSelector));
- if (rightDurationSelector == null)
- throw new ArgumentNullException(nameof(rightDurationSelector));
- if (resultSelector == null)
- throw new ArgumentNullException(nameof(resultSelector));
- return Create<TResult>(async observer =>
- {
- var subscriptions = new CompositeAsyncDisposable();
- var (leftObserver, rightObserver, disposable) = AsyncObserver.GroupJoin(observer, subscriptions, leftDurationSelector, rightDurationSelector, resultSelector);
- var leftSubscription = await left.SubscribeSafeAsync(leftObserver).ConfigureAwait(false);
- await subscriptions.AddAsync(leftSubscription).ConfigureAwait(false);
- var rightSubscription = await right.SubscribeSafeAsync(rightObserver).ConfigureAwait(false);
- await subscriptions.AddAsync(rightSubscription).ConfigureAwait(false);
- return disposable;
- });
- }
- }
- partial class AsyncObserver
- {
- public static (IAsyncObserver<TLeft>, IAsyncObserver<TRight>, IAsyncDisposable) GroupJoin<TLeft, TRight, TLeftDuration, TRightDuration, TResult>(IAsyncObserver<TResult> observer, IAsyncDisposable subscriptions, Func<TLeft, IAsyncObservable<TLeftDuration>> leftDurationSelector, Func<TRight, IAsyncObservable<TRightDuration>> rightDurationSelector, Func<TLeft, IAsyncObservable<TRight>, TResult> resultSelector)
- {
- if (observer == null)
- throw new ArgumentNullException(nameof(observer));
- if (subscriptions == null)
- throw new ArgumentNullException(nameof(subscriptions));
- if (leftDurationSelector == null)
- throw new ArgumentNullException(nameof(leftDurationSelector));
- if (rightDurationSelector == null)
- throw new ArgumentNullException(nameof(rightDurationSelector));
- if (resultSelector == null)
- throw new ArgumentNullException(nameof(resultSelector));
- var gate = new AsyncLock();
- var group = new CompositeAsyncDisposable(subscriptions);
- var refCount = new RefCountAsyncDisposable(group);
- var leftMap = new SortedDictionary<int, IAsyncObserver<TRight>>();
- var rightMap = new SortedDictionary<int, TRight>();
- var leftId = default(int);
- var rightId = default(int);
- async Task OnErrorAsync(Exception ex)
- {
- using (await gate.LockAsync().ConfigureAwait(false))
- {
- foreach (var o in leftMap)
- {
- await o.Value.OnErrorAsync(ex).ConfigureAwait(false);
- }
- await observer.OnErrorAsync(ex).ConfigureAwait(false);
- }
- }
- var leftObserver =
- Create<TLeft>(
- async x =>
- {
- var s = new SequentialSimpleAsyncSubject<TRight>();
- var theLeftId = default(int);
- var theRightId = default(int);
- using (await gate.LockAsync().ConfigureAwait(false))
- {
- theLeftId = leftId++;
- theRightId = rightId;
- leftMap.Add(theLeftId, s);
- }
- var duration = default(IAsyncObservable<TLeftDuration>);
- try
- {
- duration = leftDurationSelector(x);
- }
- catch (Exception ex)
- {
- await OnErrorAsync(ex).ConfigureAwait(false);
- return;
- }
- var sad = new SingleAssignmentAsyncDisposable();
- await group.AddAsync(sad).ConfigureAwait(false);
- var durationObserver =
- Create<TLeftDuration>(
- d => Task.CompletedTask,
- OnErrorAsync,
- async () =>
- {
- using (await gate.LockAsync().ConfigureAwait(false))
- {
- if (leftMap.Remove(theLeftId))
- {
- await s.OnCompletedAsync().ConfigureAwait(false);
- }
- }
- await group.RemoveAsync(sad).ConfigureAwait(false);
- }
- );
- var durationSubscription = await duration.FirstOrDefault().SubscribeSafeAsync(durationObserver).ConfigureAwait(false);
- await sad.AssignAsync(durationSubscription).ConfigureAwait(false);
- var window = new WindowAsyncObservable<TRight>(s, refCount);
- var result = default(TResult);
- try
- {
- result = resultSelector(x, window);
- }
- catch (Exception ex)
- {
- await OnErrorAsync(ex).ConfigureAwait(false);
- return;
- }
- using (await gate.LockAsync().ConfigureAwait(false))
- {
- await observer.OnNextAsync(result).ConfigureAwait(false);
- foreach (var rightValue in rightMap)
- {
- if (rightValue.Key < theRightId)
- {
- await s.OnNextAsync(rightValue.Value).ConfigureAwait(false);
- }
- }
- }
- },
- OnErrorAsync,
- async () =>
- {
- using (await gate.LockAsync().ConfigureAwait(false))
- {
- await observer.OnCompletedAsync().ConfigureAwait(false);
- }
- }
- );
- var rightObserver =
- Create<TRight>(
- async x =>
- {
- var theLeftId = 0;
- var theRightId = 0;
- using (await gate.LockAsync().ConfigureAwait(false))
- {
- theRightId = rightId++;
- theLeftId = leftId;
- rightMap.Add(theRightId, x);
- }
- var duration = default(IAsyncObservable<TRightDuration>);
- try
- {
- duration = rightDurationSelector(x);
- }
- catch (Exception ex)
- {
- await OnErrorAsync(ex).ConfigureAwait(false);
- return;
- }
- var sad = new SingleAssignmentAsyncDisposable();
- await group.AddAsync(sad).ConfigureAwait(false);
- var durationObserver =
- Create<TRightDuration>(
- d => Task.CompletedTask,
- OnErrorAsync,
- async () =>
- {
- using (await gate.LockAsync().ConfigureAwait(false))
- {
- rightMap.Remove(theRightId);
- }
- await group.RemoveAsync(sad).ConfigureAwait(false);
- }
- );
- var durationSubscription = await duration.FirstOrDefault().SubscribeSafeAsync(durationObserver).ConfigureAwait(false);
- await sad.AssignAsync(durationSubscription).ConfigureAwait(false);
- using (await gate.LockAsync().ConfigureAwait(false))
- {
- foreach (var o in leftMap)
- {
- if (o.Key < theLeftId)
- {
- await o.Value.OnNextAsync(x).ConfigureAwait(false);
- }
- }
- }
- },
- OnErrorAsync,
- () => Task.CompletedTask
- );
- return (leftObserver, rightObserver, refCount);
- }
- }
- }
|