From f07864b80f32f93fe92538eac8ba037f3a4a20d0 Mon Sep 17 00:00:00 2001 From: Vasil Vasilev Date: Fri, 18 Jun 2021 15:18:50 +0200 Subject: [PATCH] Port the StripedHashtableSpec to Scala.js --- .../effect/unsafe/StripedHashtableSpec.scala | 86 +++++++++++++++++++ 1 file changed, 86 insertions(+) create mode 100644 tests/js/src/test/scala/cats/effect/unsafe/StripedHashtableSpec.scala diff --git a/tests/js/src/test/scala/cats/effect/unsafe/StripedHashtableSpec.scala b/tests/js/src/test/scala/cats/effect/unsafe/StripedHashtableSpec.scala new file mode 100644 index 0000000000..3aba8629dd --- /dev/null +++ b/tests/js/src/test/scala/cats/effect/unsafe/StripedHashtableSpec.scala @@ -0,0 +1,86 @@ +/* + * Copyright 2020-2021 Typelevel + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package cats.effect +package unsafe + +import cats.syntax.parallel._ + +import scala.concurrent.{Future, Promise} +import scala.concurrent.duration._ + +class StripedHashtableSpec extends BaseSpec with Runners { + + override def executionTimeout: FiniteDuration = 30.seconds + + def hashtableRuntime(): IORuntime = + new IORuntime( + IORuntime.defaultComputeExecutionContext, + IORuntime.defaultComputeExecutionContext, + IORuntime.defaultScheduler, + () => (), + IORuntimeConfig() + ) + + "StripedHashtable" should { + "work correctly in the presence of many unsafeRuns" in real { + val iterations = 10000 + + object Boom extends RuntimeException("Boom!") + + def io(n: Int): IO[Unit] = + (n % 3) match { + case 0 => IO.unit + case 1 => IO.canceled + case 2 => IO.raiseError[Unit](Boom) + } + + Resource.make(IO(hashtableRuntime()))(rt => IO(rt.shutdown())).use { rt => + IO(new CountDownLatch(iterations)).flatMap { counter => + (0 until iterations) + .toList + .parTraverse { n => IO(io(n).unsafeRunAsync { _ => counter.countDown() }(rt)) } + .flatMap { _ => IO.fromFuture(IO.delay(counter.await())) } + .flatMap { _ => + IO.blocking { + rt.fiberErrorCbs.synchronized { + rt.fiberErrorCbs.tables.forall(_.isEmpty) mustEqual true + } + } + } + } + } + } + } + + /** + * This implementation only works on Scala.js as it relies on a single + * threaded execution model. + */ + private final class CountDownLatch(private var counter: Int) { + private val promise: Promise[Unit] = Promise() + + def countDown(): Unit = { + counter -= 1 + if (counter == 0) { + promise.success(()) + } + } + + def await(): Future[Unit] = + promise.future + } +}