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

com.adobe.acs.commons.logging.impl.JsonEventLogger Maven / Gradle / Ivy

There is a newer version: 6.10.0
Show newest version
/*
 * ACS AEM Commons
 *
 * Copyright (C) 2013 - 2023 Adobe
 *
 * 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.adobe.acs.commons.logging.impl;

import com.adobe.acs.commons.util.RequireAem;
import com.google.gson.Gson;
import com.google.gson.JsonObject;
import org.apache.commons.lang3.StringUtils;
import org.apache.felix.scr.annotations.Activate;
import org.apache.felix.scr.annotations.Component;
import org.apache.felix.scr.annotations.ConfigurationPolicy;
import org.apache.felix.scr.annotations.Deactivate;
import org.apache.felix.scr.annotations.Properties;
import org.apache.felix.scr.annotations.Property;
import org.apache.felix.scr.annotations.PropertyOption;
import org.apache.felix.scr.annotations.PropertyUnbounded;
import org.apache.felix.scr.annotations.Reference;
import org.apache.jackrabbit.util.ISO8601;
import org.apache.sling.commons.osgi.PropertiesUtil;

import org.osgi.service.event.Event;
import org.osgi.service.event.EventConstants;
import org.osgi.service.event.EventHandler;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.Arrays;
import java.util.Calendar;
import java.util.Collection;
import java.util.Date;
import java.util.Dictionary;
import java.util.Hashtable;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.function.Consumer;
import java.util.function.Supplier;

/**
 * Logs OSGi Events for any set of topics to an SLF4j Logger Category, as JSON
 * objects.
 */
@Component(metatype = true, configurationFactory = true, policy = ConfigurationPolicy.REQUIRE,
        label = "ACS AEM Commons - JSON Event Logger", description = "Logs OSGi Events for any set of topics to an SLF4j Logger Category, as JSON objects.")
@Properties({
    @Property(
            name = "webconsole.configurationFactory.nameHint",
            value = "Logger: {event.logger.category} for events matching '{event.filter}' on '{event.topics}'")
})
@SuppressWarnings("PMD.MoreThanOneLogger")
public class JsonEventLogger implements EventHandler {

    /**
     * Use this logger for tracing this service instance's own lifecycle.
     */
    private static final Logger log = LoggerFactory.getLogger(JsonEventLogger.class);

    /**
     * We add this timestamp property to all logged events
     */
    private static final String PROP_TIMESTAMP = "_timestamp";

    private static final String DEFAULT_LEVEL = "INFO";

    /**
     * A simple enum for Slf4j logging levels.
     */
    private enum LogLevel {
        TRACE, DEBUG, INFO, WARN, ERROR, NONE;

        public static LogLevel fromProperty(String prop) {
            if (prop != null) {
                for (LogLevel value : values()) {
                    if (value.name().equalsIgnoreCase(prop)) {
                        return value;
                    }
                }
            }
            return NONE;
        }
    }
    
    // Disable this feature on AEM as a Cloud Service
    @Reference(target="(distribution=classic)")
    RequireAem requireAem;

    @Property(label = "Event Topics", unbounded = PropertyUnbounded.ARRAY,
            description = "This value lists the topics handled by this logger. The value is a list of strings. If the string ends with a star, all topics in this package and all subpackages match. If the string does not end with a star, this is assumed to define an exact topic.")
    static final String OSGI_TOPICS = EventConstants.EVENT_TOPIC;

    @Property(label = "Event Filter", description = "LDAP-style event filter query. Leave blank to log all events to the configured topic or topics.")
    static final String OSGI_FILTER = EventConstants.EVENT_FILTER;

    @Property(label = "Logger Name", description = "The Sling SLF4j Logger Name or Category to send the JSON messages to. Leave empty to disable the logger.")
    static final String OSGI_CATEGORY = "event.logger.category";

    @SuppressWarnings("AEM Rules:AEM-1")
    @Property(label = "Logger Level", value = DEFAULT_LEVEL, options = {
        @PropertyOption(name = "TRACE", value = "Trace"),
        @PropertyOption(name = "DEBUG", value = "Debug"),
        @PropertyOption(name = "INFO", value = "Information"),
        @PropertyOption(name = "WARN", value = "Warnings"),
        @PropertyOption(name = "ERROR", value = "Error")
    }, description = "Select the logging level the messages should be sent with.")
    static final String OSGI_LEVEL = "event.logger.level";

    private String[] topics;
    private String filter;
    private String category;
    private String level;

    /**
     * Suppress the PMD.LoggerIsNotStaticFinal check because the point is to
     * have an SCR-configurable logger separate from the normal class-level log
     * object defined above.
     */
    @SuppressWarnings("PMD.LoggerIsNotStaticFinal")
    private Logger eventLogger;
    private Consumer logMapper = logMapperForLevel(null, null);
    private Supplier logEnabler = logEnablerForLevel(null, null);

