Skip to content

Commit 7c1a32d

Browse files
committed
https://github.com/lightbend-labs/scala-logging/issues/354
After wrapping varargs, the user code fails to compile. In Scala 2, there were no inline parameters, and the subtype of args was obtained during compilation. However, this approach may not always be accurate. Regarding #191 There is an issue with obtaining inline parameters in Scala 3. To address this, we recursively obtain the actual value of inline parameters. This necessitates the ongoing use of inlining in the parameters of the wrapper function. For instance: ```scala class LogWrapper(val underlying: Logger) { inline def info(message: String, inline args: AnyRef*): Unit = underlying.info(message, args: _*) } ``` This ensures compatibility and accurate handling of inline parameters in both Scala 2 and Scala 3.
1 parent da10c6c commit 7c1a32d

File tree

5 files changed

+115
-11
lines changed

5 files changed

+115
-11
lines changed

src/main/scala-2/com/typesafe/scalalogging/LoggerMacro.scala

+9-1
Original file line numberDiff line numberDiff line change
@@ -295,7 +295,15 @@ private[scalalogging] object LoggerMacro {
295295
private def formatArgs(c: LoggerContext)(args: c.Expr[Any]*) = {
296296
import c.universe._
297297
args.map { arg =>
298-
c.Expr[AnyRef](if (arg.tree.tpe <:< weakTypeOf[AnyRef]) q"$arg: _root_.scala.AnyRef" else q"$arg.asInstanceOf[_root_.scala.AnyRef]")
298+
// If arg is a varargs, it is also a AnyRef and we need to check the subtree.
299+
val argument = arg.tree.children.map(_.tpe) match {
300+
case head :: next :: Nil if head <:< weakTypeOf[scala.collection.Seq[_]] && next <:< weakTypeOf[AnyRef] =>
301+
q"$arg"
302+
case _ =>
303+
if (arg.tree.tpe <:< weakTypeOf[AnyRef]) q"$arg: _root_.scala.AnyRef"
304+
else q"$arg.asInstanceOf[_root_.scala.AnyRef]"
305+
}
306+
c.Expr[AnyRef](argument)
299307
}
300308
}
301309
}

src/main/scala-3/com/typesafe/scalalogging/LoggerMacro.scala

