| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120 | // 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)        {            return new AnonymousAsyncEnumerable<T>(getEnumerator);        }        public static IAsyncEnumerator<T> CreateEnumerator<T>(Func<CancellationToken, Task<bool>> moveNext, Func<T> current, Action dispose)        {            return new AnonymousAsyncIterator<T>(moveNext, current, dispose);        }        private static IAsyncEnumerator<T> CreateEnumerator<T>(Func<CancellationToken, TaskCompletionSource<bool>, Task<bool>> moveNext, Func<T> current, Action dispose)        {            var self = new AnonymousAsyncIterator<T>(                async ct =>                {                    var tcs = new TaskCompletionSource<bool>();                    var stop = new Action(                        () =>                        {                            tcs.TrySetCanceled();                        });                    using (ct.Register(stop))                    {                        return await moveNext(ct, tcs)                                   .ConfigureAwait(false);                    }                },                current,                dispose            );            return self;        }        private class AnonymousAsyncEnumerable<T> : IAsyncEnumerable<T>        {            private readonly Func<IAsyncEnumerator<T>> getEnumerator;            public AnonymousAsyncEnumerable(Func<IAsyncEnumerator<T>> getEnumerator)            {                this.getEnumerator = getEnumerator;            }            public IAsyncEnumerator<T> GetEnumerator()            {                return getEnumerator();            }        }        private sealed class AnonymousAsyncIterator<T> : AsyncIterator<T>        {            private readonly Func<T> currentFunc;            private readonly Action dispose;            private readonly Func<CancellationToken, Task<bool>> moveNext;            public AnonymousAsyncIterator(Func<CancellationToken, Task<bool>> moveNext, Func<T> currentFunc, Action dispose)            {                Debug.Assert(moveNext != null);                Debug.Assert(currentFunc != null);                Debug.Assert(dispose != null);                this.moveNext = moveNext;                this.currentFunc = currentFunc;                this.dispose = dispose;                // Explicit call to initialize enumerator mode                GetEnumerator();            }            public override AsyncIterator<T> Clone()            {                throw new NotSupportedException("Iterator only");            }            public override void Dispose()            {                dispose?.Invoke();                base.Dispose();            }            protected override async Task<bool> MoveNextCore(CancellationToken cancellationToken)            {                switch (state)                {                    case AsyncIteratorState.Allocated:                        state = AsyncIteratorState.Iterating;                        goto case AsyncIteratorState.Iterating;                    case AsyncIteratorState.Iterating:                        if (await moveNext(cancellationToken).ConfigureAwait(false))                        {                            current = currentFunc();                            return true;                        }                        Dispose();                        break;                }                return false;            }        }    }}
 |