From faf05f9279cf91a2ba70386579f690ae875526d6 Mon Sep 17 00:00:00 2001 From: Cody Allen Date: Sat, 14 May 2016 10:54:00 -0400 Subject: [PATCH] Don't combine lefts on Xor and XorT combine Resolves #888 I don't think there's really a _right_ answer here. Both methods of combining `Xor`s are straightforward and law-abiding. However, I think that since in pretty much every other context (including the `SemigroupK` instances), `Xor` does not combine failures and `Validated` does, it is less surprising behavior to not combine left values in `combine` methods for `Xor` and `XorT`. Some notes about related implementations: - Scalaz's respective methods (and semigroup instances) _do_ combine errors, so this is a deviation from that. - The `Semigroup` for `Either` in Haskell has doesn't combine values at all, but returns the first `Right` (if any), making it equivalent to the behavior of `orElse` and the `SemigroupK` instance for `Xor`. Since we have already decided to not go with `orElse`-like behavior for our `Option` semigroup, I'm inclined to not take this approach for `Xor`. See also https://github.com/typelevel/cats/pull/996#issuecomment-214010455 --- core/src/main/scala/cats/data/Xor.scala | 46 +++++++++++++++++++----- core/src/main/scala/cats/data/XorT.scala | 40 ++++++++++++++++++++- 2 files changed, 77 insertions(+), 9 deletions(-) diff --git a/core/src/main/scala/cats/data/Xor.scala b/core/src/main/scala/cats/data/Xor.scala index 1bf5e115b6..f7f2a1d467 100644 --- a/core/src/main/scala/cats/data/Xor.scala +++ b/core/src/main/scala/cats/data/Xor.scala @@ -133,13 +133,43 @@ sealed abstract class Xor[+A, +B] extends Product with Serializable { def merge[AA >: A](implicit ev: B <:< AA): AA = fold(identity, ev.apply) - final def combine[AA >: A, BB >: B](that: AA Xor BB)(implicit AA: Semigroup[AA], BB: Semigroup[BB]): AA Xor BB = this match { - case Xor.Left(a1) => that match { - case Xor.Left(a2) => Xor.Left(AA.combine(a1, a2)) - case Xor.Right(b2) => Xor.Left(a1) - } + /** + * Combine with another `Xor` value. + * + * If this `Xor` is a `Left` then it will be returned as-is. + * If this `Xor` is a `Right` and `that` `Xor` is a left, then `that` will be + * returned. + * If both `Xor`s are `Right`s, then the `Semigroup[BB]` instance will be used + * to combine both values and return them as a `Right`. + * Note: If both `Xor`s are `Left`s then their values are not combined. Use + * `Validated` if you prefer to combine `Left` values. + * + * Examples: + * {{{ + * scala> import cats.data.Xor + * scala> import cats.implicits._ + * scala> val l1: Xor[String, Int] = Xor.left("error 1") + * scala> val l2: Xor[String, Int] = Xor.left("error 2") + * scala> val r3: Xor[String, Int] = Xor.right(3) + * scala> val r4: Xor[String, Int] = Xor.right(4) + * + * scala> l1 combine l2 + * res0: Xor[String, Int] = Left(error 1) + * + * scala> l1 combine r3 + * res1: Xor[String, Int] = Left(error 1) + * + * scala> r3 combine l1 + * res2: Xor[String, Int] = Left(error 1) + * + * scala> r3 combine r4 + * res3: Xor[String, Int] = Right(7) + * }}} + */ + final def combine[AA >: A, BB >: B](that: AA Xor BB)(implicit BB: Semigroup[BB]): AA Xor BB = this match { + case l @ Xor.Left(_) => l case Xor.Right(b1) => that match { - case Xor.Left(a2) => Xor.Left(a2) + case l @ Xor.Left(_) => l case Xor.Right(b2) => Xor.Right(BB.combine(b1, b2)) } } @@ -168,7 +198,7 @@ private[data] sealed abstract class XorInstances extends XorInstances1 { def show(f: A Xor B): String = f.show } - implicit def xorMonoid[A, B](implicit A: Semigroup[A], B: Monoid[B]): Monoid[A Xor B] = + implicit def xorMonoid[A, B](implicit B: Monoid[B]): Monoid[A Xor B] = new Monoid[A Xor B] { def empty: A Xor B = Xor.Right(B.empty) def combine(x: A Xor B, y: A Xor B): A Xor B = x combine y @@ -229,7 +259,7 @@ private[data] sealed abstract class XorInstances extends XorInstances1 { private[data] sealed abstract class XorInstances1 extends XorInstances2 { - implicit def xorSemigroup[A, B](implicit A: Semigroup[A], B: Semigroup[B]): Semigroup[A Xor B] = + implicit def xorSemigroup[A, B](implicit B: Semigroup[B]): Semigroup[A Xor B] = new Semigroup[A Xor B] { def combine(x: A Xor B, y: A Xor B): A Xor B = x combine y } diff --git a/core/src/main/scala/cats/data/XorT.scala b/core/src/main/scala/cats/data/XorT.scala index a3bfb686ec..2fdf95282e 100644 --- a/core/src/main/scala/cats/data/XorT.scala +++ b/core/src/main/scala/cats/data/XorT.scala @@ -110,7 +110,45 @@ final case class XorT[F[_], A, B](value: F[A Xor B]) { def merge[AA >: A](implicit ev: B <:< AA, F: Functor[F]): F[AA] = F.map(value)(_.fold(identity, ev.apply)) - def combine(that: XorT[F, A, B])(implicit F: Apply[F], A: Semigroup[A], B: Semigroup[B]): XorT[F, A, B] = + /** + * Similar to [[Xor.combine]] but mapped over an `F` context. + * + * Examples: + * {{{ + * scala> import cats.data.XorT + * scala> import cats.implicits._ + * scala> val l1: XorT[Option, String, Int] = XorT.left(Some("error 1")) + * scala> val l2: XorT[Option, String, Int] = XorT.left(Some("error 2")) + * scala> val r3: XorT[Option, String, Int] = XorT.right(Some(3)) + * scala> val r4: XorT[Option, String, Int] = XorT.right(Some(4)) + * scala> val noneXorT: XorT[Option, String, Int] = XorT.left(None) + * + * scala> l1 combine l2 + * res0: XorT[Option, String, Int] = XorT(Some(Left(error 1))) + * + * scala> l1 combine r3 + * res1: XorT[Option, String, Int] = XorT(Some(Left(error 1))) + * + * scala> r3 combine l1 + * res2: XorT[Option, String, Int] = XorT(Some(Left(error 1))) + * + * scala> r3 combine r4 + * res3: XorT[Option, String, Int] = XorT(Some(Right(7))) + * + * scala> l1 combine noneXorT + * res4: XorT[Option, String, Int] = XorT(None) + * + * scala> noneXorT combine l1 + * res5: XorT[Option, String, Int] = XorT(None) + * + * scala> r3 combine noneXorT + * res6: XorT[Option, String, Int] = XorT(None) + * + * scala> noneXorT combine r4 + * res7: XorT[Option, String, Int] = XorT(None) + * }}} + */ + def combine(that: XorT[F, A, B])(implicit F: Apply[F], B: Semigroup[B]): XorT[F, A, B] = XorT(F.map2(this.value, that.value)(_ combine _)) def toValidated(implicit F: Functor[F]): F[Validated[A, B]] =