diff --git a/docs/project/list-of-diagnostics.md b/docs/project/list-of-diagnostics.md index c40d4a50b30025..e203738bcac936 100644 --- a/docs/project/list-of-diagnostics.md +++ b/docs/project/list-of-diagnostics.md @@ -115,6 +115,7 @@ The PR that reveals the implementation of the ` are obsolete. Use the new ones that take an IComparer\. | ## Analyzer Warnings @@ -185,7 +186,7 @@ The diagnostic id values reserved for .NET Libraries analyzer warnings are `SYSL | __`SYSLIB1059`__ | Marshaller type does not support allocating constructor | | __`SYSLIB1060`__ | Specified marshaller type is invalid | | __`SYSLIB1061`__ | Marshaller type has incompatible method signatures | -| __`SYSLIB1062`__ | Project must be updated with 'true' | +| __`SYSLIB1062`__ | Project must be updated with '\true\' | | __`SYSLIB1063`__ | _`SYSLIB1063`-`SYSLIB1069` reserved for Microsoft.Interop.LibraryImportGenerator._ | | __`SYSLIB1064`__ | _`SYSLIB1063`-`SYSLIB1069` reserved for Microsoft.Interop.LibraryImportGenerator._ | | __`SYSLIB1065`__ | _`SYSLIB1063`-`SYSLIB1069` reserved for Microsoft.Interop.LibraryImportGenerator._ | diff --git a/src/libraries/Common/src/System/Obsoletions.cs b/src/libraries/Common/src/System/Obsoletions.cs index 1947cc3324a5ba..cbba37ed07be17 100644 --- a/src/libraries/Common/src/System/Obsoletions.cs +++ b/src/libraries/Common/src/System/Obsoletions.cs @@ -192,6 +192,9 @@ internal static class Obsoletions internal const string Rfc2898DeriveBytesCtorMessage = "The constructors on Rfc2898DeriveBytes are obsolete. Use the static Pbkdf2 method instead."; internal const string Rfc2898DeriveBytesCtorDiagId = "SYSLIB0060"; + internal const string QueryableMinByMaxByTSourceObsoleteMessage = "The Queryable MinBy and MaxBy taking an IComparer are obsolete. Use the new ones that take an IComparer."; + internal const string QueryableMinByMaxByTSourceObsoleteDiagId = "SYSLIB0061"; + // When adding a new diagnostic ID, add it to the table in docs\project\list-of-diagnostics.md as well. // Keep new const identifiers above this comment. } diff --git a/src/libraries/System.Linq.Queryable/ref/System.Linq.Queryable.cs b/src/libraries/System.Linq.Queryable/ref/System.Linq.Queryable.cs index 38bff9dcfbf048..2d63b858f0bd26 100644 --- a/src/libraries/System.Linq.Queryable/ref/System.Linq.Queryable.cs +++ b/src/libraries/System.Linq.Queryable/ref/System.Linq.Queryable.cs @@ -128,12 +128,20 @@ public static partial class Queryable public static long LongCount(this System.Linq.IQueryable source) { throw null; } public static long LongCount(this System.Linq.IQueryable source, System.Linq.Expressions.Expression> predicate) { throw null; } public static TSource? MaxBy(this System.Linq.IQueryable source, System.Linq.Expressions.Expression> keySelector) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + [System.Runtime.CompilerServices.OverloadResolutionPriorityAttribute(-1)] + [System.ObsoleteAttribute("The Queryable MinBy and MaxBy taking an IComparer are obsolete. Use the new ones that take an IComparer.", DiagnosticId="SYSLIB0061", UrlFormat="https://aka.ms/dotnet-warnings/{0}")] public static TSource? MaxBy(this System.Linq.IQueryable source, System.Linq.Expressions.Expression> keySelector, System.Collections.Generic.IComparer? comparer) { throw null; } + public static TSource? MaxBy(this System.Linq.IQueryable source, System.Linq.Expressions.Expression> keySelector, System.Collections.Generic.IComparer? comparer) { throw null; } public static TSource? Max(this System.Linq.IQueryable source) { throw null; } public static TSource? Max(this System.Linq.IQueryable source, System.Collections.Generic.IComparer? comparer) { throw null; } public static TResult? Max(this System.Linq.IQueryable source, System.Linq.Expressions.Expression> selector) { throw null; } public static TSource? MinBy(this System.Linq.IQueryable source, System.Linq.Expressions.Expression> keySelector) { throw null; } + [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Never)] + [System.Runtime.CompilerServices.OverloadResolutionPriorityAttribute(-1)] + [System.ObsoleteAttribute("The Queryable MinBy and MaxBy taking an IComparer are obsolete. Use the new ones that take an IComparer.", DiagnosticId="SYSLIB0061", UrlFormat="https://aka.ms/dotnet-warnings/{0}")] public static TSource? MinBy(this System.Linq.IQueryable source, System.Linq.Expressions.Expression> keySelector, System.Collections.Generic.IComparer? comparer) { throw null; } + public static TSource? MinBy(this System.Linq.IQueryable source, System.Linq.Expressions.Expression> keySelector, System.Collections.Generic.IComparer? comparer) { throw null; } public static TSource? Min(this System.Linq.IQueryable source) { throw null; } public static TSource? Min(this System.Linq.IQueryable source, System.Collections.Generic.IComparer? comparer) { throw null; } public static TResult? Min(this System.Linq.IQueryable source, System.Linq.Expressions.Expression> selector) { throw null; } diff --git a/src/libraries/System.Linq.Queryable/src/System.Linq.Queryable.csproj b/src/libraries/System.Linq.Queryable/src/System.Linq.Queryable.csproj index 343028ced1ad2a..4023cdc363dae1 100644 --- a/src/libraries/System.Linq.Queryable/src/System.Linq.Queryable.csproj +++ b/src/libraries/System.Linq.Queryable/src/System.Linq.Queryable.csproj @@ -23,4 +23,9 @@ + + + + diff --git a/src/libraries/System.Linq.Queryable/src/System/Linq/Queryable.cs b/src/libraries/System.Linq.Queryable/src/System/Linq/Queryable.cs index 802ef47e0ee711..b686967bc259a9 100644 --- a/src/libraries/System.Linq.Queryable/src/System/Linq/Queryable.cs +++ b/src/libraries/System.Linq.Queryable/src/System/Linq/Queryable.cs @@ -3,8 +3,10 @@ using System.Collections; using System.Collections.Generic; +using System.ComponentModel; using System.Diagnostics.CodeAnalysis; using System.Linq.Expressions; +using System.Runtime.CompilerServices; namespace System.Linq { @@ -1780,11 +1782,14 @@ public static long LongCount(this IQueryable source, Expressio /// The type of key to compare elements by. /// A sequence of values to determine the minimum value of. /// A function to extract the key for each element. - /// The to compare keys. + /// The to compare elements. /// The value with the minimum key in the sequence. /// is . - /// No key extracted from implements the or interface. + /// No key extracted from implements the or interface. [DynamicDependency("MinBy`2", typeof(Enumerable))] + [Obsolete(Obsoletions.QueryableMinByMaxByTSourceObsoleteMessage, DiagnosticId=Obsoletions.QueryableMinByMaxByTSourceObsoleteDiagId, UrlFormat = Obsoletions.SharedUrlFormat)] + [EditorBrowsable(EditorBrowsableState.Never)] + [OverloadResolutionPriority(-1)] public static TSource? MinBy(this IQueryable source, Expression> keySelector, IComparer? comparer) { ArgumentNullException.ThrowIfNull(source); @@ -1799,6 +1804,30 @@ public static long LongCount(this IQueryable source, Expressio Expression.Constant(comparer, typeof(IComparer)))); } + /// Returns the minimum value in a generic according to a specified key selector function. + /// The type of the elements of . + /// The type of key to compare elements by. + /// A sequence of values to determine the minimum value of. + /// A function to extract the key for each element. + /// The to compare keys. + /// The value with the minimum key in the sequence. + /// is . + /// No key extracted from implements the or interface. + [DynamicDependency("MinBy`2", typeof(Enumerable))] + public static TSource? MinBy(this IQueryable source, Expression> keySelector, IComparer? comparer) + { + ArgumentNullException.ThrowIfNull(source); + ArgumentNullException.ThrowIfNull(keySelector); + + return source.Provider.Execute( + Expression.Call( + null, + new Func, Expression>, IComparer, TSource?>(MinBy).Method, + source.Expression, + Expression.Quote(keySelector), + Expression.Constant(comparer, typeof(IComparer)))); + } + [DynamicDependency("Max`1", typeof(Enumerable))] public static TSource? Max(this IQueryable source) { @@ -1870,11 +1899,14 @@ public static long LongCount(this IQueryable source, Expressio /// The type of key to compare elements by. /// A sequence of values to determine the maximum value of. /// A function to extract the key for each element. - /// The to compare keys. + /// The to compare elements. /// The value with the maximum key in the sequence. /// is . - /// No key extracted from implements the or interface. + /// No key extracted from implements the or interface. [DynamicDependency("MaxBy`2", typeof(Enumerable))] + [Obsolete(Obsoletions.QueryableMinByMaxByTSourceObsoleteMessage, DiagnosticId=Obsoletions.QueryableMinByMaxByTSourceObsoleteDiagId, UrlFormat = Obsoletions.SharedUrlFormat)] + [EditorBrowsable(EditorBrowsableState.Never)] + [OverloadResolutionPriority(-1)] public static TSource? MaxBy(this IQueryable source, Expression> keySelector, IComparer? comparer) { ArgumentNullException.ThrowIfNull(source); @@ -1889,6 +1921,30 @@ public static long LongCount(this IQueryable source, Expressio Expression.Constant(comparer, typeof(IComparer)))); } + /// Returns the maximum value in a generic according to a specified key selector function. + /// The type of the elements of . + /// The type of key to compare elements by. + /// A sequence of values to determine the maximum value of. + /// A function to extract the key for each element. + /// The to compare keys. + /// The value with the maximum key in the sequence. + /// is . + /// No key extracted from implements the or interface. + [DynamicDependency("MaxBy`2", typeof(Enumerable))] + public static TSource? MaxBy(this IQueryable source, Expression> keySelector, IComparer? comparer) + { + ArgumentNullException.ThrowIfNull(source); + ArgumentNullException.ThrowIfNull(keySelector); + + return source.Provider.Execute( + Expression.Call( + null, + new Func, Expression>, IComparer, TSource?>(MaxBy).Method, + source.Expression, + Expression.Quote(keySelector), + Expression.Constant(comparer, typeof(IComparer)))); + } + [DynamicDependency("Sum", typeof(Enumerable))] public static int Sum(this IQueryable source) { diff --git a/src/libraries/System.Linq.Queryable/tests/MaxTests.cs b/src/libraries/System.Linq.Queryable/tests/MaxTests.cs index 6c487c7c92f861..56db0d4a843b10 100644 --- a/src/libraries/System.Linq.Queryable/tests/MaxTests.cs +++ b/src/libraries/System.Linq.Queryable/tests/MaxTests.cs @@ -638,5 +638,26 @@ public void MaxBy_CustomComparer() IQueryable source = Enumerable.Range(1, 20).AsQueryable(); Assert.Equal(20, source.MaxBy(x => -x, Comparer.Create((x, y) => -x.CompareTo(y)))); } + + private sealed class Folk + { + public string Name { get; set; } + public int Age { get; set; } + } + + [Fact] + public void MaxBy_CustomComparer_DistinctTypes() + { + var data = new Folk[] { + new Folk { Name="Doug", Age=42 }, + new Folk { Name="John", Age=18 }, + new Folk { Name="Bob", Age=21 } + }; + + IQueryable source = data.AsQueryable(); + var result = source.MaxBy(e => e.Age, Comparer.Default); + Assert.NotNull(result); + Assert.Equal("Doug", result.Name); + } } } diff --git a/src/libraries/System.Linq.Queryable/tests/MinTests.cs b/src/libraries/System.Linq.Queryable/tests/MinTests.cs index 2cbee36d713866..be5580d9d01e9c 100644 --- a/src/libraries/System.Linq.Queryable/tests/MinTests.cs +++ b/src/libraries/System.Linq.Queryable/tests/MinTests.cs @@ -606,5 +606,26 @@ public void MinBy_CustomComparer() IQueryable source = Enumerable.Range(1, 20).AsQueryable(); Assert.Equal(1, source.MinBy(x => -x, Comparer.Create((x, y) => -x.CompareTo(y)))); } + + private sealed class Folk + { + public string Name { get; set; } + public int Age { get; set; } + } + + [Fact] + public void MinBy_CustomComparer_DistinctTypes() + { + var data = new Folk[] { + new Folk { Name="Doug", Age=42 }, + new Folk { Name="John", Age=18 }, + new Folk { Name="Bob", Age=21 } + }; + + IQueryable source = data.AsQueryable(); + var result = source.MinBy(e => e.Age, Comparer.Default); + Assert.NotNull(result); + Assert.Equal("John", result.Name); + } } } diff --git a/src/libraries/apicompat/ApiCompatBaseline.NetCoreAppLatestStable.xml b/src/libraries/apicompat/ApiCompatBaseline.NetCoreAppLatestStable.xml index 1f33b65aeebd3f..5ebda682ce7aea 100644 --- a/src/libraries/apicompat/ApiCompatBaseline.NetCoreAppLatestStable.xml +++ b/src/libraries/apicompat/ApiCompatBaseline.NetCoreAppLatestStable.xml @@ -331,4 +331,4 @@ net9.0/System.Runtime.dll net10.0/System.Runtime.dll - \ No newline at end of file +