Skip to content

Commit

Permalink
pass MDC as lazy eval param (#236)
Browse files Browse the repository at this point in the history
Co-authored-by: Denys Fakhritdinov <dfakhritdinov@evolution.com>
  • Loading branch information
dfakhritdinov and Denys Fakhritdinov authored Aug 22, 2023
1 parent cb625d8 commit 044b071
Show file tree
Hide file tree
Showing 6 changed files with 90 additions and 70 deletions.
12 changes: 11 additions & 1 deletion build.sbt
Original file line number Diff line number Diff line change
Expand Up @@ -83,7 +83,17 @@ lazy val logback = project
libraryDependencies ++= Seq(
Logback.classic,
scalatest % Test,
)
),
libraryDependencies ++= crossSettings(
scalaVersion.value,
if3 = Nil,
if2 = List(compilerPlugin("org.typelevel" % "kind-projector" % "0.13.2" cross CrossVersion.full))
),
scalacOptions ++= crossSettings(
scalaVersion.value,
if3 = Seq("-Ykind-projector:underscores", "-language:implicitConversions"),
if2 = List("-Xsource:3", "-P:kind-projector:underscore-placeholders")
),
)
.dependsOn(
core,
Expand Down
71 changes: 36 additions & 35 deletions core/src/main/scala/com/evolutiongaming/catshelper/Log.scala
Original file line number Diff line number Diff line change
Expand Up @@ -23,19 +23,19 @@ trait Log[F[_]] {

@inline def error(msg: => String, cause: Throwable): F[Unit] = error(msg, cause, mdc = Log.Mdc.empty)

def trace(msg: => String, mdc: Log.Mdc): F[Unit]
def trace(msg: => String, mdc: => Log.Mdc): F[Unit]

def debug(msg: => String, mdc: Log.Mdc): F[Unit]
def debug(msg: => String, mdc: => Log.Mdc): F[Unit]

def info(msg: => String, mdc: Log.Mdc): F[Unit]
def info(msg: => String, mdc: => Log.Mdc): F[Unit]

def warn(msg: => String, mdc: Log.Mdc): F[Unit]
def warn(msg: => String, mdc: => Log.Mdc): F[Unit]

def warn(msg: => String, cause: Throwable, mdc: Log.Mdc): F[Unit]
def warn(msg: => String, cause: Throwable, mdc: => Log.Mdc): F[Unit]

def error(msg: => String, mdc: Log.Mdc): F[Unit]
def error(msg: => String, mdc: => Log.Mdc): F[Unit]

def error(msg: => String, cause: Throwable, mdc: Log.Mdc): F[Unit]
def error(msg: => String, cause: Throwable, mdc: => Log.Mdc): F[Unit]
}

object Log {
Expand Down Expand Up @@ -91,43 +91,44 @@ object Log {
}
}

def trace(msg: => String, mdc: Log.Mdc) = {

def trace(msg: => String, mdc: => Log.Mdc) = {
Sync[F].delay {
if (logger.isTraceEnabled) withMDC(mdc) { logger.trace(msg) }
}
}

def debug(msg: => String, mdc: Log.Mdc) = {
def debug(msg: => String, mdc: => Log.Mdc) = {
Sync[F].delay {
if (logger.isDebugEnabled) withMDC(mdc) { logger.debug(msg) }
}
}

def info(msg: => String, mdc: Log.Mdc) = {
def info(msg: => String, mdc: => Log.Mdc) = {
Sync[F].delay {
if (logger.isInfoEnabled) withMDC(mdc) { logger.info(msg) }
}
}

def warn(msg: => String, mdc: Log.Mdc) = {
def warn(msg: => String, mdc: => Log.Mdc) = {
Sync[F].delay {
if (logger.isWarnEnabled) withMDC(mdc) { logger.warn(msg) }
}
}

def warn(msg: => String, cause: Throwable, mdc: Log.Mdc) = {
def warn(msg: => String, cause: Throwable, mdc: => Log.Mdc) = {
Sync[F].delay {
if (logger.isWarnEnabled) withMDC(mdc) { logger.warn(msg, cause) }
}
}

def error(msg: => String, mdc: Log.Mdc) = {
def error(msg: => String, mdc: => Log.Mdc) = {
Sync[F].delay {
if (logger.isErrorEnabled) withMDC(mdc) { logger.error(msg) }
}
}

def error(msg: => String, cause: Throwable, mdc: Log.Mdc) = {
def error(msg: => String, cause: Throwable, mdc: => Log.Mdc) = {
Sync[F].delay {
if (logger.isErrorEnabled) withMDC(mdc) { logger.error(msg, cause) }
}
Expand All @@ -136,19 +137,19 @@ object Log {

def const[F[_]](unit: F[Unit]): Log[F] = new Log[F] {

def trace(msg: => String, mdc: Log.Mdc) = unit
def trace(msg: => String, mdc: => Log.Mdc) = unit

def debug(msg: => String, mdc: Log.Mdc) = unit
def debug(msg: => String, mdc: => Log.Mdc) = unit

def info(msg: => String, mdc: Log.Mdc) = unit
def info(msg: => String, mdc: => Log.Mdc) = unit

def warn(msg: => String, mdc: Log.Mdc) = unit
def warn(msg: => String, mdc: => Log.Mdc) = unit

def warn(msg: => String, cause: Throwable, mdc: Log.Mdc) = unit
def warn(msg: => String, cause: Throwable, mdc: => Log.Mdc) = unit

def error(msg: => String, mdc: Log.Mdc) = unit
def error(msg: => String, mdc: => Log.Mdc) = unit

def error(msg: => String, cause: Throwable, mdc: Log.Mdc) = unit
def error(msg: => String, cause: Throwable, mdc: => Log.Mdc) = unit
}

def empty[F[_]: Applicative]: Log[F] = const(Applicative[F].unit)
Expand All @@ -157,36 +158,36 @@ object Log {

def mapK[G[_]](f: F ~> G): Log[G] = new Log[G] {

def trace(msg: => String, mdc: Log.Mdc) = f(self.trace(msg, mdc))
def trace(msg: => String, mdc: => Log.Mdc) = f(self.trace(msg, mdc))

def debug(msg: => String, mdc: Log.Mdc) = f(self.debug(msg, mdc))
def debug(msg: => String, mdc: => Log.Mdc) = f(self.debug(msg, mdc))

def info(msg: => String, mdc: Log.Mdc) = f(self.info(msg, mdc))
def info(msg: => String, mdc: => Log.Mdc) = f(self.info(msg, mdc))

def warn(msg: => String, mdc: Log.Mdc) = f(self.warn(msg, mdc))
def warn(msg: => String, mdc: => Log.Mdc) = f(self.warn(msg, mdc))

def warn(msg: => String, cause: Throwable, mdc: Log.Mdc) = f(self.warn(msg, cause, mdc))
def warn(msg: => String, cause: Throwable, mdc: => Log.Mdc) = f(self.warn(msg, cause, mdc))

def error(msg: => String, mdc: Log.Mdc) = f(self.error(msg, mdc))
def error(msg: => String, mdc: => Log.Mdc) = f(self.error(msg, mdc))

def error(msg: => String, cause: Throwable, mdc: Log.Mdc) = f(self.error(msg, cause, mdc))
def error(msg: => String, cause: Throwable, mdc: => Log.Mdc) = f(self.error(msg, cause, mdc))
}

def mapMsg(f: String => String): Log[F] = new Log[F] {

def trace(msg: => String, mdc: Log.Mdc) = self.trace(f(msg), mdc)
def trace(msg: => String, mdc: => Log.Mdc) = self.trace(f(msg), mdc)

def debug(msg: => String, mdc: Log.Mdc) = self.debug(f(msg), mdc)
def debug(msg: => String, mdc: => Log.Mdc) = self.debug(f(msg), mdc)

def info(msg: => String, mdc: Log.Mdc) = self.info(f(msg), mdc)
def info(msg: => String, mdc: => Log.Mdc) = self.info(f(msg), mdc)

def warn(msg: => String, mdc: Log.Mdc) = self.warn(f(msg), mdc)
def warn(msg: => String, mdc: => Log.Mdc) = self.warn(f(msg), mdc)

def warn(msg: => String, cause: Throwable, mdc: Log.Mdc) = self.warn(f(msg), cause, mdc)
def warn(msg: => String, cause: Throwable, mdc: => Log.Mdc) = self.warn(f(msg), cause, mdc)

def error(msg: => String, mdc: Log.Mdc) = self.error(f(msg), mdc)
def error(msg: => String, mdc: => Log.Mdc) = self.error(f(msg), mdc)

def error(msg: => String, cause: Throwable, mdc: Log.Mdc) = self.error(f(msg), cause, mdc)
def error(msg: => String, cause: Throwable, mdc: => Log.Mdc) = self.error(f(msg), cause, mdc)
}

def prefixed(prefix: String): Log[F] = mapMsg(msg => s"$prefix $msg")
Expand Down
44 changes: 33 additions & 11 deletions core/src/test/scala/com/evolutiongaming/catshelper/LogSpec.scala
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,9 @@ import cats.effect.IO
import scala.util.control.NoStackTrace
import org.scalatest.funsuite.AnyFunSuite
import org.scalatest.matchers.should.Matchers
import com.evolutiongaming.catshelper.IOSuite._
import com.evolutiongaming.catshelper.IOSuite.*

import scala.concurrent.duration.*

class LogSpec extends AnyFunSuite with Matchers {

Expand Down Expand Up @@ -69,15 +71,35 @@ class LogSpec extends AnyFunSuite with Matchers {
Action.OfStr("source")))
}

test("MDC cleanup") {
test("MDC ThreadLocal cleanup") {

val io = for {
logOf <- LogOf.slf4j[IO]
log <- logOf(getClass)
_ <- log.info("whatever", Log.Mdc("k" -> "v"))
} yield org.slf4j.MDC.getCopyOfContextMap
} yield {
val context = org.slf4j.MDC.getCopyOfContextMap
context shouldEqual null
}

io.unsafeRunSync()
}

test("MDC lazy evaluation") {
var used = false
def mdc = {
used = true
Log.Mdc.empty
}

val io = for {
log <- LogOf.slf4j[IO]
log <- log("test-lazy-mdc")
_ <- IO.sleep(1.second) // await for logger initialisation: https://www.slf4j.org/codes.html#replay
_ <- log.trace("should not appear in (debug) log", mdc) // trace is disabled in logback-text.xml
} yield used shouldEqual false

io.unsafeRunSync() shouldEqual null
io.unsafeRunSync()
}

test("LogOf.log") {
Expand Down Expand Up @@ -117,49 +139,49 @@ object LogSpec {
val log: Log[StateT] = {
val log = new Log[StateT] {

def trace(msg: => String, mdc: Log.Mdc) = {
def trace(msg: => String, mdc: => Log.Mdc) = {
StateT { state =>
val action = Action.Trace(msg, mdc)
(state.add(action), ())
}
}

def debug(msg: => String, mdc: Log.Mdc) = {
def debug(msg: => String, mdc: => Log.Mdc) = {
StateT { state =>
val action = Action.Debug(msg, mdc)
(state.add(action), ())
}
}

def info(msg: => String, mdc: Log.Mdc) = {
def info(msg: => String, mdc: => Log.Mdc) = {
StateT { state =>
val action = Action.Info(msg, mdc)
(state.add(action), ())
}
}

def warn(msg: => String, mdc: Log.Mdc) = {
def warn(msg: => String, mdc: => Log.Mdc) = {
StateT { state =>
val action = Action.Warn0(msg, mdc)
(state.add(action), ())
}
}

def warn(msg: => String, cause: Throwable, mdc: Log.Mdc) = {
def warn(msg: => String, cause: Throwable, mdc: => Log.Mdc) = {
StateT { state =>
val action = Action.Warn1(msg, cause, mdc)
(state.add(action), ())
}
}

def error(msg: => String, mdc: Log.Mdc) = {
def error(msg: => String, mdc: => Log.Mdc) = {
StateT { state =>
val action = Action.Error0(msg, mdc)
(state.add(action), ())
}
}

def error(msg: => String, cause: Throwable, mdc: Log.Mdc) = {
def error(msg: => String, cause: Throwable, mdc: => Log.Mdc) = {
StateT { state =>
val action = Action.Error1(msg, cause, mdc)
(state.add(action), ())
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import ch.qos.logback.classic.spi.LoggingEvent
import ch.qos.logback.classic.util.ContextInitializer
import com.evolutiongaming.catshelper.Log.Mdc

import scala.collection.JavaConverters._
import scala.collection.JavaConverters.*

/**
* ===Motivation===
Expand All @@ -31,7 +31,7 @@ object LogOfFromLogback {
log(context.getLogger(source))
}

def apply(source: Class[_]): F[Log[F]] =
def apply(source: Class[?]): F[Log[F]] =
apply(source.getName.stripSuffix("$"))
}
}
Expand All @@ -42,7 +42,7 @@ object LogOfFromLogback {
val FQCN = getClass.getName

def append(msg: => String,
mdc: Mdc,
mdc: => Mdc,
level: Level,
throwable: Throwable = null): F[Unit] = Sync[F].delay {
if (logger.isEnabledFor(level)) {
Expand All @@ -55,25 +55,25 @@ object LogOfFromLogback {
}
}

def trace(msg: => String, mdc: Mdc): F[Unit] =
def trace(msg: => String, mdc: => Mdc): F[Unit] =
append(msg, mdc, Level.TRACE)

def debug(msg: => String, mdc: Mdc): F[Unit] =
def debug(msg: => String, mdc: => Mdc): F[Unit] =
append(msg, mdc, Level.DEBUG)

def info(msg: => String, mdc: Mdc): F[Unit] =
def info(msg: => String, mdc: => Mdc): F[Unit] =
append(msg, mdc, Level.INFO)

def warn(msg: => String, mdc: Mdc): F[Unit] =
def warn(msg: => String, mdc: => Mdc): F[Unit] =
append(msg, mdc, Level.WARN)

def warn(msg: => String, cause: Throwable, mdc: Mdc): F[Unit] =
def warn(msg: => String, cause: Throwable, mdc: => Mdc): F[Unit] =
append(msg, mdc, Level.WARN, cause)

def error(msg: => String, mdc: Mdc): F[Unit] =
def error(msg: => String, mdc: => Mdc): F[Unit] =
append(msg, mdc, Level.ERROR)

def error(msg: => String, cause: Throwable, mdc: Mdc): F[Unit] =
def error(msg: => String, cause: Throwable, mdc: => Mdc): F[Unit] =
append(msg, mdc, Level.ERROR, cause)
}

Expand Down
File renamed without changes.
13 changes: 0 additions & 13 deletions logback/src/test/resources/logback.xml

This file was deleted.

0 comments on commit 044b071

Please sign in to comment.