123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250 |
- // 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.Diagnostics;
- using System.Threading;
- using System.Threading.Tasks;
- namespace System.Linq
- {
- public static partial class AsyncEnumerableEx
- {
- public static IAsyncEnumerable<TSource> Defer<TSource>(Func<IAsyncEnumerable<TSource>> factory)
- {
- if (factory == null)
- throw Error.ArgumentNull(nameof(factory));
- #if USE_ASYNC_ITERATOR
- return AsyncEnumerable.Create(Core);
- async IAsyncEnumerator<TSource> Core(CancellationToken cancellationToken)
- {
- await foreach (TSource item in factory().WithCancellation(cancellationToken).ConfigureAwait(false))
- {
- yield return item;
- }
- }
- #else
- return new DeferIterator<TSource>(factory);
- #endif
- }
- public static IAsyncEnumerable<TSource> Defer<TSource>(Func<Task<IAsyncEnumerable<TSource>>> factory)
- {
- if (factory == null)
- throw Error.ArgumentNull(nameof(factory));
- #if USE_ASYNC_ITERATOR
- return AsyncEnumerable.Create(Core);
- async IAsyncEnumerator<TSource> Core(CancellationToken cancellationToken)
- {
- await foreach (TSource item in (await factory().ConfigureAwait(false)).WithCancellation(cancellationToken).ConfigureAwait(false))
- {
- yield return item;
- }
- }
- #else
- return new AsyncDeferIterator<TSource>(factory);
- #endif
- }
- #if !NO_DEEP_CANCELLATION
- public static IAsyncEnumerable<TSource> Defer<TSource>(Func<CancellationToken, Task<IAsyncEnumerable<TSource>>> factory)
- {
- if (factory == null)
- throw Error.ArgumentNull(nameof(factory));
- #if USE_ASYNC_ITERATOR
- return AsyncEnumerable.Create(Core);
- async IAsyncEnumerator<TSource> Core(CancellationToken cancellationToken)
- {
- await foreach (TSource item in (await factory(cancellationToken).ConfigureAwait(false)).WithCancellation(cancellationToken).ConfigureAwait(false))
- {
- yield return item;
- }
- }
- #else
- return new AsyncDeferIteratorWithCancellation<TSource>(factory);
- #endif
- }
- #endif
- #if !USE_ASYNC_ITERATOR
- private sealed class DeferIterator<T> : AsyncIteratorBase<T>
- {
- private readonly Func<IAsyncEnumerable<T>> _factory;
- private IAsyncEnumerator<T> _enumerator;
- public DeferIterator(Func<IAsyncEnumerable<T>> factory)
- {
- Debug.Assert(factory != null);
- _factory = factory;
- }
- public override T Current => _enumerator == null ? default : _enumerator.Current;
- public override AsyncIteratorBase<T> Clone()
- {
- return new DeferIterator<T>(_factory);
- }
- public override async ValueTask DisposeAsync()
- {
- if (_enumerator != null)
- {
- await _enumerator.DisposeAsync().ConfigureAwait(false);
- _enumerator = null;
- }
- await base.DisposeAsync().ConfigureAwait(false);
- }
- protected override ValueTask<bool> MoveNextCore()
- {
- if (_enumerator == null)
- {
- return InitializeAndMoveNextAsync();
- }
- return _enumerator.MoveNextAsync();
- }
- private async ValueTask<bool> InitializeAndMoveNextAsync()
- {
- // NB: Using an async method to ensure any exception is reported via the task.
- try
- {
- _enumerator = _factory().GetAsyncEnumerator(_cancellationToken);
- }
- catch (Exception ex)
- {
- _enumerator = Throw<T>(ex).GetAsyncEnumerator(_cancellationToken);
- throw;
- }
- return await _enumerator.MoveNextAsync().ConfigureAwait(false);
- }
- }
- private sealed class AsyncDeferIterator<T> : AsyncIteratorBase<T>
- {
- private readonly Func<Task<IAsyncEnumerable<T>>> _factory;
- private IAsyncEnumerator<T> _enumerator;
- public AsyncDeferIterator(Func< Task<IAsyncEnumerable<T>>> factory)
- {
- Debug.Assert(factory != null);
- _factory = factory;
- }
- public override T Current => _enumerator == null ? default : _enumerator.Current;
- public override AsyncIteratorBase<T> Clone()
- {
- return new AsyncDeferIterator<T>(_factory);
- }
- public override async ValueTask DisposeAsync()
- {
- if (_enumerator != null)
- {
- await _enumerator.DisposeAsync().ConfigureAwait(false);
- _enumerator = null;
- }
- await base.DisposeAsync().ConfigureAwait(false);
- }
- protected override ValueTask<bool> MoveNextCore()
- {
- if (_enumerator == null)
- {
- return InitializeAndMoveNextAsync();
- }
- return _enumerator.MoveNextAsync();
- }
- private async ValueTask<bool> InitializeAndMoveNextAsync()
- {
- try
- {
- _enumerator = (await _factory().ConfigureAwait(false)).GetAsyncEnumerator(_cancellationToken);
- }
- catch (Exception ex)
- {
- _enumerator = Throw<T>(ex).GetAsyncEnumerator(_cancellationToken);
- throw;
- }
- return await _enumerator.MoveNextAsync().ConfigureAwait(false);
- }
- }
- #if !NO_DEEP_CANCELLATION
- private sealed class AsyncDeferIteratorWithCancellation<T> : AsyncIteratorBase<T>
- {
- private readonly Func<CancellationToken, Task<IAsyncEnumerable<T>>> _factory;
- private IAsyncEnumerator<T> _enumerator;
- public AsyncDeferIteratorWithCancellation(Func<CancellationToken, Task<IAsyncEnumerable<T>>> factory)
- {
- Debug.Assert(factory != null);
- _factory = factory;
- }
- public override T Current => _enumerator == null ? default : _enumerator.Current;
- public override AsyncIteratorBase<T> Clone()
- {
- return new AsyncDeferIteratorWithCancellation<T>(_factory);
- }
- public override async ValueTask DisposeAsync()
- {
- if (_enumerator != null)
- {
- await _enumerator.DisposeAsync().ConfigureAwait(false);
- _enumerator = null;
- }
- await base.DisposeAsync().ConfigureAwait(false);
- }
- protected override ValueTask<bool> MoveNextCore()
- {
- if (_enumerator == null)
- {
- return InitializeAndMoveNextAsync();
- }
- return _enumerator.MoveNextAsync();
- }
- private async ValueTask<bool> InitializeAndMoveNextAsync()
- {
- try
- {
- _enumerator = (await _factory(_cancellationToken).ConfigureAwait(false)).GetAsyncEnumerator(_cancellationToken);
- }
- catch (Exception ex)
- {
- _enumerator = Throw<T>(ex).GetAsyncEnumerator(_cancellationToken);
- throw;
- }
- return await _enumerator.MoveNextAsync().ConfigureAwait(false);
- }
- }
- #endif
- #endif
- }
- }
|