// 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 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; } } }