// 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; using System.Collections.Generic; using System.Threading; using System.Threading.Tasks; namespace System.Linq { public static partial class AsyncEnumerable { public static Task Count(this IAsyncEnumerable source) { if (source == null) throw Error.ArgumentNull(nameof(source)); return CountCore(source, CancellationToken.None); } public static Task Count(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); return CountCore(source, cancellationToken); } public static Task Count(this IAsyncEnumerable source, Func predicate) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return CountCore(source, predicate, CancellationToken.None); } public static Task Count(this IAsyncEnumerable source, Func predicate, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return CountCore(source, predicate, cancellationToken); } public static Task Count(this IAsyncEnumerable source, Func> predicate) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return CountCore(source, predicate, CancellationToken.None); } public static Task Count(this IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return CountCore(source, predicate, cancellationToken); } private static Task CountCore(IAsyncEnumerable source, CancellationToken cancellationToken) { switch (source) { case ICollection collection: return Task.FromResult(collection.Count); case IAsyncIListProvider listProv: return listProv.GetCountAsync(onlyIfCheap: false, cancellationToken); case ICollection collection: return Task.FromResult(collection.Count); } return Core(); async Task Core() { var count = 0; var e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { checked { count++; } } } finally { await e.DisposeAsync().ConfigureAwait(false); } return count; } } private static async Task CountCore(IAsyncEnumerable source, Func predicate, CancellationToken cancellationToken) { var count = 0; var e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { if (predicate(e.Current)) { checked { count++; } } } } finally { await e.DisposeAsync().ConfigureAwait(false); } return count; } private static async Task CountCore(this IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { var count = 0; var e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { if (await predicate(e.Current).ConfigureAwait(false)) { checked { count++; } } } } finally { await e.DisposeAsync().ConfigureAwait(false); } return count; } } }