// 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; using System.Collections.Generic; using System.Linq; using System.Threading; using System.Threading.Tasks; namespace System.Linq { public static partial class AsyncEnumerable { public static Task Min(this IAsyncEnumerable source, Func selector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return Min(source, selector, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source, Func selector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return Min(source, selector, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source, IComparer comparer, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (comparer == null) throw new ArgumentNullException(nameof(comparer)); return Min_(source, comparer, cancellationToken); } public static Task Min(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Min(source, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Min(source, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Min(source, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Min(source, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Min(source, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Min(source, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Min(source, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Min(source, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Min(source, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Min(source, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source) { if (source == null) throw new ArgumentNullException(nameof(source)); return Min(source, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source, Func selector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return Min(source, selector, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source, Func selector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return Min(source, selector, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source, Func selector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return Min(source, selector, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source, Func selector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return Min(source, selector, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source, Func selector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return Min(source, selector, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source, Func selector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return Min(source, selector, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source, Func selector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return Min(source, selector, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source, Func selector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return Min(source, selector, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); return source.Aggregate(Math.Min, cancellationToken); } public static Task Min(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); return source.Aggregate(Math.Min, cancellationToken); } public static Task Min(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); return source.Aggregate(Math.Min, cancellationToken); } public static Task Min(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); return source.Aggregate(Math.Min, cancellationToken); } public static Task Min(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); return source.Aggregate(Math.Min, cancellationToken); } public static Task Min(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); return source.Aggregate(default(int?), NullableMin, cancellationToken); } public static Task Min(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); return source.Aggregate(default(long?), NullableMin, cancellationToken); } public static Task Min(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); return source.Aggregate(default(double?), NullableMin, cancellationToken); } public static Task Min(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); return source.Aggregate(default(float?), NullableMin, cancellationToken); } public static Task Min(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); return source.Aggregate(default(decimal?), NullableMin, cancellationToken); } public static Task Min(this IAsyncEnumerable source, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); var comparer = Comparer.Default; return source.Aggregate((x, y) => comparer.Compare(x, y) <= 0 ? x : y, cancellationToken); } public static Task Min(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return source.Select(selector) .Min(cancellationToken); } public static Task Min(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return source.Select(selector) .Min(cancellationToken); } public static Task Min(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return source.Select(selector) .Min(cancellationToken); } public static Task Min(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return source.Select(selector) .Min(cancellationToken); } public static Task Min(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return source.Select(selector) .Min(cancellationToken); } public static Task Min(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return source.Select(selector) .Min(cancellationToken); } public static Task Min(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return source.Select(selector) .Min(cancellationToken); } public static Task Min(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return source.Select(selector) .Min(cancellationToken); } public static Task Min(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return source.Select(selector) .Min(cancellationToken); } public static Task Min(this IAsyncEnumerable source, IComparer comparer) { if (source == null) throw new ArgumentNullException(nameof(source)); if (comparer == null) throw new ArgumentNullException(nameof(comparer)); return source.Min(comparer, CancellationToken.None); } public static Task Min(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return source.Select(selector) .Min(cancellationToken); } public static Task Min(this IAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return source.Select(selector) .Min(cancellationToken); } public static Task Min(this IAsyncEnumerable source, Func selector) { if (source == null) throw new ArgumentNullException(nameof(source)); if (selector == null) throw new ArgumentNullException(nameof(selector)); return Min(source, selector, CancellationToken.None); } 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 source.MinBy(keySelector, CancellationToken.None); } 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 source.MinBy(keySelector, comparer, 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 MinBy(source, keySelector, Comparer.Default, cancellationToken); } 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 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(); using (var e = source.GetEnumerator()) { if (!await e.MoveNext(cancellationToken) .ConfigureAwait(false)) throw new InvalidOperationException(Strings.NO_ELEMENTS); var current = e.Current; var resKey = keySelector(current); result.Add(current); while (await e.MoveNext(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; } } } return result; } private static async Task Min_(IAsyncEnumerable source, IComparer comparer, CancellationToken cancellationToken) { return (await MinBy(source, x => x, comparer, cancellationToken) .ConfigureAwait(false)).First(); } private static T? NullableMin(T? x, T? y) where T : struct, IComparable { if (!x.HasValue) return y; if (!y.HasValue) return x; if (x.Value.CompareTo(y.Value) <= 0) return x; return y; } } }