// 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 ValueTask CountAsync(this IAsyncEnumerable source, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); switch (source) { case ICollection collection: return new ValueTask(collection.Count); case IAsyncIListProvider listProv: return listProv.GetCountAsync(onlyIfCheap: false, cancellationToken); case ICollection collection: return new ValueTask(collection.Count); } return Core(source, cancellationToken); static async ValueTask Core(IAsyncEnumerable _source, CancellationToken _cancellationToken) { var count = 0; await foreach (var item in AsyncEnumerableExtensions.WithCancellation(_source, _cancellationToken).ConfigureAwait(false)) { checked { count++; } } return count; } } public static ValueTask CountAsync(this IAsyncEnumerable source, Func predicate, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return Core(source, predicate, cancellationToken); static async ValueTask Core(IAsyncEnumerable _source, Func _predicate, CancellationToken _cancellationToken) { var count = 0; await foreach (var item in AsyncEnumerableExtensions.WithCancellation(_source, _cancellationToken).ConfigureAwait(false)) { if (_predicate(item)) { checked { count++; } } } return count; } } internal static ValueTask CountAwaitAsyncCore(this IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return Core(source, predicate, cancellationToken); static async ValueTask Core(IAsyncEnumerable _source, Func> _predicate, CancellationToken _cancellationToken) { var count = 0; await foreach (var item in AsyncEnumerableExtensions.WithCancellation(_source, _cancellationToken).ConfigureAwait(false)) { if (await _predicate(item).ConfigureAwait(false)) { checked { count++; } } } return count; } } #if !NO_DEEP_CANCELLATION internal static ValueTask CountAwaitWithCancellationAsyncCore(this IAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (predicate == null) throw Error.ArgumentNull(nameof(predicate)); return Core(source, predicate, cancellationToken); static async ValueTask Core(IAsyncEnumerable _source, Func> _predicate, CancellationToken _cancellationToken) { var count = 0; await foreach (var item in AsyncEnumerableExtensions.WithCancellation(_source, _cancellationToken).ConfigureAwait(false)) { if (await _predicate(item, _cancellationToken).ConfigureAwait(false)) { checked { count++; } } } return count; } } #endif } }