// 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; using System.Collections; using System.Collections.Generic; using System.Diagnostics; using System.Linq; using System.Threading; using System.Threading.Tasks; namespace System.Linq { public static partial class AsyncEnumerable { public static IAsyncEnumerable ToAsyncEnumerable(this IEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); // optimize these adapters for lists and collections var ilist = source as IList; if (ilist != null) return new AsyncIListEnumerableAdapter(ilist); var icoll = source as ICollection; if (icoll != null) return new AsyncICollectionEnumerableAdapter(icoll); return new AsyncEnumerableAdapter(source); } public static IAsyncEnumerable ToAsyncEnumerable(this Task task) { if (task == null) throw new ArgumentNullException(nameof(task)); return CreateEnumerable( () => { var called = 0; var value = default(TSource); return CreateEnumerator( async ct => { if (Interlocked.CompareExchange(ref called, 1, 0) == 0) { value = await task.ConfigureAwait(false); return true; } return false; }, () => value, () => { }); }); } public static IEnumerable ToEnumerable(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return ToEnumerable_(source); } private static IEnumerable ToEnumerable_(IAsyncEnumerable source) { using (var e = source.GetEnumerator()) { while (true) { if (!e.MoveNext(CancellationToken.None) .Result) break; var c = e.Current; yield return c; } } } internal sealed class AsyncEnumerableAdapter : AsyncIterator, IIListProvider { private readonly IEnumerable source; private IEnumerator enumerator; public AsyncEnumerableAdapter(IEnumerable source) { Debug.Assert(source != null); this.source = source; } public override AsyncIterator Clone() { return new AsyncEnumerableAdapter(source); } public override void Dispose() { if (enumerator != null) { enumerator.Dispose(); enumerator = null; } base.Dispose(); } protected override Task MoveNextCore(CancellationToken cancellationToken) { switch (state) { case AsyncIteratorState.Allocated: enumerator = source.GetEnumerator(); state = AsyncIteratorState.Iterating; goto case AsyncIteratorState.Iterating; case AsyncIteratorState.Iterating: if (enumerator.MoveNext()) { current = enumerator.Current; return Task.FromResult(true); } Dispose(); break; } return Task.FromResult(false); } // These optimizations rely on the Sys.Linq impls from IEnumerable to optimize // and short circuit as appropriate public Task ToArrayAsync(CancellationToken cancellationToken) { return Task.FromResult(source.ToArray()); } public Task> ToListAsync(CancellationToken cancellationToken) { return Task.FromResult(source.ToList()); } public Task GetCountAsync(bool onlyIfCheap, CancellationToken cancellationToken) { return Task.FromResult(source.Count()); } } internal sealed class AsyncIListEnumerableAdapter : AsyncIterator, IIListProvider, IList { private readonly IList source; private IEnumerator enumerator; public AsyncIListEnumerableAdapter(IList source) { Debug.Assert(source != null); this.source = source; } public override AsyncIterator Clone() { return new AsyncEnumerableAdapter(source); } public override void Dispose() { if (enumerator != null) { enumerator.Dispose(); enumerator = null; } base.Dispose(); } protected override Task MoveNextCore(CancellationToken cancellationToken) { switch (state) { case AsyncIteratorState.Allocated: enumerator = source.GetEnumerator(); state = AsyncIteratorState.Iterating; goto case AsyncIteratorState.Iterating; case AsyncIteratorState.Iterating: if (enumerator.MoveNext()) { current = enumerator.Current; return Task.FromResult(true); } Dispose(); break; } return Task.FromResult(false); } // These optimizations rely on the Sys.Linq impls from IEnumerable to optimize // and short circuit as appropriate public Task ToArrayAsync(CancellationToken cancellationToken) { return Task.FromResult(source.ToArray()); } public Task> ToListAsync(CancellationToken cancellationToken) { return Task.FromResult(source.ToList()); } public Task GetCountAsync(bool onlyIfCheap, CancellationToken cancellationToken) { return Task.FromResult(source.Count()); } IEnumerator IEnumerable.GetEnumerator() => source.GetEnumerator(); IEnumerator IEnumerable.GetEnumerator() => source.GetEnumerator(); void ICollection.Add(T item) => source.Add(item); void ICollection.Clear() => source.Clear(); bool ICollection.Contains(T item) => source.Contains(item); void ICollection.CopyTo(T[] array, int arrayIndex) => source.CopyTo(array, arrayIndex); bool ICollection.Remove(T item) => source.Remove(item); int ICollection.Count => source.Count; bool ICollection.IsReadOnly => source.IsReadOnly; int IList.IndexOf(T item) => source.IndexOf(item); void IList.Insert(int index, T item) => source.Insert(index, item); void IList.RemoveAt(int index) => source.RemoveAt(index); T IList.this[int index] { get { return source[index]; } set { source[index] = value; } } } internal sealed class AsyncICollectionEnumerableAdapter : AsyncIterator, IIListProvider, ICollection { private readonly ICollection source; private IEnumerator enumerator; public AsyncICollectionEnumerableAdapter(ICollection source) { Debug.Assert(source != null); this.source = source; } public override AsyncIterator Clone() { return new AsyncEnumerableAdapter(source); } public override void Dispose() { if (enumerator != null) { enumerator.Dispose(); enumerator = null; } base.Dispose(); } protected override Task MoveNextCore(CancellationToken cancellationToken) { switch (state) { case AsyncIteratorState.Allocated: enumerator = source.GetEnumerator(); state = AsyncIteratorState.Iterating; goto case AsyncIteratorState.Iterating; case AsyncIteratorState.Iterating: if (enumerator.MoveNext()) { current = enumerator.Current; return Task.FromResult(true); } Dispose(); break; } return Task.FromResult(false); } // These optimizations rely on the Sys.Linq impls from IEnumerable to optimize // and short circuit as appropriate public Task ToArrayAsync(CancellationToken cancellationToken) { return Task.FromResult(source.ToArray()); } public Task> ToListAsync(CancellationToken cancellationToken) { return Task.FromResult(source.ToList()); } public Task GetCountAsync(bool onlyIfCheap, CancellationToken cancellationToken) { return Task.FromResult(source.Count()); } IEnumerator IEnumerable.GetEnumerator() => source.GetEnumerator(); IEnumerator IEnumerable.GetEnumerator() => source.GetEnumerator(); void ICollection.Add(T item) => source.Add(item); void ICollection.Clear() => source.Clear(); bool ICollection.Contains(T item) => source.Contains(item); void ICollection.CopyTo(T[] array, int arrayIndex) => source.CopyTo(array, arrayIndex); bool ICollection.Remove(T item) => source.Remove(item); int ICollection.Count => source.Count; bool ICollection.IsReadOnly => source.IsReadOnly; } } }