spire.laws.BaseLaws.scala Maven / Gradle / Ivy
package spire
package laws
import spire.algebra._
import spire.implicits._
import org.typelevel.discipline.Laws
import org.scalacheck.Arbitrary
import org.scalacheck.Prop._
object BaseLaws {
def apply[A : Eq : Arbitrary] = new BaseLaws[A] {
def Equ = Eq[A]
def Arb = implicitly[Arbitrary[A]]
}
}
trait BaseLaws[A] extends Laws {
implicit def Equ: Eq[A]
implicit def Arb: Arbitrary[A]
def signed(implicit A: Signed[A]) = new SimpleRuleSet(
name = "signed",
"abs non-negative" → forAll((x: A) =>
x.abs.sign != Sign.Negative
),
"signum returns -1/0/1" → forAll((x: A) =>
x.signum.abs <= 1
),
"signum is sign.toInt" → forAll((x: A) =>
x.signum == x.sign.toInt
)
)
def metricSpace[R](implicit MSA: MetricSpace[A, R], SR: Signed[R], OR: Order[R], ASR: AdditiveSemigroup[R]) = new SimpleRuleSet(
name = "metricSpace",
"non-negative" → forAll((a1: A, a2: A) =>
MSA.distance(a1, a2).sign != Sign.Negative
),
"identity" → forAll((a: A) =>
MSA.distance(a, a).sign == Sign.Zero
),
"equality" → forAll((a1: A, a2: A) =>
// generating equal values is hard, and Scalacheck will give up if it can't
// hence, not using `==>` here
a1 =!= a2 || MSA.distance(a1, a2).sign == Sign.Zero
),
"symmetry" → forAll((a1: A, a2: A) =>
MSA.distance(a1, a2) === MSA.distance(a2, a1)
),
"triangleInequality" → forAll((a1: A, a2: A, a3: A) =>
(MSA.distance(a1, a2) + MSA.distance(a2, a3)) >= MSA.distance(a1, a3)
)
)
}
// vim: expandtab:ts=2:sw=2
© 2015 - 2025 Weber Informatics LLC | Privacy Policy