Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Breaking change]: Fix type of the IComparer<T> on System.Linq.Queryable MinBy and MaxBy #45535

Closed
3 tasks done
IDisposable opened this issue Mar 26, 2025 · 8 comments · Fixed by #45565
Closed
3 tasks done
Assignees
Labels
breaking-change Indicates a .NET Core breaking change in-pr This issue will be closed (fixed) by an active pull request. ⌚ Not Triaged Not triaged

Comments

@IDisposable
Copy link
Contributor

IDisposable commented Mar 26, 2025

Description

Change the generic type of the IComparer used in the MinBy and MaxBy in System.Linq.Queryable to the correct TKey instead of the current (and wrong) TSource.

Specifically, change wrong use of IComparer<TSource> for comparer

public static TSource? MinBy<TSource, TKey>(this IQueryable<TSource> source, Expression<Func<TSource, TKey>> keySelector, IComparer<TSource>? comparer)

public static TSource? MaxBy<TSource, TKey>(this IQueryable<TSource> source, Expression<Func<TSource, TKey>> keySelector, IComparer<TSource>? comparer)

to the correct IComparer<TKey> for comparer

public static TSource? MinBy<TSource, TKey>(this IQueryable<TSource> source, Expression<Func<TSource, TKey>> keySelector, IComparer<TKey>? comparer)

public static TSource? MaxBy<TSource, TKey>(this IQueryable<TSource> source, Expression<Func<TSource, TKey>> keySelector, IComparer<TKey>? comparer)

Version

.NET 10 Preview 4

Previous behavior

The versions of MaxBy and MinBy in System.Linq.Queryable that accepted a custom comparer were declared as taking an IComparer<TSource> when they SHOULD have been IComparer<TKey> to match what the keySelector was projecting.

Prior attempts to use a custom comparer would have only worked if TSource and TKey were the same type (e.g. both string or int). Other type combinations would result in an System.IndexOutOfRangeException as documented here in Dotnet Runtime Issue #113878 because the correct backing MaxBy or backing MinBy Method could not be located.

Note, this was never a compile-time error, it would only fail at runtime.

Unhandled exception. System.IndexOutOfRangeException: Index was outside the bounds of the array.
   at System.Linq.EnumerableRewriter.FindEnumerableMethodForQueryable(String name, ReadOnlyCollection`1 args, Type[] typeArgs)
   at System.Linq.EnumerableRewriter.VisitMethodCall(MethodCallExpression m)
   at System.Linq.EnumerableExecutor`1.Execute()
   at System.Linq.EnumerableQuery`1.System.Linq.IQueryProvider.Execute[TElement](Expression expression)
   at System.Linq.Queryable.MaxBy[TSource,TKey](IQueryable`1 source, Expression`1 keySelector, IComparer`1 comparer)
   at Program.<Main>$(String[] args) in C:\Users\eitsarpa\source\repos\ConsoleApp1\ConsoleApp1\Program.cs:line 1

Examples of this working correctly the TSource and TKey types are the both same type:

New behavior

You can use a different type for TSource and TKey in the MaxBy and MinBy that accepted a custom comparer.

This means that any use will work correctly (as long as there is an IComparer<TKey>) and will no longer throw an System.IndexOutOfRangeException.

Type of breaking change

  • Binary incompatible: Existing binaries might encounter a breaking change in behavior, such as failure to load or execute, and if so, require recompilation.
  • Source incompatible: When recompiled using the new SDK or component or to target the new runtime, existing source code might require source changes to compile successfully.
  • Behavioral change: Existing binaries might behave differently at run time.

Reason for change

The original implementation was incorrect see comment on Dotnet Runtime Issue #113878 as specified here and implemented here

Recommended action

None required, things will just quietly work now.

Feature area

LINQ

Affected APIs

System.Linq.Queryable.MinBy and System.Linq.Queryable.MaxBy Linq extension methods.

@teo-tsirpanis
Copy link
Contributor

This is technically both a binary and a source breaking change since the method's signature will be altered, but the previous one was broken. 🤔

@IDisposable
Copy link
Contributor Author

Added those checkmarks :)

@eiriktsarpalis
Copy link
Member

We should hold off from taking any action until any changes are made to the product itself.

@IDisposable
Copy link
Contributor Author

Reserved SYSLIB0061 for this change in dotnet/runtime#113996

@eiriktsarpalis
Copy link
Member

Thanks. Closing this issue since I don't believe obsoleting an API counts as a breaking change.

@teo-tsirpanis
Copy link
Contributor

teo-tsirpanis commented Mar 28, 2025

I don't believe obsoleting an API counts as a breaking change

Some obsoletions have been documented as breaking changes (example 1, example 2, example 3).

@CamSoper
Copy link
Contributor

@eiriktsarpalis We document obsoletions as breaking changes all the time. If you want to reopen, go right ahead! 🙂

@dotnet-policy-service dotnet-policy-service bot added the in-pr This issue will be closed (fixed) by an active pull request. label Mar 28, 2025
@CamSoper CamSoper moved this from 🔖 Ready to 🏗 In progress in dotnet/docs April 2025 sprint project Mar 28, 2025
@dotnetrepoman dotnetrepoman bot added the 🗺️ mapQUEST Only used as a way to mark an issue as updated for quest. RepoMan should instantly remove it. label Mar 28, 2025
@dotnet-policy-service dotnet-policy-service bot removed the 🗺️ mapQUEST Only used as a way to mark an issue as updated for quest. RepoMan should instantly remove it. label Mar 28, 2025
@dotnetrepoman dotnetrepoman bot added the 🗺️ mapQUEST Only used as a way to mark an issue as updated for quest. RepoMan should instantly remove it. label Mar 28, 2025
@dotnet-policy-service dotnet-policy-service bot removed the 🗺️ mapQUEST Only used as a way to mark an issue as updated for quest. RepoMan should instantly remove it. label Mar 28, 2025
@IDisposable
Copy link
Contributor Author

Documentation completed at #45565

@github-project-automation github-project-automation bot moved this from 🏗 In progress to ✅ Done in dotnet/docs April 2025 sprint project Apr 1, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
breaking-change Indicates a .NET Core breaking change in-pr This issue will be closed (fixed) by an active pull request. ⌚ Not Triaged Not triaged
Projects
Development

Successfully merging a pull request may close this issue.

4 participants