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

com.comoyo.logging.context.json.JsonEventFormatter Maven / Gradle / Ivy

/**
 * Copyright (C) 2014 Telenor Digital AS
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.comoyo.logging.context.json;

import com.comoyo.commons.logging.context.LoggingContext;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.SortedMap;
import java.util.TimeZone;
import java.util.TreeMap;
import java.util.concurrent.Semaphore;
import java.util.logging.Formatter;
import java.util.logging.LogManager;
import java.util.logging.LogRecord;
import javax.json.Json;
import javax.json.stream.JsonGenerator;
import javax.json.stream.JsonGeneratorFactory;

/**
 * LoggingContext-aware log formatter for producing logstash-ready JSON.
 * 
 * This is just a formatter, shipping the product off to logstash is left as an
 * exercise for the reader. One recommended option is log with a regular FileHandler
 * and asynchronously transport those to logstash, an alternative is to send the events
 * directly into a Redis list with a custom LogHandler.
 *
 * JSON structure and names based on JSONEventLayoutV1.java from log4j-jsonevent-layout.
 *
 * @author anders
 */
public class JsonEventFormatter extends Formatter {
    public static final String JSON_KEY_VERSION = "@version";
    public static final String JSON_KEY_TIMESTAMP = "@timestamp";
    public static final String JSON_KEY_LEVEL = "level";
    public static final String JSON_KEY_MESSAGE = "message";
    public static final String JSON_KEY_SOURCE = "source";
    public static final String JSON_KEY_SOURCE_HOST = "source_host";
    public static final String JSON_KEY_LOGGER_NAME = "logger_name";
    public static final String JSON_KEY_SOURCE_METHOD = "source_method";
    public static final String JSON_KEY_SOURCE_CLASS = "source_class";
    public static final String JSON_KEY_CONTEXT = "context";
    public static final String JSON_KEY_EXCEPTION = "exception";
    public static final String JSON_KEY_CAUSE = "cause";
    public static final String JSON_KEY_SUPPRESSED = "suppressed";
    public static final String JSON_KEY_EXCEPTION_CLASS = "exception_class";
    public static final String JSON_KEY_EXCEPTION_MESSAGE = "exception_message";
    public static final String JSON_KEY_EXCEPTION_STACKTRACE = "stacktrace";
    public static final String JSON_KEY_THREAD_ID = "thread_id";
    public static final String JSON_KEY_THREAD_NAME = "thread_name";

    static final int LOGSTASH_JSON_VERSION = 1;
    static final String ISO8601_DATE_FORMAT = "yyyy-MM-dd'T'HH:mm:ss.SSSZZ";
    static final String PROPERTY_PREFIX = JsonEventFormatter.class.getName() + ".";
    static final String TAG_PROPERTY = "tags";
    static final String PRETTY_PROPERTY = "pretty";
    static final String SOURCE_PROPERTY = "source";
    static final String SOURCEHOST_PROPERTY = "source_host";
    static final String TAG_SEPARATOR_REGEX = "[, ]+";

    // MUST NOT be modified after initialization - clone()d without synchronization on all format/1 calls.
    private static final SimpleDateFormat dateFormatterMaster = new SimpleDateFormat(ISO8601_DATE_FORMAT);
    static {
        dateFormatterMaster.setTimeZone(TimeZone.getTimeZone("UTC"));
    }

    // All of these are to be immutable as well.
    private final JsonGeneratorFactory jsonFactory;
    private final String hostName;
    private final String sourceName;
    private final List tags;

    // Immutable and replaceable
    private volatile SortedMap threadNames;
    private final Semaphore threadNamesLock;

