From 269681c254f178cd5899b65b14e60511219bb188 Mon Sep 17 00:00:00 2001 From: Emrys Ingersoll Date: Fri, 27 Nov 2020 14:30:31 -0600 Subject: [PATCH 1/2] Raise exception when IOApp main fiber is canceled --- .../js/src/main/scala/cats/effect/IOApp.scala | 13 +++++++--- .../src/main/scala/cats/effect/IOApp.scala | 24 ++++++++++++------- .../test/scala/cats/effect/IOAppSpec.scala | 11 +++++++++ 3 files changed, 36 insertions(+), 12 deletions(-) diff --git a/core/js/src/main/scala/cats/effect/IOApp.scala b/core/js/src/main/scala/cats/effect/IOApp.scala index 937fef5615..c441e3d851 100644 --- a/core/js/src/main/scala/cats/effect/IOApp.scala +++ b/core/js/src/main/scala/cats/effect/IOApp.scala @@ -42,11 +42,18 @@ trait IOApp { else args.toList - IO.race(run(argList), keepAlive) + Spawn[IO].raceOutcome[ExitCode, Nothing](run(argList), keepAlive) + .flatMap { + case Left(Outcome.Canceled()) => + IO.raiseError(new RuntimeException("IOApp main fiber canceled")) + case Left(Outcome.Errored(t)) => IO.raiseError(t) + case Left(Outcome.Succeeded(code)) => code + case Right(Outcome.Errored(t)) => IO.raiseError(t) + case Right(_) => sys.error("impossible") + } .unsafeRunAsync({ case Left(t) => throw t - case Right(Left(code)) => reportExitCode(code) - case Right(Right(_)) => sys.error("impossible") + case Right(code) => reportExitCode(code) })(unsafe.IORuntime.global) } diff --git a/core/jvm/src/main/scala/cats/effect/IOApp.scala b/core/jvm/src/main/scala/cats/effect/IOApp.scala index 31d04620a8..14057e6e03 100644 --- a/core/jvm/src/main/scala/cats/effect/IOApp.scala +++ b/core/jvm/src/main/scala/cats/effect/IOApp.scala @@ -34,15 +34,21 @@ trait IOApp { val ioa = run(args.toList) - val fiber = ioa.unsafeRunFiber( - { t => - error = t - latch.countDown() - }, - { a => - result = a - latch.countDown() - })(runtime) + val fiber = + ioa.onCancel( + IO { + error = new RuntimeException("IOApp main fiber canceled") + latch.countDown() + }) + .unsafeRunFiber( + { t => + error = t + latch.countDown() + }, + { a => + result = a + latch.countDown() + })(runtime) def handleShutdown(): Unit = { if (latch.getCount() > 0) { diff --git a/core/jvm/src/test/scala/cats/effect/IOAppSpec.scala b/core/jvm/src/test/scala/cats/effect/IOAppSpec.scala index 2ae2c645c8..bfdd3d86f3 100644 --- a/core/jvm/src/test/scala/cats/effect/IOAppSpec.scala +++ b/core/jvm/src/test/scala/cats/effect/IOAppSpec.scala @@ -108,6 +108,12 @@ class IOAppSpec extends Specification { h.awaitStatus() mustEqual 1 h.stderr() must contain("Boom!") } + + "exit on canceled" in { + val h = java(Canceled, List.empty) + h.awaitStatus() mustEqual 1 + h.stderr() must contain("canceled") + } } } @@ -179,4 +185,9 @@ package examples { _ <- IO.never[Unit] } yield ExitCode.Success } + + object Canceled extends IOApp { + def run(args: List[String]): IO[ExitCode] = + IO.canceled.as(ExitCode.Success) + } } From 9815784573662a390e6b46950079855eb0db18e1 Mon Sep 17 00:00:00 2001 From: Emrys Ingersoll Date: Fri, 27 Nov 2020 15:17:53 -0600 Subject: [PATCH 2/2] Apply scalafmt --- .../js/src/main/scala/cats/effect/IOApp.scala | 3 ++- .../src/main/scala/cats/effect/IOApp.scala | 22 +++++++++---------- 2 files changed, 13 insertions(+), 12 deletions(-) diff --git a/core/js/src/main/scala/cats/effect/IOApp.scala b/core/js/src/main/scala/cats/effect/IOApp.scala index c441e3d851..9b293af7dc 100644 --- a/core/js/src/main/scala/cats/effect/IOApp.scala +++ b/core/js/src/main/scala/cats/effect/IOApp.scala @@ -42,7 +42,8 @@ trait IOApp { else args.toList - Spawn[IO].raceOutcome[ExitCode, Nothing](run(argList), keepAlive) + Spawn[IO] + .raceOutcome[ExitCode, Nothing](run(argList), keepAlive) .flatMap { case Left(Outcome.Canceled()) => IO.raiseError(new RuntimeException("IOApp main fiber canceled")) diff --git a/core/jvm/src/main/scala/cats/effect/IOApp.scala b/core/jvm/src/main/scala/cats/effect/IOApp.scala index 14057e6e03..98f324a6f7 100644 --- a/core/jvm/src/main/scala/cats/effect/IOApp.scala +++ b/core/jvm/src/main/scala/cats/effect/IOApp.scala @@ -35,20 +35,20 @@ trait IOApp { val ioa = run(args.toList) val fiber = - ioa.onCancel( - IO { + ioa + .onCancel(IO { error = new RuntimeException("IOApp main fiber canceled") latch.countDown() }) - .unsafeRunFiber( - { t => - error = t - latch.countDown() - }, - { a => - result = a - latch.countDown() - })(runtime) + .unsafeRunFiber( + { t => + error = t + latch.countDown() + }, + { a => + result = a + latch.countDown() + })(runtime) def handleShutdown(): Unit = { if (latch.getCount() > 0) {