From 39fbd284e091d767482eda4bb0652aee6e81f2ad Mon Sep 17 00:00:00 2001 From: David Gregory Date: Fri, 11 Mar 2016 08:12:54 +0000 Subject: [PATCH] Adding more contravariant instances --- core/src/main/scala/cats/data/Cokleisli.scala | 7 +++- core/src/main/scala/cats/data/Func.scala | 13 ++++++++ core/src/main/scala/cats/data/Prod.scala | 12 +++++++ core/src/main/scala/cats/data/WriterT.scala | 18 +++++++++- .../scala/cats/functor/Contravariant.scala | 20 +++++++++++ .../cats/laws/discipline/Arbitrary.scala | 13 ++++++++ .../main/scala/cats/laws/discipline/Eq.scala | 33 +++++++++++++++++++ .../scala/cats/tests/CokleisliTests.scala | 5 ++- .../scala/cats/tests/CoproductTests.scala | 12 +------ .../src/test/scala/cats/tests/FuncTests.scala | 7 ++++ .../cats/tests/KernelContravariantTests.scala | 15 +++++++++ .../src/test/scala/cats/tests/ProdTests.scala | 5 +++ .../test/scala/cats/tests/WriterTTests.scala | 7 +++- 13 files changed, 152 insertions(+), 15 deletions(-) create mode 100644 tests/src/test/scala/cats/tests/KernelContravariantTests.scala diff --git a/core/src/main/scala/cats/data/Cokleisli.scala b/core/src/main/scala/cats/data/Cokleisli.scala index 6a9f5bcdd3..464b636bfb 100644 --- a/core/src/main/scala/cats/data/Cokleisli.scala +++ b/core/src/main/scala/cats/data/Cokleisli.scala @@ -2,7 +2,7 @@ package cats package data import cats.arrow.{Arrow, Split} -import cats.functor.Profunctor +import cats.functor.{Contravariant, Profunctor} import cats.{CoflatMap, Comonad, Functor, Monad} /** @@ -71,6 +71,11 @@ private[data] sealed abstract class CokleisliInstances0 { implicit def catsDataSemigroupKForCokleisli[F[_]](implicit ev: CoflatMap[F]): SemigroupK[λ[α => Cokleisli[F, α, α]]] = new CokleisliSemigroupK[F] { def F: CoflatMap[F] = ev } + + implicit def catsDataContravariantForCokleisli[F[_]: Functor, A]: Contravariant[Cokleisli[F, ?, A]] = + new Contravariant[Cokleisli[F, ?, A]] { + def contramap[B, C](fbc: Cokleisli[F, B, A])(f: C => B): Cokleisli[F, C, A] = fbc.lmap(f) + } } private trait CokleisliArrow[F[_]] extends Arrow[Cokleisli[F, ?, ?]] with CokleisliSplit[F] with CokleisliProfunctor[F] { diff --git a/core/src/main/scala/cats/data/Func.scala b/core/src/main/scala/cats/data/Func.scala index de64c0f941..f8c7e530e0 100644 --- a/core/src/main/scala/cats/data/Func.scala +++ b/core/src/main/scala/cats/data/Func.scala @@ -1,6 +1,8 @@ package cats package data +import cats.functor.Contravariant + /** * [[Func]] is a function `A => F[B]`. * @@ -50,6 +52,11 @@ private[data] abstract class FuncInstances1 { new FuncFunctor[F, C] { def F: Functor[F] = FF } + + implicit def catsDataContravariantForFunc[F[_], C](implicit FC: Contravariant[F]): Contravariant[λ[α => Func[F, α, C]]] = + new FuncContravariant[F, C] { + def F: Contravariant[F] = FC + } } sealed trait FuncFunctor[F[_], C] extends Functor[λ[α => Func[F, C, α]]] { @@ -58,6 +65,12 @@ sealed trait FuncFunctor[F[_], C] extends Functor[λ[α => Func[F, C, α]]] { fa.map(f)(F) } +sealed trait FuncContravariant[F[_], C] extends Contravariant[λ[α => Func[F, α, C]]] { + def F: Contravariant[F] + def contramap[A, B](fa: Func[F, A, C])(f: B => A): Func[F, B, C] = + Func.func(a => fa.run(f(a))) +} + sealed trait FuncApply[F[_], C] extends Apply[λ[α => Func[F, C, α]]] with FuncFunctor[F, C] { def F: Apply[F] def ap[A, B](f: Func[F, C, A => B])(fa: Func[F, C, A]): Func[F, C, B] = diff --git a/core/src/main/scala/cats/data/Prod.scala b/core/src/main/scala/cats/data/Prod.scala index 74f3139086..c7a0aaaa82 100644 --- a/core/src/main/scala/cats/data/Prod.scala +++ b/core/src/main/scala/cats/data/Prod.scala @@ -1,6 +1,8 @@ package cats package data +import cats.functor.Contravariant + /** * [[Prod]] is a product to two independent functor values. * @@ -55,6 +57,10 @@ private[data] sealed abstract class ProdInstances4 { def F: Functor[F] = FF def G: Functor[G] = GG } + implicit def catsDataContravariantForProd[F[_], G[_]](implicit FC: Contravariant[F], GC: Contravariant[G]): Contravariant[λ[α => Prod[F, G, α]]] = new ProdContravariant[F, G] { + def F: Contravariant[F] = FC + def G: Contravariant[G] = GC + } } sealed trait ProdFunctor[F[_], G[_]] extends Functor[λ[α => Prod[F, G, α]]] { @@ -63,6 +69,12 @@ sealed trait ProdFunctor[F[_], G[_]] extends Functor[λ[α => Prod[F, G, α]]] { override def map[A, B](fa: Prod[F, G, A])(f: A => B): Prod[F, G, B] = Prod(F.map(fa.first)(f), G.map(fa.second)(f)) } +sealed trait ProdContravariant[F[_], G[_]] extends Contravariant[λ[α => Prod[F, G, α]]] { + def F: Contravariant[F] + def G: Contravariant[G] + def contramap[A, B](fa: Prod[F, G, A])(f: B => A): Prod[F, G, B] = Prod(F.contramap(fa.first)(f), G.contramap(fa.second)(f)) +} + sealed trait ProdApply[F[_], G[_]] extends Apply[λ[α => Prod[F, G, α]]] with ProdFunctor[F, G] { def F: Apply[F] def G: Apply[G] diff --git a/core/src/main/scala/cats/data/WriterT.scala b/core/src/main/scala/cats/data/WriterT.scala index 9df6c5013a..37e7adddda 100644 --- a/core/src/main/scala/cats/data/WriterT.scala +++ b/core/src/main/scala/cats/data/WriterT.scala @@ -2,7 +2,7 @@ package cats package data import cats.kernel.std.tuple._ -import cats.functor.Bifunctor +import cats.functor.{Bifunctor, Contravariant} final case class WriterT[F[_], L, V](run: F[(L, V)]) { def written(implicit functorF: Functor[F]): F[L] = @@ -22,6 +22,11 @@ final case class WriterT[F[_], L, V](run: F[(L, V)]) { functorF.map(run) { z => (z._1, fn(z._2)) } } + def contramap[Z](fn: Z => V)(implicit F: Contravariant[F]): WriterT[F, L, Z] = + WriterT { + F.contramap(run) { z => (z._1, fn(z._2)) } + } + def flatMap[U](f: V => WriterT[F, L, U])(implicit flatMapF: FlatMap[F], semigroupL: Semigroup[L]): WriterT[F, L, U] = WriterT { flatMapF.flatMap(run) { lv => @@ -176,6 +181,10 @@ private[data] sealed abstract class WriterTInstances7 { new WriterTCoflatMap[F, L] { implicit val F0: Functor[F] = F } + + implicit def catsDataContravariantForWriterT[F[_], L](implicit F: Contravariant[F]): Contravariant[WriterT[F, L, ?]] = new WriterTContravariant[F, L] { + implicit val F0: Contravariant[F] = F + } } private[data] sealed trait WriterTFunctor[F[_], L] extends Functor[WriterT[F, L, ?]] { @@ -185,6 +194,13 @@ private[data] sealed trait WriterTFunctor[F[_], L] extends Functor[WriterT[F, L, fa.map(f) } +private[data] sealed trait WriterTContravariant[F[_], L] extends Contravariant[WriterT[F, L, ?]] { + implicit def F0: Contravariant[F] + + def contramap[A, B](fa: WriterT[F, L, A])(f: B => A): WriterT[F, L, B] = + fa.contramap(f) +} + private[data] sealed trait WriterTApply[F[_], L] extends WriterTFunctor[F, L] with Apply[WriterT[F, L, ?]] { override implicit def F0: Apply[F] implicit def L0: Semigroup[L] diff --git a/core/src/main/scala/cats/functor/Contravariant.scala b/core/src/main/scala/cats/functor/Contravariant.scala index cf231e38a2..59329b3b58 100644 --- a/core/src/main/scala/cats/functor/Contravariant.scala +++ b/core/src/main/scala/cats/functor/Contravariant.scala @@ -22,3 +22,23 @@ import simulacrum.typeclass val G = Functor[G] } } + +object Contravariant { + implicit val catsFunctorContravariantForPartialOrder: Contravariant[PartialOrder] = + new Contravariant[PartialOrder] { + /** Derive a `PartialOrder` for `B` given a `PartialOrder[A]` and a function `B => A`. + * + * Note: resulting instances are law-abiding only when the functions used are injective (represent a one-to-one mapping) + */ + def contramap[A, B](fa: PartialOrder[A])(f: B => A): PartialOrder[B] = fa.on(f) + } + + implicit val catsFunctorContravariantForOrder: Contravariant[Order] = + new Contravariant[Order] { + /** Derive an `Order` for `B` given an `Order[A]` and a function `B => A`. + * + * Note: resulting instances are law-abiding only when the functions used are injective (represent a one-to-one mapping) + */ + def contramap[A, B](fa: Order[A])(f: B => A): Order[B] = fa.on(f) + } +} diff --git a/laws/src/main/scala/cats/laws/discipline/Arbitrary.scala b/laws/src/main/scala/cats/laws/discipline/Arbitrary.scala index 7e823c6a06..e874bfc8d4 100644 --- a/laws/src/main/scala/cats/laws/discipline/Arbitrary.scala +++ b/laws/src/main/scala/cats/laws/discipline/Arbitrary.scala @@ -77,6 +77,19 @@ object arbitrary extends ArbitraryInstances0 { implicit def showArbitrary[A: Arbitrary]: Arbitrary[Show[A]] = Arbitrary(Show.fromToString[A]) + implicit def partialOrderArbitrary[A: Arbitrary]: Arbitrary[PartialOrder[A]] = + Arbitrary(Gen.oneOf( + PartialOrder.from[A]((_: A, _: A) => Double.NaN), + PartialOrder.from[A]((_: A, _: A) => -1.0), + PartialOrder.from[A]((_: A, _: A) => 0.0), + PartialOrder.from[A]((_: A, _: A) => 1.0))) + + implicit def orderArbitrary[A: Arbitrary]: Arbitrary[Order[A]] = + Arbitrary(Gen.oneOf( + Order.from[A]((_: A, _: A) => -1), + Order.from[A]((_: A, _: A) => 0), + Order.from[A]((_: A, _: A) => 1))) + implicit def function0Arbitrary[A: Arbitrary]: Arbitrary[() => A] = Arbitrary(getArbitrary[A].map(() => _)) diff --git a/laws/src/main/scala/cats/laws/discipline/Eq.scala b/laws/src/main/scala/cats/laws/discipline/Eq.scala index 432b1d7a80..ece842cc52 100644 --- a/laws/src/main/scala/cats/laws/discipline/Eq.scala +++ b/laws/src/main/scala/cats/laws/discipline/Eq.scala @@ -24,6 +24,39 @@ object eq { } } + /** + * Create an approximation of Eq[PartialOrder[A]] by generating 100 values for A + * and comparing the application of the two compare functions + */ + implicit def partialOrderEq[A](implicit arbA: Arbitrary[(A, A)], optIntEq: Eq[Option[Int]]): Eq[PartialOrder[A]] = new Eq[PartialOrder[A]] { + def eqv(f: PartialOrder[A], g: PartialOrder[A]): Boolean = { + val samples = List.fill(100)(arbA.arbitrary.sample).collect { + case Some(a) => a + case None => sys.error("Could not generate arbitrary values to compare two PartialOrder[A]") + } + samples.forall { + case (l, r) => optIntEq.eqv(f.tryCompare(l, r), g.tryCompare(l, r)) + } + } + } + + /** + * Create an approximation of Eq[Order[A]] by generating 100 values for A + * and comparing the application of the two compare functions + */ + implicit def orderEq[A](implicit arbA: Arbitrary[(A, A)], intEq: Eq[Int]): Eq[Order[A]] = new Eq[Order[A]] { + def eqv(f: Order[A], g: Order[A]): Boolean = { + val samples = List.fill(100)(arbA.arbitrary.sample).collect { + case Some(a) => a + case None => sys.error("Could not generate arbitrary values to compare two Order[A]") + } + samples.forall { + case (l, r) => intEq.eqv(f.compare(l, r), g.compare(l, r)) + } + } + } + + /** Create an approximation of Eq[Show[A]] by using function1Eq[A, String] */ implicit def showEq[A: Arbitrary]: Eq[Show[A]] = { val xyz = function1Eq[A, String] diff --git a/tests/src/test/scala/cats/tests/CokleisliTests.scala b/tests/src/test/scala/cats/tests/CokleisliTests.scala index 212de1f56b..a2781129e7 100644 --- a/tests/src/test/scala/cats/tests/CokleisliTests.scala +++ b/tests/src/test/scala/cats/tests/CokleisliTests.scala @@ -3,7 +3,7 @@ package tests import cats.arrow.{Arrow, Split} import cats.data.{Cokleisli, NonEmptyList} -import cats.functor.Profunctor +import cats.functor.{Contravariant, Profunctor} import cats.laws.discipline._ import cats.laws.discipline.arbitrary._ import cats.laws.discipline.eq._ @@ -32,6 +32,9 @@ class CokleisliTests extends SlowCatsSuite { checkAll("Cokleisli[Option, Int, Int]", SplitTests[Cokleisli[Option, ?, ?]].split[Int, Int, Int, Int, Int, Int]) checkAll("Split[Cokleisli[Option, ?, ?]", SerializableTests.serializable(Split[Cokleisli[Option, ?, ?]])) + checkAll("Cokleisli[Option, Int, Int]", ContravariantTests[Cokleisli[Option, ?, Int]].contravariant[Int, Int, Int]) + checkAll("Contravariant[Cokleisli[Option, ?, Int]]", SerializableTests.serializable(Contravariant[Cokleisli[Option, ?, Int]])) + { // Ceremony to help scalac to do the right thing, see also #267. type CokleisliNEL[A, B] = Cokleisli[NonEmptyList, A, B] diff --git a/tests/src/test/scala/cats/tests/CoproductTests.scala b/tests/src/test/scala/cats/tests/CoproductTests.scala index ff6a40e324..dc2e4c4d55 100644 --- a/tests/src/test/scala/cats/tests/CoproductTests.scala +++ b/tests/src/test/scala/cats/tests/CoproductTests.scala @@ -6,7 +6,7 @@ import cats.data.Coproduct import cats.functor.Contravariant import cats.laws.discipline._ import cats.laws.discipline.arbitrary._ -import org.scalacheck.Arbitrary +import cats.laws.discipline.eq._ class CoproductTests extends CatsSuite { @@ -31,16 +31,6 @@ class CoproductTests extends CatsSuite { checkAll("Coproduct[Option, Option, Int]", OrderLaws[Coproduct[Option, Option, Int]].eqv) checkAll("Eq[Coproduct[Option, Option, Int]]", SerializableTests.serializable(Eq[Coproduct[Option, Option, Int]])) - implicit def showEq[A](implicit arbA: Arbitrary[A], stringEq: Eq[String]): Eq[Show[A]] = new Eq[Show[A]] { - def eqv(f: Show[A], g: Show[A]): Boolean = { - val samples = List.fill(100)(arbA.arbitrary.sample).collect { - case Some(a) => a - case None => sys.error("Could not generate arbitrary values to compare two Show[A]") - } - samples.forall(s => stringEq.eqv(f.show(s), g.show(s))) - } - } - checkAll("Coproduct[Show, Show, ?]", ContravariantTests[Coproduct[Show, Show, ?]].contravariant[Int, Int, Int]) checkAll("Contravariant[Coproduct[Show, Show, ?]]", SerializableTests.serializable(Contravariant[Coproduct[Show, Show, ?]])) diff --git a/tests/src/test/scala/cats/tests/FuncTests.scala b/tests/src/test/scala/cats/tests/FuncTests.scala index fe1216e03a..1f779bd51f 100644 --- a/tests/src/test/scala/cats/tests/FuncTests.scala +++ b/tests/src/test/scala/cats/tests/FuncTests.scala @@ -2,6 +2,7 @@ package cats package tests import cats.data.{ Func, AppFunc } +import cats.functor.Contravariant import Func.appFunc import cats.laws.discipline._ import cats.laws.discipline.arbitrary._ @@ -37,6 +38,12 @@ class FuncTests extends CatsSuite { checkAll("Functor[Func[Option, Int, ?]]", SerializableTests.serializable(Functor[Func[Option, Int, ?]])) } + { + implicit val funcContravariant = Func.catsDataContravariantForFunc[Show, Int] + checkAll("Func[Show, Int, Int]", ContravariantTests[Func[Show, ?, Int]].contravariant[Int, Int, Int]) + checkAll("Contravariant[Func[Show, ?, Int]]", SerializableTests.serializable(Contravariant[Func[Show, ?, Int]])) + } + { implicit val appFuncApp = AppFunc.appFuncApplicative[Option, Int] implicit val iso = CartesianTests.Isomorphisms.invariant[AppFunc[Option, Int, ?]] diff --git a/tests/src/test/scala/cats/tests/KernelContravariantTests.scala b/tests/src/test/scala/cats/tests/KernelContravariantTests.scala new file mode 100644 index 0000000000..92be09409e --- /dev/null +++ b/tests/src/test/scala/cats/tests/KernelContravariantTests.scala @@ -0,0 +1,15 @@ +package cats +package tests + +import cats.functor.Contravariant +import cats.laws.discipline.arbitrary._ +import cats.laws.discipline.{ContravariantTests, SerializableTests} +import cats.laws.discipline.eq._ + +class KernelContravariantTests extends CatsSuite { + checkAll("Contravariant[PartialOrder]", ContravariantTests[PartialOrder].contravariant[Int, Int, Int]) + checkAll("Contravariant[PartialOrder]", SerializableTests.serializable(Contravariant[PartialOrder])) + + checkAll("Contravariant[Order]", ContravariantTests[Order].contravariant[Int, Int, Int]) + checkAll("Contravariant[Order]", SerializableTests.serializable(Contravariant[Order])) +} diff --git a/tests/src/test/scala/cats/tests/ProdTests.scala b/tests/src/test/scala/cats/tests/ProdTests.scala index d6e02e9d84..1d04e7bea4 100644 --- a/tests/src/test/scala/cats/tests/ProdTests.scala +++ b/tests/src/test/scala/cats/tests/ProdTests.scala @@ -2,8 +2,10 @@ package cats package tests import cats.data.Prod +import cats.functor.Contravariant import cats.laws.discipline._ import cats.laws.discipline.arbitrary._ +import cats.laws.discipline.eq._ class ProdTests extends CatsSuite { implicit val iso = CartesianTests.Isomorphisms.invariant[Prod[Option, List, ?]] @@ -13,6 +15,9 @@ class ProdTests extends CatsSuite { checkAll("Prod[Option, List, Int]", AlternativeTests[λ[α => Prod[Option, List, α]]].alternative[Int, Int, Int]) checkAll("Alternative[Prod[Option, List, Int]]", SerializableTests.serializable(Alternative[λ[α => Prod[Option, List, α]]])) + checkAll("Prod[Show, Order, Int]", ContravariantTests[λ[α => Prod[Show, Order, α]]].contravariant[Int, Int, Int]) + checkAll("Contravariant[Prod[Show, Order, Int]]", SerializableTests.serializable(Contravariant[λ[α => Prod[Show, Order, α]]])) + { implicit val monoidK = ListWrapper.monoidK checkAll("Prod[ListWrapper, ListWrapper, ?]", MonoidKTests[Prod[ListWrapper, ListWrapper, ?]].monoidK[Int]) diff --git a/tests/src/test/scala/cats/tests/WriterTTests.scala b/tests/src/test/scala/cats/tests/WriterTTests.scala index e567248e73..b3f7630403 100644 --- a/tests/src/test/scala/cats/tests/WriterTTests.scala +++ b/tests/src/test/scala/cats/tests/WriterTTests.scala @@ -2,9 +2,10 @@ package cats package tests import cats.data.{Writer, WriterT} -import cats.functor.Bifunctor +import cats.functor.{Bifunctor, Contravariant} import cats.laws.discipline._ import cats.laws.discipline.arbitrary._ +import cats.laws.discipline.eq._ import cats.kernel.laws.OrderLaws @@ -19,6 +20,10 @@ class WriterTTests extends CatsSuite { checkAll("WriterT[List, Int, Int]", OrderLaws[WriterT[List, Int, Int]].eqv) checkAll("Eq[WriterT[List, Int, Int]]", SerializableTests.serializable(Eq[WriterT[List, Int, Int]])) + + checkAll("WriterT[Show, Int, Int]", ContravariantTests[WriterT[Show, Int, ?]].contravariant[Int, Int, Int]) + checkAll("Contravariant[WriterT[Show, Int, Int]]", SerializableTests.serializable(Contravariant[WriterT[Show, Int, ?]])) + // check that this resolves Eq[Writer[Int, Int]]