All Downloads are FREE. Search and download functionalities are using the official Maven repository.

horan.kse3-testing_3.0.2.11.source-code.TestUtilities.scala Maven / Gradle / Ivy

// This file is distributed under the BSD 3-clause license.  See file LICENSE.
// Copyright (c) 2022-23 Rex Kerr and Calico Life Sciences LLC.

package kse.testutilities

import scala.collection.generic.IsIterable
import scala.reflect.{ClassTag, TypeTest}
import scala.util.{Try, Success, Failure}
import scala.util.control.ControlThrowable


/** This provides a way to write unit tests.
  *
  * It's really easy to use once you're used to it!
  * 
  * Unfortunately, it doesn't have any documentation at all.
  * You can read the unit tests for kse.flow etc. to see how it is used.
  * 
  * See build.sc for how to include it in your own tests.
  */
object TestUtilities {
  class Asserter(
        assertEq:    (String, Any, Any) => Unit,
        assertNotEq: (String, Any, Any) => Unit,
    val assertTrue:  (String, Boolean)  => Unit,
        isScalaEquality: Boolean = false
  ) {
    /** Checks Scala equality before passing on to error handler */
    def assertEquals(message: String, expect: Any, actual: Any): Unit =
      if isScalaEquality || expect != actual then assertEq(message, expect, actual)

    def assertNotEquals(message: String, expect: Any, actual: Any): Unit =
      if isScalaEquality || expect == actual then assertNotEq(message, expect, actual)
  }

  trait Approximation[A] {
    def approx(a0: A, a1: A): Boolean
  }
  object Approximation {
    class OfFloat(eps: Float, thresh: Float, tol: Float) extends Approximation[Float] {
      def approx(a0: Float, a1: Float) =
        (a0 == a1) ||
        (a0.isNaN && a1.isNaN) ||
        {
          val delta = math.abs(a0 - a1)
          val a = math.abs(a0) max math.abs(a1)
          if a > thresh then delta < eps*a
          else delta < tol
        }
    }

    class OfDouble(eps: Double, thresh: Double, tol: Double) extends Approximation[Double] {
      def approx(a0: Double, a1: Double) =
        (a0 == a1) ||
        (a0.isNaN && a1.isNaN) ||
        {
          val delta = math.abs(a0 - a1)
          val a = math.abs(a0) max math.abs(a1)
          val ans =
            if a > thresh then delta < eps*a
            else delta < tol
          ans
        }
    }

    given defaultFloatApprox: Approximation[Float] = new OfFloat(1e-6f, 1f, 1e-6f)

    given defaultDoubleApprox: Approximation[Double] = new OfDouble(1e-9, 1.0, 1e-9)
  }

  case class Thrown(tag: ClassTag[?])(val classname: String) extends ControlThrowable(classname) {}
  def thrown[A](using tag: ClassTag[A]): Thrown = Thrown(tag)(tag.runtimeClass.getName)

  class TypeGen[B, C](val typed: Typed[B], val gen: () => C) {}

  case class Typed[A](unit: Unit = ()) {
    def --:[C](c: => C) = new TypeGen[A, C](this, () => c)

    def :--[B](b: B) = (this, b)

    def :==:[B](b: B)(using B =:= A): Unit = {}
  }
  def typed[A]: Typed[A] = new Typed[A]()
  def typedLike[A](a: A): Typed[A] = new Typed[A]()

  case class RunType[A](tag: ClassTag[A]) {}
  def runtype[A](using tag: ClassTag[A]): RunType[A] = RunType(tag)

  trait Messaging {
    def message: String = if mline.isEmpty then mline else s"${mline}\n"
    def mline: String
  }

  class Labeled[A](val mline: String, val value: () => A)(using asr: Asserter, ln: sourcecode.Line, fl: sourcecode.FileName) extends Messaging {
    import asr._

    override def message = s"error at ${fl.value}:${ln.value}\n" + super.message

    def isEqual[B](b: => B): Unit =
      val ta = Try{ value() }
      val vb = b
      vb match
        case t @ Thrown(tag) => ta match
          case Failure(x) =>
            if !tag.runtimeClass.isAssignableFrom(x.getClass) then
              assertEquals(message, ta, Failure(t))
          case _ => assertEquals(message, ta, Failure(t))
        case _ => assertEquals(message, ta.get, vb)

