From 2d90c65241704b84639112067a4e2e5246845c30 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Piotr=20Paradzi=C5=84ski?= Date: Sun, 25 Nov 2018 23:21:28 +0100 Subject: [PATCH 1/4] add Strong Laws based --- .../src/main/scala/cats/laws/StrongLaws.scala | 51 +++++++++++++++++++ 1 file changed, 51 insertions(+) diff --git a/laws/src/main/scala/cats/laws/StrongLaws.scala b/laws/src/main/scala/cats/laws/StrongLaws.scala index 77a2a1bb3e..83f638d50c 100644 --- a/laws/src/main/scala/cats/laws/StrongLaws.scala +++ b/laws/src/main/scala/cats/laws/StrongLaws.scala @@ -21,6 +21,57 @@ trait StrongLaws[F[_, _]] extends ProfunctorLaws[F] { 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]) + + private def swapTuple[X, Y]: Tuple2[X, Y] => Tuple2[Y, X] = _.swap + + /** first' == dimap swap swap . second' */ + def firstIsSwappedSecond[A, B, C](fab: F[A, B]): IsEq[F[(A, C), (B, C)]] = + fab.first[C] <-> fab.second[C].dimap(swapTuple[A, C])(swapTuple[C, B]) + + /** second' == dimap swap swap . first' */ + def secondIsSwappedFirst[A, B, C](fab: F[A, B]): IsEq[F[(C, A), (C, B)]] = + fab.second[C] <-> fab.first[C].dimap(swapTuple[C, A])(swapTuple[B, C]) + + /** lmap fst == rmap fst . first' */ + def lmapEqualsFirstAndThenRmap[A, B, C](fab: F[A, B]): IsEq[F[(A, C), B]] = + fab.lmap[(A, C)]({ case (a, _) => a }) <-> fab.first[C].rmap[B](_._1) + + /** lmap snd == rmap snd . second' */ + def lmapEqualsSecondAndThenRmap[A, B, C](fab: F[A, B]): IsEq[F[(C, A), B]] = + fab.lmap[(C, A)]({ case (_, b) => b }) <-> fab.second[C].rmap[B](_._2) + + /** lmap (second f) . first == rmap (second f) . first */ + def dinaturalityFirst[A, B, C, D](fab: F[A, B], + f: C => D)(implicit PF: Strong[Function1]): IsEq[F[(A, C), (B, D)]] = { + val idbf: ((B, C)) => (B, D) = PF.second(f) + val idaf: ((A, C)) => (A, D) = PF.second(f) + fab.first[C].rmap(idbf) <-> fab.first[D].lmap(idaf) + } + + /** lmap (first f) . second == rmap (first f) . second */ + def dinaturalitySecond[A, B, C, D](fab: F[A, B], + f: C => D)(implicit PF: Strong[Function1]): IsEq[F[(C, A), (D, B)]] = { + val idbf: ((C, B)) => (D, B) = PF.first(f) + val idaf: ((C, A)) => (D, A) = PF.first(f) + fab.second[C].rmap(idbf) <-> fab.second[D].lmap(idaf) + } + + private def assoc[A, B, C]: (((A, B), C)) => (A, (B, C)) = { case ((a, c), d) => (a, (c, d)) } + private def unassoc[A, B, C]: ((A, (B, C))) => ((A, B), C) = { case (a, (c, d)) => ((a, c), d) } + + /** first' . first' == dimap assoc unassoc . first' where + * assoc ((a,b),c) = (a,(b,c)) + * unassoc (a,(b,c)) = ((a,b),c) + */ + def firstFirstIsDimap[A, B, C, D](fab: F[A, B]): IsEq[F[((A, C), D), ((B, C), D)]] = + fab.first[C].first[D] <-> fab.first[(C, D)].dimap[((A, C), D), ((B, C), D)](assoc)(unassoc) + + /** second' . second' == dimap unassoc assoc . second' where + * assoc ((a,b),c) = (a,(b,c)) + * unassoc (a,(b,c)) = ((a,b),c) + */ + def secondSecondIsDimap[A, B, C, D](fab: F[A, B]): IsEq[F[(D, (C, A)), (D, (C, B))]] = + fab.second[C].second[D] <-> fab.second[(D, C)].dimap[(D, (C, A)), (D, (C, B))](unassoc)(assoc) } object StrongLaws { From bae4ccd0f653d1e3deb4bed651587c9bf1e86000 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Piotr=20Paradzi=C5=84ski?= Date: Sun, 25 Nov 2018 23:41:17 +0100 Subject: [PATCH 2/4] add reference to Notions of Compuations as monoids in StrongLaws --- laws/src/main/scala/cats/laws/StrongLaws.scala | 2 ++ 1 file changed, 2 insertions(+) diff --git a/laws/src/main/scala/cats/laws/StrongLaws.scala b/laws/src/main/scala/cats/laws/StrongLaws.scala index 83f638d50c..6cb84a20b3 100644 --- a/laws/src/main/scala/cats/laws/StrongLaws.scala +++ b/laws/src/main/scala/cats/laws/StrongLaws.scala @@ -8,6 +8,8 @@ import cats.instances.function._ /** * Laws that must be obeyed by any `cats.functor.Strong`. + * + * See: [[https://arxiv.org/abs/1406.4823 E. Rivas, M. Jaskelioff Notions of Computation as Monoids, Chapter 7]] */ trait StrongLaws[F[_, _]] extends ProfunctorLaws[F] { implicit override def F: Strong[F] From 0b133d6ffcbda203373f81217b98f3b7ecbd6b17 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Piotr=20Paradzi=C5=84ski?= Date: Mon, 26 Nov 2018 00:32:46 +0100 Subject: [PATCH 3/4] use new laws in Arrow, CommutativeArrow, ArrowChoice test --- .../src/main/scala/cats/laws/StrongLaws.scala | 25 +++++-------------- .../laws/discipline/ArrowChoiceTests.scala | 7 ++++-- .../cats/laws/discipline/ArrowTests.scala | 9 ++++--- .../discipline/CommutativeArrowTests.scala | 9 ++++--- .../cats/laws/discipline/StrongTests.scala | 21 ++++++++++++---- 5 files changed, 39 insertions(+), 32 deletions(-) diff --git a/laws/src/main/scala/cats/laws/StrongLaws.scala b/laws/src/main/scala/cats/laws/StrongLaws.scala index 6cb84a20b3..d4dce9dc85 100644 --- a/laws/src/main/scala/cats/laws/StrongLaws.scala +++ b/laws/src/main/scala/cats/laws/StrongLaws.scala @@ -4,7 +4,6 @@ package laws import cats.arrow.Strong import cats.syntax.profunctor._ import cats.syntax.strong._ -import cats.instances.function._ /** * Laws that must be obeyed by any `cats.functor.Strong`. @@ -14,16 +13,6 @@ import cats.instances.function._ 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]) - private def swapTuple[X, Y]: Tuple2[X, Y] => Tuple2[Y, X] = _.swap /** first' == dimap swap swap . second' */ @@ -43,18 +32,16 @@ trait StrongLaws[F[_, _]] extends ProfunctorLaws[F] { fab.lmap[(C, A)]({ case (_, b) => b }) <-> fab.second[C].rmap[B](_._2) /** lmap (second f) . first == rmap (second f) . first */ - def dinaturalityFirst[A, B, C, D](fab: F[A, B], - f: C => D)(implicit PF: Strong[Function1]): IsEq[F[(A, C), (B, D)]] = { - val idbf: ((B, C)) => (B, D) = PF.second(f) - val idaf: ((A, C)) => (A, D) = PF.second(f) + def dinaturalityFirst[A, B, C, D](fab: F[A, B], f: C => D): IsEq[F[(A, C), (B, D)]] = { + val idbf: ((B, C)) => (B, D) = { case (b, c) => (b, f(c)) } + val idaf: ((A, C)) => (A, D) = { case (a, c) => (a, f(c)) } fab.first[C].rmap(idbf) <-> fab.first[D].lmap(idaf) } /** lmap (first f) . second == rmap (first f) . second */ - def dinaturalitySecond[A, B, C, D](fab: F[A, B], - f: C => D)(implicit PF: Strong[Function1]): IsEq[F[(C, A), (D, B)]] = { - val idbf: ((C, B)) => (D, B) = PF.first(f) - val idaf: ((C, A)) => (D, A) = PF.first(f) + def dinaturalitySecond[A, B, C, D](fab: F[A, B], f: C => D): IsEq[F[(C, A), (D, B)]] = { + val idbf: ((C, B)) => (D, B) = { case (c, b) => (f(c), b) } + val idaf: ((C, A)) => (D, A) = { case (c, a) => (f(c), a) } fab.second[C].rmap(idbf) <-> fab.second[D].lmap(idaf) } diff --git a/laws/src/main/scala/cats/laws/discipline/ArrowChoiceTests.scala b/laws/src/main/scala/cats/laws/discipline/ArrowChoiceTests.scala index 234ef9889a..f1e23137f1 100644 --- a/laws/src/main/scala/cats/laws/discipline/ArrowChoiceTests.scala +++ b/laws/src/main/scala/cats/laws/discipline/ArrowChoiceTests.scala @@ -34,10 +34,13 @@ trait ArrowChoiceTests[F[_, _]] extends ArrowTests[F] with ChoiceTests[F] { EqFACBD: Eq[F[(A, C), (B, D)]], EqFADCD: Eq[F[(A, D), (C, D)]], EqFADCG: Eq[F[(A, D), (C, G)]], - EqFAEDE: Eq[F[(A, E), (D, E)]], + EqFDADB: Eq[F[(D, A), (D, B)]], + EqFCADB: Eq[F[(C, A), (D, B)]], EqFABC: Eq[F[A, (B, C)]], - EqFEAED: Eq[F[(E, A), (E, D)]], EqFACDBCD: Eq[F[((A, C), D), (B, (C, D))]], + EqFACDBCD2: Eq[F[((A, C), D), ((B, C), D)]], + EqFDCADCB: Eq[F[(D, (C, A)), (D, (C, B))]], + EqFCAB: Eq[F[(C, A), B]], EqFEitherABD: Eq[F[Either[A, B], D]], EqFEitherCoABC: Eq[F[A, Either[B, C]]], REqFEitherACD: Eq[F[Either[A, D], Either[C, D]]], diff --git a/laws/src/main/scala/cats/laws/discipline/ArrowTests.scala b/laws/src/main/scala/cats/laws/discipline/ArrowTests.scala index 26db8bac4b..a627c45b29 100644 --- a/laws/src/main/scala/cats/laws/discipline/ArrowTests.scala +++ b/laws/src/main/scala/cats/laws/discipline/ArrowTests.scala @@ -32,10 +32,13 @@ trait ArrowTests[F[_, _]] extends CategoryTests[F] with StrongTests[F] { EqFACBD: Eq[F[(A, C), (B, D)]], EqFADCD: Eq[F[(A, D), (C, D)]], EqFADCG: Eq[F[(A, D), (C, G)]], - EqFAEDE: Eq[F[(A, E), (D, E)]], + EqFDADB: Eq[F[(D, A), (D, B)]], + EqFCADB: Eq[F[(C, A), (D, B)]], EqFABC: Eq[F[A, (B, C)]], - EqFEAED: Eq[F[(E, A), (E, D)]], - EqFACDBCD: Eq[F[((A, C), D), (B, (C, D))]] + EqFCAB: Eq[F[(C, A), B]], + EqFACDBCD: Eq[F[((A, C), D), (B, (C, D))]], + EqFACDBCD2: Eq[F[((A, C), D), ((B, C), D)]], + EqFDCADCB: Eq[F[(D, (C, A)), (D, (C, B))]] ): RuleSet = new RuleSet { def name: String = "arrow" diff --git a/laws/src/main/scala/cats/laws/discipline/CommutativeArrowTests.scala b/laws/src/main/scala/cats/laws/discipline/CommutativeArrowTests.scala index 0185d7ff8b..53695eee37 100644 --- a/laws/src/main/scala/cats/laws/discipline/CommutativeArrowTests.scala +++ b/laws/src/main/scala/cats/laws/discipline/CommutativeArrowTests.scala @@ -33,9 +33,12 @@ trait CommutativeArrowTests[F[_, _]] extends ArrowTests[F] { EqFACBD: Eq[F[(A, C), (B, D)]], EqFADCD: Eq[F[(A, D), (C, D)]], EqFADCG: Eq[F[(A, D), (C, G)]], - EqFAEDE: Eq[F[(A, E), (D, E)]], - EqFEAED: Eq[F[(E, A), (E, D)]], - EqFACDBCD: Eq[F[((A, C), D), (B, (C, D))]] + EqFDADB: Eq[F[(D, A), (D, B)]], + EqFCADB: Eq[F[(C, A), (D, B)]], + EqFACDBCD: Eq[F[((A, C), D), (B, (C, D))]], + EqFACDBCD2: Eq[F[((A, C), D), ((B, C), D)]], + EqFDCADCB: Eq[F[(D, (C, A)), (D, (C, B))]], + EqFCAB: Eq[F[(C, A), B]] ): RuleSet = new DefaultRuleSet(name = "commutative arrow", parent = Some(arrow[A, B, C, D, E, G]), diff --git a/laws/src/main/scala/cats/laws/discipline/StrongTests.scala b/laws/src/main/scala/cats/laws/discipline/StrongTests.scala index d6d4c7fa02..7c4ff859c1 100644 --- a/laws/src/main/scala/cats/laws/discipline/StrongTests.scala +++ b/laws/src/main/scala/cats/laws/discipline/StrongTests.scala @@ -12,7 +12,6 @@ trait StrongTests[F[_, _]] extends ProfunctorTests[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]], CogenA: Cogen[A], CogenB: Cogen[B], @@ -22,14 +21,26 @@ trait StrongTests[F[_, _]] extends ProfunctorTests[F] { EqFAB: Eq[F[A, B]], EqFAD: Eq[F[A, D]], EqFAG: Eq[F[A, G]], - EqFAEDE: Eq[F[(A, E), (D, E)]], - EqFEAED: Eq[F[(E, A), (E, D)]] + EqFACBC: Eq[F[(A, C), (B, C)]], + EqFDADB: Eq[F[(D, A), (D, B)]], + EqFACBD: Eq[F[(A, C), (B, D)]], + EqFCADB: Eq[F[(C, A), (D, B)]], + EqFACB: Eq[F[(A, C), B]], + EqFCAB: Eq[F[(C, A), B]], + EqFACDBCD: Eq[F[((A, C), D), ((B, C), D)]], + EqFDCADCB: Eq[F[(D, (C, A)), (D, (C, B))]] ): RuleSet = new DefaultRuleSet( name = "strong", parent = Some(profunctor[A, B, C, D, E, G]), - "strong first distributivity" -> forAll(laws.strongFirstDistributivity[A, B, C, D, E] _), - "strong second distributivity" -> forAll(laws.strongSecondDistributivity[A, B, C, D, E] _) + "first is swapped second" -> forAll(laws.firstIsSwappedSecond[A, B, C] _), + "second is swapped first" -> forAll(laws.secondIsSwappedFirst[A, B, D] _), + "lmap equals first and then rmap" -> forAll(laws.lmapEqualsFirstAndThenRmap[A, B, C] _), + "lmap equals second and then rmap" -> forAll(laws.lmapEqualsSecondAndThenRmap[A, B, C] _), + "dinaturality of first" -> forAll(laws.dinaturalityFirst[A, B, C, D] _), + "dinaturality of second" -> forAll(laws.dinaturalitySecond[A, B, C, D] _), + "first first is dimap" -> forAll(laws.firstFirstIsDimap[A, B, C, D] _), + "second second is dimap" -> forAll(laws.secondSecondIsDimap[A, B, C, D] _) ) } From ee6f12878c1048222dbb5d65309508c5c2db3b91 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Piotr=20Paradzi=C5=84ski?= Date: Mon, 26 Nov 2018 09:30:08 +0100 Subject: [PATCH 4/4] add link to Haskell impl, refactor common parts --- laws/src/main/scala/cats/laws/StrongLaws.scala | 18 ++++++++---------- 1 file changed, 8 insertions(+), 10 deletions(-) diff --git a/laws/src/main/scala/cats/laws/StrongLaws.scala b/laws/src/main/scala/cats/laws/StrongLaws.scala index d4dce9dc85..82fb89750f 100644 --- a/laws/src/main/scala/cats/laws/StrongLaws.scala +++ b/laws/src/main/scala/cats/laws/StrongLaws.scala @@ -9,6 +9,7 @@ import cats.syntax.strong._ * Laws that must be obeyed by any `cats.functor.Strong`. * * See: [[https://arxiv.org/abs/1406.4823 E. Rivas, M. Jaskelioff Notions of Computation as Monoids, Chapter 7]] + * See: [[http://hackage.haskell.org/package/profunctors/docs/Data-Profunctor-Strong.html Haskell Data.Profunctor.Strong]] */ trait StrongLaws[F[_, _]] extends ProfunctorLaws[F] { implicit override def F: Strong[F] @@ -31,19 +32,16 @@ trait StrongLaws[F[_, _]] extends ProfunctorLaws[F] { def lmapEqualsSecondAndThenRmap[A, B, C](fab: F[A, B]): IsEq[F[(C, A), B]] = fab.lmap[(C, A)]({ case (_, b) => b }) <-> fab.second[C].rmap[B](_._2) + private def mapFirst[X, Y, Z](f: X => Z)(cb: (X, Y)): (Z, Y) = (f(cb._1), cb._2) + private def mapSecond[X, Y, Z](f: Y => Z)(cb: (X, Y)): (X, Z) = (cb._1, f(cb._2)) + /** lmap (second f) . first == rmap (second f) . first */ - def dinaturalityFirst[A, B, C, D](fab: F[A, B], f: C => D): IsEq[F[(A, C), (B, D)]] = { - val idbf: ((B, C)) => (B, D) = { case (b, c) => (b, f(c)) } - val idaf: ((A, C)) => (A, D) = { case (a, c) => (a, f(c)) } - fab.first[C].rmap(idbf) <-> fab.first[D].lmap(idaf) - } + def dinaturalityFirst[A, B, C, D](fab: F[A, B], f: C => D): IsEq[F[(A, C), (B, D)]] = + fab.first[C].rmap(mapSecond(f)) <-> fab.first[D].lmap(mapSecond(f)) /** lmap (first f) . second == rmap (first f) . second */ - def dinaturalitySecond[A, B, C, D](fab: F[A, B], f: C => D): IsEq[F[(C, A), (D, B)]] = { - val idbf: ((C, B)) => (D, B) = { case (c, b) => (f(c), b) } - val idaf: ((C, A)) => (D, A) = { case (c, a) => (f(c), a) } - fab.second[C].rmap(idbf) <-> fab.second[D].lmap(idaf) - } + def dinaturalitySecond[A, B, C, D](fab: F[A, B], f: C => D): IsEq[F[(C, A), (D, B)]] = + fab.second[C].rmap(mapFirst(f)) <-> fab.second[D].lmap(mapFirst(f)) private def assoc[A, B, C]: (((A, B), C)) => (A, (B, C)) = { case ((a, c), d) => (a, (c, d)) } private def unassoc[A, B, C]: ((A, (B, C))) => ((A, B), C) = { case (a, (c, d)) => ((a, c), d) }