// 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.Threading; using System.Threading.Tasks; namespace System.Linq { public static partial class AsyncEnumerable { public static Task AggregateAsync(this IAsyncEnumerable source, Func accumulator, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return Core(); async Task Core() { var e = source.GetAsyncEnumerator(cancellationToken); try { if (!await e.MoveNextAsync().ConfigureAwait(false)) { throw Error.NoElements(); } var acc = e.Current; while (await e.MoveNextAsync().ConfigureAwait(false)) { acc = accumulator(acc, e.Current); } return acc; } finally { await e.DisposeAsync().ConfigureAwait(false); } } } public static Task AggregateAsync(this IAsyncEnumerable source, Func> accumulator, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return Core(); async Task Core() { var e = source.GetAsyncEnumerator(cancellationToken); try { if (!await e.MoveNextAsync().ConfigureAwait(false)) { throw Error.NoElements(); } var acc = e.Current; while (await e.MoveNextAsync().ConfigureAwait(false)) { acc = await accumulator(acc, e.Current).ConfigureAwait(false); } return acc; } finally { await e.DisposeAsync().ConfigureAwait(false); } } } #if !NO_DEEP_CANCELLATION public static Task AggregateAsync(this IAsyncEnumerable source, Func> accumulator, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return Core(); async Task Core() { var e = source.GetAsyncEnumerator(cancellationToken); try { if (!await e.MoveNextAsync().ConfigureAwait(false)) { throw Error.NoElements(); } var acc = e.Current; while (await e.MoveNextAsync().ConfigureAwait(false)) { acc = await accumulator(acc, e.Current, cancellationToken).ConfigureAwait(false); } return acc; } finally { await e.DisposeAsync().ConfigureAwait(false); } } } #endif public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func accumulator, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return Core(); async Task Core() { var acc = seed; var e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { acc = accumulator(acc, e.Current); } } finally { await e.DisposeAsync().ConfigureAwait(false); } return acc; } } public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func> accumulator, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return Core(); async Task Core() { var acc = seed; var e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { acc = await accumulator(acc, e.Current).ConfigureAwait(false); } } finally { await e.DisposeAsync().ConfigureAwait(false); } return acc; } } #if !NO_DEEP_CANCELLATION public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func> accumulator, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return Core(); async Task Core() { var acc = seed; var e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { acc = await accumulator(acc, e.Current, cancellationToken).ConfigureAwait(false); } } finally { await e.DisposeAsync().ConfigureAwait(false); } return acc; } } #endif public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func accumulator, Func resultSelector, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); if (resultSelector == null) throw Error.ArgumentNull(nameof(resultSelector)); return Core(); async Task Core() { var acc = seed; var e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { acc = accumulator(acc, e.Current); } } finally { await e.DisposeAsync().ConfigureAwait(false); } return resultSelector(acc); } } public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func> accumulator, Func> resultSelector, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); if (resultSelector == null) throw Error.ArgumentNull(nameof(resultSelector)); return Core(); async Task Core() { var acc = seed; var e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { acc = await accumulator(acc, e.Current).ConfigureAwait(false); } } finally { await e.DisposeAsync().ConfigureAwait(false); } return await resultSelector(acc).ConfigureAwait(false); } } #if !NO_DEEP_CANCELLATION public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func> accumulator, Func> resultSelector, CancellationToken cancellationToken = default) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); if (resultSelector == null) throw Error.ArgumentNull(nameof(resultSelector)); return Core(); async Task Core() { var acc = seed; var e = source.GetAsyncEnumerator(cancellationToken); try { while (await e.MoveNextAsync().ConfigureAwait(false)) { acc = await accumulator(acc, e.Current, cancellationToken).ConfigureAwait(false); } } finally { await e.DisposeAsync().ConfigureAwait(false); } return await resultSelector(acc, cancellationToken).ConfigureAwait(false); } } #endif } }