scribe.format.FormatBlock.scala Maven / Gradle / Ivy
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