From 99e9a71e54ca6da97d22bfcfd444e0c49cc8fd28 Mon Sep 17 00:00:00 2001 From: mox692 Date: Wed, 12 Jul 2023 03:50:55 +0900 Subject: [PATCH 01/28] fix apply syntax allocation --- core/src/main/scala/cats/syntax/apply.scala | 25 +++++++++++++++------ 1 file changed, 18 insertions(+), 7 deletions(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 868647b5dc..98ee24b7eb 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -23,13 +23,8 @@ package cats package syntax trait ApplySyntax extends TupleSemigroupalSyntax { - implicit final def catsSyntaxApply[F[_], A](fa: F[A])(implicit F: Apply[F]): Apply.Ops[F, A] = - new Apply.Ops[F, A] { - type TypeClassType = Apply[F] - - val self = fa - val typeClassInstance = F - } + implicit final def applySyntaxBinCompat1[F[_], A](fa: F[A]): ApplySyntaxBinCompat1[F, A] = + new ApplySyntaxBinCompat1(fa) implicit final def catsSyntaxApplyOps[F[_], A](fa: F[A]): ApplyOps[F, A] = new ApplyOps(fa) @@ -40,6 +35,22 @@ private[syntax] trait ApplySyntaxBinCompat0 { new IfApplyOps[F](fa) } +final class ApplySyntaxBinCompat1[F[_], A](private val fa: F[A]) extends AnyVal { + def ap[B, C](fb: F[B])(implicit ev$1: A <:< (B => C), F: Apply[F]): F[C] = + F.ap(fa.asInstanceOf[F[B => C]])(fb) + def productR[B](fb: F[B])(implicit F: Apply[F]): F[B] = F.productR[A, B](fa)(fb) + def productL[B](fb: F[B])(implicit F: Apply[F]): F[A] = F.productL[A, B](fa)(fb) + @inline final def <*>[B, C](fa: F[B])(implicit ev$1: A <:< (B => C), F: Apply[F]): F[C] = + F.<*>[B, C](fa.asInstanceOf[F[B => C]])(fa) + @inline def *>[B](fb: F[B])(implicit F: Apply[F]): F[B] = F.*>[A, B](fa)(fb) + @inline def <*[B](fb: F[B])(implicit F: Apply[F]): F[A] = F.<*[A, B](fa)(fb) + def ap2[B, C, D](fa: F[B], fb: F[C])(implicit ev$1: A <:< ((B, C) => D), F: Apply[F]): F[D] = + F.ap2[B, C, D](fa.asInstanceOf[F[(B, C) => D]])(fa, fb) + def map2[B, C](fb: F[B])(f: (A, B) => C)(implicit F: Apply[F]): F[C] = F.map2[A, B, C](fa, fb)(f) + def map2Eval[B, C](fb: Eval[F[B]])(f: (A, B) => C)(implicit F: Apply[F]): Eval[F[C]] = + F.map2Eval[A, B, C](fa, fb)(f) +} + final class IfApplyOps[F[_]](private val fcond: F[Boolean]) extends AnyVal { @deprecated("Dangerous method, use ifM (a flatMap) or ifF (a map) instead", "2.6.2") From af45152a53e8331fbe5216adfbd422b9f9be83e8 Mon Sep 17 00:00:00 2001 From: mox692 Date: Wed, 12 Jul 2023 05:19:19 +0900 Subject: [PATCH 02/28] not remove original ops --- core/src/main/scala/cats/syntax/apply.scala | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 98ee24b7eb..739c846bfc 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -23,6 +23,14 @@ package cats package syntax trait ApplySyntax extends TupleSemigroupalSyntax { + final def catsSyntaxApply[F[_], A](fa: F[A])(implicit F: Apply[F]): Apply.Ops[F, A] = + new Apply.Ops[F, A] { + type TypeClassType = Apply[F] + + val self = fa + val typeClassInstance = F + } + implicit final def applySyntaxBinCompat1[F[_], A](fa: F[A]): ApplySyntaxBinCompat1[F, A] = new ApplySyntaxBinCompat1(fa) From ef328b8870b3557e8d27ae794848b5794092692a Mon Sep 17 00:00:00 2001 From: mox692 Date: Wed, 12 Jul 2023 05:23:38 +0900 Subject: [PATCH 03/28] fix ApplyBinCompat1 naming --- core/src/main/scala/cats/syntax/apply.scala | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 739c846bfc..2efa7b20be 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -31,8 +31,8 @@ trait ApplySyntax extends TupleSemigroupalSyntax { val typeClassInstance = F } - implicit final def applySyntaxBinCompat1[F[_], A](fa: F[A]): ApplySyntaxBinCompat1[F, A] = - new ApplySyntaxBinCompat1(fa) + implicit final def catsSyntaxApplyBinCompat1[F[_], A](fa: F[A]): ApplyBinCompat1[F, A] = + new ApplyBinCompat1(fa) implicit final def catsSyntaxApplyOps[F[_], A](fa: F[A]): ApplyOps[F, A] = new ApplyOps(fa) @@ -43,7 +43,7 @@ private[syntax] trait ApplySyntaxBinCompat0 { new IfApplyOps[F](fa) } -final class ApplySyntaxBinCompat1[F[_], A](private val fa: F[A]) extends AnyVal { +final class ApplyBinCompat1[F[_], A](private val fa: F[A]) extends AnyVal { def ap[B, C](fb: F[B])(implicit ev$1: A <:< (B => C), F: Apply[F]): F[C] = F.ap(fa.asInstanceOf[F[B => C]])(fb) def productR[B](fb: F[B])(implicit F: Apply[F]): F[B] = F.productR[A, B](fa)(fb) From 0ba479bcc6cb4c24f5a173bc9f729fd2b0e9eb8a Mon Sep 17 00:00:00 2001 From: mox692 Date: Wed, 19 Jul 2023 21:02:17 +0900 Subject: [PATCH 04/28] remove implicit for F --- core/src/main/scala/cats/syntax/apply.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 2efa7b20be..aef44aa5e0 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -23,7 +23,7 @@ package cats package syntax trait ApplySyntax extends TupleSemigroupalSyntax { - final def catsSyntaxApply[F[_], A](fa: F[A])(implicit F: Apply[F]): Apply.Ops[F, A] = + final def catsSyntaxApply[F[_], A](fa: F[A])(F: Apply[F]): Apply.Ops[F, A] = new Apply.Ops[F, A] { type TypeClassType = Apply[F] From 0c642f63307351582c0da2bff2a2f1cd5b10b884 Mon Sep 17 00:00:00 2001 From: kim / Motoyuki Kimura <55653825+mox692@users.noreply.github.com> Date: Fri, 21 Jul 2023 00:09:43 +0900 Subject: [PATCH 05/28] Add deprecate annotation to old ops Co-authored-by: Arman Bilge --- core/src/main/scala/cats/syntax/apply.scala | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index aef44aa5e0..4575347693 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -23,7 +23,8 @@ package cats package syntax trait ApplySyntax extends TupleSemigroupalSyntax { - final def catsSyntaxApply[F[_], A](fa: F[A])(F: Apply[F]): Apply.Ops[F, A] = + @deprecated("Use `catsSyntaxApplyBinCompat1`", "2.10.0") + final def catsSyntaxApply[F[_], A](fa: F[A], F: Apply[F]): Apply.Ops[F, A] = new Apply.Ops[F, A] { type TypeClassType = Apply[F] From e7d544f0b12fbf1651a415a477f35d0b649cc48a Mon Sep 17 00:00:00 2001 From: mox692 Date: Sun, 23 Jul 2023 18:42:27 +0900 Subject: [PATCH 06/28] create another syntax classes --- core/src/main/scala/cats/syntax/all.scala | 1 + core/src/main/scala/cats/syntax/apply.scala | 59 +++++++++++++------ core/src/main/scala/cats/syntax/package.scala | 2 +- 3 files changed, 43 insertions(+), 19 deletions(-) diff --git a/core/src/main/scala/cats/syntax/all.scala b/core/src/main/scala/cats/syntax/all.scala index 60d6d2b150..8f7b3453db 100644 --- a/core/src/main/scala/cats/syntax/all.scala +++ b/core/src/main/scala/cats/syntax/all.scala @@ -111,6 +111,7 @@ trait AllSyntaxBinCompat3 extends UnorderedFoldableSyntax with Function1Syntax trait AllSyntaxBinCompat4 extends TraverseFilterSyntaxBinCompat0 with ApplySyntaxBinCompat0 + with ApplySyntaxBinCompat1 with ParallelApplySyntax with FoldableSyntaxBinCompat0 with ReducibleSyntaxBinCompat0 diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 4575347693..422672e683 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -23,7 +23,7 @@ package cats package syntax trait ApplySyntax extends TupleSemigroupalSyntax { - @deprecated("Use `catsSyntaxApplyBinCompat1`", "2.10.0") + @deprecated("Use `ApplySyntaxBinCompat1`", "2.10.0") final def catsSyntaxApply[F[_], A](fa: F[A], F: Apply[F]): Apply.Ops[F, A] = new Apply.Ops[F, A] { type TypeClassType = Apply[F] @@ -32,9 +32,6 @@ trait ApplySyntax extends TupleSemigroupalSyntax { val typeClassInstance = F } - implicit final def catsSyntaxApplyBinCompat1[F[_], A](fa: F[A]): ApplyBinCompat1[F, A] = - new ApplyBinCompat1(fa) - implicit final def catsSyntaxApplyOps[F[_], A](fa: F[A]): ApplyOps[F, A] = new ApplyOps(fa) } @@ -44,20 +41,46 @@ private[syntax] trait ApplySyntaxBinCompat0 { new IfApplyOps[F](fa) } -final class ApplyBinCompat1[F[_], A](private val fa: F[A]) extends AnyVal { - def ap[B, C](fb: F[B])(implicit ev$1: A <:< (B => C), F: Apply[F]): F[C] = - F.ap(fa.asInstanceOf[F[B => C]])(fb) - def productR[B](fb: F[B])(implicit F: Apply[F]): F[B] = F.productR[A, B](fa)(fb) - def productL[B](fb: F[B])(implicit F: Apply[F]): F[A] = F.productL[A, B](fa)(fb) - @inline final def <*>[B, C](fa: F[B])(implicit ev$1: A <:< (B => C), F: Apply[F]): F[C] = - F.<*>[B, C](fa.asInstanceOf[F[B => C]])(fa) - @inline def *>[B](fb: F[B])(implicit F: Apply[F]): F[B] = F.*>[A, B](fa)(fb) - @inline def <*[B](fb: F[B])(implicit F: Apply[F]): F[A] = F.<*[A, B](fa)(fb) - def ap2[B, C, D](fa: F[B], fb: F[C])(implicit ev$1: A <:< ((B, C) => D), F: Apply[F]): F[D] = - F.ap2[B, C, D](fa.asInstanceOf[F[(B, C) => D]])(fa, fb) - def map2[B, C](fb: F[B])(f: (A, B) => C)(implicit F: Apply[F]): F[C] = F.map2[A, B, C](fa, fb)(f) - def map2Eval[B, C](fb: Eval[F[B]])(f: (A, B) => C)(implicit F: Apply[F]): Eval[F[C]] = - F.map2Eval[A, B, C](fa, fb)(f) +private[syntax] trait ApplySyntaxBinCompat1 { + implicit final def applyFABOps[F[_], A, B](fab: F[A => B]): ApplyFABOps[F, A, B] = + new ApplyFABOps[F, A, B](fab) + + implicit final def apply2Ops[F[_], A, B, C](ff: F[(A, B) => C]): Apply2Ops[F, A, B, C] = + new Apply2Ops[F, A, B, C](ff) + + implicit final def productOps[F[_], A, B](fa: F[A]): ProductOps[F, A, B] = + new ProductOps[F, A, B](fa) + + implicit final def map2Ops[F[_], A, B, C](fa: F[A]): Map2Ops[F, A, B, C] = + new Map2Ops[F, A, B, C](fa) +} + +final class ApplyFABOps[F[_], A, B](private val fab: F[A => B]) extends AnyVal { + def ap(fa: F[A])(implicit F: Apply[F]): F[B] = F.ap(fab)(fa) + + def <*>(fa: F[A])(implicit F: Apply[F]): F[B] = F.<*>[A, B](fab)(fa) +} + +final class Apply2Ops[F[_], A, B, C](private val ff: F[(A, B) => C]) extends AnyVal { + def ap2(fa: F[A], fb: F[B])(implicit F: Apply[F]): F[C] = F.ap2(ff)(fa, fb) +} + +final class ProductOps[F[_], A, B](private val fa: F[A]) extends AnyVal { + def productR(fb: F[B])(implicit F: Apply[F]): F[B] = F.productR(fa)(fb) + + def productL(fb: F[B])(implicit F: Apply[F]): F[A] = F.productL(fa)(fb) + + def *>(fb: F[B])(implicit F: Apply[F]): F[B] = F.*>(fa)(fb) + + def <*(fb: F[B])(implicit F: Apply[F]): F[A] = F.<*(fa)(fb) +} + +final class Map2Ops[F[_], A, B, C](private val fa: F[A]) extends AnyVal { + def map2(fb: F[B])(f: (A, B) => C)(implicit F: Apply[F]): F[C] = + F.map2(fa, fb)(f) + + def map2Eval(fb: Eval[F[B]])(f: (A, B) => C)(implicit F: Apply[F]): Eval[F[C]] = + F.map2Eval(fa, fb)(f) } final class IfApplyOps[F[_]](private val fcond: F[Boolean]) extends AnyVal { diff --git a/core/src/main/scala/cats/syntax/package.scala b/core/src/main/scala/cats/syntax/package.scala index 46b743455f..5d02b8f134 100644 --- a/core/src/main/scala/cats/syntax/package.scala +++ b/core/src/main/scala/cats/syntax/package.scala @@ -27,7 +27,7 @@ package object syntax { object alternative extends AlternativeSyntax object applicative extends ApplicativeSyntax object applicativeError extends ApplicativeErrorSyntax - object apply extends ApplySyntax with ApplySyntaxBinCompat0 + object apply extends ApplySyntax with ApplySyntaxBinCompat0 with ApplySyntaxBinCompat1 object arrow extends ArrowSyntax object arrowChoice extends ArrowChoiceSyntax object bifunctor extends BifunctorSyntax From 7ca674352b48bfb0ef9ff6674b082b8f08269d01 Mon Sep 17 00:00:00 2001 From: mox692 Date: Sun, 23 Jul 2023 19:42:38 +0900 Subject: [PATCH 07/28] Add doc comments --- core/src/main/scala/cats/syntax/apply.scala | 163 ++++++++++++++++++++ 1 file changed, 163 insertions(+) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 422672e683..f34a858b56 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -56,29 +56,192 @@ private[syntax] trait ApplySyntaxBinCompat1 { } final class ApplyFABOps[F[_], A, B](private val fab: F[A => B]) extends AnyVal { + + /** + * @see [[Apply.ap]]. + * + * Example: + * {{{ + * scala> import cats.syntax.all._ + * + * scala> val someF: Option[Int => Long] = Some(_.toLong + 1L) + * scala> val noneF: Option[Int => Long] = None + * scala> val someInt: Option[Int] = Some(3) + * scala> val noneInt: Option[Int] = None + * + * scala> someF.ap(someInt) + * res0: Option[Long] = Some(4) + * + * scala> noneF.ap(someInt) + * res1: Option[Long] = None + * + * scala> someF.ap(noneInt) + * res2: Option[Long] = None + * + * scala> noneF.ap(noneInt) + * res3: Option[Long] = None + * }}} + */ def ap(fa: F[A])(implicit F: Apply[F]): F[B] = F.ap(fab)(fa) + /** + * Alias for [[ap]]. + */ def <*>(fa: F[A])(implicit F: Apply[F]): F[B] = F.<*>[A, B](fab)(fa) } final class Apply2Ops[F[_], A, B, C](private val ff: F[(A, B) => C]) extends AnyVal { + + /** + * @see [[Apply.ap2]]. + * + * Example: + * {{{ + * scala> import cats.syntax.all._ + * + * scala> val someF: Option[(Int, Int) => Long] = Some((a, b) => (a + b).toLong) + * scala> val noneF: Option[(Int, Int) => Long] = None + * scala> val someInt1: Option[Int] = Some(3) + * scala> val someInt2: Option[Int] = Some(2) + * scala> val noneInt: Option[Int] = None + * + * scala> someF.ap2(someInt1, someInt2) + * res0: Option[Long] = Some(5) + * + * scala> noneF.ap2(someInt1, someInt2) + * res1: Option[Long] = None + * + * scala> someF.ap2(noneInt, noneInt) + * res2: Option[Long] = None + * + * scala> noneF.ap2(noneInt, noneInt) + * res3: Option[Long] = None + * }}} + * + */ def ap2(fa: F[A], fb: F[B])(implicit F: Apply[F]): F[C] = F.ap2(ff)(fa, fb) } final class ProductOps[F[_], A, B](private val fa: F[A]) extends AnyVal { + + /** + * @see [[Apply.productR]]. + * + * Example: + * {{{ + * scala> import cats.syntax.all._ + * scala> import cats.data.Validated + * scala> import Validated.{Valid, Invalid} + * + * scala> type ErrOr[A] = Validated[String, A] + * + * scala> val validInt: ErrOr[Int] = Valid(3) + * scala> val validBool: ErrOr[Boolean] = Valid(true) + * scala> val invalidInt: ErrOr[Int] = Invalid("Invalid int.") + * scala> val invalidBool: ErrOr[Boolean] = Invalid("Invalid boolean.") + * + * scala> validInt.productR(validBool) + * res0: ErrOr[Boolean] = Valid(true) + * + * scala> invalidInt.productR(validBool) + * res1: ErrOr[Boolean] = Invalid(Invalid int.) + * + * scala> validInt.productR(invalidBool) + * res2: ErrOr[Boolean] = Invalid(Invalid boolean.) + * + * scala> invalidInt.productR(invalidBool) + * res3: ErrOr[Boolean] = Invalid(Invalid int.Invalid boolean.) + * }}} + */ def productR(fb: F[B])(implicit F: Apply[F]): F[B] = F.productR(fa)(fb) + /** + * @see [[Apply.productL]]. + * + * Example: + * {{{ + * scala> import cats.syntax.all._ + * scala> import cats.data.Validated + * scala> import Validated.{Valid, Invalid} + * + * scala> type ErrOr[A] = Validated[String, A] + * + * scala> val validInt: ErrOr[Int] = Valid(3) + * scala> val validBool: ErrOr[Boolean] = Valid(true) + * scala> val invalidInt: ErrOr[Int] = Invalid("Invalid int.") + * scala> val invalidBool: ErrOr[Boolean] = Invalid("Invalid boolean.") + * + * scala> validInt.productL(validBool) + * res0: ErrOr[Int] = Valid(3) + * + * scala> invalidInt.productL(validBool) + * res1: ErrOr[Boolean] = Invalid(Invalid int.) + * + * scala> validInt.productL(invalidBool) + * res2: ErrOr[Int] = Invalid(Invalid boolean.) + * + * scala> invalidInt.productL(invalidBool) + * res3: ErrOr[Int] = Invalid(Invalid int.Invalid boolean.) + * }}} + */ def productL(fb: F[B])(implicit F: Apply[F]): F[A] = F.productL(fa)(fb) + /** + * Alias for [[productR]]. + */ def *>(fb: F[B])(implicit F: Apply[F]): F[B] = F.*>(fa)(fb) + /** + * Alias for [[productL]]. + */ def <*(fb: F[B])(implicit F: Apply[F]): F[A] = F.<*(fa)(fb) } final class Map2Ops[F[_], A, B, C](private val fa: F[A]) extends AnyVal { + + /** + * @see [[Apply.map2]]. + * + * Example: + * {{{ + * scala> import cats.syntax.all._ + * + * scala> val someInt: Option[Int] = Some(3) + * scala> val noneInt: Option[Int] = None + * scala> val someLong: Option[Long] = Some(4L) + * scala> val noneLong: Option[Long] = None + * + * scala> someInt.map2(someLong)((i, l) => i.toString + l.toString) + * res0: Option[String] = Some(34) + * + * scala> someInt.map2(noneLong)((i, l) => i.toString + l.toString) + * res0: Option[String] = None + * + * scala> noneInt.map2(noneLong)((i, l) => i.toString + l.toString) + * res0: Option[String] = None + * + * scala> noneInt.map2(someLong)((i, l) => i.toString + l.toString) + * res0: Option[String] = None + * }}} + */ def map2(fb: F[B])(f: (A, B) => C)(implicit F: Apply[F]): F[C] = F.map2(fa, fb)(f) + /** + * @see [[Apply.map2Eval]]. + * + * Example: + * {{{ + * scala> import cats.{Eval, Later} + * scala> import cats.syntax.all._ + * + * scala> val bomb: Eval[Option[Int]] = Later(sys.error("boom")) + * scala> val x: Option[Int] = None + * + * scala> x.map2Eval(bomb)(_ + _).value + * res0: Option[Int] = None + * }}} + */ def map2Eval(fb: Eval[F[B]])(f: (A, B) => C)(implicit F: Apply[F]): Eval[F[C]] = F.map2Eval(fa, fb)(f) } From 2f5dc8f092c6613649ec0022e379e022499cbc35 Mon Sep 17 00:00:00 2001 From: mox692 Date: Sun, 23 Jul 2023 20:49:48 +0900 Subject: [PATCH 08/28] fix naming --- core/src/main/scala/cats/syntax/apply.scala | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index f34a858b56..dbfd0921a1 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -42,16 +42,16 @@ private[syntax] trait ApplySyntaxBinCompat0 { } private[syntax] trait ApplySyntaxBinCompat1 { - implicit final def applyFABOps[F[_], A, B](fab: F[A => B]): ApplyFABOps[F, A, B] = + implicit final def catsSyntaxApplyFABOps[F[_], A, B](fab: F[A => B]): ApplyFABOps[F, A, B] = new ApplyFABOps[F, A, B](fab) - implicit final def apply2Ops[F[_], A, B, C](ff: F[(A, B) => C]): Apply2Ops[F, A, B, C] = + implicit final def catsSyntaxApply2Ops[F[_], A, B, C](ff: F[(A, B) => C]): Apply2Ops[F, A, B, C] = new Apply2Ops[F, A, B, C](ff) - implicit final def productOps[F[_], A, B](fa: F[A]): ProductOps[F, A, B] = + implicit final def catsSyntaxProductOps[F[_], A, B](fa: F[A]): ProductOps[F, A, B] = new ProductOps[F, A, B](fa) - implicit final def map2Ops[F[_], A, B, C](fa: F[A]): Map2Ops[F, A, B, C] = + implicit final def catsSyntaxMap2Ops[F[_], A, B, C](fa: F[A]): Map2Ops[F, A, B, C] = new Map2Ops[F, A, B, C](fa) } From 0ee124ee806b3739ea22c521fb8a260f6eaf16cf Mon Sep 17 00:00:00 2001 From: mox692 Date: Mon, 24 Jul 2023 20:13:54 +0900 Subject: [PATCH 09/28] Fix: implicit method type parameter --- core/src/main/scala/cats/syntax/apply.scala | 24 ++++++++++----------- 1 file changed, 12 insertions(+), 12 deletions(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index dbfd0921a1..ff8e7f366f 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -48,11 +48,11 @@ private[syntax] trait ApplySyntaxBinCompat1 { implicit final def catsSyntaxApply2Ops[F[_], A, B, C](ff: F[(A, B) => C]): Apply2Ops[F, A, B, C] = new Apply2Ops[F, A, B, C](ff) - implicit final def catsSyntaxProductOps[F[_], A, B](fa: F[A]): ProductOps[F, A, B] = - new ProductOps[F, A, B](fa) + implicit final def catsSyntaxProductOps[F[_], A](fa: F[A]): ProductOps[F, A] = + new ProductOps[F, A](fa) - implicit final def catsSyntaxMap2Ops[F[_], A, B, C](fa: F[A]): Map2Ops[F, A, B, C] = - new Map2Ops[F, A, B, C](fa) + implicit final def catsSyntaxMap2Ops[F[_], A, B, C](fa: F[A]): Map2Ops[F, A] = + new Map2Ops[F, A](fa) } final class ApplyFABOps[F[_], A, B](private val fab: F[A => B]) extends AnyVal { @@ -122,7 +122,7 @@ final class Apply2Ops[F[_], A, B, C](private val ff: F[(A, B) => C]) extends Any def ap2(fa: F[A], fb: F[B])(implicit F: Apply[F]): F[C] = F.ap2(ff)(fa, fb) } -final class ProductOps[F[_], A, B](private val fa: F[A]) extends AnyVal { +final class ProductOps[F[_], A](private val fa: F[A]) extends AnyVal { /** * @see [[Apply.productR]]. @@ -153,7 +153,7 @@ final class ProductOps[F[_], A, B](private val fa: F[A]) extends AnyVal { * res3: ErrOr[Boolean] = Invalid(Invalid int.Invalid boolean.) * }}} */ - def productR(fb: F[B])(implicit F: Apply[F]): F[B] = F.productR(fa)(fb) + def productR[B](fb: F[B])(implicit F: Apply[F]): F[B] = F.productR(fa)(fb) /** * @see [[Apply.productL]]. @@ -184,20 +184,20 @@ final class ProductOps[F[_], A, B](private val fa: F[A]) extends AnyVal { * res3: ErrOr[Int] = Invalid(Invalid int.Invalid boolean.) * }}} */ - def productL(fb: F[B])(implicit F: Apply[F]): F[A] = F.productL(fa)(fb) + def productL[B](fb: F[B])(implicit F: Apply[F]): F[A] = F.productL(fa)(fb) /** * Alias for [[productR]]. */ - def *>(fb: F[B])(implicit F: Apply[F]): F[B] = F.*>(fa)(fb) + def *>[B](fb: F[B])(implicit F: Apply[F]): F[B] = F.*>(fa)(fb) /** * Alias for [[productL]]. */ - def <*(fb: F[B])(implicit F: Apply[F]): F[A] = F.<*(fa)(fb) + def <*[B](fb: F[B])(implicit F: Apply[F]): F[A] = F.<*(fa)(fb) } -final class Map2Ops[F[_], A, B, C](private val fa: F[A]) extends AnyVal { +final class Map2Ops[F[_], A](private val fa: F[A]) extends AnyVal { /** * @see [[Apply.map2]]. @@ -224,7 +224,7 @@ final class Map2Ops[F[_], A, B, C](private val fa: F[A]) extends AnyVal { * res0: Option[String] = None * }}} */ - def map2(fb: F[B])(f: (A, B) => C)(implicit F: Apply[F]): F[C] = + def map2[B, C](fb: F[B])(f: (A, B) => C)(implicit F: Apply[F]): F[C] = F.map2(fa, fb)(f) /** @@ -242,7 +242,7 @@ final class Map2Ops[F[_], A, B, C](private val fa: F[A]) extends AnyVal { * res0: Option[Int] = None * }}} */ - def map2Eval(fb: Eval[F[B]])(f: (A, B) => C)(implicit F: Apply[F]): Eval[F[C]] = + def map2Eval[B, C](fb: Eval[F[B]])(f: (A, B) => C)(implicit F: Apply[F]): Eval[F[C]] = F.map2Eval(fa, fb)(f) } From f97affd283ed04481b9bb4a5c97559f72099882e Mon Sep 17 00:00:00 2001 From: mox692 Date: Mon, 24 Jul 2023 20:23:40 +0900 Subject: [PATCH 10/28] Fix: group together F[A] syntax --- core/src/main/scala/cats/syntax/apply.scala | 12 +++--------- 1 file changed, 3 insertions(+), 9 deletions(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index ff8e7f366f..05290a1072 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -48,11 +48,8 @@ private[syntax] trait ApplySyntaxBinCompat1 { implicit final def catsSyntaxApply2Ops[F[_], A, B, C](ff: F[(A, B) => C]): Apply2Ops[F, A, B, C] = new Apply2Ops[F, A, B, C](ff) - implicit final def catsSyntaxProductOps[F[_], A](fa: F[A]): ProductOps[F, A] = - new ProductOps[F, A](fa) - - implicit final def catsSyntaxMap2Ops[F[_], A, B, C](fa: F[A]): Map2Ops[F, A] = - new Map2Ops[F, A](fa) + implicit final def catsSyntaxApplyOps2[F[_], A](fa: F[A]): ApplyOps2[F, A] = + new ApplyOps2[F, A](fa) } final class ApplyFABOps[F[_], A, B](private val fab: F[A => B]) extends AnyVal { @@ -122,7 +119,7 @@ final class Apply2Ops[F[_], A, B, C](private val ff: F[(A, B) => C]) extends Any def ap2(fa: F[A], fb: F[B])(implicit F: Apply[F]): F[C] = F.ap2(ff)(fa, fb) } -final class ProductOps[F[_], A](private val fa: F[A]) extends AnyVal { +final class ApplyOps2[F[_], A](private val fa: F[A]) extends AnyVal { /** * @see [[Apply.productR]]. @@ -195,9 +192,6 @@ final class ProductOps[F[_], A](private val fa: F[A]) extends AnyVal { * Alias for [[productL]]. */ def <*[B](fb: F[B])(implicit F: Apply[F]): F[A] = F.<*(fa)(fb) -} - -final class Map2Ops[F[_], A](private val fa: F[A]) extends AnyVal { /** * @see [[Apply.map2]]. From e1ca16498d3bfc88ee333c4f0383b79f119ba51c Mon Sep 17 00:00:00 2001 From: mox692 Date: Mon, 24 Jul 2023 21:51:45 +0900 Subject: [PATCH 11/28] Add: new Semigroupal syntax with no unnecessary allocations --- .../main/scala/cats/syntax/semigroupal.scala | 41 ++++++++++++++++++- 1 file changed, 40 insertions(+), 1 deletion(-) diff --git a/core/src/main/scala/cats/syntax/semigroupal.scala b/core/src/main/scala/cats/syntax/semigroupal.scala index bd53dbd7e8..413689e61d 100644 --- a/core/src/main/scala/cats/syntax/semigroupal.scala +++ b/core/src/main/scala/cats/syntax/semigroupal.scala @@ -23,13 +23,52 @@ package cats package syntax trait SemigroupalSyntax { - implicit final def catsSyntaxSemigroupal[F[_], A](fa: F[A])(implicit F: Semigroupal[F]): SemigroupalOps[F, A] = + @deprecated("Use `catsSntaxSemigroupalOps2`", "2.10.0") + final def catsSyntaxSemigroupal[F[_], A](fa: F[A])(implicit F: Semigroupal[F]): SemigroupalOps[F, A] = new SemigroupalOps[F, A] { type TypeClassType = Semigroupal[F] val self = fa val typeClassInstance = F } + + implicit def catsSntaxSemigroupalOps2[F[_], A](fa: F[A]): SemigroupalOps2[F, A] = + new SemigroupalOps2[F, A](fa) + +} + +final class SemigroupalOps2[F[_], A](private val fa: F[A]) extends AnyVal { + + /** + * @see [[Semigroupal.product]] + * + * Example: + * {{{ + * scala> import cats.syntax.all._ + * + * scala> val noneInt: Option[Int] = None + * scala> val some3: Option[Int] = Some(3) + * scala> val noneString: Option[String] = None + * scala> val someFoo: Option[String] = Some("foo") + * + * scala> noneInt.product(noneString) + * res0: Option[(Int, String)] = None + * + * scala> noneInt.product(someFoo) + * res1: Option[(Int, String)] = None + * + * scala> some3.product(noneString) + * res2: Option[(Int, String)] = None + * + * scala> some3.product(someFoo) + * res3: Option[(Int, String)] = Some((3,foo)) + * }}} + */ + def product[B](fb: F[B])(implicit F: Semigroupal[F]): F[(A, B)] = F.product(fa, fb) + + @deprecated("Replaced by an apply syntax, e.g. instead of (a |@| b).map(...) use (a, b).mapN(...)", "1.0.0-MF") + def |@|[B](fb: F[B]): SemigroupalBuilder[F]#SemigroupalBuilder2[A, B] = + new SemigroupalBuilder[F] |@| fa |@| fb } abstract class SemigroupalOps[F[_], A] extends Semigroupal.Ops[F, A] { From 5603fee20352bda2c97b8bed13b5ee34e0705737 Mon Sep 17 00:00:00 2001 From: mox692 Date: Mon, 24 Jul 2023 23:24:11 +0900 Subject: [PATCH 12/28] Fix: failing doctest --- core/src/main/scala/cats/syntax/apply.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 05290a1072..2016b10e80 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -172,7 +172,7 @@ final class ApplyOps2[F[_], A](private val fa: F[A]) extends AnyVal { * res0: ErrOr[Int] = Valid(3) * * scala> invalidInt.productL(validBool) - * res1: ErrOr[Boolean] = Invalid(Invalid int.) + * res1: ErrOr[Int] = Invalid(Invalid int.) * * scala> validInt.productL(invalidBool) * res2: ErrOr[Int] = Invalid(Invalid boolean.) From fc6ac27a85b8815841b02f48b553c58643acb6db Mon Sep 17 00:00:00 2001 From: mox692 Date: Tue, 25 Jul 2023 20:32:35 +0900 Subject: [PATCH 13/28] Fix: failing doc test --- core/src/main/scala/cats/syntax/apply.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 2016b10e80..499cef6e84 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -233,7 +233,7 @@ final class ApplyOps2[F[_], A](private val fa: F[A]) extends AnyVal { * scala> val x: Option[Int] = None * * scala> x.map2Eval(bomb)(_ + _).value - * res0: Option[Int] = None + * res0: Option[Int] = None * }}} */ def map2Eval[B, C](fb: Eval[F[B]])(f: (A, B) => C)(implicit F: Apply[F]): Eval[F[C]] = From 342996fbed56fb96b47b5e65a7af1297bba90144 Mon Sep 17 00:00:00 2001 From: kim / Motoyuki Kimura <55653825+mox692@users.noreply.github.com> Date: Wed, 26 Jul 2023 20:28:06 +0900 Subject: [PATCH 14/28] remove implicit from catsSyntaxSemigroupal Co-authored-by: Sergey Torgashov --- core/src/main/scala/cats/syntax/semigroupal.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/src/main/scala/cats/syntax/semigroupal.scala b/core/src/main/scala/cats/syntax/semigroupal.scala index 413689e61d..11bef26f9c 100644 --- a/core/src/main/scala/cats/syntax/semigroupal.scala +++ b/core/src/main/scala/cats/syntax/semigroupal.scala @@ -24,7 +24,7 @@ package syntax trait SemigroupalSyntax { @deprecated("Use `catsSntaxSemigroupalOps2`", "2.10.0") - final def catsSyntaxSemigroupal[F[_], A](fa: F[A])(implicit F: Semigroupal[F]): SemigroupalOps[F, A] = + final def catsSyntaxSemigroupal[F[_], A](fa: F[A])(F: Semigroupal[F]): SemigroupalOps[F, A] = new SemigroupalOps[F, A] { type TypeClassType = Semigroupal[F] From 649c38c1420a7698082148277c835a8e40850612 Mon Sep 17 00:00:00 2001 From: mox692 Date: Wed, 26 Jul 2023 21:28:20 +0900 Subject: [PATCH 15/28] Workaround to the failing test --- .../src/test/scala/cats/tests/SyntaxSerializationSuite.scala | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/tests/shared/src/test/scala/cats/tests/SyntaxSerializationSuite.scala b/tests/shared/src/test/scala/cats/tests/SyntaxSerializationSuite.scala index fdf9c1afe6..4ca0f940dc 100644 --- a/tests/shared/src/test/scala/cats/tests/SyntaxSerializationSuite.scala +++ b/tests/shared/src/test/scala/cats/tests/SyntaxSerializationSuite.scala @@ -23,6 +23,7 @@ package cats.tests import cats.laws.discipline.SerializableTests import cats.syntax.either._ +import cats.Semigroupal /** * Test that our syntax implicits are serializable. @@ -36,7 +37,7 @@ class SyntaxSerializationSuite extends CatsSuite { ) checkAll("SemigroupalOps[Option, Int]", - SerializableTests.serializable(cats.syntax.all.catsSyntaxSemigroupal[Option, Int](None)) + SerializableTests.serializable(cats.syntax.all.catsSyntaxSemigroupal[Option, Int](None)(Semigroupal[Option])) ) checkAll( From 9ebe3a79dc339168445383db2437ae561a25a779 Mon Sep 17 00:00:00 2001 From: mox692 Date: Wed, 26 Jul 2023 21:33:41 +0900 Subject: [PATCH 16/28] Deprecate some synax causing allocation --- core/src/main/scala/cats/syntax/contravariantMonoidal.scala | 5 +++-- .../main/scala/cats/syntax/contravariantSemigroupal.scala | 5 +++-- 2 files changed, 6 insertions(+), 4 deletions(-) diff --git a/core/src/main/scala/cats/syntax/contravariantMonoidal.scala b/core/src/main/scala/cats/syntax/contravariantMonoidal.scala index 0253d1c18a..1c4a317092 100644 --- a/core/src/main/scala/cats/syntax/contravariantMonoidal.scala +++ b/core/src/main/scala/cats/syntax/contravariantMonoidal.scala @@ -25,9 +25,10 @@ package syntax import cats.ContravariantMonoidal trait ContravariantMonoidalSyntax { - implicit final def catsSyntaxContravariantMonoidal[F[_], A]( + @deprecated("Kept for binary compatibility", "2.10.0") + final def catsSyntaxContravariantMonoidal[F[_], A]( fa: F[A] - )(implicit F: ContravariantMonoidal[F]): ContravariantMonoidalOps[F, A] = + )(F: ContravariantMonoidal[F]): ContravariantMonoidalOps[F, A] = new ContravariantMonoidalOps[F, A] { type TypeClassType = ContravariantMonoidal[F] diff --git a/core/src/main/scala/cats/syntax/contravariantSemigroupal.scala b/core/src/main/scala/cats/syntax/contravariantSemigroupal.scala index 688a279caa..cd50153163 100644 --- a/core/src/main/scala/cats/syntax/contravariantSemigroupal.scala +++ b/core/src/main/scala/cats/syntax/contravariantSemigroupal.scala @@ -25,9 +25,10 @@ package syntax import cats.ContravariantSemigroupal trait ContravariantSemigroupalSyntax extends TupleSemigroupalSyntax { - implicit final def catsSyntaxContravariantSemigroupal[F[_], A]( + @deprecated("Kept for binary compatibility", "2.10.0") + final def catsSyntaxContravariantSemigroupal[F[_], A]( fa: F[A] - )(implicit F: ContravariantSemigroupal[F]): ContravariantSemigroupal.Ops[F, A] = + )(F: ContravariantSemigroupal[F]): ContravariantSemigroupal.Ops[F, A] = new ContravariantSemigroupal.Ops[F, A] { type TypeClassType = ContravariantSemigroupal[F] From 1962dc3fa3cbc8a83bd731fadf43b05d5f0cb1a1 Mon Sep 17 00:00:00 2001 From: mox692 Date: Fri, 28 Jul 2023 20:49:48 +0900 Subject: [PATCH 17/28] Fix: typo --- core/src/main/scala/cats/syntax/semigroupal.scala | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/core/src/main/scala/cats/syntax/semigroupal.scala b/core/src/main/scala/cats/syntax/semigroupal.scala index 11bef26f9c..b051ebde7a 100644 --- a/core/src/main/scala/cats/syntax/semigroupal.scala +++ b/core/src/main/scala/cats/syntax/semigroupal.scala @@ -23,7 +23,7 @@ package cats package syntax trait SemigroupalSyntax { - @deprecated("Use `catsSntaxSemigroupalOps2`", "2.10.0") + @deprecated("Use `catsSyntaxSemigroupalOps2`", "2.10.0") final def catsSyntaxSemigroupal[F[_], A](fa: F[A])(F: Semigroupal[F]): SemigroupalOps[F, A] = new SemigroupalOps[F, A] { type TypeClassType = Semigroupal[F] @@ -32,7 +32,7 @@ trait SemigroupalSyntax { val typeClassInstance = F } - implicit def catsSntaxSemigroupalOps2[F[_], A](fa: F[A]): SemigroupalOps2[F, A] = + implicit def catsSytaxSemigroupalOps2[F[_], A](fa: F[A]): SemigroupalOps2[F, A] = new SemigroupalOps2[F, A](fa) } From 1d5102f410985186a3e546a56cabade96aa3dde3 Mon Sep 17 00:00:00 2001 From: mox692 Date: Fri, 28 Jul 2023 20:57:14 +0900 Subject: [PATCH 18/28] Fix: delete ApplySyntaxBinCompat1 --- core/src/main/scala/cats/syntax/all.scala | 1 - core/src/main/scala/cats/syntax/apply.scala | 21 ++++++++----------- core/src/main/scala/cats/syntax/package.scala | 2 +- 3 files changed, 10 insertions(+), 14 deletions(-) diff --git a/core/src/main/scala/cats/syntax/all.scala b/core/src/main/scala/cats/syntax/all.scala index 8f7b3453db..60d6d2b150 100644 --- a/core/src/main/scala/cats/syntax/all.scala +++ b/core/src/main/scala/cats/syntax/all.scala @@ -111,7 +111,6 @@ trait AllSyntaxBinCompat3 extends UnorderedFoldableSyntax with Function1Syntax trait AllSyntaxBinCompat4 extends TraverseFilterSyntaxBinCompat0 with ApplySyntaxBinCompat0 - with ApplySyntaxBinCompat1 with ParallelApplySyntax with FoldableSyntaxBinCompat0 with ReducibleSyntaxBinCompat0 diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 499cef6e84..01c1deaf44 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -23,7 +23,7 @@ package cats package syntax trait ApplySyntax extends TupleSemigroupalSyntax { - @deprecated("Use `ApplySyntaxBinCompat1`", "2.10.0") + @deprecated("Kept for binary compatibility", "2.10.0") final def catsSyntaxApply[F[_], A](fa: F[A], F: Apply[F]): Apply.Ops[F, A] = new Apply.Ops[F, A] { type TypeClassType = Apply[F] @@ -34,25 +34,22 @@ trait ApplySyntax extends TupleSemigroupalSyntax { implicit final def catsSyntaxApplyOps[F[_], A](fa: F[A]): ApplyOps[F, A] = new ApplyOps(fa) -} - -private[syntax] trait ApplySyntaxBinCompat0 { - implicit final def catsSyntaxIfApplyOps[F[_]](fa: F[Boolean]): IfApplyOps[F] = - new IfApplyOps[F](fa) -} - -private[syntax] trait ApplySyntaxBinCompat1 { - implicit final def catsSyntaxApplyFABOps[F[_], A, B](fab: F[A => B]): ApplyFABOps[F, A, B] = - new ApplyFABOps[F, A, B](fab) implicit final def catsSyntaxApply2Ops[F[_], A, B, C](ff: F[(A, B) => C]): Apply2Ops[F, A, B, C] = new Apply2Ops[F, A, B, C](ff) + implicit final def catsSyntaxApplyFABOps[F[_], A, B](fab: F[A => B]): ApplyFABOps[F, A, B] = + new ApplyFABOps[F, A, B](fab) implicit final def catsSyntaxApplyOps2[F[_], A](fa: F[A]): ApplyOps2[F, A] = new ApplyOps2[F, A](fa) } -final class ApplyFABOps[F[_], A, B](private val fab: F[A => B]) extends AnyVal { +private[syntax] trait ApplySyntaxBinCompat0 { + implicit final def catsSyntaxIfApplyOps[F[_]](fa: F[Boolean]): IfApplyOps[F] = + new IfApplyOps[F](fa) +} + +final class ApplyFABOps[F[_], A, B](private val fab: F[A => B]) extends AnyVal with Serializable { /** * @see [[Apply.ap]]. diff --git a/core/src/main/scala/cats/syntax/package.scala b/core/src/main/scala/cats/syntax/package.scala index 5d02b8f134..46b743455f 100644 --- a/core/src/main/scala/cats/syntax/package.scala +++ b/core/src/main/scala/cats/syntax/package.scala @@ -27,7 +27,7 @@ package object syntax { object alternative extends AlternativeSyntax object applicative extends ApplicativeSyntax object applicativeError extends ApplicativeErrorSyntax - object apply extends ApplySyntax with ApplySyntaxBinCompat0 with ApplySyntaxBinCompat1 + object apply extends ApplySyntax with ApplySyntaxBinCompat0 object arrow extends ArrowSyntax object arrowChoice extends ArrowChoiceSyntax object bifunctor extends BifunctorSyntax From c18c981f241efd3f9d86bc87af71e363785f197c Mon Sep 17 00:00:00 2001 From: mox692 Date: Fri, 28 Jul 2023 21:00:49 +0900 Subject: [PATCH 19/28] Move ApplyOps2 --- laws/src/main/scala/cats/laws/InvariantSemigroupalLaws.scala | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/laws/src/main/scala/cats/laws/InvariantSemigroupalLaws.scala b/laws/src/main/scala/cats/laws/InvariantSemigroupalLaws.scala index 216c2a06c5..99139895e4 100644 --- a/laws/src/main/scala/cats/laws/InvariantSemigroupalLaws.scala +++ b/laws/src/main/scala/cats/laws/InvariantSemigroupalLaws.scala @@ -35,6 +35,11 @@ trait InvariantSemigroupalLaws[F[_]] extends InvariantLaws[F] with SemigroupalLa .product(fb) .product(fc) .imap { case ((a, b), c) => (a, (b, c)) } { case (a, (b, c)) => ((a, b), c) } + + def aa[A, B, C](fa: F[A], fb: F[B], fc: F[C]) = { + fa.product(fb.product(fc)) + fa |@| fb + } } object InvariantSemigroupalLaws { From e74b3016ac1b350f5daf0e2e6abb2ca19bf33c46 Mon Sep 17 00:00:00 2001 From: mox692 Date: Fri, 28 Jul 2023 21:02:19 +0900 Subject: [PATCH 20/28] Rename Apply2Ops to ApplyFABCOps --- core/src/main/scala/cats/syntax/apply.scala | 53 +++++++++++---------- 1 file changed, 27 insertions(+), 26 deletions(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 01c1deaf44..ea7c286132 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -35,8 +35,9 @@ trait ApplySyntax extends TupleSemigroupalSyntax { implicit final def catsSyntaxApplyOps[F[_], A](fa: F[A]): ApplyOps[F, A] = new ApplyOps(fa) - implicit final def catsSyntaxApply2Ops[F[_], A, B, C](ff: F[(A, B) => C]): Apply2Ops[F, A, B, C] = - new Apply2Ops[F, A, B, C](ff) + implicit final def catsSyntaxApply2Ops[F[_], A, B, C](ff: F[(A, B) => C]): ApplyFABCOps[F, A, B, C] = + new ApplyFABCOps[F, A, B, C](ff) + implicit final def catsSyntaxApplyFABOps[F[_], A, B](fab: F[A => B]): ApplyFABOps[F, A, B] = new ApplyFABOps[F, A, B](fab) @@ -84,7 +85,7 @@ final class ApplyFABOps[F[_], A, B](private val fab: F[A => B]) extends AnyVal w def <*>(fa: F[A])(implicit F: Apply[F]): F[B] = F.<*>[A, B](fab)(fa) } -final class Apply2Ops[F[_], A, B, C](private val ff: F[(A, B) => C]) extends AnyVal { +final class ApplyFABCOps[F[_], A, B, C](private val ff: F[(A, B) => C]) extends AnyVal { /** * @see [[Apply.ap2]]. @@ -116,6 +117,29 @@ final class Apply2Ops[F[_], A, B, C](private val ff: F[(A, B) => C]) extends Any def ap2(fa: F[A], fb: F[B])(implicit F: Apply[F]): F[C] = F.ap2(ff)(fa, fb) } +final class IfApplyOps[F[_]](private val fcond: F[Boolean]) extends AnyVal { + + @deprecated("Dangerous method, use ifM (a flatMap) or ifF (a map) instead", "2.6.2") + def ifA[A](ifTrue: F[A], ifFalse: F[A])(implicit F: Apply[F]): F[A] = F.ifA(fcond)(ifTrue, ifFalse) +} + +final class ApplyOps[F[_], A](private val fa: F[A]) extends AnyVal { + + /** + * Alias for [[Apply.productR]]. + */ + @deprecated("Use *> or productR instead.", "1.0.0-RC2") + @inline private[syntax] def followedBy[B](fb: F[B])(implicit F: Apply[F]): F[B] = + F.productR(fa)(fb) + + /** + * Alias for [[Apply.productL]]. + */ + @deprecated("Use <* or productL instead.", "1.0.0-RC2") + @inline private[syntax] def forEffect[B](fb: F[B])(implicit F: Apply[F]): F[A] = + F.productL(fa)(fb) +} + final class ApplyOps2[F[_], A](private val fa: F[A]) extends AnyVal { /** @@ -236,26 +260,3 @@ final class ApplyOps2[F[_], A](private val fa: F[A]) extends AnyVal { def map2Eval[B, C](fb: Eval[F[B]])(f: (A, B) => C)(implicit F: Apply[F]): Eval[F[C]] = F.map2Eval(fa, fb)(f) } - -final class IfApplyOps[F[_]](private val fcond: F[Boolean]) extends AnyVal { - - @deprecated("Dangerous method, use ifM (a flatMap) or ifF (a map) instead", "2.6.2") - def ifA[A](ifTrue: F[A], ifFalse: F[A])(implicit F: Apply[F]): F[A] = F.ifA(fcond)(ifTrue, ifFalse) -} - -final class ApplyOps[F[_], A](private val fa: F[A]) extends AnyVal { - - /** - * Alias for [[Apply.productR]]. - */ - @deprecated("Use *> or productR instead.", "1.0.0-RC2") - @inline private[syntax] def followedBy[B](fb: F[B])(implicit F: Apply[F]): F[B] = - F.productR(fa)(fb) - - /** - * Alias for [[Apply.productL]]. - */ - @deprecated("Use <* or productL instead.", "1.0.0-RC2") - @inline private[syntax] def forEffect[B](fb: F[B])(implicit F: Apply[F]): F[A] = - F.productL(fa)(fb) -} From 0fbc093626a94ae98dbbe375739a5d0d793074aa Mon Sep 17 00:00:00 2001 From: mox692 Date: Fri, 28 Jul 2023 21:45:15 +0900 Subject: [PATCH 21/28] Uncurry implicits --- core/src/main/scala/cats/syntax/contravariantMonoidal.scala | 5 +++-- .../main/scala/cats/syntax/contravariantSemigroupal.scala | 5 +++-- core/src/main/scala/cats/syntax/semigroupal.scala | 2 +- .../src/test/scala/cats/tests/SyntaxSerializationSuite.scala | 2 +- 4 files changed, 8 insertions(+), 6 deletions(-) diff --git a/core/src/main/scala/cats/syntax/contravariantMonoidal.scala b/core/src/main/scala/cats/syntax/contravariantMonoidal.scala index 1c4a317092..ab5b76c754 100644 --- a/core/src/main/scala/cats/syntax/contravariantMonoidal.scala +++ b/core/src/main/scala/cats/syntax/contravariantMonoidal.scala @@ -27,8 +27,9 @@ import cats.ContravariantMonoidal trait ContravariantMonoidalSyntax { @deprecated("Kept for binary compatibility", "2.10.0") final def catsSyntaxContravariantMonoidal[F[_], A]( - fa: F[A] - )(F: ContravariantMonoidal[F]): ContravariantMonoidalOps[F, A] = + fa: F[A], + F: ContravariantMonoidal[F] + ): ContravariantMonoidalOps[F, A] = new ContravariantMonoidalOps[F, A] { type TypeClassType = ContravariantMonoidal[F] diff --git a/core/src/main/scala/cats/syntax/contravariantSemigroupal.scala b/core/src/main/scala/cats/syntax/contravariantSemigroupal.scala index cd50153163..29d92a4d26 100644 --- a/core/src/main/scala/cats/syntax/contravariantSemigroupal.scala +++ b/core/src/main/scala/cats/syntax/contravariantSemigroupal.scala @@ -27,8 +27,9 @@ import cats.ContravariantSemigroupal trait ContravariantSemigroupalSyntax extends TupleSemigroupalSyntax { @deprecated("Kept for binary compatibility", "2.10.0") final def catsSyntaxContravariantSemigroupal[F[_], A]( - fa: F[A] - )(F: ContravariantSemigroupal[F]): ContravariantSemigroupal.Ops[F, A] = + fa: F[A], + F: ContravariantSemigroupal[F] + ): ContravariantSemigroupal.Ops[F, A] = new ContravariantSemigroupal.Ops[F, A] { type TypeClassType = ContravariantSemigroupal[F] diff --git a/core/src/main/scala/cats/syntax/semigroupal.scala b/core/src/main/scala/cats/syntax/semigroupal.scala index b051ebde7a..4a44ec7bcb 100644 --- a/core/src/main/scala/cats/syntax/semigroupal.scala +++ b/core/src/main/scala/cats/syntax/semigroupal.scala @@ -24,7 +24,7 @@ package syntax trait SemigroupalSyntax { @deprecated("Use `catsSyntaxSemigroupalOps2`", "2.10.0") - final def catsSyntaxSemigroupal[F[_], A](fa: F[A])(F: Semigroupal[F]): SemigroupalOps[F, A] = + final def catsSyntaxSemigroupal[F[_], A](fa: F[A], F: Semigroupal[F]): SemigroupalOps[F, A] = new SemigroupalOps[F, A] { type TypeClassType = Semigroupal[F] diff --git a/tests/shared/src/test/scala/cats/tests/SyntaxSerializationSuite.scala b/tests/shared/src/test/scala/cats/tests/SyntaxSerializationSuite.scala index 4ca0f940dc..5ea5b3e4d7 100644 --- a/tests/shared/src/test/scala/cats/tests/SyntaxSerializationSuite.scala +++ b/tests/shared/src/test/scala/cats/tests/SyntaxSerializationSuite.scala @@ -37,7 +37,7 @@ class SyntaxSerializationSuite extends CatsSuite { ) checkAll("SemigroupalOps[Option, Int]", - SerializableTests.serializable(cats.syntax.all.catsSyntaxSemigroupal[Option, Int](None)(Semigroupal[Option])) + SerializableTests.serializable(cats.syntax.all.catsSyntaxSemigroupal[Option, Int](None, Semigroupal[Option])) ) checkAll( From 3f1e8e57067501aad94133eaecd5c5baaf0cefce Mon Sep 17 00:00:00 2001 From: mox692 Date: Fri, 28 Jul 2023 22:14:46 +0900 Subject: [PATCH 22/28] Fix: typo --- core/src/main/scala/cats/syntax/semigroupal.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/src/main/scala/cats/syntax/semigroupal.scala b/core/src/main/scala/cats/syntax/semigroupal.scala index 4a44ec7bcb..e80bd78fd1 100644 --- a/core/src/main/scala/cats/syntax/semigroupal.scala +++ b/core/src/main/scala/cats/syntax/semigroupal.scala @@ -32,7 +32,7 @@ trait SemigroupalSyntax { val typeClassInstance = F } - implicit def catsSytaxSemigroupalOps2[F[_], A](fa: F[A]): SemigroupalOps2[F, A] = + implicit def catsSyntaxSemigroupalOps2[F[_], A](fa: F[A]): SemigroupalOps2[F, A] = new SemigroupalOps2[F, A](fa) } From f09b400fc3bda998497503a1543e7842460a2391 Mon Sep 17 00:00:00 2001 From: mox692 Date: Fri, 28 Jul 2023 22:19:50 +0900 Subject: [PATCH 23/28] Tidying --- core/src/main/scala/cats/syntax/apply.scala | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index ea7c286132..622755a03f 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -32,14 +32,14 @@ trait ApplySyntax extends TupleSemigroupalSyntax { val typeClassInstance = F } - implicit final def catsSyntaxApplyOps[F[_], A](fa: F[A]): ApplyOps[F, A] = - new ApplyOps(fa) + implicit final def catsSyntaxApplyFABOps[F[_], A, B](fab: F[A => B]): ApplyFABOps[F, A, B] = + new ApplyFABOps[F, A, B](fab) - implicit final def catsSyntaxApply2Ops[F[_], A, B, C](ff: F[(A, B) => C]): ApplyFABCOps[F, A, B, C] = + implicit final def catsSyntaxApplyFABCOps[F[_], A, B, C](ff: F[(A, B) => C]): ApplyFABCOps[F, A, B, C] = new ApplyFABCOps[F, A, B, C](ff) - implicit final def catsSyntaxApplyFABOps[F[_], A, B](fab: F[A => B]): ApplyFABOps[F, A, B] = - new ApplyFABOps[F, A, B](fab) + implicit final def catsSyntaxApplyOps[F[_], A](fa: F[A]): ApplyOps[F, A] = + new ApplyOps(fa) implicit final def catsSyntaxApplyOps2[F[_], A](fa: F[A]): ApplyOps2[F, A] = new ApplyOps2[F, A](fa) From 8ae9b9c350ae551f9dfe15cfd4a0615013236b91 Mon Sep 17 00:00:00 2001 From: mox692 Date: Fri, 28 Jul 2023 22:24:30 +0900 Subject: [PATCH 24/28] Remove unrelated diff --- laws/src/main/scala/cats/laws/InvariantSemigroupalLaws.scala | 5 ----- 1 file changed, 5 deletions(-) diff --git a/laws/src/main/scala/cats/laws/InvariantSemigroupalLaws.scala b/laws/src/main/scala/cats/laws/InvariantSemigroupalLaws.scala index 99139895e4..216c2a06c5 100644 --- a/laws/src/main/scala/cats/laws/InvariantSemigroupalLaws.scala +++ b/laws/src/main/scala/cats/laws/InvariantSemigroupalLaws.scala @@ -35,11 +35,6 @@ trait InvariantSemigroupalLaws[F[_]] extends InvariantLaws[F] with SemigroupalLa .product(fb) .product(fc) .imap { case ((a, b), c) => (a, (b, c)) } { case (a, (b, c)) => ((a, b), c) } - - def aa[A, B, C](fa: F[A], fb: F[B], fc: F[C]) = { - fa.product(fb.product(fc)) - fa |@| fb - } } object InvariantSemigroupalLaws { From a0fdb0b42933405b2c174efce9c9e7a4d2398988 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Wed, 2 Aug 2023 06:14:29 +0000 Subject: [PATCH 25/28] Merge `ApplyOps` and `ApplyOps2` --- core/src/main/scala/cats/syntax/apply.scala | 5 ----- 1 file changed, 5 deletions(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 622755a03f..5747db98d3 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -41,8 +41,6 @@ trait ApplySyntax extends TupleSemigroupalSyntax { implicit final def catsSyntaxApplyOps[F[_], A](fa: F[A]): ApplyOps[F, A] = new ApplyOps(fa) - implicit final def catsSyntaxApplyOps2[F[_], A](fa: F[A]): ApplyOps2[F, A] = - new ApplyOps2[F, A](fa) } private[syntax] trait ApplySyntaxBinCompat0 { @@ -138,9 +136,6 @@ final class ApplyOps[F[_], A](private val fa: F[A]) extends AnyVal { @deprecated("Use <* or productL instead.", "1.0.0-RC2") @inline private[syntax] def forEffect[B](fb: F[B])(implicit F: Apply[F]): F[A] = F.productL(fa)(fb) -} - -final class ApplyOps2[F[_], A](private val fa: F[A]) extends AnyVal { /** * @see [[Apply.productR]]. From 119f350b8dd1656b8ea243982032e95e7d558d33 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Thu, 3 Aug 2023 07:41:41 +0000 Subject: [PATCH 26/28] Remove spurious whitespace change --- core/src/main/scala/cats/syntax/apply.scala | 1 - 1 file changed, 1 deletion(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 5747db98d3..28e4c9f168 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -40,7 +40,6 @@ trait ApplySyntax extends TupleSemigroupalSyntax { implicit final def catsSyntaxApplyOps[F[_], A](fa: F[A]): ApplyOps[F, A] = new ApplyOps(fa) - } private[syntax] trait ApplySyntaxBinCompat0 { From 5c06b8a074b315f6ed3f13f9759f68e6f297fe51 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Thu, 3 Aug 2023 07:43:32 +0000 Subject: [PATCH 27/28] Remove `Serializable` from `ApplyFABOps` --- core/src/main/scala/cats/syntax/apply.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/src/main/scala/cats/syntax/apply.scala b/core/src/main/scala/cats/syntax/apply.scala index 28e4c9f168..df9b7d336c 100644 --- a/core/src/main/scala/cats/syntax/apply.scala +++ b/core/src/main/scala/cats/syntax/apply.scala @@ -47,7 +47,7 @@ private[syntax] trait ApplySyntaxBinCompat0 { new IfApplyOps[F](fa) } -final class ApplyFABOps[F[_], A, B](private val fab: F[A => B]) extends AnyVal with Serializable { +final class ApplyFABOps[F[_], A, B](private val fab: F[A => B]) extends AnyVal { /** * @see [[Apply.ap]]. From 288587e7f3586a8e31ca4cd06065602f5c25ae4f Mon Sep 17 00:00:00 2001 From: mox692 Date: Sat, 12 Aug 2023 18:58:23 +0900 Subject: [PATCH 28/28] Add tests related to the change --- .../test/scala/cats/tests/SyntaxSuite.scala | 23 +++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/tests/shared/src/test/scala/cats/tests/SyntaxSuite.scala b/tests/shared/src/test/scala/cats/tests/SyntaxSuite.scala index 7687cf0eb0..10fe574128 100644 --- a/tests/shared/src/test/scala/cats/tests/SyntaxSuite.scala +++ b/tests/shared/src/test/scala/cats/tests/SyntaxSuite.scala @@ -368,6 +368,8 @@ object SyntaxSuite { val f = mock[(A, B, C) => Z] val ff = mock[F[(A, B, C) => Z]] + fa.productR(fb) + fa.productL(fb) fa *> fb fb <* fc @@ -394,6 +396,19 @@ object SyntaxSuite { thabcde.imapN(f5)(g5) (ha, hb, hc, hd, he).imapN(f5)(g5) + + val tfab = mock[F[A => B]] + tfab.ap(fa) + tfab <*> fa + + val tabcf = mock[F[(A, B) => C]] + tabcf.ap2(fa, fb) + + val tabc = mock[(A, B) => C] + fa.map2(fb)(tabc) + + val tEvalfb = mock[Eval[F[B]]] + fa.map2Eval(tEvalfb)(tabc) } def testBifoldable[F[_, _]: Bifoldable, A, B, C, D: Monoid]: Unit = { @@ -688,4 +703,12 @@ object SyntaxSuite { val result: Option[List[B]] = list.traverseCollect(f) } + + def testSemigroupal[F[_]: Semigroupal, A, B]: Unit = { + val fa = mock[F[A]] + val fb = mock[F[B]] + + fa.product(fb) + fa |@| fb + } }