| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259 | // Licensed to the .NET Foundation under one or more agreements.// The .NET Foundation licenses this file to you under the Apache 2.0 License.// See the LICENSE file in the project root for more information. using System.Collections.Generic;using System.Reactive.Disposables;using System.Threading;using System.Threading.Tasks;namespace System.Reactive.Linq{    partial class AsyncObservable    {        public static IAsyncObservable<TResult> Join<TLeft, TRight, TLeftDuration, TRightDuration, TResult>(this IAsyncObservable<TLeft> left, IAsyncObservable<TRight> right, Func<TLeft, IAsyncObservable<TLeftDuration>> leftDurationSelector, Func<TRight, IAsyncObservable<TRightDuration>> rightDurationSelector, Func<TLeft, 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.Join(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) Join<TLeft, TRight, TLeftDuration, TRightDuration, TResult>(IAsyncObserver<TResult> observer, IAsyncDisposable subscriptions, Func<TLeft, IAsyncObservable<TLeftDuration>> leftDurationSelector, Func<TRight, IAsyncObservable<TRightDuration>> rightDurationSelector, Func<TLeft, 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 leftMap = new SortedDictionary<int, TLeft>();            var rightMap = new SortedDictionary<int, TRight>();            var leftDone = false;            var rightDone = false;            var leftId = default(int);            var rightId = default(int);            async Task OnErrorAsync(Exception ex)            {                using (await gate.LockAsync().ConfigureAwait(false))                {                    await observer.OnErrorAsync(ex).ConfigureAwait(false);                }            }            var leftObserver =                Create<TLeft>(                    async x =>                    {                        var theLeftId = default(int);                        var theRightId = default(int);                        using (await gate.LockAsync().ConfigureAwait(false))                        {                            theLeftId = leftId++;                            theRightId = rightId;                            leftMap.Add(theLeftId, x);                        }                        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) && leftMap.Count == 0 && leftDone)                                        {                                            await observer.OnCompletedAsync().ConfigureAwait(false);                                        }                                    }                                    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 rightValue in rightMap)                            {                                if (rightValue.Key < theRightId)                                {                                    var result = default(TResult);                                    try                                    {                                        result = resultSelector(x, rightValue.Value);                                    }                                    catch (Exception ex)                                    {                                        await OnErrorAsync(ex).ConfigureAwait(false);                                        return;                                    }                                    await observer.OnNextAsync(result).ConfigureAwait(false);                                }                            }                        }                    },                    OnErrorAsync,                    async () =>                    {                        using (await gate.LockAsync().ConfigureAwait(false))                        {                            leftDone = true;                            if (rightDone || leftMap.Count == 0)                            {                                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))                                    {                                        if (rightMap.Remove(theRightId) && rightMap.Count == 0 && rightDone)                                        {                                            await observer.OnCompletedAsync().ConfigureAwait(false);                                        }                                    }                                    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 leftValue in leftMap)                            {                                if (leftValue.Key < theLeftId)                                {                                    var result = default(TResult);                                    try                                    {                                        result = resultSelector(leftValue.Value, x);                                    }                                    catch (Exception ex)                                    {                                        await OnErrorAsync(ex).ConfigureAwait(false);                                        return;                                    }                                    await observer.OnNextAsync(result).ConfigureAwait(false);                                }                            }                        }                    },                    OnErrorAsync,                    async () =>                    {                        using (await gate.LockAsync().ConfigureAwait(false))                        {                            rightDone = true;                            if (leftDone || rightMap.Count == 0)                            {                                await observer.OnCompletedAsync().ConfigureAwait(false);                            }                        }                    }                );            return (leftObserver, rightObserver, group);        }    }}
 |