+9-10
Original file line numberDiff line numberDiff line change
@@ -264,19 +264,18 @@ private[scalalogging] object LoggerMacro {
264264
def formatArgs(args: Expr[Seq[Any]])(using q: Quotes): Seq[Expr[AnyRef]] = {
265265
import quotes.reflect.*
266266
import util.*
267-
268-
args.asTerm match {
269-
case p@Inlined(_, _, Typed(Repeated(v, _),_)) =>
270-
v.map{
271-
case t if t.tpe <:< TypeRepr.of[AnyRef] => t.asExprOf[AnyRef]
272-
case t => '{${t.asExpr}.asInstanceOf[AnyRef]}
273-
}
274-
case Repeated(v, _) =>
275-
v.map{
267+
// we recursively obtain the actual value of inline parameters
268+
def rec(tree: Term): Option[Seq[Expr[AnyRef]]] = tree match {
269+
case Repeated(elems, _) => Some(
270+
elems.map {
276271
case t if t.tpe <:< TypeRepr.of[AnyRef] => t.asExprOf[AnyRef]
277272
case t => '{${t.asExpr}.asInstanceOf[AnyRef]}
278273
}
279-
case _ => Seq.empty
274+
)
275+
case Typed(e, _) => rec(e)
276+
case Inlined(_, Nil, e) => rec(e)
277+
case _ => None
280278
}
279+
rec(args.asTerm).getOrElse(Seq.empty)
281280
}
282281
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
package com.typesafe.scalalogging
2+
3+
import org.mockito.Mockito._
4+
import org.scalatest.matchers.should.Matchers
5+
import org.scalatest.wordspec.AnyWordSpec
6+
import org.scalatestplus.mockito.MockitoSugar
7+
import org.slf4j.{ Logger => Underlying }
8+
9+
class Scala2LoggerSpec extends AnyWordSpec with Matchers with Varargs with MockitoSugar {
10+
11+
// Info
12+
"Calling info with an interpolated message, only scala 2" should {
13+
"fix Varargs compilation error issue 354 only scala 2" in {
14+
val f = fixture(_.isInfoEnabled, isEnabled = true)
15+
import f._
16+
class LogWrapper(val underlying: Logger) {
17+
def info(message: String, args: AnyRef*): Unit = underlying.info(message, args: _*)
18+
}
19+
val logWrapper = new LogWrapper(logger)
20+
logWrapper.info("""Hello {}""", arg5ref)
21+
verify(underlying).info("""Hello {}""", forceVarargs(arg5ref): _*)
22+
}
23+
}
24+
25+
private def fixture(p: Underlying => Boolean, isEnabled: Boolean) = new LoggerF(p, isEnabled)
26+
27+
private class LoggerF(p: Underlying => Boolean, isEnabled: Boolean) {
28+
val msg = "msg"
29+
val cause = new RuntimeException("cause")
30+
val arg1 = "arg1"
31+
val arg2: Integer = Integer.valueOf(1)
32+
val arg3 = "arg3"
33+
val arg4 = 4
34+
val arg4ref: AnyRef = arg4.asInstanceOf[AnyRef]
35+
val arg5 = true
36+
val arg5ref: AnyRef = arg5.asInstanceOf[AnyRef]
37+
val arg6 = 6L
38+
val arg6ref: AnyRef = arg6.asInstanceOf[AnyRef]
39+
val arg7 = new Throwable
40+
val arg7ref: AnyRef = arg7.asInstanceOf[AnyRef]
41+
val underlying: Underlying = mock[org.slf4j.Logger]
42+
when(p(underlying)).thenReturn(isEnabled)
43+
val logger: Logger = Logger(underlying)
44+
}
45+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
package com.typesafe.scalalogging
2+
3+
import org.mockito.Mockito._
4+
import org.scalatest.matchers.should.Matchers
5+
import org.scalatest.wordspec.AnyWordSpec
6+
import org.scalatestplus.mockito.MockitoSugar
7+
import org.slf4j.{Logger => Underlying}
8+
9+
class Scala3LoggerSpec extends AnyWordSpec with Matchers with Varargs with MockitoSugar {
10+
11+
// Info
12+
"Calling info with an interpolated message, only scala 3" should {
13+
"fix Varargs compilation error issue 354 only scala 3" in {
14+
val f = fixture(_.isInfoEnabled, isEnabled = true)
15+
import f._
16+
class LogWrapper(val underlying: Logger) {
17+
inline def info(message: String, inline args: AnyRef*): Unit = underlying.info(message, args: _*)
18+
}
19+
val logWrapper = new LogWrapper(logger)
20+
logWrapper.info("""Hello {}""", arg5ref)
21+
verify(underlying).info("""Hello {}""", arg5ref)
22+
}
23+
}
24+
25+
private def fixture(p: Underlying => Boolean, isEnabled: Boolean) = new LoggerF(p, isEnabled)
26+
private class LoggerF(p: Underlying => Boolean, isEnabled: Boolean) {
27+
val msg = "msg"
28+
val cause = new RuntimeException("cause")
29+
val arg1 = "arg1"
30+
val arg2: Integer = Integer.valueOf(1)
31+
val arg3 = "arg3"
32+
val arg4 = 4
33+
val arg4ref: AnyRef = arg4.asInstanceOf[AnyRef]
34+
val arg5 = true
35+
val arg5ref: AnyRef = arg5.asInstanceOf[AnyRef]
36+
val arg6 = 6L
37+
val arg6ref: AnyRef = arg6.asInstanceOf[AnyRef]
38+
val arg7 = new Throwable
39+
val arg7ref: AnyRef = arg7.asInstanceOf[AnyRef]
40+
val underlying: Underlying = mock[org.slf4j.Logger]
41+
when(p(underlying)).thenReturn(isEnabled)
42+
val logger: Logger = Logger(underlying)
43+
}
44+
}

src/test/scala/com/typesafe/scalalogging/LoggerSpec.scala

+8
Original file line numberDiff line numberDiff line change
@@ -207,6 +207,14 @@ class LoggerSpec extends AnyWordSpec with Matchers with Varargs with MockitoSuga
207207
logger.info(s"msg $arg1 $arg2")
208208
verify(underlying).info("msg {} {}", forceVarargs(arg1, arg2): _*)
209209
}
210+
211+
"fix Varargs compilation error, issue 191" in {
212+
val f = fixture(_.isInfoEnabled, isEnabled = true)
213+
import f._
214+
val message = "Hello"
215+
logger.info(s"Message with id=$message, submittedAt=$message will be dropped.")
216+
verify(underlying).info("""Message with id={}, submittedAt={} will be dropped.""", forceVarargs(message, message): _*)
217+
}
210218
}
211219

212220
"Calling info with a message and cause" should {

0 commit comments

Comments
 (0)