    /**
     * Return a logging function appropriate for the specified loglevel.
     *
     * @param logLevel the specified loglegel
     * @param logger   the logger to map to
     * @return a string comsuming logger function
     */
    static Consumer logMapperForLevel(final LogLevel logLevel, final Logger logger) {
        if (logger == null) {
            return (message) -> { /* do nothing */ };
        }
        switch (logLevel) {
            case ERROR:
                return logger::error;
            case WARN:
                return logger::warn;
            case INFO:
                return logger::info;
            case DEBUG:
                return logger::debug;
            case TRACE:
                return logger::trace;
            default:
                return (message) -> { /* do nothing */ };
        }
    }

    /**
     * Return a logging function appropriate for the specified log level.
     *
     * @param logLevel the specified log level
     * @return a boolean-supplying function
     */
    static Supplier logEnablerForLevel(final LogLevel logLevel, final Logger logger) {
        if (logger == null) {
            return () -> false;
        }
        switch (logLevel) {
            case ERROR:
                return logger::isErrorEnabled;
            case WARN:
                return logger::isWarnEnabled;
            case INFO:
                return logger::isInfoEnabled;
            case DEBUG:
                return logger::isDebugEnabled;
            case TRACE:
                return logger::isTraceEnabled;
            default:
                return () -> false;
        }
    }

    /**
     * Serializes an OSGi {@link org.osgi.service.event.Event} into a JSON
     * object string
     *
     * @param event the event to be serialized as
     * @return a serialized JSON object
     */
    protected static String constructMessage(Event event) {
        Map eventProperties = new LinkedHashMap<>();
        for (String prop : event.getPropertyNames()) {
            Object val = event.getProperty(prop);
            Object converted = convertValue(val);
            eventProperties.put(prop, converted == null ? val : converted);
        }
        eventProperties.put(PROP_TIMESTAMP, ISO8601.format(Calendar.getInstance()));
        Gson gson = new Gson();
        return gson.toJson(eventProperties);
    }

    /**
     * Converts individual java objects to JSONObjects using reflection and
     * recursion
     *
     * @param val an untyped Java object to try to convert
     * @return {@code val} if not handled, or return a converted JSONObject,
     * JSONArray, or String
     */
    @SuppressWarnings({"unchecked", "squid:S3776"})
    protected static Object convertValue(Object val) {
        if (val instanceof Calendar) {
            return ISO8601.format((Calendar) val);
        } else if (val instanceof Date) {
            Calendar calendar = Calendar.getInstance();
            calendar.setTime((Date) val);
            return ISO8601.format(calendar);
        }

        return val;
    }

    //
    // ---------------------------------------------------------< EventHandler methods >-----
    //

    /**
     * {@inheritDoc}
     */
    @Override
    public void handleEvent(Event event) {
        if (event.getProperty("event.application") == null && this.logEnabler.get()) {
            logMapper.accept(constructMessage(event));
        }
    }

    //
    // ---------------------------------------------------------< SCR methods >-------------
    //
    @Activate
    @SuppressWarnings("squid:S1149")
    protected void activate(final Map config) {
        log.trace("[activate] entered activate method.");
        this.topics = PropertiesUtil.toStringArray(config.get(OSGI_TOPICS));
        this.filter = PropertiesUtil.toString(config.get(OSGI_FILTER), "").trim();
        this.category = PropertiesUtil.toString(config.get(OSGI_CATEGORY), "").trim();
        this.level = PropertiesUtil.toString(config.get(OSGI_LEVEL), DEFAULT_LEVEL);

        if (StringUtils.isNotEmpty(this.category)) {
            this.eventLogger = LoggerFactory.getLogger(this.category);
        } else {
            log.warn("No event.logger.category specified. No events will be logged.");
        }

        final LogLevel logLevel = LogLevel.fromProperty(this.level);
        this.logEnabler = logEnablerForLevel(logLevel, this.eventLogger);
        this.logMapper = logMapperForLevel(logLevel, this.eventLogger);
        log.trace("[activate] logger state: {}", this);
    }

    @Deactivate
    protected void deactivate() {
        log.trace("[deactivate] entered deactivate method.");
        this.logEnabler = logEnablerForLevel(LogLevel.NONE, this.eventLogger);
        this.logMapper = logMapperForLevel(LogLevel.NONE, this.eventLogger);
        this.eventLogger = null;
    }

    //
    // ---------------------------------------------------------< Object methods >-------------
    //
    @Override
    public String toString() {
        return "JsonEventLogger{"
                + "topics=" + Arrays.toString(topics)
                + ", filter='" + filter + '\''
                + ", category='" + category + '\''
                + ", level='" + level + '\''
                + ", enabled=" + logEnabler.get()
                + '}';
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy