This is a followup to Matching on a context function.
Using @Jasper-M 's help, I was able to implement what I need, namely a withThreads
context that runs code on an implicit thread pool but behaves differently whether that code produces a future or not.
This demo application (available at GitHub - charpov/UnusedWarning: A confusing Scala warning; the warning disappears if Test2 is uncommented.) produces an interesting warning:
package foo
import tinyscalautils.threads.{ Execute, withThreads }
@main def Test1() =
withThreads():
Execute(println("X"))
/*
def Test2() =
withThreads():
scala.concurrent.Future(println("X"))
*/
sbt run
executes the program but produces this warning:
[warn] /.../UnusedWarning/S/src/main/scala/tinyscalautils/threads/withThreads.scala: unused local definition
Note that:
- The message refers to a non-existing local directory and to a source file from the library where
withThreads
is implemented. - Uncommenting
Test2
makes the warning disappear.
Further investigation shows the warning to refer to this code:
transparent inline def _withThreads[A](n: Int, awaitTermination: Boolean)(
code: ExecutionContextExecutorService ?=> A
) =
val f = exec => code(using exec)
inline f match
case g: (ExecutionContextExecutorService => Future[?]) =>
_withThreadsAndWait(n, awaitTermination)(g)
case _ =>
_withThreadsAndWait(n, awaitTermination)(exec => { code(using exec); Future.unit })
and the warning is:
[warn] 117 | val f = exec => code(using exec)
[warn] | ^
[warn] | unused local definition
What seems to be happening is that if withThreads
is used somewhere with a future, the first case
is matched and f
is considered being used. However, if all uses of withThreads
are with non-future code, only the second case
is used and f
is considered unused.
This feels buggy. It’s confusing to get warnings in an application that refers to (inlined) code from a library. Also, f
is clearly used (and needed) in the body of _withThreads
.
(As a workaround, I think I can rewrite code(using exec)
into f(exec)
in my library to convince the compiler that f
is being used in the second case
; that seems to make the warning go away.)