    /**
     * Default constructor for use by java.util.logging - retrieves configuration
     * from logging properties:
     *
     * 
  • com.comoyo.logging.context.json.JsonEventFormatter.source_host
    : explicit * configuration of host name in log entries - default auto-determined. *
  • com.comoyo.logging.context.json.JsonEventFormatter.source
    : Log source, i.e. * application name *
  • com.comoyo.logging.context.json.JsonEventFormatter.tags
    : Comma/space separated * list of strings to tag log entries with, i.e. runtime environment *
  • com.comoyo.logging.context.json.JsonEventFormatter.pretty
    : If "true" or "yes", * output JSON formatted with indentation and line breaks for human consumption. Possibly * useful for debugging, not recommended for machine consumption. */ public JsonEventFormatter() { this( getLoggerPropertyOrDefault(SOURCEHOST_PROPERTY, defaultHostName()), getLoggerPropertyOrDefault(SOURCE_PROPERTY, "java"), defaultTags(), defaultJsonFactory()); } private static String defaultHostName() { try { return InetAddress.getLocalHost().getHostName(); } catch (UnknownHostException ex) { return "unknown-host"; } } private static List defaultTags() { final String configuredTagsString = getLoggerPropertyOrDefault(TAG_PROPERTY, ""); if (!"".equals(configuredTagsString)) { final String[] configuredTags = configuredTagsString.split(TAG_SEPARATOR_REGEX); return Arrays.asList(configuredTags); } else { return new ArrayList<>(); } } private static JsonGeneratorFactory defaultJsonFactory() { final String prettyProperty = getLoggerPropertyOrDefault(PRETTY_PROPERTY, "false"); final HashMap jsonConfig = new HashMap<>(1); if ("true".equalsIgnoreCase(prettyProperty) || "yes".equalsIgnoreCase(prettyProperty)) { jsonConfig.put(JsonGenerator.PRETTY_PRINTING, prettyProperty); } return Json.createGeneratorFactory(jsonConfig); } /** * Construct a JSON log event formatter with explicit configuration, without looking at the * java.util.logging properties. * @param hostName Host name to use in log entries. * @param source Log source - i.e. application name. * @param tags Strings to tag log entries with, i.e. runtime environment: "production", "test" * @param prettyJson If true; output more human-readable JSON. Not recommended for machine * consumption, but perhaps more convenient when debugging. * @param jsonFactory - a JsonFactory configured to your liking, i.e. *
         * final HashMap jsonConfig = new HashMap<>(1);
         * jsonConfig.put(JsonGenerator.PRETTY_PRINTING, "true");
         * final JsonFactory jsonFactory = Json.createGeneratorFactory(jsonConfig);
         * 
    * to get pretty-printed JSON with whitespace and indentation. */ public JsonEventFormatter(final String hostName, final String source, final List tags, JsonGeneratorFactory jsonFactory) { assert hostName != null; assert source != null; assert tags != null; assert jsonFactory != null; this.hostName = hostName; this.sourceName = source; this.tags = Collections.unmodifiableList(new ArrayList(tags)); this.jsonFactory = jsonFactory; threadNames = Collections.unmodifiableSortedMap(new TreeMap()); threadNamesLock = new Semaphore(1); } /** * {@inheritDoc} * @param record A Java log record. * @return A string containing a logstash-style JSON log event. */ @Override public String format(final LogRecord record) { final SimpleDateFormat dateFormatter = (SimpleDateFormat) dateFormatterMaster.clone(); final String timeStampString = dateFormatter.format(new Date(record.getMillis())); final String formattedMessage = formatMessage(record); final StringWriter writer = new StringWriter(); final JsonGenerator json = jsonFactory.createGenerator(writer); { json.writeStartObject() .write(JSON_KEY_VERSION, LOGSTASH_JSON_VERSION) .write(JSON_KEY_TIMESTAMP, timeStampString) .write(JSON_KEY_SOURCE_HOST, hostName) .write(JSON_KEY_SOURCE, sourceName) .write(JSON_KEY_LEVEL, record.getLevel().getName()) .write(JSON_KEY_MESSAGE, formattedMessage) .write(JSON_KEY_THREAD_ID, record.getThreadID()); final String threadName = getThreadName(record.getThreadID()); if (null != threadName) { json.write(JSON_KEY_THREAD_NAME, threadName); } if (null != record.getLoggerName()) { json.write(JSON_KEY_LOGGER_NAME, record.getLoggerName()); } if (null != record.getSourceClassName()) { json.write(JSON_KEY_SOURCE_CLASS, record.getSourceClassName()); } if (null != record.getSourceMethodName()) { json.write(JSON_KEY_SOURCE_METHOD, record.getSourceMethodName()); } if (!tags.isEmpty()) { json.writeStartArray("tags"); for (String tag: tags) { json.write(tag); } json.writeEnd(); } final Throwable thrown = record.getThrown(); if (thrown != null) { json.writeStartObject(JSON_KEY_EXCEPTION); boolean withDescription = true; writeThrowableToJson(json, thrown, withDescription); json.writeEnd(); } final Map context = thrown != null ? LoggingContext.getLastEnteredContext() : LoggingContext.getContext(); if (context != null) { writeContextToJson(json, context); } json.writeEnd(); } json.flush(); writer.append('\n'); return writer.toString(); } /** * Write a set of key/value pairs to JSON object called "context". No output is made * if there are no key/value pairs. * @param json A JsonGenerator to output the JSON object to. * @param context A map to get the context data from. */ private static void writeContextToJson(JsonGenerator json, Map context) { if (!context.isEmpty()) { json.writeStartObject(JSON_KEY_CONTEXT); for (Map.Entry entry: context.entrySet()) { json.write(entry.getKey(), entry.getValue()); } json.writeEnd(); } } /** * Write a Throwable to a JSON generator stream. In the normal case a trivial * affair, but as Throwables can have both causes (recursively) and zero or more * suppressed "sibling Throwables", the rabbit hole is potentially deep. JSON is a pretty good * format for representing all of this, though. * * The description field is skipped on all nested/suppressed exceptions, as their stack traces * are already contained in the description of the root exception. * * Assumes that the start of the object has already been written, and that the end of the object * will be written after its return. (To facilitate writing Throwables in different contexts, * which is exploited when this function calls itself to write Throwables in both sub-objects * and JSON arrays). * * @param json A JsonGenerator to output the JSON structure to. * @param thrown A Throwable to process. */ private static void writeThrowableToJson( final JsonGenerator json, final Throwable thrown, boolean withDescription) { final String exceptionClass = thrown.getClass().getCanonicalName(); final String exceptionMessage = thrown.getMessage(); if (exceptionClass != null) { json.write(JSON_KEY_EXCEPTION_CLASS, exceptionClass); } if (exceptionMessage != null) { json.write(JSON_KEY_EXCEPTION_MESSAGE, exceptionMessage); } if (withDescription) { final StringWriter stackWriter = new StringWriter(); thrown.printStackTrace(new PrintWriter(stackWriter)); String stack = stackWriter.toString(); if (!stack.isEmpty()) { json.write(JSON_KEY_EXCEPTION_STACKTRACE, stack); } } final boolean noDescription = false; final Throwable[] suppressed = thrown.getSuppressed(); if (null != suppressed && suppressed.length > 0) { json.writeStartArray(JSON_KEY_SUPPRESSED); for (Throwable oneSuppressed: suppressed) { json.writeStartObject(); writeThrowableToJson(json, oneSuppressed, noDescription); json.writeEnd(); } json.writeEnd(); } final Throwable cause = thrown.getCause(); if (null != cause) { json.writeStartObject(JSON_KEY_CAUSE); writeThrowableToJson(json, cause, noDescription); json.writeEnd(); } } private static String getLoggerPropertyOrDefault(final String propertyName, final String defaultValue) { final String foundProperty = LogManager.getLogManager().getProperty(PROPERTY_PREFIX + propertyName); return foundProperty != null ? foundProperty : defaultValue; } /** * Best-effort attempt at resolving a thread id to a thread name. * Will return null if the thread id cannot be resolved, e.g if * the thread in question has already exited. May potentially * return misleading data if thread ids are reused. This is not * guaranteed not to happen, but the Java 7 Thread implementation * in practice obeys this constraint. * * @param tid A thread id * @return The corresponding name for the thread in question */ private String getThreadName(long tid) { if (!threadNames.containsKey(tid)) { if (threadNamesLock.tryAcquire()) { try { updateThreadNames(); } finally { threadNamesLock.release(); } } } return threadNames.get(tid); } private void updateThreadNames() { ThreadGroup root = Thread.currentThread().getThreadGroup(); while (true) { final ThreadGroup parent = root.getParent(); if (null == parent) { break; } root = parent; } final int active = root.activeCount(); final Thread[] threads = new Thread[active + 100]; final int enumerated = root.enumerate(threads); final TreeMap names = new TreeMap<>(); for (int i = 0; i < enumerated; i++) { final Thread thread = threads[i]; names.put(thread.getId(), thread.getName()); } threadNames = Collections.unmodifiableSortedMap(names); } // Package private accessors for test usage. String getHostName() { return this.hostName; } String getSourceName() { return this.sourceName; } List getTags() { return new ArrayList<>(this.tags); } }




  • © 2015 - 2025 Weber Informatics LLC | Privacy Policy