From 7ef5e5b1af9555852c0f31e281622bdc23f4d775 Mon Sep 17 00:00:00 2001 From: peterneyens Date: Sat, 7 May 2016 17:58:02 +0200 Subject: [PATCH] Add Traverse.traverseM and add examples to ScalaDoc for traverse syntax. --- core/src/main/scala/cats/Traverse.scala | 6 ++ .../src/main/scala/cats/syntax/traverse.scala | 87 ++++++++++++++++++- .../test/scala/cats/tests/SyntaxTests.scala | 9 +- 3 files changed, 98 insertions(+), 4 deletions(-) diff --git a/core/src/main/scala/cats/Traverse.scala b/core/src/main/scala/cats/Traverse.scala index 08ad00ddb3..299acb844b 100644 --- a/core/src/main/scala/cats/Traverse.scala +++ b/core/src/main/scala/cats/Traverse.scala @@ -29,6 +29,12 @@ import simulacrum.typeclass def traverseU[A, GB](fa: F[A])(f: A => GB)(implicit U: Unapply[Applicative, GB]): U.M[F[U.A]] = U.TC.traverse(fa)(a => U.subst(f(a)))(this) + /** + * A traverse followed by flattening the inner result. + */ + def traverseM[G[_], A, B](fa: F[A])(f: A => G[F[B]])(implicit G: Applicative[G], F: FlatMap[F]): G[F[B]] = + G.map(traverse(fa)(f))(F.flatten) + /** * Thread all the G effects through the F structure to invert the * structure from F[G[A]] to G[F[A]]. diff --git a/core/src/main/scala/cats/syntax/traverse.scala b/core/src/main/scala/cats/syntax/traverse.scala index 0fb0c866cc..56ac664682 100644 --- a/core/src/main/scala/cats/syntax/traverse.scala +++ b/core/src/main/scala/cats/syntax/traverse.scala @@ -16,17 +16,98 @@ trait TraverseSyntax extends TraverseSyntax1 { } final class TraverseOps[F[_], A](fa: F[A])(implicit F: Traverse[F]) { - def traverse[G[_]: Applicative, B](f: A => G[B]): G[F[B]] = F.traverse(fa)(f) + /** + * @see [[Traverse.traverse]] + * + * Example: + * {{{ + * scala> import cats.data.Xor + * scala> import cats.std.list._ + * scala> import cats.std.option._ + * scala> import cats.syntax.traverse._ + * scala> def parseInt(s: String): Option[Int] = Xor.catchOnly[NumberFormatException](s.toInt).toOption + * scala> List("1", "2", "3").traverse(parseInt) + * res0: Option[List[Int]] = Some(List(1, 2, 3)) + * scala> List("1", "two", "3").traverse(parseInt) + * res1: Option[List[Int]] = None + * }}} + */ + def traverse[G[_]: Applicative, B](f: A => G[B]): G[F[B]] = + F.traverse(fa)(f) + /** + * @see [[Traverse.traverse]] + * + * Example: + * {{{ + * scala> import cats.data.Xor + * scala> import cats.std.list._ + * scala> import cats.syntax.traverse._ + * scala> def parseInt(s: String): Xor[String, Int] = Xor.catchOnly[NumberFormatException](s.toInt).leftMap(_ => "no number") + * scala> val ns = List("1", "2", "3") + * scala> ns.traverseU(parseInt) + * res0: Xor[String, List[Int]] = Right(List(1, 2, 3)) + * scala> ns.traverse[Xor[String, ?], Int](parseInt) + * res1: Xor[String, List[Int]] = Right(List(1, 2, 3)) + * }}} + */ def traverseU[GB](f: A => GB)(implicit U: Unapply[Applicative, GB]): U.M[F[U.A]] = F.traverseU[A, GB](fa)(f)(U) - def sequence[G[_], B](implicit G: Applicative[G], ev: A =:= G[B]): G[F[B]] = + /** + * @see [[Traverse.traverseM]] + * + * Example: + * {{{ + * scala> import cats.data.Xor + * scala> import cats.std.list._ + * scala> import cats.std.option._ + * scala> import cats.syntax.traverse._ + * scala> def parseInt(s: String): Option[Int] = Xor.catchOnly[NumberFormatException](s.toInt).toOption + * scala> val x = Option(List("1", "two", "3")) + * scala> x.traverseM(_.map(parseInt)) + * res0: List[Option[Int]] = List(Some(1), None, Some(3)) + * }}} + */ + def traverseM[G[_]: Applicative, B](f: A => G[F[B]])(implicit F2: FlatMap[F]): G[F[B]] = + F.traverseM(fa)(f) + + /** + * @see [[Traverse.sequence]] + * + * Example: + * {{{ + * scala> import cats.std.list._ + * scala> import cats.std.option._ + * scala> import cats.syntax.traverse._ + * scala> val x: List[Option[Int]] = List(Some(1), Some(2)) + * scala> val y: List[Option[Int]] = List(None, Some(2)) + * scala> x.sequence + * res0: Option[List[Int]] = Some(List(1, 2)) + * scala> y.sequence + * res1: Option[List[Int]] = None + * }}} + */ + def sequence[G[_], B](implicit G: Applicative[G], ev: A =:= G[B]): G[F[B]] = F.sequence(fa.asInstanceOf[F[G[B]]]) + /** + * @see [[Traverse.sequenceU]] + * + * Example: + * {{{ + * scala> import cats.data.{Validated, ValidatedNel} + * scala> import cats.std.list._ + * scala> import cats.syntax.traverse._ + * scala> val x: List[ValidatedNel[String, Int]] = List(Validated.valid(1), Validated.invalid("a"), Validated.invalid("b")).map(_.toValidatedNel) + * scala> x.sequenceU + * res0: cats.data.ValidatedNel[String,List[Int]] = Invalid(OneAnd(a,List(b))) + * scala> x.sequence[ValidatedNel[String, ?], Int] + * res1: cats.data.ValidatedNel[String,List[Int]] = Invalid(OneAnd(a,List(b))) + * }}} + */ def sequenceU(implicit U: Unapply[Applicative,A]): U.M[F[U.A]] = F.sequenceU[A](fa)(U) - } final class NestedTraverseOps[F[_], G[_], A](fga: F[G[A]])(implicit F: Traverse[F]) { diff --git a/tests/src/test/scala/cats/tests/SyntaxTests.scala b/tests/src/test/scala/cats/tests/SyntaxTests.scala index 9208c26b06..8228c73e68 100644 --- a/tests/src/test/scala/cats/tests/SyntaxTests.scala +++ b/tests/src/test/scala/cats/tests/SyntaxTests.scala @@ -110,7 +110,14 @@ class SyntaxTests extends AllInstances with AllSyntax { val as2: List[A] = fa.dropWhile_(f5) } - def testTraverse[F[_]: Traverse, G[_]: Applicative, A]: Unit = { + def testTraverse[F[_]: Traverse: FlatMap, G[_]: Applicative, A, B]: Unit = { + val fa = mock[F[A]] + val f1 = mock[A => G[B]] + val gfb: G[F[B]] = fa.traverse(f1) + + val f2 = mock[A => G[F[B]]] + val gfb2: G[F[B]] = fa.traverseM(f2) + val fga = mock[F[G[A]]] val gunit: G[F[A]] = fga.sequence }