Skip to content

Commit

Permalink
fix: Retry to retry non-IOException
Browse files Browse the repository at this point in the history
**Problem**
Retry(...) only handles IOExceptions.

**Solution**
This fixes it to handle NonFatal.
  • Loading branch information
eed3si9n committed Dec 29, 2024
1 parent 4aab2d8 commit 9503542
Show file tree
Hide file tree
Showing 5 changed files with 19 additions and 10 deletions.
3 changes: 2 additions & 1 deletion build.sbt
Original file line number Diff line number Diff line change
Expand Up @@ -80,8 +80,9 @@ val io = (project in file("io"))
commonSettings,
name := "IO",
libraryDependencies ++= {
Vector(scalaCompiler.value % Test, scalaCheck % Test, scalatest % Test)
Vector(scalaCompiler.value % Test, scalaVerify % Test, scalaCheck % Test, scalatest % Test)
} ++ Vector(swovalFiles),
testFrameworks += new TestFramework("verify.runner.Framework"),
Test / fork := System.getProperty("sbt.test.fork", "false") == "true",
Test / testForkedParallel := true,
Compile / generateContrabands / sourceManaged := baseDirectory.value / "src" / "main" / "contraband-scala",
Expand Down
6 changes: 3 additions & 3 deletions io/src/main/scala/sbt/internal/io/Retry.scala
Original file line number Diff line number Diff line change
Expand Up @@ -34,19 +34,19 @@ private[sbt] object Retry {
excludedExceptions: Class[? <: IOException]*,
): T = {
require(limit >= 1, "limit must be 1 or higher: was: " + limit)
def filter(e: Exception): Boolean = excludedExceptions match {
def filter(e: Throwable): Boolean = excludedExceptions match {
case s if s.nonEmpty =>
!excludedExceptions.exists(_.isAssignableFrom(e.getClass))
case _ =>
true
}
var attempt = 1
var firstException: IOException = null
var firstException: Throwable = null
while (attempt <= limit) {
try {
return f
} catch {
case e: IOException if filter(e) =>
case NonFatal(e) if filter(e) =>
if (firstException == null) firstException = e
// https://github.com/sbt/io/issues/295
// On Windows, we're seeing java.nio.file.AccessDeniedException with sleep(0).
Expand Down
17 changes: 12 additions & 5 deletions io/src/test/scala/sbt/internal/io/RetrySpec.scala
Original file line number Diff line number Diff line change
Expand Up @@ -14,11 +14,9 @@ package sbt.internal.io
import java.io.IOException
import java.util.concurrent.atomic.AtomicInteger

import org.scalatest.flatspec.AnyFlatSpec

final class RetrySpec extends AnyFlatSpec {
object RetrySpec extends verify.BasicTestSuite {
private val noExcluded: List[Class[? <: IOException]] = List[Class[? <: IOException]]()
"retry" should "throw first exception after number of failures" in {
test("retry should throw first exception after number of failures") {
val i = new AtomicInteger()
def throww(): Any = throw new IOException(i.incrementAndGet().toString)
try {
Expand All @@ -31,7 +29,7 @@ final class RetrySpec extends AnyFlatSpec {
}
}

"retry" should "throw recover" in {
test("retry should throw recover") {
for (recoveryStep <- (1 to 14)) {
val i = new AtomicInteger()
val value = Retry(
Expand All @@ -46,4 +44,13 @@ final class RetrySpec extends AnyFlatSpec {
assert(value == "recover")
}
}

test("retry should recover from non-IO exceptions") {
val i = new AtomicInteger()
def throww(): Any =
if (i.incrementAndGet() == 5) 0
else ???
Retry(throww())
()
}
}
1 change: 1 addition & 0 deletions project/Dependencies.scala
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ object Dependencies {
"org.scala-lang" % "scala-compiler" % v
}

val scalaVerify = "com.eed3si9n.verify" %% "verify" % "1.0.0"
val scalaCheck = "org.scalacheck" %% "scalacheck" % "1.18.1"
val scalatest = "org.scalatest" %% "scalatest" % "3.2.19"
val swovalFiles = "com.swoval" % "file-tree-views" % "2.1.12"
Expand Down
2 changes: 1 addition & 1 deletion project/build.properties
Original file line number Diff line number Diff line change
@@ -1 +1 @@
sbt.version=1.9.8
sbt.version=1.10.7

0 comments on commit 9503542

Please sign in to comment.