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

Fix alleycats Set Functor ambiguous implicits #4678

Merged

Conversation

LaurenceWarne
Copy link
Contributor

Hi, on 2.12.0 I find that I get the error:

ambiguous implicit values:
  both value alleyCatsStdSetMonad in trait SetInstances of type cats.Monad[Set] with cats.Alternative[Set]
  and value alleyCatsSetTraverse in trait SetInstances of type cats.Traverse[Set]
  match expected type cats.Functor[Set] (lsp)

With for example the setup:

import alleycats.std.set._

object test {
  implicitly[cats.Functor[Set]]
}

To try and fix I've used the approach here: https://typelevel.org/cats/guidelines.html#implicit-instance-priority, which fixes the original problem though introduces binary compatibility issues which I don't know how to fix!:

[error]  * abstract synthetic method alleycats$std$SetInstances0$_setter_$alleyCatsStdSetMonad_=(cats.Monad)Unit in interface alleycats.std.SetInstances0 is inherited by class SetInstances in current version.
[error]    filter with: ProblemFilters.exclude[InheritedNewAbstractMethodProblem]("alleycats.std.SetInstances.alleycats$std$SetInstances0$_setter_$alleyCatsStdSetMonad_=")
[error]  * abstract synthetic method alleycats$std$SetInstances1$_setter_$alleyCatsSetTraverse_=(cats.Traverse)Unit in interface alleycats.std.SetInstances1 is inherited by class SetInstances in current version.
[error]    filter with: ProblemFilters.exclude[InheritedNewAbstractMethodProblem]("alleycats.std.SetInstances.alleycats$std$SetInstances1$_setter_$alleyCatsSetTraverse_=")

Suggestions very welcome! Thanks

@satorg
Copy link
Contributor

satorg commented Nov 24, 2024

@LaurenceWarne , thank you for your help!

Regarding the original issue:

ambiguous implicit values:
both value alleyCatsStdSetMonad in trait SetInstances of type cats.Monad[Set] with cats.Alternative[Set]
and value alleyCatsSetTraverse in trait SetInstances of type cats.Traverse[Set]
match expected type cats.Functor[Set] (lsp)

In theory, all these instances (i.e., Traverse, Monad, Alternative, etc) can be (and usually are) gathered in a single implementation class and therefore exposed by a single implicit val. For instance, see

