com.jcabi.log.MulticolorLayout Maven / Gradle / Ivy
/*
* Copyright (c) 2012-2024, jcabi.com
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met: 1) Redistributions of source code must retain the above
* copyright notice, this list of conditions and the following
* disclaimer. 2) Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following
* disclaimer in the documentation and/or other materials provided
* with the distribution. 3) Neither the name of the jcabi.com nor
* the names of its contributors may be used to endorse or promote
* products derived from this software without specific prior written
* permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT
* NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
* FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
* THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
* OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package com.jcabi.log;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import org.apache.log4j.EnhancedPatternLayout;
import org.apache.log4j.Level;
import org.apache.log4j.spi.LoggingEvent;
/**
* Multi-color layout for LOG4J.
*
* Use it in your LOG4J configuration:
*
*
log4j.rootLogger=INFO, CONSOLE
* log4j.appender.CONSOLE=org.apache.log4j.ConsoleAppender
* log4j.appender.CONSOLE.layout=com.jcabi.log.MulticolorLayout
* log4j.appender.CONSOLE.layout.ConversionPattern=[%color{%-5p}] %c: %m%n
*
* The part of the message wrapped with {@code %color{...}}
* will change its color according to the logging level of the event. Without
* this highlighting the behavior of the layout is identical to
* {@link EnhancedPatternLayout}. You can use {@code %color-red{...}} if you
* want to use specifically red color for the wrapped piece of text. Supported
* colors are: {@code red}, {@code blue}, {@code yellow}, {@code cyan},
* {@code black}, and {@code white}.
*
*
Besides that you can specify any ANSI color you like with
* {@code %color-;;{...}}, where
* {@code } is a binary mask of attributes,
* {@code } is a background color, and
* {@code } is a foreground color. Read more about
* ANSI escape code.
*
* This class or its parents are not serializable.
*
*
Maven dependency for this class is
* (see How
* to use with Maven instructions):
*
*
<dependency>
* <groupId>com.jcabi</groupId>
* <artifactId>jcabi-log</artifactId>
* </dependency>
* @since 0.1.10
* @see ANSI escape code
* @see PatternLayout from LOG4J
* @see How to use with Maven
*/
@SuppressWarnings("PMD.NonStaticInitializer")
public final class MulticolorLayout extends EnhancedPatternLayout {
/**
* Name of the property that is used to disable log coloring.
*/
private static final String COLORING_PROPERY = "com.jcabi.log.coloring";
/**
* Colors of levels.
*/
private final transient ConcurrentMap levels =
MulticolorLayout.levelMap();
/**
* Store original conversation pattern to be able
* to recalculate it, if new colors are provided.
*/
private transient String base;
/**
* Color human readable data.
*/
private final transient Colors colors = new Colors();
@Override
public void setConversionPattern(final String pattern) {
this.base = pattern;
super.setConversionPattern(
new ConversionPattern(this.base, this.colors).generate()
);
}
/**
* Allow to overwrite or specify new ANSI color names
* in a javascript map like format.
*
* @param cols JavaScript like map of color names
* @since 0.9
*/
@SuppressWarnings("PMD.UseConcurrentHashMap")
public void setColors(final String cols) {
final Map parsed = new ParseableInformation(
cols
).information();
for (final Map.Entry entry : parsed.entrySet()) {
this.colors.addColor(entry.getKey(), entry.getValue());
}
if (this.base != null) {
this.setConversionPattern(this.base);
}
}
/**
* Allow to overwrite the ANSI color values for the log levels
* in a javascript map like format.
* @param lev JavaScript like map of levels
* @since 0.9
*/
@SuppressWarnings("PMD.UseConcurrentHashMap")
public void setLevels(final String lev) {
final Map parsed = new ParseableLevelInformation(
lev
).information();
for (final Map.Entry entry : parsed.entrySet()) {
this.levels.put(entry.getKey(), entry.getValue());
}
}
@Override
public String format(final LoggingEvent event) {
final Formatted formatted;
if (MulticolorLayout.isColoringEnabled()) {
formatted = this.colorfulFormatting(event);
} else {
formatted = this.dullFormatting(event);
}
return formatted.format();
}
/**
* Generate a dull {@code Formatted}.
* @param event Event to be formatted
* @return A {@link Formatted} to format the event
* @checkstyle NonStaticMethodCheck (10 lines)
*/
private Formatted dullFormatting(final LoggingEvent event) {
return new DullyFormatted(super.format(event));
}
/**
* Generate a colorful {@code Formatted}.
* @param event Event to be formatted
* @return Text of a log event, probably colored with ANSI color codes
*/
private Formatted colorfulFormatting(final LoggingEvent event) {
return new ColorfullyFormatted(
super.format(event),
this.levels.get(event.getLevel().toString())
);
}
/**
* Level map.
* @return Map of levels
*/
private static ConcurrentMap levelMap() {
final ConcurrentMap map = new ConcurrentHashMap<>(0);
map.put(Level.TRACE.toString(), "2;33");
map.put(Level.DEBUG.toString(), "2;37");
map.put(Level.INFO.toString(), "0;37");
map.put(Level.WARN.toString(), "0;33");
map.put(Level.ERROR.toString(), "0;31");
map.put(Level.FATAL.toString(), "0;35");
return map;
}
/**
* Should the logged text be colored or not.
* @return True if the coloring is enabled, or false otherwise.
*/
private static boolean isColoringEnabled() {
return !"false".equals(
System.getProperty(MulticolorLayout.COLORING_PROPERY)
);
}
}