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

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) ); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy