// 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) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return AggregateCore(source, accumulator, CancellationToken.None); } public static Task AggregateAsync(this IAsyncEnumerable source, Func accumulator, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return AggregateCore(source, accumulator, cancellationToken); } public static Task AggregateAsync(this IAsyncEnumerable source, Func> accumulator) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return AggregateCore(source, accumulator, CancellationToken.None); } public static Task AggregateAsync(this IAsyncEnumerable source, Func> accumulator, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return AggregateCore(source, accumulator, cancellationToken); } #if !NO_DEEP_CANCELLATION public static Task AggregateAsync(this IAsyncEnumerable source, Func> accumulator, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return AggregateCore(source, accumulator, cancellationToken); } #endif public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func accumulator) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return AggregateCore(source, seed, accumulator, x => x, CancellationToken.None); } public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func accumulator, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return AggregateCore(source, seed, accumulator, x => x, cancellationToken); } public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func> accumulator) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return AggregateCore(source, seed, accumulator, CancellationToken.None); } public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func> accumulator, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return AggregateCore(source, seed, accumulator, cancellationToken); } #if !NO_DEEP_CANCELLATION public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func> accumulator, CancellationToken cancellationToken) { if (source == null) throw Error.ArgumentNull(nameof(source)); if (accumulator == null) throw Error.ArgumentNull(nameof(accumulator)); return AggregateCore(source, seed, accumulator, cancellationToken); } #endif public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func accumulator, Func resultSelector) { 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 AggregateCore(source, seed, accumulator, resultSelector, CancellationToken.None); } public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func accumulator, Func resultSelector, CancellationToken cancellationToken) { 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 AggregateCore(source, seed, accumulator, resultSelector, cancellationToken); } public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func> accumulator, Func> resultSelector) { 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 AggregateCore(source, seed, accumulator, resultSelector, CancellationToken.None); } public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func> accumulator, Func> resultSelector, CancellationToken cancellationToken) { 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 AggregateCore(source, seed, accumulator, resultSelector, cancellationToken); } #if !NO_DEEP_CANCELLATION public static Task AggregateAsync(this IAsyncEnumerable source, TAccumulate seed, Func> accumulator, Func> resultSelector, CancellationToken cancellationToken) { 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 AggregateCore(source, seed, accumulator, resultSelector, cancellationToken); } #endif private static async Task AggregateCore(IAsyncEnumerable source, TAccumulate seed, Func accumulator, Func resultSelector, CancellationToken cancellationToken) { 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); } private static async Task AggregateCore(IAsyncEnumerable source, Func accumulator, CancellationToken cancellationToken) { 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); } } private static async Task AggregateCore(IAsyncEnumerable source, TResult seed, Func> accumulator, CancellationToken cancellationToken) { 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 private static async Task AggregateCore(IAsyncEnumerable source, TResult seed, Func> accumulator, CancellationToken cancellationToken) { 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 private static async Task AggregateCore(IAsyncEnumerable source, TAccumulate seed, Func> accumulator, Func> resultSelector, CancellationToken cancellationToken) { 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 private static async Task AggregateCore(IAsyncEnumerable source, TAccumulate seed, Func> accumulator, Func> resultSelector, CancellationToken cancellationToken) { 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 private static async Task AggregateCore(IAsyncEnumerable source, Func> accumulator, CancellationToken cancellationToken) { 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 private static async Task AggregateCore(IAsyncEnumerable source, Func> accumulator, CancellationToken cancellationToken) { 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 } }