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

scribe.format.FormatBlock.scala Maven / Gradle / Ivy

There is a newer version: 3.15.3
Show newest version
package scribe.format

import perfolation._
import scribe._
import scribe.mdc.MDC
import scribe.output.{CompositeOutput, EmptyOutput, LogOutput, TextOutput}
import scribe.util.Abbreviator

import scala.language.implicitConversions

trait FormatBlock {
  def format(record: LogRecord): LogOutput

  def abbreviate(maxLength: Int,
                 padded: Boolean = false,
                 separator: Char = '.',
                 removeEntries: Boolean = true,
                 abbreviateName: Boolean = false): FormatBlock = {
    val block = new AbbreviateBlock(this, maxLength, separator, removeEntries, abbreviateName)
    if (padded) {
      new RightPaddingBlock(block, maxLength, ' ')
    } else {
      block
    }
  }

  def map(f: LogOutput => LogOutput): FormatBlock = FormatBlock.Mapped(this, f)

  def mapPlain(f: String => String): FormatBlock = FormatBlock.MappedPlain(this, f)

  def padRight(length: Int, padding: Char = ' '): FormatBlock = new RightPaddingBlock(this, length, padding)
}

object FormatBlock {
  def apply(f: LogRecord => LogOutput): FormatBlock = new FormatBlock {
    override def format(record: LogRecord): LogOutput = f(record)
  }

  case class Mapped(block: FormatBlock, f: LogOutput => LogOutput) extends FormatBlock {
    override def format(record: LogRecord): LogOutput = f(block.format(record))
  }

  case class MappedPlain(block: FormatBlock, f: String => String) extends FormatBlock {
    override def format(record: LogRecord): LogOutput = block.format(record).map(f)
  }

  case class RawString(s: String) extends FormatBlock {
    override def format(record: LogRecord): LogOutput = new TextOutput(s)
  }

  object TimeStamp extends FormatBlock {
    override def format(record: LogRecord): LogOutput = new TextOutput(record.timeStamp.toString)
  }

  object Time extends FormatBlock {
    override def format(record: LogRecord): LogOutput = new TextOutput(s"${record.timeStamp.t.T}")
  }

  object Date {
    object Standard extends CachingFormatBlock {
      override protected def cacheLength: Long = 1000L

      override protected def formatCached(record: LogRecord): LogOutput = {
        val l = record.timeStamp
        val d = s"${l.t.Y}.${l.t.m}.${l.t.d} ${l.t.T}"
        new TextOutput(d)
      }
    }
    object ISO8601 extends CachingFormatBlock {
      override protected def cacheLength: Long = 1000L

      override protected def formatCached(record: LogRecord): LogOutput = {
        val l = record.timeStamp
        val d = s"${l.t.Y}-${l.t.m}-${l.t.d}T${l.t.T}${l.t.Z}"
        new TextOutput(d)
      }
    }
    object Full extends FormatBlock {
      override def format(record: LogRecord): LogOutput = {
        val l = record.timeStamp
        val d = s"${l.t.Y}.${l.t.m}.${l.t.d} ${l.t.T}:${l.t.L}"
        new TextOutput(d)
      }
    }
    object Counter extends FormatBlock {
      private lazy val cache = new ThreadLocal[String] {
        override def initialValue(): String = ""
      }
      private lazy val lastValue = new ThreadLocal[Long] {
        override def initialValue(): Long = 0L
      }
      private lazy val counter = new ThreadLocal[Long] {
        override def initialValue(): Long = 0L
      }

      override def format(record: LogRecord): LogOutput = {
        val l = record.timeStamp
        if (l - lastValue.get() > 1000L) {
          counter.set(0L)
          val base = s"${l.t.Y}.${l.t.m}.${l.t.d} ${l.t.T}"
          val d = s"$base:0000"
          cache.set(base)
          lastValue.set(l)
          new TextOutput(d)
        } else {
          val c = counter.get() + 1
          counter.set(c)
          val cnt = c match {
            case _ if c >= 1000 => c.toString
            case _ if c >= 100 => s"0$c"
            case _ if c >= 10 => s"00$c"
            case _ => s"000$c"
          }
          new TextOutput(s"${cache.get()}:$cnt")
        }
      }
    }
  }

  object ThreadName extends FormatBlock {
    override def format(record: LogRecord): LogOutput = new TextOutput(record.thread.getName)
  }

  object Level extends FormatBlock {
    override def format(record: LogRecord): LogOutput = new TextOutput(record.level.name)

    object PaddedRight extends FormatBlock {
      override def format(record: LogRecord): LogOutput = new TextOutput(record.level.namePadded)
    }
  }

  object FileName extends FormatBlock {
    override def format(record: LogRecord): LogOutput = new TextOutput(record.fileName)
  }

  object ClassName extends FormatBlock {
    override def format(record: LogRecord): LogOutput = new TextOutput(record.className)
  }

  object ClassNameSimple extends FormatBlock {
    override def format(record: LogRecord): LogOutput = {
      val cn = record.className
      val index = cn.lastIndexOf('.')
      val simple = if (index != -1) {
        cn.substring(index + 1)
      } else {
        cn
      }
      new TextOutput(simple)
    }
  }

  object MethodName extends FormatBlock {
    override def format(record: LogRecord): LogOutput = new TextOutput(record.methodName.getOrElse(""))
  }

  object ClassAndMethodName extends FormatBlock {
    override def format(record: LogRecord): LogOutput = {
      val className = ClassName.format(record).plainText
      val methodName = if (record.methodName.nonEmpty) {
        s".${MethodName.format(record).plainText}"
      } else {
        ""
      }
      new TextOutput(s"$className$methodName")
    }
  }