    def ====(n: Null): Unit =
      isEqual(null)

    def ====[B, C](bc: TypeGen[B, C])(using B =:= A): Unit =
      isEqual(bc.gen())

    def ====[B](t: Typed[B])(using B =:= A): Unit = {}

    def ====[B](t: RunType[B]): Unit =
      val v = value()
      if !t.tag.runtimeClass.isAssignableFrom(v.getClass) then 
        assertEquals(message, v.getClass, t.tag.runtimeClass)

    def ====[B](b: => B): Unit =
      isEqual(b)

    def =!!=[B](b: => B): Unit = Try{ value() } match
      case Success(v) => assertNotEquals(message, v, b)
      case Failure(x) => b match
        case t @ Thrown(tag) =>
          if tag.runtimeClass.isAssignableFrom(x.getClass) then
            assertTrue(s"${message}Did not expect $x\nto be a ${t.classname}", false)
        case _ =>

    def =~~=[B >: A](b: => B)(using apx: Approximation[B]): Unit =
      val va = value()
      val vb = b
      if !apx.approx(va, vb) then assertEquals(message, va, vb)
  }

  class LabeledCollection[C, I <: IsIterable[C]](val mline: String, val value: () => C, val ii: I)(using asr: Asserter, ln: sourcecode.Line, fl: sourcecode.FileName) extends Messaging {
    import asr._

    override def message = s"error at ${fl.value}:${ln.value}\n" + super.message

    private def pickMoreElements[A](count: Int, index: Int, it: Iterator[A], acc: List[String] = Nil): List[String] =
      if count <= 0 || ! it.hasNext then
        var result = acc
        while result.forall(_ startsWith "   ") do
          result = result.map(_.drop(1))
        if it.hasNext then ("  ..." :: result).reverse else result.reverse
      else if !it.hasNext then acc.reverse
      else
        val longest = (count + index).toString
        var istr = "  #" + index.toString
        if istr.length < 3+longest.length then istr = " "*(3+longest.length - istr.length) + istr
        pickMoreElements(count-1, index+1, it, s"$istr = ${it.next}" :: acc)

    def =**=[D, J <: IsIterable[D]](d: => D)(using jj: J): Unit =
      var i = 0
      val ia = ii(value()).iterator
      val ib = jj(d).iterator
      while ia.hasNext && ib.hasNext do
        val va = ia.next
        val vb = ib.next
        if va != vb then
          assertEquals(message + s"\nerror at index $i", va, vb)
        i += 1
      if ia.hasNext then
        val elts = pickMoreElements(4, i, ia)
        val plural = if elts.length > 1 then "Too many elements.  Extra:" else "One extra element:"
        val lines = elts.mkString("", "\n", "\n")
        assertTrue(s"${message}$plural\n$lines", false)
      if ib.hasNext then
        val elts = pickMoreElements(4, i, ib)
        val plural = if elts.length > 1 then "Not enough elements.  Missed:" else "Missed one element:"
        val lines = elts.mkString("", "\n", "\n")
        assertTrue(s"${message}$plural\n$lines", false)

    def contains[B](b: => B): Unit =
      val ia = ii(value()).iterator
      val vb = b
      while ia.hasNext do
        val va = ia.next
        if va == vb then
          return
      assertTrue(s"${message}could not find an element matching $vb", false )

    def exists(f: ii.A => Boolean): Unit =
      val ia = ii(value()).iterator
      while ia.hasNext do
        if f(ia.next) then return
      assertTrue(s"${message}collection never passed test", false )
  }

  trait GenLabeled {
    def message: String

    def ~[A](a: => A)(using asr: Asserter, ln: sourcecode.Line, fl: sourcecode.FileName): Labeled[A] =
      Labeled(message, () => a)

    def ~[A](a: => A)(using ii: IsIterable[A], asr: Asserter, ln: sourcecode.Line, fl: sourcecode.FileName): LabeledCollection[A, ii.type] =
      LabeledCollection[A, ii.type](message, () => a, ii)
  }

  object T extends GenLabeled {
    def message = ""

    def apply(msg: String): GenLabeled = new GenLabeled { def message = msg }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy