/pl4j

Pretty Logger for Java

Primary LanguageJavaApache License 2.0Apache-2.0

Pretty Logger for Java (PL4J)

(Now with themes support!)

Loggers Loggers Loggers

Description

Pretty Logger for Java is a slf4j decorator that enables pretty printing using ANSI formatting through jansi. PL4J can be used either through Markers or simply prefixing the log messages.

Table of Contents

Usage

Add it as a Maven dependency in your pom.xml file:

<dependency>
    <groupId>io.github.ludovicianul</groupId>
    <artifactId>pretty-logger</artifactId>
    <version>LATEST</version>
</dependency>
 public class TestClass {

    public static void main(String... args) {
        PrettyLogger prettyLogger = PrettyLoggerFactory.getLogger(TestClass.class); //same declaration as SLF4J

        prettyLogger.success("received response from: {}", "http://google.com");
        prettyLogger.awaiting("parsing input data");
        prettyLogger.complete("finish processing");
        prettyLogger.debug("value is: {}", "190");
        prettyLogger.error("not able to connect to: {}", "http://google.com");
        prettyLogger.fatal("something went terribly wrong");
        prettyLogger.info("url to connect to: {}", "http://google.com");
        prettyLogger.note("remember to run CATS");
        prettyLogger.pause("process was paused");
        prettyLogger.santa("ho! ho! ho!");
        prettyLogger.star("run CATS next time");
        prettyLogger.start("process started");
        prettyLogger.stop("process paused");
        prettyLogger.warning("unable to normalize string");
    }
}

Configuration

Marker vs Prefix

PL4J can be used either through Markers or as simple prefix for the messages. Depending on your logging implementation you can choose one or the other. This can be configured through the pl4j.use-markers property. Default value is true.

The way you set this property will influence how the log patterns should be defined:

  • when using Markers, make sure the log pattern also includes the %marker
  • when not using Markers, it's enough to just include the %msg

Log Pattern

This is a sample logback configuration file that was used to display the above console output when using Markers (i.e. pl4j.use-markers = true):

<?xml version="1.0" encoding="UTF-8"?>
<configuration debug="false">
    <statusListener class="ch.qos.logback.core.status.NopStatusListener"/>
    <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
        <withJansi>true</withJansi>
        <encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder">
            <pattern>%-27marker %msg%n</pattern>
        </encoder>
    </appender>
    <root level="trace">
        <appender-ref ref="STDOUT"/>
    </root>
</configuration>

If you want to achieve the same thing using prefixes, you need to set the following properties in the pl4j.properties:

pl4j.use-markers=false
pl4j.prefix-format=%1$-29s

And the log pattern will need to remove the %marker:

<?xml version="1.0" encoding="UTF-8"?>
<configuration debug="false">
    <statusListener class="ch.qos.logback.core.status.NopStatusListener"/>
    <appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
        <withJansi>true</withJansi>
        <encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder">
            <pattern>%msg%n</pattern>
        </encoder>
    </appender>
    <root level="trace">
        <appender-ref ref="STDOUT"/>
    </root>
</configuration>

The next sections will use the name Markers generically, as a way to identify both the Markers and prefixes cases. You can get back to this section and the Marker vs Prefix section to see how you can configure any of them.

Log Level and Markers

PL4J uses the following mapping between the SLF4J log levels and the markers:

SLF4J Log Level PL4J Marker
debug debug
info awaiting, complete, info, fav, note, pause, pending, santa, star, start, stop, success, skipping, time, timeEnd
warn warning
error error, fatal

Markers Configuration

Individual configuration

You can override the default Markers as follows:

import io.github.ludovicianul.prettylogger.config.level.PrettyMarker;
import io.github.ludovicianul.prettylogger.PrettyLogger;
import io.github.ludovicianul.prettylogger.config.level.ConfigFactory;

import java.util.HashMap;

public class TestClass {

    public static void main(String... args) {
        PrettyLogger prettyLogger = PrettyLoggerFactory.getLogger(TestClass.class);
        //option 1
        PrettyMarker config = ConfigFactory.error().label("err");// we change the label to `err` instead of `error`
        prettyLogger.log(config, "this is an error");//note that we use log() instead of error()
        //option 2
        Map<PrettyMarker.ConfigKey, Object> configMap = new HashMap<>();
        configMap.put(PrettyMarker.ConfigKey.UNDERLINE, true);
        prettyLogger.skip(configMap, "skip processing for id 1");
    }
}

You can override both the symbol and the label. The following flags can also be configured (all booleans):

  • bold
  • underline
  • showLabel
  • showSymbol
  • uppercaseLabel

The flags can be configured both individually, for each marker:

import io.github.ludovicianul.prettylogger.config.level.PrettyMarker;
import io.github.ludovicianul.prettylogger.PrettyLogger;
import io.github.ludovicianul.prettylogger.config.level.ConfigFactory;

public class TestClass {

    public static void main(String... args) {
        PrettyLogger prettyLogger = PrettyLoggerFactory.getLogger(TestClass.class);
        PrettyMarker config = ConfigFactory.error().bold(false).underline(true).showLabel(true).showSymbol(true).uppercaseLabel(true);
        prettyLogger.log(config, "this is an error");//note that we use log() instead of error()
    }
}

As well as globally as shown in Global Configuration

Global configuration

You can configure the above flags globally through a file called pl4j.properties which must be present in the classpath. The following properties can be used to change the flag values (all booleans:

  • pl4j.show-labels
  • pl4j.show-symbols
  • pl4j.bold
  • pl4j.underline
  • pl4j.theme
  • pl4j.uppercase-label
  • pl4j.use-markers
  • pl4j.prefix-format

Default values

If no global or individual configuration is supplied the default values are as follows:

  • bold = true
  • underline = false
  • showLabel = true
  • showSymbol = true
  • theme = default
  • uppercaseLabel = false
  • use-markers = true
  • prefix-format = %1$-29s

Themes

PL4J supports themes, meaning that you can create your own combination of symbol, label and color for each MarkerType. In order to do this you must create a file named pl4j-themename.theme following the example of the default theme, place it into the classpath and configure the theme name in pl4j.properties as follows:

pl4j.theme=themename

If PL4J doesn't find the pl4j-themename.theme inside the classpath it will default to the default theme. This is the list of out-of-the-box supported themes:

Timers

You can also automatically measure the duration of specific tasks using timers. This is an example on how to use timers:

import io.github.ludovicianul.prettylogger.config.level.PrettyMarker;
import io.github.ludovicianul.prettylogger.PrettyLogger;
import io.github.ludovicianul.prettylogger.config.level.ConfigFactory;

public class TestClass {

    public static void main(String... args) {
        PrettyLogger prettyLogger = PrettyLoggerFactory.getLogger(TestClass.class);
        prettyLogger.time("job1");
        // stuff happening
        prettyLogger.time("job1");
        //other stuff happening
        prettyLogger.timeEnd("job1");
    }
}

If you call the time() method multiple times using the same timer key, pl4j will display interim timers between the starting time, and the moment you call the method again. The above program will print something like:

👍 job1      Timer starting...
✋ job1      Interim timer run for: 2004ms
✋ job1      Timer run for: 4506ms

NO_COLOR

If NO_COLOR is set, then all ANSI styling is disabled.

Projects using PL4J

Credits

Inspired by signale.