diff --git a/README.md b/README.md index 1de70133f7..b0f2051e47 100644 --- a/README.md +++ b/README.md @@ -117,6 +117,7 @@ functionality, you can pick-and-choose from amongst these modules * `cats-laws`: Laws for testing type class instances. * `cats-free`: Free structures such as the free monad, and supporting type classes. * `cats-testkit`: lib for writing tests for type class instances using laws. + * `algebra`: Type classes to represent algebraic structures. * `alleycats-core`: Cats instances and classes which are not lawful. There are several other Cats modules that are in separate repos so that they can @@ -140,10 +141,11 @@ Links: 2. ScalaDoc: [typelevel.org/cats/api/](https://typelevel.org/cats/api/) 3. Type classes: [typelevel.org/cats/typeclasses](https://typelevel.org/cats/typeclasses.html) 4. Data types: [typelevel.org/cats/datatypes.html](https://typelevel.org/cats/datatypes.html) -5. Glossary: [typelevel.org/cats/nomenclature.html](https://typelevel.org/cats/nomenclature.html) -6. Resources for Learners: [typelevel.org/cats/resources_for_learners.html](https://typelevel.org/cats/resources_for_learners.html) -7. FAQ: [typelevel.org/cats/faq.html](https://typelevel.org/cats/faq.html) -8. The Typelevel Ecosystem: [typelevel.org/cats/typelevelEcosystem.html](https://typelevel.org/cats/typelevelEcosystem.html) +5. Algebra overview: [typelevel.org/cats/algebra.html](https://typelevel.org/cats/algebra.html) +6. Glossary: [typelevel.org/cats/nomenclature.html](https://typelevel.org/cats/nomenclature.html) +7. Resources for Learners: [typelevel.org/cats/resources_for_learners.html](https://typelevel.org/cats/resources_for_learners.html) +8. FAQ: [typelevel.org/cats/faq.html](https://typelevel.org/cats/faq.html) +9. The Typelevel Ecosystem: [typelevel.org/cats/typelevelEcosystem.html](https://typelevel.org/cats/typelevelEcosystem.html) ### Community diff --git a/algebra-core/src/main/scala/algebra/Priority.scala b/algebra-core/src/main/scala/algebra/Priority.scala new file mode 100644 index 0000000000..1c5f930273 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/Priority.scala @@ -0,0 +1,69 @@ +package algebra + +import scala.annotation.nowarn + +/** + * Priority is a type class for prioritized implicit search. + * + * This type class will attempt to provide an implicit instance of `P` + * (the preferred type). If that type is not available it will + * fallback to `F` (the fallback type). If neither type is available + * then a `Priority[P, F]` instance will not be available. + * + * This type can be useful for problems where multiple algorithms can + * be used, depending on the type classes available. + */ +sealed trait Priority[+P, +F] { + + import Priority.{Fallback, Preferred} + + def fold[B](f1: P => B)(f2: F => B): B = + this match { + case Preferred(x) => f1(x) + case Fallback(y) => f2(y) + } + + def join[U >: P with F]: U = + fold(_.asInstanceOf[U])(_.asInstanceOf[U]) + + def bimap[P2, F2](f1: P => P2)(f2: F => F2): Priority[P2, F2] = + this match { + case Preferred(x) => Preferred(f1(x)) + case Fallback(y) => Fallback(f2(y)) + } + + def toEither: Either[P, F] = + fold[Either[P, F]](p => Left(p))(f => Right(f)) + + def isPreferred: Boolean = + fold(_ => true)(_ => false) + + def isFallback: Boolean = + fold(_ => false)(_ => true) + + def getPreferred: Option[P] = + fold[Option[P]](p => Some(p))(_ => None) + + def getFallback: Option[F] = + fold[Option[F]](_ => None)(f => Some(f)) +} + +object Priority extends FindPreferred { + + case class Preferred[P](get: P) extends Priority[P, Nothing] + case class Fallback[F](get: F) extends Priority[Nothing, F] + + def apply[P, F](implicit ev: Priority[P, F]): Priority[P, F] = ev +} + +private[algebra] trait FindPreferred extends FindFallback { + @nowarn("msg=deprecated") + implicit def preferred[P](implicit ev: P): Priority[P, Nothing] = + Priority.Preferred(ev) +} + +private[algebra] trait FindFallback { + @nowarn("msg=deprecated") + implicit def fallback[F](implicit ev: F): Priority[Nothing, F] = + Priority.Fallback(ev) +} diff --git a/algebra-core/src/main/scala/algebra/instances/StaticMethods.scala b/algebra-core/src/main/scala/algebra/instances/StaticMethods.scala new file mode 100644 index 0000000000..02eb30d555 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/StaticMethods.scala @@ -0,0 +1,27 @@ +package algebra.instances + +import scala.annotation.tailrec + +object StaticMethods { + + /** + * Exponentiation function, e.g. x^y + * + * If base^ex doesn't fit in a Long, the result will overflow (unlike + * Math.pow which will return +/- Infinity). + */ + final def pow(base: Long, exponent: Long): Long = { + @tailrec def loop(t: Long, b: Long, e: Long): Long = + if (e == 0L) t + else if ((e & 1) == 1) loop(t * b, b * b, e >>> 1L) + else loop(t, b * b, e >>> 1L) + + if (exponent >= 0L) loop(1L, base, exponent) + else { + if (base == 0L) throw new ArithmeticException("zero can't be raised to negative power") + else if (base == 1L) 1L + else if (base == -1L) if ((exponent & 1L) == 0L) -1L else 1L + else 0L + } + } +} diff --git a/algebra-core/src/main/scala/algebra/instances/all.scala b/algebra-core/src/main/scala/algebra/instances/all.scala new file mode 100644 index 0000000000..be730a79cf --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/all.scala @@ -0,0 +1,25 @@ +package algebra +package instances + +package object all extends AllInstances + +trait AllInstances + extends ArrayInstances + with BigDecimalInstances + with BigIntInstances + with BitSetInstances + with BooleanInstances + with ByteInstances + with CharInstances + with DoubleInstances + with FloatInstances + with IntInstances + with ListInstances + with LongInstances + with MapInstances + with OptionInstances + with SetInstances + with ShortInstances + with StringInstances + with TupleInstances + with UnitInstances diff --git a/algebra-core/src/main/scala/algebra/instances/array.scala b/algebra-core/src/main/scala/algebra/instances/array.scala new file mode 100644 index 0000000000..2e4a27226a --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/array.scala @@ -0,0 +1,70 @@ +package algebra +package instances + +import scala.{specialized => sp} + +package object array extends ArrayInstances + +trait ArrayInstances { + implicit def arrayEq[@sp A: Eq]: Eq[Array[A]] = + new ArrayEq[A] + implicit def arrayOrder[@sp A: Order]: Order[Array[A]] = + new ArrayOrder[A] + implicit def arrayPartialOrder[@sp A: PartialOrder]: PartialOrder[Array[A]] = + new ArrayPartialOrder[A] +} + +private object ArraySupport { + + private def signum(x: Int): Int = + if (x < 0) -1 + else if (x > 0) 1 + else 0 + + def eqv[@sp A](x: Array[A], y: Array[A])(implicit ev: Eq[A]): Boolean = { + var i = 0 + if (x.length != y.length) return false + while (i < x.length && i < y.length && ev.eqv(x(i), y(i))) i += 1 + i == x.length + } + + def compare[@sp A](x: Array[A], y: Array[A])(implicit ev: Order[A]): Int = { + var i = 0 + while (i < x.length && i < y.length) { + val cmp = ev.compare(x(i), y(i)) + if (cmp != 0) return cmp + i += 1 + } + signum(x.length - y.length) + } + + def partialCompare[@sp A](x: Array[A], y: Array[A])(implicit ev: PartialOrder[A]): Double = { + var i = 0 + while (i < x.length && i < y.length) { + val cmp = ev.partialCompare(x(i), y(i)) + // Double.NaN is also != 0.0 + if (cmp != 0.0) return cmp + i += 1 + } + signum(x.length - y.length).toDouble + } +} + +final private class ArrayEq[@sp A: Eq] extends Eq[Array[A]] with Serializable { + def eqv(x: Array[A], y: Array[A]): Boolean = + ArraySupport.eqv(x, y) +} + +final private class ArrayOrder[@sp A: Order] extends Order[Array[A]] with Serializable { + override def eqv(x: Array[A], y: Array[A]): Boolean = + ArraySupport.eqv(x, y) + def compare(x: Array[A], y: Array[A]): Int = + ArraySupport.compare(x, y) +} + +final private class ArrayPartialOrder[@sp A: PartialOrder] extends PartialOrder[Array[A]] with Serializable { + override def eqv(x: Array[A], y: Array[A]): Boolean = + ArraySupport.eqv(x, y) + override def partialCompare(x: Array[A], y: Array[A]): Double = + ArraySupport.partialCompare(x, y) +} diff --git a/algebra-core/src/main/scala/algebra/instances/bigDecimal.scala b/algebra-core/src/main/scala/algebra/instances/bigDecimal.scala new file mode 100644 index 0000000000..89d7a6e3f2 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/bigDecimal.scala @@ -0,0 +1,31 @@ +package algebra +package instances + +import java.math.MathContext + +import algebra.ring._ + +package object bigDecimal extends BigDecimalInstances + +trait BigDecimalInstances extends cats.kernel.instances.BigDecimalInstances { + implicit val bigDecimalAlgebra: BigDecimalAlgebra = new BigDecimalAlgebra() +} + +class BigDecimalAlgebra(mc: MathContext) extends Field[BigDecimal] with Serializable { + def this() = this(MathContext.UNLIMITED) + + val zero: BigDecimal = BigDecimal(0, mc) + val one: BigDecimal = BigDecimal(1, mc) + + def plus(a: BigDecimal, b: BigDecimal): BigDecimal = a + b + def negate(a: BigDecimal): BigDecimal = -a + override def minus(a: BigDecimal, b: BigDecimal): BigDecimal = a - b + + def times(a: BigDecimal, b: BigDecimal): BigDecimal = a * b + def div(a: BigDecimal, b: BigDecimal): BigDecimal = a / b + + override def pow(a: BigDecimal, k: Int): BigDecimal = a.pow(k) + + override def fromInt(n: Int): BigDecimal = BigDecimal(n, mc) + override def fromBigInt(n: BigInt): BigDecimal = BigDecimal(n, mc) +} diff --git a/algebra-core/src/main/scala/algebra/instances/bigInt.scala b/algebra-core/src/main/scala/algebra/instances/bigInt.scala new file mode 100644 index 0000000000..3c5696319b --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/bigInt.scala @@ -0,0 +1,63 @@ +package algebra +package instances + +import algebra.ring._ + +package object bigInt extends BigIntInstances + +trait BigIntInstances extends cats.kernel.instances.BigIntInstances { + implicit val bigIntAlgebra: BigIntAlgebra = new BigIntTruncatedDivison + implicit def bigIntTruncatedDivision: TruncatedDivision[BigInt] = + bigIntAlgebra.asInstanceOf[BigIntTruncatedDivison] // Bin-compat hack to avoid allocation +} + +class BigIntAlgebra extends EuclideanRing[BigInt] with Serializable { + + val zero: BigInt = BigInt(0) + val one: BigInt = BigInt(1) + + def plus(a: BigInt, b: BigInt): BigInt = a + b + def negate(a: BigInt): BigInt = -a + override def minus(a: BigInt, b: BigInt): BigInt = a - b + + def times(a: BigInt, b: BigInt): BigInt = a * b + + override def pow(a: BigInt, k: Int): BigInt = a.pow(k) + + override def fromInt(n: Int): BigInt = BigInt(n) + override def fromBigInt(n: BigInt): BigInt = n + + override def lcm(a: BigInt, b: BigInt)(implicit ev: Eq[BigInt]): BigInt = + if (a.signum == 0 || b.signum == 0) zero else (a / a.gcd(b)) * b + override def gcd(a: BigInt, b: BigInt)(implicit ev: Eq[BigInt]): BigInt = a.gcd(b) + + def euclideanFunction(a: BigInt): BigInt = a.abs + + override def equotmod(a: BigInt, b: BigInt): (BigInt, BigInt) = { + val (qt, rt) = a /% b // truncated quotient and remainder + if (rt.signum >= 0) (qt, rt) + else if (b.signum > 0) (qt - 1, rt + b) + else (qt + 1, rt - b) + } + + def equot(a: BigInt, b: BigInt): BigInt = { + val (qt, rt) = a /% b // truncated quotient and remainder + if (rt.signum >= 0) qt + else if (b.signum > 0) qt - 1 + else qt + 1 + } + + def emod(a: BigInt, b: BigInt): BigInt = { + val rt = a % b // truncated remainder + if (rt.signum >= 0) rt + else if (b > 0) rt + b + else rt - b + } + +} + +class BigIntTruncatedDivison extends BigIntAlgebra with TruncatedDivision.forCommutativeRing[BigInt] { + override def tquot(x: BigInt, y: BigInt): BigInt = x / y + override def tmod(x: BigInt, y: BigInt): BigInt = x % y + override def order: Order[BigInt] = cats.kernel.instances.bigInt.catsKernelStdOrderForBigInt +} diff --git a/algebra-core/src/main/scala/algebra/instances/bitSet.scala b/algebra-core/src/main/scala/algebra/instances/bitSet.scala new file mode 100644 index 0000000000..a14e6a441b --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/bitSet.scala @@ -0,0 +1,21 @@ +package algebra +package instances + +import scala.collection.immutable.BitSet + +import algebra.lattice._ + +package object bitSet extends BitSetInstances + +trait BitSetInstances extends cats.kernel.instances.BitSetInstances { + implicit val bitSetAlgebra: BitSetAlgebra = + new BitSetAlgebra +} + +class BitSetAlgebra extends GenBool[BitSet] with Serializable { + val zero: BitSet = BitSet.empty + def and(a: BitSet, b: BitSet): BitSet = a & b + def or(a: BitSet, b: BitSet): BitSet = a | b + def without(a: BitSet, b: BitSet): BitSet = a -- b + override def xor(a: BitSet, b: BitSet): BitSet = a ^ b +} diff --git a/algebra-core/src/main/scala/algebra/instances/boolean.scala b/algebra-core/src/main/scala/algebra/instances/boolean.scala new file mode 100644 index 0000000000..f4bf0e295a --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/boolean.scala @@ -0,0 +1,42 @@ +package algebra +package instances + +import algebra.lattice.Bool +import algebra.ring.BoolRing +import algebra.ring.CommutativeRig + +package object boolean extends BooleanInstances + +trait BooleanInstances extends cats.kernel.instances.BooleanInstances { + implicit val booleanAlgebra: BooleanAlgebra = + new BooleanAlgebra + + val booleanRing = new BoolRing[Boolean] { + def zero: Boolean = false + def one: Boolean = true + def plus(x: Boolean, y: Boolean): Boolean = x ^ y + def times(x: Boolean, y: Boolean): Boolean = x && y + } +} + +/** + * This commutative rig is different than the one obtained from GF(2). + * + * It uses || for plus, and && for times. + */ +class BooleanAlgebra extends Bool[Boolean] with CommutativeRig[Boolean] { + + def zero: Boolean = false + def one: Boolean = true + + override def isZero(x: Boolean)(implicit ev: Eq[Boolean]): Boolean = !x + override def isOne(x: Boolean)(implicit ev: Eq[Boolean]): Boolean = x + + def and(x: Boolean, y: Boolean): Boolean = x && y + def or(x: Boolean, y: Boolean): Boolean = x || y + def complement(x: Boolean): Boolean = !x + + def plus(a: Boolean, b: Boolean): Boolean = a || b + override def pow(a: Boolean, b: Int): Boolean = a + override def times(a: Boolean, b: Boolean): Boolean = a && b +} diff --git a/algebra-core/src/main/scala/algebra/instances/byte.scala b/algebra-core/src/main/scala/algebra/instances/byte.scala new file mode 100644 index 0000000000..4e3197fe67 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/byte.scala @@ -0,0 +1,32 @@ +package algebra +package instances + +import algebra.lattice._ +import algebra.ring._ + +package object byte extends ByteInstances + +trait ByteInstances extends cats.kernel.instances.ByteInstances { + implicit val byteAlgebra: ByteAlgebra = new ByteAlgebra + + val ByteMinMaxLattice: BoundedDistributiveLattice[Byte] = + BoundedDistributiveLattice.minMax[Byte](Byte.MinValue, Byte.MaxValue) +} + +class ByteAlgebra extends CommutativeRing[Byte] with Serializable { + + def zero: Byte = 0 + def one: Byte = 1 + + def plus(x: Byte, y: Byte): Byte = (x + y).toByte + def negate(x: Byte): Byte = (-x).toByte + override def minus(x: Byte, y: Byte): Byte = (x - y).toByte + + def times(x: Byte, y: Byte): Byte = (x * y).toByte + + override def pow(x: Byte, y: Int): Byte = + Math.pow(x.toDouble, y.toDouble).toByte + + override def fromInt(n: Int): Byte = n.toByte + override def fromBigInt(n: BigInt): Byte = n.toByte +} diff --git a/algebra-core/src/main/scala/algebra/instances/char.scala b/algebra-core/src/main/scala/algebra/instances/char.scala new file mode 100644 index 0000000000..25701780a3 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/char.scala @@ -0,0 +1,6 @@ +package algebra +package instances + +package object char extends CharInstances + +trait CharInstances extends cats.kernel.instances.CharInstances diff --git a/algebra-core/src/main/scala/algebra/instances/double.scala b/algebra-core/src/main/scala/algebra/instances/double.scala new file mode 100644 index 0000000000..9c59145f79 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/double.scala @@ -0,0 +1,43 @@ +package algebra +package instances + +import algebra.lattice.DistributiveLattice +import algebra.ring.Field + +import java.lang.Math + +trait DoubleInstances extends cats.kernel.instances.DoubleInstances { + implicit val doubleAlgebra: Field[Double] = + new DoubleAlgebra + + // This is not Bounded due to the presence of NaN + val DoubleMinMaxLattice: DistributiveLattice[Double] = + DistributiveLattice.minMax[Double] +} + +/** + * Due to the way floating-point equality works, this instance is not + * lawful under equality, but is correct when taken as an + * approximation of an exact value. + * + * If you would prefer an absolutely lawful fractional value, you'll + * need to investigate rational numbers or more exotic types. + */ +class DoubleAlgebra extends Field[Double] with Serializable { + + def zero: Double = 0.0 + def one: Double = 1.0 + + def plus(x: Double, y: Double): Double = x + y + def negate(x: Double): Double = -x + override def minus(x: Double, y: Double): Double = x - y + + def times(x: Double, y: Double): Double = x * y + def div(x: Double, y: Double): Double = x / y + override def reciprocal(x: Double): Double = 1.0 / x + override def pow(x: Double, y: Int): Double = Math.pow(x, y.toDouble) + + override def fromInt(x: Int): Double = x.toDouble + override def fromBigInt(n: BigInt): Double = n.toDouble + override def fromDouble(x: Double): Double = x +} diff --git a/algebra-core/src/main/scala/algebra/instances/float.scala b/algebra-core/src/main/scala/algebra/instances/float.scala new file mode 100644 index 0000000000..a8267dd7ec --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/float.scala @@ -0,0 +1,44 @@ +package algebra +package instances + +import algebra.lattice.DistributiveLattice +import algebra.ring.Field +import java.lang.Math + +trait FloatInstances extends cats.kernel.instances.FloatInstances { + implicit val floatAlgebra: Field[Float] = + new FloatAlgebra + + // Not bounded due to the presence of NaN + val FloatMinMaxLattice: DistributiveLattice[Float] = + DistributiveLattice.minMax[Float] +} + +/** + * Due to the way floating-point equality works, this instance is not + * lawful under equality, but is correct when taken as an + * approximation of an exact value. + * + * If you would prefer an absolutely lawful fractional value, you'll + * need to investigate rational numbers or more exotic types. + */ +class FloatAlgebra extends Field[Float] with Serializable { + + def zero: Float = 0.0f + def one: Float = 1.0f + + def plus(x: Float, y: Float): Float = x + y + def negate(x: Float): Float = -x + override def minus(x: Float, y: Float): Float = x - y + + def times(x: Float, y: Float): Float = x * y + def div(x: Float, y: Float): Float = x / y + override def reciprocal(x: Float): Float = 1.0f / x + + override def pow(x: Float, y: Int): Float = + Math.pow(x.toDouble, y.toDouble).toFloat + + override def fromInt(x: Int): Float = x.toFloat + override def fromBigInt(n: BigInt): Float = n.toFloat + override def fromDouble(x: Double): Float = x.toFloat +} diff --git a/algebra-core/src/main/scala/algebra/instances/int.scala b/algebra-core/src/main/scala/algebra/instances/int.scala new file mode 100644 index 0000000000..77762f7ac7 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/int.scala @@ -0,0 +1,33 @@ +package algebra +package instances + +import algebra.lattice._ +import algebra.ring._ + +package object int extends IntInstances + +trait IntInstances extends cats.kernel.instances.IntInstances { + implicit val intAlgebra: IntAlgebra = + new IntAlgebra + + val IntMinMaxLattice: BoundedDistributiveLattice[Int] = + BoundedDistributiveLattice.minMax[Int](Int.MinValue, Int.MaxValue) +} + +class IntAlgebra extends CommutativeRing[Int] with Serializable { + + def zero: Int = 0 + def one: Int = 1 + + def plus(x: Int, y: Int): Int = x + y + def negate(x: Int): Int = -x + override def minus(x: Int, y: Int): Int = x - y + + def times(x: Int, y: Int): Int = x * y + + override def pow(x: Int, y: Int): Int = + StaticMethods.pow(x.toLong, y.toLong).toInt + + override def fromInt(n: Int): Int = n + override def fromBigInt(n: BigInt): Int = n.toInt +} diff --git a/algebra-core/src/main/scala/algebra/instances/list.scala b/algebra-core/src/main/scala/algebra/instances/list.scala new file mode 100644 index 0000000000..7bf52cbc81 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/list.scala @@ -0,0 +1,6 @@ +package algebra +package instances + +package object list extends ListInstances + +trait ListInstances extends cats.kernel.instances.ListInstances diff --git a/algebra-core/src/main/scala/algebra/instances/long.scala b/algebra-core/src/main/scala/algebra/instances/long.scala new file mode 100644 index 0000000000..3c5f917b9c --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/long.scala @@ -0,0 +1,32 @@ +package algebra +package instances + +import algebra.lattice._ +import algebra.ring._ + +package object long extends LongInstances + +trait LongInstances extends cats.kernel.instances.LongInstances { + implicit val longAlgebra: LongAlgebra = + new LongAlgebra + + val LongMinMaxLattice: BoundedDistributiveLattice[Long] = + BoundedDistributiveLattice.minMax[Long](Long.MinValue, Long.MaxValue) +} + +class LongAlgebra extends CommutativeRing[Long] with Serializable { + + def zero: Long = 0 + def one: Long = 1 + + def plus(x: Long, y: Long): Long = x + y + def negate(x: Long): Long = -x + override def minus(x: Long, y: Long): Long = x - y + + def times(x: Long, y: Long): Long = x * y + + override def pow(x: Long, y: Int): Long = StaticMethods.pow(x, y.toLong) + + override def fromInt(n: Int): Long = n.toLong + override def fromBigInt(n: BigInt): Long = n.toLong +} diff --git a/algebra-core/src/main/scala/algebra/instances/map.scala b/algebra-core/src/main/scala/algebra/instances/map.scala new file mode 100644 index 0000000000..026c51998d --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/map.scala @@ -0,0 +1,132 @@ +package algebra +package instances + +import scala.annotation.nowarn +import scala.collection.mutable + +import algebra.ring._ + +package object map extends MapInstances + +trait MapInstances extends cats.kernel.instances.MapInstances with MapInstances3 + +trait MapInstances3 extends MapInstances2 {} + +trait MapInstances2 extends MapInstances1 { + implicit def mapSemiring[K, V: Semiring]: MapSemiring[K, V] = + new MapSemiring[K, V] +} + +trait MapInstances1 extends MapInstances0 {} + +trait MapInstances0 { + implicit def mapAdditiveMonoid[K, V: AdditiveSemigroup]: MapAdditiveMonoid[K, V] = + new MapAdditiveMonoid[K, V] +} + +class MapAdditiveMonoid[K, V](implicit V: AdditiveSemigroup[V]) extends AdditiveMonoid[Map[K, V]] { + def zero: Map[K, V] = Map.empty + + def plus(xs: Map[K, V], ys: Map[K, V]): Map[K, V] = + if (xs.size <= ys.size) { + xs.foldLeft(ys) { case (my, (k, x)) => + my.updated(k, + my.get(k) match { + case Some(y) => V.plus(x, y) + case None => x + } + ) + } + } else { + ys.foldLeft(xs) { case (mx, (k, y)) => + mx.updated(k, + mx.get(k) match { + case Some(x) => V.plus(x, y) + case None => y + } + ) + } + } + + override def sumN(a: Map[K, V], n: Int): Map[K, V] = + if (n > 0) a.map { case (k, v) => (k, V.sumN(v, n)) } + else if (n == 0) zero + else throw new IllegalArgumentException("Illegal negative exponent to sumN: %s".format(n)) + + @nowarn("msg=deprecated") + override def sum(as: TraversableOnce[Map[K, V]]): Map[K, V] = { + val acc = mutable.Map.empty[K, V] + as.foreach { m => + val it = m.iterator + while (it.hasNext) { + val (k, y) = it.next() + acc.get(k) match { + case None => acc(k) = y + case Some(x) => acc(k) = V.plus(x, y) + } + } + } + cats.kernel.instances.StaticMethods.wrapMutableMap(acc) + } +} + +class MapSemiring[K, V](implicit V: Semiring[V]) extends MapAdditiveMonoid[K, V] with Semiring[Map[K, V]] { + + def times(xs: Map[K, V], ys: Map[K, V]): Map[K, V] = + // we figure out which of our maps is smaller, iterate over its + // keys, see which of those are in the larger map, and add the + // resulting product to our result map. + if (xs.size <= ys.size) { + xs.foldLeft(Map.empty[K, V]) { case (m, (k, x)) => + ys.get(k) match { + case Some(y) => m.updated(k, V.times(x, y)) + case None => m + } + } + } else { + ys.foldLeft(Map.empty[K, V]) { case (m, (k, y)) => + xs.get(k) match { + case Some(x) => m.updated(k, V.times(x, y)) + case None => m + } + } + } + + override def pow(x: Map[K, V], n: Int): Map[K, V] = + if (n < 1) throw new IllegalArgumentException(s"non-positive exponent: $n") + else if (n == 1) x + else x.map { case (k, v) => (k, V.pow(v, n)) } + + @nowarn("msg=deprecated") + override def tryProduct(as: TraversableOnce[Map[K, V]]): Option[Map[K, V]] = + if (as.isEmpty) { + None + } else { + val acc = mutable.Map.empty[K, V] + var ready: Boolean = false + as.foreach { m => + if (ready) { + // at this point all we can do is modify or remove + // keys. since any "missing" key is effectively zero, and + // since 0 * x = 0, we ignore any keys not already in our + // accumulator. + val it = acc.iterator + while (it.hasNext) { + val (k, x) = it.next() + m.get(k) match { + case None => acc -= k + case Some(y) => acc(k) = V.times(x, y) + } + } + } else { + // we have to initialize our accumulator (`acc`) with the + // very first element of `as`. if there is only one map in + // our collection we want to return exactly those values. + val it = m.iterator + while (it.hasNext) acc += it.next() + ready = true + } + } + Some(cats.kernel.instances.StaticMethods.wrapMutableMap(acc)) + } +} diff --git a/algebra-core/src/main/scala/algebra/instances/option.scala b/algebra-core/src/main/scala/algebra/instances/option.scala new file mode 100644 index 0000000000..d6db1b1e05 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/option.scala @@ -0,0 +1,6 @@ +package algebra +package instances + +package object option extends OptionInstances + +trait OptionInstances extends cats.kernel.instances.OptionInstances diff --git a/algebra-core/src/main/scala/algebra/instances/set.scala b/algebra-core/src/main/scala/algebra/instances/set.scala new file mode 100644 index 0000000000..c2eba6f151 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/set.scala @@ -0,0 +1,36 @@ +package algebra +package instances + +import algebra.lattice.GenBool +import algebra.ring.{BoolRng, Semiring} + +package object set extends SetInstances + +trait SetInstances extends cats.kernel.instances.SetInstances { + + implicit def setLattice[A]: GenBool[Set[A]] = new SetLattice[A] + implicit def setSemiring[A]: Semiring[Set[A]] = new SetSemiring[A] + + // this instance is not compatible with setSemiring, so it is not + // marked as implicit to avoid an ambiguity. + def setBoolRng[A]: BoolRng[Set[A]] = new SetBoolRng[A] +} + +class SetLattice[A] extends GenBool[Set[A]] { + def zero: Set[A] = Set.empty + def or(lhs: Set[A], rhs: Set[A]): Set[A] = lhs | rhs + def and(lhs: Set[A], rhs: Set[A]): Set[A] = lhs & rhs + def without(lhs: Set[A], rhs: Set[A]): Set[A] = lhs -- rhs +} + +class SetSemiring[A] extends Semiring[Set[A]] { + def zero: Set[A] = Set.empty + def plus(x: Set[A], y: Set[A]): Set[A] = x | y + def times(x: Set[A], y: Set[A]): Set[A] = x & y +} + +class SetBoolRng[A] extends BoolRng[Set[A]] { + def zero: Set[A] = Set.empty + def plus(x: Set[A], y: Set[A]): Set[A] = (x -- y) | (y -- x) // this is xor + def times(x: Set[A], y: Set[A]): Set[A] = x & y +} diff --git a/algebra-core/src/main/scala/algebra/instances/short.scala b/algebra-core/src/main/scala/algebra/instances/short.scala new file mode 100644 index 0000000000..e40f36cb53 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/short.scala @@ -0,0 +1,33 @@ +package algebra +package instances + +import algebra.lattice._ +import algebra.ring._ + +package object short extends ShortInstances + +trait ShortInstances extends cats.kernel.instances.ShortInstances { + implicit val shortAlgebra: ShortAlgebra = + new ShortAlgebra + + val ShortMinMaxLattice: BoundedDistributiveLattice[Short] = + BoundedDistributiveLattice.minMax[Short](Short.MinValue, Short.MaxValue) +} + +class ShortAlgebra extends CommutativeRing[Short] with Serializable { + + def zero: Short = 0 + def one: Short = 1 + + def plus(x: Short, y: Short): Short = (x + y).toShort + def negate(x: Short): Short = (-x).toShort + override def minus(x: Short, y: Short): Short = (x - y).toShort + + def times(x: Short, y: Short): Short = (x * y).toShort + + override def pow(x: Short, y: Int): Short = + Math.pow(x.toDouble, y.toDouble).toShort + + override def fromInt(n: Int): Short = n.toShort + override def fromBigInt(n: BigInt): Short = n.toShort +} diff --git a/algebra-core/src/main/scala/algebra/instances/string.scala b/algebra-core/src/main/scala/algebra/instances/string.scala new file mode 100644 index 0000000000..8e9817035f --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/string.scala @@ -0,0 +1,6 @@ +package algebra +package instances + +package object string extends StringInstances + +trait StringInstances extends cats.kernel.instances.StringInstances diff --git a/algebra-core/src/main/scala/algebra/instances/tuple.scala b/algebra-core/src/main/scala/algebra/instances/tuple.scala new file mode 100644 index 0000000000..a73a87c67a --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/tuple.scala @@ -0,0 +1,4 @@ +package algebra +package instances + +package object tuple extends TupleInstances diff --git a/algebra-core/src/main/scala/algebra/instances/unit.scala b/algebra-core/src/main/scala/algebra/instances/unit.scala new file mode 100644 index 0000000000..6ee3fcedb8 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/instances/unit.scala @@ -0,0 +1,25 @@ +package algebra +package instances + +import algebra.ring.CommutativeRing + +package object unit extends UnitInstances + +trait UnitInstances extends cats.kernel.instances.UnitInstances { + implicit val unitRing: CommutativeRing[Unit] = + new UnitAlgebra +} + +class UnitAlgebra extends CommutativeRing[Unit] { + + def zero: Unit = () + def one: Unit = () + + override def isZero(x: Unit)(implicit ev: Eq[Unit]): Boolean = true + override def isOne(x: Unit)(implicit ev: Eq[Unit]): Boolean = true + + def plus(a: Unit, b: Unit): Unit = () + def negate(x: Unit): Unit = () + def times(a: Unit, b: Unit): Unit = () + override def pow(a: Unit, b: Int): Unit = () +} diff --git a/algebra-core/src/main/scala/algebra/lattice/Bool.scala b/algebra-core/src/main/scala/algebra/lattice/Bool.scala new file mode 100644 index 0000000000..a1b5dd2ff0 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/Bool.scala @@ -0,0 +1,93 @@ +package algebra +package lattice + +import ring.BoolRing +import scala.{specialized => sp} + +/** + * Boolean algebras are Heyting algebras with the additional + * constraint that the law of the excluded middle is true + * (equivalently, double-negation is true). + * + * This means that in addition to the laws Heyting algebras obey, + * boolean algebras also obey the following: + * + * - (a ∨ ¬a) = 1 + * - ¬¬a = a + * + * Boolean algebras generalize classical logic: one is equivalent to + * "true" and zero is equivalent to "false". Boolean algebras provide + * additional logical operators such as `xor`, `nand`, `nor`, and + * `nxor` which are commonly used. + * + * Every boolean algebras has a dual algebra, which involves reversing + * true/false as well as and/or. + */ +trait Bool[@sp(Int, Long) A] extends Any with Heyting[A] with GenBool[A] { self => + def imp(a: A, b: A): A = or(complement(a), b) + def without(a: A, b: A): A = and(a, complement(b)) + + // xor is already defined in both Heyting and GenBool. + // In Bool, the definitions coincide, so we just use one of them. + override def xor(a: A, b: A): A = + or(without(a, b), without(b, a)) + + override def dual: Bool[A] = new DualBool(this) + + /** + * Every Boolean algebra is a BoolRing, with multiplication defined as + * `and` and addition defined as `xor`. Bool does not extend BoolRing + * because, e.g. we might want a Bool[Int] and CommutativeRing[Int] to + * refer to different structures, by default. + * + * Note that the ring returned by this method is not an extension of + * the `Rig` returned from `BoundedDistributiveLattice.asCommutativeRig`. + */ + override def asBoolRing: BoolRing[A] = new BoolRingFromBool(self) +} + +class DualBool[@sp(Int, Long) A](orig: Bool[A]) extends Bool[A] { + def one: A = orig.zero + def zero: A = orig.one + def and(a: A, b: A): A = orig.or(a, b) + def or(a: A, b: A): A = orig.and(a, b) + def complement(a: A): A = orig.complement(a) + override def xor(a: A, b: A): A = orig.complement(orig.xor(a, b)) + + override def imp(a: A, b: A): A = orig.and(orig.complement(a), b) + override def nand(a: A, b: A): A = orig.nor(a, b) + override def nor(a: A, b: A): A = orig.nand(a, b) + override def nxor(a: A, b: A): A = orig.xor(a, b) + + override def dual: Bool[A] = orig +} + +class BoolRingFromBool[A](orig: Bool[A]) extends BoolRngFromGenBool(orig) with BoolRing[A] { + def one: A = orig.one +} + +/** + * Every Boolean ring gives rise to a Boolean algebra: + * - 0 and 1 are preserved; + * - ring multiplication (`times`) corresponds to `and`; + * - ring addition (`plus`) corresponds to `xor`; + * - `a or b` is then defined as `a xor b xor (a and b)`; + * - complement (`¬a`) is defined as `a xor 1`. + */ +class BoolFromBoolRing[A](orig: BoolRing[A]) extends GenBoolFromBoolRng(orig) with Bool[A] { + def one: A = orig.one + def complement(a: A): A = orig.plus(orig.one, a) + override def without(a: A, b: A): A = super[GenBoolFromBoolRng].without(a, b) + override def asBoolRing: BoolRing[A] = orig + + override def meet(a: A, b: A): A = super[GenBoolFromBoolRng].meet(a, b) + override def join(a: A, b: A): A = super[GenBoolFromBoolRng].join(a, b) +} + +object Bool extends HeytingFunctions[Bool] with GenBoolFunctions[Bool] { + + /** + * Access an implicit `Bool[A]`. + */ + @inline final def apply[@sp(Int, Long) A](implicit ev: Bool[A]): Bool[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/lattice/BoundedDistributiveLattice.scala b/algebra-core/src/main/scala/algebra/lattice/BoundedDistributiveLattice.scala new file mode 100644 index 0000000000..6d3838bfb9 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/BoundedDistributiveLattice.scala @@ -0,0 +1,56 @@ +package algebra +package lattice + +import scala.{specialized => sp} +import algebra.ring.CommutativeRig + +/** + * A bounded distributive lattice is a lattice that both bounded and distributive + */ +trait BoundedDistributiveLattice[@sp(Int, Long, Float, Double) A] + extends Any + with BoundedLattice[A] + with DistributiveLattice[A] { self => + + /** + * Return a CommutativeRig using join and meet. Note this must obey the commutative rig laws since + * meet(a, one) = a, and meet and join are associative, commutative and distributive. + */ + private[algebra] def asCommutativeRig: CommutativeRig[A] = + new CommutativeRig[A] { + def zero: A = self.zero + def one: A = self.one + def plus(x: A, y: A): A = self.join(x, y) + def times(x: A, y: A): A = self.meet(x, y) + } + + override def dual: BoundedDistributiveLattice[A] = new BoundedDistributiveLattice[A] { + def meet(a: A, b: A) = self.join(a, b) + def join(a: A, b: A) = self.meet(a, b) + def one = self.zero + def zero = self.one + override def dual = self + } +} + +object BoundedDistributiveLattice + extends BoundedMeetSemilatticeFunctions[BoundedDistributiveLattice] + with BoundedJoinSemilatticeFunctions[BoundedDistributiveLattice] { + + /** + * Access an implicit `BoundedDistributiveLattice[A]`. + */ + @inline final def apply[@sp(Int, Long, Float, Double) A](implicit + ev: BoundedDistributiveLattice[A] + ): BoundedDistributiveLattice[A] = ev + + def minMax[@sp(Int, Long, Float, Double) A](min: A, max: A)(implicit ord: Order[A]): BoundedDistributiveLattice[A] = + new MinMaxBoundedDistributiveLattice(min, max) +} + +class MinMaxBoundedDistributiveLattice[A](min: A, max: A)(implicit o: Order[A]) + extends MinMaxLattice[A] + with BoundedDistributiveLattice[A] { + def zero = min + def one = max +} diff --git a/algebra-core/src/main/scala/algebra/lattice/BoundedJoinSemilattice.scala b/algebra-core/src/main/scala/algebra/lattice/BoundedJoinSemilattice.scala new file mode 100644 index 0000000000..dc8189a3f9 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/BoundedJoinSemilattice.scala @@ -0,0 +1,31 @@ +package algebra +package lattice + +import scala.{specialized => sp} + +trait BoundedJoinSemilattice[@sp(Int, Long, Float, Double) A] extends Any with JoinSemilattice[A] { self => + def zero: A + def isZero(a: A)(implicit ev: Eq[A]): Boolean = ev.eqv(a, zero) + + override def joinSemilattice: BoundedSemilattice[A] = + new BoundedSemilattice[A] { + def empty: A = self.zero + def combine(x: A, y: A): A = join(x, y) + } +} + +trait BoundedJoinSemilatticeFunctions[B[A] <: BoundedJoinSemilattice[A]] extends JoinSemilatticeFunctions[B] { + def zero[@sp(Int, Long, Float, Double) A](implicit ev: B[A]): A = ev.zero +} + +object BoundedJoinSemilattice + extends JoinSemilatticeFunctions[BoundedJoinSemilattice] + with BoundedJoinSemilatticeFunctions[BoundedJoinSemilattice] { + + /** + * Access an implicit `BoundedJoinSemilattice[A]`. + */ + @inline final def apply[@sp(Int, Long, Float, Double) A](implicit + ev: BoundedJoinSemilattice[A] + ): BoundedJoinSemilattice[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/lattice/BoundedLattice.scala b/algebra-core/src/main/scala/algebra/lattice/BoundedLattice.scala new file mode 100644 index 0000000000..f1dab0da4a --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/BoundedLattice.scala @@ -0,0 +1,41 @@ +package algebra +package lattice + +import scala.{specialized => sp} + +/** + * A bounded lattice is a lattice that additionally has one element + * that is the bottom (zero, also written as ⊥), and one element that + * is the top (one, also written as ⊤). + * + * This means that for any a in A: + * + * join(zero, a) = a = meet(one, a) + * + * Or written using traditional notation: + * + * (0 ∨ a) = a = (1 ∧ a) + */ +trait BoundedLattice[@sp(Int, Long, Float, Double) A] + extends Any + with Lattice[A] + with BoundedMeetSemilattice[A] + with BoundedJoinSemilattice[A] { self => + override def dual: BoundedLattice[A] = new BoundedLattice[A] { + def meet(a: A, b: A) = self.join(a, b) + def join(a: A, b: A) = self.meet(a, b) + def one = self.zero + def zero = self.one + override def dual = self + } +} + +object BoundedLattice + extends BoundedMeetSemilatticeFunctions[BoundedLattice] + with BoundedJoinSemilatticeFunctions[BoundedLattice] { + + /** + * Access an implicit `BoundedLattice[A]`. + */ + @inline final def apply[@sp(Int, Long, Float, Double) A](implicit ev: BoundedLattice[A]): BoundedLattice[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/lattice/BoundedMeetSemilattice.scala b/algebra-core/src/main/scala/algebra/lattice/BoundedMeetSemilattice.scala new file mode 100644 index 0000000000..1b2f37a6ef --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/BoundedMeetSemilattice.scala @@ -0,0 +1,30 @@ +package algebra +package lattice + +import scala.{specialized => sp} + +trait BoundedMeetSemilattice[@sp(Int, Long, Float, Double) A] extends Any with MeetSemilattice[A] { self => + def one: A + def isOne(a: A)(implicit ev: Eq[A]): Boolean = ev.eqv(a, one) + + override def meetSemilattice: BoundedSemilattice[A] = + new BoundedSemilattice[A] { + def empty: A = self.one + def combine(x: A, y: A): A = meet(x, y) + } +} + +trait BoundedMeetSemilatticeFunctions[B[A] <: BoundedMeetSemilattice[A]] extends MeetSemilatticeFunctions[B] { + def one[@sp(Int, Long, Float, Double) A](implicit ev: B[A]): A = + ev.one +} + +object BoundedMeetSemilattice extends BoundedMeetSemilatticeFunctions[BoundedMeetSemilattice] { + + /** + * Access an implicit `BoundedMeetSemilattice[A]`. + */ + @inline final def apply[@sp(Int, Long, Float, Double) A](implicit + ev: BoundedMeetSemilattice[A] + ): BoundedMeetSemilattice[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/lattice/DeMorgan.scala b/algebra-core/src/main/scala/algebra/lattice/DeMorgan.scala new file mode 100644 index 0000000000..e5b30906a5 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/DeMorgan.scala @@ -0,0 +1,61 @@ +package algebra +package lattice + +import scala.{specialized => sp} + +/** + * De Morgan algebras are bounded lattices that are also equipped with + * a De Morgan involution. + * + * De Morgan involution obeys the following laws: + * + * - ¬¬a = a + * - ¬(x∧y) = ¬x∨¬y + * + * However, in De Morgan algebras this involution does not necessarily + * provide the law of the excluded middle. This means that there is no + * guarantee that (a ∨ ¬a) = 1. De Morgan algebra do not not necessarily + * provide the law of non contradiction either. This means that there is + * no guarantee that (a ∧ ¬a) = 0. + * + * De Morgan algebras are useful to model fuzzy logic. For a model of + * classical logic, see the boolean algebra type class implemented as + * [[Bool]]. + */ +trait DeMorgan[@sp(Int, Long) A] extends Any with Logic[A] { self => + def meet(a: A, b: A): A = and(a, b) + + def join(a: A, b: A): A = or(a, b) + + def imp(a: A, b: A): A = or(not(a), b) +} + +trait DeMorganFunctions[H[A] <: DeMorgan[A]] + extends BoundedMeetSemilatticeFunctions[H] + with BoundedJoinSemilatticeFunctions[H] + with LogicFunctions[H] + +object DeMorgan extends DeMorganFunctions[DeMorgan] { + + /** + * Access an implicit `DeMorgan[A]`. + */ + @inline final def apply[@sp(Int, Long) A](implicit ev: DeMorgan[A]): DeMorgan[A] = ev + + /** + * Turn a [[Bool]] into a `DeMorgan` + * Used for binary compatibility. + */ + final def fromBool[@sp(Int, Long) A](bool: Bool[A]): DeMorgan[A] = + new DeMorgan[A] { + def and(a: A, b: A): A = bool.and(a, b) + + def or(a: A, b: A): A = bool.or(a, b) + + def not(a: A): A = bool.complement(a) + + def one: A = bool.one + + def zero: A = bool.zero + } +} diff --git a/algebra-core/src/main/scala/algebra/lattice/DistributiveLattice.scala b/algebra-core/src/main/scala/algebra/lattice/DistributiveLattice.scala new file mode 100644 index 0000000000..c2a764d8c7 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/DistributiveLattice.scala @@ -0,0 +1,32 @@ +package algebra +package lattice + +import scala.{specialized => sp} + +/** + * A distributive lattice a lattice where join and meet distribute: + * + * - a ∨ (b ∧ c) = (a ∨ b) ∧ (a ∨ c) + * - a ∧ (b ∨ c) = (a ∧ b) ∨ (a ∧ c) + */ +trait DistributiveLattice[@sp(Int, Long, Float, Double) A] extends Any with Lattice[A] + +object DistributiveLattice + extends JoinSemilatticeFunctions[DistributiveLattice] + with MeetSemilatticeFunctions[DistributiveLattice] { + + /** + * Access an implicit `Lattice[A]`. + */ + @inline final def apply[@sp(Int, Long, Float, Double) A](implicit + ev: DistributiveLattice[A] + ): DistributiveLattice[A] = ev + + def minMax[@sp(Int, Long, Float, Double) A: Order]: DistributiveLattice[A] = + new MinMaxLattice[A] +} + +class MinMaxLattice[@sp(Int, Long, Float, Double) A](implicit order: Order[A]) extends DistributiveLattice[A] { + def join(x: A, y: A): A = order.max(x, y) + def meet(x: A, y: A): A = order.min(x, y) +} diff --git a/algebra-core/src/main/scala/algebra/lattice/GenBool.scala b/algebra-core/src/main/scala/algebra/lattice/GenBool.scala new file mode 100644 index 0000000000..a939610ef2 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/GenBool.scala @@ -0,0 +1,76 @@ +package algebra +package lattice + +import ring.BoolRng +import scala.{specialized => sp} + +/** + * Generalized Boolean algebra, that is, a Boolean algebra without + * the top element. Generalized Boolean algebras do not (in general) + * have (absolute) complements, but they have ''relative complements'' + * (see [[GenBool.without]]). + */ +trait GenBool[@sp(Int, Long) A] extends Any with DistributiveLattice[A] with BoundedJoinSemilattice[A] { self => + def and(a: A, b: A): A + override def meet(a: A, b: A): A = and(a, b) + + def or(a: A, b: A): A + override def join(a: A, b: A): A = or(a, b) + + /** + * The operation of ''relative complement'', symbolically often denoted + * `a\b` (the symbol for set-theoretic difference, which is the + * meaning of relative complement in the lattice of sets). + */ + def without(a: A, b: A): A + + /** + * Logical exclusive or, set-theoretic symmetric difference. + * Defined as `a\b ∨ b\a`. + */ + def xor(a: A, b: A): A = or(without(a, b), without(b, a)) + + /** + * Every generalized Boolean algebra is also a `BoolRng`, with + * multiplication defined as `and` and addition defined as `xor`. + */ + @deprecated("See typelevel/algebra#108 for discussion", since = "2.7.0") + def asBoolRing: BoolRng[A] = new BoolRngFromGenBool(self) +} + +/** + * Every Boolean rng gives rise to a Boolean algebra without top: + * - 0 is preserved; + * - ring multiplication (`times`) corresponds to `and`; + * - ring addition (`plus`) corresponds to `xor`; + * - `a or b` is then defined as `a xor b xor (a and b)`; + * - relative complement `a\b` is defined as `a xor (a and b)`. + * + * `BoolRng.asBool.asBoolRing` gives back the original `BoolRng`. + * + * @see [[algebra.lattice.GenBool.asBoolRing]] + */ +class GenBoolFromBoolRng[A](orig: BoolRng[A]) extends GenBool[A] { + def zero: A = orig.zero + def and(a: A, b: A): A = orig.times(a, b) + def or(a: A, b: A): A = orig.plus(orig.plus(a, b), orig.times(a, b)) + def without(a: A, b: A): A = orig.plus(a, orig.times(a, b)) + override def asBoolRing: BoolRng[A] = orig +} + +class BoolRngFromGenBool[@sp(Int, Long) A](orig: GenBool[A]) extends BoolRng[A] { + def zero: A = orig.zero + def plus(x: A, y: A): A = orig.xor(x, y) + def times(x: A, y: A): A = orig.and(x, y) +} + +trait GenBoolFunctions[G[A] <: GenBool[A]] extends BoundedJoinSemilatticeFunctions[G] with MeetSemilatticeFunctions[G] { + def and[@sp(Int, Long) A](x: A, y: A)(implicit ev: G[A]): A = ev.and(x, y) + def or[@sp(Int, Long) A](x: A, y: A)(implicit ev: G[A]): A = ev.or(x, y) + def without[@sp(Int, Long) A](x: A, y: A)(implicit ev: G[A]): A = ev.without(x, y) + def xor[@sp(Int, Long) A](x: A, y: A)(implicit ev: G[A]): A = ev.xor(x, y) +} + +object GenBool extends GenBoolFunctions[GenBool] { + @inline final def apply[@sp(Int, Long) A](implicit ev: GenBool[A]): GenBool[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/lattice/Heyting.scala b/algebra-core/src/main/scala/algebra/lattice/Heyting.scala new file mode 100644 index 0000000000..d5dbe0c0a0 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/Heyting.scala @@ -0,0 +1,84 @@ +package algebra +package lattice + +import scala.{specialized => sp} + +/** + * Heyting algebras are bounded lattices that are also equipped with + * an additional binary operation `imp` (for implication, also + * written as →). + * + * Implication obeys the following laws: + * + * - a → a = 1 + * - a ∧ (a → b) = a ∧ b + * - b ∧ (a → b) = b + * - a → (b ∧ c) = (a → b) ∧ (a → c) + * + * In heyting algebras, `and` is equivalent to `meet` and `or` is + * equivalent to `join`; both methods are available. + * + * Heyting algebra also define `complement` operation (sometimes + * written as ¬a). The complement of `a` is equivalent to `(a → 0)`, + * and the following laws hold: + * + * - a ∧ ¬a = 0 + * + * However, in Heyting algebras this operation is only a + * pseudo-complement, since Heyting algebras do not necessarily + * provide the law of the excluded middle. This means that there is no + * guarantee that (a ∨ ¬a) = 1. + * + * Heyting algebras model intuitionistic logic. For a model of + * classical logic, see the boolean algebra type class implemented as + * `Bool`. + */ +trait Heyting[@sp(Int, Long) A] extends Any with BoundedDistributiveLattice[A] { self => + def and(a: A, b: A): A + def meet(a: A, b: A): A = and(a, b) + + def or(a: A, b: A): A + def join(a: A, b: A): A = or(a, b) + + def imp(a: A, b: A): A + def complement(a: A): A + + def xor(a: A, b: A): A = or(and(a, complement(b)), and(complement(a), b)) + def nand(a: A, b: A): A = complement(and(a, b)) + def nor(a: A, b: A): A = complement(or(a, b)) + def nxor(a: A, b: A): A = complement(xor(a, b)) +} + +trait HeytingGenBoolOverlap[H[A] <: Heyting[A]] { + def and[@sp(Int, Long) A](x: A, y: A)(implicit ev: H[A]): A = + ev.and(x, y) + def or[@sp(Int, Long) A](x: A, y: A)(implicit ev: H[A]): A = + ev.or(x, y) + def xor[@sp(Int, Long) A](x: A, y: A)(implicit ev: H[A]): A = + ev.xor(x, y) +} + +trait HeytingFunctions[H[A] <: Heyting[A]] + extends BoundedMeetSemilatticeFunctions[H] + with BoundedJoinSemilatticeFunctions[H] { + + def complement[@sp(Int, Long) A](x: A)(implicit ev: H[A]): A = + ev.complement(x) + + def imp[@sp(Int, Long) A](x: A, y: A)(implicit ev: H[A]): A = + ev.imp(x, y) + def nor[@sp(Int, Long) A](x: A, y: A)(implicit ev: H[A]): A = + ev.nor(x, y) + def nxor[@sp(Int, Long) A](x: A, y: A)(implicit ev: H[A]): A = + ev.nxor(x, y) + def nand[@sp(Int, Long) A](x: A, y: A)(implicit ev: H[A]): A = + ev.nand(x, y) +} + +object Heyting extends HeytingFunctions[Heyting] with HeytingGenBoolOverlap[Heyting] { + + /** + * Access an implicit `Heyting[A]`. + */ + @inline final def apply[@sp(Int, Long) A](implicit ev: Heyting[A]): Heyting[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/lattice/JoinSemilattice.scala b/algebra-core/src/main/scala/algebra/lattice/JoinSemilattice.scala new file mode 100644 index 0000000000..ae7c019ae5 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/JoinSemilattice.scala @@ -0,0 +1,35 @@ +package algebra +package lattice + +import scala.{specialized => sp} + +/** + * A join-semilattice (or upper semilattice) is a semilattice whose + * operation is called "join", and which can be thought of as a least + * upper bound. + */ +trait JoinSemilattice[@sp(Int, Long, Float, Double) A] extends Any with Serializable { self => + def join(lhs: A, rhs: A): A + + def joinSemilattice: Semilattice[A] = + new Semilattice[A] { + def combine(x: A, y: A): A = self.join(x, y) + } + + def joinPartialOrder(implicit ev: Eq[A]): PartialOrder[A] = + joinSemilattice.asJoinPartialOrder +} + +trait JoinSemilatticeFunctions[J[A] <: JoinSemilattice[A]] { + def join[@sp(Int, Long, Float, Double) A](x: A, y: A)(implicit ev: J[A]): A = + ev.join(x, y) +} + +object JoinSemilattice extends JoinSemilatticeFunctions[JoinSemilattice] { + + /** + * Access an implicit `JoinSemilattice[A]`. + */ + @inline final def apply[@sp(Int, Long, Float, Double) A](implicit ev: JoinSemilattice[A]): JoinSemilattice[A] = ev + +} diff --git a/algebra-core/src/main/scala/algebra/lattice/Lattice.scala b/algebra-core/src/main/scala/algebra/lattice/Lattice.scala new file mode 100644 index 0000000000..040f3d66fd --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/Lattice.scala @@ -0,0 +1,38 @@ +package algebra +package lattice + +import scala.{specialized => sp} + +/** + * A lattice is a set `A` together with two operations (meet and + * join). Both operations individually constitute semilattices (join- + * and meet-semilattices respectively): each operation is commutative, + * associative, and idempotent. + * + * Join can be thought of as finding a least upper bound (supremum), + * and meet can be thought of as finding a greatest lower bound + * (infimum). + * + * The join and meet operations are also linked by absorption laws: + * + * meet(a, join(a, b)) = join(a, meet(a, b)) = a + */ +trait Lattice[@sp(Int, Long, Float, Double) A] extends Any with JoinSemilattice[A] with MeetSemilattice[A] { self => + + /** + * This is the lattice with meet and join swapped + */ + def dual: Lattice[A] = new Lattice[A] { + def meet(a: A, b: A) = self.join(a, b) + def join(a: A, b: A) = self.meet(a, b) + override def dual = self + } +} + +object Lattice extends JoinSemilatticeFunctions[Lattice] with MeetSemilatticeFunctions[Lattice] { + + /** + * Access an implicit `Lattice[A]`. + */ + @inline final def apply[@sp(Int, Long, Float, Double) A](implicit ev: Lattice[A]): Lattice[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/lattice/Logic.scala b/algebra-core/src/main/scala/algebra/lattice/Logic.scala new file mode 100644 index 0000000000..1fd07eb8c0 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/Logic.scala @@ -0,0 +1,69 @@ +package algebra +package lattice + +import scala.{specialized => sp} + +/** + * Logic models a logic generally. It is a bounded distributive + * lattice with an extra negation operator. + * + * The negation operator obeys the weak De Morgan laws: + * - ¬(x∨y) = ¬x∧¬y + * - ¬(x∧y) = ¬¬(¬x∨¬y) + * + * For intuitionistic logic see [[Heyting]] + * For fuzzy logic see [[DeMorgan]] + */ +trait Logic[@sp(Int, Long) A] extends Any with BoundedDistributiveLattice[A] { self => + def and(a: A, b: A): A + + def or(a: A, b: A): A + + def not(a: A): A + + def xor(a: A, b: A): A = or(and(a, not(b)), and(not(a), b)) + def nand(a: A, b: A): A = not(and(a, b)) + def nor(a: A, b: A): A = not(or(a, b)) + def nxor(a: A, b: A): A = not(xor(a, b)) +} + +trait LogicFunctions[H[A] <: Logic[A]] { + def complement[@sp(Int, Long) A](x: A)(implicit ev: H[A]): A = + ev.not(x) + + def nor[@sp(Int, Long) A](x: A, y: A)(implicit ev: H[A]): A = + ev.nor(x, y) + def nxor[@sp(Int, Long) A](x: A, y: A)(implicit ev: H[A]): A = + ev.nxor(x, y) + def nand[@sp(Int, Long) A](x: A, y: A)(implicit ev: H[A]): A = + ev.nand(x, y) +} + +object Logic extends LogicFunctions[Logic] { + + /** + * Access an implicit `Logic[A]`. + */ + @inline final def apply[@sp(Int, Long) A](implicit ev: Logic[A]): Logic[A] = ev + + /** + * Turn a [[Heyting]] into a `Logic`. + * Used for binary compatibility. + */ + final def fromHeyting[@sp(Int, Long) A](h: Heyting[A]): Logic[A] = + new Logic[A] { + def and(a: A, b: A): A = h.and(a, b) + + def or(a: A, b: A): A = h.or(a, b) + + def not(a: A): A = h.complement(a) + + def zero: A = h.zero + + def one: A = h.one + + def meet(lhs: A, rhs: A): A = h.meet(lhs, rhs) + + def join(lhs: A, rhs: A): A = h.join(lhs, rhs) + } +} diff --git a/algebra-core/src/main/scala/algebra/lattice/MeetSemilattice.scala b/algebra-core/src/main/scala/algebra/lattice/MeetSemilattice.scala new file mode 100644 index 0000000000..1dc72df29f --- /dev/null +++ b/algebra-core/src/main/scala/algebra/lattice/MeetSemilattice.scala @@ -0,0 +1,34 @@ +package algebra +package lattice + +import scala.{specialized => sp} + +/** + * A meet-semilattice (or lower semilattice) is a semilattice whose + * operation is called "meet", and which can be thought of as a + * greatest lower bound. + */ +trait MeetSemilattice[@sp(Int, Long, Float, Double) A] extends Any with Serializable { self => + def meet(lhs: A, rhs: A): A + + def meetSemilattice: Semilattice[A] = + new Semilattice[A] { + def combine(x: A, y: A): A = self.meet(x, y) + } + + def meetPartialOrder(implicit ev: Eq[A]): PartialOrder[A] = + meetSemilattice.asMeetPartialOrder +} + +trait MeetSemilatticeFunctions[M[A] <: MeetSemilattice[A]] { + def meet[@sp(Int, Long, Float, Double) A](x: A, y: A)(implicit ev: M[A]): A = + ev.meet(x, y) +} + +object MeetSemilattice extends MeetSemilatticeFunctions[MeetSemilattice] { + + /** + * Access an implicit `MeetSemilattice[A]`. + */ + @inline final def apply[@sp(Int, Long, Float, Double) A](implicit ev: MeetSemilattice[A]): MeetSemilattice[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/package.scala b/algebra-core/src/main/scala/algebra/package.scala new file mode 100644 index 0000000000..0f720497ef --- /dev/null +++ b/algebra-core/src/main/scala/algebra/package.scala @@ -0,0 +1,38 @@ +package object algebra { + + type Band[A] = cats.kernel.Band[A] + val Band = cats.kernel.Band + + type BoundedSemilattice[A] = cats.kernel.BoundedSemilattice[A] + val BoundedSemilattice = cats.kernel.BoundedSemilattice + + type CommutativeGroup[A] = cats.kernel.CommutativeGroup[A] + val CommutativeGroup = cats.kernel.CommutativeGroup + + type CommutativeMonoid[A] = cats.kernel.CommutativeMonoid[A] + val CommutativeMonoid = cats.kernel.CommutativeMonoid + + type CommutativeSemigroup[A] = cats.kernel.CommutativeSemigroup[A] + val CommutativeSemigroup = cats.kernel.CommutativeSemigroup + + type Eq[A] = cats.kernel.Eq[A] + val Eq = cats.kernel.Eq + + type Group[A] = cats.kernel.Group[A] + val Group = cats.kernel.Group + + type Monoid[A] = cats.kernel.Monoid[A] + val Monoid = cats.kernel.Monoid + + type Order[A] = cats.kernel.Order[A] + val Order = cats.kernel.Order + + type PartialOrder[A] = cats.kernel.PartialOrder[A] + val PartialOrder = cats.kernel.PartialOrder + + type Semigroup[A] = cats.kernel.Semigroup[A] + val Semigroup = cats.kernel.Semigroup + + type Semilattice[A] = cats.kernel.Semilattice[A] + val Semilattice = cats.kernel.Semilattice +} diff --git a/algebra-core/src/main/scala/algebra/ring/Additive.scala b/algebra-core/src/main/scala/algebra/ring/Additive.scala new file mode 100644 index 0000000000..ef5acc7801 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/Additive.scala @@ -0,0 +1,251 @@ +package algebra +package ring + +import scala.{specialized => sp} +import scala.annotation.{nowarn, tailrec} + +trait AdditiveSemigroup[@sp(Int, Long, Float, Double) A] extends Any with Serializable { + def additive: Semigroup[A] = new Semigroup[A] { + def combine(x: A, y: A): A = plus(x, y) + @nowarn("msg=deprecated") + override def combineAllOption(as: TraversableOnce[A]): Option[A] = trySum(as) + } + + def plus(x: A, y: A): A + + def sumN(a: A, n: Int): A = + if (n > 0) positiveSumN(a, n) + else throw new IllegalArgumentException("Illegal non-positive exponent to sumN: %s".format(n)) + + protected[this] def positiveSumN(a: A, n: Int): A = { + @tailrec def loop(b: A, k: Int, extra: A): A = + if (k == 1) plus(b, extra) + else { + val x = if ((k & 1) == 1) plus(b, extra) else extra + loop(plus(b, b), k >>> 1, x) + } + if (n == 1) a else loop(a, n - 1, a) + } + + /** + * Given a sequence of `as`, combine them and return the total. + * + * If the sequence is empty, returns None. Otherwise, returns Some(total). + */ + @nowarn("msg=deprecated") + def trySum(as: TraversableOnce[A]): Option[A] = + as.toIterator.reduceOption(plus) +} + +trait AdditiveCommutativeSemigroup[@sp(Int, Long, Float, Double) A] extends Any with AdditiveSemigroup[A] { + override def additive: CommutativeSemigroup[A] = new CommutativeSemigroup[A] { + def combine(x: A, y: A): A = plus(x, y) + @nowarn("msg=deprecated") + override def combineAllOption(as: TraversableOnce[A]): Option[A] = trySum(as) + } +} + +trait AdditiveMonoid[@sp(Int, Long, Float, Double) A] extends Any with AdditiveSemigroup[A] { + override def additive: Monoid[A] = new Monoid[A] { + def empty = zero + def combine(x: A, y: A): A = plus(x, y) + @nowarn("msg=deprecated") + override def combineAllOption(as: TraversableOnce[A]): Option[A] = trySum(as) + @nowarn("msg=deprecated") + override def combineAll(as: TraversableOnce[A]): A = sum(as) + } + + def zero: A + + /** + * Tests if `a` is zero. + */ + def isZero(a: A)(implicit ev: Eq[A]): Boolean = ev.eqv(a, zero) + + override def sumN(a: A, n: Int): A = + if (n > 0) positiveSumN(a, n) + else if (n == 0) zero + else throw new IllegalArgumentException("Illegal negative exponent to sumN: %s".format(n)) + + /** + * Given a sequence of `as`, compute the sum. + */ + @nowarn("msg=deprecated") + def sum(as: TraversableOnce[A]): A = + as.foldLeft(zero)(plus) + + @nowarn("msg=deprecated") + override def trySum(as: TraversableOnce[A]): Option[A] = + if (as.isEmpty) None else Some(sum(as)) +} + +trait AdditiveCommutativeMonoid[@sp(Int, Long, Float, Double) A] + extends Any + with AdditiveMonoid[A] + with AdditiveCommutativeSemigroup[A] { + override def additive: CommutativeMonoid[A] = new CommutativeMonoid[A] { + def empty = zero + def combine(x: A, y: A): A = plus(x, y) + @nowarn("msg=deprecated") + override def combineAllOption(as: TraversableOnce[A]): Option[A] = trySum(as) + @nowarn("msg=deprecated") + override def combineAll(as: TraversableOnce[A]): A = sum(as) + } +} + +trait AdditiveGroup[@sp(Int, Long, Float, Double) A] extends Any with AdditiveMonoid[A] { + override def additive: Group[A] = new Group[A] { + def empty = zero + def combine(x: A, y: A): A = plus(x, y) + override def remove(x: A, y: A): A = minus(x, y) + def inverse(x: A): A = negate(x) + @nowarn("msg=deprecated") + override def combineAllOption(as: TraversableOnce[A]): Option[A] = trySum(as) + @nowarn("msg=deprecated") + override def combineAll(as: TraversableOnce[A]): A = sum(as) + } + + def negate(x: A): A + def minus(x: A, y: A): A = plus(x, negate(y)) + + override def sumN(a: A, n: Int): A = + if (n > 0) positiveSumN(a, n) + else if (n == 0) zero + else if (n == Int.MinValue) positiveSumN(negate(plus(a, a)), 1073741824) + else positiveSumN(negate(a), -n) +} + +trait AdditiveCommutativeGroup[@sp(Int, Long, Float, Double) A] + extends Any + with AdditiveGroup[A] + with AdditiveCommutativeMonoid[A] { + override def additive: CommutativeGroup[A] = new CommutativeGroup[A] { + def empty = zero + def combine(x: A, y: A): A = plus(x, y) + override def remove(x: A, y: A): A = minus(x, y) + def inverse(x: A): A = negate(x) + @nowarn("msg=deprecated") + override def combineAllOption(as: TraversableOnce[A]): Option[A] = trySum(as) + @nowarn("msg=deprecated") + override def combineAll(as: TraversableOnce[A]): A = sum(as) + } +} + +trait AdditiveSemigroupFunctions[S[T] <: AdditiveSemigroup[T]] { + + def isAdditiveCommutative[A](implicit ev: S[A]): Boolean = + ev.isInstanceOf[AdditiveCommutativeSemigroup[_]] + + def plus[@sp(Int, Long, Float, Double) A](x: A, y: A)(implicit ev: S[A]): A = + ev.plus(x, y) + + def sumN[@sp(Int, Long, Float, Double) A](a: A, n: Int)(implicit ev: S[A]): A = + ev.sumN(a, n) + + @nowarn("msg=deprecated") + def trySum[A](as: TraversableOnce[A])(implicit ev: S[A]): Option[A] = + ev.trySum(as) +} + +trait AdditiveMonoidFunctions[M[T] <: AdditiveMonoid[T]] extends AdditiveSemigroupFunctions[M] { + def zero[@sp(Int, Long, Float, Double) A](implicit ev: M[A]): A = + ev.zero + + def isZero[@sp(Int, Long, Float, Double) A](a: A)(implicit ev0: M[A], ev1: Eq[A]): Boolean = + ev0.isZero(a) + + @nowarn("msg=deprecated") + def sum[@sp(Int, Long, Float, Double) A](as: TraversableOnce[A])(implicit ev: M[A]): A = + ev.sum(as) +} + +trait AdditiveGroupFunctions[G[T] <: AdditiveGroup[T]] extends AdditiveMonoidFunctions[G] { + def negate[@sp(Int, Long, Float, Double) A](x: A)(implicit ev: G[A]): A = + ev.negate(x) + def minus[@sp(Int, Long, Float, Double) A](x: A, y: A)(implicit ev: G[A]): A = + ev.minus(x, y) +} + +object AdditiveSemigroup extends AdditiveSemigroupFunctions[AdditiveSemigroup] { + @inline final def apply[A](implicit ev: AdditiveSemigroup[A]): AdditiveSemigroup[A] = ev + + /** + * This method converts an additive instance into a generic + * instance. + * + * Given an implicit `AdditiveSemigroup[A]`, this method returns a + * `Semigroup[A]`. + */ + @inline final def additive[A](implicit ev: AdditiveSemigroup[A]): Semigroup[A] = + ev.additive +} + +object AdditiveCommutativeSemigroup extends AdditiveSemigroupFunctions[AdditiveCommutativeSemigroup] { + @inline final def apply[A](implicit ev: AdditiveCommutativeSemigroup[A]): AdditiveCommutativeSemigroup[A] = ev + + /** + * This method converts an additive instance into a generic + * instance. + * + * Given an implicit `AdditiveCommutativeSemigroup[A]`, this method returns a + * `CommutativeSemigroup[A]`. + */ + @inline final def additive[A](implicit ev: AdditiveCommutativeSemigroup[A]): CommutativeSemigroup[A] = + ev.additive +} + +object AdditiveMonoid extends AdditiveMonoidFunctions[AdditiveMonoid] { + @inline final def apply[A](implicit ev: AdditiveMonoid[A]): AdditiveMonoid[A] = ev + + /** + * This method converts an additive instance into a generic + * instance. + * + * Given an implicit `AdditiveMonoid[A]`, this method returns a + * `Monoid[A]`. + */ + @inline final def additive[A](implicit ev: AdditiveMonoid[A]): Monoid[A] = + ev.additive +} + +object AdditiveCommutativeMonoid extends AdditiveMonoidFunctions[AdditiveCommutativeMonoid] { + @inline final def apply[A](implicit ev: AdditiveCommutativeMonoid[A]): AdditiveCommutativeMonoid[A] = ev + + /** + * This method converts an additive instance into a generic + * instance. + * + * Given an implicit `AdditiveCommutativeMonoid[A]`, this method returns a + * `CommutativeMonoid[A]`. + */ + @inline final def additive[A](implicit ev: AdditiveCommutativeMonoid[A]): CommutativeMonoid[A] = + ev.additive +} + +object AdditiveGroup extends AdditiveGroupFunctions[AdditiveGroup] { + @inline final def apply[A](implicit ev: AdditiveGroup[A]): AdditiveGroup[A] = ev + + /** + * This method converts an additive instance into a generic + * instance. + * + * Given an implicit `AdditiveGroup[A]`, this method returns a + * `Group[A]`. + */ + @inline final def additive[A](implicit ev: AdditiveGroup[A]): Group[A] = + ev.additive +} + +object AdditiveCommutativeGroup extends AdditiveGroupFunctions[AdditiveCommutativeGroup] { + @inline final def apply[A](implicit ev: AdditiveCommutativeGroup[A]): AdditiveCommutativeGroup[A] = ev + + /** + * This method converts an additive instance into a generic + * instance. + * + * Given an implicit `AdditiveCommutativeGroup[A]`, this method returns a + * `CommutativeGroup[A]`. + */ + @inline final def additive[A](implicit ev: AdditiveCommutativeGroup[A]): CommutativeGroup[A] = + ev.additive +} diff --git a/algebra-core/src/main/scala/algebra/ring/BoolRing.scala b/algebra-core/src/main/scala/algebra/ring/BoolRing.scala new file mode 100644 index 0000000000..354c336d44 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/BoolRing.scala @@ -0,0 +1,16 @@ +package algebra +package ring + +/** + * A Boolean ring is a ring whose multiplication is idempotent, that is + * `a⋅a = a` for all elements ''a''. This property also implies `a+a = 0` + * for all ''a'', and `a⋅b = b⋅a` (commutativity of multiplication). + * + * Every Boolean ring is equivalent to a Boolean algebra. + * See `algebra.lattice.BoolFromBoolRing` for details. + */ +trait BoolRing[A] extends Any with BoolRng[A] with CommutativeRing[A] + +object BoolRing extends RingFunctions[BoolRing] { + @inline final def apply[A](implicit r: BoolRing[A]): BoolRing[A] = r +} diff --git a/algebra-core/src/main/scala/algebra/ring/BoolRng.scala b/algebra-core/src/main/scala/algebra/ring/BoolRng.scala new file mode 100644 index 0000000000..78ff3d0c73 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/BoolRng.scala @@ -0,0 +1,18 @@ +package algebra +package ring + +/** + * A Boolean rng is a rng whose multiplication is idempotent, that is + * `a⋅a = a` for all elements ''a''. This property also implies `a+a = 0` + * for all ''a'', and `a⋅b = b⋅a` (commutativity of multiplication). + * + * Every `BoolRng` is equivalent to `algebra.lattice.GenBool`. + * See `algebra.lattice.GenBoolFromBoolRng` for details. + */ +trait BoolRng[A] extends Any with CommutativeRng[A] { self => + final override def negate(x: A): A = x +} + +object BoolRng extends AdditiveGroupFunctions[BoolRng] with MultiplicativeSemigroupFunctions[BoolRng] { + @inline final def apply[A](implicit r: BoolRng[A]): BoolRng[A] = r +} diff --git a/algebra-core/src/main/scala/algebra/ring/CommutativeRig.scala b/algebra-core/src/main/scala/algebra/ring/CommutativeRig.scala new file mode 100644 index 0000000000..751b2d5950 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/CommutativeRig.scala @@ -0,0 +1,19 @@ +package algebra +package ring + +import scala.{specialized => sp} + +/** + * CommutativeRig is a Rig that is commutative under multiplication. + */ +trait CommutativeRig[@sp(Int, Long, Float, Double) A] + extends Any + with Rig[A] + with CommutativeSemiring[A] + with MultiplicativeCommutativeMonoid[A] + +object CommutativeRig + extends AdditiveMonoidFunctions[CommutativeRig] + with MultiplicativeMonoidFunctions[CommutativeRig] { + @inline final def apply[A](implicit r: CommutativeRig[A]): CommutativeRig[A] = r +} diff --git a/algebra-core/src/main/scala/algebra/ring/CommutativeRing.scala b/algebra-core/src/main/scala/algebra/ring/CommutativeRing.scala new file mode 100644 index 0000000000..59d03576c0 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/CommutativeRing.scala @@ -0,0 +1,17 @@ +package algebra +package ring + +import scala.{specialized => sp} + +/** + * CommutativeRing is a Ring that is commutative under multiplication. + */ +trait CommutativeRing[@sp(Int, Long, Float, Double) A] + extends Any + with Ring[A] + with CommutativeRig[A] + with CommutativeRng[A] + +object CommutativeRing extends RingFunctions[CommutativeRing] { + @inline final def apply[A](implicit r: CommutativeRing[A]): CommutativeRing[A] = r +} diff --git a/algebra-core/src/main/scala/algebra/ring/CommutativeRng.scala b/algebra-core/src/main/scala/algebra/ring/CommutativeRng.scala new file mode 100644 index 0000000000..f7ad06d4f4 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/CommutativeRng.scala @@ -0,0 +1,15 @@ +package algebra +package ring + +import scala.{specialized => sp} + +/** + * CommutativeRng is a Rng that is commutative under multiplication. + */ +trait CommutativeRng[@sp(Int, Long, Float, Double) A] extends Any with Rng[A] with CommutativeSemiring[A] + +object CommutativeRng + extends AdditiveGroupFunctions[CommutativeRng] + with MultiplicativeSemigroupFunctions[CommutativeRng] { + @inline final def apply[A](implicit r: CommutativeRng[A]): CommutativeRng[A] = r +} diff --git a/algebra-core/src/main/scala/algebra/ring/CommutativeSemiring.scala b/algebra-core/src/main/scala/algebra/ring/CommutativeSemiring.scala new file mode 100644 index 0000000000..571e674e12 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/CommutativeSemiring.scala @@ -0,0 +1,18 @@ +package algebra +package ring + +import scala.{specialized => sp} + +/** + * CommutativeSemiring is a Semiring that is commutative under multiplication. + */ +trait CommutativeSemiring[@sp(Int, Long, Float, Double) A] + extends Any + with Semiring[A] + with MultiplicativeCommutativeSemigroup[A] + +object CommutativeSemiring + extends AdditiveMonoidFunctions[CommutativeSemiring] + with MultiplicativeSemigroupFunctions[CommutativeSemiring] { + @inline final def apply[A](implicit r: CommutativeSemiring[A]): CommutativeSemiring[A] = r +} diff --git a/algebra-core/src/main/scala/algebra/ring/DivisionRing.scala b/algebra-core/src/main/scala/algebra/ring/DivisionRing.scala new file mode 100644 index 0000000000..88bbf3d2f5 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/DivisionRing.scala @@ -0,0 +1,30 @@ +package algebra +package ring + +import scala.{specialized => sp} + +trait DivisionRing[@sp(Byte, Short, Int, Long, Float, Double) A] extends Any with Ring[A] with MultiplicativeGroup[A] { + self => + + /** + * This is implemented in terms of basic Ring ops. However, this is + * probably significantly less efficient than can be done with a + * specific type. So, it is recommended that this method be + * overriden. + * + * This is possible because a Double is a rational number. + */ + def fromDouble(a: Double): A = DivisionRing.defaultFromDouble[A](a)(self, self) + +} + +trait DivisionRingFunctions[F[T] <: DivisionRing[T]] extends RingFunctions[F] with MultiplicativeGroupFunctions[F] { + def fromDouble[@sp(Int, Long, Float, Double) A](n: Double)(implicit ev: F[A]): A = + ev.fromDouble(n) +} + +object DivisionRing extends DivisionRingFunctions[DivisionRing] { + + @inline final def apply[A](implicit f: DivisionRing[A]): DivisionRing[A] = f + +} diff --git a/algebra-core/src/main/scala/algebra/ring/EuclideanRing.scala b/algebra-core/src/main/scala/algebra/ring/EuclideanRing.scala new file mode 100644 index 0000000000..450e34e90f --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/EuclideanRing.scala @@ -0,0 +1,55 @@ +package algebra +package ring + +import scala.annotation.tailrec +import scala.{specialized => sp} + +/** + * EuclideanRing implements a Euclidean domain. + * + * The formal definition says that every euclidean domain A has (at + * least one) euclidean function f: A -> N (the natural numbers) where: + * + * (for every x and non-zero y) x = yq + r, and r = 0 or f(r) < f(y). + * + * This generalizes the Euclidean division of integers, where f represents + * a measure of length (or absolute value), and the previous equation + * represents finding the quotient and remainder of x and y. So: + * + * quot(x, y) = q + * mod(x, y) = r + */ +trait EuclideanRing[@sp(Int, Long, Float, Double) A] extends Any with GCDRing[A] { self => + def euclideanFunction(a: A): BigInt + def equot(a: A, b: A): A + def emod(a: A, b: A): A + def equotmod(a: A, b: A): (A, A) = (equot(a, b), emod(a, b)) + def gcd(a: A, b: A)(implicit ev: Eq[A]): A = + EuclideanRing.euclid(a, b)(ev, self) + def lcm(a: A, b: A)(implicit ev: Eq[A]): A = + if (isZero(a) || isZero(b)) zero else times(equot(a, gcd(a, b)), b) +} + +trait EuclideanRingFunctions[R[T] <: EuclideanRing[T]] extends GCDRingFunctions[R] { + def euclideanFunction[@sp(Int, Long, Float, Double) A](a: A)(implicit ev: R[A]): BigInt = + ev.euclideanFunction(a) + def equot[@sp(Int, Long, Float, Double) A](a: A, b: A)(implicit ev: R[A]): A = + ev.equot(a, b) + def emod[@sp(Int, Long, Float, Double) A](a: A, b: A)(implicit ev: R[A]): A = + ev.emod(a, b) + def equotmod[@sp(Int, Long, Float, Double) A](a: A, b: A)(implicit ev: R[A]): (A, A) = + ev.equotmod(a, b) +} + +object EuclideanRing extends EuclideanRingFunctions[EuclideanRing] { + + @inline final def apply[A](implicit e: EuclideanRing[A]): EuclideanRing[A] = e + + /** + * Simple implementation of Euclid's algorithm for gcd + */ + @tailrec final def euclid[@sp(Int, Long, Float, Double) A: Eq: EuclideanRing](a: A, b: A): A = { + if (EuclideanRing[A].isZero(b)) a else euclid(b, EuclideanRing[A].emod(a, b)) + } + +} diff --git a/algebra-core/src/main/scala/algebra/ring/Field.scala b/algebra-core/src/main/scala/algebra/ring/Field.scala new file mode 100644 index 0000000000..fa809f682f --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/Field.scala @@ -0,0 +1,39 @@ +package algebra +package ring + +import scala.{specialized => sp} + +trait Field[@sp(Int, Long, Float, Double) A] + extends Any + with EuclideanRing[A] + with DivisionRing[A] + with MultiplicativeCommutativeGroup[A] { + self => + + // default implementations for GCD + + override def gcd(a: A, b: A)(implicit eqA: Eq[A]): A = + if (isZero(a) && isZero(b)) zero else one + override def lcm(a: A, b: A)(implicit eqA: Eq[A]): A = times(a, b) + + // default implementations for Euclidean division in a field (as every nonzero element is a unit!) + + def euclideanFunction(a: A): BigInt = BigInt(0) + def equot(a: A, b: A): A = div(a, b) + def emod(a: A, b: A): A = zero + override def equotmod(a: A, b: A): (A, A) = (div(a, b), zero) + + // needed for bin-compat + override def fromDouble(a: Double): A = + DivisionRing.defaultFromDouble[A](a)(self, self) + +} + +trait FieldFunctions[F[T] <: Field[T]] extends EuclideanRingFunctions[F] with MultiplicativeGroupFunctions[F] { + def fromDouble[@sp(Int, Long, Float, Double) A](n: Double)(implicit ev: F[A]): A = + ev.fromDouble(n) +} + +object Field extends FieldFunctions[Field] { + @inline final def apply[A](implicit ev: Field[A]): Field[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/ring/GCDRing.scala b/algebra-core/src/main/scala/algebra/ring/GCDRing.scala new file mode 100644 index 0000000000..a2031f36a4 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/GCDRing.scala @@ -0,0 +1,41 @@ +package algebra +package ring + +import scala.{specialized => sp} + +/** + * GCDRing implements a GCD ring. + * + * For two elements x and y in a GCD ring, we can choose two elements d and m + * such that: + * + * d = gcd(x, y) + * m = lcm(x, y) + * + * d * m = x * y + * + * Additionally, we require: + * + * gcd(0, 0) = 0 + * lcm(x, 0) = lcm(0, x) = 0 + * + * and commutativity: + * + * gcd(x, y) = gcd(y, x) + * lcm(x, y) = lcm(y, x) + */ +trait GCDRing[@sp(Int, Long, Float, Double) A] extends Any with CommutativeRing[A] { + def gcd(a: A, b: A)(implicit ev: Eq[A]): A + def lcm(a: A, b: A)(implicit ev: Eq[A]): A +} + +trait GCDRingFunctions[R[T] <: GCDRing[T]] extends RingFunctions[R] { + def gcd[@sp(Int, Long, Float, Double) A](a: A, b: A)(implicit ev: R[A], eqA: Eq[A]): A = + ev.gcd(a, b)(eqA) + def lcm[@sp(Int, Long, Float, Double) A](a: A, b: A)(implicit ev: R[A], eqA: Eq[A]): A = + ev.lcm(a, b)(eqA) +} + +object GCDRing extends GCDRingFunctions[GCDRing] { + @inline final def apply[A](implicit ev: GCDRing[A]): GCDRing[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/ring/Multiplicative.scala b/algebra-core/src/main/scala/algebra/ring/Multiplicative.scala new file mode 100644 index 0000000000..d9d51014eb --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/Multiplicative.scala @@ -0,0 +1,232 @@ +package algebra +package ring + +import scala.{specialized => sp} +import scala.annotation.{nowarn, tailrec} + +trait MultiplicativeSemigroup[@sp(Int, Long, Float, Double) A] extends Any with Serializable { + def multiplicative: Semigroup[A] = + new Semigroup[A] { + def combine(x: A, y: A): A = times(x, y) + } + + def times(x: A, y: A): A + + def pow(a: A, n: Int): A = + if (n > 0) positivePow(a, n) + else throw new IllegalArgumentException("Illegal non-positive exponent to pow: %s".format(n)) + + protected[this] def positivePow(a: A, n: Int): A = { + @tailrec def loop(b: A, k: Int, extra: A): A = + if (k == 1) times(b, extra) + else { + val x = if ((k & 1) == 1) times(b, extra) else extra + loop(times(b, b), k >>> 1, x) + } + if (n == 1) a else loop(a, n - 1, a) + } + + /** + * Given a sequence of `as`, combine them and return the total. + * + * If the sequence is empty, returns None. Otherwise, returns Some(total). + */ + @nowarn("msg=deprecated") + def tryProduct(as: TraversableOnce[A]): Option[A] = + as.toIterator.reduceOption(times) +} + +trait MultiplicativeCommutativeSemigroup[@sp(Int, Long, Float, Double) A] extends Any with MultiplicativeSemigroup[A] { + override def multiplicative: CommutativeSemigroup[A] = new CommutativeSemigroup[A] { + def combine(x: A, y: A): A = times(x, y) + } +} + +trait MultiplicativeMonoid[@sp(Int, Long, Float, Double) A] extends Any with MultiplicativeSemigroup[A] { + override def multiplicative: Monoid[A] = new Monoid[A] { + def empty = one + def combine(x: A, y: A): A = times(x, y) + } + + def one: A + + /** + * Tests if `a` is one. + */ + def isOne(a: A)(implicit ev: Eq[A]): Boolean = ev.eqv(a, one) + + override def pow(a: A, n: Int): A = + if (n > 0) positivePow(a, n) + else if (n == 0) one + else throw new IllegalArgumentException("Illegal negative exponent to pow: %s".format(n)) + + /** + * Given a sequence of `as`, compute the product. + */ + @nowarn("msg=deprecated") + def product(as: TraversableOnce[A]): A = + as.foldLeft(one)(times) + + @nowarn("msg=deprecated") + override def tryProduct(as: TraversableOnce[A]): Option[A] = + if (as.isEmpty) None else Some(product(as)) +} + +trait MultiplicativeCommutativeMonoid[@sp(Int, Long, Float, Double) A] + extends Any + with MultiplicativeMonoid[A] + with MultiplicativeCommutativeSemigroup[A] { + override def multiplicative: CommutativeMonoid[A] = new CommutativeMonoid[A] { + def empty = one + def combine(x: A, y: A): A = times(x, y) + } +} + +trait MultiplicativeGroup[@sp(Int, Long, Float, Double) A] extends Any with MultiplicativeMonoid[A] { + override def multiplicative: Group[A] = new Group[A] { + def empty = one + def combine(x: A, y: A): A = times(x, y) + override def remove(x: A, y: A): A = div(x, y) + def inverse(x: A): A = reciprocal(x) + } + + def reciprocal(x: A): A = div(one, x) + def div(x: A, y: A): A + + override def pow(a: A, n: Int): A = + if (n > 0) positivePow(a, n) + else if (n == 0) one + else if (n == Int.MinValue) positivePow(reciprocal(times(a, a)), 1073741824) + else positivePow(reciprocal(a), -n) +} + +trait MultiplicativeCommutativeGroup[@sp(Int, Long, Float, Double) A] + extends Any + with MultiplicativeGroup[A] + with MultiplicativeCommutativeMonoid[A] { + override def multiplicative: CommutativeGroup[A] = new CommutativeGroup[A] { + def empty = one + def combine(x: A, y: A): A = times(x, y) + override def remove(x: A, y: A): A = div(x, y) + def inverse(x: A): A = reciprocal(x) + } +} + +trait MultiplicativeSemigroupFunctions[S[T] <: MultiplicativeSemigroup[T]] { + def isMultiplicativeCommutative[A](implicit ev: S[A]): Boolean = + ev.isInstanceOf[MultiplicativeCommutativeSemigroup[A]] + + def times[@sp(Int, Long, Float, Double) A](x: A, y: A)(implicit ev: S[A]): A = + ev.times(x, y) + def pow[@sp(Int, Long, Float, Double) A](a: A, n: Int)(implicit ev: S[A]): A = + ev.pow(a, n) + + @nowarn("msg=deprecated") + def tryProduct[A](as: TraversableOnce[A])(implicit ev: S[A]): Option[A] = + ev.tryProduct(as) +} + +trait MultiplicativeMonoidFunctions[M[T] <: MultiplicativeMonoid[T]] extends MultiplicativeSemigroupFunctions[M] { + def one[@sp(Int, Long, Float, Double) A](implicit ev: M[A]): A = + ev.one + + def isOne[@sp(Int, Long, Float, Double) A](a: A)(implicit ev0: M[A], ev1: Eq[A]): Boolean = + ev0.isOne(a) + + @nowarn("msg=deprecated") + def product[@sp(Int, Long, Float, Double) A](as: TraversableOnce[A])(implicit ev: M[A]): A = + ev.product(as) +} + +trait MultiplicativeGroupFunctions[G[T] <: MultiplicativeGroup[T]] extends MultiplicativeMonoidFunctions[G] { + def reciprocal[@sp(Int, Long, Float, Double) A](x: A)(implicit ev: G[A]): A = + ev.reciprocal(x) + def div[@sp(Int, Long, Float, Double) A](x: A, y: A)(implicit ev: G[A]): A = + ev.div(x, y) +} + +object MultiplicativeSemigroup extends MultiplicativeSemigroupFunctions[MultiplicativeSemigroup] { + @inline final def apply[A](implicit ev: MultiplicativeSemigroup[A]): MultiplicativeSemigroup[A] = ev + + /** + * This method converts a multiplicative instance into a generic + * instance. + * + * Given an implicit `MultiplicativeSemigroup[A]`, this method returns + * a `Semigroup[A]`. + */ + @inline final def multiplicative[A](implicit ev: MultiplicativeSemigroup[A]): Semigroup[A] = + ev.multiplicative +} + +object MultiplicativeCommutativeSemigroup extends MultiplicativeSemigroupFunctions[MultiplicativeCommutativeSemigroup] { + @inline final def apply[A](implicit + ev: MultiplicativeCommutativeSemigroup[A] + ): MultiplicativeCommutativeSemigroup[A] = ev + + /** + * This method converts a multiplicative instance into a generic + * instance. + * + * Given an implicit `MultiplicativeCommutativeSemigroup[A]`, this method returns + * a `CommutativeSemigroup[A]`. + */ + @inline final def multiplicative[A](implicit ev: MultiplicativeCommutativeSemigroup[A]): CommutativeSemigroup[A] = + ev.multiplicative +} + +object MultiplicativeMonoid extends MultiplicativeMonoidFunctions[MultiplicativeMonoid] { + @inline final def apply[A](implicit ev: MultiplicativeMonoid[A]): MultiplicativeMonoid[A] = ev + + /** + * This method converts a multiplicative instance into a generic + * instance. + * + * Given an implicit `MultiplicativeMonoid[A]`, this method returns + * a `Monoid[A]`. + */ + @inline final def multiplicative[A](implicit ev: MultiplicativeMonoid[A]): Monoid[A] = + ev.multiplicative +} + +object MultiplicativeCommutativeMonoid extends MultiplicativeMonoidFunctions[MultiplicativeCommutativeMonoid] { + @inline final def apply[A](implicit ev: MultiplicativeCommutativeMonoid[A]): MultiplicativeCommutativeMonoid[A] = ev + + /** + * This method converts a multiplicative instance into a generic + * instance. + * + * Given an implicit `MultiplicativeCommutativeMonoid[A]`, this method returns + * a `CommutativeMonoid[A]`. + */ + @inline final def multiplicative[A](implicit ev: MultiplicativeCommutativeMonoid[A]): CommutativeMonoid[A] = + ev.multiplicative +} + +object MultiplicativeGroup extends MultiplicativeGroupFunctions[MultiplicativeGroup] { + @inline final def apply[A](implicit ev: MultiplicativeGroup[A]): MultiplicativeGroup[A] = ev + + /** + * This method converts a multiplicative instance into a generic + * instance. + * + * Given an implicit `MultiplicativeGroup[A]`, this method returns + * a `Group[A]`. + */ + @inline final def multiplicative[A](implicit ev: MultiplicativeGroup[A]): Group[A] = + ev.multiplicative +} + +object MultiplicativeCommutativeGroup extends MultiplicativeGroupFunctions[MultiplicativeCommutativeGroup] { + @inline final def apply[A](implicit ev: MultiplicativeCommutativeGroup[A]): MultiplicativeCommutativeGroup[A] = ev + + /** + * This method converts a multiplicative instance into a generic + * instance. + * + * Given an implicit `MultiplicativeCommutativeGroup[A]`, this method returns + * a `CommutativeGroup[A]`. + */ + @inline final def multiplicative[A](implicit ev: MultiplicativeCommutativeGroup[A]): CommutativeGroup[A] = + ev.multiplicative +} diff --git a/algebra-core/src/main/scala/algebra/ring/Rig.scala b/algebra-core/src/main/scala/algebra/ring/Rig.scala new file mode 100644 index 0000000000..532866acaf --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/Rig.scala @@ -0,0 +1,22 @@ +package algebra +package ring + +import scala.{specialized => sp} + +/** + * Rig consists of: + * + * - a commutative monoid for addition (+) + * - a monoid for multiplication (*) + * + * Alternately, a Rig can be thought of as a ring without + * multiplicative or additive inverses (or as a semiring with a + * multiplicative identity). + * + * Mnemonic: "Rig is a Ring without 'N'egation." + */ +trait Rig[@sp(Int, Long, Float, Double) A] extends Any with Semiring[A] with MultiplicativeMonoid[A] + +object Rig extends AdditiveMonoidFunctions[Rig] with MultiplicativeMonoidFunctions[Rig] { + @inline final def apply[A](implicit ev: Rig[A]): Rig[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/ring/Ring.scala b/algebra-core/src/main/scala/algebra/ring/Ring.scala new file mode 100644 index 0000000000..6606817046 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/Ring.scala @@ -0,0 +1,118 @@ +package algebra +package ring + +import scala.{specialized => sp} +import scala.annotation.tailrec + +/** + * Ring consists of: + * + * - a commutative group for addition (+) + * - a monoid for multiplication (*) + * + * Additionally, multiplication must distribute over addition. + * + * Ring implements some methods (for example fromInt) in terms of + * other more fundamental methods (zero, one and plus). Where + * possible, these methods should be overridden by more efficient + * implementations. + */ +trait Ring[@sp(Int, Long, Float, Double) A] extends Any with Rig[A] with Rng[A] { + + /** + * Convert the given integer to an instance of A. + * + * Defined to be equivalent to `sumN(one, n)`. + * + * That is, `n` repeated summations of this ring's `one`, or `-n` + * summations of `-one` if `n` is negative. + * + * Most type class instances should consider overriding this method + * for performance reasons. + */ + def fromInt(n: Int): A = sumN(one, n) + + /** + * Convert the given BigInt to an instance of A. + * + * This is equivalent to `n` repeated summations of this ring's `one`, or + * `-n` summations of `-one` if `n` is negative. + * + * Most type class instances should consider overriding this method for + * performance reasons. + */ + def fromBigInt(n: BigInt): A = Ring.defaultFromBigInt(n)(this) +} + +trait RingFunctions[R[T] <: Ring[T]] extends AdditiveGroupFunctions[R] with MultiplicativeMonoidFunctions[R] { + def fromInt[@sp(Int, Long, Float, Double) A](n: Int)(implicit ev: R[A]): A = + ev.fromInt(n) + + def fromBigInt[@sp(Int, Long, Float, Double) A](n: BigInt)(implicit ev: R[A]): A = + ev.fromBigInt(n) + + final def defaultFromBigInt[@sp(Int, Long, Float, Double) A](n: BigInt)(implicit ev: R[A]): A = { + if (n.isValidInt) { + ev.fromInt(n.toInt) + } else { + val d = ev.fromInt(1 << 30) + val mask = (1L << 30) - 1 + @tailrec def loop(k: A, x: BigInt, acc: A): A = + if (x.isValidInt) { + ev.plus(ev.times(k, ev.fromInt(x.toInt)), acc) + } else { + val y = x >> 30 + val r = ev.fromInt((x & mask).toInt) + loop(ev.times(d, k), y, ev.plus(ev.times(k, r), acc)) + } + + val absValue = loop(one, n.abs, zero) + if (n.signum < 0) ev.negate(absValue) else absValue + } + } + + /** + * Returns the given Double, understood as a rational number, in the provided + * (division) ring. + * + * This is implemented in terms of basic ops. However, this is + * probably significantly less efficient than can be done with a specific + * type. So, it is recommended to specialize this general method. + */ + final def defaultFromDouble[A](a: Double)(implicit ringA: Ring[A], mgA: MultiplicativeGroup[A]): A = + if (a == 0.0) ringA.zero + else if (a.isValidInt) ringA.fromInt(a.toInt) + else { + import java.lang.Double.{doubleToLongBits, isInfinite, isNaN} + import java.lang.Long.numberOfTrailingZeros + require(!isInfinite(a) && !isNaN(a), "Double must be representable as a fraction.") + val bits = doubleToLongBits(a) + val expBits = ((bits >> 52) & 0x7ff).toInt + val mBits = bits & 0x000fffffffffffffL + // If expBits is 0, then this is a subnormal and we drop the implicit + // 1 bit. + val m = if (expBits > 0) mBits | 0x0010000000000000L else mBits + val zeros = numberOfTrailingZeros(m) + val value = m >>> zeros + // If expBits is 0, then this is a subnormal with expBits = 1. + val exp = math.max(1, expBits) - 1075 + zeros // 1023 + 52 + + val high = ringA.times(ringA.fromInt((value >>> 30).toInt), ringA.fromInt(1 << 30)) + val low = ringA.fromInt((value & 0x3fffffff).toInt) + val num = ringA.plus(high, low) + val unsigned = if (exp > 0) { + ringA.times(num, ringA.pow(ringA.fromInt(2), exp)) + } else if (exp < 0) { + mgA.div(num, ringA.pow(ringA.fromInt(2), -exp)) + } else { + num + } + + if (a < 0) ringA.negate(unsigned) else unsigned + } + +} + +object Ring extends RingFunctions[Ring] { + @inline final def apply[A](implicit ev: Ring[A]): Ring[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/ring/Rng.scala b/algebra-core/src/main/scala/algebra/ring/Rng.scala new file mode 100644 index 0000000000..94585327b5 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/Rng.scala @@ -0,0 +1,22 @@ +package algebra +package ring + +import scala.{specialized => sp} + +/** + * Rng (pronounced "Rung") consists of: + * + * - a commutative group for addition (+) + * - a semigroup for multiplication (*) + * + * Alternately, a Rng can be thought of as a ring without a + * multiplicative identity (or as a semiring with an additive + * inverse). + * + * Mnemonic: "Rng is a Ring without multiplicative 'I'dentity." + */ +trait Rng[@sp(Int, Long, Float, Double) A] extends Any with Semiring[A] with AdditiveCommutativeGroup[A] + +object Rng extends AdditiveGroupFunctions[Rng] with MultiplicativeSemigroupFunctions[Rng] { + @inline final def apply[A](implicit ev: Rng[A]): Rng[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/ring/Semiring.scala b/algebra-core/src/main/scala/algebra/ring/Semiring.scala new file mode 100644 index 0000000000..ef9ca42420 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/Semiring.scala @@ -0,0 +1,26 @@ +package algebra +package ring + +import scala.{specialized => sp} + +/** + * Semiring consists of: + * + * - a commutative monoid for addition (+) + * - a semigroup for multiplication (*) + * + * Alternately, a Semiring can be thought of as a ring without a + * multiplicative identity or an additive inverse. + * + * A Semiring with an additive inverse (-) is a Rng. + * A Semiring with a multiplicative identity (1) is a Rig. + * A Semiring with both of those is a Ring. + */ +trait Semiring[@sp(Int, Long, Float, Double) A] + extends Any + with AdditiveCommutativeMonoid[A] + with MultiplicativeSemigroup[A] + +object Semiring extends AdditiveMonoidFunctions[Semiring] with MultiplicativeSemigroupFunctions[Semiring] { + @inline final def apply[A](implicit ev: Semiring[A]): Semiring[A] = ev +} diff --git a/algebra-core/src/main/scala/algebra/ring/Signed.scala b/algebra-core/src/main/scala/algebra/ring/Signed.scala new file mode 100644 index 0000000000..1326cb24a3 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/Signed.scala @@ -0,0 +1,152 @@ +package algebra.ring + +import algebra.{CommutativeMonoid, Eq, Order} + +import scala.{specialized => sp} + +/** + * A trait that expresses the existence of signs and absolute values on linearly ordered additive commutative monoids + * (i.e. types with addition and a zero). + * + * The following laws holds: + * + * (1) if `a <= b` then `a + c <= b + c` (linear order), + * (2) `signum(x) = -1` if `x < 0`, `signum(x) = 1` if `x > 0`, `signum(x) = 0` otherwise, + * + * Negative elements only appear when the scalar is taken from a additive abelian group. Then: + * + * (3) `abs(x) = -x` if `x < 0`, or `x` otherwise, + * + * Laws (1) and (2) lead to the triange inequality: + * + * (4) `abs(a + b) <= abs(a) + abs(b)` + * + * Signed should never be extended in implementations, rather the [[Signed.forAdditiveCommutativeMonoid]] and + * [[Signed.forAdditiveCommutativeGroup subtraits]]. + * + * It's better to have the Signed hierarchy separate from the Ring/Order hierarchy, so that + * we do not end up with duplicate implicits. + */ +trait Signed[@sp(Byte, Short, Int, Long, Float, Double) A] extends Any { + + def additiveCommutativeMonoid: AdditiveCommutativeMonoid[A] + def order: Order[A] + + /** + * Returns Zero if `a` is 0, Positive if `a` is positive, and Negative is `a` is negative. + */ + def sign(a: A): Signed.Sign = Signed.Sign(signum(a)) + + /** + * Returns 0 if `a` is 0, 1 if `a` is positive, and -1 is `a` is negative. + */ + def signum(a: A): Int + + /** + * An idempotent function that ensures an object has a non-negative sign. + */ + def abs(a: A): A + + def isSignZero(a: A): Boolean = signum(a) == 0 + def isSignPositive(a: A): Boolean = signum(a) > 0 + def isSignNegative(a: A): Boolean = signum(a) < 0 + + def isSignNonZero(a: A): Boolean = signum(a) != 0 + def isSignNonPositive(a: A): Boolean = signum(a) <= 0 + def isSignNonNegative(a: A): Boolean = signum(a) >= 0 +} + +trait SignedFunctions[S[T] <: Signed[T]] extends cats.kernel.OrderFunctions[Order] { + def sign[@sp(Int, Long, Float, Double) A](a: A)(implicit ev: S[A]): Signed.Sign = + ev.sign(a) + def signum[@sp(Int, Long, Float, Double) A](a: A)(implicit ev: S[A]): Int = + ev.signum(a) + def abs[@sp(Int, Long, Float, Double) A](a: A)(implicit ev: S[A]): A = + ev.abs(a) + def isSignZero[@sp(Int, Long, Float, Double) A](a: A)(implicit ev: S[A]): Boolean = + ev.isSignZero(a) + def isSignPositive[@sp(Int, Long, Float, Double) A](a: A)(implicit ev: S[A]): Boolean = + ev.isSignPositive(a) + def isSignNegative[@sp(Int, Long, Float, Double) A](a: A)(implicit ev: S[A]): Boolean = + ev.isSignNegative(a) + def isSignNonZero[@sp(Int, Long, Float, Double) A](a: A)(implicit ev: S[A]): Boolean = + ev.isSignNonZero(a) + def isSignNonPositive[@sp(Int, Long, Float, Double) A](a: A)(implicit ev: S[A]): Boolean = + ev.isSignNonPositive(a) + def isSignNonNegative[@sp(Int, Long, Float, Double) A](a: A)(implicit ev: S[A]): Boolean = + ev.isSignNonNegative(a) +} + +object Signed extends SignedFunctions[Signed] { + + /** + * Signed implementation for additive commutative monoids + */ + trait forAdditiveCommutativeMonoid[A] extends Any with Signed[A] with AdditiveCommutativeMonoid[A] { + final override def additiveCommutativeMonoid = this + def signum(a: A): Int = { + val c = order.compare(a, zero) + if (c < 0) -1 + else if (c > 0) 1 + else 0 + } + } + + /** + * Signed implementation for additive commutative groups + */ + trait forAdditiveCommutativeGroup[A] + extends Any + with forAdditiveCommutativeMonoid[A] + with AdditiveCommutativeGroup[A] { + def abs(a: A): A = if (order.compare(a, zero) < 0) negate(a) else a + } + + def apply[A](implicit s: Signed[A]): Signed[A] = s + + /** + * A simple ADT representing the `Sign` of an object. + */ + sealed abstract class Sign(val toInt: Int) { + def unary_- : Sign = this match { + case Positive => Negative + case Negative => Positive + case Zero => Zero + } + + def *(that: Sign): Sign = Sign(this.toInt * that.toInt) + + def **(that: Int): Sign = this match { + case Positive => Positive + case Zero if that == 0 => Positive + case Zero => Zero + case Negative if (that % 2) == 0 => Positive + case Negative => Negative + } + } + + case object Zero extends Sign(0) + case object Positive extends Sign(1) + case object Negative extends Sign(-1) + + object Sign { + implicit def sign2int(s: Sign): Int = s.toInt + + def apply(i: Int): Sign = + if (i == 0) Zero else if (i > 0) Positive else Negative + + private val instance: CommutativeMonoid[Sign] with MultiplicativeCommutativeMonoid[Sign] with Eq[Sign] = + new CommutativeMonoid[Sign] with MultiplicativeCommutativeMonoid[Sign] with Eq[Sign] { + def eqv(x: Sign, y: Sign): Boolean = x == y + def empty: Sign = Positive + def combine(x: Sign, y: Sign): Sign = x * y + def one: Sign = Positive + def times(x: Sign, y: Sign): Sign = x * y + } + + implicit final def signMultiplicativeMonoid: MultiplicativeCommutativeMonoid[Sign] = instance + implicit final def signMonoid: CommutativeMonoid[Sign] = instance + implicit final def signEq: Eq[Sign] = instance + } + +} diff --git a/algebra-core/src/main/scala/algebra/ring/TruncatedDivision.scala b/algebra-core/src/main/scala/algebra/ring/TruncatedDivision.scala new file mode 100644 index 0000000000..443bcdb387 --- /dev/null +++ b/algebra-core/src/main/scala/algebra/ring/TruncatedDivision.scala @@ -0,0 +1,86 @@ +package algebra.ring + +import scala.{specialized => sp} + +/** + * Division and modulus for computer scientists + * taken from https://www.microsoft.com/en-us/research/wp-content/uploads/2016/02/divmodnote-letter.pdf + * + * For two numbers x (dividend) and y (divisor) on an ordered ring with y != 0, + * there exists a pair of numbers q (quotient) and r (remainder) + * such that these laws are satisfied: + * + * (1) q is an integer + * (2) x = y * q + r (division rule) + * (3) |r| < |y|, + * (4t) r = 0 or sign(r) = sign(x), + * (4f) r = 0 or sign(r) = sign(y). + * + * where sign is the sign function, and the absolute value + * function |x| is defined as |x| = x if x >=0, and |x| = -x otherwise. + * + * We define functions tmod and tquot such that: + * q = tquot(x, y) and r = tmod(x, y) obey rule (4t), + * (which truncates effectively towards zero) + * and functions fmod and fquot such that: + * q = fquot(x, y) and r = fmod(x, y) obey rule (4f) + * (which floors the quotient and effectively rounds towards negative infinity). + * + * Law (4t) corresponds to ISO C99 and Haskell's quot/rem. + * Law (4f) is described by Knuth and used by Haskell, + * and fmod corresponds to the REM function of the IEEE floating-point standard. + */ +trait TruncatedDivision[@sp(Byte, Short, Int, Long, Float, Double) A] extends Any with Signed[A] { + def tquot(x: A, y: A): A + def tmod(x: A, y: A): A + def tquotmod(x: A, y: A): (A, A) = (tquot(x, y), tmod(x, y)) + + def fquot(x: A, y: A): A + def fmod(x: A, y: A): A + def fquotmod(x: A, y: A): (A, A) = (fquot(x, y), fmod(x, y)) +} + +trait TruncatedDivisionFunctions[S[T] <: TruncatedDivision[T]] extends SignedFunctions[S] { + def tquot[@sp(Int, Long, Float, Double) A](x: A, y: A)(implicit ev: TruncatedDivision[A]): A = + ev.tquot(x, y) + def tmod[@sp(Int, Long, Float, Double) A](x: A, y: A)(implicit ev: TruncatedDivision[A]): A = + ev.tmod(x, y) + def tquotmod[@sp(Int, Long, Float, Double) A](x: A, y: A)(implicit ev: TruncatedDivision[A]): (A, A) = + ev.tquotmod(x, y) + def fquot[@sp(Int, Long, Float, Double) A](x: A, y: A)(implicit ev: TruncatedDivision[A]): A = + ev.fquot(x, y) + def fmod[@sp(Int, Long, Float, Double) A](x: A, y: A)(implicit ev: TruncatedDivision[A]): A = + ev.fmod(x, y) + def fquotmod[@sp(Int, Long, Float, Double) A](x: A, y: A)(implicit ev: TruncatedDivision[A]): (A, A) = + ev.fquotmod(x, y) +} + +object TruncatedDivision extends TruncatedDivisionFunctions[TruncatedDivision] { + trait forCommutativeRing[@sp(Byte, Short, Int, Long, Float, Double) A] + extends Any + with TruncatedDivision[A] + with Signed.forAdditiveCommutativeGroup[A] + with CommutativeRing[A] { self => + + def fmod(x: A, y: A): A = { + val tm = tmod(x, y) + if (signum(tm) == -signum(y)) plus(tm, y) else tm + } + + def fquot(x: A, y: A): A = { + val (tq, tm) = tquotmod(x, y) + if (signum(tm) == -signum(y)) minus(tq, one) else tq + } + + override def fquotmod(x: A, y: A): (A, A) = { + val (tq, tm) = tquotmod(x, y) + val signsDiffer = signum(tm) == -signum(y) + val fq = if (signsDiffer) minus(tq, one) else tq + val fm = if (signsDiffer) plus(tm, y) else tm + (fq, fm) + } + + } + + def apply[A](implicit ev: TruncatedDivision[A]): TruncatedDivision[A] = ev +} diff --git a/algebra-core/src/test/scala/algebra/Instances.scala b/algebra-core/src/test/scala/algebra/Instances.scala new file mode 100644 index 0000000000..b7dd2e1854 --- /dev/null +++ b/algebra-core/src/test/scala/algebra/Instances.scala @@ -0,0 +1,32 @@ +package algebra + +object Instances { + + def t2HasSemigroup[A, B](implicit eva: Semigroup[A], evb: Semigroup[B]) = + new Semigroup[(A, B)] { + def combine(x: (A, B), y: (A, B)): (A, B) = + (eva.combine(x._1, y._1), evb.combine(x._2, y._2)) + } + + val stringHasMonoid = + new Monoid[String] { + def empty: String = "" + def combine(x: String, y: String): String = x + y + } + + def f1ComposeMonoid[A] = + new Monoid[A => A] { + def empty: A => A = + a => a + def combine(x: A => A, y: A => A): A => A = + a => y(x(a)) + } + + def f1HomomorphismMonoid[A, B](implicit ev: Monoid[B]) = + new Monoid[A => B] { + def empty: A => B = + _ => ev.empty + def combine(x: A => B, y: A => B): A => B = + a => ev.combine(x(a), y(a)) + } +} diff --git a/algebra-core/src/test/scala/algebra/ring/RingTest.scala b/algebra-core/src/test/scala/algebra/ring/RingTest.scala new file mode 100644 index 0000000000..865a3e56f8 --- /dev/null +++ b/algebra-core/src/test/scala/algebra/ring/RingTest.scala @@ -0,0 +1,13 @@ +package algebra.ring + +import algebra.instances.bigInt._ + +import org.scalacheck.Prop._ + +class RingTest extends munit.DisciplineSuite { + test("Ring.defaultFromBigInt") { + forAll { (n: BigInt) => + assertEquals(Ring.defaultFromBigInt[BigInt](n), n) + } + } +} diff --git a/algebra-laws/js/src/main/scala/algebra/laws/platform/Platform.scala b/algebra-laws/js/src/main/scala/algebra/laws/platform/Platform.scala new file mode 100644 index 0000000000..adeea18df3 --- /dev/null +++ b/algebra-laws/js/src/main/scala/algebra/laws/platform/Platform.scala @@ -0,0 +1,10 @@ +package algebra.laws.platform + +private[laws] object Platform { + // using `final val` makes compiler constant-fold any use of these values, dropping dead code automatically + // $COVERAGE-OFF$ + final val isJvm = false + final val isJs = true + final val isNative = false + // $COVERAGE-ON$ +} diff --git a/algebra-laws/jvm/src/main/scala/algebra/laws/platform/Platform.scala b/algebra-laws/jvm/src/main/scala/algebra/laws/platform/Platform.scala new file mode 100644 index 0000000000..e4a14708bc --- /dev/null +++ b/algebra-laws/jvm/src/main/scala/algebra/laws/platform/Platform.scala @@ -0,0 +1,10 @@ +package algebra.laws.platform + +private[laws] object Platform { + // using `final val` makes compiler constant-fold any use of these values, dropping dead code automatically + // $COVERAGE-OFF$ + final val isJvm = true + final val isJs = false + final val isNative = false + // $COVERAGE-ON$ +} diff --git a/algebra-laws/native/src/main/scala/algebra/laws/platform/Platform.scala b/algebra-laws/native/src/main/scala/algebra/laws/platform/Platform.scala new file mode 100644 index 0000000000..dac1c22e89 --- /dev/null +++ b/algebra-laws/native/src/main/scala/algebra/laws/platform/Platform.scala @@ -0,0 +1,10 @@ +package algebra.laws.platform + +private[laws] object Platform { + // using `final val` makes compiler constant-fold any use of these values, dropping dead code automatically + // $COVERAGE-OFF$ + final val isJvm = false + final val isJs = false + final val isNative = true + // $COVERAGE-ON$ +} diff --git a/algebra-laws/shared/src/main/scala/algebra/laws/BaseLaws.scala b/algebra-laws/shared/src/main/scala/algebra/laws/BaseLaws.scala new file mode 100644 index 0000000000..11b2c89a3a --- /dev/null +++ b/algebra-laws/shared/src/main/scala/algebra/laws/BaseLaws.scala @@ -0,0 +1,30 @@ +package algebra.laws + +import cats.kernel._ + +import org.typelevel.discipline.Laws + +import org.scalacheck.{Arbitrary, Prop} + +@deprecated("No replacement", since = "2.7.0") +object BaseLaws { + def apply[A: Eq: Arbitrary]: BaseLaws[A] = new BaseLaws[A] { + def Equ = Eq[A] + def Arb = implicitly[Arbitrary[A]] + } +} + +@deprecated("No replacement", since = "2.7.0") +trait BaseLaws[A] extends Laws { + + implicit def Equ: Eq[A] + implicit def Arb: Arbitrary[A] + + class BaseRuleSet( + val name: String, + val parent: Option[RuleSet], + val bases: Seq[(String, Laws#RuleSet)], + val props: (String, Prop)* + ) extends RuleSet + with HasOneParent +} diff --git a/algebra-laws/shared/src/main/scala/algebra/laws/CheckSupport.scala b/algebra-laws/shared/src/main/scala/algebra/laws/CheckSupport.scala new file mode 100644 index 0000000000..6f95634f97 --- /dev/null +++ b/algebra-laws/shared/src/main/scala/algebra/laws/CheckSupport.scala @@ -0,0 +1,12 @@ +package algebra.laws + +/** + * This object contains Arbitrary instances for types defined in + * algebra.instances, as well as anything else we'd like to import to assist + * in running ScalaCheck tests. + * + * (Since algebra-instances has no dependencies, its types can't + * define Arbitrary instances in companions.) + */ +@deprecated("No replacement", since = "2.7.0") +object CheckSupport {} diff --git a/algebra-laws/shared/src/main/scala/algebra/laws/DeMorganLaws.scala b/algebra-laws/shared/src/main/scala/algebra/laws/DeMorganLaws.scala new file mode 100644 index 0000000000..cd75fd8092 --- /dev/null +++ b/algebra-laws/shared/src/main/scala/algebra/laws/DeMorganLaws.scala @@ -0,0 +1,51 @@ +package algebra.laws + +import algebra._ +import algebra.lattice._ +import org.scalacheck.{Arbitrary, Prop} +import org.scalacheck.Prop._ +import org.typelevel.discipline.Laws + +@deprecated("Laws moved to LogicLaws", since = "2.7.0") +object DeMorganLaws { + def apply[A: Eq: Arbitrary: LatticeLaws] = new DeMorganLaws[A] { + def Equ = Eq[A] + def Arb = implicitly[Arbitrary[A]] + def LL = implicitly[LatticeLaws[A]] + } +} + +@deprecated("Laws moved to LogicLaws", since = "2.7.0") +trait DeMorganLaws[A] extends Laws { + + implicit def Equ: Eq[A] + implicit def Arb: Arbitrary[A] + def LL: LatticeLaws[A] + + def logic(implicit A: Logic[A]) = new DeMorganProperties( + name = "logic", + parents = Seq(), + ll = LL.boundedDistributiveLattice, + Rules.distributive(A.or)(A.and), + "¬(x∨y) = ¬x∧¬y" -> forAll { (x: A, y: A) => A.not(A.or(x, y)) ?== A.and(A.not(x), A.not(y)) }, + "¬(x∧y) = ¬¬(¬x∨¬y)" -> forAll { (x: A, y: A) => A.not(A.and(x, y)) ?== A.not(A.not(A.or(A.not(x), A.not(y)))) } + ) + + def deMorgan(implicit A: DeMorgan[A]) = new DeMorganProperties( + name = "deMorgan", + parents = Seq(logic), + ll = LL.boundedDistributiveLattice, + Rules.distributive(A.or)(A.and), + "involutive" -> forAll { (x: A) => A.not(A.not(x)) ?== x } + ) + + class DeMorganProperties( + val name: String, + val parents: Seq[DeMorganProperties], + val ll: LatticeLaws[A]#LatticeProperties, + val props: (String, Prop)* + ) extends RuleSet { + val bases = Seq("lattice" -> ll) + } + +} diff --git a/algebra-laws/shared/src/main/scala/algebra/laws/GroupLaws.scala b/algebra-laws/shared/src/main/scala/algebra/laws/GroupLaws.scala new file mode 100644 index 0000000000..14fac2e575 --- /dev/null +++ b/algebra-laws/shared/src/main/scala/algebra/laws/GroupLaws.scala @@ -0,0 +1,102 @@ +package algebra.laws + +import cats.kernel._ +import cats.kernel.instances.option._ + +import org.typelevel.discipline.Laws + +import org.scalacheck.{Arbitrary, Prop} +import org.scalacheck.Prop._ + +@deprecated("Provided by cats.kernel.laws", since = "2.7.0") +object GroupLaws { + def apply[A: Eq: Arbitrary]: GroupLaws[A] = new GroupLaws[A] { + def Equ = Eq[A] + def Arb = implicitly[Arbitrary[A]] + } +} + +@deprecated("Provided by cats.kernel.laws", since = "2.7.0") +trait GroupLaws[A] extends Laws { + + implicit def Equ: Eq[A] + implicit def Arb: Arbitrary[A] + + // groups + + def semigroup(implicit A: Semigroup[A]): GroupProperties = new GroupProperties( + name = "semigroup", + parents = Nil, + Rules.serializable(A), + Rules.associativity(A.combine), + Rules.repeat1("combineN")(A.combineN), + Rules.repeat2("combineN", "|+|")(A.combineN)(A.combine), + "combineAllOption" -> forAll { (xs: Vector[A]) => + A.combineAllOption(xs) ?== xs.reduceOption(A.combine) + } + ) + + def band(implicit A: Band[A]): GroupProperties = new GroupProperties( + name = "band", + parents = List(semigroup), + Rules.idempotence(A.combine), + "isIdempotent" -> Semigroup.isIdempotent[A] + ) + + def commutativeSemigroup(implicit A: CommutativeSemigroup[A]): GroupProperties = new GroupProperties( + name = "commutative semigroup", + parents = List(semigroup), + Rules.commutative(A.combine) + ) + + def semilattice(implicit A: Semilattice[A]): GroupProperties = new GroupProperties( + name = "semilattice", + parents = List(band, commutativeSemigroup) + ) + + def monoid(implicit A: Monoid[A]): GroupProperties = new GroupProperties( + name = "monoid", + parents = List(semigroup), + Rules.leftIdentity(A.empty)(A.combine), + Rules.rightIdentity(A.empty)(A.combine), + Rules.repeat0("combineN", "id", A.empty)(A.combineN), + Rules.collect0("combineAll", "id", A.empty)(A.combineAll), + Rules.isId("isEmpty", A.empty)(A.isEmpty), + "combineAll" -> forAll { (xs: Vector[A]) => + A.combineAll(xs) ?== (A.empty +: xs).reduce(A.combine) + } + ) + + def commutativeMonoid(implicit A: CommutativeMonoid[A]): GroupProperties = new GroupProperties( + name = "commutative monoid", + parents = List(monoid, commutativeSemigroup) + ) + + def boundedSemilattice(implicit A: BoundedSemilattice[A]): GroupProperties = new GroupProperties( + name = "boundedSemilattice", + parents = List(commutativeMonoid, semilattice) + ) + + def group(implicit A: Group[A]): GroupProperties = new GroupProperties( + name = "group", + parents = List(monoid), + Rules.leftInverse(A.empty)(A.combine)(A.inverse), + Rules.rightInverse(A.empty)(A.combine)(A.inverse), + Rules.consistentInverse("remove")(A.remove)(A.combine)(A.inverse) + ) + + def commutativeGroup(implicit A: CommutativeGroup[A]): GroupProperties = new GroupProperties( + name = "commutative group", + parents = List(group, commutativeMonoid) + ) + + // property classes + + class GroupProperties( + val name: String, + val parents: Seq[GroupProperties], + val props: (String, Prop)* + ) extends RuleSet { + val bases = Nil + } +} diff --git a/algebra-laws/shared/src/main/scala/algebra/laws/LatticeLaws.scala b/algebra-laws/shared/src/main/scala/algebra/laws/LatticeLaws.scala new file mode 100644 index 0000000000..76212e1733 --- /dev/null +++ b/algebra-laws/shared/src/main/scala/algebra/laws/LatticeLaws.scala @@ -0,0 +1,115 @@ +package algebra.laws + +import algebra._ +import algebra.lattice._ + +import org.scalacheck.{Arbitrary, Prop} +import org.scalacheck.Prop._ +import scala.annotation.nowarn + +object LatticeLaws { + def apply[A: Eq: Arbitrary] = new LatticeLaws[A] { + def Equ = Eq[A] + def Arb = implicitly[Arbitrary[A]] + } +} + +@nowarn("msg=deprecated") +trait LatticeLaws[A] extends GroupLaws[A] { + + implicit def Equ: Eq[A] + implicit def Arb: Arbitrary[A] + + def joinSemilattice(implicit A: JoinSemilattice[A]) = new LatticeProperties( + name = "joinSemilattice", + parents = Nil, + join = Some(semilattice(A.joinSemilattice)), + meet = None, + Rules.serializable(A) + ) + + def meetSemilattice(implicit A: MeetSemilattice[A]) = new LatticeProperties( + name = "meetSemilattice", + parents = Nil, + join = None, + meet = Some(semilattice(A.meetSemilattice)), + Rules.serializable(A) + ) + + def lattice(implicit A: Lattice[A]) = new LatticeProperties( + name = "lattice", + parents = Seq(joinSemilattice, meetSemilattice), + join = Some(semilattice(A.joinSemilattice)), + meet = Some(semilattice(A.meetSemilattice)), + "absorption" -> forAll { (x: A, y: A) => + (A.join(x, A.meet(x, y)) ?== x) && (A.meet(x, A.join(x, y)) ?== x) + } + ) + + def distributiveLattice(implicit A: DistributiveLattice[A]) = new LatticeProperties( + name = "distributiveLattice", + parents = Seq(lattice), + join = Some(semilattice(A.joinSemilattice)), + meet = Some(semilattice(A.meetSemilattice)), + "distributive" -> forAll { (x: A, y: A, z: A) => + (A.join(x, A.meet(y, z)) ?== A.meet(A.join(x, y), A.join(x, z))) && + (A.meet(x, A.join(y, z)) ?== A.join(A.meet(x, y), A.meet(x, z))) + } + ) + + def boundedJoinSemilattice(implicit A: BoundedJoinSemilattice[A]) = new LatticeProperties( + name = "boundedJoinSemilattice", + parents = Seq(joinSemilattice), + join = Some(boundedSemilattice(A.joinSemilattice)), + meet = None + ) + + def boundedMeetSemilattice(implicit A: BoundedMeetSemilattice[A]) = new LatticeProperties( + name = "boundedMeetSemilattice", + parents = Seq(meetSemilattice), + join = None, + meet = Some(boundedSemilattice(A.meetSemilattice)) + ) + + def boundedJoinLattice(implicit A: Lattice[A] with BoundedJoinSemilattice[A]) = new LatticeProperties( + name = "boundedJoinLattice", + parents = Seq(boundedJoinSemilattice, lattice), + join = Some(boundedSemilattice(A.joinSemilattice)), + meet = Some(semilattice(A.meetSemilattice)) + ) + + def boundedMeetLattice(implicit A: Lattice[A] with BoundedMeetSemilattice[A]) = new LatticeProperties( + name = "boundedMeetLattice", + parents = Seq(boundedMeetSemilattice, lattice), + join = Some(semilattice(A.joinSemilattice)), + meet = Some(boundedSemilattice(A.meetSemilattice)) + ) + + def boundedLattice(implicit A: BoundedLattice[A]) = new LatticeProperties( + name = "boundedLattice", + parents = Seq(boundedJoinSemilattice, boundedMeetSemilattice, lattice), + join = Some(boundedSemilattice(A.joinSemilattice)), + meet = Some(boundedSemilattice(A.meetSemilattice)) + ) + + def boundedDistributiveLattice(implicit A: BoundedDistributiveLattice[A]) = new LatticeProperties( + name = "boundedLattice", + parents = Seq(boundedLattice, distributiveLattice), + join = Some(boundedSemilattice(A.joinSemilattice)), + meet = Some(boundedSemilattice(A.meetSemilattice)) + ) + + class LatticeProperties( + val name: String, + val parents: Seq[LatticeProperties], + val join: Option[GroupProperties], + val meet: Option[GroupProperties], + val props: (String, Prop)* + ) extends RuleSet { + private val _m = meet.map { "meet" -> _ } + private val _j = join.map { "join" -> _ } + + val bases = _m.toList ::: _j.toList + } + +} diff --git a/algebra-laws/shared/src/main/scala/algebra/laws/LatticePartialOrderLaws.scala b/algebra-laws/shared/src/main/scala/algebra/laws/LatticePartialOrderLaws.scala new file mode 100644 index 0000000000..db3d17a649 --- /dev/null +++ b/algebra-laws/shared/src/main/scala/algebra/laws/LatticePartialOrderLaws.scala @@ -0,0 +1,87 @@ +package algebra.laws + +import algebra._ +import algebra.lattice._ + +import org.typelevel.discipline.Laws + +import org.scalacheck.{Arbitrary, Prop} +import org.scalacheck.Prop._ + +import algebra.instances.boolean._ + +object LatticePartialOrderLaws { + def apply[A: Eq: Arbitrary] = new LatticePartialOrderLaws[A] { + def Equ = Eq[A] + def Arb = implicitly[Arbitrary[A]] + } +} + +trait LatticePartialOrderLaws[A] extends Laws { + + implicit def Equ: Eq[A] + implicit def Arb: Arbitrary[A] + + def joinSemilatticePartialOrder(implicit A: JoinSemilattice[A], P: PartialOrder[A]) = + new LatticePartialOrderProperties( + name = "joinSemilatticePartialOrder", + parents = Seq.empty, + "join+lteqv" -> forAll { (x: A, y: A) => + P.lteqv(x, y) ?== P.eqv(y, A.join(x, y)) + } + ) + + def meetSemilatticePartialOrder(implicit A: MeetSemilattice[A], P: PartialOrder[A]) = + new LatticePartialOrderProperties( + name = "meetSemilatticePartialOrder", + parents = Seq.empty, + "meet+lteqv" -> forAll { (x: A, y: A) => + P.lteqv(x, y) ?== P.eqv(x, A.meet(x, y)) + } + ) + + def latticePartialOrder(implicit A: Lattice[A], P: PartialOrder[A]) = new LatticePartialOrderProperties( + name = "latticePartialOrder", + parents = Seq(joinSemilatticePartialOrder, meetSemilatticePartialOrder) + ) + + def boundedJoinSemilatticePartialOrder(implicit A: BoundedJoinSemilattice[A], P: PartialOrder[A]) = + new LatticePartialOrderProperties( + name = "boundedJoinSemilatticePartialOrder", + parents = Seq(joinSemilatticePartialOrder), + "lteqv+zero" -> forAll { (x: A) => A.zero ?<= x } + ) + + def boundedMeetSemilatticePartialOrder(implicit A: BoundedMeetSemilattice[A], P: PartialOrder[A]) = + new LatticePartialOrderProperties( + name = "boundedMeetSemilatticePartialOrder", + parents = Seq(meetSemilatticePartialOrder), + "lteqv+one" -> forAll { (x: A) => x ?<= A.one } + ) + + def boundedBelowLatticePartialOrder(implicit A: Lattice[A] with BoundedJoinSemilattice[A], P: PartialOrder[A]) = + new LatticePartialOrderProperties( + name = "boundedBelowLatticePartialOrder", + parents = Seq(boundedJoinSemilatticePartialOrder, latticePartialOrder) + ) + + def boundedAboveLatticePartialOrder(implicit A: Lattice[A] with BoundedMeetSemilattice[A], P: PartialOrder[A]) = + new LatticePartialOrderProperties( + name = "boundedAboveLatticePartialOrder", + parents = Seq(boundedMeetSemilatticePartialOrder, latticePartialOrder) + ) + + def boundedLatticePartialOrder(implicit A: BoundedLattice[A], P: PartialOrder[A]) = new LatticePartialOrderProperties( + name = "boundedLatticePartialOrder", + parents = Seq(boundedJoinSemilatticePartialOrder, boundedMeetSemilatticePartialOrder) + ) + + class LatticePartialOrderProperties( + val name: String, + val parents: Seq[LatticePartialOrderProperties], + val props: (String, Prop)* + ) extends RuleSet { + def bases = Nil + } + +} diff --git a/algebra-laws/shared/src/main/scala/algebra/laws/LogicLaws.scala b/algebra-laws/shared/src/main/scala/algebra/laws/LogicLaws.scala new file mode 100644 index 0000000000..7b42fe3ff7 --- /dev/null +++ b/algebra-laws/shared/src/main/scala/algebra/laws/LogicLaws.scala @@ -0,0 +1,84 @@ +package algebra.laws + +import algebra._ +import algebra.lattice.{Bool, GenBool, Heyting} + +import org.scalacheck.{Arbitrary, Prop} +import org.scalacheck.Prop._ +import scala.annotation.nowarn + +object LogicLaws { + def apply[A: Eq: Arbitrary] = new LogicLaws[A] { + def Equ = Eq[A] + def Arb = implicitly[Arbitrary[A]] + } +} + +@nowarn("msg=deprecated") +trait LogicLaws[A] extends LatticeLaws[A] with DeMorganLaws[A] { + + final def LL: LatticeLaws[A] = this + + def heyting(implicit A: Heyting[A]) = new LogicProperties( + name = "heyting", + parents = Seq(), + ll = boundedDistributiveLattice, + Rules.distributive(A.or)(A.and), + "consistent" -> forAll { (x: A) => A.and(x, A.complement(x)) ?== A.zero }, + "¬x = (x → 0)" -> forAll { (x: A) => A.complement(x) ?== A.imp(x, A.zero) }, + "x → x = 1" -> forAll { (x: A) => A.imp(x, x) ?== A.one }, + "if x → y and y → x then x=y" -> forAll { (x: A, y: A) => + (A.imp(x, y) ?!= A.one) || (A.imp(y, x) ?!= A.one) || (x ?== y) + }, + "if (1 → x)=1 then x=1" -> forAll { (x: A) => + (A.imp(A.one, x) ?!= A.one) || (x ?== A.one) + }, + "x → (y → x) = 1" -> forAll { (x: A, y: A) => A.imp(x, A.imp(y, x)) ?== A.one }, + "(x→(y→z)) → ((x→y)→(x→z)) = 1" -> forAll { (x: A, y: A, z: A) => + A.imp(A.imp(x, A.imp(y, z)), A.imp(A.imp(x, y), A.imp(x, z))) ?== A.one + }, + "x∧y → x = 1" -> forAll { (x: A, y: A) => A.imp(A.and(x, y), x) ?== A.one }, + "x∧y → y = 1" -> forAll { (x: A, y: A) => A.imp(A.and(x, y), y) ?== A.one }, + "x → y → (x∧y) = 1" -> forAll { (x: A, y: A) => A.imp(x, A.imp(y, A.and(x, y))) ?== A.one }, + "x → x∨y" -> forAll { (x: A, y: A) => A.imp(x, A.or(x, y)) ?== A.one }, + "y → x∨y" -> forAll { (x: A, y: A) => A.imp(y, A.or(x, y)) ?== A.one }, + "(x → z) → ((y → z) → ((x | y) → z)) = 1" -> forAll { (x: A, y: A, z: A) => + A.imp(A.imp(x, z), A.imp(A.imp(y, z), A.imp(A.or(x, y), z))) ?== A.one + }, + "(0 → x) = 1" -> forAll { (x: A) => A.imp(A.zero, x) ?== A.one } + ) + + def generalizedBool(implicit A: GenBool[A]) = new LogicProperties( + name = "generalized bool", + parents = Seq(), + ll = new LatticeProperties( + name = "lowerBoundedDistributiveLattice", + parents = Seq(boundedJoinSemilattice, distributiveLattice), + join = Some(boundedSemilattice(A.joinSemilattice)), + meet = Some(semilattice(A.meetSemilattice)) + ), + """x\y ∧ y = 0""" -> forAll { (x: A, y: A) => + A.and(A.without(x, y), y) ?== A.zero + }, + """x\y ∨ y = x ∨ y""" -> forAll { (x: A, y: A) => + A.or(A.without(x, y), y) ?== A.or(x, y) + } + ) + + def bool(implicit A: Bool[A]) = new LogicProperties( + name = "bool", + parents = Seq(heyting, generalizedBool), + ll = boundedDistributiveLattice, + "excluded middle" -> forAll { (x: A) => A.or(x, A.complement(x)) ?== A.one } + ) + + class LogicProperties( + val name: String, + val parents: Seq[LogicProperties], + val ll: LatticeProperties, + val props: (String, Prop)* + ) extends RuleSet { + val bases = Seq("lattice" -> ll) + } + +} diff --git a/algebra-laws/shared/src/main/scala/algebra/laws/OrderLaws.scala b/algebra-laws/shared/src/main/scala/algebra/laws/OrderLaws.scala new file mode 100644 index 0000000000..5a2aecae10 --- /dev/null +++ b/algebra-laws/shared/src/main/scala/algebra/laws/OrderLaws.scala @@ -0,0 +1,227 @@ +package algebra.laws + +import cats.kernel._ + +import org.typelevel.discipline.Laws + +import org.scalacheck.{Arbitrary, Cogen, Prop} +import org.scalacheck.Prop._ + +import cats.kernel.instances.all._ +import algebra.ring.Signed +import algebra.ring.CommutativeRing +import algebra.ring.TruncatedDivision +import algebra.ring.AdditiveCommutativeGroup +import algebra.ring.GCDRing + +@deprecated("Provided by cats.kernel.laws", since = "2.7.0") +object OrderLaws { + def apply[A: Eq: Arbitrary: Cogen]: OrderLaws[A] = + new OrderLaws[A] { + def Equ = Eq[A] + def Arb = implicitly[Arbitrary[A]] + def Cog = implicitly[Cogen[A]] + } +} + +@deprecated("Provided by cats.kernel.laws", since = "2.7.0") +trait OrderLaws[A] extends Laws { + + implicit def Equ: Eq[A] + implicit def Arb: Arbitrary[A] + implicit def Cog: Cogen[A] + + def eqv: OrderProperties = new OrderProperties( + name = "eq", + parent = None, + Rules.serializable(Equ), + "reflexitivity-eq" -> forAll { (x: A) => + x ?== x + }, + "symmetry-eq" -> forAll { (x: A, y: A) => + Equ.eqv(x, y) ?== Equ.eqv(y, x) + }, + "antisymmetry-eq" -> forAll { (x: A, y: A, f: A => A) => + !Equ.eqv(x, y) ?|| Equ.eqv(f(x), f(y)) + }, + "transitivity-eq" -> forAll { (x: A, y: A, z: A) => + !(Equ.eqv(x, y) && Equ.eqv(y, z)) ?|| Equ.eqv(x, z) + } + ) + + def partialOrder(implicit A: PartialOrder[A]): OrderProperties = new OrderProperties( + name = "partialOrder", + parent = Some(eqv), + Rules.serializable(A), + "reflexitivity" -> forAll { (x: A) => + x ?<= x + }, + "antisymmetry" -> forAll { (x: A, y: A) => + !(A.lteqv(x, y) && A.lteqv(y, x)) ?|| A.eqv(x, y) + }, + "transitivity" -> forAll { (x: A, y: A, z: A) => + !(A.lteqv(x, y) && A.lteqv(y, z)) ?|| A.lteqv(x, z) + }, + "gteqv" -> forAll { (x: A, y: A) => + A.lteqv(x, y) ?== A.gteqv(y, x) + }, + "lt" -> forAll { (x: A, y: A) => + A.lt(x, y) ?== (A.lteqv(x, y) && A.neqv(x, y)) + }, + "gt" -> forAll { (x: A, y: A) => + A.lt(x, y) ?== A.gt(y, x) + }, + "partialCompare" -> forAll { (x: A, y: A) => + val c = A.partialCompare(x, y) + ((c < 0) ?== A.lt(x, y)) && ((c == 0) ?== A.eqv(x, y)) && ((c > 0) ?== A.gt(x, y)) + }, + "pmin" -> forAll { (x: A, y: A) => + val c = A.partialCompare(x, y) + val m = A.pmin(x, y) + if (c < 0) m ?== Some(x) + else if (c == 0) (m ?== Some(x)) && (m ?== Some(y)) + else if (c > 0) m ?== Some(y) + else m ?== None + }, + "pmax" -> forAll { (x: A, y: A) => + val c = A.partialCompare(x, y) + val m = A.pmax(x, y) + if (c < 0) m ?== Some(y) + else if (c == 0) (m ?== Some(x)) && (m ?== Some(y)) + else if (c > 0) m ?== Some(x) + else m ?== None + } + ) + + def order(implicit A: Order[A]): OrderProperties = new OrderProperties( + name = "order", + parent = Some(partialOrder), + "totality" -> forAll { (x: A, y: A) => + A.lteqv(x, y) ?|| A.lteqv(y, x) + }, + "compare" -> forAll { (x: A, y: A) => + val c = A.compare(x, y) + ((c < 0) ?== A.lt(x, y)) && ((c == 0) ?== A.eqv(x, y)) && ((c > 0) ?== A.gt(x, y)) + }, + "min" -> forAll { (x: A, y: A) => + val c = A.compare(x, y) + val m = A.min(x, y) + if (c < 0) m ?== x + else if (c == 0) (m ?== x) && (m ?== y) + else m ?== y + }, + "max" -> forAll { (x: A, y: A) => + val c = A.compare(x, y) + val m = A.max(x, y) + if (c < 0) m ?== y + else if (c == 0) (m ?== x) && (m ?== y) + else m ?== x + } + ) + + def signed(implicit A: Signed[A]) = new OrderProperties( + name = "signed", + parent = Some(order(A.order)), + "abs non-negative" -> forAll((x: A) => A.sign(A.abs(x)) != Signed.Negative), + "signum returns -1/0/1" -> forAll((x: A) => A.signum(A.abs(x)) <= 1), + "signum is sign.toInt" -> forAll((x: A) => A.signum(x) == A.sign(x).toInt), + "ordered group" -> forAll { (x: A, y: A, z: A) => + A.order.lteqv(x, y) ==> A.order.lteqv(A.additiveCommutativeMonoid.plus(x, z), + A.additiveCommutativeMonoid.plus(y, z) + ) + }, + "triangle inequality" -> forAll { (x: A, y: A) => + A.order.lteqv(A.abs(A.additiveCommutativeMonoid.plus(x, y)), A.additiveCommutativeMonoid.plus(A.abs(x), A.abs(y))) + } + ) + + def signedAdditiveCommutativeGroup(implicit signedA: Signed[A], A: AdditiveCommutativeGroup[A]) = new DefaultRuleSet( + name = "signedAdditiveAbGroup", + parent = Some(signed), + "abs(x) equals abs(-x)" -> forAll { (x: A) => + signedA.abs(x) ?== signedA.abs(A.negate(x)) + } + ) + + // more a convention: as GCD is defined up to a unit, so up to a sign, + // on an ordered GCD ring we require gcd(x, y) >= 0, which is the common + // behavior of computer algebra systems + def signedGCDRing(implicit signedA: Signed[A], A: GCDRing[A]) = new DefaultRuleSet( + name = "signedGCDRing", + parent = Some(signedAdditiveCommutativeGroup), + "gcd(x, y) >= 0" -> forAll { (x: A, y: A) => + signedA.isSignNonNegative(A.gcd(x, y)) + }, + "gcd(x, 0) === abs(x)" -> forAll { (x: A) => + A.gcd(x, A.zero) ?== signedA.abs(x) + } + ) + + def truncatedDivision(implicit ring: CommutativeRing[A], A: TruncatedDivision[A]) = new DefaultRuleSet( + name = "truncatedDivision", + parent = Some(signed), + "division rule (tquotmod)" -> forAll { (x: A, y: A) => + A.isSignNonZero(y) ==> { + val (q, r) = A.tquotmod(x, y) + x ?== ring.plus(ring.times(y, q), r) + } + }, + "division rule (fquotmod)" -> forAll { (x: A, y: A) => + A.isSignNonZero(y) ==> { + val (q, r) = A.fquotmod(x, y) + x ?== ring.plus(ring.times(y, q), r) + } + }, + "|r| < |y| (tmod)" -> forAll { (x: A, y: A) => + A.isSignNonZero(y) ==> { + val r = A.tmod(x, y) + A.order.lt(A.abs(r), A.abs(y)) + } + }, + "|r| < |y| (fmod)" -> forAll { (x: A, y: A) => + A.isSignNonZero(y) ==> { + val r = A.fmod(x, y) + A.order.lt(A.abs(r), A.abs(y)) + } + }, + "r = 0 or sign(r) = sign(x) (tmod)" -> forAll { (x: A, y: A) => + A.isSignNonZero(y) ==> { + val r = A.tmod(x, y) + A.isSignZero(r) || (A.sign(r) ?== A.sign(x)) + } + }, + "r = 0 or sign(r) = sign(y) (fmod)" -> forAll { (x: A, y: A) => + A.isSignNonZero(y) ==> { + val r = A.fmod(x, y) + A.isSignZero(r) || (A.sign(r) ?== A.sign(y)) + } + }, + "tquot" -> forAll { (x: A, y: A) => + A.isSignNonZero(y) ==> { + A.tquotmod(x, y)._1 ?== A.tquot(x, y) + } + }, + "tmod" -> forAll { (x: A, y: A) => + A.isSignNonZero(y) ==> { + A.tquotmod(x, y)._2 ?== A.tmod(x, y) + } + }, + "fquot" -> forAll { (x: A, y: A) => + A.isSignNonZero(y) ==> { + A.fquotmod(x, y)._1 ?== A.fquot(x, y) + } + }, + "fmod" -> forAll { (x: A, y: A) => + A.isSignNonZero(y) ==> { + A.fquotmod(x, y)._2 ?== A.fmod(x, y) + } + } + ) + + class OrderProperties( + name: String, + parent: Option[RuleSet], + props: (String, Prop)* + ) extends DefaultRuleSet(name, parent, props: _*) + +} diff --git a/algebra-laws/shared/src/main/scala/algebra/laws/RingLaws.scala b/algebra-laws/shared/src/main/scala/algebra/laws/RingLaws.scala new file mode 100644 index 0000000000..22df657e10 --- /dev/null +++ b/algebra-laws/shared/src/main/scala/algebra/laws/RingLaws.scala @@ -0,0 +1,378 @@ +package algebra +package laws + +import algebra.ring._ + +import algebra.laws.platform.Platform + +import org.typelevel.discipline.Predicate + +import org.scalacheck.{Arbitrary, Prop} +import org.scalacheck.Arbitrary._ +import org.scalacheck.Prop._ +import scala.annotation.nowarn + +object RingLaws { + def apply[A: Eq: Arbitrary: AdditiveMonoid]: RingLaws[A] = + withPred[A](new Predicate[A] { + def apply(a: A): Boolean = Eq[A].neqv(a, AdditiveMonoid[A].zero) + }) + + def withPred[A](pred0: Predicate[A])(implicit eqv: Eq[A], arb: Arbitrary[A]): RingLaws[A] = new RingLaws[A] { + def Arb = arb + def pred = pred0 + @nowarn("msg=deprecated") + val nonZeroLaws = new GroupLaws[A] { + def Arb = Arbitrary(arbitrary[A](arb).filter(pred)) + def Equ = eqv + } + } +} + +@nowarn("msg=deprecated") +trait RingLaws[A] extends GroupLaws[A] { self => + + // must be a val (stable identifier) + val nonZeroLaws: GroupLaws[A] + def pred: Predicate[A] + + def withPred(pred0: Predicate[A], replace: Boolean = true): RingLaws[A] = + RingLaws.withPred(if (replace) pred0 else pred && pred0)(Equ, Arb) + + def setNonZeroParents(props: nonZeroLaws.GroupProperties, + parents: Seq[nonZeroLaws.GroupProperties] + ): nonZeroLaws.GroupProperties = + new nonZeroLaws.GroupProperties( + name = props.name, + parents = parents, + props = props.props: _* + ) + + implicit def Arb: Arbitrary[A] + implicit def Equ: Eq[A] = nonZeroLaws.Equ + + // additive groups + + def additiveSemigroup(implicit A: AdditiveSemigroup[A]) = new AdditiveProperties( + base = semigroup(A.additive), + parents = Nil, + Rules.serializable(A), + Rules.repeat1("sumN")(A.sumN), + Rules.repeat2("sumN", "+")(A.sumN)(A.plus) + ) + + def additiveCommutativeSemigroup(implicit A: AdditiveCommutativeSemigroup[A]) = new AdditiveProperties( + base = commutativeSemigroup(A.additive), + parents = List(additiveSemigroup) + ) + + def additiveMonoid(implicit A: AdditiveMonoid[A]) = new AdditiveProperties( + base = monoid(A.additive), + parents = List(additiveSemigroup), + Rules.repeat0("sumN", "zero", A.zero)(A.sumN), + Rules.collect0("sum", "zero", A.zero)(A.sum) + ) + + def additiveCommutativeMonoid(implicit A: AdditiveCommutativeMonoid[A]) = new AdditiveProperties( + base = commutativeMonoid(A.additive), + parents = List(additiveMonoid) + ) + + def additiveGroup(implicit A: AdditiveGroup[A]) = new AdditiveProperties( + base = group(A.additive), + parents = List(additiveMonoid), + Rules.consistentInverse("subtract")(A.minus)(A.plus)(A.negate) + ) + + def additiveCommutativeGroup(implicit A: AdditiveCommutativeGroup[A]) = new AdditiveProperties( + base = commutativeGroup(A.additive), + parents = List(additiveGroup) + ) + + // multiplicative groups + + def multiplicativeSemigroup(implicit A: MultiplicativeSemigroup[A]) = new MultiplicativeProperties( + base = semigroup(A.multiplicative), + nonZeroBase = None, + parent = None, + Rules.serializable(A), + Rules.repeat1("pow")(A.pow), + Rules.repeat2("pow", "*")(A.pow)(A.times) + ) + + def multiplicativeCommutativeSemigroup(implicit A: MultiplicativeCommutativeSemigroup[A]) = + new MultiplicativeProperties( + base = semigroup(A.multiplicative), + nonZeroBase = None, + parent = Some(multiplicativeSemigroup) + ) + + def multiplicativeMonoid(implicit A: MultiplicativeMonoid[A]) = new MultiplicativeProperties( + base = monoid(A.multiplicative), + nonZeroBase = None, + parent = Some(multiplicativeSemigroup), + Rules.repeat0("pow", "one", A.one)(A.pow), + Rules.collect0("product", "one", A.one)(A.product) + ) + + def multiplicativeCommutativeMonoid(implicit A: MultiplicativeCommutativeMonoid[A]) = new MultiplicativeProperties( + base = commutativeMonoid(A.multiplicative), + nonZeroBase = None, + parent = Some(multiplicativeMonoid) + ) + + def multiplicativeGroup(implicit A: MultiplicativeGroup[A]) = new MultiplicativeProperties( + base = monoid(A.multiplicative), + nonZeroBase = Some(setNonZeroParents(nonZeroLaws.group(A.multiplicative), Nil)), + parent = Some(multiplicativeMonoid), + // pred is used to ensure y is not zero. + "consistent division" -> forAll { (x: A, y: A) => + pred(y) ==> (A.div(x, y) ?== A.times(x, A.reciprocal(y))) + } + ) + + def multiplicativeCommutativeGroup(implicit A: MultiplicativeCommutativeGroup[A]) = new MultiplicativeProperties( + base = commutativeMonoid(A.multiplicative), + nonZeroBase = + Some(setNonZeroParents(nonZeroLaws.commutativeGroup(A.multiplicative), multiplicativeGroup.nonZeroBase.toSeq)), + parent = Some(multiplicativeGroup) + ) + + // rings + + def semiring(implicit A: Semiring[A]) = new RingProperties( + name = "semiring", + al = additiveCommutativeMonoid, + ml = multiplicativeSemigroup, + parents = Seq.empty, + Rules.distributive(A.plus)(A.times) + ) + + def rng(implicit A: Rng[A]) = new RingProperties( + name = "rng", + al = additiveCommutativeGroup, + ml = multiplicativeSemigroup, + parents = Seq(semiring) + ) + + def rig(implicit A: Rig[A]) = new RingProperties( + name = "rig", + al = additiveCommutativeMonoid, + ml = multiplicativeMonoid, + parents = Seq(semiring) + ) + + def ring(implicit A: Ring[A]) = new RingProperties( + // TODO fromParents + name = "ring", + al = additiveCommutativeGroup, + ml = multiplicativeMonoid, + parents = Seq(rig, rng), + "fromInt" -> forAll { (n: Int) => + Ring.fromInt[A](n) ?== A.sumN(A.one, n) + }, + "fromBigInt" -> forAll { (ns: List[Int]) => + val actual = Ring.fromBigInt[A](ns.map(BigInt(_)).foldLeft(BigInt(1))(_ * _)) + val expected = ns.map(A.fromInt).foldLeft(A.one)(A.times) + actual ?== expected + } + ) + + // commutative rings + + def commutativeSemiring(implicit A: CommutativeSemiring[A]) = new RingProperties( + name = "commutativeSemiring", + al = additiveCommutativeMonoid, + ml = multiplicativeCommutativeSemigroup, + parents = Seq(semiring) + ) + + def commutativeRng(implicit A: CommutativeRng[A]) = new RingProperties( + name = "commutativeRng", + al = additiveCommutativeMonoid, + ml = multiplicativeCommutativeSemigroup, + parents = Seq(rng, commutativeSemiring) + ) + + def commutativeRig(implicit A: CommutativeRig[A]) = new RingProperties( + name = "commutativeRig", + al = additiveCommutativeMonoid, + ml = multiplicativeCommutativeMonoid, + parents = Seq(rig, commutativeSemiring) + ) + + def commutativeRing(implicit A: CommutativeRing[A]) = new RingProperties( + name = "commutative ring", + al = additiveCommutativeGroup, + ml = multiplicativeCommutativeMonoid, + parents = Seq(ring, commutativeRig, commutativeRng) + ) + + def gcdRing(implicit A: GCDRing[A]) = RingProperties.fromParent( + name = "gcd domain", + parent = commutativeRing, + "gcd/lcm" -> forAll { (x: A, y: A) => + val d = A.gcd(x, y) + val m = A.lcm(x, y) + A.times(x, y) ?== A.times(d, m) + }, + "gcd is commutative" -> forAll { (x: A, y: A) => + A.gcd(x, y) ?== A.gcd(y, x) + }, + "lcm is commutative" -> forAll { (x: A, y: A) => + A.lcm(x, y) ?== A.lcm(y, x) + }, + "gcd(0, 0)" -> (A.gcd(A.zero, A.zero) ?== A.zero), + "lcm(0, 0) === 0" -> (A.lcm(A.zero, A.zero) ?== A.zero), + "lcm(x, 0) === 0" -> forAll { (x: A) => A.lcm(x, A.zero) ?== A.zero } + ) + + def euclideanRing(implicit A: EuclideanRing[A]) = RingProperties.fromParent( + name = "euclidean ring", + parent = gcdRing, + "euclidean division rule" -> forAll { (x: A, y: A) => + pred(y) ==> { + val (q, r) = A.equotmod(x, y) + x ?== A.plus(A.times(y, q), r) + } + }, + "equot" -> forAll { (x: A, y: A) => + pred(y) ==> { + A.equotmod(x, y)._1 ?== A.equot(x, y) + } + }, + "emod" -> forAll { (x: A, y: A) => + pred(y) ==> { + A.equotmod(x, y)._2 ?== A.emod(x, y) + } + }, + "euclidean function" -> forAll { (x: A, y: A) => + pred(y) ==> { + val (_, r) = A.equotmod(x, y) + A.isZero(r) || (A.euclideanFunction(r) < A.euclideanFunction(y)) + } + }, + "submultiplicative function" -> forAll { (x: A, y: A) => + (pred(x) && pred(y)) ==> { + A.euclideanFunction(x) <= A.euclideanFunction(A.times(x, y)) + } + } + ) + + def divisionRing(implicit A: DivisionRing[A]) = new RingProperties( + name = "division ring", + al = additiveCommutativeGroup, + ml = multiplicativeGroup, + parents = Seq(ring), + "fromDouble" -> forAll { (n: Double) => + if (Platform.isJvm) { + // TODO: BigDecimal(n) is busted in scalajs, so we skip this test. + val bd = new java.math.BigDecimal(n) + val unscaledValue = new BigInt(bd.unscaledValue) + val expected = + if (bd.scale > 0) { + A.div(A.fromBigInt(unscaledValue), A.fromBigInt(BigInt(10).pow(bd.scale))) + } else { + A.fromBigInt(unscaledValue * BigInt(10).pow(-bd.scale)) + } + DivisionRing.fromDouble[A](n) ?== expected + } else { + Prop(true) + } + } + ) + + // boolean rings + + def boolRng(implicit A: BoolRng[A]) = RingProperties.fromParent( + name = "boolean rng", + parent = commutativeRng, + Rules.idempotence(A.times) + ) + + def boolRing(implicit A: BoolRing[A]) = RingProperties.fromParent( + name = "boolean ring", + parent = commutativeRing, + Rules.idempotence(A.times) + ) + + // Everything below fields (e.g. rings) does not require their multiplication + // operation to be a group. Hence, we do not check for the existence of an + // inverse. On the other hand, fields require their multiplication to be an + // abelian group. Now we have to worry about zero. + // + // The usual text book definition says: Fields consist of two abelian groups + // (set, +, zero) and (set \ zero, *, one). We do the same thing here. + // However, since law checking for the multiplication does not include zero + // any more, it is not immediately clear that desired properties like + // zero * x == x * zero hold. + // Luckily, these follow from the other field and group axioms. + def field(implicit A: Field[A]) = new RingProperties( + name = "field", + al = additiveCommutativeGroup, + ml = multiplicativeCommutativeGroup, + parents = Seq(euclideanRing, divisionRing) + ) + + // Approximate fields such a Float or Double, even through filtered using FPFilter, do not work well with + // Euclidean ring tests + def approxField(implicit A: Field[A]) = new RingProperties( + name = "field", + al = additiveCommutativeGroup, + ml = multiplicativeCommutativeGroup, + parents = Seq(commutativeRing), + "fromDouble" -> forAll { (n: Double) => + if (Platform.isJvm) { + // TODO: BigDecimal(n) is busted in scalajs, so we skip this test. + val bd = new java.math.BigDecimal(n) + val unscaledValue = new BigInt(bd.unscaledValue) + val expected = + if (bd.scale > 0) { + A.div(A.fromBigInt(unscaledValue), A.fromBigInt(BigInt(10).pow(bd.scale))) + } else { + A.fromBigInt(unscaledValue * BigInt(10).pow(-bd.scale)) + } + Field.fromDouble[A](n) ?== expected + } else { + Prop(true) + } + } + ) + + // property classes + + class AdditiveProperties( + val base: GroupLaws[A]#GroupProperties, + val parents: Seq[AdditiveProperties], + val props: (String, Prop)* + ) extends RuleSet { + val name = "additive " + base.name + val bases = List("base" -> base) + } + + class MultiplicativeProperties( + val base: GroupLaws[A]#GroupProperties, + val nonZeroBase: Option[nonZeroLaws.GroupProperties], + val parent: Option[MultiplicativeProperties], + val props: (String, Prop)* + ) extends RuleSet + with HasOneParent { + val name = "multiplicative " + base.name + val bases = Seq("base" -> base) ++ nonZeroBase.map("non-zero base" -> _) + } + + object RingProperties { + def fromParent(name: String, parent: RingProperties, props: (String, Prop)*) = + new RingProperties(name, parent.al, parent.ml, Seq(parent), props: _*) + } + + class RingProperties( + val name: String, + val al: AdditiveProperties, + val ml: MultiplicativeProperties, + val parents: Seq[RingProperties], + val props: (String, Prop)* + ) extends RuleSet { + def bases = Seq("additive" -> al, "multiplicative" -> ml) + } +} diff --git a/algebra-laws/shared/src/main/scala/algebra/laws/Rules.scala b/algebra-laws/shared/src/main/scala/algebra/laws/Rules.scala new file mode 100644 index 0000000000..32a62bbc9e --- /dev/null +++ b/algebra-laws/shared/src/main/scala/algebra/laws/Rules.scala @@ -0,0 +1,99 @@ +package algebra.laws + +import cats.kernel._ +import org.scalacheck.Prop._ +import org.scalacheck.{Arbitrary, Prop} +import cats.kernel.instances.boolean._ +import cats.kernel.laws.discipline.SerializableTests + +object Rules { + + // Comparison operators for testing are supplied by CheckEqOps and + // CheckOrderOps in package.scala. They are: + // + // ?== Ensure that x equals y + // ?!= Ensure that x does not equal y + // ?< Ensure that x < y + // ?<= Ensure that x <= y + // ?> Ensure that x > y + // ?>= Ensure that x >= y + // + // The reason to prefer these operators is that when tests fail, we + // will get more detaild output about what the failing values were + // (in addition to the input values generated by ScalaCheck). + + def associativity[A: Arbitrary: Eq](f: (A, A) => A): (String, Prop) = + "associativity" -> forAll { (x: A, y: A, z: A) => + f(f(x, y), z) ?== f(x, f(y, z)) + } + + def leftIdentity[A: Arbitrary: Eq](id: A)(f: (A, A) => A): (String, Prop) = + "leftIdentity" -> forAll { (x: A) => + f(id, x) ?== x + } + + def rightIdentity[A: Arbitrary: Eq](id: A)(f: (A, A) => A): (String, Prop) = + "rightIdentity" -> forAll { (x: A) => + f(x, id) ?== x + } + + def leftInverse[A: Arbitrary: Eq](id: A)(f: (A, A) => A)(inv: A => A): (String, Prop) = + "left inverse" -> forAll { (x: A) => + id ?== f(inv(x), x) + } + + def rightInverse[A: Arbitrary: Eq](id: A)(f: (A, A) => A)(inv: A => A): (String, Prop) = + "right inverse" -> forAll { (x: A) => + id ?== f(x, inv(x)) + } + + def commutative[A: Arbitrary: Eq](f: (A, A) => A): (String, Prop) = + "commutative" -> forAll { (x: A, y: A) => + f(x, y) ?== f(y, x) + } + + def idempotence[A: Arbitrary: Eq](f: (A, A) => A): (String, Prop) = + "idempotence" -> forAll { (x: A) => + f(x, x) ?== x + } + + def consistentInverse[A: Arbitrary: Eq](name: String)(m: (A, A) => A)(f: (A, A) => A)(inv: A => A): (String, Prop) = + s"consistent $name" -> forAll { (x: A, y: A) => + m(x, y) ?== f(x, inv(y)) + } + + def repeat0[A: Arbitrary: Eq](name: String, sym: String, id: A)(r: (A, Int) => A): (String, Prop) = + s"$name(a, 0) == $sym" -> forAll { (a: A) => + r(a, 0) ?== id + } + + def repeat1[A: Arbitrary: Eq](name: String)(r: (A, Int) => A): (String, Prop) = + s"$name(a, 1) == a" -> forAll { (a: A) => + r(a, 1) ?== a + } + + def repeat2[A: Arbitrary: Eq](name: String, sym: String)(r: (A, Int) => A)(f: (A, A) => A): (String, Prop) = + s"$name(a, 2) == a $sym a" -> forAll { (a: A) => + r(a, 2) ?== f(a, a) + } + + def collect0[A: Arbitrary: Eq](name: String, sym: String, id: A)(c: Seq[A] => A): (String, Prop) = + s"$name(Nil) == $sym" -> forAll { (a: A) => + c(Nil) ?== id + } + + def isId[A: Arbitrary: Eq](name: String, id: A)(p: A => Boolean): (String, Prop) = + name -> forAll { (x: A) => + Eq.eqv(x, id) ?== p(x) + } + + def distributive[A: Arbitrary: Eq](a: (A, A) => A)(m: (A, A) => A): (String, Prop) = + "distributive" -> forAll { (x: A, y: A, z: A) => + (m(x, a(y, z)) ?== a(m(x, y), m(x, z))) && + (m(a(x, y), z) ?== a(m(x, z), m(y, z))) + } + + @deprecated("Provided by cats.kernel.laws", since = "2.7.0") + def serializable[M](m: M): (String, Prop) = + SerializableTests.serializable[M](m).props.head +} diff --git a/algebra-laws/shared/src/main/scala/algebra/laws/package.scala b/algebra-laws/shared/src/main/scala/algebra/laws/package.scala new file mode 100644 index 0000000000..304e97d17e --- /dev/null +++ b/algebra-laws/shared/src/main/scala/algebra/laws/package.scala @@ -0,0 +1,40 @@ +package algebra + +import org.scalacheck._ +import org.scalacheck.util.Pretty +import Prop.{False, Proof, Result} + +package object laws { + + lazy val proved = Prop(Result(status = Proof)) + + lazy val falsified = Prop(Result(status = False)) + + object Ops { + def run[A](sym: String)(lhs: A, rhs: A)(f: (A, A) => Boolean): Prop = + if (f(lhs, rhs)) proved + else + falsified :| { + val exp = Pretty.pretty(lhs, Pretty.Params(0)) + val got = Pretty.pretty(rhs, Pretty.Params(0)) + s"($exp $sym $got) failed" + } + } + + implicit class CheckEqOps[A](lhs: A)(implicit ev: Eq[A], pp: A => Pretty) { + def ?==(rhs: A): Prop = Ops.run("?==")(lhs, rhs)(ev.eqv) + def ?!=(rhs: A): Prop = Ops.run("?!=")(lhs, rhs)(ev.neqv) + } + + implicit class CheckOrderOps[A](lhs: A)(implicit ev: PartialOrder[A], pp: A => Pretty) { + def ?<(rhs: A): Prop = Ops.run("?<")(lhs, rhs)(ev.lt) + def ?<=(rhs: A): Prop = Ops.run("?<=")(lhs, rhs)(ev.lteqv) + def ?>(rhs: A): Prop = Ops.run("?>")(lhs, rhs)(ev.gt) + def ?>=(rhs: A): Prop = Ops.run("?>=")(lhs, rhs)(ev.gteqv) + } + + implicit class BooleanOps[A](lhs: Boolean)(implicit pp: Boolean => Pretty) { + def ?&&(rhs: Boolean): Prop = Ops.run("?&&")(lhs, rhs)(_ && _) + def ?||(rhs: Boolean): Prop = Ops.run("?||")(lhs, rhs)(_ || _) + } +} diff --git a/algebra-laws/shared/src/test/scala/algebra/laws/FPApprox.scala b/algebra-laws/shared/src/test/scala/algebra/laws/FPApprox.scala new file mode 100644 index 0000000000..0cedcd686e --- /dev/null +++ b/algebra-laws/shared/src/test/scala/algebra/laws/FPApprox.scala @@ -0,0 +1,145 @@ +package algebra.laws + +import java.lang.{Double => JDouble, Float => JFloat} +import java.math.MathContext + +import org.scalacheck.Arbitrary + +import algebra._ +import algebra.ring._ + +/** + * A wrapper type for approximate floating point values like Float, Double, and + * BigDecimal which maintains an error bound on the current approximation. The + * `Eq` instance for this type returns true if 2 values could be equal to each + * other, given the error bounds, rather than if they actually are equal. So, + * if x == 0.5, and y = 0.6, and the error bound of (x - y) is greater than or + * equal to 0.1, then it's plausible they could be equal to each other, so we + * return true. On the other hand, if the error bound is less than 0.1, then we + * can definitely say they cannot be equal to each other. + */ +case class FPApprox[A](approx: A, mes: A, ind: BigInt) { + import FPApprox.{abs, Epsilon} + + private def timesWithUnderflowCheck(x: A, y: A)(implicit ev: Semiring[A], eq: Eq[A], eps: Epsilon[A]): A = { + val z = ev.times(x, y) + if (eps.nearZero(z) && !ev.isZero(x) && !ev.isZero(y)) eps.minValue + else z + } + + def unary_-(implicit ev: Rng[A]): FPApprox[A] = + FPApprox(ev.negate(approx), mes, ind) + def +(that: FPApprox[A])(implicit ev: Semiring[A]): FPApprox[A] = + FPApprox(ev.plus(approx, that.approx), ev.plus(mes, that.mes), ind.max(that.ind) + 1) + def -(that: FPApprox[A])(implicit ev: Rng[A]): FPApprox[A] = + FPApprox(ev.minus(approx, that.approx), ev.plus(mes, that.mes), ind.max(that.ind) + 1) + def *(that: FPApprox[A])(implicit ev: Semiring[A], eq: Eq[A], eps: Epsilon[A]): FPApprox[A] = + FPApprox(ev.times(approx, that.approx), timesWithUnderflowCheck(mes, that.mes), ind + that.ind + 1) + def /(that: FPApprox[A])(implicit ev: Field[A], ord: Order[A], eps: Epsilon[A]): FPApprox[A] = { + val tmp = abs(that.approx) + val mesApx = ev.plus(ev.div(abs(approx), tmp), ev.div(mes, that.mes)) + val mesCorrection = ev.minus(ev.div(tmp, that.mes), ev.times(ev.fromBigInt(that.ind + 1), eps.epsilon)) + val mes0 = ev.div(mesApx, mesCorrection) + val ind0 = ind.max(that.ind + 1) + 1 + FPApprox(ev.div(approx, that.approx), mes0, ind0) + } + + def reciprocal(implicit ev: Field[A], ord: Order[A], eps: Epsilon[A]): FPApprox[A] = { + val tmp = abs(approx) + val mes0 = ev.div(ev.plus(ev.div(ev.one, tmp), ev.div(ev.one, mes)), ev.minus(ev.div(tmp, mes), eps.epsilon)) + FPApprox(ev.reciprocal(approx), mes0, ind.max(1) + 1) + } + + def pow(k: Int)(implicit ev: Field[A]): FPApprox[A] = { + val k0 = if (k >= 0) BigInt(k) else -BigInt(k) + FPApprox(ev.pow(approx, k), ev.pow(mes, k), (ind + 1) * k0 - 1) + } + + def error(implicit ev: Ring[A], eps: Epsilon[A]): A = + ev.times(ev.times(mes, ev.fromBigInt(ind)), eps.epsilon) +} + +object FPApprox { + final def abs[A](x: A)(implicit ev: Rng[A], ord: Order[A]): A = + if (ord.lt(x, ev.zero)) ev.negate(x) else x + + def exact[A: Rng: Order](a: A): FPApprox[A] = FPApprox(a, abs(a), 0) + def approx[A: Rng: Order](a: A): FPApprox[A] = FPApprox(a, abs(a), 1) + + trait Epsilon[A] extends Serializable { + def minValue: A + def epsilon: A + def isFinite(a: A): Boolean + def nearZero(a: A): Boolean + } + + object Epsilon { + def isFinite[A](a: A)(implicit eps: Epsilon[A]): Boolean = eps.isFinite(a) + + def instance[A](min: A, eps: A, isFin: A => Boolean, zero: A => Boolean): Epsilon[A] = + new Epsilon[A] { + def minValue: A = min + def epsilon: A = eps + def isFinite(a: A): Boolean = isFin(a) + def nearZero(a: A): Boolean = zero(a) + } + + private def isFin[A](a: A)(implicit f: A => Double): Boolean = + !JDouble.isInfinite(f(a)) && !JDouble.isNaN(f(a)) + + // These are not the actual minimums, but closest we can get without + // causing problems. + private val minFloat: Float = JFloat.intBitsToFloat(1 << 23) + private val minDouble: Double = JDouble.longBitsToDouble(1L << 52) + + implicit val floatEpsilon: Epsilon[Float] = + instance(minFloat, 1.1920929e-7f, isFin(_), x => math.abs(x) < minFloat) + implicit val doubleEpsilon: Epsilon[Double] = + instance(minDouble, 2.220446049250313e-16, isFin(_), x => math.abs(x) < minDouble) + def bigDecimalEpsilon(mc: MathContext): Epsilon[BigDecimal] = + instance(BigDecimal(1, Int.MaxValue, mc), BigDecimal(1, mc.getPrecision - 1, mc), _ => true, _ == 0) + } + + implicit def fpApproxAlgebra[A: Field: Order: Epsilon]: FPApproxAlgebra[A] = new FPApproxAlgebra[A] + + // An Eq instance that returns true if 2 values *could* be equal. + implicit def fpApproxEq[A: Field: Order: Epsilon]: Eq[FPApprox[A]] = + new Eq[FPApprox[A]] { + def eqv(x: FPApprox[A], y: FPApprox[A]): Boolean = { + // We want to check if z +/- error contains 0 + if (x.approx == y.approx) { + true + } else { + val z = x - y + val err = z.error + if (Epsilon.isFinite(err)) { + Order.lteqv(Ring[A].minus(z.approx, err), Ring[A].zero) && + Order.gteqv(Ring[A].plus(z.approx, err), Ring[A].zero) + } else { + true + } + } + } + } + + implicit def arbFPApprox[A: Rng: Order: Arbitrary]: Arbitrary[FPApprox[A]] = + Arbitrary(Arbitrary.arbitrary[A].map(FPApprox.exact[A](_))) +} + +class FPApproxAlgebra[A: Order: FPApprox.Epsilon](implicit ev: Field[A]) extends Field[FPApprox[A]] with Serializable { + def zero: FPApprox[A] = FPApprox.exact(ev.zero) + def one: FPApprox[A] = FPApprox.exact(ev.one) + + def plus(x: FPApprox[A], y: FPApprox[A]): FPApprox[A] = x + y + def negate(x: FPApprox[A]): FPApprox[A] = -x + override def minus(x: FPApprox[A], y: FPApprox[A]): FPApprox[A] = x - y + + def times(x: FPApprox[A], y: FPApprox[A]): FPApprox[A] = x * y + def div(x: FPApprox[A], y: FPApprox[A]): FPApprox[A] = x / y + override def reciprocal(x: FPApprox[A]): FPApprox[A] = x.reciprocal // one / x + override def pow(x: FPApprox[A], y: Int): FPApprox[A] = x.pow(y) + + override def fromInt(x: Int): FPApprox[A] = FPApprox.approx(ev.fromInt(x)) + override def fromBigInt(x: BigInt): FPApprox[A] = FPApprox.approx(ev.fromBigInt(x)) + override def fromDouble(x: Double): FPApprox[A] = FPApprox.approx(ev.fromDouble(x)) +} diff --git a/algebra-laws/shared/src/test/scala/algebra/laws/LawTests.scala b/algebra-laws/shared/src/test/scala/algebra/laws/LawTests.scala new file mode 100644 index 0000000000..2982b74a0d --- /dev/null +++ b/algebra-laws/shared/src/test/scala/algebra/laws/LawTests.scala @@ -0,0 +1,200 @@ +package algebra +package laws + +import algebra.lattice._ +import algebra.ring._ +import algebra.instances.all._ +import algebra.instances.BigDecimalAlgebra + +import algebra.laws.platform.Platform + +import org.scalacheck.{Arbitrary, Cogen} +import Arbitrary.arbitrary +import scala.collection.immutable.BitSet +import scala.util.Random + +class LawTests extends munit.DisciplineSuite { + + implicit val byteLattice: Lattice[Byte] = ByteMinMaxLattice + implicit val shortLattice: Lattice[Short] = ShortMinMaxLattice + implicit val intLattice: BoundedDistributiveLattice[Int] = IntMinMaxLattice + implicit val longLattice: BoundedDistributiveLattice[Long] = LongMinMaxLattice + + implicit def logicLaws[A: Eq: Arbitrary]: LogicLaws[A] = LogicLaws[A] + + implicit def latticeLaws[A: Eq: Arbitrary]: LatticeLaws[A] = LatticeLaws[A] + implicit def ringLaws[A: Eq: Arbitrary: AdditiveMonoid]: RingLaws[A] = RingLaws[A] + implicit def latticePartialOrderLaws[A: Eq: Arbitrary]: LatticePartialOrderLaws[A] = LatticePartialOrderLaws[A] + + case class HasEq[A](a: A) + + object HasEq { + implicit def hasEq[A: Eq]: Eq[HasEq[A]] = Eq.by(_.a) + implicit def hasEqArbitrary[A: Arbitrary]: Arbitrary[HasEq[A]] = + Arbitrary(arbitrary[A].map(HasEq(_))) + implicit def hasEqCogen[A: Cogen]: Cogen[HasEq[A]] = + Cogen[A].contramap[HasEq[A]](_.a) + } + + case class HasPartialOrder[A](a: A) + + object HasPartialOrder { + implicit def hasPartialOrder[A: PartialOrder]: PartialOrder[HasPartialOrder[A]] = PartialOrder.by(_.a) + implicit def hasPartialOrderArbitrary[A: Arbitrary]: Arbitrary[HasPartialOrder[A]] = + Arbitrary(arbitrary[A].map(HasPartialOrder(_))) + implicit def hasPartialOrderCogen[A: Cogen]: Cogen[HasPartialOrder[A]] = + Cogen[A].contramap[HasPartialOrder[A]](_.a) + } + + checkAll("Boolean", LogicLaws[Boolean].bool) + checkAll("SimpleHeyting", DeMorganLaws[SimpleHeyting].logic(Logic.fromHeyting(Heyting[SimpleHeyting]))) + checkAll("SimpleHeyting", LogicLaws[SimpleHeyting].heyting) + checkAll("SimpleDeMorgan", DeMorganLaws[SimpleDeMorgan].deMorgan) + checkAll("Boolean", DeMorganLaws[Boolean].deMorgan(DeMorgan.fromBool(Bool[Boolean]))) + checkAll("Boolean", LatticePartialOrderLaws[Boolean].boundedLatticePartialOrder) + checkAll("Boolean", RingLaws[Boolean].boolRing(booleanRing)) + + // ensure that Bool[A].asBoolRing is a valid BoolRing + checkAll("Boolean-ring-from-bool", RingLaws[Boolean].boolRing(new BoolRingFromBool[Boolean](Bool[Boolean]))) + + // ensure that BoolRing[A].asBool is a valid Bool + checkAll("Boolean- bool-from-ring", LogicLaws[Boolean].bool(new BoolFromBoolRing(booleanRing))) + + checkAll("Set[Byte]", LogicLaws[Set[Byte]].generalizedBool) + checkAll("Set[Byte]", RingLaws[Set[Byte]].boolRng(setBoolRng[Byte])) + checkAll("Set[Byte]-bool-from-rng", LogicLaws[Set[Byte]].generalizedBool(new GenBoolFromBoolRng(setBoolRng))) + checkAll("Set[Byte]-rng-from-bool", RingLaws[Set[Byte]].boolRng(new BoolRngFromGenBool(GenBool[Set[Byte]]))) + checkAll("Set[Int]", RingLaws[Set[Int]].semiring) + checkAll("Set[String]", RingLaws[Set[String]].semiring) + + checkAll("Map[Char, Int]", RingLaws[Map[Char, Int]].semiring) + checkAll("Map[Int, BigInt]", RingLaws[Map[Int, BigInt]].semiring) + + checkAll("Byte", RingLaws[Byte].commutativeRing) + checkAll("Byte", LatticeLaws[Byte].lattice) + + checkAll("Short", RingLaws[Short].commutativeRing) + checkAll("Short", LatticeLaws[Short].lattice) + + checkAll("Int", RingLaws[Int].commutativeRing) + checkAll("Int", LatticeLaws[Int].boundedDistributiveLattice) + + { + checkAll("Int", RingLaws[Int].commutativeRig) + } + + checkAll("Long", RingLaws[Long].commutativeRing) + checkAll("Long", LatticeLaws[Long].boundedDistributiveLattice) + + checkAll("BigInt", RingLaws[BigInt].euclideanRing) + + checkAll("FPApprox[Float]", RingLaws[FPApprox[Float]].approxField) + checkAll("FPApprox[Double]", RingLaws[FPApprox[Double]].approxField) + + // let's limit our BigDecimal-related tests to the JVM for now. + if (Platform.isJvm) { + + { + // we need a less intense arbitrary big decimal implementation. + // this keeps the values relatively small/simple and avoids some + // of the numerical errors we might hit. + implicit val arbBigDecimal: Arbitrary[BigDecimal] = + Arbitrary(arbitrary[Int].map(x => BigDecimal(x, java.math.MathContext.UNLIMITED))) + + // BigDecimal does have numerical errors, so we can't pass all of + // the field laws. + checkAll("BigDecimal", RingLaws[BigDecimal].ring) + } + + { + // We check the full field laws using a FPApprox. + val mc = java.math.MathContext.DECIMAL32 + implicit val arbBigDecimal: Arbitrary[BigDecimal] = + Arbitrary(arbitrary[Double].map(x => BigDecimal(x, mc))) + implicit val epsBigDecimal = FPApprox.Epsilon.bigDecimalEpsilon(mc) + implicit val algebra: FPApproxAlgebra[BigDecimal] = + FPApprox.fpApproxAlgebra(new BigDecimalAlgebra(mc), Order[BigDecimal], epsBigDecimal) + checkAll("FPApprox[BigDecimal]", RingLaws[FPApprox[BigDecimal]].field(algebra)) + } + } else () + + { + implicit val arbBitSet: Arbitrary[BitSet] = + Arbitrary(arbitrary[List[Byte]].map(s => BitSet(s.map(_ & 0xff): _*))) + checkAll("BitSet", LogicLaws[BitSet].generalizedBool) + } + + checkAll("(Int, Int)", RingLaws[(Int, Int)].ring) + + checkAll("Unit", RingLaws[Unit].commutativeRing) + checkAll("Unit", RingLaws[Unit].multiplicativeMonoid) + checkAll("Unit", LatticeLaws[Unit].boundedSemilattice) + + { + // In order to check the monoid laws for `Order[N]`, we need + // `Arbitrary[Order[N]]` and `Eq[Order[N]]` instances. + // Here we have a bit of a hack to create these instances. + val nMax: Int = 13 + final case class N(n: Int) { require(n >= 0 && n < nMax) } + // The arbitrary `Order[N]` values are created by mapping N values to random + // integers. + implicit val arbNOrder: Arbitrary[Order[N]] = Arbitrary(arbitrary[Int].map { seed => + val order = new Random(seed).shuffle(Vector.range(0, nMax)) + Order.by { (n: N) => order(n.n) } + }) + // The arbitrary `Eq[N]` values are created by mapping N values to random + // integers. + implicit val arbNEq: Arbitrary[Eq[N]] = Arbitrary(arbitrary[Int].map { seed => + val mapping = new Random(seed).shuffle(Vector.range(0, nMax)) + Eq.by { (n: N) => mapping(n.n) } + }) + // needed because currently we don't have Vector instances + implicit val vectorNEq: Eq[Vector[N]] = Eq.fromUniversalEquals + // The `Eq[Order[N]]` instance enumerates all possible `N` values in a + // `Vector` and considers two `Order[N]` instances to be equal if they + // result in the same sorting of that vector. + implicit val NOrderEq: Eq[Order[N]] = Eq.by { (order: Order[N]) => + Vector.tabulate(nMax)(N).sorted(order.toOrdering) + } + implicit val NEqEq: Eq[Eq[N]] = new Eq[Eq[N]] { + def eqv(a: Eq[N], b: Eq[N]) = + Iterator + .tabulate(nMax)(N) + .flatMap { x => Iterator.tabulate(nMax)(N).map((x, _)) } + .forall { case (x, y) => a.eqv(x, y) == b.eqv(x, y) } + } + + implicit val monoidOrderN: Monoid[Order[N]] = Order.whenEqualMonoid[N] + checkAll("Order[N]", GroupLaws[Order[N]].monoid) + + { + implicit val bsEqN: BoundedSemilattice[Eq[N]] = Eq.allEqualBoundedSemilattice[N] + checkAll("Eq[N]", GroupLaws[Eq[N]].boundedSemilattice) + } + { + implicit val sEqN: Semilattice[Eq[N]] = Eq.anyEqualSemilattice[N] + checkAll("Eq[N]", GroupLaws[Eq[N]].semilattice) + } + } + + // checkAll("Int", "fromOrdering", OrderLaws[Int].order(Order.fromOrdering[Int])) + checkAll("Array[Int]", OrderLaws[Array[Int]].order) + checkAll("Array[Int]", OrderLaws[Array[Int]].partialOrder) + + // Rational tests do not return on Scala-js, so we make them JVM only. + if (Platform.isJvm) checkAll("Rat", RingLaws[Rat].field) + else () + + test("Field.fromDouble with subnormal") { + val n = 1.9726888167225064e-308 + val bd = new java.math.BigDecimal(n) + val unscaledValue = new BigInt(bd.unscaledValue) + val expected = + if (bd.scale > 0) { + Ring[Rat].fromBigInt(unscaledValue) / Ring[Rat].fromBigInt(BigInt(10).pow(bd.scale)) + } else { + Ring[Rat].fromBigInt(unscaledValue * BigInt(10).pow(-bd.scale)) + } + assert(Field.fromDouble[Rat](n) == expected) + } +} diff --git a/algebra-laws/shared/src/test/scala/algebra/laws/Rat.scala b/algebra-laws/shared/src/test/scala/algebra/laws/Rat.scala new file mode 100644 index 0000000000..8180c2c00b --- /dev/null +++ b/algebra-laws/shared/src/test/scala/algebra/laws/Rat.scala @@ -0,0 +1,136 @@ +package algebra +package laws + +import algebra.lattice.DistributiveLattice +import algebra.ring._ +import org.scalacheck.{Arbitrary, Gen} +import org.scalacheck.Arbitrary.arbitrary + +class Rat(val num: BigInt, val den: BigInt) extends Serializable { lhs => + + override def toString: String = + if (den == 1) s"$num" else s"$num/$den" + + override def equals(that: Any): Boolean = + that match { + case r: Rat => num == r.num && den == r.den + case _ => false + } + + override def hashCode(): Int = (num, den).## + + def isZero: Boolean = num == 0 + + def isOne: Boolean = num == 1 && den == 1 + + def compare(rhs: Rat): Int = + (lhs.num * rhs.den).compare(rhs.num * lhs.den) + + def abs: Rat = Rat(num.abs, den) + + def signum: Int = num.signum + + def +(rhs: Rat): Rat = + Rat((lhs.num * rhs.den) + (rhs.num * lhs.den), (lhs.den * rhs.den)) + + def unary_- : Rat = + Rat(-lhs.num, lhs.den) + + def *(rhs: Rat): Rat = + Rat(lhs.num * rhs.num, lhs.den * rhs.den) + + def /~(rhs: Rat) = lhs / rhs + + def %(rhs: Rat) = Rat.Zero + + def reciprocal: Rat = + if (num == 0) throw new ArithmeticException("/0") else Rat(den, num) + + def /(rhs: Rat): Rat = + lhs * rhs.reciprocal + + def **(k: Int): Rat = + Rat(num.pow(k), den.pow(k)) + + def toDouble: Double = num.toDouble / den.toDouble + + def toInt: Int = toDouble.toInt + + def isWhole: Boolean = den == 1 + + def ceil: Rat = + if (num >= 0) Rat((num + den - 1) / den, 1) + else Rat(num / den, 1) + + def floor: Rat = + if (num >= 0) Rat(num / den, 1) + else Rat((num - den + 1) / den, 1) + + def round: Rat = + if (num >= 0) Rat((num + (den / 2)) / den, 1) + else Rat((num - (den / 2)) / den, 1) + +} + +object Rat { + + val MinusOne: Rat = Rat(-1) + val Zero: Rat = Rat(0) + val One: Rat = Rat(1) + val Two: Rat = Rat(2) + + def apply(n: BigInt): Rat = + Rat(n, 1) + + def apply(num: BigInt, den: BigInt): Rat = + if (den == 0) throw new ArithmeticException("/0") + else if (den < 0) apply(-num, -den) + else if (num == 0) new Rat(0, 1) + else { + val g = num.gcd(den) + new Rat(num / g, den / g) + } + + def unapply(r: Rat): Some[(BigInt, BigInt)] = Some((r.num, r.den)) + + implicit val ratAlgebra: RatAlgebra = + new RatAlgebra + + val RatMinMaxLattice: DistributiveLattice[Rat] = + DistributiveLattice.minMax[Rat](ratAlgebra) + + // Is this horrible? Yes. Am I ashamed? Yes. + private[this] def genNonZero: Gen[BigInt] = + arbitrary[BigInt].flatMap { x => + if (x != 0) Gen.const(x) + else genNonZero + } + + implicit val ratArbitrary: Arbitrary[Rat] = + Arbitrary(for { + n <- arbitrary[BigInt] + d <- genNonZero + } yield Rat(n, d)) +} + +class RatAlgebra extends Field[Rat] with Order[Rat] with Serializable { + + def compare(x: Rat, y: Rat): Int = x.compare(y) + + val zero: Rat = Rat.Zero + val one: Rat = Rat.One + + def plus(a: Rat, b: Rat): Rat = a + b + def negate(a: Rat): Rat = -a + def times(a: Rat, b: Rat): Rat = a * b + override def reciprocal(a: Rat): Rat = a.reciprocal + def div(a: Rat, b: Rat): Rat = a / b + + override def fromInt(n: Int): Rat = Rat(n) + override def fromBigInt(n: BigInt): Rat = Rat(n) + + def isWhole(a: Rat): Boolean = a.isWhole + def ceil(a: Rat): Rat = a.ceil + def floor(a: Rat): Rat = a.floor + def round(a: Rat): Rat = a.round +} diff --git a/algebra-laws/shared/src/test/scala/algebra/laws/SimpleDeMorgan.scala b/algebra-laws/shared/src/test/scala/algebra/laws/SimpleDeMorgan.scala new file mode 100644 index 0000000000..70fa772cca --- /dev/null +++ b/algebra-laws/shared/src/test/scala/algebra/laws/SimpleDeMorgan.scala @@ -0,0 +1,53 @@ +package algebra +package laws + +import algebra.lattice.DeMorgan + +import org.scalacheck.Arbitrary +import org.scalacheck.Gen.oneOf + +/** + * The simplest De Morgan algebra that is not already a Boolean algebra. + * It is the standard three valued logic. + * Taken from https://en.wikipedia.org/wiki/De_Morgan_algebra#Kleene_algebra + */ +sealed trait SimpleDeMorgan + +object SimpleDeMorgan { + private case object False extends SimpleDeMorgan + private case object Unknown extends SimpleDeMorgan + private case object True extends SimpleDeMorgan + + implicit val deMorgan: DeMorgan[SimpleDeMorgan] = new DeMorgan[SimpleDeMorgan] { + def zero: SimpleDeMorgan = False + def one: SimpleDeMorgan = True + + def and(a: SimpleDeMorgan, b: SimpleDeMorgan): SimpleDeMorgan = (a, b) match { + case (False, _) => False + case (_, False) => False + case (Unknown, _) => Unknown + case (_, Unknown) => Unknown + case _ => True + } + + def or(a: SimpleDeMorgan, b: SimpleDeMorgan): SimpleDeMorgan = (a, b) match { + case (False, x) => x + case (x, False) => x + case (Unknown, x) => x + case (x, Unknown) => x + case _ => True + } + + def not(a: SimpleDeMorgan): SimpleDeMorgan = a match { + case False => True + case Unknown => Unknown + case True => False + } + } + + implicit val arbitrary: Arbitrary[SimpleDeMorgan] = Arbitrary(oneOf(False, Unknown, True)) + + implicit val eq: Eq[SimpleDeMorgan] = new Eq[SimpleDeMorgan] { + def eqv(x: SimpleDeMorgan, y: SimpleDeMorgan): Boolean = x == y + } +} diff --git a/algebra-laws/shared/src/test/scala/algebra/laws/SimpleHeyting.scala b/algebra-laws/shared/src/test/scala/algebra/laws/SimpleHeyting.scala new file mode 100644 index 0000000000..5b8d20ff65 --- /dev/null +++ b/algebra-laws/shared/src/test/scala/algebra/laws/SimpleHeyting.scala @@ -0,0 +1,59 @@ +package algebra +package laws + +import algebra.lattice.Heyting + +import org.scalacheck.Arbitrary +import org.scalacheck.Gen.oneOf + +/** + * The simplest Heyting algebra that is not already a Boolean algebra. + * Taken from https://en.wikipedia.org/wiki/Heyting_algebra#Examples + */ +sealed trait SimpleHeyting + +object SimpleHeyting { + private case object Zero extends SimpleHeyting + private case object Half extends SimpleHeyting + private case object One extends SimpleHeyting + + implicit val heyting: Heyting[SimpleHeyting] = new Heyting[SimpleHeyting] { + def zero: SimpleHeyting = Zero + def one: SimpleHeyting = One + + def and(a: SimpleHeyting, b: SimpleHeyting): SimpleHeyting = (a, b) match { + case (Zero, _) => Zero + case (_, Zero) => Zero + case (Half, _) => Half + case (_, Half) => Half + case _ => One + } + + def or(a: SimpleHeyting, b: SimpleHeyting): SimpleHeyting = (a, b) match { + case (Zero, x) => x + case (x, Zero) => x + case (Half, x) => x + case (x, Half) => x + case _ => One + } + + def complement(a: SimpleHeyting): SimpleHeyting = a match { + case Zero => One + case Half => Zero + case One => Zero + } + + def imp(a: SimpleHeyting, b: SimpleHeyting): SimpleHeyting = (a, b) match { + case (Zero, _) => One + case (_, Zero) => Zero + case (Half, _) => One + case (One, x) => x + } + } + + implicit val arbitrary: Arbitrary[SimpleHeyting] = Arbitrary(oneOf(Zero, Half, One)) + + implicit val eq: Eq[SimpleHeyting] = new Eq[SimpleHeyting] { + def eqv(x: SimpleHeyting, y: SimpleHeyting): Boolean = x == y + } +} diff --git a/build.sbt b/build.sbt index 90a9fd725b..7c5085249c 100644 --- a/build.sbt +++ b/build.sbt @@ -504,6 +504,10 @@ def mimaSettings(moduleName: String, includeCats1: Boolean = true) = exclude[MissingClassProblem]("cats.syntax.EqOps$"), exclude[MissingClassProblem]("cats.syntax.EqOps$mcF$sp"), exclude[MissingClassProblem]("cats.syntax.EqOps$mcI$sp") + ) ++ // https://github.com/typelevel/cats/pull/3918 + Seq( + exclude[MissingClassProblem]("algebra.laws.IsSerializable"), + exclude[MissingClassProblem]("algebra.laws.IsSerializable$") ) } ) @@ -540,21 +544,26 @@ lazy val catsJVM = project .settings(noPublishSettings) .settings(catsSettings) .settings(commonJvmSettings) - .aggregate(kernel.jvm, - kernelLaws.jvm, - core.jvm, - laws.jvm, - free.jvm, - testkit.jvm, - tests.jvm, - alleycatsCore.jvm, - alleycatsLaws.jvm, - alleycatsTests.jvm, - jvm + .aggregate( + kernel.jvm, + kernelLaws.jvm, + algebra.jvm, + algebraLaws.jvm, + core.jvm, + laws.jvm, + free.jvm, + testkit.jvm, + tests.jvm, + alleycatsCore.jvm, + alleycatsLaws.jvm, + alleycatsTests.jvm, + jvm ) .dependsOn( kernel.jvm, kernelLaws.jvm, + algebra.jvm, + algebraLaws.jvm, core.jvm, laws.jvm, free.jvm, @@ -574,6 +583,8 @@ lazy val catsJS = project .settings(commonJsSettings) .aggregate(kernel.js, kernelLaws.js, + algebra.js, + algebraLaws.js, core.js, laws.js, free.js, @@ -587,6 +598,8 @@ lazy val catsJS = project .dependsOn( kernel.js, kernelLaws.js, + algebra.js, + algebraLaws.js, core.js, laws.js, free.js, @@ -608,6 +621,8 @@ lazy val catsNative = project .aggregate( kernel.native, kernelLaws.native, + algebra.native, + algebraLaws.native, core.native, laws.native, free.native, @@ -621,6 +636,8 @@ lazy val catsNative = project .dependsOn( kernel.native, kernelLaws.native, + algebra.native, + algebraLaws.native, core.native, laws.native, free.native, @@ -662,6 +679,44 @@ lazy val kernelLaws = crossProject(JSPlatform, JVMPlatform, NativePlatform) .dependsOn(kernel) .nativeSettings(commonNativeSettings) +lazy val algebra = crossProject(JSPlatform, JVMPlatform, NativePlatform) + .crossType(CrossType.Pure) + .in(file("algebra-core")) + .settings(moduleName := "algebra", name := "Cats algebra") + .dependsOn(kernel) + .settings(commonSettings) + .settings(publishSettings) + .settings(Compile / sourceGenerators += (Compile / sourceManaged).map(AlgebraBoilerplate.gen).taskValue) + .settings(includeGeneratedSrc) + .jsSettings(commonJsSettings) + .jvmSettings( + commonJvmSettings ++ mimaSettings("algebra") ++ Seq( + mimaPreviousArtifacts := Set("org.typelevel" %% "algebra" % "2.2.3") + ) + ) + .nativeSettings(commonNativeSettings) + .settings(testingDependencies) + .settings( + libraryDependencies += "org.scalacheck" %%% "scalacheck" % scalaCheckVersion % Test + ) + +lazy val algebraLaws = crossProject(JSPlatform, JVMPlatform, NativePlatform) + .in(file("algebra-laws")) + .settings(moduleName := "algebra-laws", name := "Cats algebra laws") + .settings(commonSettings) + .settings(publishSettings) + .settings(disciplineDependencies) + .settings(testingDependencies) + .settings(Test / scalacOptions := (Test / scalacOptions).value.filter(_ != "-Xfatal-warnings")) + .jsSettings(commonJsSettings) + .jvmSettings( + commonJvmSettings ++ mimaSettings("algebra-laws") ++ Seq( + mimaPreviousArtifacts := Set("org.typelevel" %% "algebra-laws" % "2.2.3") + ) + ) + .dependsOn(kernelLaws, algebra) + .nativeSettings(commonNativeSettings) + lazy val core = crossProject(JSPlatform, JVMPlatform, NativePlatform) .crossType(CrossType.Pure) .dependsOn(kernel) diff --git a/docs/src/main/mdoc/algebra.md b/docs/src/main/mdoc/algebra.md new file mode 100644 index 0000000000..9607cc7685 --- /dev/null +++ b/docs/src/main/mdoc/algebra.md @@ -0,0 +1,113 @@ +--- +layout: docs +title: "Algebra Overview" +section: "algebra" +--- + +# Algebra Overview + +Algebra uses type classes to represent algebraic structures. You can use these type classes to represent the abstract capabilities (and requirements) you want generic parameters to possess. + +This section will explain the structures available. + +## algebraic properties and terminology + +We will be talking about properties like *associativity* and *commutativity*. Here is a quick explanation of what those properties mean: + +|Name |Description | +|-------------|--------------------------------------------------------------------------------| +|Associative | If `⊕` is associative, then `a ⊕ (b ⊕ c)` = `(a ⊕ b) ⊕ c`. | +|Commutative | If `⊕` is commutative, then `a ⊕ b` = `b ⊕ a`. | +|Identity | If `id` is an identity for `⊕`, then `a ⊕ id` = `id ⊕ a` = `a`. | +|Inverse | If `¬` is an inverse for `⊕` and `id`, then `a ⊕ ¬a` = `¬a ⊕ a` = `id`. | +|Distributive | If `⊕` and `⊙` distribute, then `a ⊙ (b ⊕ c)` = `(a ⊙ b) ⊕ (a ⊙ c)` and `(a ⊕ b) ⊙ c` = `(a ⊙ c) ⊕ (b ⊙ c)`. | +|Idempotent | If `⊕` is idempotent, then `a ⊕ a` = `a`. If `f` is idempotent, then `f(f(a))` = `f(a)` | + +Though these properties are illustrated with symbolic operators, they work equally-well with functions. When you see `a ⊕ b` that is equivalent to `f(a, b)`: `⊕` is an infix representation of the binary function `f`, and `a` and `b` are values (of some type `A`). + +Similarly, when you see `¬a` that is equivalent to `g(a)`: `¬` is a prefix representation of the unary function `g`, and `a` is a value (of some type `A`). + +## basic algebraic structures + +The most basic structures can be found in the `algebra` package. They all implement a method called `combine`, which is associative. The identity element (if present) will be called `empty`, and the inverse method (if present) will be called `inverse`. + +|Name |Associative?|Commutative?|Identity?|Inverse?|Idempotent?| +|--------------------|------------|------------|---------|--------|-----------| +|Semigroup | ✓| | | | | +|CommutativeSemigroup| ✓| ✓| | | | +|Monoid | ✓| | ✓| | | +|Band | ✓| | | | ✓| +|Semilattice | ✓| ✓| | | ✓| +|Group | ✓| | ✓| ✓| | +|CommutativeMonoid | ✓| ✓| ✓| | | +|CommutativeGroup | ✓| ✓| ✓| ✓| | +|BoundedSemilattice | ✓| ✓| ✓| | ✓| + +(For a description of what each column means, see [§algebraic properties and terminology](#algebraic-properties-and-terminology).) + +## ring-like structures + +The `algebra.ring` package contains more sophisticated structures which combine an *additive* operation (called `plus`) and a *multiplicative* operation (called `times`). Additive identity and inverses will be called `zero` and `negate` (respectively); multiplicative identity and inverses will be called `one` and `reciprocal` (respectively). + +All ring-like structures are associative for both `+` and `*`, have commutative `+`, and have a `zero` element (an identity for `+`). + +|Name |Has `negate`?|Has `1`?|Has `reciprocal`?|Commutative `*`?| +|--------------------|-------------|--------|-----------------|----------------| +|Semiring | | | | | +|Rng | ✓| | | | +|Rig | | ✓| | | +|CommutativeRig | | ✓| | ✓| +|Ring | ✓| ✓| | | +|CommutativeRing | ✓| ✓| | ✓| +|Field | ✓| ✓| ✓| ✓| + +With the exception of `CommutativeRig` and `Rng`, every lower structure is also an instance of the structures above it. For example, every `Ring` is a `Rig`, every `Field` is a `CommutativeRing`, and so on. + +(For a description of what the terminology in each column means, see [§algebraic properties and terminology](#algebraic-properties-and-terminology).) + +## lattice-like structures + +The `algebra.lattice` package contains more structures that can be somewhat ring-like. Rather than `plus` and `times` we have `meet` and `join` both of which are always associative, commutative and idempotent, and as such each can be viewed as a semilattice. Meet can be thought of as the greatest lower bound of two items while join can be thought of as the least upper bound between two items. + +When `zero` is present, `join(a, zero)` = `a`. When `one` is present `meet(a, one)` = `a`. + +When `meet` and `join` are both present, they obey the absorption law: + + - `meet(a, join(a, b))` = `join(a, meet(a, b)) = a` + +Sometimes meet and join distribute, we say it is distributive in this case: + + - `meet(a, join(b, c))` = `join(meet(a, b), meet(a, c))` + - `join(a, meet(b, c))` = `meet(join(a, b), join(a, c))` + +Sometimes an additional binary operation `imp` (for impliciation, also written as →, meet written as ∧) is present. Implication obeys the following laws: + + - `a → a` = `1` + - `a ∧ (a → b)` = `a ∧ b` + - `b ∧ (a → b)` = `b` + - `a → (b ∧ c)` = `(a → b) ∧ (a → c)` + +The law of the excluded middle can be expressed as: + + - `(a ∨ (a → 0))` = `1` + +|Name |Has `join`?|Has `meet`?|Has `zero`?|Has `one`?|Distributive|Has `imp`?|Excludes middle?| +|--------------------------|-----------|-----------|-----------|----------|------------|----------|----------------| +|JoinSemilattice | ✓| | | | | | | +|MeetSemilattice | | ✓| | | | | | +|BoundedJoinSemilattice | ✓| | ✓| | | | | +|BoundedMeetSemilattice | | ✓| | ✓| | | | +|Lattice | ✓| ✓| | | | | | +|DistributiveLattice | ✓| ✓| | | ✓| | | +|BoundedLattice | ✓| ✓| ✓| ✓| | | | +|BoundedDistributiveLattice| ✓| ✓| ✓| ✓| ✓| | | +|Heyting | ✓| ✓| ✓| ✓| ✓| ✓| | +|Bool | ✓| ✓| ✓| ✓| ✓| ✓| ✓| + +Note that a `BoundedDistributiveLattice` gives you a `CommutativeRig`, but not the other way around: rigs aren't distributive with `a + (b * c) = (a + b) * (a + c)`. + +Also, a `Bool` gives rise to a `BoolRing`, since each element can be defined as its own negation. Note, Bool's `.asBoolRing` is not an extension of the `.asCommutativeRig` method as the `plus` operations are defined differently. + +### Documentation Help + +We'd love your help with this documentation! You can edit this page in your browser by clicking [this link](https://github.com/typelevel/cats/edit/master/docs/src/main/mdoc/algebra.md). diff --git a/project/AlgebraBoilerplate.scala b/project/AlgebraBoilerplate.scala new file mode 100644 index 0000000000..cac6c3f7ee --- /dev/null +++ b/project/AlgebraBoilerplate.scala @@ -0,0 +1,161 @@ +import sbt._ + +/** + * Generate a range of boilerplate classes that would be tedious to write and maintain by hand. + * + * Copied, with some modifications, from + * [[https://github.com/milessabin/shapeless/blob/master/project/Boilerplate.scala Shapeless]]. + * + * @author Miles Sabin + * @author Kevin Wright + */ +object AlgebraBoilerplate { + import scala.StringContext._ + + implicit class BlockHelper(val sc: StringContext) extends AnyVal { + def block(args: Any*): String = { + val interpolated = sc.standardInterpolator(treatEscapes, args) + val rawLines = interpolated.split('\n') + val trimmedLines = rawLines.map(_.dropWhile(_.isWhitespace)) + trimmedLines.mkString("\n") + } + } + + val templates: Seq[Template] = Seq( + GenTupleInstances + ) + + val header = "// auto-generated boilerplate" + val maxArity = 22 + + /** + * Return a sequence of the generated files. + * + * As a side-effect, it actually generates them... + */ + def gen(dir: File): Seq[File] = templates.map { template => + val tgtFile = template.filename(dir) + IO.write(tgtFile, template.body) + tgtFile + } + + class TemplateVals(val arity: Int) { + val synTypes = (0 until arity).map(n => s"A$n") + val synVals = (0 until arity).map(n => s"a$n") + val `A..N` = synTypes.mkString(", ") + val `a..n` = synVals.mkString(", ") + val `_.._` = Seq.fill(arity)("_").mkString(", ") + val `(A..N)` = if (arity == 1) "Tuple1[A0]" else synTypes.mkString("(", ", ", ")") + val `(_.._)` = if (arity == 1) "Tuple1[_]" else Seq.fill(arity)("_").mkString("(", ", ", ")") + val `(a..n)` = if (arity == 1) "Tuple1(a)" else synVals.mkString("(", ", ", ")") + } + + /** + * Blocks in the templates below use a custom interpolator, combined with post-processing to + * produce the body. + * + * - The contents of the `header` val is output first + * - Then the first block of lines beginning with '|' + * - Then the block of lines beginning with '-' is replicated once for each arity, + * with the `templateVals` already pre-populated with relevant relevant vals for that arity + * - Then the last block of lines prefixed with '|' + * + * The block otherwise behaves as a standard interpolated string with regards to variable + * substitution. + */ + trait Template { + def filename(root: File): File + def content(tv: TemplateVals): String + def range: IndexedSeq[Int] = 1 to maxArity + def body: String = { + val headerLines = header.split('\n') + val raw = range.map(n => content(new TemplateVals(n)).split('\n').filterNot(_.isEmpty)) + val preBody = raw.head.takeWhile(_.startsWith("|")).map(_.tail) + val instances = raw.flatMap(_.filter(_.startsWith("-")).map(_.tail)) + val postBody = raw.head.dropWhile(_.startsWith("|")).dropWhile(_.startsWith("-")).map(_.tail) + (headerLines ++ preBody ++ instances ++ postBody).mkString("\n") + } + } + + object GenTupleInstances extends Template { + override def range: IndexedSeq[Int] = 1 to maxArity + + def filename(root: File): File = root / "algebra" / "instances" / "TupleAlgebra.scala" + + def content(tv: TemplateVals): String = { + import tv._ + + def constraints(constraint: String) = + synTypes.map(tpe => s"${tpe}: ${constraint}[${tpe}]").mkString(", ") + + def tuple(results: TraversableOnce[String]) = { + val resultsVec = results.toVector + val a = synTypes.size + val r = s"${0.until(a).map(i => resultsVec(i)).mkString(", ")}" + if (a == 1) "Tuple1(" ++ r ++ ")" + else s"(${r})" + } + + def binMethod(name: String) = + synTypes.zipWithIndex.iterator.map { case (tpe, i) => + val j = i + 1 + s"${tpe}.${name}(x._${j}, y._${j})" + } + + def binTuple(name: String) = + tuple(binMethod(name)) + + def unaryTuple(name: String) = { + val m = synTypes.zipWithIndex.map { case (tpe, i) => s"${tpe}.${name}(x._${i + 1})" } + tuple(m) + } + + def nullaryTuple(name: String) = { + val m = synTypes.map(tpe => s"${tpe}.${name}") + tuple(m) + } + + block""" + |package algebra + |package instances + | + |import algebra.ring.{Rig, Ring, Rng, Semiring} + | + |trait TupleInstances extends cats.kernel.instances.TupleInstances { + - + - implicit def tuple${arity}Rig[${`A..N`}](implicit ${constraints("Rig")}): Rig[${`(A..N)`}] = + - new Rig[${`(A..N)`}] { + - def one: ${`(A..N)`} = ${nullaryTuple("one")} + - def plus(x: ${`(A..N)`}, y: ${`(A..N)`}): ${`(A..N)`} = ${binTuple("plus")} + - def times(x: ${`(A..N)`}, y: ${`(A..N)`}): ${`(A..N)`} = ${binTuple("times")} + - def zero: ${`(A..N)`} = ${nullaryTuple("zero")} + - } + - + - implicit def tuple${arity}Ring[${`A..N`}](implicit ${constraints("Ring")}): Ring[${`(A..N)`}] = + - new Ring[${`(A..N)`}] { + - def one: ${`(A..N)`} = ${nullaryTuple("one")} + - def plus(x: ${`(A..N)`}, y: ${`(A..N)`}): ${`(A..N)`} = ${binTuple("plus")} + - def times(x: ${`(A..N)`}, y: ${`(A..N)`}): ${`(A..N)`} = ${binTuple("times")} + - def zero: ${`(A..N)`} = ${nullaryTuple("zero")} + - def negate(x: ${`(A..N)`}): ${`(A..N)`} = ${unaryTuple("negate")} + - } + - + - implicit def tuple${arity}Rng[${`A..N`}](implicit ${constraints("Rng")}): Rng[${`(A..N)`}] = + - new Rng[${`(A..N)`}] { + - def plus(x: ${`(A..N)`}, y: ${`(A..N)`}): ${`(A..N)`} = ${binTuple("plus")} + - def times(x: ${`(A..N)`}, y: ${`(A..N)`}): ${`(A..N)`} = ${binTuple("times")} + - def zero: ${`(A..N)`} = ${nullaryTuple("zero")} + - def negate(x: ${`(A..N)`}): ${`(A..N)`} = ${unaryTuple("negate")} + - } + - + - implicit def tuple${arity}Semiring[${`A..N`}](implicit ${constraints("Semiring")}): Semiring[${`(A..N)`}] = + - new Semiring[${`(A..N)`}] { + - def plus(x: ${`(A..N)`}, y: ${`(A..N)`}): ${`(A..N)`} = ${binTuple("plus")} + - def times(x: ${`(A..N)`}, y: ${`(A..N)`}): ${`(A..N)`} = ${binTuple("times")} + - def zero: ${`(A..N)`} = ${nullaryTuple("zero")} + - } + |} + """ + } + } +}