com.adobe.acs.commons.logging.impl.JsonEventLogger Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of acs-aem-commons-bundle Show documentation
Show all versions of acs-aem-commons-bundle Show documentation
Main ACS AEM Commons OSGi Bundle. Includes commons utilities.
/*
* 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