From 16ede17746fdf90ada1a9a2910d94b053df31de7 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Sat, 24 Jun 2023 12:43:55 -0700 Subject: [PATCH 01/46] Add release notes config [ci skip] --- .github/release.yml | 15 +++++++++++++++ 1 file changed, 15 insertions(+) create mode 100644 .github/release.yml diff --git a/.github/release.yml b/.github/release.yml new file mode 100644 index 0000000000..6dc5f68c19 --- /dev/null +++ b/.github/release.yml @@ -0,0 +1,15 @@ +changelog: + categories: + - title: Features + labels: + - :mushroom: enhancement + - title: Bug Fixes + labels: + - :beetle: bug + - title: Behind the Scenes + labels: + - :gear: infrastructure + - :robot: + - title: Documentation + labels: + - :books: docs From 9f81c229eba72f2c2f407c029b40bf16223d9bca Mon Sep 17 00:00:00 2001 From: Daniel Spiewak Date: Sat, 24 Jun 2023 14:45:01 -0500 Subject: [PATCH 02/46] Added script for automatically posting release to discord --- scripts/post-release-discord.sh | 19 +++++++++++++++++++ 1 file changed, 19 insertions(+) create mode 100755 scripts/post-release-discord.sh diff --git a/scripts/post-release-discord.sh b/scripts/post-release-discord.sh new file mode 100755 index 0000000000..fd912bb09a --- /dev/null +++ b/scripts/post-release-discord.sh @@ -0,0 +1,19 @@ +#!/usr/bin/env bash + +set -euo pipefail +IFS=$'\n\t' + +if [[ $# -ne 1 ]] || [[ "$1" == "--help" ]]; then + echo "usage: $0 new-version" + exit 1 +fi + +# defunct example: +# +# export DISCORD_WEBHOOK_URL='https://discord.com/api/webhooks/1122248098014564483/qv_hfoMpexcTjX_8FX23uisvpqrt_N_UD8VtYFLzUo8ROthEWk5cqECQPB3OCJ9MNUxB' +# +# this url should be considered a secret and handled with appropriate care + +data="{\"content\":\"https://github.com/typelevel/cats-effect/releases/tag/v$1\"}" + +exec curl -H "Content-Type: application/json" -X POST -d "$data" "$DISCORD_WEBHOOK_URL" From 612a97479d2c240c2284de00cf85d5d4fc8f54ee Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Sun, 25 Jun 2023 17:08:39 +0000 Subject: [PATCH 03/46] Fix major branch PR in release script [ci skip] --- scripts/make-release-prs.sh | 14 +++++++++----- 1 file changed, 9 insertions(+), 5 deletions(-) diff --git a/scripts/make-release-prs.sh b/scripts/make-release-prs.sh index c8384d3469..fcc2fe705f 100755 --- a/scripts/make-release-prs.sh +++ b/scripts/make-release-prs.sh @@ -16,28 +16,32 @@ new_version="$2" minor_base=series/$(echo $new_version | sed -E 's/([0-9]+).([0-9]+).[0-9]+/\1.\2.x/') major_base=series/$(echo $new_version | sed -E 's/([0-9]+).[0-9]+.[0-9]+/\1.x/') -branch="release/$new_version-minor" +minor_branch="release/$new_version-minor" +major_branch="release/$new_version-major" cd "$(mktemp -d)" git clone git@github.com:typelevel/cats-effect.git cd 'cats-effect' -git checkout -b $branch origin/$minor_base +git checkout -b $minor_branch origin/$minor_base "$primary_base/scripts/update-versions.sh" --base . $old_version $new_version git commit -a -m "Update versions for $new_version" -git push origin $branch +git push origin $minor_branch gh pr create \ --fill \ --base $minor_base \ --repo typelevel/cats-effect \ - --head typelevel:$branch \ + --head typelevel:$minor_branch \ --label ':robot:' +git checkout -b $major_branch +git push origin $major_branch + gh pr create \ --title "Merge changes from $new_version into $major_base" \ --body '' \ --base $major_base \ --repo typelevel/cats-effect \ - --head typelevel:$branch \ + --head typelevel:$major_branch \ --label ':robot:' From 1c9d4fcacc4b8eed97b4da3943b869e1107d4d16 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Thu, 29 Jun 2023 01:51:42 +0000 Subject: [PATCH 04/46] Optimize `liveTraces()` on JS --- .../unsafe/BatchingMacrotaskExecutor.scala | 18 ++++------ .../cats/effect/unsafe/JSArrayQueue.scala | 21 ++++++++++++ .../cats/effect/unsafe/JSArrayQueueSpec.scala | 33 ++++++++++++++++++- 3 files changed, 59 insertions(+), 13 deletions(-) diff --git a/core/js/src/main/scala/cats/effect/unsafe/BatchingMacrotaskExecutor.scala b/core/js/src/main/scala/cats/effect/unsafe/BatchingMacrotaskExecutor.scala index d550d2c389..f56123cfec 100644 --- a/core/js/src/main/scala/cats/effect/unsafe/BatchingMacrotaskExecutor.scala +++ b/core/js/src/main/scala/cats/effect/unsafe/BatchingMacrotaskExecutor.scala @@ -62,17 +62,11 @@ private[effect] final class BatchingMacrotaskExecutor( var i = 0 while (i < batchSize && !fibers.isEmpty()) { val fiber = fibers.take() - - if (LinkingInfo.developmentMode) - if (fiberBag ne null) - fiberBag -= fiber - try fiber.run() catch { case t if NonFatal(t) => reportFailure(t) case t: Throwable => IOFiber.onFatalFailure(t) } - i += 1 } @@ -99,10 +93,6 @@ private[effect] final class BatchingMacrotaskExecutor( * batch. */ def schedule(fiber: IOFiber[_]): Unit = { - if (LinkingInfo.developmentMode) - if (fiberBag ne null) - fiberBag += fiber - fibers.offer(fiber) if (needsReschedule) { @@ -116,8 +106,12 @@ private[effect] final class BatchingMacrotaskExecutor( def reportFailure(t: Throwable): Unit = reportFailure0(t) - def liveTraces(): Map[IOFiber[_], Trace] = - fiberBag.iterator.filterNot(_.isDone).map(f => f -> f.captureTrace()).toMap + def liveTraces(): Map[IOFiber[_], Trace] = { + val traces = Map.newBuilder[IOFiber[_], Trace] + fibers.foreach(f => if (!f.isDone) traces += f -> f.captureTrace()) + fiberBag.foreach(f => if (!f.isDone) traces += f -> f.captureTrace()) + traces.result() + } @inline private[this] def monitor(runnable: Runnable): Runnable = if (LinkingInfo.developmentMode) diff --git a/core/js/src/main/scala/cats/effect/unsafe/JSArrayQueue.scala b/core/js/src/main/scala/cats/effect/unsafe/JSArrayQueue.scala index 65505f1b61..ead6c6979e 100644 --- a/core/js/src/main/scala/cats/effect/unsafe/JSArrayQueue.scala +++ b/core/js/src/main/scala/cats/effect/unsafe/JSArrayQueue.scala @@ -69,4 +69,25 @@ private final class JSArrayQueue[A] { } } + @inline def foreach(f: A => Unit): Unit = + if (empty) () + else if (startIndex < endIndex) { // consecutive in middle of buffer + var i = startIndex + while (i < endIndex) { + f(buffer(i)) + i += 1 + } + } else { // split across tail and init of buffer + var i = startIndex + while (i < buffer.length) { + f(buffer(i)) + i += 1 + } + i = 0 + while (i < endIndex) { + f(buffer(i)) + i += 1 + } + } + } diff --git a/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala b/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala index e52a54a624..d28c9f20e9 100644 --- a/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala +++ b/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala @@ -20,7 +20,7 @@ package unsafe import org.scalacheck.Prop.forAll import org.specs2.ScalaCheck -import scala.collection.mutable.ListBuffer +import scala.collection.mutable.{ArrayDeque, ListBuffer} class JSArrayQueueSpec extends BaseSpec with ScalaCheck { @@ -41,6 +41,37 @@ class JSArrayQueueSpec extends BaseSpec with ScalaCheck { taken.toList must beEqualTo(stuff.flatten) } } + + "iterate over contents in foreach" in { + forAll { (stuff: List[Option[Int]]) => + val queue = new JSArrayQueue[Int] + val shadow = new ArrayDeque[Int] + + def checkContents() = { + val builder = List.newBuilder[Int] + queue.foreach(builder += _) + builder.result() must beEqualTo(shadow.toList) + } + + checkContents() + + stuff.foreach { + case Some(i) => + queue.offer(i) + shadow.append(i) + checkContents() + case None => + if (!shadow.isEmpty) { + val got = queue.take() + val expected = shadow.removeHead() + got must beEqualTo(expected) + checkContents() + } + } + + true must beTrue + } + } } } From 1f3ca6e07bd6b53e7d6780d4d2d192784206e268 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Thu, 29 Jun 2023 02:08:03 +0000 Subject: [PATCH 05/46] Fix 2.12 compile --- .../test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala b/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala index d28c9f20e9..ade4d40d03 100644 --- a/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala +++ b/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala @@ -20,7 +20,7 @@ package unsafe import org.scalacheck.Prop.forAll import org.specs2.ScalaCheck -import scala.collection.mutable.{ArrayDeque, ListBuffer} +import scala.collection.mutable.{ListBuffer, Queue} class JSArrayQueueSpec extends BaseSpec with ScalaCheck { @@ -45,7 +45,7 @@ class JSArrayQueueSpec extends BaseSpec with ScalaCheck { "iterate over contents in foreach" in { forAll { (stuff: List[Option[Int]]) => val queue = new JSArrayQueue[Int] - val shadow = new ArrayDeque[Int] + val shadow = new Queue[Int] def checkContents() = { val builder = List.newBuilder[Int] @@ -58,12 +58,12 @@ class JSArrayQueueSpec extends BaseSpec with ScalaCheck { stuff.foreach { case Some(i) => queue.offer(i) - shadow.append(i) + shadow.enqueue(i) checkContents() case None => if (!shadow.isEmpty) { val got = queue.take() - val expected = shadow.removeHead() + val expected = shadow.dequeue() got must beEqualTo(expected) checkContents() } From 907b121665c7de6dd7bfd019693c7188bab12109 Mon Sep 17 00:00:00 2001 From: Daniel Urban Date: Sat, 1 Jul 2023 13:23:23 +0200 Subject: [PATCH 06/46] Remove unused code --- .../unsafe/FiberAwareExecutionContext.scala | 45 ------------------- 1 file changed, 45 deletions(-) delete mode 100644 core/js/src/main/scala/cats/effect/unsafe/FiberAwareExecutionContext.scala diff --git a/core/js/src/main/scala/cats/effect/unsafe/FiberAwareExecutionContext.scala b/core/js/src/main/scala/cats/effect/unsafe/FiberAwareExecutionContext.scala deleted file mode 100644 index 3718b8d4c7..0000000000 --- a/core/js/src/main/scala/cats/effect/unsafe/FiberAwareExecutionContext.scala +++ /dev/null @@ -1,45 +0,0 @@ -/* - * Copyright 2020-2023 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 scala.collection.mutable -import scala.concurrent.ExecutionContext - -private final class FiberAwareExecutionContext(ec: ExecutionContext) extends ExecutionContext { - - def liveTraces(): Map[IOFiber[_], Trace] = - fiberBag.iterator.filterNot(_.isDone).map(f => f -> f.captureTrace()).toMap - - private[this] val fiberBag = mutable.Set.empty[IOFiber[_]] - - def execute(runnable: Runnable): Unit = runnable match { - case r: IOFiber[_] => - fiberBag += r - ec execute { () => - // We have to remove r _before_ running it, b/c it may be re-enqueued while running - // B/c JS is single-threaded, nobody can observe the bag while it is running anyway - fiberBag -= r - r.run() - } - - case r => r.run() - } - - def reportFailure(cause: Throwable): Unit = ec.reportFailure(cause) - -} From 5a3a942a66a47136deb7a649b8b366ddfb40fa0b Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Sat, 1 Jul 2023 07:57:17 -0700 Subject: [PATCH 07/46] ok ok --- .../js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala b/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala index ade4d40d03..df0b7edd52 100644 --- a/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala +++ b/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala @@ -69,7 +69,7 @@ class JSArrayQueueSpec extends BaseSpec with ScalaCheck { } } - true must beTrue + ok } } } From 809d837c738a5ad93b4d753fc0b7bf807303e6eb Mon Sep 17 00:00:00 2001 From: Daniel Urban Date: Fri, 7 Jul 2023 16:27:39 +0200 Subject: [PATCH 08/46] Fix typo --- kernel/shared/src/main/scala/cats/effect/kernel/Async.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/kernel/shared/src/main/scala/cats/effect/kernel/Async.scala b/kernel/shared/src/main/scala/cats/effect/kernel/Async.scala index b047f591e0..06b534f929 100644 --- a/kernel/shared/src/main/scala/cats/effect/kernel/Async.scala +++ b/kernel/shared/src/main/scala/cats/effect/kernel/Async.scala @@ -46,7 +46,7 @@ import java.util.concurrent.atomic.AtomicReference * * {{{async(k)}}} is semantically blocked until the callback is invoked. * - * `async_` is somewhat contrained however. We can't perform any `F[_]` effects in the process + * `async_` is somewhat constrained however. We can't perform any `F[_]` effects in the process * of registering the callback and we also can't register a finalizer to eg cancel the * asynchronous task in the event that the fiber running `async_` is canceled. * From 67b32756083733ab3b78504e1f6038cfc86d40e6 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bj=C3=B8rn=20Madsen?= Date: Mon, 10 Jul 2023 10:24:30 +0200 Subject: [PATCH 09/46] Increase blocking thread expiration --- .../test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/tests/jvm/src/test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala b/tests/jvm/src/test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala index ebd19cb893..5817d10e34 100644 --- a/tests/jvm/src/test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala +++ b/tests/jvm/src/test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala @@ -20,6 +20,8 @@ import cats.effect.{BaseSpec, IO} import cats.effect.testkit.TestInstances import cats.syntax.all._ +import scala.concurrent.duration._ + class WorkerThreadNameSpec extends BaseSpec with TestInstances { override def runtime(): IORuntime = { @@ -33,7 +35,8 @@ class WorkerThreadNameSpec extends BaseSpec with TestInstances { IORuntime.createWorkStealingComputeThreadPool( threads = 1, threadPrefix = s"io-compute-${getClass.getName}", - blockerThreadPrefix = s"io-blocker-${getClass.getName}") + blockerThreadPrefix = s"io-blocker-${getClass.getName}", + runtimeBlockingExpiration = 10.minutes) IORuntime( compute, From c2e6a496f9ac3413ca31e2822c433c5fcc78f7fe Mon Sep 17 00:00:00 2001 From: ag91 Date: Sun, 9 Jul 2023 23:22:43 +0100 Subject: [PATCH 10/46] update Deferred.complete type Fixes https://github.com/typelevel/cats-effect/issues/3720 --- docs/thread-model.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/thread-model.md b/docs/thread-model.md index 6335595502..765a946851 100644 --- a/docs/thread-model.md +++ b/docs/thread-model.md @@ -121,7 +121,7 @@ purely functional promise that can only be completed once. trait Deferred[F[_], A] { def get: F[A] - def complete(a: A): F[Unit] + def complete(a: A): F[Boolean] } ``` From c5fbd51fb69880891e34ac18d8cdbf012f0e072a Mon Sep 17 00:00:00 2001 From: Sam Pillsworth Date: Thu, 13 Jul 2023 20:20:43 -0400 Subject: [PATCH 11/46] remove test sleep I believe that this sleep is no longer necessary now that the `alive` resource, and it's associated race condition, has been removed. --- .../shared/src/test/scala/cats/effect/std/DispatcherSpec.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/shared/src/test/scala/cats/effect/std/DispatcherSpec.scala b/tests/shared/src/test/scala/cats/effect/std/DispatcherSpec.scala index cf95a3ead9..ea27049c25 100644 --- a/tests/shared/src/test/scala/cats/effect/std/DispatcherSpec.scala +++ b/tests/shared/src/test/scala/cats/effect/std/DispatcherSpec.scala @@ -360,7 +360,7 @@ class DispatcherSpec extends BaseSpec with DetectPlatform { implicit ticker => val test = dispatcher.allocated.flatMap { case (runner, release) => - IO(runner.unsafeRunAndForget(IO.sleep(50.millis) *> release)) *> + IO(runner.unsafeRunAndForget(release)) *> IO.sleep(100.millis) *> IO(runner.unsafeRunAndForget(IO(ko)) must throwAn[IllegalStateException]) } From a93b88a9bb17dd91f0a674250f84987af0065456 Mon Sep 17 00:00:00 2001 From: Sam Pillsworth Date: Fri, 4 Aug 2023 08:12:58 -0400 Subject: [PATCH 12/46] update to graalvm@17 --- .github/workflows/ci.yml | 78 ++++++++++++++++++++-------------------- build.sbt | 2 +- 2 files changed, 40 insertions(+), 40 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index c5297949f6..49fcc0ae04 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -29,7 +29,7 @@ jobs: matrix: os: [ubuntu-latest, windows-latest, macos-latest] scala: [3.2.2, 2.12.17, 2.13.10] - java: [temurin@8, temurin@11, temurin@17, graalvm@11] + java: [temurin@8, temurin@11, temurin@17, graalvm@17] ci: [ciJVM, ciNative, ciJS, ciFirefox, ciChrome] exclude: - scala: 3.2.2 @@ -39,7 +39,7 @@ jobs: - scala: 2.12.17 java: temurin@17 - scala: 2.12.17 - java: graalvm@11 + java: graalvm@17 - os: windows-latest scala: 3.2.2 - os: macos-latest @@ -61,7 +61,7 @@ jobs: - ci: ciJS java: temurin@17 - ci: ciJS - java: graalvm@11 + java: graalvm@17 - os: windows-latest ci: ciJS - os: macos-latest @@ -71,7 +71,7 @@ jobs: - ci: ciFirefox java: temurin@17 - ci: ciFirefox - java: graalvm@11 + java: graalvm@17 - os: windows-latest ci: ciFirefox - os: macos-latest @@ -81,7 +81,7 @@ jobs: - ci: ciChrome java: temurin@17 - ci: ciChrome - java: graalvm@11 + java: graalvm@17 - os: windows-latest ci: ciChrome - os: macos-latest @@ -91,7 +91,7 @@ jobs: - ci: ciNative java: temurin@17 - ci: ciNative - java: graalvm@11 + java: graalvm@17 - os: windows-latest ci: ciNative - os: macos-latest @@ -101,7 +101,7 @@ jobs: ci: ciNative scala: 3.2.2 - os: windows-latest - java: graalvm@11 + java: graalvm@17 runs-on: ${{ matrix.os }} timeout-minutes: 60 steps: @@ -184,26 +184,26 @@ jobs: shell: bash run: sbt '++ ${{ matrix.scala }}' reload +update - - name: Download Java (graalvm@11) - id: download-java-graalvm-11 - if: matrix.java == 'graalvm@11' + - name: Download Java (graalvm@17) + id: download-java-graalvm-17 + if: matrix.java == 'graalvm@17' uses: typelevel/download-java@v2 with: distribution: graalvm - java-version: 11 + java-version: 17 - - name: Setup Java (graalvm@11) - id: setup-java-graalvm-11 - if: matrix.java == 'graalvm@11' + - name: Setup Java (graalvm@17) + id: setup-java-graalvm-17 + if: matrix.java == 'graalvm@17' uses: actions/setup-java@v3 with: distribution: jdkfile - java-version: 11 - jdkFile: ${{ steps.download-java-graalvm-11.outputs.jdkFile }} + java-version: 17 + jdkFile: ${{ steps.download-java-graalvm-17.outputs.jdkFile }} cache: sbt - name: sbt update - if: matrix.java == 'graalvm@11' && steps.setup-java-graalvm-11.outputs.cache-hit == 'false' + if: matrix.java == 'graalvm@17' && steps.setup-java-graalvm-17.outputs.cache-hit == 'false' shell: bash run: sbt '++ ${{ matrix.scala }}' reload +update @@ -219,7 +219,7 @@ jobs: run: npm install - name: Install GraalVM Native Image - if: matrix.java == 'graalvm@11' + if: matrix.java == 'graalvm@17' shell: bash run: gu install native-image @@ -268,7 +268,7 @@ jobs: run: example/test-js.sh ${{ matrix.scala }} - name: Test GraalVM Native Image - if: matrix.scala == '2.13.10' && matrix.java == 'graalvm@11' && matrix.os == 'ubuntu-latest' + if: matrix.scala == '2.13.10' && matrix.java == 'graalvm@17' && matrix.os == 'ubuntu-latest' shell: bash run: sbt '++ ${{ matrix.scala }}' graalVMExample/nativeImage graalVMExample/nativeImageRun @@ -387,26 +387,26 @@ jobs: if: matrix.java == 'temurin@17' && steps.setup-java-temurin-17.outputs.cache-hit == 'false' run: sbt '++ ${{ matrix.scala }}' reload +update - - name: Download Java (graalvm@11) - id: download-java-graalvm-11 - if: matrix.java == 'graalvm@11' + - name: Download Java (graalvm@17) + id: download-java-graalvm-17 + if: matrix.java == 'graalvm@17' uses: typelevel/download-java@v2 with: distribution: graalvm - java-version: 11 + java-version: 17 - - name: Setup Java (graalvm@11) - id: setup-java-graalvm-11 - if: matrix.java == 'graalvm@11' + - name: Setup Java (graalvm@17) + id: setup-java-graalvm-17 + if: matrix.java == 'graalvm@17' uses: actions/setup-java@v3 with: distribution: jdkfile - java-version: 11 - jdkFile: ${{ steps.download-java-graalvm-11.outputs.jdkFile }} + java-version: 17 + jdkFile: ${{ steps.download-java-graalvm-17.outputs.jdkFile }} cache: sbt - name: sbt update - if: matrix.java == 'graalvm@11' && steps.setup-java-graalvm-11.outputs.cache-hit == 'false' + if: matrix.java == 'graalvm@17' && steps.setup-java-graalvm-17.outputs.cache-hit == 'false' run: sbt '++ ${{ matrix.scala }}' reload +update - name: Download target directories (3.2.2, ciJVM) @@ -631,26 +631,26 @@ jobs: if: matrix.java == 'temurin@17' && steps.setup-java-temurin-17.outputs.cache-hit == 'false' run: sbt '++ ${{ matrix.scala }}' reload +update - - name: Download Java (graalvm@11) - id: download-java-graalvm-11 - if: matrix.java == 'graalvm@11' + - name: Download Java (graalvm@17) + id: download-java-graalvm-17 + if: matrix.java == 'graalvm@17' uses: typelevel/download-java@v2 with: distribution: graalvm - java-version: 11 + java-version: 17 - - name: Setup Java (graalvm@11) - id: setup-java-graalvm-11 - if: matrix.java == 'graalvm@11' + - name: Setup Java (graalvm@17) + id: setup-java-graalvm-17 + if: matrix.java == 'graalvm@17' uses: actions/setup-java@v3 with: distribution: jdkfile - java-version: 11 - jdkFile: ${{ steps.download-java-graalvm-11.outputs.jdkFile }} + java-version: 17 + jdkFile: ${{ steps.download-java-graalvm-17.outputs.jdkFile }} cache: sbt - name: sbt update - if: matrix.java == 'graalvm@11' && steps.setup-java-graalvm-11.outputs.cache-hit == 'false' + if: matrix.java == 'graalvm@17' && steps.setup-java-graalvm-17.outputs.cache-hit == 'false' run: sbt '++ ${{ matrix.scala }}' reload +update - name: Submit Dependencies diff --git a/build.sbt b/build.sbt index fa5d1d2dd0..7cbd88e36d 100644 --- a/build.sbt +++ b/build.sbt @@ -137,7 +137,7 @@ val LTSJava = JavaSpec.temurin("11") val LatestJava = JavaSpec.temurin("17") val ScalaJSJava = OldGuardJava val ScalaNativeJava = OldGuardJava -val GraalVM = JavaSpec.graalvm("11") +val GraalVM = JavaSpec.graalvm("17") ThisBuild / githubWorkflowJavaVersions := Seq(OldGuardJava, LTSJava, LatestJava, GraalVM) ThisBuild / githubWorkflowOSes := Seq(PrimaryOS, Windows, MacOS) From d42975a4ba8d7c0422bd9a8b6cc5a20ad342f7fc Mon Sep 17 00:00:00 2001 From: Gavin Bisesi Date: Fri, 4 Aug 2023 16:44:08 -0400 Subject: [PATCH 13/46] Print warning when we detect a non-main-thread main This should help people not run facefirst into https://github.com/sbt/sbt/issues/6242 It's a frequent question for new users --- .../src/main/scala/cats/effect/IOApp.scala | 24 +++++++++++++++++++ docs/core/io-runtime-config.md | 3 ++- 2 files changed, 26 insertions(+), 1 deletion(-) diff --git a/core/jvm/src/main/scala/cats/effect/IOApp.scala b/core/jvm/src/main/scala/cats/effect/IOApp.scala index 6f1a0857af..8d98bf7421 100644 --- a/core/jvm/src/main/scala/cats/effect/IOApp.scala +++ b/core/jvm/src/main/scala/cats/effect/IOApp.scala @@ -314,6 +314,29 @@ trait IOApp { protected def onCpuStarvationWarn(metrics: CpuStarvationWarningMetrics): IO[Unit] = CpuStarvationCheck.logWarning(metrics) + /** + * Defines what to do when IOApp detects that `main` is being invoked on a `Thread` which + * isn't the main process thread. This condition can happen when we are running inside of an + * `sbt run` with `fork := false` + */ + private def onNonMainThreadDetected(): Unit = { + val shouldPrint = + Option(System.getProperty("cats.effect.warnOnNonMainThreadDetected")) + .map(_.equalsIgnoreCase("true")) + .getOrElse(true) + if (shouldPrint) + System + .err + .println( + """|Warning: IOApp `main` is running on a thread other than the main thread. + |This may prevent correct resource cleanup after `main` completes. + |This condition could be caused by executing `run` in an interactive sbt session with `fork := false`. + |Set `Compile / run / fork := true` in this project to resolve this. + |""".stripMargin + ) + else () + } + /** * The entry point for your application. Will be called by the runtime when the process is * started. If the underlying runtime supports it, any arguments passed to the process will be @@ -333,6 +356,7 @@ trait IOApp { final def main(args: Array[String]): Unit = { // checked in openjdk 8-17; this attempts to detect when we're running under artificial environments, like sbt val isForked = Thread.currentThread().getId() == 1 + if (!isForked) onNonMainThreadDetected() val installed = if (runtime == null) { import unsafe.IORuntime diff --git a/docs/core/io-runtime-config.md b/docs/core/io-runtime-config.md index 0a48199b48..39828ad429 100644 --- a/docs/core/io-runtime-config.md +++ b/docs/core/io-runtime-config.md @@ -28,7 +28,8 @@ This can be done for example with the [EnvironmentPlugin for Webpack](https://we | `cats.effect.detectBlockedThreads`
N/A | `Boolean` (`false`) | Whether or not we should detect blocked threads. | | `cats.effect.logNonDaemonThreadsOnExit`
N/A | `Boolean` (`true`) | Whether or not we should check for non-daemon threads on JVM exit. | | `cats.effect.logNonDaemonThreads.sleepIntervalMillis`
N/A | `Long` (`10000L`) | Time to sleep between checking for presence of non-daemon threads. | -| `cats.effect.cancelation.check.threshold `
`CATS_EFFECT_CANCELATION_CHECK_THRESHOLD` | `Int` (`512`) | Configure how often cancellation is checked. By default, every 512 iterations of the run loop. | +| `cats.effect.warnOnNonMainThreadDetected`
N/A | `Boolean` (`true`) | Print a warning message when IOApp `main` runs on a non-main thread | +| `cats.effect.cancelation.check.threshold`
`CATS_EFFECT_CANCELATION_CHECK_THRESHOLD` | `Int` (`512`) | Configure how often cancellation is checked. By default, every 512 iterations of the run loop. | | `cats.effect.auto.yield.threshold.multiplier`
`CATS_EFFECT_AUTO_YIELD_THRESHOLD_MULTIPLIER` | `Int` (`2`) | `autoYieldThreshold = autoYieldThresholdMultiplier x cancelationCheckThreshold`. See [thread model](../thread-model.md). | | `cats.effect.tracing.exceptions.enhanced`
`CATS_EFFECT_TRACING_EXCEPTIONS_ENHANCED` | `Boolean` (`true`) | Augment the stack traces of caught exceptions to include frames from the asynchronous stack traces. See [tracing](../tracing.md). | | `cats.effect.tracing.buffer.size`
`CATS_EFFECT_TRACING_BUFFER_SIZE` | `Int` (`16`) | Number of stack frames retained in the tracing buffer. Will be rounded up to next power of two. | From f521fb24294297f7573be626965d86f443c4434b Mon Sep 17 00:00:00 2001 From: Gavin Bisesi Date: Sat, 5 Aug 2023 06:37:29 -0400 Subject: [PATCH 14/46] Add note on how to silence main thread warning Co-authored-by: Arman Bilge --- core/jvm/src/main/scala/cats/effect/IOApp.scala | 3 +++ 1 file changed, 3 insertions(+) diff --git a/core/jvm/src/main/scala/cats/effect/IOApp.scala b/core/jvm/src/main/scala/cats/effect/IOApp.scala index 8d98bf7421..dcd01d6804 100644 --- a/core/jvm/src/main/scala/cats/effect/IOApp.scala +++ b/core/jvm/src/main/scala/cats/effect/IOApp.scala @@ -332,6 +332,9 @@ trait IOApp { |This may prevent correct resource cleanup after `main` completes. |This condition could be caused by executing `run` in an interactive sbt session with `fork := false`. |Set `Compile / run / fork := true` in this project to resolve this. + | + |To silence this warning set the system property: + |`-Dcats.effect.warnOnNonMainThreadDetected=false`. |""".stripMargin ) else () From 55044195c8d6bdfdadcd7411aa6ebe9e97bcf045 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Sun, 6 Aug 2023 00:29:53 +0000 Subject: [PATCH 15/46] Elide thunk allocation when using `sleepInternal` --- .../effect/unsafe/WorkStealingThreadPool.scala | 2 +- .../scala/cats/effect/unsafe/TimerSkipList.scala | 7 +++++-- .../effect/unsafe/WorkStealingThreadPool.scala | 6 ++++-- .../scala/cats/effect/unsafe/WorkerThread.scala | 4 +++- .../src/main/scala/cats/effect/IOFiber.scala | 16 ++++++++++------ 5 files changed, 23 insertions(+), 12 deletions(-) diff --git a/core/js-native/src/main/scala/cats/effect/unsafe/WorkStealingThreadPool.scala b/core/js-native/src/main/scala/cats/effect/unsafe/WorkStealingThreadPool.scala index e8118e1cee..f47fc7889a 100644 --- a/core/js-native/src/main/scala/cats/effect/unsafe/WorkStealingThreadPool.scala +++ b/core/js-native/src/main/scala/cats/effect/unsafe/WorkStealingThreadPool.scala @@ -30,7 +30,7 @@ private[effect] sealed abstract class WorkStealingThreadPool private () private[effect] def reschedule(runnable: Runnable): Unit private[effect] def sleepInternal( delay: FiniteDuration, - callback: Right[Nothing, Unit] => Unit): Runnable + callback: Right[Nothing, Unit] => Unit): Function0[Unit] with Runnable private[effect] def sleep( delay: FiniteDuration, task: Runnable, diff --git a/core/jvm/src/main/scala/cats/effect/unsafe/TimerSkipList.scala b/core/jvm/src/main/scala/cats/effect/unsafe/TimerSkipList.scala index 5cd5cd884a..e02a2422b9 100644 --- a/core/jvm/src/main/scala/cats/effect/unsafe/TimerSkipList.scala +++ b/core/jvm/src/main/scala/cats/effect/unsafe/TimerSkipList.scala @@ -68,18 +68,21 @@ private final class TimerSkipList() extends AtomicLong(MARKER + 1L) { sequenceNu cb: Callback, next: Node ) extends TimerSkipListNodeBase[Callback, Node](cb, next) + with Function0[Unit] with Runnable { /** * Cancels the timer */ - final override def run(): Unit = { + final def apply(): Unit = { // TODO: We could null the callback here directly, // TODO: and the do the lookup after (for unlinking). TimerSkipList.this.doRemove(triggerTime, sequenceNum) () } + final def run() = apply() + private[TimerSkipList] final def isMarker: Boolean = { // note: a marker node also has `triggerTime == MARKER`, // but that's also a valid trigger time, so we need @@ -158,7 +161,7 @@ private final class TimerSkipList() extends AtomicLong(MARKER + 1L) { sequenceNu delay: Long, callback: Right[Nothing, Unit] => Unit, tlr: ThreadLocalRandom - ): Runnable = { + ): Function0[Unit] with Runnable = { require(delay >= 0L) // we have to check for overflow: val triggerTime = computeTriggerTime(now = now, delay = delay) diff --git a/core/jvm/src/main/scala/cats/effect/unsafe/WorkStealingThreadPool.scala b/core/jvm/src/main/scala/cats/effect/unsafe/WorkStealingThreadPool.scala index 909746243d..cd740b0b41 100644 --- a/core/jvm/src/main/scala/cats/effect/unsafe/WorkStealingThreadPool.scala +++ b/core/jvm/src/main/scala/cats/effect/unsafe/WorkStealingThreadPool.scala @@ -621,7 +621,9 @@ private[effect] final class WorkStealingThreadPool( /** * Tries to call the current worker's `sleep`, but falls back to `sleepExternal` if needed. */ - def sleepInternal(delay: FiniteDuration, callback: Right[Nothing, Unit] => Unit): Runnable = { + def sleepInternal( + delay: FiniteDuration, + callback: Right[Nothing, Unit] => Unit): Function0[Unit] with Runnable = { val thread = Thread.currentThread() if (thread.isInstanceOf[WorkerThread]) { val worker = thread.asInstanceOf[WorkerThread] @@ -642,7 +644,7 @@ private[effect] final class WorkStealingThreadPool( */ private[this] final def sleepExternal( delay: FiniteDuration, - callback: Right[Nothing, Unit] => Unit): Runnable = { + callback: Right[Nothing, Unit] => Unit): Function0[Unit] with Runnable = { val random = ThreadLocalRandom.current() val idx = random.nextInt(threadCount) val tsl = sleepers(idx) diff --git a/core/jvm/src/main/scala/cats/effect/unsafe/WorkerThread.scala b/core/jvm/src/main/scala/cats/effect/unsafe/WorkerThread.scala index 31b36e408e..849e71a2d4 100644 --- a/core/jvm/src/main/scala/cats/effect/unsafe/WorkerThread.scala +++ b/core/jvm/src/main/scala/cats/effect/unsafe/WorkerThread.scala @@ -152,7 +152,9 @@ private final class WorkerThread( } } - def sleep(delay: FiniteDuration, callback: Right[Nothing, Unit] => Unit): Runnable = { + def sleep( + delay: FiniteDuration, + callback: Right[Nothing, Unit] => Unit): Function0[Unit] with Runnable = { // take the opportunity to update the current time, just in case other timers can benefit val _now = System.nanoTime() now = _now diff --git a/core/shared/src/main/scala/cats/effect/IOFiber.scala b/core/shared/src/main/scala/cats/effect/IOFiber.scala index ef4957bbaf..f6214b50f8 100644 --- a/core/shared/src/main/scala/cats/effect/IOFiber.scala +++ b/core/shared/src/main/scala/cats/effect/IOFiber.scala @@ -925,13 +925,17 @@ private final class IOFiber[A]( IO { val scheduler = runtime.scheduler - val cancel = - if (scheduler.isInstanceOf[WorkStealingThreadPool]) - scheduler.asInstanceOf[WorkStealingThreadPool].sleepInternal(delay, cb) - else - scheduler.sleep(delay, () => cb(RightUnit)) + val cancelIO = + if (scheduler.isInstanceOf[WorkStealingThreadPool]) { + val cancel = + scheduler.asInstanceOf[WorkStealingThreadPool].sleepInternal(delay, cb) + IO.Delay(cancel, null) + } else { + val cancel = scheduler.sleep(delay, () => cb(RightUnit)) + IO(cancel.run()) + } - Some(IO(cancel.run())) + Some(cancelIO) } } else IO.cede From 8dbd2bcd0befdb6073dea85c3d77541e62b89f14 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Sun, 6 Aug 2023 00:53:00 +0000 Subject: [PATCH 16/46] Add MiMa filters --- build.sbt | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/build.sbt b/build.sbt index 7cbd88e36d..82a522f7a7 100644 --- a/build.sbt +++ b/build.sbt @@ -643,7 +643,12 @@ lazy val core = crossProject(JSPlatform, JVMPlatform, NativePlatform) "cats.effect.IOFiberConstants.ExecuteRunnableR"), ProblemFilters.exclude[ReversedMissingMethodProblem]("cats.effect.IOLocal.scope"), ProblemFilters.exclude[DirectMissingMethodProblem]( - "cats.effect.IOFiberConstants.ContStateResult") + "cats.effect.IOFiberConstants.ContStateResult"), + // #3775, changes to internal timers APIs + ProblemFilters.exclude[IncompatibleResultTypeProblem]( + "cats.effect.unsafe.TimerSkipList.insert"), + ProblemFilters.exclude[IncompatibleResultTypeProblem]( + "cats.effect.unsafe.WorkerThread.sleep") ) ++ { if (tlIsScala3.value) { // Scala 3 specific exclusions From 499300a6259bb092f55122bb8e0d169367f6226c Mon Sep 17 00:00:00 2001 From: Victor Grigoriu Date: Mon, 24 Jul 2023 17:12:21 +0300 Subject: [PATCH 17/46] fix typo --- docs/tutorial.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/tutorial.md b/docs/tutorial.md index 89d3ca7781..4f8b48cf99 100644 --- a/docs/tutorial.md +++ b/docs/tutorial.md @@ -719,7 +719,7 @@ Both producer and consumer will access the same shared state instance, which will be carried and safely modified by an instance of `Ref`. Consumer shall work as follows: 1. If `queue` is not empty, it will extract and return its head. The new state - will keep the tail of the queue, not change on `takers` will be needed. + will keep the tail of the queue, no change on `takers` will be needed. 2. If `queue` is empty it will use a new `Deferred` instance as a new `taker`, add it to the `takers` queue, and 'block' the caller by invoking `taker.get` From da5c71fd9f116e8f3a01441229a6d05acb9f2c2d Mon Sep 17 00:00:00 2001 From: Kamil Kloch Date: Thu, 3 Aug 2023 09:33:33 +0200 Subject: [PATCH 18/46] Update `reportFailure` scaladoc - add explanation of an unhandled error. h/t @armanbilge https://github.com/typelevel/cats-effect/issues/3768 --- .../src/main/scala/cats/effect/IOApp.scala | 24 +++++++++++++++---- 1 file changed, 19 insertions(+), 5 deletions(-) diff --git a/core/jvm/src/main/scala/cats/effect/IOApp.scala b/core/jvm/src/main/scala/cats/effect/IOApp.scala index 6f1a0857af..c63b6a3585 100644 --- a/core/jvm/src/main/scala/cats/effect/IOApp.scala +++ b/core/jvm/src/main/scala/cats/effect/IOApp.scala @@ -230,11 +230,25 @@ trait IOApp { ) /** - * Configures the action to perform when unhandled errors are caught by the runtime. By - * default, this simply delegates to [[cats.effect.std.Console!.printStackTrace]]. It is safe - * to perform any `IO` action within this handler; it will not block the progress of the - * runtime. With that said, some care should be taken to avoid raising unhandled errors as a - * result of handling unhandled errors, since that will result in the obvious chaos. + * Configures the action to perform when unhandled errors are caught by the runtime. An + * unhandled error is an error that is raised on a Fiber that nobody is joining. + * + * For example: + * + * {{{ + * import scala.concurrent.duration._ + * override def run: IO[Unit] = IO(throw new Exception("")).start *> IO.sleep(1.second) + * }}} + * + * In this case, the exception is raised on a Fiber with no listeners. So nobody would be + * notified about that error. Therefore it is unhandled, and it goes through the reportFailure + * mechanism. + * + * By default, `reportFailure` simply delegates to + * [[cats.effect.std.Console!.printStackTrace]]. It is safe to perform any `IO` action within + * this handler; it will not block the progress of the runtime. With that said, some care + * should be taken to avoid raising unhandled errors as a result of handling unhandled errors, + * since that will result in the obvious chaos. */ protected def reportFailure(err: Throwable): IO[Unit] = Console[IO].printStackTrace(err) From 7e4bcbb9e1d94c63dad8adf4643a6ecc649f0484 Mon Sep 17 00:00:00 2001 From: Kamil Kloch Date: Mon, 7 Aug 2023 11:21:29 +0200 Subject: [PATCH 19/46] Update scaladoc. --- core/jvm/src/main/scala/cats/effect/IOApp.scala | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/core/jvm/src/main/scala/cats/effect/IOApp.scala b/core/jvm/src/main/scala/cats/effect/IOApp.scala index c63b6a3585..d512280972 100644 --- a/core/jvm/src/main/scala/cats/effect/IOApp.scala +++ b/core/jvm/src/main/scala/cats/effect/IOApp.scala @@ -230,8 +230,8 @@ trait IOApp { ) /** - * Configures the action to perform when unhandled errors are caught by the runtime. An - * unhandled error is an error that is raised on a Fiber that nobody is joining. + * Configures the action to perform when unhandled errors are caught by the runtime. + * An unhandled error is an error that is raised (and not handled) on a Fiber that nobody is joining. * * For example: * @@ -240,9 +240,8 @@ trait IOApp { * override def run: IO[Unit] = IO(throw new Exception("")).start *> IO.sleep(1.second) * }}} * - * In this case, the exception is raised on a Fiber with no listeners. So nobody would be - * notified about that error. Therefore it is unhandled, and it goes through the reportFailure - * mechanism. + * In this case, the exception is raised on a Fiber with no listeners. Nobody would be notified + * about that error. Therefore it is unhandled, and it goes through the reportFailure mechanism. * * By default, `reportFailure` simply delegates to * [[cats.effect.std.Console!.printStackTrace]]. It is safe to perform any `IO` action within From 492fbd1c04aa50366569b257b196cb4002e88d80 Mon Sep 17 00:00:00 2001 From: Kamil Kloch Date: Mon, 7 Aug 2023 11:34:42 +0200 Subject: [PATCH 20/46] scalafmt. --- core/jvm/src/main/scala/cats/effect/IOApp.scala | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/core/jvm/src/main/scala/cats/effect/IOApp.scala b/core/jvm/src/main/scala/cats/effect/IOApp.scala index d512280972..0557102baa 100644 --- a/core/jvm/src/main/scala/cats/effect/IOApp.scala +++ b/core/jvm/src/main/scala/cats/effect/IOApp.scala @@ -230,8 +230,9 @@ trait IOApp { ) /** - * Configures the action to perform when unhandled errors are caught by the runtime. - * An unhandled error is an error that is raised (and not handled) on a Fiber that nobody is joining. + * Configures the action to perform when unhandled errors are caught by the runtime. An + * unhandled error is an error that is raised (and not handled) on a Fiber that nobody is + * joining. * * For example: * @@ -240,8 +241,9 @@ trait IOApp { * override def run: IO[Unit] = IO(throw new Exception("")).start *> IO.sleep(1.second) * }}} * - * In this case, the exception is raised on a Fiber with no listeners. Nobody would be notified - * about that error. Therefore it is unhandled, and it goes through the reportFailure mechanism. + * In this case, the exception is raised on a Fiber with no listeners. Nobody would be + * notified about that error. Therefore it is unhandled, and it goes through the reportFailure + * mechanism. * * By default, `reportFailure` simply delegates to * [[cats.effect.std.Console!.printStackTrace]]. It is safe to perform any `IO` action within From f0f23bea5e1ba44b8ce484d1a91aa6337a220bc6 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Tue, 22 Aug 2023 04:02:03 +0000 Subject: [PATCH 21/46] Remove `Thunk.asFunction0` utility --- build.sbt | 4 ++- .../cats/effect/IOCompanionPlatform.scala | 8 ++--- .../src/main/scala/cats/effect/IO.scala | 2 +- .../src/main/scala/cats/effect/Thunk.scala | 29 ------------------- .../test/scala/cats/effect/ThunkSpec.scala | 4 +-- 5 files changed, 10 insertions(+), 37 deletions(-) delete mode 100644 core/shared/src/main/scala/cats/effect/Thunk.scala diff --git a/build.sbt b/build.sbt index 82a522f7a7..dec1f87f15 100644 --- a/build.sbt +++ b/build.sbt @@ -648,7 +648,9 @@ lazy val core = crossProject(JSPlatform, JVMPlatform, NativePlatform) ProblemFilters.exclude[IncompatibleResultTypeProblem]( "cats.effect.unsafe.TimerSkipList.insert"), ProblemFilters.exclude[IncompatibleResultTypeProblem]( - "cats.effect.unsafe.WorkerThread.sleep") + "cats.effect.unsafe.WorkerThread.sleep"), + // #3787, internal utility that was no longer needed + ProblemFilters.exclude[MissingClassProblem]("cats.effect.Thunk") ) ++ { if (tlIsScala3.value) { // Scala 3 specific exclusions diff --git a/core/jvm/src/main/scala/cats/effect/IOCompanionPlatform.scala b/core/jvm/src/main/scala/cats/effect/IOCompanionPlatform.scala index cf3a5303ac..ad9d75c8b7 100644 --- a/core/jvm/src/main/scala/cats/effect/IOCompanionPlatform.scala +++ b/core/jvm/src/main/scala/cats/effect/IOCompanionPlatform.scala @@ -47,14 +47,14 @@ private[effect] abstract class IOCompanionPlatform { this: IO.type => * Implements [[cats.effect.kernel.Sync.blocking]]. */ def blocking[A](thunk: => A): IO[A] = { - val fn = Thunk.asFunction0(thunk) + val fn = () => thunk Blocking(TypeBlocking, fn, Tracing.calculateTracingEvent(fn.getClass)) } // this cannot be marked private[effect] because of static forwarders in Java @deprecated("use interruptible / interruptibleMany instead", "3.3.0") def interruptible[A](many: Boolean, thunk: => A): IO[A] = { - val fn = Thunk.asFunction0(thunk) + val fn = () => thunk Blocking( if (many) TypeInterruptibleMany else TypeInterruptibleOnce, fn, @@ -80,7 +80,7 @@ private[effect] abstract class IOCompanionPlatform { this: IO.type => * Implements [[cats.effect.kernel.Sync.interruptible[A](thunk:=>A):*]] */ def interruptible[A](thunk: => A): IO[A] = { - val fn = Thunk.asFunction0(thunk) + val fn = () => thunk Blocking(TypeInterruptibleOnce, fn, Tracing.calculateTracingEvent(fn.getClass)) } @@ -104,7 +104,7 @@ private[effect] abstract class IOCompanionPlatform { this: IO.type => * Implements [[cats.effect.kernel.Sync!.interruptibleMany]] */ def interruptibleMany[A](thunk: => A): IO[A] = { - val fn = Thunk.asFunction0(thunk) + val fn = () => thunk Blocking(TypeInterruptibleMany, fn, Tracing.calculateTracingEvent(fn.getClass)) } diff --git a/core/shared/src/main/scala/cats/effect/IO.scala b/core/shared/src/main/scala/cats/effect/IO.scala index 372630f7bb..ea520aa21f 100644 --- a/core/shared/src/main/scala/cats/effect/IO.scala +++ b/core/shared/src/main/scala/cats/effect/IO.scala @@ -1130,7 +1130,7 @@ object IO extends IOCompanionPlatform with IOLowPriorityImplicits { * Any exceptions thrown by the effect will be caught and sequenced into the `IO`. */ def delay[A](thunk: => A): IO[A] = { - val fn = Thunk.asFunction0(thunk) + val fn = () => thunk Delay(fn, Tracing.calculateTracingEvent(fn)) } diff --git a/core/shared/src/main/scala/cats/effect/Thunk.scala b/core/shared/src/main/scala/cats/effect/Thunk.scala deleted file mode 100644 index d60a39a080..0000000000 --- a/core/shared/src/main/scala/cats/effect/Thunk.scala +++ /dev/null @@ -1,29 +0,0 @@ -/* - * Copyright 2020-2023 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 - -/** - * A utility to convert a by-name `thunk: => A` to a `Function0[A]` (its binary representation). - * Scala 2 performs this optimization automatically but on Scala 3 the thunk is wrapped inside - * of a new `Function0`. See https://github.com/typelevel/cats-effect/pull/2226 - */ -private object Thunk { - private[this] val impl = - ((x: Any) => x).asInstanceOf[(=> Any) => Function0[Any]] - - def asFunction0[A](thunk: => A): Function0[A] = impl(thunk).asInstanceOf[Function0[A]] -} diff --git a/tests/shared/src/test/scala/cats/effect/ThunkSpec.scala b/tests/shared/src/test/scala/cats/effect/ThunkSpec.scala index 92624ab5bd..f51f7b1e98 100644 --- a/tests/shared/src/test/scala/cats/effect/ThunkSpec.scala +++ b/tests/shared/src/test/scala/cats/effect/ThunkSpec.scala @@ -18,11 +18,11 @@ package cats.effect class ThunkSpec extends BaseSpec { - "Thunk.asFunction0" should { + "IO.delay" should { "return the same function" in { var i = 0 val f = () => i += 1 - Thunk.asFunction0(f()) eq f + IO.delay(f()).asInstanceOf[IO.Delay[Unit]].thunk eq f } } From 88594c62c5d45a17deada61526c9bfc2fd47e2e5 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Tue, 22 Aug 2023 05:35:14 -0700 Subject: [PATCH 22/46] Another MiMa filter --- build.sbt | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/build.sbt b/build.sbt index dec1f87f15..679df2d278 100644 --- a/build.sbt +++ b/build.sbt @@ -650,7 +650,8 @@ lazy val core = crossProject(JSPlatform, JVMPlatform, NativePlatform) ProblemFilters.exclude[IncompatibleResultTypeProblem]( "cats.effect.unsafe.WorkerThread.sleep"), // #3787, internal utility that was no longer needed - ProblemFilters.exclude[MissingClassProblem]("cats.effect.Thunk") + ProblemFilters.exclude[MissingClassProblem]("cats.effect.Thunk"), + ProblemFilters.exclude[MissingClassProblem]("cats.effect.Thunk$") ) ++ { if (tlIsScala3.value) { // Scala 3 specific exclusions From 7918ca7f5d53bbc21740bbd32d3483b84688eb2b Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Tue, 29 Aug 2023 07:36:32 -0700 Subject: [PATCH 23/46] Changeup Cirrus CI config --- .cirrus.yml | 62 +++++++++++++++++++---------------------------------- 1 file changed, 22 insertions(+), 40 deletions(-) diff --git a/.cirrus.yml b/.cirrus.yml index 768443c3b6..c55dc27661 100644 --- a/.cirrus.yml +++ b/.cirrus.yml @@ -4,68 +4,50 @@ jvm_highcore_task: cpu: 4 memory: 8G matrix: - - name: JVM high-core-count 2.12 - script: sbt '++ 2.12' testsJVM/test - name: JVM high-core-count 2.13 - script: sbt '++ 2.13' testsJVM/test stressTests/Jcstress/run + script: sbt '++ 2.13' testsJVM/test - name: JVM high-core-count 3 script: sbt '++ 3' testsJVM/test -# jvm_arm_highcore_task: -# arm_container: -# image: sbtscala/scala-sbt:eclipse-temurin-jammy-17.0.5_8_1.9.0_3.3.0 -# cpu: 4 -# memory: 8G -# matrix: -# - name: JVM ARM high-core-count 2.12 -# script: sbt '++ 2.12' testsJVM/test -# - name: JVM ARM high-core-count 2.13 -# script: sbt '++ 2.13' testsJVM/test stressTests/Jcstress/run -# - name: JVM ARM high-core-count 3 -# script: sbt '++ 3' testsJVM/test +jvm_arm_highcore_task: + arm_container: + image: sbtscala/scala-sbt:eclipse-temurin-jammy-17.0.5_8_1.9.0_3.3.0 + cpu: 4 + memory: 8G + matrix: + - name: JVM ARM high-core-count 2.13 + script: sbt '++ 2.13' testsJVM/test + - name: JVM ARM high-core-count 3 + script: sbt '++ 3' testsJVM/test jvm_macos_highcore_task: macos_instance: image: ghcr.io/cirruslabs/macos-ventura-base:latest - cpu: 4 - memory: 8G matrix: - - name: JVM Apple Silicon high-core-count 2.12 - script: - - brew install sbt - - sbt '++ 2.12' testsJVM/test - name: JVM Apple Silicon high-core-count 2.13 script: - brew install sbt - - sbt '++ 2.13' testsJVM/test stressTests/Jcstress/run + - sbt '++ 2.13' testsJVM/test - name: JVM Apple Silicon high-core-count 3 script: - brew install sbt - sbt '++ 3' testsJVM/test -# native_arm_task: -# arm_container: -# dockerfile: .cirrus/Dockerfile -# cpu: 2 -# memory: 8G -# matrix: -# - name: Native ARM 2.12 -# script: sbt '++ 2.12' testsNative/test -# - name: Native ARM 2.13 -# script: sbt '++ 2.13' testsNative/test -# - name: Native ARM 3 -# script: sbt '++ 3' testsNative/test +native_arm_task: + arm_container: + dockerfile: .cirrus/Dockerfile + cpu: 2 + memory: 8G + matrix: + - name: Native ARM 2.13 + script: sbt '++ 2.13' testsNative/test + - name: Native ARM 3 + script: sbt '++ 3' testsNative/test native_macos_task: macos_instance: image: ghcr.io/cirruslabs/macos-ventura-base:latest - cpu: 2 - memory: 8G matrix: - - name: Native Apple Silicon 2.12 - script: - - brew install sbt - - sbt '++ 2.12' testsNative/test - name: Native Apple Silicon 2.13 script: - brew install sbt From c575b8db139b793ddfd7b701cd02e9c115974bd1 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Tue, 29 Aug 2023 07:47:32 -0700 Subject: [PATCH 24/46] Drop JVM Apple Silicon Scala 3 --- .cirrus.yml | 4 ---- 1 file changed, 4 deletions(-) diff --git a/.cirrus.yml b/.cirrus.yml index c55dc27661..cbf8936a60 100644 --- a/.cirrus.yml +++ b/.cirrus.yml @@ -28,10 +28,6 @@ jvm_macos_highcore_task: script: - brew install sbt - sbt '++ 2.13' testsJVM/test - - name: JVM Apple Silicon high-core-count 3 - script: - - brew install sbt - - sbt '++ 3' testsJVM/test native_arm_task: arm_container: From 3b3a4daef11b65321cc7a8967d45eed56161a69a Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Tue, 29 Aug 2023 08:04:02 -0700 Subject: [PATCH 25/46] More cuts --- .cirrus.yml | 10 ++-------- 1 file changed, 2 insertions(+), 8 deletions(-) diff --git a/.cirrus.yml b/.cirrus.yml index cbf8936a60..6e3b151060 100644 --- a/.cirrus.yml +++ b/.cirrus.yml @@ -24,10 +24,10 @@ jvm_macos_highcore_task: macos_instance: image: ghcr.io/cirruslabs/macos-ventura-base:latest matrix: - - name: JVM Apple Silicon high-core-count 2.13 + - name: JVM Apple Silicon high-core-count 3 script: - brew install sbt - - sbt '++ 2.13' testsJVM/test + - sbt '++ 3' testsJVM/test native_arm_task: arm_container: @@ -35,8 +35,6 @@ native_arm_task: cpu: 2 memory: 8G matrix: - - name: Native ARM 2.13 - script: sbt '++ 2.13' testsNative/test - name: Native ARM 3 script: sbt '++ 3' testsNative/test @@ -44,10 +42,6 @@ native_macos_task: macos_instance: image: ghcr.io/cirruslabs/macos-ventura-base:latest matrix: - - name: Native Apple Silicon 2.13 - script: - - brew install sbt - - sbt '++ 2.13' testsNative/test - name: Native Apple Silicon 3 script: - brew install sbt From a27262415c9ef966f0955d321c9bf2bfc2f7e9f9 Mon Sep 17 00:00:00 2001 From: William Rose Date: Tue, 29 Aug 2023 16:39:43 -0700 Subject: [PATCH 26/46] Changes to Hotswap to prevent swap from blocking get --- .../src/main/scala/cats/effect/std/Hotswap.scala | 12 ++++++------ .../src/test/scala/cats/effect/std/HotswapSpec.scala | 10 ++++++++++ 2 files changed, 16 insertions(+), 6 deletions(-) diff --git a/std/shared/src/main/scala/cats/effect/std/Hotswap.scala b/std/shared/src/main/scala/cats/effect/std/Hotswap.scala index 15ac983442..e84301a34d 100644 --- a/std/shared/src/main/scala/cats/effect/std/Hotswap.scala +++ b/std/shared/src/main/scala/cats/effect/std/Hotswap.scala @@ -131,12 +131,12 @@ object Hotswap { new Hotswap[F, R] { override def swap(next: Resource[F, R]): F[R] = - exclusive.surround { - F.uncancelable { poll => - poll(next.allocated).flatMap { - case (r, fin) => - swapFinalizer(Acquired(r, fin)).as(r) - } + F.uncancelable { poll => + poll(next.allocated).flatMap { + case (r, fin) => + poll(exclusive.surround { + swapFinalizer(Acquired(r, fin)).uncancelable.as(r) + }).onCancel(fin) } } diff --git a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala index 67ef946db8..d306712a40 100644 --- a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala +++ b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala @@ -18,6 +18,7 @@ package cats package effect package std +import cats.effect.Resource import cats.effect.kernel.Ref import scala.concurrent.duration._ @@ -104,6 +105,15 @@ class HotswapSpec extends BaseSpec { outer => go must completeAs(()) } + + "not block current resource while swap is instantiating new one" in ticked { + implicit ticker => + val go = Hotswap.create[IO, Unit].use { hs => + hs.swap(Resource.eval(IO.sleep(1.minute) *> IO.unit)).start *> + hs.get.use_.timeout(1.second) *> IO.unit + } + go must completeAs(()) + } } } From 79c600a32e2839fee43c8887c992b2f8632165a7 Mon Sep 17 00:00:00 2001 From: William Rose Date: Tue, 29 Aug 2023 17:56:18 -0700 Subject: [PATCH 27/46] Shift Hotswap#swap onCancel execution of fin to the Semaphore lock rather than the entire lock and swapFinalizer call --- std/shared/src/main/scala/cats/effect/std/Hotswap.scala | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/std/shared/src/main/scala/cats/effect/std/Hotswap.scala b/std/shared/src/main/scala/cats/effect/std/Hotswap.scala index e84301a34d..cf77c72345 100644 --- a/std/shared/src/main/scala/cats/effect/std/Hotswap.scala +++ b/std/shared/src/main/scala/cats/effect/std/Hotswap.scala @@ -134,9 +134,9 @@ object Hotswap { F.uncancelable { poll => poll(next.allocated).flatMap { case (r, fin) => - poll(exclusive.surround { + poll(exclusive.onCancel(Resource.eval(fin)).surround { swapFinalizer(Acquired(r, fin)).uncancelable.as(r) - }).onCancel(fin) + }) } } From 5eddc3cac368f2759627d0b94f2781278f8c11a4 Mon Sep 17 00:00:00 2001 From: William Rose Date: Wed, 30 Aug 2023 12:36:16 -0700 Subject: [PATCH 28/46] Adds test for successful Hotswap#swap cancelation while blocked by Hotswap#get --- .../scala/cats/effect/std/HotswapSpec.scala | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) diff --git a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala index d306712a40..08a74540dc 100644 --- a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala +++ b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala @@ -110,10 +110,27 @@ class HotswapSpec extends BaseSpec { outer => implicit ticker => val go = Hotswap.create[IO, Unit].use { hs => hs.swap(Resource.eval(IO.sleep(1.minute) *> IO.unit)).start *> + IO.sleep(5.seconds) *> hs.get.use_.timeout(1.second) *> IO.unit } go must completeAs(()) } + + "successfully cancel during swap and run finalizer if cancelation is requested while waiting for get to release" in ticked { + implicit ticker => + val go = (for { + log <- Ref.of[IO, List[String]](List()).toResource + (hs, _) <- Hotswap[IO, Unit](logged(log, "a")) + _ <- hs.get.evalMap(_ => IO.sleep(1.minute)).use_.start.toResource + _ <- IO.sleep(3.seconds).toResource + swapFib <- hs.swap(logged(log, "b")).start.toResource + _ <- IO.sleep(3.seconds).toResource + _ <- swapFib.cancel.toResource + value <- log.get.toResource + } yield value).use(res => IO(res)) + + go must completeAs(List("open a", "open b", "close b")) + } } } From 79e9a72d1008f7e463e4960709bef350e2043f36 Mon Sep 17 00:00:00 2001 From: William Rose Date: Wed, 30 Aug 2023 15:37:19 -0700 Subject: [PATCH 29/46] HotswapSpec test cleanup --- .../scala/cats/effect/std/HotswapSpec.scala | 22 +++++++++---------- 1 file changed, 11 insertions(+), 11 deletions(-) diff --git a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala index 08a74540dc..5dd208d108 100644 --- a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala +++ b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala @@ -109,7 +109,7 @@ class HotswapSpec extends BaseSpec { outer => "not block current resource while swap is instantiating new one" in ticked { implicit ticker => val go = Hotswap.create[IO, Unit].use { hs => - hs.swap(Resource.eval(IO.sleep(1.minute) *> IO.unit)).start *> + hs.swap(IO.sleep(1.minute).toResource).start *> IO.sleep(5.seconds) *> hs.get.use_.timeout(1.second) *> IO.unit } @@ -118,16 +118,16 @@ class HotswapSpec extends BaseSpec { outer => "successfully cancel during swap and run finalizer if cancelation is requested while waiting for get to release" in ticked { implicit ticker => - val go = (for { - log <- Ref.of[IO, List[String]](List()).toResource - (hs, _) <- Hotswap[IO, Unit](logged(log, "a")) - _ <- hs.get.evalMap(_ => IO.sleep(1.minute)).use_.start.toResource - _ <- IO.sleep(3.seconds).toResource - swapFib <- hs.swap(logged(log, "b")).start.toResource - _ <- IO.sleep(3.seconds).toResource - _ <- swapFib.cancel.toResource - value <- log.get.toResource - } yield value).use(res => IO(res)) + val go = Ref.of[IO, List[String]](List()).flatMap { log => + Hotswap[IO, Unit](logged(log, "a")).use { case (hs, _) => + for { + _ <- hs.get.evalMap(_ => IO.sleep(1.minute)).use_.start + _ <- IO.sleep(2.seconds) + _ <- hs.swap(logged(log, "b")).timeoutTo(1.second, IO.unit) + value <- log.get + } yield value + } + } go must completeAs(List("open a", "open b", "close b")) } From a1106aa56e3afc2ca9689a56797d715f514c4f98 Mon Sep 17 00:00:00 2001 From: William Rose Date: Thu, 31 Aug 2023 10:04:25 -0700 Subject: [PATCH 30/46] scalafmt --- .../test/scala/cats/effect/std/HotswapSpec.scala | 15 ++++++++------- 1 file changed, 8 insertions(+), 7 deletions(-) diff --git a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala index 5dd208d108..ee7773f22a 100644 --- a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala +++ b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala @@ -119,13 +119,14 @@ class HotswapSpec extends BaseSpec { outer => "successfully cancel during swap and run finalizer if cancelation is requested while waiting for get to release" in ticked { implicit ticker => val go = Ref.of[IO, List[String]](List()).flatMap { log => - Hotswap[IO, Unit](logged(log, "a")).use { case (hs, _) => - for { - _ <- hs.get.evalMap(_ => IO.sleep(1.minute)).use_.start - _ <- IO.sleep(2.seconds) - _ <- hs.swap(logged(log, "b")).timeoutTo(1.second, IO.unit) - value <- log.get - } yield value + Hotswap[IO, Unit](logged(log, "a")).use { + case (hs, _) => + for { + _ <- hs.get.evalMap(_ => IO.sleep(1.minute)).use_.start + _ <- IO.sleep(2.seconds) + _ <- hs.swap(logged(log, "b")).timeoutTo(1.second, IO.unit) + value <- log.get + } yield value } } From d528afbd95078ba7e8c55ef79efbe3095cbbf63f Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Sat, 2 Sep 2023 00:10:09 +0000 Subject: [PATCH 31/46] Add failing test for `swap` cancelation --- .../test/scala/cats/effect/std/HotswapSpec.scala | 15 +++++++++++++++ 1 file changed, 15 insertions(+) diff --git a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala index ee7773f22a..6f3ee3d287 100644 --- a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala +++ b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala @@ -20,6 +20,8 @@ package std import cats.effect.Resource import cats.effect.kernel.Ref +import cats.effect.testkit.TestControl +import cats.effect.unsafe.IORuntimeConfig import scala.concurrent.duration._ @@ -132,6 +134,19 @@ class HotswapSpec extends BaseSpec { outer => go must completeAs(List("open a", "open b", "close b")) } + + "swap is safe to concurrent cancelation" in ticked { implicit ticker => + val go = IO.ref(false).flatMap { open => + Hotswap[IO, Unit](Resource.unit) + .use { + case (hs, _) => + hs.swap(Resource.make(open.set(true))(_ => open.set(false))) + } + .race(IO.unit) *> open.get.map(_ must beFalse) + } + + TestControl.executeEmbed(go, IORuntimeConfig(1, 2)).replicateA_(1000) must completeAs(()) + } } } From 7a0d8c35b6a15a01ea4ce6bffb765cfa5f7ea249 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Sat, 2 Sep 2023 08:59:24 -0700 Subject: [PATCH 32/46] Make rendering of `[WARNING]` consistent Co-authored-by: Daniel Spiewak --- core/jvm/src/main/scala/cats/effect/IOApp.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/jvm/src/main/scala/cats/effect/IOApp.scala b/core/jvm/src/main/scala/cats/effect/IOApp.scala index dcd01d6804..55794ad571 100644 --- a/core/jvm/src/main/scala/cats/effect/IOApp.scala +++ b/core/jvm/src/main/scala/cats/effect/IOApp.scala @@ -328,7 +328,7 @@ trait IOApp { System .err .println( - """|Warning: IOApp `main` is running on a thread other than the main thread. + """|[WARNING] IOApp `main` is running on a thread other than the main thread. |This may prevent correct resource cleanup after `main` completes. |This condition could be caused by executing `run` in an interactive sbt session with `fork := false`. |Set `Compile / run / fork := true` in this project to resolve this. From f0cdcb2a1ee8ba7fdc64f1c891c61aa71fd1f855 Mon Sep 17 00:00:00 2001 From: Will Rose <3587401+forkedcancel@users.noreply.github.com> Date: Thu, 7 Sep 2023 09:53:11 -0700 Subject: [PATCH 33/46] Update tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala Co-authored-by: Arman Bilge --- tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala index 6f3ee3d287..b7c1130392 100644 --- a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala +++ b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala @@ -140,7 +140,7 @@ class HotswapSpec extends BaseSpec { outer => Hotswap[IO, Unit](Resource.unit) .use { case (hs, _) => - hs.swap(Resource.make(open.set(true))(_ => open.set(false))) + hs.swap(Resource.make(open.set(true))(_ => open.getAndSet(false).map(_ should beTrue).void)) } .race(IO.unit) *> open.get.map(_ must beFalse) } From 4c82c1900aa3c3ab4bd8305319b0fc66c4f3ecbf Mon Sep 17 00:00:00 2001 From: Will Rose <3587401+forkedcancel@users.noreply.github.com> Date: Thu, 7 Sep 2023 09:55:13 -0700 Subject: [PATCH 34/46] Update std/shared/src/main/scala/cats/effect/std/Hotswap.scala Co-authored-by: Arman Bilge --- std/shared/src/main/scala/cats/effect/std/Hotswap.scala | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/std/shared/src/main/scala/cats/effect/std/Hotswap.scala b/std/shared/src/main/scala/cats/effect/std/Hotswap.scala index cf77c72345..51c81f52bc 100644 --- a/std/shared/src/main/scala/cats/effect/std/Hotswap.scala +++ b/std/shared/src/main/scala/cats/effect/std/Hotswap.scala @@ -134,8 +134,8 @@ object Hotswap { F.uncancelable { poll => poll(next.allocated).flatMap { case (r, fin) => - poll(exclusive.onCancel(Resource.eval(fin)).surround { - swapFinalizer(Acquired(r, fin)).uncancelable.as(r) + exclusive.mapK(poll).onCancel(Resource.eval(fin)).surround { + swapFinalizer(Acquired(r, fin)).as(r) }) } } From feceb1954d4c7cfc70c34c47cd4c3f4434fc0e42 Mon Sep 17 00:00:00 2001 From: Kamil Kloch Date: Mon, 11 Sep 2023 16:31:21 +0200 Subject: [PATCH 35/46] Extract parasitic EC outside `unsafeRunAsync`, inline `unsafeRunAndForget`. --- .../scala/cats/effect/std/Dispatcher.scala | 36 +++++++++---------- 1 file changed, 18 insertions(+), 18 deletions(-) diff --git a/std/shared/src/main/scala/cats/effect/std/Dispatcher.scala b/std/shared/src/main/scala/cats/effect/std/Dispatcher.scala index 4d08927e33..d9b61211cd 100644 --- a/std/shared/src/main/scala/cats/effect/std/Dispatcher.scala +++ b/std/shared/src/main/scala/cats/effect/std/Dispatcher.scala @@ -17,16 +17,16 @@ package cats.effect.std import cats.effect.kernel.{Async, Outcome, Resource} +import cats.effect.std.Dispatcher.parasiticEC import cats.syntax.all._ +import java.util.concurrent.ThreadLocalRandom +import java.util.concurrent.atomic.{AtomicBoolean, AtomicReference} import scala.annotation.tailrec import scala.collection.mutable import scala.concurrent.{ExecutionContext, Future, Promise} import scala.util.{Failure, Success} -import java.util.concurrent.ThreadLocalRandom -import java.util.concurrent.atomic.{AtomicBoolean, AtomicReference} - /** * A fiber-based supervisor utility for evaluating effects across an impure boundary. This is * useful when working with reactive interfaces that produce potentially many values (as opposed @@ -68,28 +68,28 @@ trait Dispatcher[F[_]] extends DispatcherPlatform[F] { /** * Submits an effect to be executed with fire-and-forget semantics. */ - def unsafeRunAndForget[A](fa: F[A]): Unit = { - unsafeRunAsync(fa) { - case Left(t) => t.printStackTrace() - case Right(_) => () - } - } + def unsafeRunAndForget[A](fa: F[A]): Unit = + unsafeToFutureCancelable(fa) + ._1 + .onComplete { + case Failure(ex) => ex.printStackTrace() + case _ => () + }(parasiticEC) // package-private because it's just an internal utility which supports specific implementations // anyone who needs this type of thing should use unsafeToFuture and then onComplete - private[std] def unsafeRunAsync[A](fa: F[A])(cb: Either[Throwable, A] => Unit): Unit = { - // this is safe because the only invocation will be cb - implicit val parasitic: ExecutionContext = new ExecutionContext { - def execute(runnable: Runnable) = runnable.run() - def reportFailure(t: Throwable) = t.printStackTrace() - } - - unsafeToFuture(fa).onComplete(t => cb(t.toEither)) - } + private[std] def unsafeRunAsync[A](fa: F[A])(cb: Either[Throwable, A] => Unit): Unit = + unsafeToFutureCancelable(fa)._1.onComplete(t => cb(t.toEither))(parasiticEC) } object Dispatcher { + private val parasiticEC: ExecutionContext = new ExecutionContext { + def execute(runnable: Runnable) = runnable.run() + + def reportFailure(t: Throwable) = t.printStackTrace() + } + private[this] val Cpus: Int = Runtime.getRuntime().availableProcessors() private[this] val Noop: () => Unit = () => () From 102f9fe0b7c06402458f632d07475c2a1a10e6c6 Mon Sep 17 00:00:00 2001 From: Kamil Kloch Date: Mon, 11 Sep 2023 16:41:57 +0200 Subject: [PATCH 36/46] Cosmetics. --- std/shared/src/main/scala/cats/effect/std/Dispatcher.scala | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/std/shared/src/main/scala/cats/effect/std/Dispatcher.scala b/std/shared/src/main/scala/cats/effect/std/Dispatcher.scala index d9b61211cd..3937f9b1bc 100644 --- a/std/shared/src/main/scala/cats/effect/std/Dispatcher.scala +++ b/std/shared/src/main/scala/cats/effect/std/Dispatcher.scala @@ -20,13 +20,14 @@ import cats.effect.kernel.{Async, Outcome, Resource} import cats.effect.std.Dispatcher.parasiticEC import cats.syntax.all._ -import java.util.concurrent.ThreadLocalRandom -import java.util.concurrent.atomic.{AtomicBoolean, AtomicReference} import scala.annotation.tailrec import scala.collection.mutable import scala.concurrent.{ExecutionContext, Future, Promise} import scala.util.{Failure, Success} +import java.util.concurrent.ThreadLocalRandom +import java.util.concurrent.atomic.{AtomicBoolean, AtomicReference} + /** * A fiber-based supervisor utility for evaluating effects across an impure boundary. This is * useful when working with reactive interfaces that produce potentially many values (as opposed From 279fc23dcc480e70ea0be1b2bdf550e422189ca2 Mon Sep 17 00:00:00 2001 From: Kamil Kloch Date: Mon, 11 Sep 2023 17:00:41 +0200 Subject: [PATCH 37/46] Cosmetics. --- .../src/main/scala/cats/effect/std/Dispatcher.scala | 12 +++++------- 1 file changed, 5 insertions(+), 7 deletions(-) diff --git a/std/shared/src/main/scala/cats/effect/std/Dispatcher.scala b/std/shared/src/main/scala/cats/effect/std/Dispatcher.scala index 3937f9b1bc..943aa74889 100644 --- a/std/shared/src/main/scala/cats/effect/std/Dispatcher.scala +++ b/std/shared/src/main/scala/cats/effect/std/Dispatcher.scala @@ -70,17 +70,15 @@ trait Dispatcher[F[_]] extends DispatcherPlatform[F] { * Submits an effect to be executed with fire-and-forget semantics. */ def unsafeRunAndForget[A](fa: F[A]): Unit = - unsafeToFutureCancelable(fa) - ._1 - .onComplete { - case Failure(ex) => ex.printStackTrace() - case _ => () - }(parasiticEC) + unsafeToFuture(fa).onComplete { + case Failure(ex) => ex.printStackTrace() + case _ => () + }(parasiticEC) // package-private because it's just an internal utility which supports specific implementations // anyone who needs this type of thing should use unsafeToFuture and then onComplete private[std] def unsafeRunAsync[A](fa: F[A])(cb: Either[Throwable, A] => Unit): Unit = - unsafeToFutureCancelable(fa)._1.onComplete(t => cb(t.toEither))(parasiticEC) + unsafeToFuture(fa).onComplete(t => cb(t.toEither))(parasiticEC) } object Dispatcher { From f916dc6c34286d193eb45f535ac4e5265698717e Mon Sep 17 00:00:00 2001 From: William Rose Date: Thu, 14 Sep 2023 15:43:13 -0700 Subject: [PATCH 38/46] Syntax/formatting issues from accepted code suggestions --- std/shared/src/main/scala/cats/effect/std/Hotswap.scala | 2 +- tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala | 3 ++- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/std/shared/src/main/scala/cats/effect/std/Hotswap.scala b/std/shared/src/main/scala/cats/effect/std/Hotswap.scala index 51c81f52bc..454744b4ba 100644 --- a/std/shared/src/main/scala/cats/effect/std/Hotswap.scala +++ b/std/shared/src/main/scala/cats/effect/std/Hotswap.scala @@ -136,7 +136,7 @@ object Hotswap { case (r, fin) => exclusive.mapK(poll).onCancel(Resource.eval(fin)).surround { swapFinalizer(Acquired(r, fin)).as(r) - }) + } } } diff --git a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala index b7c1130392..e78041a967 100644 --- a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala +++ b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala @@ -140,7 +140,8 @@ class HotswapSpec extends BaseSpec { outer => Hotswap[IO, Unit](Resource.unit) .use { case (hs, _) => - hs.swap(Resource.make(open.set(true))(_ => open.getAndSet(false).map(_ should beTrue).void)) + hs.swap(Resource.make(open.set(true))(_ => + open.getAndSet(false).map(_ should beTrue).void)) } .race(IO.unit) *> open.get.map(_ must beFalse) } From 5025c28ea6cde5d30dc68327aa648e5e3bdb936d Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Sun, 17 Sep 2023 08:53:00 -0700 Subject: [PATCH 39/46] Drop macOS jobs from Cirrus --- .cirrus.yml | 18 ------------------ 1 file changed, 18 deletions(-) diff --git a/.cirrus.yml b/.cirrus.yml index 6e3b151060..d9f22944bf 100644 --- a/.cirrus.yml +++ b/.cirrus.yml @@ -20,15 +20,6 @@ jvm_arm_highcore_task: - name: JVM ARM high-core-count 3 script: sbt '++ 3' testsJVM/test -jvm_macos_highcore_task: - macos_instance: - image: ghcr.io/cirruslabs/macos-ventura-base:latest - matrix: - - name: JVM Apple Silicon high-core-count 3 - script: - - brew install sbt - - sbt '++ 3' testsJVM/test - native_arm_task: arm_container: dockerfile: .cirrus/Dockerfile @@ -37,12 +28,3 @@ native_arm_task: matrix: - name: Native ARM 3 script: sbt '++ 3' testsNative/test - -native_macos_task: - macos_instance: - image: ghcr.io/cirruslabs/macos-ventura-base:latest - matrix: - - name: Native Apple Silicon 3 - script: - - brew install sbt - - sbt '++ 3' testsNative/test From 366fbf70f120e83a3374ccd7986f9d028ca34ce6 Mon Sep 17 00:00:00 2001 From: Daniel Spiewak Date: Sun, 17 Sep 2023 13:42:36 -0500 Subject: [PATCH 40/46] Skip `WorkerNameSpec` --- .../cats/effect/unsafe/WorkerThreadNameSpec.scala | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/tests/jvm/src/test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala b/tests/jvm/src/test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala index 5817d10e34..b9d3ece97d 100644 --- a/tests/jvm/src/test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala +++ b/tests/jvm/src/test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala @@ -16,9 +16,9 @@ package cats.effect.unsafe -import cats.effect.{BaseSpec, IO} +import cats.effect.{BaseSpec/*, IO*/} import cats.effect.testkit.TestInstances -import cats.syntax.all._ +// import cats.syntax.all._ import scala.concurrent.duration._ @@ -55,7 +55,7 @@ class WorkerThreadNameSpec extends BaseSpec with TestInstances { } "WorkerThread" should { - "rename itself when entering and exiting blocking region" in real { + "rename itself when entering and exiting blocking region" in skipped("this test is quite flaky in CI") /*real { for { _ <- IO.cede computeThread <- threadInfo @@ -91,10 +91,10 @@ class WorkerThreadNameSpec extends BaseSpec with TestInstances { resetBlockerThread must beSome((_: String).endsWith("-0")) .setMessage("blocker thread index was not correct") } - } + }*/ } - private val threadInfo = - IO((Thread.currentThread().getName(), Thread.currentThread().getId())) + /*private val threadInfo = + IO((Thread.currentThread().getName(), Thread.currentThread().getId()))*/ } From 38f49760826a1cfd2206b4ecbc2b7290e4303425 Mon Sep 17 00:00:00 2001 From: Daniel Spiewak Date: Sun, 17 Sep 2023 13:55:48 -0500 Subject: [PATCH 41/46] Skip starvation check on Node --- .../test/scala/cats/effect/IOAppSpec.scala | 29 ++++++++++--------- 1 file changed, 15 insertions(+), 14 deletions(-) diff --git a/tests/jvm/src/test/scala/cats/effect/IOAppSpec.scala b/tests/jvm/src/test/scala/cats/effect/IOAppSpec.scala index 75688054fd..82b75f7a1a 100644 --- a/tests/jvm/src/test/scala/cats/effect/IOAppSpec.scala +++ b/tests/jvm/src/test/scala/cats/effect/IOAppSpec.scala @@ -203,20 +203,6 @@ class IOAppSpec extends Specification { h.stderr() must not(contain("boom")) } - "warn on cpu starvation" in { - val h = platform(CpuStarvation, List.empty) - h.awaitStatus() - val err = h.stderr() - err must not(contain("[WARNING] Failed to register Cats Effect CPU")) - err must contain("[WARNING] Your app's responsiveness") - // we use a regex because time has too many corner cases - a test run at just the wrong - // moment on new year's eve, etc - err must beMatching( - // (?s) allows matching across line breaks - """(?s)^\d{4}-[01]\d-[0-3]\dT[012]\d:[0-6]\d:[0-6]\d(?:\.\d{1,3})?Z \[WARNING\] Your app's responsiveness.*""" - ) - } - "custom runtime installed as global" in { val h = platform(CustomRuntime, List.empty) h.awaitStatus() mustEqual 0 @@ -285,6 +271,8 @@ class IOAppSpec extends Specification { "support main thread evaluation" in skipped( "JavaScript is all main thread, all the time") + "warn on cpu starvation" in skipped( + "starvation detection works on Node, but the test struggles with determinism") } else { val isJava8 = sys.props.get("java.version").filter(_.startsWith("1.8")).isDefined @@ -337,6 +325,19 @@ class IOAppSpec extends Specification { "[WARNING] A Cats Effect worker thread was detected to be in a blocked state") } + "warn on cpu starvation" in { + val h = platform(CpuStarvation, List.empty) + h.awaitStatus() + val err = h.stderr() + err must not(contain("[WARNING] Failed to register Cats Effect CPU")) + err must contain("[WARNING] Your app's responsiveness") + // we use a regex because time has too many corner cases - a test run at just the wrong + // moment on new year's eve, etc + err must beMatching( + // (?s) allows matching across line breaks + """(?s)^\d{4}-[01]\d-[0-3]\dT[012]\d:[0-6]\d:[0-6]\d(?:\.\d{1,3})?Z \[WARNING\] Your app's responsiveness.*""" + ) + } } } () From 6bf2acc031217a20f1b8d1ac02725103bb1817c3 Mon Sep 17 00:00:00 2001 From: Daniel Spiewak Date: Sun, 17 Sep 2023 13:58:35 -0500 Subject: [PATCH 42/46] =?UTF-8?q?Sigh=E2=80=A6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../scala/cats/effect/unsafe/WorkerThreadNameSpec.scala | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/tests/jvm/src/test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala b/tests/jvm/src/test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala index b9d3ece97d..b2be8ec719 100644 --- a/tests/jvm/src/test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala +++ b/tests/jvm/src/test/scala/cats/effect/unsafe/WorkerThreadNameSpec.scala @@ -16,9 +16,8 @@ package cats.effect.unsafe -import cats.effect.{BaseSpec/*, IO*/} +import cats.effect.{BaseSpec /*, IO*/} import cats.effect.testkit.TestInstances -// import cats.syntax.all._ import scala.concurrent.duration._ @@ -55,7 +54,9 @@ class WorkerThreadNameSpec extends BaseSpec with TestInstances { } "WorkerThread" should { - "rename itself when entering and exiting blocking region" in skipped("this test is quite flaky in CI") /*real { + "rename itself when entering and exiting blocking region" in skipped( + "this test is quite flaky in CI" + ) /*real { for { _ <- IO.cede computeThread <- threadInfo From 854e2b81cc6ddf53c430427f5e9c5841b175dbac Mon Sep 17 00:00:00 2001 From: Will Rose <3587401+forkedcancel@users.noreply.github.com> Date: Mon, 18 Sep 2023 09:28:56 -0700 Subject: [PATCH 43/46] Update HotswapSpec.scala Co-authored-by: Daniel Urban --- tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala index e78041a967..550e3b0222 100644 --- a/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala +++ b/tests/shared/src/test/scala/cats/effect/std/HotswapSpec.scala @@ -113,7 +113,7 @@ class HotswapSpec extends BaseSpec { outer => val go = Hotswap.create[IO, Unit].use { hs => hs.swap(IO.sleep(1.minute).toResource).start *> IO.sleep(5.seconds) *> - hs.get.use_.timeout(1.second) *> IO.unit + hs.get.use_.timeout(1.second).void } go must completeAs(()) } From f7d7bc922ea4ad40d2dfb417dbec168e3d1d9b25 Mon Sep 17 00:00:00 2001 From: Arman Bilge Date: Fri, 22 Sep 2023 16:20:10 +0000 Subject: [PATCH 44/46] Run Cirrus only for labeled PRs or tags --- .cirrus.yml | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/.cirrus.yml b/.cirrus.yml index d9f22944bf..9bc3b32c4b 100644 --- a/.cirrus.yml +++ b/.cirrus.yml @@ -1,4 +1,6 @@ jvm_highcore_task: + only_if: $CIRRUS_TAG != '' || $CIRRUS_PR != '' + required_pr_labels: Cirrus JVM container: image: sbtscala/scala-sbt:eclipse-temurin-jammy-17.0.5_8_1.9.0_3.3.0 cpu: 4 @@ -10,6 +12,8 @@ jvm_highcore_task: script: sbt '++ 3' testsJVM/test jvm_arm_highcore_task: + only_if: $CIRRUS_TAG != '' || $CIRRUS_PR != '' + required_pr_labels: Cirrus JVM arm_container: image: sbtscala/scala-sbt:eclipse-temurin-jammy-17.0.5_8_1.9.0_3.3.0 cpu: 4 @@ -21,6 +25,8 @@ jvm_arm_highcore_task: script: sbt '++ 3' testsJVM/test native_arm_task: + only_if: $CIRRUS_TAG != '' || $CIRRUS_PR != '' + required_pr_labels: Cirrus Native arm_container: dockerfile: .cirrus/Dockerfile cpu: 2 From 778186565fc86ccd0c8faa4a44928b64a7368386 Mon Sep 17 00:00:00 2001 From: Daniel Spiewak Date: Sat, 23 Sep 2023 14:47:06 -0500 Subject: [PATCH 45/46] Updated versions script to look for `**` pattern --- README.md | 2 +- scripts/update-versions.sh | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index b182c620d1..b167aa2390 100644 --- a/README.md +++ b/README.md @@ -11,7 +11,7 @@ ## Getting Started -- Wired: **3.4.11** +- Wired: **3.5.1** - Tired: **2.5.5** (end of life) ```scala diff --git a/scripts/update-versions.sh b/scripts/update-versions.sh index e5b19ec70c..02b1603861 100755 --- a/scripts/update-versions.sh +++ b/scripts/update-versions.sh @@ -22,3 +22,4 @@ new_version="$2" # perl is ironically more portable than sed because of GNU/BSD differences # the quote reduce the false positive rate find . -type f -name '*.md' -exec perl -pi -e "s/\"$old_version\"/\"$new_version\"/g" {} \; +find . -type f -name '*.md' -exec perl -pi -e "s/\\*\\*$old_version\\*\\*/\\*\\*$new_version\\*\\*/g" {} \; From f436fe691418164f61eb4d3dcd3b9eea761ac2cf Mon Sep 17 00:00:00 2001 From: Daniel Spiewak Date: Mon, 25 Sep 2023 14:53:26 -0500 Subject: [PATCH 46/46] Fix discarded value issue --- .../js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala | 2 ++ 1 file changed, 2 insertions(+) diff --git a/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala b/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala index df0b7edd52..432e82bdf8 100644 --- a/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala +++ b/tests/js/src/test/scala/cats/effect/unsafe/JSArrayQueueSpec.scala @@ -66,6 +66,8 @@ class JSArrayQueueSpec extends BaseSpec with ScalaCheck { val expected = shadow.dequeue() got must beEqualTo(expected) checkContents() + } else { + ok } }