diff --git a/core/src/main/scala/cats/FunctorFilter.scala b/core/src/main/scala/cats/FunctorFilter.scala new file mode 100644 index 0000000000..4abcc08b25 --- /dev/null +++ b/core/src/main/scala/cats/FunctorFilter.scala @@ -0,0 +1,69 @@ +package cats + +import simulacrum.typeclass + +/** + * `FunctorFilter[F]` allows you to `map` and filter out elements simultaneously. + */ +@typeclass +trait FunctorFilter[F[_]] extends Serializable { + def functor: Functor[F] + + /** + * A combined `map` and `filter`. Filtering is handled via `Option` + * instead of `Boolean` such that the output type `B` can be different than + * the input type `A`. + * + * Example: + * {{{ + * scala> import cats.implicits._ + * scala> val m: Map[Int, String] = Map(1 -> "one", 3 -> "three") + * scala> val l: List[Int] = List(1, 2, 3, 4) + * scala> def asString(i: Int): Option[String] = m.get(i) + * scala> l.mapFilter(i => m.get(i)) + * res0: List[String] = List(one, three) + * }}} + */ + def mapFilter[A, B](fa: F[A])(f: A => Option[B]): F[B] + + /** + * Similar to [[mapFilter]] but uses a partial function instead of a function + * that returns an `Option`. + * + * Example: + * {{{ + * scala> import cats.implicits._ + * scala> val l: List[Int] = List(1, 2, 3, 4) + * scala> FunctorFilter[List].collect(l){ + * | case 1 => "one" + * | case 3 => "three" + * | } + * res0: List[String] = List(one, three) + * }}} + */ + def collect[A, B](fa: F[A])(f: PartialFunction[A, B]): F[B] = + mapFilter(fa)(f.lift) + + /** + * "Flatten" out a structure by collapsing `Option`s. + * Equivalent to using `mapFilter` with `identity`. + * + * Example: + * {{{ + * scala> import cats.implicits._ + * scala> val l: List[Option[Int]] = List(Some(1), None, Some(3), None) + * scala> l.flattenOption + * res0: List[Int] = List(1, 3) + * }}} + */ + def flattenOption[A](fa: F[Option[A]]): F[A] = + mapFilter(fa)(identity) + + /** + * Apply a filter to a structure such that the output structure contains all + * `A` elements in the input structure that satisfy the predicate `f` but none + * that don't. + */ + def filter[A](fa: F[A])(f: A => Boolean): F[A] = + mapFilter(fa)(a => if (f(a)) Some(a) else None) +} diff --git a/core/src/main/scala/cats/TraverseFilter.scala b/core/src/main/scala/cats/TraverseFilter.scala new file mode 100644 index 0000000000..ea2848de63 --- /dev/null +++ b/core/src/main/scala/cats/TraverseFilter.scala @@ -0,0 +1,62 @@ +package cats + +import simulacrum.typeclass + +/** + * `TraverseFilter`, also known as `Witherable`, represents list-like structures + * that can essentially have a `traverse` and a `filter` applied as a single + * combined operation (`traverseFilter`). + * + * Based on Haskell's [[https://hackage.haskell.org/package/witherable-0.1.3.3/docs/Data-Witherable.html Data.Witherable]] + */ + +@typeclass +trait TraverseFilter[F[_]] extends FunctorFilter[F] { + def traverse: Traverse[F] + + final override def functor: Functor[F] = traverse + + /** + * A combined [[traverse]] and [[filter]]. Filtering is handled via `Option` + * instead of `Boolean` such that the output type `B` can be different than + * the input type `A`. + * + * Example: + * {{{ + * scala> import cats.implicits._ + * scala> val m: Map[Int, String] = Map(1 -> "one", 3 -> "three") + * scala> val l: List[Int] = List(1, 2, 3, 4) + * scala> def asString(i: Int): Eval[Option[String]] = Now(m.get(i)) + * scala> val result: Eval[List[String]] = l.traverseFilter(asString) + * scala> result.value + * res0: List[String] = List(one, three) + * }}} + */ + def traverseFilter[G[_], A, B](fa: F[A])(f: A => G[Option[B]])(implicit G: Applicative[G]): G[F[B]] + + /** + * + * Filter values inside a `G` context. + * + * This is a generalized version of Haskell's [[http://hackage.haskell.org/package/base-4.9.0.0/docs/Control-Monad.html#v:filterM filterM]]. + * [[http://stackoverflow.com/questions/28872396/haskells-filterm-with-filterm-x-true-false-1-2-3 This StackOverflow question]] about `filterM` may be helpful in understanding how it behaves. + * + * Example: + * {{{ + * scala> import cats.implicits._ + * scala> val l: List[Int] = List(1, 2, 3, 4) + * scala> def odd(i: Int): Eval[Boolean] = Now(i % 2 == 1) + * scala> val res: Eval[List[Int]] = l.filterA(odd) + * scala> res.value + * res0: List[Int] = List(1, 3) + * + * scala> List(1, 2, 3).filterA(_ => List(true, false)) + * res1: List[List[Int]] = List(List(1, 2, 3), List(1, 2), List(1, 3), List(1), List(2, 3), List(2), List(3), List()) + * }}} + */ + def filterA[G[_], A](fa: F[A])(f: A => G[Boolean])(implicit G: Applicative[G]): G[F[A]] = + traverseFilter(fa)(a => G.map(f(a))(if (_) Some(a) else None)) + + override def mapFilter[A, B](fa: F[A])(f: A => Option[B]): F[B] = + traverseFilter[Id, A, B](fa)(f) +} diff --git a/core/src/main/scala/cats/data/Chain.scala b/core/src/main/scala/cats/data/Chain.scala index 361f0e31b6..904856852d 100644 --- a/core/src/main/scala/cats/data/Chain.scala +++ b/core/src/main/scala/cats/data/Chain.scala @@ -597,6 +597,29 @@ private[data] sealed abstract class ChainInstances extends ChainInstances1 { } } + implicit val catsDataTraverseFilterForChain: TraverseFilter[Chain] = new TraverseFilter[Chain] { + def traverse: Traverse[Chain] = Chain.catsDataInstancesForChain + + override def filter[A](fa: Chain[A])(f: A => Boolean): Chain[A] = fa.filter(f) + + override def collect[A, B](fa: Chain[A])(f: PartialFunction[A, B]): Chain[B] = fa.collect(f) + + override def mapFilter[A, B](fa: Chain[A])(f: A => Option[B]): Chain[B] = fa.collect(Function.unlift(f)) + + override def flattenOption[A](fa: Chain[Option[A]]): Chain[A] = fa.collect { case Some(a) => a } + + def traverseFilter[G[_], A, B](fa: Chain[A])(f: A => G[Option[B]])(implicit G: Applicative[G]): G[Chain[B]] = + fa.foldRight(G.pure(Chain.empty[B]))( + (a, gcb) => G.map2(f(a), gcb)((ob, cb) => ob.fold(cb)(_ +: cb)) + ) + + override def filterA[G[_], A](fa: Chain[A])(f: A => G[Boolean])(implicit G: Applicative[G]): G[Chain[A]] = + fa.foldRight(G.pure(Chain.empty[A]))( + (a, gca) => + G.map2(f(a), gca)((b, chain) => if (b) a +: chain else chain)) + + } + } private[data] sealed abstract class ChainInstances1 extends ChainInstances2 { diff --git a/core/src/main/scala/cats/data/Const.scala b/core/src/main/scala/cats/data/Const.scala index 305f1fcffb..9532cc7440 100644 --- a/core/src/main/scala/cats/data/Const.scala +++ b/core/src/main/scala/cats/data/Const.scala @@ -87,6 +87,25 @@ private[data] sealed abstract class ConstInstances extends ConstInstances0 { fa.traverse(f) } + implicit def catsDataTraverseFilterForConst[C]: TraverseFilter[Const[C, ?]] = new TraverseFilter[Const[C, ?]] { + + override def mapFilter[A, B](fa: Const[C, A])(f: (A) => Option[B]): Const[C, B] = fa.retag + + override def collect[A, B](fa: Const[C, A])(f: PartialFunction[A, B]): Const[C, B] = fa.retag + + override def flattenOption[A](fa: Const[C, Option[A]]): Const[C, A] = fa.retag + + override def filter[A](fa: Const[C, A])(f: (A) => Boolean): Const[C, A] = fa.retag + + def traverseFilter[G[_], A, B](fa: Const[C, A])(f: (A) => G[Option[B]])(implicit G: Applicative[G]): G[Const[C, B]] = + G.pure(fa.retag[B]) + + override def filterA[G[_], A](fa: Const[C, A])(f: (A) => G[Boolean])(implicit G: Applicative[G]): G[Const[C, A]] = + G.pure(fa) + + val traverse: Traverse[Const[C, ?]] = Const.catsDataTraverseForConst[C] + } + implicit def catsDataMonoidForConst[A: Monoid, B]: Monoid[Const[A, B]] = new Monoid[Const[A, B]]{ def empty: Const[A, B] = Const.empty diff --git a/core/src/main/scala/cats/data/Nested.scala b/core/src/main/scala/cats/data/Nested.scala index f237a56b7a..e780bd6aa7 100644 --- a/core/src/main/scala/cats/data/Nested.scala +++ b/core/src/main/scala/cats/data/Nested.scala @@ -2,7 +2,6 @@ package cats package data - /** Similar to [[cats.data.Tuple2K]], but for nested composition. * * For instance, since both `List` and `Option` have a `Functor`, then so does @@ -54,6 +53,12 @@ private[data] sealed abstract class NestedInstances extends NestedInstances0 { def defer[A](fa: => Nested[F, G, A]): Nested[F, G, A] = Nested(F.defer(fa.value)) } + + implicit def catsDataTraverseFilterForNested[F[_], G[_]](implicit F0: Traverse[F], G0: TraverseFilter[G]): TraverseFilter[Nested[F, G, ?]] = + new NestedTraverseFilter[F, G] { + implicit val F: Traverse[F] = F0 + implicit val G: TraverseFilter[G] = G0 + } } private[data] sealed abstract class NestedInstances0 extends NestedInstances1 { @@ -61,6 +66,12 @@ private[data] sealed abstract class NestedInstances0 extends NestedInstances1 { new NestedTraverse[F, G] { val FG: Traverse[λ[α => F[G[α]]]] = Traverse[F].compose[G] } + + implicit def catsDataFunctorFilterForNested[F[_], G[_]](implicit F0: Functor[F], G0: FunctorFilter[G]): FunctorFilter[Nested[F, G, ?]] = + new NestedFunctorFilter[F, G] { + implicit val F: Functor[F] = F0 + implicit val G: FunctorFilter[G] = G0 + } } private[data] sealed abstract class NestedInstances1 extends NestedInstances2 { @@ -315,3 +326,44 @@ private[data] trait NestedInvariantSemigroupalApply[F[_], G[_]] extends Invarian def product[A, B](fa: Nested[F, G, A], fb: Nested[F, G, B]): Nested[F, G, (A, B)] = Nested(FG.product(fa.value, fb.value)) } + +private[data] abstract class NestedFunctorFilter[F[_], G[_]] extends FunctorFilter[Nested[F, G, ?]] { + implicit val F: Functor[F] + + implicit val G: FunctorFilter[G] + + def functor: Functor[Nested[F, G, ?]] = Nested.catsDataFunctorForNested(F, G.functor) + + def mapFilter[A, B](fa: Nested[F, G, A])(f: (A) => Option[B]): Nested[F, G, B] = + Nested[F, G, B](F.map(fa.value)(G.mapFilter(_)(f))) + + override def collect[A, B](fa: Nested[F, G, A])(f: PartialFunction[A, B]): Nested[F, G, B] = + Nested[F, G, B](F.map(fa.value)(G.collect(_)(f))) + + override def flattenOption[A](fa: Nested[F, G, Option[A]]): Nested[F, G, A] = + Nested[F, G, A](F.map(fa.value)(G.flattenOption)) + + override def filter[A](fa: Nested[F, G, A])(f: (A) => Boolean): Nested[F, G, A] = + Nested[F, G, A](F.map(fa.value)(G.filter(_)(f))) +} + +private[data] abstract class NestedTraverseFilter[F[_], G[_]] + extends NestedFunctorFilter[F, G] with TraverseFilter[Nested[F, G, ?]] { + implicit val F: Traverse[F] + + implicit val G: TraverseFilter[G] + + def traverse: Traverse[Nested[F, G, ?]] = Nested.catsDataTraverseForNested(F, G.traverse) + + override def filterA[H[_], A] + (fa: Nested[F, G, A]) + (f: A => H[Boolean]) + (implicit H: Applicative[H]): H[Nested[F, G, A]] = + H.map(F.traverse(fa.value)(G.filterA[H, A](_)(f)))(Nested[F, G, A]) + + def traverseFilter[H[_], A, B] + (fga: Nested[F, G, A]) + (f: A => H[Option[B]]) + (implicit H: Applicative[H]): H[Nested[F, G, B]] = + H.map(F.traverse[H, G[A], G[B]](fga.value)(ga => G.traverseFilter(ga)(f)))(Nested[F, G, B]) +} diff --git a/core/src/main/scala/cats/data/OptionT.scala b/core/src/main/scala/cats/data/OptionT.scala index 70157cd36a..9cac0f5c2a 100644 --- a/core/src/main/scala/cats/data/OptionT.scala +++ b/core/src/main/scala/cats/data/OptionT.scala @@ -1,7 +1,7 @@ package cats package data -import cats.instances.option.{catsStdInstancesForOption => optionInstance} +import cats.instances.option.{catsStdInstancesForOption => optionInstance, catsStdTraverseFilterForOption} import cats.syntax.either._ /** @@ -234,6 +234,26 @@ private[data] sealed abstract class OptionTInstances extends OptionTInstances0 { def defer[A](fa: => OptionT[F, A]): OptionT[F, A] = OptionT(F.defer(fa.value)) } + + implicit def catsDateTraverseFilterForOptionT[F[_]](implicit F0: Traverse[F]): TraverseFilter[OptionT[F, ?]] = + new OptionTFunctorFilter[F] with TraverseFilter[OptionT[F, ?]] { + implicit def F: Functor[F] = F0 + + val traverse: Traverse[OptionT[F, ?]] = OptionT.catsDataTraverseForOptionT[F] + + def traverseFilter[G[_], A, B](fa: OptionT[F, A]) + (f: A => G[Option[B]]) + (implicit G: Applicative[G]): G[OptionT[F, B]] = + G.map(Traverse[F].traverse[G, Option[A], Option[B]](fa.value) { + oa => TraverseFilter[Option].traverseFilter(oa)(f) + })(OptionT[F, B]) + + override def filterA[G[_], A](fa: OptionT[F, A]) + (f: A => G[Boolean]) + (implicit G: Applicative[G]): G[OptionT[F, A]] = + G.map(Traverse[F].traverse(fa.value)(TraverseFilter[Option].filterA[G, A](_)(f)))(OptionT[F, A]) + + } } private[data] sealed abstract class OptionTInstances0 extends OptionTInstances1 { @@ -251,6 +271,9 @@ private[data] sealed abstract class OptionTInstances0 extends OptionTInstances1 implicit def catsDataPartialOrderForOptionT[F[_], A](implicit F0: PartialOrder[F[Option[A]]]): PartialOrder[OptionT[F, A]] = new OptionTPartialOrder[F, A] { implicit val F = F0 } + + implicit def catsDateFunctorFilterForOptionT[F[_]](implicit F0: Functor[F]): FunctorFilter[OptionT[F, ?]] = + new OptionTFunctorFilter[F] { implicit val F = F0 } } private[data] sealed abstract class OptionTInstances1 extends OptionTInstances2 { @@ -387,6 +410,20 @@ private[data] sealed trait OptionTPartialOrder[F[_], A] extends PartialOrder[Opt override def partialCompare(x: OptionT[F, A], y: OptionT[F, A]): Double = x partialCompare y } +private[data] sealed trait OptionTFunctorFilter[F[_]] extends FunctorFilter[OptionT[F, ?]] { + implicit def F: Functor[F] + + def functor: Functor[OptionT[F, ?]] = OptionT.catsDataFunctorForOptionT[F] + + def mapFilter[A, B](fa: OptionT[F, A])(f: (A) => Option[B]): OptionT[F, B] = fa.subflatMap(f) + + override def collect[A, B](fa: OptionT[F, A])(f: PartialFunction[A, B]): OptionT[F, B] = fa.subflatMap(f.lift) + + override def flattenOption[A](fa: OptionT[F, Option[A]]): OptionT[F, A] = fa.subflatMap(identity) + + override def filter[A](fa: OptionT[F, A])(f: (A) => Boolean): OptionT[F, A] = fa.filter(f) +} + private[data] sealed trait OptionTOrder[F[_], A] extends Order[OptionT[F, A]] with OptionTPartialOrder[F, A]{ override implicit def F: Order[F[Option[A]]] diff --git a/core/src/main/scala/cats/instances/all.scala b/core/src/main/scala/cats/instances/all.scala index 5d82c6d715..95fa9bc6c8 100644 --- a/core/src/main/scala/cats/instances/all.scala +++ b/core/src/main/scala/cats/instances/all.scala @@ -38,4 +38,9 @@ trait AllInstancesBinCompat0 with Tuple2InstancesBinCompat0 trait AllInstancesBinCompat1 - extends MapInstancesBinCompat0 + extends OptionInstancesBinCompat0 + with ListInstancesBinCompat0 + with VectorInstancesBinCompat0 + with StreamInstancesBinCompat0 + with MapInstancesBinCompat0 + with SortedMapInstancesBinCompat0 diff --git a/core/src/main/scala/cats/instances/list.scala b/core/src/main/scala/cats/instances/list.scala index 4ba1e829d8..7ef2972a0b 100644 --- a/core/src/main/scala/cats/instances/list.scala +++ b/core/src/main/scala/cats/instances/list.scala @@ -143,3 +143,29 @@ trait ListInstances extends cats.kernel.instances.ListInstances { fa.iterator.map(_.show).mkString("List(", ", ", ")") } } + +trait ListInstancesBinCompat0 { + implicit val catsStdTraverseFilterForList: TraverseFilter[List] = new TraverseFilter[List] { + val traverse: Traverse[List] = cats.instances.list.catsStdInstancesForList + + override def mapFilter[A, B](fa: List[A])(f: (A) => Option[B]): List[B] = fa.collect(Function.unlift(f)) + + override def filter[A](fa: List[A])(f: (A) => Boolean): List[A] = fa.filter(f) + + override def collect[A, B](fa: List[A])(f: PartialFunction[A, B]): List[B] = fa.collect(f) + + override def flattenOption[A](fa: List[Option[A]]): List[A] = fa.flatten + + def traverseFilter[G[_], A, B](fa: List[A])(f: (A) => G[Option[B]])(implicit G: Applicative[G]): G[List[B]] = + fa.foldRight(Eval.now(G.pure(List.empty[B])))( + (x, xse) => + G.map2Eval(f(x), xse)((i, o) => i.fold(o)(_ :: o)) + ).value + + override def filterA[G[_], A](fa: List[A])(f: (A) => G[Boolean])(implicit G: Applicative[G]): G[List[A]] = + fa.foldRight(Eval.now(G.pure(List.empty[A])))( + (x, xse) => + G.map2Eval(f(x), xse)((b, list) => if (b) x :: list else list) + ).value + } +} diff --git a/core/src/main/scala/cats/instances/map.scala b/core/src/main/scala/cats/instances/map.scala index dd044faf36..fda0d30052 100644 --- a/core/src/main/scala/cats/instances/map.scala +++ b/core/src/main/scala/cats/instances/map.scala @@ -107,7 +107,26 @@ trait MapInstancesBinCompat0 { } } } + } + + implicit def catsStdFunctorFilterForMap[K]: FunctorFilter[Map[K, ?]] = { + new FunctorFilter[Map[K, ?]] { + + val functor: Functor[Map[K, ?]] = cats.instances.map.catsStdInstancesForMap[K] + + def mapFilter[A, B](fa: Map[K, A])(f: A => Option[B]) = + fa.collect(scala.Function.unlift(t => f(t._2).map(t._1 -> _))) + override def collect[A, B](fa: Map[K, A])(f: PartialFunction[A, B]) = + fa.collect(scala.Function.unlift(t => f.lift(t._2).map(t._1 -> _))) + + override def flattenOption[A](fa: Map[K, Option[A]]) = + fa.collect(scala.Function.unlift(t => t._2.map(t._1 -> _))) + + override def filter[A](fa: Map[K, A])(f: A => Boolean) = + fa.filter { case (_, v) => f(v) } + + } } } diff --git a/core/src/main/scala/cats/instances/option.scala b/core/src/main/scala/cats/instances/option.scala index 7ca772bc13..0a84a02a65 100644 --- a/core/src/main/scala/cats/instances/option.scala +++ b/core/src/main/scala/cats/instances/option.scala @@ -121,3 +121,30 @@ trait OptionInstances extends cats.kernel.instances.OptionInstances { } } } + +trait OptionInstancesBinCompat0 { + implicit val catsStdTraverseFilterForOption: TraverseFilter[Option] = new TraverseFilter[Option] { + val traverse: Traverse[Option] = cats.instances.option.catsStdInstancesForOption + + override def mapFilter[A, B](fa: Option[A])(f: (A) => Option[B]): Option[B] = fa.flatMap(f) + + override def filter[A](fa: Option[A])(f: (A) => Boolean): Option[A] = fa.filter(f) + + override def collect[A, B](fa: Option[A])(f: PartialFunction[A, B]): Option[B] = fa.collect(f) + + override def flattenOption[A](fa: Option[Option[A]]): Option[A] = fa.flatten + + def traverseFilter[G[_], A, B](fa: Option[A])(f: (A) => G[Option[B]])(implicit G: Applicative[G]): G[Option[B]] = + fa match { + case None => G.pure(Option.empty[B]) + case Some(a) => f(a) + } + + override def filterA[G[_], A](fa: Option[A])(f: (A) => G[Boolean])(implicit G: Applicative[G]): G[Option[A]] = + fa match { + case None => G.pure(Option.empty[A]) + case Some(a) => G.map(f(a))(b => if (b) Some(a) else None) + } + + } +} diff --git a/core/src/main/scala/cats/instances/package.scala b/core/src/main/scala/cats/instances/package.scala index f398b3ad65..6aaefa4373 100644 --- a/core/src/main/scala/cats/instances/package.scala +++ b/core/src/main/scala/cats/instances/package.scala @@ -14,15 +14,14 @@ package object instances { object eq extends EqInstances object equiv extends EquivInstances object float extends FloatInstances - object function extends FunctionInstances - with FunctionInstancesBinCompat0 + object function extends FunctionInstances with FunctionInstancesBinCompat0 object future extends FutureInstances object int extends IntInstances object invariant extends InvariantMonoidalInstances - object list extends ListInstances + object list extends ListInstances with ListInstancesBinCompat0 object long extends LongInstances + object option extends OptionInstances with OptionInstancesBinCompat0 object map extends MapInstances with MapInstancesBinCompat0 - object option extends OptionInstances object order extends OrderInstances object ordering extends OrderingInstances object parallel extends ParallelInstances @@ -33,12 +32,11 @@ package object instances { object short extends ShortInstances object sortedMap extends SortedMapInstances object sortedSet extends SortedSetInstances - object stream extends StreamInstances + object stream extends StreamInstances with StreamInstancesBinCompat0 object string extends StringInstances object try_ extends TryInstances - object tuple extends TupleInstances - with Tuple2InstancesBinCompat0 + object tuple extends TupleInstances with Tuple2InstancesBinCompat0 object unit extends UnitInstances object uuid extends UUIDInstances - object vector extends VectorInstances + object vector extends VectorInstances with VectorInstancesBinCompat0 } diff --git a/core/src/main/scala/cats/instances/sortedMap.scala b/core/src/main/scala/cats/instances/sortedMap.scala index df27ea937c..b3d83a4e92 100644 --- a/core/src/main/scala/cats/instances/sortedMap.scala +++ b/core/src/main/scala/cats/instances/sortedMap.scala @@ -1,6 +1,6 @@ package cats.instances -import cats.{Always, Applicative, Eval, FlatMap, Foldable, Monoid, Show, Traverse} +import cats.{Always, Applicative, Eval, FlatMap, Foldable, Monoid, Order, Show, Traverse, TraverseFilter} import cats.kernel._ import cats.kernel.instances.StaticMethods @@ -166,3 +166,42 @@ class SortedMapMonoid[K, V](implicit V: Semigroup[V], O: Order[K]) extends Monoi } } + +trait SortedMapInstancesBinCompat0 { + implicit def catsStdTraverseFilterForSortedMap[K: Order]: TraverseFilter[SortedMap[K, ?]] = + new TraverseFilter[SortedMap[K, ?]] { + + implicit val ordering: Ordering[K] = Order[K].toOrdering + + val traverse: Traverse[SortedMap[K, ?]] = cats.instances.sortedMap.catsStdInstancesForSortedMap[K] + + override def traverseFilter[G[_], A, B] + (fa: SortedMap[K, A]) + (f: A => G[Option[B]]) + (implicit G: Applicative[G]): G[SortedMap[K, B]] = { + val gba: Eval[G[SortedMap[K, B]]] = Always(G.pure(SortedMap.empty)) + Foldable.iterateRight(fa, gba) { (kv, lbuf) => + G.map2Eval(f(kv._2), lbuf)({ (ob, buf) => ob.fold(buf)(b => buf + (kv._1 -> b)) }) + }.value + } + + override def mapFilter[A, B](fa: SortedMap[K, A])(f: (A) => Option[B]): SortedMap[K, B] = + fa.collect(scala.Function.unlift(t => f(t._2).map(t._1 -> _))) + + override def collect[A, B](fa: SortedMap[K, A])(f: PartialFunction[A, B]): SortedMap[K, B] = + fa.collect(scala.Function.unlift(t => f.lift(t._2).map(t._1 -> _))) + + override def flattenOption[A](fa: SortedMap[K, Option[A]]): SortedMap[K, A] = + fa.collect(scala.Function.unlift(t => t._2.map(t._1 -> _))) + + override def filter[A](fa: SortedMap[K, A])(f: (A) => Boolean): SortedMap[K, A] = + fa.filter { case (_, v) => f(v) } + + + override def filterA[G[_], A] + (fa: SortedMap[K, A]) + (f: (A) => G[Boolean]) + (implicit G: Applicative[G]): G[SortedMap[K, A]] = + traverseFilter(fa)(a => G.map(f(a))(if (_) Some(a) else None)) + } +} diff --git a/core/src/main/scala/cats/instances/stream.scala b/core/src/main/scala/cats/instances/stream.scala index 8a4e79c4b3..1d97c90fb0 100644 --- a/core/src/main/scala/cats/instances/stream.scala +++ b/core/src/main/scala/cats/instances/stream.scala @@ -153,3 +153,32 @@ trait StreamInstances extends cats.kernel.instances.StreamInstances { def show(fa: Stream[A]): String = if (fa.isEmpty) "Stream()" else s"Stream(${fa.head.show}, ?)" } } + +trait StreamInstancesBinCompat0 { + implicit val catsStdTraverseFilterForStream: TraverseFilter[Stream] = new TraverseFilter[Stream] { + val traverse: Traverse[Stream] = cats.instances.stream.catsStdInstancesForStream + + override def mapFilter[A, B](fa: Stream[A])(f: (A) => Option[B]): Stream[B] = { + fa.collect(Function.unlift(f)) + } + + override def filter[A](fa: Stream[A])(f: (A) => Boolean): Stream[A] = fa.filter(f) + + override def collect[A, B](fa: Stream[A])(f: PartialFunction[A, B]): Stream[B] = fa.collect(f) + + override def flattenOption[A](fa: Stream[Option[A]]): Stream[A] = fa.flatten + + def traverseFilter[G[_], A, B](fa: Stream[A])(f: (A) => G[Option[B]])(implicit G: Applicative[G]): G[Stream[B]] = { + fa.foldRight(Eval.now(G.pure(Stream.empty[B])))( + (x, xse) => G.map2Eval(f(x), xse)((i, o) => i.fold(o)(_ +: o)) + ).value + } + + override def filterA[G[_], A](fa: Stream[A])(f: (A) => G[Boolean])(implicit G: Applicative[G]): G[Stream[A]] = { + fa.foldRight(Eval.now(G.pure(Stream.empty[A])))( + (x, xse) => G.map2Eval(f(x), xse)((b, as) => if (b) x +: as else as) + ).value + } + + } +} diff --git a/core/src/main/scala/cats/instances/vector.scala b/core/src/main/scala/cats/instances/vector.scala index a27175daf1..1b1ddec740 100644 --- a/core/src/main/scala/cats/instances/vector.scala +++ b/core/src/main/scala/cats/instances/vector.scala @@ -2,6 +2,7 @@ package cats package instances import cats.syntax.show._ + import scala.annotation.tailrec import scala.collection.+: import scala.collection.immutable.VectorBuilder @@ -113,3 +114,29 @@ trait VectorInstances extends cats.kernel.instances.VectorInstances { fa.iterator.map(_.show).mkString("Vector(", ", ", ")") } } + +trait VectorInstancesBinCompat0 { + implicit val catsStdTraverseFilterForVector: TraverseFilter[Vector] = new TraverseFilter[Vector] { + val traverse: Traverse[Vector] = cats.instances.vector.catsStdInstancesForVector + + override def mapFilter[A, B](fa: Vector[A])(f: (A) => Option[B]): Vector[B] = + fa.collect(Function.unlift(f)) + + override def filter[A](fa: Vector[A])(f: (A) => Boolean): Vector[A] = fa.filter(f) + + override def collect[A, B](fa: Vector[A])(f: PartialFunction[A, B]): Vector[B] = fa.collect(f) + + override def flattenOption[A](fa: Vector[Option[A]]): Vector[A] = fa.flatten + + def traverseFilter[G[_], A, B](fa: Vector[A])(f: (A) => G[Option[B]])(implicit G: Applicative[G]): G[Vector[B]] = + fa.foldRight(Eval.now(G.pure(Vector.empty[B])))( + (x, xse) => G.map2Eval(f(x), xse)((i, o) => i.fold(o)(_ +: o)) + ).value + + override def filterA[G[_], A](fa: Vector[A])(f: (A) => G[Boolean])(implicit G: Applicative[G]): G[Vector[A]] = + fa.foldRight(Eval.now(G.pure(Vector.empty[A])))( + (x, xse) => + G.map2Eval(f(x), xse)((b, vec) => if (b) x +: vec else vec) + ).value + } +} diff --git a/core/src/main/scala/cats/syntax/all.scala b/core/src/main/scala/cats/syntax/all.scala index 78a67f51df..dd270b7ef5 100644 --- a/core/src/main/scala/cats/syntax/all.scala +++ b/core/src/main/scala/cats/syntax/all.scala @@ -72,6 +72,8 @@ trait AllSyntaxBinCompat1 trait AllSyntaxBinCompat2 extends ParallelTraverseSyntax + with TraverseFilterSyntax + with FunctorFilterSyntax with EitherSyntaxBinCompat0 with ListSyntaxBinCompat0 with ValidatedSyntaxBincompat0 diff --git a/core/src/main/scala/cats/syntax/functorFilter.scala b/core/src/main/scala/cats/syntax/functorFilter.scala new file mode 100644 index 0000000000..a80b6c2618 --- /dev/null +++ b/core/src/main/scala/cats/syntax/functorFilter.scala @@ -0,0 +1,4 @@ +package cats +package syntax + +trait FunctorFilterSyntax extends FunctorFilter.ToFunctorFilterOps diff --git a/core/src/main/scala/cats/syntax/package.scala b/core/src/main/scala/cats/syntax/package.scala index 03b4413f22..771706bee0 100644 --- a/core/src/main/scala/cats/syntax/package.scala +++ b/core/src/main/scala/cats/syntax/package.scala @@ -28,6 +28,7 @@ package object syntax { object flatMap extends FlatMapSyntax object foldable extends FoldableSyntax object functor extends FunctorSyntax + object functorFilter extends FunctorFilterSyntax object group extends GroupSyntax object invariant extends InvariantSyntax object ior extends IorSyntax @@ -49,6 +50,7 @@ package object syntax { object show extends ShowSyntax object strong extends StrongSyntax object traverse extends TraverseSyntax + object traverseFilter extends TraverseFilterSyntax object nonEmptyTraverse extends NonEmptyTraverseSyntax object unorderedTraverse extends UnorderedTraverseSyntax object validated extends ValidatedSyntax with ValidatedExtensionSyntax with ValidatedSyntaxBincompat0 diff --git a/core/src/main/scala/cats/syntax/traverseFilter.scala b/core/src/main/scala/cats/syntax/traverseFilter.scala new file mode 100644 index 0000000000..794caeddee --- /dev/null +++ b/core/src/main/scala/cats/syntax/traverseFilter.scala @@ -0,0 +1,4 @@ +package cats +package syntax + +trait TraverseFilterSyntax extends TraverseFilter.ToTraverseFilterOps diff --git a/laws/src/main/scala/cats/laws/FunctorFilterLaws.scala b/laws/src/main/scala/cats/laws/FunctorFilterLaws.scala new file mode 100644 index 0000000000..32af361b1d --- /dev/null +++ b/laws/src/main/scala/cats/laws/FunctorFilterLaws.scala @@ -0,0 +1,38 @@ +package cats +package laws + + +trait FunctorFilterLaws[F[_]] { + implicit def F: FunctorFilter[F] + + + implicit def functor: Functor[F] = F.functor + + def mapFilterComposition[A, B, C](fa: F[A], f: A => Option[B], g: B => Option[C]): IsEq[F[C]] = { + val lhs: F[C] = F.mapFilter(F.mapFilter(fa)(f))(g) + val rhs: F[C] = F.mapFilter(fa)(a => f(a).flatMap(g)) + lhs <-> rhs + } + + def mapFilterMapConsistency[A, B](fa: F[A], f: A => B): IsEq[F[B]] = { + F.mapFilter(fa)(f andThen (x => Some(x): Option[B])) <-> functor.map(fa)(f) + } + + def collectConsistentWithMapFilter[A, B](fa: F[A], f: PartialFunction[A, B]): IsEq[F[B]] = { + F.collect(fa)(f) <-> F.mapFilter(fa)(f.lift) + } + + def flattenOptionConsistentWithMapFilter[A](fa: F[Option[A]]): IsEq[F[A]] = { + F.flattenOption(fa) <-> F.mapFilter(fa)(identity) + } + + def filterConsistentWithMapFilter[A](fa: F[A], f: A => Boolean): IsEq[F[A]] = { + F.filter(fa)(f) <-> + F.mapFilter(fa)(a => if (f(a)) Some(a) else None) + } +} + +object FunctorFilterLaws { + def apply[F[_]](implicit ev: FunctorFilter[F]): FunctorFilterLaws[F] = + new FunctorFilterLaws[F] { def F: FunctorFilter[F] = ev } +} diff --git a/laws/src/main/scala/cats/laws/TraverseFilterLaws.scala b/laws/src/main/scala/cats/laws/TraverseFilterLaws.scala new file mode 100644 index 0000000000..bd799fa8f6 --- /dev/null +++ b/laws/src/main/scala/cats/laws/TraverseFilterLaws.scala @@ -0,0 +1,41 @@ +package cats +package laws + +import cats.data.Nested +import cats.syntax.all._ +import cats.instances.option._ + +trait TraverseFilterLaws[F[_]] extends FunctorFilterLaws[F] { + implicit override def F: TraverseFilter[F] + + def traverseFilterIdentity[G[_]: Applicative, A](fa: F[A]): IsEq[G[F[A]]] = { + fa.traverseFilter(_.some.pure[G]) <-> fa.pure[G] + } + + def traverseFilterConsistentWithTraverse[G[_]: Applicative, A](fa: F[A], f: A => G[A]): IsEq[G[F[A]]] = { + fa.traverseFilter(a => f(a).map(_.some)) <-> F.traverse.traverse(fa)(f) + } + + def traverseFilterComposition[A, B, C, M[_], N[_]](fa: F[A], + f: A => M[Option[B]], + g: B => N[Option[C]] + )(implicit + M: Applicative[M], + N: Applicative[N] + ): IsEq[Nested[M, N, F[C]]] = { + val lhs = Nested[M, N, F[C]](fa.traverseFilter(f).map(_.traverseFilter(g))) + val rhs: Nested[M, N, F[C]] = fa.traverseFilter[Nested[M, N, ?], C](a => + Nested[M, N, Option[C]](f(a).map(_.traverseFilter(g))) + ) + lhs <-> rhs + } + + def filterAConsistentWithTraverseFilter[G[_]: Applicative, A](fa: F[A], f: A => G[Boolean]): IsEq[G[F[A]]] = { + fa.filterA(f) <-> fa.traverseFilter(a => f(a).map(if (_) Some(a) else None)) + } +} + +object TraverseFilterLaws { + def apply[F[_]](implicit ev: TraverseFilter[F]): TraverseFilterLaws[F] = + new TraverseFilterLaws[F] { def F: TraverseFilter[F] = ev } +} diff --git a/laws/src/main/scala/cats/laws/discipline/FunctorFilterTests.scala b/laws/src/main/scala/cats/laws/discipline/FunctorFilterTests.scala new file mode 100644 index 0000000000..01fead48a2 --- /dev/null +++ b/laws/src/main/scala/cats/laws/discipline/FunctorFilterTests.scala @@ -0,0 +1,39 @@ +package cats +package laws +package discipline + +import org.scalacheck.Prop.forAll +import org.scalacheck.Arbitrary +import org.typelevel.discipline.Laws + +trait FunctorFilterTests[F[_]] extends Laws { + def laws: FunctorFilterLaws[F] + + def functorFilter[A, B, C](implicit + ArbFA: Arbitrary[F[A]], + ArbFABoo: Arbitrary[PartialFunction[A, B]], + ArbFOA: Arbitrary[F[Option[A]]], + ArbAOB: Arbitrary[A => Option[B]], + ArbBOC: Arbitrary[B => Option[C]], + ArbAB: Arbitrary[A => B], + ArbABoo: Arbitrary[A => Boolean], + EqFA: Eq[F[A]], + EqFB: Eq[F[B]], + EqFC: Eq[F[C]] + ): RuleSet = { + new DefaultRuleSet( + name = "functorFilter", + parent = None, + "mapFilter composition" -> forAll(laws.mapFilterComposition[A, B, C] _), + "mapFilter map consistency" -> forAll(laws.mapFilterMapConsistency[A, B] _), + "collect mapFilter consistency" -> forAll(laws.collectConsistentWithMapFilter[A, B] _), + "flattenOption mapFilter consistency" -> forAll(laws.flattenOptionConsistentWithMapFilter[A] _), + "filter mapFilter consistency" -> forAll(laws.filterConsistentWithMapFilter[A] _) + ) + } +} + +object FunctorFilterTests { + def apply[F[_]: FunctorFilter]: FunctorFilterTests[F] = + new FunctorFilterTests[F] { def laws: FunctorFilterLaws[F] = FunctorFilterLaws[F] } +} diff --git a/laws/src/main/scala/cats/laws/discipline/TraverseFilterTests.scala b/laws/src/main/scala/cats/laws/discipline/TraverseFilterTests.scala new file mode 100644 index 0000000000..77dc281eec --- /dev/null +++ b/laws/src/main/scala/cats/laws/discipline/TraverseFilterTests.scala @@ -0,0 +1,45 @@ +package cats +package laws +package discipline + +import cats.data.Nested +import org.scalacheck.Prop.forAll +import org.scalacheck.Arbitrary +import cats.instances.option._ + +trait TraverseFilterTests[F[_]] extends FunctorFilterTests[F] { + def laws: TraverseFilterLaws[F] + + def traverseFilter[A, B, C](implicit + ArbFA: Arbitrary[F[A]], + ArbFOA: Arbitrary[F[Option[A]]], + ArbFABoo: Arbitrary[PartialFunction[A, B]], + ArbAOB: Arbitrary[A => Option[B]], + ArbAOA: Arbitrary[A => Option[A]], + ArbAOOB: Arbitrary[A => Option[Option[B]]], + ArbBOC: Arbitrary[B => Option[C]], + ArbBOOC: Arbitrary[B => Option[Option[C]]], + ArbAB: Arbitrary[A => B], + ArbABoo: Arbitrary[A => Boolean], + ArbAOBoo: Arbitrary[A => Option[Boolean]], + EqFA: Eq[F[A]], + EqFB: Eq[F[B]], + EqFC: Eq[F[C]], + EqGFA: Eq[Option[F[A]]], + EqMNFC: Eq[Nested[Option, Option, F[C]]] + ): RuleSet = { + new DefaultRuleSet( + name = "traverseFilter", + parent = Some(functorFilter[A, B, C]), + "traverseFilter identity" -> forAll(laws.traverseFilterIdentity[Option, A] _), + "traverseFilter nested composition" -> forAll(laws.traverseFilterComposition[A, B, C, Option, Option] _), + "traverseFilter consistent with traverse" -> forAll(laws.traverseFilterConsistentWithTraverse[Option, A] _), + "filterA consistent with traverseFilter" -> forAll(laws.filterAConsistentWithTraverseFilter[Option, A] _) + ) + } +} + +object TraverseFilterTests { + def apply[F[_]: TraverseFilter]: TraverseFilterTests[F] = + new TraverseFilterTests[F] { def laws: TraverseFilterLaws[F] = TraverseFilterLaws[F] } +} diff --git a/tests/src/test/scala/cats/tests/ChainSuite.scala b/tests/src/test/scala/cats/tests/ChainSuite.scala index 0a4c23e4cd..b776354939 100644 --- a/tests/src/test/scala/cats/tests/ChainSuite.scala +++ b/tests/src/test/scala/cats/tests/ChainSuite.scala @@ -2,8 +2,8 @@ package cats package tests import cats.data.Chain +import cats.laws.discipline.{AlternativeTests, CoflatMapTests, MonadTests, SerializableTests, TraverseFilterTests, TraverseTests} import cats.kernel.laws.discipline.{EqTests, MonoidTests, OrderTests, PartialOrderTests} -import cats.laws.discipline.{AlternativeTests, CoflatMapTests, MonadTests, SerializableTests, TraverseTests} import cats.laws.discipline.arbitrary._ class ChainSuite extends CatsSuite { @@ -25,6 +25,10 @@ class ChainSuite extends CatsSuite { checkAll("Chain[Int]", OrderTests[Chain[Int]].order) checkAll("Order[Chain]", SerializableTests.serializable(Order[Chain[Int]])) + + checkAll("Chain[Int]", TraverseFilterTests[Chain].traverseFilter[Int, Int, Int]) + checkAll("TraverseFilter[Chain]", SerializableTests.serializable(TraverseFilter[Chain])) + { implicit val partialOrder = ListWrapper.partialOrder[Int] checkAll("Chain[ListWrapper[Int]]", @@ -41,6 +45,7 @@ class ChainSuite extends CatsSuite { SerializableTests.serializable(Eq[Chain[ListWrapper[Int]]])) } + test("show"){ Show[Chain[Int]].show(Chain(1, 2, 3)) should === ("Chain(1, 2, 3)") Chain.empty[Int].show should === ("Chain()") diff --git a/tests/src/test/scala/cats/tests/ConstSuite.scala b/tests/src/test/scala/cats/tests/ConstSuite.scala index 27d1e2c5d3..d233f91ac5 100644 --- a/tests/src/test/scala/cats/tests/ConstSuite.scala +++ b/tests/src/test/scala/cats/tests/ConstSuite.scala @@ -21,6 +21,9 @@ class ConstSuite extends CatsSuite { checkAll("Const[String, Int] with Option", TraverseTests[Const[String, ?]].traverse[Int, Int, Int, Int, Option, Option]) checkAll("Traverse[Const[String, ?]]", SerializableTests.serializable(Traverse[Const[String, ?]])) + checkAll("Const[String, Int]", TraverseFilterTests[Const[String, ?]].traverseFilter[Int, Int, Int]) + checkAll("TraverseFilter[Const[String, ?]]", SerializableTests.serializable(TraverseFilter[Const[String, ?]])) + // Get Apply[Const[C : Semigroup, ?]], not Applicative[Const[C : Monoid, ?]] { implicit def nonEmptyListSemigroup[A]: Semigroup[NonEmptyList[A]] = SemigroupK[NonEmptyList].algebra diff --git a/tests/src/test/scala/cats/tests/ListSuite.scala b/tests/src/test/scala/cats/tests/ListSuite.scala index a75de04879..1e8ed70e04 100644 --- a/tests/src/test/scala/cats/tests/ListSuite.scala +++ b/tests/src/test/scala/cats/tests/ListSuite.scala @@ -2,7 +2,7 @@ package cats package tests import cats.data.{NonEmptyList, ZipList} -import cats.laws.discipline.{CommutativeApplyTests, TraverseTests, CoflatMapTests, AlternativeTests, MonadTests, SerializableTests, SemigroupalTests} +import cats.laws.discipline.{AlternativeTests, CoflatMapTests, CommutativeApplyTests, MonadTests, SemigroupalTests, SerializableTests, TraverseFilterTests, TraverseTests} import cats.laws.discipline.arbitrary._ class ListSuite extends CatsSuite { @@ -22,6 +22,10 @@ class ListSuite extends CatsSuite { checkAll("List[Int]", MonadTests[List].monad[Int, Int, Int]) checkAll("Monad[List]", SerializableTests.serializable(Monad[List])) + checkAll("List[Int]", TraverseFilterTests[List].traverseFilter[Int, Int, Int]) + checkAll("TraverseFilter[List]", SerializableTests.serializable(TraverseFilter[List])) + + checkAll("ZipList[Int]", CommutativeApplyTests[ZipList].commutativeApply[Int, Int, Int]) test("nel => list => nel returns original nel")( diff --git a/tests/src/test/scala/cats/tests/ListWrapper.scala b/tests/src/test/scala/cats/tests/ListWrapper.scala index 212d9d61ce..2f081d7735 100644 --- a/tests/src/test/scala/cats/tests/ListWrapper.scala +++ b/tests/src/test/scala/cats/tests/ListWrapper.scala @@ -56,6 +56,26 @@ object ListWrapper { } } + val traverseFilter: TraverseFilter[ListWrapper] = { + val F = TraverseFilter[List] + + new TraverseFilter[ListWrapper] { + def traverse = ListWrapper.traverse + def traverseFilter[G[_], A, B](fa: ListWrapper[A])(f: A => G[Option[B]])(implicit G: Applicative[G]): G[ListWrapper[B]] = + G.map(F.traverseFilter(fa.list)(f))(ListWrapper.apply) + } + } + + val functorFilter: FunctorFilter[ListWrapper] = { + val F = FunctorFilter[List] + + new FunctorFilter[ListWrapper] { + def functor = ListWrapper.functor + def mapFilter[A, B](fa: ListWrapper[A])(f: A => Option[B]): ListWrapper[B] = + ListWrapper(F.mapFilter(fa.list)(f)) + } + } + val foldable: Foldable[ListWrapper] = traverse val functor: Functor[ListWrapper] = traverse diff --git a/tests/src/test/scala/cats/tests/MapSuite.scala b/tests/src/test/scala/cats/tests/MapSuite.scala index 53fdb73b75..ec9144ae17 100644 --- a/tests/src/test/scala/cats/tests/MapSuite.scala +++ b/tests/src/test/scala/cats/tests/MapSuite.scala @@ -1,8 +1,10 @@ package cats package tests + +import cats.laws.discipline.{FlatMapTests, FunctorFilterTests, SemigroupalTests, SerializableTests, UnorderedTraverseTests, ComposeTests} +import cats.laws.discipline.arbitrary._ import cats.arrow.Compose -import cats.laws.discipline.{FlatMapTests, SemigroupalTests, SerializableTests, UnorderedTraverseTests, ComposeTests} class MapSuite extends CatsSuite { implicit val iso = SemigroupalTests.Isomorphisms.invariant[Map[Int, ?]] @@ -16,10 +18,14 @@ class MapSuite extends CatsSuite { checkAll("Map[Int, Int] with Option", UnorderedTraverseTests[Map[Int, ?]].unorderedTraverse[Int, Int, Int, Option, Option]) checkAll("UnorderedTraverse[Map[Int, ?]]", SerializableTests.serializable(UnorderedTraverse[Map[Int, ?]])) + checkAll("Map[Int, Int]", FunctorFilterTests[Map[Int, ?]].functorFilter[Int, Int, Int]) + checkAll("FunctorFilter[Map[Int, ?]]", SerializableTests.serializable(FunctorFilter[Map[Int, ?]])) + checkAll("Map[Int, Long]", ComposeTests[Map].compose[Int, Long, String, Double]) checkAll("Compose[Map]", SerializableTests.serializable(Compose[Map])) + test("show isn't empty and is formatted as expected") { forAll { (map: Map[Int, String]) => map.show.nonEmpty should === (true) diff --git a/tests/src/test/scala/cats/tests/NestedSuite.scala b/tests/src/test/scala/cats/tests/NestedSuite.scala index df200b48a7..e5b2c25bd2 100644 --- a/tests/src/test/scala/cats/tests/NestedSuite.scala +++ b/tests/src/test/scala/cats/tests/NestedSuite.scala @@ -25,6 +25,26 @@ class NestedSuite extends CatsSuite { checkAll("Invariant[Nested[ListWrapper, ListWrapper, ?]]", SerializableTests.serializable(Invariant[Nested[ListWrapper, ListWrapper, ?]])) } + { + // FunctorFilter composition + implicit val instance = ListWrapper.functorFilter + implicit val functorInstance = ListWrapper.functor + checkAll("Nested[ListWrapper, ListWrapper]", + FunctorFilterTests[Nested[ListWrapper, ListWrapper, ?]].functorFilter[Int, Int, Int]) + checkAll("FunctorFilter[Nested[ListWrapper, ListWrapper, ?]]", + SerializableTests.serializable(FunctorFilter[Nested[ListWrapper, ListWrapper, ?]])) + } + + { + // TraverseFilter composition + implicit val instance = ListWrapper.traverseFilter + implicit val traverseInstance = ListWrapper.traverse + checkAll("Nested[ListWrapper, ListWrapper]", + TraverseFilterTests[Nested[ListWrapper, ListWrapper, ?]].traverseFilter[Int, Int, Int]) + checkAll("TraverseFilter[Nested[ListWrapper, ListWrapper, ?]]", + SerializableTests.serializable(TraverseFilter[Nested[ListWrapper, ListWrapper, ?]])) + } + { // Invariant + Covariant = Invariant val instance = Nested.catsDataInvariantForCovariantNested(ListWrapper.invariant, ListWrapper.functor) diff --git a/tests/src/test/scala/cats/tests/OptionSuite.scala b/tests/src/test/scala/cats/tests/OptionSuite.scala index a8f1dbc3aa..ae36f20129 100644 --- a/tests/src/test/scala/cats/tests/OptionSuite.scala +++ b/tests/src/test/scala/cats/tests/OptionSuite.scala @@ -3,6 +3,7 @@ package tests import cats.laws.{ApplicativeLaws, CoflatMapLaws, FlatMapLaws, MonadLaws} import cats.laws.discipline._ +import cats.laws.discipline.arbitrary._ class OptionSuite extends CatsSuite { checkAll("Option[Int]", SemigroupalTests[Option].semigroupal[Int, Int, Int]) @@ -20,6 +21,9 @@ class OptionSuite extends CatsSuite { checkAll("Option[Int] with Option", TraverseTests[Option].traverse[Int, Int, Int, Int, Option, Option]) checkAll("Traverse[Option]", SerializableTests.serializable(Traverse[Option])) + checkAll("Option[Int] with Option", TraverseFilterTests[Option].traverseFilter[Int, Int, Int]) + checkAll("TraverseFilter[Option]", SerializableTests.serializable(TraverseFilter[Option])) + checkAll("Option with Unit", MonadErrorTests[Option, Unit].monadError[Int, Int, Int]) checkAll("MonadError[Option, Unit]", SerializableTests.serializable(MonadError[Option, Unit])) diff --git a/tests/src/test/scala/cats/tests/OptionTSuite.scala b/tests/src/test/scala/cats/tests/OptionTSuite.scala index de4467088d..c8df8136dd 100644 --- a/tests/src/test/scala/cats/tests/OptionTSuite.scala +++ b/tests/src/test/scala/cats/tests/OptionTSuite.scala @@ -10,6 +10,31 @@ class OptionTSuite extends CatsSuite { implicit val iso = SemigroupalTests.Isomorphisms.invariant[OptionT[ListWrapper, ?]](OptionT.catsDataFunctorForOptionT(ListWrapper.functor)) checkAll("OptionT[Eval, ?]", DeferTests[OptionT[Eval, ?]].defer[Int]) + checkAll("OptionT[Eval, ?]", FunctorFilterTests[OptionT[Eval, ?]].functorFilter[Int, Int, Int]) + + + { + //If a Functor for F is defined + implicit val F = ListWrapper.functor + + checkAll("OptionT[ListWrapper, ?]", + FunctorFilterTests[OptionT[ListWrapper, ?]].functorFilter[Int, Int, Int]) + checkAll("FunctorFilter[OptionT[ListWrapper, ?]]", + SerializableTests.serializable(FunctorFilter[OptionT[ListWrapper, ?]])) + + } + + + { + //If a Traverse for F is defined + implicit val F = ListWrapper.traverse + + checkAll("OptionT[ListWrapper, ?]", + TraverseFilterTests[OptionT[ListWrapper, ?]].traverseFilter[Int, Int, Int]) + checkAll("TraverseFilter[OptionT[ListWrapper, ?]]", + SerializableTests.serializable(TraverseFilter[OptionT[ListWrapper, ?]])) + + } { implicit val F = ListWrapper.eqv[Option[Int]] diff --git a/tests/src/test/scala/cats/tests/SortedMapSuite.scala b/tests/src/test/scala/cats/tests/SortedMapSuite.scala index 33693c990b..5c4af5c6fd 100644 --- a/tests/src/test/scala/cats/tests/SortedMapSuite.scala +++ b/tests/src/test/scala/cats/tests/SortedMapSuite.scala @@ -2,8 +2,8 @@ package cats package tests import cats.kernel.CommutativeMonoid -import cats.kernel.laws.discipline.{HashTests, CommutativeMonoidTests, MonoidTests} -import cats.laws.discipline.{FlatMapTests, SemigroupalTests, SerializableTests, TraverseTests} +import cats.kernel.laws.discipline.{CommutativeMonoidTests, HashTests, MonoidTests} +import cats.laws.discipline.{FlatMapTests, SemigroupalTests, SerializableTests, TraverseFilterTests, TraverseTests} import cats.laws.discipline.arbitrary._ import scala.collection.immutable.SortedMap @@ -20,6 +20,9 @@ class SortedMapSuite extends CatsSuite { checkAll("SortedMap[Int, Int] with Option", TraverseTests[SortedMap[Int, ?]].traverse[Int, Int, Int, Int, Option, Option]) checkAll("Traverse[SortedMap[Int, ?]]", SerializableTests.serializable(Traverse[SortedMap[Int, ?]])) + checkAll("SortedMap[Int, Int]", TraverseFilterTests[SortedMap[Int, ?]].traverseFilter[Int, Int, Int]) + checkAll("TraverseFilter[SortedMap[Int, ?]]", SerializableTests.serializable(TraverseFilter[SortedMap[Int, ?]])) + test("show isn't empty and is formatted as expected") { forAll { (map: SortedMap[Int, String]) => map.show.nonEmpty should === (true) diff --git a/tests/src/test/scala/cats/tests/StreamSuite.scala b/tests/src/test/scala/cats/tests/StreamSuite.scala index bfc4cb51ef..25f7c1bb4d 100644 --- a/tests/src/test/scala/cats/tests/StreamSuite.scala +++ b/tests/src/test/scala/cats/tests/StreamSuite.scala @@ -1,9 +1,8 @@ package cats package tests -import cats.laws.discipline.{CommutativeApplyTests, CoflatMapTests, MonadTests, AlternativeTests, SerializableTests, TraverseTests, SemigroupalTests} +import cats.laws.discipline.{AlternativeTests, CoflatMapTests, CommutativeApplyTests, MonadTests, SemigroupalTests, SerializableTests, TraverseFilterTests, TraverseTests} import cats.data.ZipStream - import cats.laws.discipline.arbitrary._ class StreamSuite extends CatsSuite { @@ -22,6 +21,9 @@ class StreamSuite extends CatsSuite { checkAll("Stream[Int] with Option", TraverseTests[Stream].traverse[Int, Int, Int, Set[Int], Option, Option]) checkAll("Traverse[Stream]", SerializableTests.serializable(Traverse[Stream])) + checkAll("Stream[Int]", TraverseFilterTests[Stream].traverseFilter[Int, Int, Int]) + checkAll("TraverseFilter[Stream]", SerializableTests.serializable(TraverseFilter[Stream])) + // Can't test applicative laws as they don't terminate checkAll("ZipStream[Int]", CommutativeApplyTests[ZipStream].apply[Int, Int, Int]) diff --git a/tests/src/test/scala/cats/tests/VectorSuite.scala b/tests/src/test/scala/cats/tests/VectorSuite.scala index 2ae3ce5f6a..1d9562f8fa 100644 --- a/tests/src/test/scala/cats/tests/VectorSuite.scala +++ b/tests/src/test/scala/cats/tests/VectorSuite.scala @@ -2,7 +2,7 @@ package cats package tests import cats.data.{NonEmptyVector, ZipVector} -import cats.laws.discipline.{CommutativeApplyTests, AlternativeTests, CoflatMapTests, MonadTests, SerializableTests, TraverseTests, SemigroupalTests} +import cats.laws.discipline.{AlternativeTests, CoflatMapTests, CommutativeApplyTests, MonadTests, SemigroupalTests, SerializableTests, TraverseFilterTests, TraverseTests} import cats.laws.discipline.arbitrary._ class VectorSuite extends CatsSuite { @@ -21,6 +21,9 @@ class VectorSuite extends CatsSuite { checkAll("Vector[Int]", MonadTests[Vector].monad[Int, Int, Int]) checkAll("Monad[Vector]", SerializableTests.serializable(Monad[Vector])) + checkAll("Vector[Int]", TraverseFilterTests[Vector].traverseFilter[Int, Int, Int]) + checkAll("TraverseFilter[Vector]", SerializableTests.serializable(TraverseFilter[Vector])) + checkAll("ZipVector[Int]", CommutativeApplyTests[ZipVector].commutativeApply[Int, Int, Int]) test("show") {