R/logging.R

#' @title Simple logging utility
#'
#' @description
#' Creates Logger object which can be used for logging with different level of
#' verbosity. Log messages are in JSON format.
#'
#' @export
#'
#' @seealso [lgr::Logger]
#'
#' @examples
#' # init logger
#' logger = Logger$new("info")
#' # write info message
#' logger$info("hello world")
#' # write extended log entry
#' logger$info("", context = list(message = "hello world", code = 0L))
#'
Logger = R6::R6Class(
  classname = "Logger",
  public = list(
    #' @description
    #' Creates Logger object.
    #' @param level Log level. Allowed values: info, fatal, error, warn, debug,
    #'   trace, off, all.
    #' @param name Logger name.
    #' @param printer Logger with sink defined by `printer` function.
    #'   It should have signature `function(timestamp, level, logger_name, pid, message)`.
    #'   By default when `printer = NULL` logger writes message in JSON format to `stdout`.
    initialize = function(
      level = c("info", "fatal", "error", "warn", "debug", "trace", "off", "all"),
      name = "ROOT", printer = NULL) {
      self$set_log_level(level)
      self$set_name(name)
      self$set_printer(printer)
    },
    #' @description
    #' Sets logger name.
    #' @param name Logger name.
    set_name = function(name = "ROOT") {
      private$name = as.character(x = name)
      invisible(self)
    },
    #' @description
    #' Sets log level.
    #' @param level Log level. Allowed values: info, fatal, error, warn, debug,
    #'   trace, off, all.
    set_log_level = function(level = c("info", "fatal", "error", "warn", "debug", "trace", "off", "all")) {
      level = match.arg(level)
      level = logging_constants[[level]]
      private$level = level
      invisible(self)
    },
    #' @description
    #' Sets function which defines how to print logs.
    #' @param FUN Printer function. Should be a function with 6 formal arguments:
    #'   timestamp, level, logger_name, pid, message.
    set_printer = function(FUN = NULL) {
      if (is.null(FUN)) {
        FUN = function(timestamp, level, logger_name, pid, message, ...) {
          log_msg = list(
            timestamp = format(timestamp, "%Y-%m-%d %H:%M:%OS6"),
            level = as.character(level),
            name = as.character(logger_name),
            pid = as.integer(pid),
            msg = message
          )
          extra = list(...)
          if (length(extra) > 0) {
            log_msg = c(log_msg, extra)
          }
          x = to_json(log_msg)
          writeLines(x)
          flush(stdout())
        }
      }
      if (!is.function(FUN))
        stop("'FUN' should function or NULL")
      if (length(formals(FUN)) != 6L)
        stop("FUN should be a function with 6 formal arguments - (timestamp, level, logger_name, pid, message, ...)")
      private$printer = FUN
      return(invisible(self))
    },
    #' @description
    #' Write trace message.
    #' @param msg Log message.
    #' @param ... Additionals params.
    trace = function(msg, ...) {
      private$log_base(msg, ..., log_level = logging_constants$trace, log_level_tag = "TRACE")
    },
    #' @description
    #' Write debug message.
    #' @param msg Log message.
    #' @param ... Additionals params.
    debug = function(msg, ...) {
      private$log_base(msg, ..., log_level = logging_constants$debug, log_level_tag = "DEBUG")
    },
    #' @description
    #' Write information message.
    #' @param msg Log message.
    #' @param ... Additionals params.
    info = function(msg, ...) {
      private$log_base(msg, ..., log_level = logging_constants$info, log_level_tag = "INFO")
    },
    #' @description
    #' Write warning message.
    #' @param msg Log message.
    #' @param ... Additionals params.
    warn = function(msg, ...) {
      private$log_base(msg, ..., log_level = logging_constants$warn, log_level_tag = "WARN")
    },
    #' @description
    #' Write error message.
    #' @param msg Log message.
    #' @param ... Additionals params.
    error = function(msg, ...) {
      private$log_base(msg, ..., log_level = logging_constants$error, log_level_tag = "ERROR")
    },
    #' @description
    #' Write fatal error message.
    #' @param msg Log message.
    #' @param ... Additionals params.
    fatal = function(msg, ...) {
      private$log_base(msg, ..., log_level = logging_constants$fatal, log_level_tag = "FATAL")
    }
  ),
  private = list(
    printer = NULL,
    level = NULL,
    name = NULL,
    log_base = function(msg, ..., log_level, log_level_tag) {
      if (isTRUE(private$level >= log_level) || is.na(private$level)) {
        private$printer(Sys.time(), log_level_tag, private$name, Sys.getpid(), msg, ...)
      }
      invisible(msg)
    }
  )
)

logging_constants = list(
  "fatal" = 100,
  "error" = 200,
  "warn" = 300,
  "info" = 400,
  "debug" = 500,
  "trace" = 600,
  "off" = 0,
  "all" = NA_real_
)
rexyai/RestRserve documentation built on April 22, 2024, 10:29 p.m.