All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.permutive.logging.slf4j.odin.GlobalLogger.scala Maven / Gradle / Ivy

/*
 * Copyright 2022-2024 Permutive Ltd. 
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.permutive.logging.slf4j.odin

import java.util.concurrent.atomic.AtomicReference

import scala.util.control.NonFatal

import cats.effect.kernel.Async
import cats.effect.kernel.Resource
import cats.effect.kernel.Sync
import cats.effect.std.Dispatcher
import cats.effect.syntax.all._
import cats.syntax.applicativeError._
import cats.syntax.flatMap._

import io.odin.Level
import io.odin.Logger
import io.odin.formatter.Formatter

/** Utilities for getting a logger that can be used by SLF4J. */
@SuppressWarnings(Array("scalafix:DisableSyntax.null", "scalafix:DisableSyntax.==", "scalafix:DisableSyntax.!="))
object GlobalLogger {

  private val default: OdinTranslator =
    OdinTranslator.unsafeConsole(level = Level.Info, Formatter.default)

  private val ref = new AtomicReference[OdinTranslator]

  // the logger may not yet be allocated, so do a null check
  // all in try/catch for efficiency
  private[odin] def get: OdinTranslator =
    try {
      val log = ref.get()
      if (log == null) default else log
    } catch {
      case NonFatal(th) =>
        // Use default logger if there is an error getting the logger from the atomic ref
        default.run(
          this.getClass.getName,
          Level.Error,
          "Failed to get logger from atomic ref",
          Some(th)
        )
        default
    }

  def isSet[F[_]: Sync]: F[Boolean] = Sync[F].delay(ref.get() != null)

  // sets the logger in the atomic ref
  // sets the ref to null when the resource is de-allocated in order to avoid resource leak of the underlying logger
  def setLogger[F[_]: Async](logger: Logger[F]): Resource[F, Unit] = {
    val set = Dispatcher
      .sequential[F]
      .map(implicit dis => OdinTranslator[F](logger))
      .flatMap(log => Resource.make(Sync[F].delay(ref.set(log)))(_ => Sync[F].delay(ref.set(null))))

    val error: Resource[F, Unit] = new RuntimeException(
      "Logger already set: Cannot set the global logger multiple times"
    ).raiseError[F, Unit].toResource

    Resource.eval(Sync[F].delay(ref.get())).map(_ == null).ifM(set, error)
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy