From 24dc3c3385c9a7d6ee0906a16800cfc26d8a13d8 Mon Sep 17 00:00:00 2001 From: "Frank S. Thomas" Date: Tue, 24 Feb 2015 22:01:52 +0100 Subject: [PATCH] Add StrongLaws and StrongTests --- build.sbt | 2 +- core/src/main/scala/cats/functor/Strong.scala | 3 ++ .../src/main/scala/cats/laws/StrongLaws.scala | 24 +++++++++++++ .../cats/laws/discipline/SplitTests.scala | 2 +- .../cats/laws/discipline/StrongTests.scala | 35 +++++++++++++++++++ .../test/scala/cats/tests/FunctionTests.scala | 2 +- .../test/scala/cats/tests/KleisliTests.scala | 2 +- 7 files changed, 66 insertions(+), 4 deletions(-) create mode 100644 laws/src/main/scala/cats/laws/StrongLaws.scala create mode 100644 laws/src/main/scala/cats/laws/discipline/StrongTests.scala diff --git a/build.sbt b/build.sbt index f683f04a9e..c6f1a00e86 100644 --- a/build.sbt +++ b/build.sbt @@ -97,7 +97,7 @@ lazy val core = project.dependsOn(macros) sourceGenerators in Compile <+= (sourceManaged in Compile).map(Boilerplate.gen) ) -lazy val laws = project.dependsOn(macros, core, data) +lazy val laws = project.dependsOn(macros, core, data, std) .settings(moduleName := "cats-laws") .settings(catsSettings: _*) .settings( diff --git a/core/src/main/scala/cats/functor/Strong.scala b/core/src/main/scala/cats/functor/Strong.scala index d617f81a32..209f72d6ef 100644 --- a/core/src/main/scala/cats/functor/Strong.scala +++ b/core/src/main/scala/cats/functor/Strong.scala @@ -1,6 +1,9 @@ package cats package functor +/** + * Must obey the laws defined in [[laws.StrongLaws]]. + */ trait Strong[F[_, _]] extends Profunctor[F] { def first[A, B, C](fa: F[A, B]): F[(A, C), (B, C)] def second[A, B, C](fa: F[A, B]): F[(C, A), (C, B)] diff --git a/laws/src/main/scala/cats/laws/StrongLaws.scala b/laws/src/main/scala/cats/laws/StrongLaws.scala new file mode 100644 index 0000000000..ad2513b79c --- /dev/null +++ b/laws/src/main/scala/cats/laws/StrongLaws.scala @@ -0,0 +1,24 @@ +package cats.laws + +import cats.functor.Strong +import cats.syntax.profunctor._ +import cats.syntax.strong._ +import cats.std.function._ + +/** + * Laws that must be obeyed by any [[cats.functor.Strong]]. + */ +trait StrongLaws[F[_, _]] extends ProfunctorLaws[F] { + implicit override def F: Strong[F] + + def strongFirstDistributivity[A0, A1, B1, B2, C](fab: F[A1, B1], f: A0 => A1, g: B1 => B2): IsEq[F[(A0, C), (B2, C)]] = + fab.dimap(f)(g).first[C] <-> fab.first[C].dimap(f.first[C])(g.first[C]) + + def strongSecondDistributivity[A0, A1, B1, B2, C](fab: F[A1, B1], f: A0 => A1, g: B1 => B2): IsEq[F[(C, A0), (C, B2)]] = + fab.dimap(f)(g).second[C] <-> fab.second[C].dimap(f.second[C])(g.second[C]) +} + +object StrongLaws { + def apply[F[_, _]](implicit ev: Strong[F]): StrongLaws[F] = + new StrongLaws[F] { def F = ev } +} diff --git a/laws/src/main/scala/cats/laws/discipline/SplitTests.scala b/laws/src/main/scala/cats/laws/discipline/SplitTests.scala index 86f9bdb6b4..7a74e36120 100644 --- a/laws/src/main/scala/cats/laws/discipline/SplitTests.scala +++ b/laws/src/main/scala/cats/laws/discipline/SplitTests.scala @@ -16,7 +16,7 @@ trait SplitTests[F[_, _]] extends ComposeTests[F] { ArbFDE: Arbitrary[F[D, E]], ArbFEG: Arbitrary[F[E, G]], EqFAD: Eq[F[A, D]], - EqFADCG: Eq[F[(A, D),(C, G)]] + EqFADCG: Eq[F[(A, D), (C, G)]] ): RuleSet = new RuleSet { def name = "split" diff --git a/laws/src/main/scala/cats/laws/discipline/StrongTests.scala b/laws/src/main/scala/cats/laws/discipline/StrongTests.scala new file mode 100644 index 0000000000..3854c0b322 --- /dev/null +++ b/laws/src/main/scala/cats/laws/discipline/StrongTests.scala @@ -0,0 +1,35 @@ +package cats.laws +package discipline + +import cats.Eq +import cats.functor.Strong +import org.scalacheck.Arbitrary +import org.scalacheck.Prop._ + +trait StrongTests[F[_, _]] extends ProfunctorTests[F] { + def laws: StrongLaws[F] + + def strong[A: Arbitrary, B: Arbitrary, C: Arbitrary, D: Arbitrary, E: Arbitrary, G: Arbitrary](implicit + ArbFAB: Arbitrary[F[A, B]], + ArbFBC: Arbitrary[F[B, C]], + ArbFCD: Arbitrary[F[C, D]], + EqFAB: Eq[F[A, B]], + EqFAG: Eq[F[A, G]], + EqFAEDE: Eq[F[(A, E), (D, E)]], + EqFEAED: Eq[F[(E, A), (E, D)]] + ): RuleSet = + new RuleSet { + def name = "strong" + def bases = Nil + def parents = Seq(profunctor[A, B, C, D, E, G]) + def props = Seq( + "strong first distributivity" -> forAll(laws.strongFirstDistributivity[A, B, C, D, E] _), + "strong second distributivity" -> forAll(laws.strongSecondDistributivity[A, B, C, D, E] _) + ) + } +} + +object StrongTests { + def apply[F[_, _]: Strong]: StrongTests[F] = + new StrongTests[F] { def laws = StrongLaws[F] } +} diff --git a/tests/src/test/scala/cats/tests/FunctionTests.scala b/tests/src/test/scala/cats/tests/FunctionTests.scala index 4d18cdb793..54ad67d001 100644 --- a/tests/src/test/scala/cats/tests/FunctionTests.scala +++ b/tests/src/test/scala/cats/tests/FunctionTests.scala @@ -8,5 +8,5 @@ class FunctionTests extends CatsSuite { checkAll("Function0[Int]", MonadTests[Function0].monad[Int, Int, Int]) checkAll("Function1[Int, Int]", CategoryTests[Function1].category[Int, Int, Int, Int]) checkAll("Function1[Int, Int]", SplitTests[Function1].split[Int, Int, Int, Int, Int, Int]) - checkAll("Function1[Int, Int]", ProfunctorTests[Function1].profunctor[Int, Int, Int, Int, Int, Int]) + checkAll("Function1[Int, Int]", StrongTests[Function1].strong[Int, Int, Int, Int, Int, Int]) } diff --git a/tests/src/test/scala/cats/tests/KleisliTests.scala b/tests/src/test/scala/cats/tests/KleisliTests.scala index 3509936ae2..7707878c3f 100644 --- a/tests/src/test/scala/cats/tests/KleisliTests.scala +++ b/tests/src/test/scala/cats/tests/KleisliTests.scala @@ -13,5 +13,5 @@ class KleisliTests extends CatsSuite { Eq.by[Kleisli[F, A, B], A => F[B]](_.run) checkAll("Kleisli[Option, Int, Int]", ApplicativeTests[Kleisli[Option, Int, ?]].applicative[Int, Int, Int]) - checkAll("Kleisli[Option, Int, Int]", ProfunctorTests[Kleisli[Option, ?, ?]].profunctor[Int, Int, Int, Int, Int, Int]) + checkAll("Kleisli[Option, Int, Int]", StrongTests[Kleisli[Option, ?, ?]].strong[Int, Int, Int, Int, Int, Int]) }