|  | @@ -0,0 +1,132 @@
 | 
	
		
			
				|  |  | +// 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.Reactive.Disposables;
 | 
	
		
			
				|  |  | +using System.Threading;
 | 
	
		
			
				|  |  | +using System.Threading.Tasks;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +namespace System.Reactive.Linq
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +    partial class AsyncObservable
 | 
	
		
			
				|  |  | +    {
 | 
	
		
			
				|  |  | +        public static Task ForEachAsync<TSource>(this IAsyncObservable<TSource> source, Action<TSource> onNext, CancellationToken token = default(CancellationToken))
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            if (source == null)
 | 
	
		
			
				|  |  | +                throw new ArgumentNullException(nameof(source));
 | 
	
		
			
				|  |  | +            if (onNext == null)
 | 
	
		
			
				|  |  | +                throw new ArgumentNullException(nameof(onNext));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            return ForEachAsyncCore(source, (x, i) => { onNext(x); return Task.CompletedTask; }, token);
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        public static Task ForEachAsync<TSource>(this IAsyncObservable<TSource> source, Func<TSource, Task> onNext, CancellationToken token = default(CancellationToken))
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            if (source == null)
 | 
	
		
			
				|  |  | +                throw new ArgumentNullException(nameof(source));
 | 
	
		
			
				|  |  | +            if (onNext == null)
 | 
	
		
			
				|  |  | +                throw new ArgumentNullException(nameof(onNext));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            return ForEachAsyncCore(source, (x, i) => onNext(x), token);
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        public static Task ForEachAsync<TSource>(this IAsyncObservable<TSource> source, Action<TSource, int> onNext, CancellationToken token = default(CancellationToken))
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            if (source == null)
 | 
	
		
			
				|  |  | +                throw new ArgumentNullException(nameof(source));
 | 
	
		
			
				|  |  | +            if (onNext == null)
 | 
	
		
			
				|  |  | +                throw new ArgumentNullException(nameof(onNext));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            return ForEachAsyncCore(source, (x, i) => { onNext(x, i); return Task.CompletedTask; }, token);
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        public static Task ForEachAsync<TSource>(this IAsyncObservable<TSource> source, Func<TSource, int, Task> onNext, CancellationToken token = default(CancellationToken))
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            if (source == null)
 | 
	
		
			
				|  |  | +                throw new ArgumentNullException(nameof(source));
 | 
	
		
			
				|  |  | +            if (onNext == null)
 | 
	
		
			
				|  |  | +                throw new ArgumentNullException(nameof(onNext));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            return ForEachAsyncCore(source, onNext, token);
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        private static async Task ForEachAsyncCore<TSource>(IAsyncObservable<TSource> source, Func<TSource, int, Task> onNext, CancellationToken token)
 | 
	
		
			
				|  |  | +        {
 | 
	
		
			
				|  |  | +            token.ThrowIfCancellationRequested();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            var tcs = new TaskCompletionSource<object>();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            var subscription = new SingleAssignmentAsyncDisposable();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            using (token.Register(() =>
 | 
	
		
			
				|  |  | +            {
 | 
	
		
			
				|  |  | +                tcs.TrySetCanceled(token);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                subscription.DisposeAsync().ContinueWith(t =>
 | 
	
		
			
				|  |  | +                {
 | 
	
		
			
				|  |  | +                    if (t.Exception != null)
 | 
	
		
			
				|  |  | +                    {
 | 
	
		
			
				|  |  | +                        // TODO: Trace?
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                });
 | 
	
		
			
				|  |  | +            }))
 | 
	
		
			
				|  |  | +            {
 | 
	
		
			
				|  |  | +                var i = 0;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                var o = AsyncObserver.Create<TSource>(
 | 
	
		
			
				|  |  | +                    async x =>
 | 
	
		
			
				|  |  | +                    {
 | 
	
		
			
				|  |  | +                        try
 | 
	
		
			
				|  |  | +                        {
 | 
	
		
			
				|  |  | +                            await onNext(x, checked(i++)).ConfigureAwait(false);
 | 
	
		
			
				|  |  | +                        }
 | 
	
		
			
				|  |  | +                        catch (Exception ex)
 | 
	
		
			
				|  |  | +                        {
 | 
	
		
			
				|  |  | +                            try
 | 
	
		
			
				|  |  | +                            {
 | 
	
		
			
				|  |  | +                                tcs.TrySetException(ex);
 | 
	
		
			
				|  |  | +                            }
 | 
	
		
			
				|  |  | +                            finally
 | 
	
		
			
				|  |  | +                            {
 | 
	
		
			
				|  |  | +                                await subscription.DisposeAsync().ConfigureAwait(false);
 | 
	
		
			
				|  |  | +                            }
 | 
	
		
			
				|  |  | +                        }
 | 
	
		
			
				|  |  | +                    },
 | 
	
		
			
				|  |  | +                    async ex =>
 | 
	
		
			
				|  |  | +                    {
 | 
	
		
			
				|  |  | +                        try
 | 
	
		
			
				|  |  | +                        {
 | 
	
		
			
				|  |  | +                            tcs.TrySetException(ex);
 | 
	
		
			
				|  |  | +                        }
 | 
	
		
			
				|  |  | +                        finally
 | 
	
		
			
				|  |  | +                        {
 | 
	
		
			
				|  |  | +                            await subscription.DisposeAsync().ConfigureAwait(false);
 | 
	
		
			
				|  |  | +                        }
 | 
	
		
			
				|  |  | +                    },
 | 
	
		
			
				|  |  | +                    async () =>
 | 
	
		
			
				|  |  | +                    {
 | 
	
		
			
				|  |  | +                        try
 | 
	
		
			
				|  |  | +                        {
 | 
	
		
			
				|  |  | +                            tcs.TrySetResult(null);
 | 
	
		
			
				|  |  | +                        }
 | 
	
		
			
				|  |  | +                        finally
 | 
	
		
			
				|  |  | +                        {
 | 
	
		
			
				|  |  | +                            await subscription.DisposeAsync().ConfigureAwait(false);
 | 
	
		
			
				|  |  | +                        }
 | 
	
		
			
				|  |  | +                    }
 | 
	
		
			
				|  |  | +                );
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                //
 | 
	
		
			
				|  |  | +                // NB: If any of the lines below throw, the result will go into the Task returned from the async method.
 | 
	
		
			
				|  |  | +                //     There's also no need to use SubscribeSafeAsync here; the exception will propagate just fine.
 | 
	
		
			
				|  |  | +                //
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                var d = await source.SubscribeAsync(o).ConfigureAwait(false);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                await subscription.AssignAsync(d).ConfigureAwait(false);
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            await tcs.Task.ConfigureAwait(false);
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +}
 |