// 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 AsyncEnumerableEx { public static Task> MinBy(this IAsyncEnumerable source, Func keySelector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); return MinByCore(source, keySelector, Comparer.Default, CancellationToken.None); } public static Task> MinBy(this IAsyncEnumerable source, Func keySelector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); return MinByCore(source, keySelector, Comparer.Default, cancellationToken); } public static Task> MinBy(this IAsyncEnumerable source, Func keySelector, IComparer comparer) { if (source == null) throw new ArgumentNullException(nameof(source)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); if (comparer == null) throw new ArgumentNullException(nameof(comparer)); return MinByCore(source, keySelector, comparer, CancellationToken.None); } public static Task> MinBy(this IAsyncEnumerable source, Func keySelector, IComparer comparer, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); if (comparer == null) throw new ArgumentNullException(nameof(comparer)); return MinByCore(source, keySelector, comparer, cancellationToken); } public static Task> MinBy(this IAsyncEnumerable source, Func> keySelector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); return MinByCore(source, keySelector, Comparer.Default, CancellationToken.None); } public static Task> MinBy(this IAsyncEnumerable source, Func> keySelector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); return MinByCore(source, keySelector, Comparer.Default, cancellationToken); } public static Task> MinBy(this IAsyncEnumerable source, Func> keySelector, IComparer comparer) { if (source == null) throw new ArgumentNullException(nameof(source)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); if (comparer == null) throw new ArgumentNullException(nameof(comparer)); return MinByCore(source, keySelector, comparer, CancellationToken.None); } public static Task> MinBy(this IAsyncEnumerable source, Func> keySelector, IComparer comparer, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (keySelector == null) throw new ArgumentNullException(nameof(keySelector)); if (comparer == null) throw new ArgumentNullException(nameof(comparer)); return MinByCore(source, keySelector, comparer, cancellationToken); } private static Task> MinByCore(IAsyncEnumerable source, Func keySelector, IComparer comparer, CancellationToken cancellationToken) { return ExtremaBy(source, keySelector, (key, minValue) => -comparer.Compare(key, minValue), cancellationToken); } private static Task> MinByCore(IAsyncEnumerable source, Func> keySelector, IComparer comparer, CancellationToken cancellationToken) { return ExtremaBy(source, keySelector, (key, minValue) => -comparer.Compare(key, minValue), cancellationToken); } private static async Task> ExtremaBy(IAsyncEnumerable source, Func keySelector, Func compare, CancellationToken cancellationToken) { var result = new List(); var e = source.GetAsyncEnumerator(cancellationToken); try { if (!await e.MoveNextAsync(cancellationToken).ConfigureAwait(false)) throw new InvalidOperationException(Strings.NO_ELEMENTS); var current = e.Current; var resKey = keySelector(current); result.Add(current); while (await e.MoveNextAsync(cancellationToken).ConfigureAwait(false)) { var cur = e.Current; var key = keySelector(cur); var cmp = compare(key, resKey); if (cmp == 0) { result.Add(cur); } else if (cmp > 0) { result = new List { cur }; resKey = key; } } } finally { await e.DisposeAsync().ConfigureAwait(false); } return result; } private static async Task> ExtremaBy(IAsyncEnumerable source, Func> keySelector, Func compare, CancellationToken cancellationToken) { var result = new List(); var e = source.GetAsyncEnumerator(cancellationToken); try { if (!await e.MoveNextAsync(cancellationToken).ConfigureAwait(false)) throw new InvalidOperationException(Strings.NO_ELEMENTS); var current = e.Current; var resKey = await keySelector(current).ConfigureAwait(false); result.Add(current); while (await e.MoveNextAsync(cancellationToken).ConfigureAwait(false)) { var cur = e.Current; var key = await keySelector(cur).ConfigureAwait(false); var cmp = compare(key, resKey); if (cmp == 0) { result.Add(cur); } else if (cmp > 0) { result = new List { cur }; resKey = key; } } } finally { await e.DisposeAsync().ConfigureAwait(false); } return result; } } }