aboutsummaryrefslogblamecommitdiff
path: root/tests/run/t6196.scala
blob: 9ded3b9c0c48abc017fd8cf581f14356abbd03e0 (plain) (tree)
1
2
3
4
5
6
7
8
9
10

                                         
                                             






                                                               
                                                                      













                                                                                                         
                             





                                                                         
                               


































                                                                                                                 
import scala.collection.immutable.HashSet

object Test extends dotty.runtime.LegacyApp {

  case class Collision(value: Int) extends Ordered[Collision] {
    def compare(that:Collision) = value compare that.value

    override def hashCode = value / 5
  }

  def testCorrectness[T : Ordering](n: Int, mkKey: Int => T): Unit = {
    val o = implicitly[Ordering[T]]
    val s = HashSet.empty[T] ++ (0 until n).map(mkKey)
    for (i <- 0 until n) {
      val ki = mkKey(i)
      val a = s.filter(o.lt(_,ki))
      val b = s.filterNot(o.lt(_,ki))
      require(a.size == i && (0 until i).forall(i => a.contains(mkKey(i))))
      require(b.size == n - i && (i until n).forall(i => b.contains(mkKey(i))))
    }
  }

  // this tests the structural sharing of the new filter
  // I could not come up with a simple test that tests structural sharing when only parts are reused, but
  // at least this fails with the old and passes with the new implementation
  def testSharing(): Unit = {
    val s = HashSet.empty[Int] ++ (0 until 100)
    require(s.filter(_ => true) eq s)
    require(s.filterNot(_ => false) eq s)
  }

  // this tests that neither hashCode nor equals are called during filter
  def testNoHashing(): Unit = {
    var hashCount = 0
    var equalsCount = 0
    case class HashCounter(value:Int) extends Ordered[HashCounter] {
      def compare(that:HashCounter) = value compare that.value

      override def hashCode = {
        hashCount += 1
        value
      }

      override def equals(that:Any) = {
        equalsCount += 1
        that match {
          case HashCounter(value) => this.value == value
          case _ => false
        }
      }
    }

    val s = HashSet.empty[HashCounter] ++ (0 until 100).map(HashCounter)
    val hashCount0 = hashCount
    val equalsCount0 = equalsCount
    val t = s.filter(_<HashCounter(50))
    require(hashCount == hashCount0)
    require(equalsCount == equalsCount0)
  }

  // this tests correctness of filter and filterNot for integer keys
  testCorrectness[Int](100, identity _)
  // this tests correctness of filter and filterNot for keys with lots of collisions
  // this is necessary because usually collisions are rare so the collision-related code is not thoroughly tested
  testCorrectness[Collision](100, Collision.apply _)
  testSharing()
  testNoHashing()
}