Log4jApiLogFactory.java
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 org.apache.commons.logging.impl;
import java.io.IOException;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.logging.log4j.Level;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Marker;
import org.apache.logging.log4j.MarkerManager;
import org.apache.logging.log4j.spi.AbstractLoggerAdapter;
import org.apache.logging.log4j.spi.ExtendedLogger;
import org.apache.logging.log4j.spi.LoggerAdapter;
import org.apache.logging.log4j.spi.LoggerContext;
import org.apache.logging.log4j.util.StackLocatorUtil;
/**
* Logger factory hardcoded to send everything to Log4j API.
* <p>
* Based on the `log4j-jcl` artifact from Apache Logging Services.
* </p>
*
* @since 1.3.0
*/
public final class Log4jApiLogFactory extends LogFactory {
private static final class Log4j2Log implements Log {
private static final String FQCN = Log4j2Log.class.getName();
private final ExtendedLogger logger;
public Log4j2Log(final ExtendedLogger logger) {
this.logger = logger;
}
@Override
public void debug(final Object message) {
logIfEnabled(Level.DEBUG, message, null);
}
@Override
public void debug(final Object message, final Throwable t) {
logIfEnabled(Level.DEBUG, message, t);
}
@Override
public void error(final Object message) {
logIfEnabled(Level.ERROR, message, null);
}
@Override
public void error(final Object message, final Throwable t) {
logIfEnabled(Level.ERROR, message, t);
}
@Override
public void fatal(final Object message) {
logIfEnabled(Level.FATAL, message, null);
}
@Override
public void fatal(final Object message, final Throwable t) {
logIfEnabled(Level.FATAL, message, t);
}
@Override
public void info(final Object message) {
logIfEnabled(Level.INFO, message, null);
}
@Override
public void info(final Object message, final Throwable t) {
logIfEnabled(Level.INFO, message, t);
}
@Override
public boolean isDebugEnabled() {
return isEnabled(Level.DEBUG);
}
private boolean isEnabled(final Level level) {
return logger.isEnabled(level, MARKER, null);
}
@Override
public boolean isErrorEnabled() {
return isEnabled(Level.ERROR);
}
@Override
public boolean isFatalEnabled() {
return isEnabled(Level.FATAL);
}
@Override
public boolean isInfoEnabled() {
return isEnabled(Level.INFO);
}
@Override
public boolean isTraceEnabled() {
return isEnabled(Level.TRACE);
}
@Override
public boolean isWarnEnabled() {
return isEnabled(Level.WARN);
}
private void logIfEnabled(final Level level, final Object message, final Throwable t) {
if (message instanceof CharSequence) {
logger.logIfEnabled(FQCN, level, MARKER, (CharSequence) message, t);
} else {
logger.logIfEnabled(FQCN, level, MARKER, message, t);
}
}
@Override
public void trace(final Object message) {
logIfEnabled(Level.TRACE, message, null);
}
@Override
public void trace(final Object message, final Throwable t) {
logIfEnabled(Level.TRACE, message, t);
}
@Override
public void warn(final Object message) {
logIfEnabled(Level.WARN, message, null);
}
@Override
public void warn(final Object message, final Throwable t) {
logIfEnabled(Level.WARN, message, t);
}
}
private static final class LogAdapter extends AbstractLoggerAdapter<Log> {
@Override
protected LoggerContext getContext() {
return getContext(LogManager.getFactory().isClassLoaderDependent() ? StackLocatorUtil.getCallerClass(
LogFactory.class) : null);
}
@Override
protected Log newLogger(final String name, final LoggerContext context) {
return new Log4j2Log(context.getLogger(name));
}
}
private static final String[] EMPTY_ARRAY = {};
/**
* Marker used by all messages coming from Apache Commons Logging.
*/
private static final Marker MARKER = MarkerManager.getMarker("COMMONS-LOGGING");
/**
* Caches Log instances
*/
private final LoggerAdapter<Log> adapter = new LogAdapter();
private final ConcurrentMap<String, Object> attributes = new ConcurrentHashMap<>();
/**
* Constructs a new instance.
*/
public Log4jApiLogFactory() {
// empty
}
@Override
public Object getAttribute(final String name) {
return attributes.get(name);
}
@Override
public String[] getAttributeNames() {
return attributes.keySet().toArray(EMPTY_ARRAY);
}
@Override
public Log getInstance(final Class<?> clazz) {
return getInstance(clazz.getName());
}
@Override
public Log getInstance(final String name) {
return adapter.getLogger(name);
}
/**
* This method is supposed to clear all loggers. In this implementation it will clear all the logger
* wrappers but the loggers managed by the underlying logger context will not be.
*/
@Override
public void release() {
try {
adapter.close();
} catch (final IOException ignored) {
// Ignore
}
}
@Override
public void removeAttribute(final String name) {
attributes.remove(name);
}
@Override
public void setAttribute(final String name, final Object value) {
if (value != null) {
attributes.put(name, value);
} else {
removeAttribute(name);
}
}
}