implicit val catsStdInstancesForList
: Traverse[List] with Alternative[List] with Monad[List] with CoflatMap[List] with Align[List] =
new Traverse[List] with Alternative[List] with Monad[List] with CoflatMap[List] with Align[List] {

I'm not sure why in alleycats these instances were separated in the first place. Perhaps, it was just an oversight. I seems that if we merged them in just one implementor, then it would solve the issue. Perhaps, there were a reason to keep them separated, but I feel it may make sense to give it a shot and merge them. And if the effort fails for some reason, it would be nice to leave a comment about it in the sources, because for now it is not obvious at all.

Regarding the binary compatibility issue:
In the case when an implicit val is intentionally removed from a class or trait (and Mima apparently does not like it), the simplest way to overcome that is to remove implicit specifier from the val and furthermore make that val package-private. It should make Mima pretty happy. In other words, do not remove the implicit val physically, but rather un-implicit it and hide.

@LaurenceWarne
Copy link
Contributor Author

In theory, all these instances (i.e., Traverse, Monad, Alternative, etc) can be (and usually are) gathered in a single implementation class and therefore exposed by a single implicit val. For instance, see

implicit val catsStdInstancesForList
: Traverse[List] with Alternative[List] with Monad[List] with CoflatMap[List] with Align[List] =
new Traverse[List] with Alternative[List] with Monad[List] with CoflatMap[List] with Align[List] {

I'm not sure why in alleycats these instances were separated in the first place. Perhaps, it was just an oversight. I seems that if we merged them in just one implementor, then it would solve the issue. Perhaps, there were a reason to keep them separated, but I feel it may make sense to give it a shot and merge them. And if the effort fails for some reason, it would be nice to leave a comment about it in the sources, because for now it is not obvious at all.

Makes sense, and FWIW moving all to one implementor doesn't appear to cause any test failures.

Regarding the binary compatibility issue: In the case when an implicit val is intentionally removed from a class or trait (and Mima apparently does not like it), the simplest way to overcome that is to remove implicit specifier from the val and furthermore make that val package-private. It should make Mima pretty happy. In other words, do not remove the implicit val physically, but rather un-implicit it and hide.

I may have misunderstood 😅 , but after moving to a common implementer and changing the old implicits:

trait SetInstances {
  private[std] val alleyCatsSetTraverse: Traverse[Set] = alleycatsStdInstancesForSet
  private[std] val alleyCatsStdSetMonad: Monad[Set] with Alternative[Set] = alleycatsStdInstancesForSet
  private[std] val alleyCatsSetTraverseFilter: TraverseFilter[Set] = alleycatsStdInstancesForSet

  implicit val alleycatsStdInstancesForSet
    : Monad[Set] with Alternative[Set] with Traverse[Set] with TraverseFilter[Set] = ...

I run into a few different mima issues:

[error]  * abstract synthetic method alleycats$std$SetInstances$_setter_$alleycatsStdInstancesForSet_=(cats.Monad)Unit in interface alleycats.std.SetInstances is present only in current version
[error]    filter with: ProblemFilters.exclude[ReversedMissingMethodProblem]("alleycats.std.SetInstances.alleycats$std$SetInstances$_setter_$alleycatsStdInstancesForSet_=")
[error]  * abstract method alleycatsStdInstancesForSet()cats.Monad in interface alleycats.std.SetInstances is present only in current version
[error]    filter with: ProblemFilters.exclude[ReversedMissingMethodProblem]("alleycats.std.SetInstances.alleycatsStdInstancesForSet")
[error]  * static method alleyCatsSetTraverseFilter()cats.TraverseFilter in class alleycats.std.all does not have a correspondent in current version
[error]    filter with: ProblemFilters.exclude[DirectMissingMethodProblem]("alleycats.std.all.alleyCatsSetTraverseFilter")
[error]  * static method alleyCatsSetTraverse()cats.Traverse in class alleycats.std.all does not have a correspondent in current version
[error]    filter with: ProblemFilters.exclude[DirectMissingMethodProblem]("alleycats.std.all.alleyCatsSetTraverse")
[error]  * static method alleyCatsStdSetMonad()cats.Monad in class alleycats.std.all does not have a correspondent in current version
[error]    filter with: ProblemFilters.exclude[DirectMissingMethodProblem]("alleycats.std.all.alleyCatsStdSetMonad")
[error]  * static method alleyCatsSetTraverseFilter()cats.TraverseFilter in class alleycats.std.set does not have a correspondent in current version
[error]    filter with: ProblemFilters.exclude[DirectMissingMethodProblem]("alleycats.std.set.alleyCatsSetTraverseFilter")
[error]  * static method alleyCatsSetTraverse()cats.Traverse in class alleycats.std.set does not have a correspondent in current version
[error]    filter with: ProblemFilters.exclude[DirectMissingMethodProblem]("alleycats.std.set.alleyCatsSetTraverse")
[error]  * static method alleyCatsStdSetMonad()cats.Monad in class alleycats.std.set does not have a correspondent in current version
[error]    filter with: ProblemFilters.exclude[DirectMissingMethodProblem]("alleycats.std.set.alleyCatsStdSetMonad")

@satorg
Copy link
Contributor

satorg commented Nov 26, 2024

Ah, that's right – Mima gets triggered if you change visibility from public to package-private. I misguided you, sorry. Now I'm recalling that in cases like this one the approach is usually not to change the visibility but rather mark it @deprecated (along with stripping off all implicit whatever), e.g.:

@deprecated("Use catsKernelStdHashForSortedMap override without Order", "2.2.0-M3")
def catsKernelStdHashForSortedMap[K, V](hashK: Hash[K], orderK: Order[K], hashV: Hash[V]): Hash[SortedMap[K, V]] =
new SortedMapHash[K, V]()(hashV, hashK)

Here, hashK: Hash[K], orderK: Order[K], hashV: Hash[V] along with catsKernelStdHashForSortedMap itself were implicit before it was deprecated.

@satorg
Copy link
Contributor

satorg commented Nov 26, 2024

Note, that it is not necessary to merge the TraverseFilter instance with all the others because it has no base type class in common. So technically, you can keep it separated as before. On the other hand I guess it won't hurt if you merge them, probably.

@LaurenceWarne
Copy link
Contributor Author

Ah, that's right – Mima gets triggered if you change visibility from public to package-private. I misguided you, sorry. Now I'm recalling that in cases like this one the approach is usually not to change the visibility but rather mark it @deprecated (along with stripping off all implicit whatever), e.g.:

Nice, thanks that looks to have worked! Another thing was that I had to use implicit def rather than implicit val for the new combined instance, else I got a ReversedMissingMethodProblem error from mima, I hope that's ok.

Also, I wasn't sure what best to put in for since in the deprecations - please shout if it should be something else.

@satorg
Copy link
Contributor

satorg commented Nov 27, 2024

Hmm.. Even without Mima the statement like

val alleyCatsSetTraverse: Traverse[Set] = alleycatsStdInstancesForSet

where alleycatsStdInstancesForSet is val and defined down below in the trait
won't work
because it leads to the forward reference issue.
Perhaps, Mima tries to tell just about that but in its own way.

On the other hand, using implicit def is not exactly very optimal because it will result in reallocating a new instance every time it is demanded. Certainly, there are some cases where implicit def cannot be avoided, but as a rule of thumb Cats attempts to keep the number of allocations as minimal as possible.

Would you give it a shot and make implicit val alleycatsStdInstancesForSet again and move it before any dependent declaration occurs in the trait please?

@LaurenceWarne
Copy link
Contributor Author

Would you give it a shot and make implicit val alleycatsStdInstancesForSet again and move it before any dependent declaration occurs in the trait please?

Sure, I find I appear get the same errors:

[error]  * abstract synthetic method alleycats$std$SetInstances$_setter_$alleycatsStdInstancesForSet_=(cats.Monad)Unit in interface alleycats.std.SetInstances is present only in current version
[error]    filter with: ProblemFilters.exclude[ReversedMissingMethodProblem]("alleycats.std.SetInstances.alleycats$std$SetInstances$_setter_$alleycatsStdInstancesForSet_=")
[error]  * abstract method alleycatsStdInstancesForSet()cats.Monad in interface alleycats.std.SetInstances is present only in current version
[error]    filter with: ProblemFilters.exclude[ReversedMissingMethodProblem]("alleycats.std.SetInstances.alleycatsStdInstancesForSet")

@LaurenceWarne
Copy link
Contributor Author

I pushed it up if you wanted to take a look @satorg 🙂

@satorg
Copy link
Contributor

satorg commented Dec 4, 2024

@LaurenceWarne ,
it was quite tricky, but looks like I managed to overcome it:
a7c7ce1

Looks like moving the instance value into a companion object somehow resolves the compatibility conundrum (or maybe just makes Mima believing it).

Feel free to take this commit and adjust it accordingly, if necessary.
Hopefully, it works as expected.

@LaurenceWarne
Copy link
Contributor Author

@LaurenceWarne , it was quite tricky, but looks like I managed to overcome it: a7c7ce1

Looks like moving the instance value into a companion object somehow resolves the compatibility conundrum (or maybe just makes Mima believing it).

Feel free to take this commit and adjust it accordingly, if necessary. Hopefully, it works as expected.

Great, thanks! I've applied it, only adding a small comment explaining why the new instance isn't a def

@satorg
Copy link
Contributor

satorg commented Dec 4, 2024

Tests failed on ubuntu-latest, 2.13, temurin@17, catsJVM:

2024-12-04T18:13:57.2635351Z ==> X cats.jvm.tests.FutureSuite.Future: coflatMap.coflatten throughMap  3.012s munit.FailException: Failing seed: 2fAhXEoX8givw6BzXBkSKZe_3YtAuY5ouy7u7RWMA2B=
2024-12-04T18:13:57.2636731Z You can reproduce this failure by adding the following override to your suite:
2024-12-04T18:13:57.2637201Z 
2024-12-04T18:13:57.2637592Z   override def scalaCheckInitialSeed = "2fAhXEoX8givw6BzXBkSKZe_3YtAuY5ouy7u7RWMA2B="
2024-12-04T18:13:57.2638137Z 
2024-12-04T18:13:57.2638326Z Exception raised on property evaluation.
2024-12-04T18:13:57.2638860Z > ARG_0: Future(Failure(java.lang.Exception))
2024-12-04T18:13:57.2639612Z > Exception: java.util.concurrent.TimeoutException: Future timed out after [3 seconds]
2024-12-04T18:13:57.2640866Z     at munit.ScalaCheckSuite.propToTry(ScalaCheckSuite.scala:98)
2024-12-04T18:13:57.2642215Z Caused by: java.util.concurrent.TimeoutException: Future timed out after [3 seconds]
2024-12-04T18:13:57.2643573Z     at scala.concurrent.impl.Promise$DefaultPromise.tryAwait0(Promise.scala:248)
2024-12-04T18:13:57.2645051Z     at scala.concurrent.impl.Promise$DefaultPromise.result(Promise.scala:261)
2024-12-04T18:13:57.2646421Z     at scala.concurrent.Await$.$anonfun$result$1(package.scala:201)
2024-12-04T18:13:57.2647838Z     at scala.concurrent.BlockContext$DefaultBlockContext$.blockOn(BlockContext.scala:62)
2024-12-04T18:13:57.2649135Z     at scala.concurrent.Await$.result(package.scala:124)
2024-12-04T18:13:57.2650849Z     at cats.jvm.tests.FutureSuite.cats$jvm$tests$FutureSuite$$$anonfun$eqfa$1(FutureSuite.scala:45)

But it doesn't look related to the PR. Perhaps, some flakiness in FutureSuite.

satorg
satorg previously approved these changes Dec 5, 2024
Copy link
Contributor

@satorg satorg left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks good to me, thanks!

@satorg
Copy link
Contributor

satorg commented Dec 22, 2024

@LaurenceWarne , it looks like your PR got conflicts due to recent merges to the main branch. Sorry for that 🤷 Would you be open to resolve the conflicts in order to make your PR mergeable again (since it looks pretty good-to-go otherwise). Thank you!

@LaurenceWarne
Copy link
Contributor Author

Sure, done!

Copy link
Contributor

@satorg satorg left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you!

@satorg satorg merged commit c7af0bc into typelevel:main Dec 23, 2024
16 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants