| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216 | // 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 AsyncEnumerable    {        public static IAsyncEnumerable<T> CreateEnumerable<T>(Func<IAsyncEnumerator<T>> getEnumerator)        {            if (getEnumerator == null)                throw new ArgumentNullException(nameof(getEnumerator));            return new AnonymousAsyncEnumerable<T>(getEnumerator);        }        public static IAsyncEnumerable<T> CreateEnumerable<T>(Func<Task<IAsyncEnumerator<T>>> getEnumerator)        {            if (getEnumerator == null)                throw new ArgumentNullException(nameof(getEnumerator));            return new AnonymousAsyncEnumerableWithTask<T>(getEnumerator);        }        public static IAsyncEnumerator<T> CreateEnumerator<T>(Func<Task<bool>> moveNext, Func<T> current, Func<Task> dispose)        {            if (moveNext == null)                throw new ArgumentNullException(nameof(moveNext));            // Note: Many methods pass null in for the second two params. We're assuming            // That the caller is responsible and knows what they're doing            return new AnonymousAsyncIterator<T>(moveNext, current, dispose);        }        private static IAsyncEnumerator<T> CreateEnumerator<T>(Func<TaskCompletionSource<bool>, Task<bool>> moveNext, Func<T> current, Func<Task> dispose)        {            var self = new AnonymousAsyncIterator<T>(                async () =>                {                    var tcs = new TaskCompletionSource<bool>();                    var stop = new Action(() => tcs.TrySetCanceled());                    return await moveNext(tcs).ConfigureAwait(false);                },                current,                dispose            );            return self;        }        private sealed class AnonymousAsyncEnumerable<T> : IAsyncEnumerable<T>        {            private readonly Func<IAsyncEnumerator<T>> getEnumerator;            public AnonymousAsyncEnumerable(Func<IAsyncEnumerator<T>> getEnumerator)            {                Debug.Assert(getEnumerator != null);                this.getEnumerator = getEnumerator;            }            public IAsyncEnumerator<T> GetAsyncEnumerator() => getEnumerator();        }        private sealed class AnonymousAsyncEnumerableWithTask<T> : IAsyncEnumerable<T>        {            private readonly Func<Task<IAsyncEnumerator<T>>> getEnumerator;            public AnonymousAsyncEnumerableWithTask(Func<Task<IAsyncEnumerator<T>>> getEnumerator)            {                Debug.Assert(getEnumerator != null);                this.getEnumerator = getEnumerator;            }            public IAsyncEnumerator<T> GetAsyncEnumerator() => new Enumerator(getEnumerator);            private sealed class Enumerator : IAsyncEnumerator<T>            {                private Func<Task<IAsyncEnumerator<T>>> getEnumerator;                private IAsyncEnumerator<T> enumerator;                public Enumerator(Func<Task<IAsyncEnumerator<T>>> getEnumerator)                {                    Debug.Assert(getEnumerator != null);                    this.getEnumerator = getEnumerator;                }                public T Current                {                    get                    {                        if (enumerator == null)                            throw new InvalidOperationException();                        return enumerator.Current;                    }                }                public async Task DisposeAsync()                {                    var old = Interlocked.Exchange(ref enumerator, DisposedEnumerator.Instance);                    if (enumerator != null)                    {                        await enumerator.DisposeAsync().ConfigureAwait(false);                    }                }                public Task<bool> MoveNextAsync()                {                    if (enumerator == null)                    {                        return InitAndMoveNextAsync();                    }                    return enumerator.MoveNextAsync();                }                private async Task<bool> InitAndMoveNextAsync()                {                    try                    {                        enumerator = await getEnumerator().ConfigureAwait(false);                    }                    catch (Exception ex)                    {                        enumerator = Throw<T>(ex).GetAsyncEnumerator();                        throw;                    }                    finally                    {                        getEnumerator = null;                    }                    return await enumerator.MoveNextAsync().ConfigureAwait(false);                }                private sealed class DisposedEnumerator : IAsyncEnumerator<T>                {                    public static readonly DisposedEnumerator Instance = new DisposedEnumerator();                    public T Current => throw new ObjectDisposedException("this");                    public Task DisposeAsync() => TaskExt.CompletedTask;                    public Task<bool> MoveNextAsync() => throw new ObjectDisposedException("this");                }            }        }        private sealed class AnonymousAsyncIterator<T> : AsyncIterator<T>        {            private readonly Func<T> currentFunc;            private readonly Func<Task> dispose;            private readonly Func<Task<bool>> moveNext;            public AnonymousAsyncIterator(Func<Task<bool>> moveNext, Func<T> currentFunc, Func<Task> dispose)            {                Debug.Assert(moveNext != null);                this.moveNext = moveNext;                this.currentFunc = currentFunc;                this.dispose = dispose;                // Explicit call to initialize enumerator mode                GetAsyncEnumerator();            }            public override AsyncIterator<T> Clone()            {                throw new NotSupportedException("AnonymousAsyncIterator cannot be cloned. It is only intended for use as an iterator.");            }            public override async Task DisposeAsync()            {                if (dispose != null)                {                    await dispose().ConfigureAwait(false);                }                await base.DisposeAsync().ConfigureAwait(false);            }            protected override async Task<bool> MoveNextCore()            {                switch (state)                {                    case AsyncIteratorState.Allocated:                        state = AsyncIteratorState.Iterating;                        goto case AsyncIteratorState.Iterating;                    case AsyncIteratorState.Iterating:                        if (await moveNext().ConfigureAwait(false))                        {                            current = currentFunc();                            return true;                        }                        await DisposeAsync().ConfigureAwait(false);                        break;                }                return false;            }        }    }}
 |