  object ClassAndMethodNameSimple extends FormatBlock {
    override def format(record: LogRecord): LogOutput = {
      val className = ClassNameSimple.format(record).plainText
      val methodName = if (record.methodName.nonEmpty) {
        s".${MethodName.format(record).plainText}"
      } else {
        ""
      }
      new TextOutput(s"$className$methodName")
    }
  }

  object Position extends FormatBlock {
    override def format(record: LogRecord): LogOutput = {
      val line = if (record.line.nonEmpty) {
        s":${LineNumber.format(record).plainText}"
      } else {
        ""
      }
      val column = if (record.column.nonEmpty) {
        s":${ColumnNumber.format(record).plainText}"
      } else {
        ""
      }
      new TextOutput(s"${ClassAndMethodName.format(record).plainText}$line$column")
    }

    override def abbreviate(maxLength: Int,
                            padded: Boolean = false,
                            separator: Char = '.',
                            removeEntries: Boolean = false,
                            abbreviateName: Boolean = false): FormatBlock = apply { record =>
      val line = if (record.line.nonEmpty) {
        s":${LineNumber.format(record).plainText}"
      } else {
        ""
      }
      val column = if (record.column.nonEmpty) {
        s":${ColumnNumber.format(record).plainText}"
      } else {
        ""
      }
      val className = ClassName.format(record).plainText
      val methodName = MethodName.format(record).plainText
      val cna = Abbreviator(className, maxLength - line.length, separator, removeEntries, abbreviateName)
      new TextOutput(s"$cna.$methodName$line$column")
    }
  }

  object PositionSimple extends FormatBlock {
    override def format(record: LogRecord): LogOutput = {
      val line = if (record.line.nonEmpty) {
        s":${LineNumber.format(record).plainText}"
      } else {
        ""
      }
      val column = if (record.column.nonEmpty) {
        s":${ColumnNumber.format(record).plainText}"
      } else {
        ""
      }
      new TextOutput(s"${ClassAndMethodNameSimple.format(record).plainText}$line$column")
    }
  }

  object LineNumber extends FormatBlock {
    override def format(record: LogRecord): LogOutput = new TextOutput(record.line.fold("")(_.toString))
  }

  object ColumnNumber extends FormatBlock {
    override def format(record: LogRecord): LogOutput = new TextOutput(record.column.fold("")(_.toString))
  }

  object Messages extends FormatBlock {
    override def format(record: LogRecord): LogOutput = record.logOutput
  }

  case class MDCReference(key: String,
                          default: () => Any,
                          prefix: Option[FormatBlock],
                          postfix: Option[FormatBlock]) extends FormatBlock {
    override def format(record: LogRecord): LogOutput = MDC.get(key) match {
      case Some(value) => {
        val text = new TextOutput(value.toString)
        if (prefix.nonEmpty || postfix.nonEmpty) {
          new CompositeOutput(List(prefix.map(_.format(record)), Option(text), postfix.map(_.format(record))).flatten)
        } else {
          text
        }
      }
      case None => new TextOutput(default().toString)
    }
  }

  object MDCAll extends FormatBlock {
    override def format(record: LogRecord): LogOutput = {
      val map = MDC.map ++ record.data
      if (map.nonEmpty) {
        new TextOutput(map.map {
          case (key, value) => s"$key: ${value()}"
        }.mkString(" (", ", ", ")"))
      } else {
        EmptyOutput
      }
    }
  }

  object MDCMultiLine extends FormatBlock {
    override def format(record: LogRecord): LogOutput = {
      val map = MDC.map ++ record.data
      if (map.nonEmpty) {
        val nl = newLine.format(record)
        val prefix = green(bold(string("["))).format(record)
        val postfix = green(bold(string("]"))).format(record)
        val entries = nl :: prefix :: map.toList.flatMap {
          case (key, value) => List(
            new TextOutput(", "),
            brightWhite(string(s"$key: ")).format(record),
            new TextOutput(String.valueOf(value()))
          )
        }.tail ::: List(postfix)
        new CompositeOutput(entries)
      } else {
        EmptyOutput
      }
    }
  }

  object NewLine extends FormatBlock {
    override def format(record: LogRecord): LogOutput = new TextOutput(System.lineSeparator)
  }

  case class MultiLine(maxChars: () => Int = MultiLine.PlatformColumns, prefix: String = "    ", blocks: List[FormatBlock]) extends FormatBlock {
    override def format(record: LogRecord): LogOutput = {
      val pre = new TextOutput(prefix)
      val max = maxChars() - prefix.length
      val newLine = NewLine.format(record)
      val outputs = MultiLine.splitNewLines(blocks.map(_.format(record)))
      val list = outputs.flatMap { output =>
        var current = output
        var list = List.empty[LogOutput]
        while (current.length > max) {
          val (left, right) = current.splitAt(max)
          list = list ::: List(pre, left, newLine)
          current = right
        }
        list = list ::: List(pre, current)
        list
      }
      new CompositeOutput(list)
    }
  }

  object MultiLine {
    val DefaultMaxChars: Int = 120
    val PlatformColumns: () => Int = () => scribe.Platform.columns

    def splitNewLines(outputs: List[LogOutput]): List[LogOutput] = outputs.flatMap { output =>
      var lo = output
      var plainText = output.plainText
      var splitted = List.empty[LogOutput]
      def process(): Unit = {
        val index = plainText.indexOf('\n')
        if (index == -1) {
          splitted = lo :: splitted
          // Finished
        } else {
          val (one, two) = lo.splitAt(index + 1)
          splitted = one :: splitted
          lo = two
          plainText = plainText.substring(index + 1)
          process()
        }
      }
      process()
      splitted.reverse
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy