org.apache.commons.logging.LogFactory Maven / Gradle / Ivy
Show all versions of commons-logging-jboss-logging Show documentation
/*
* JBoss, Home of Professional Open Source.
*
* Copyright 2018 Red Hat, Inc., and individual contributors
* as indicated by the @author tags.
*
* 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 org.apache.commons.logging;
import java.util.Hashtable;
import org.apache.commons.logging.impl.JBossLogFactory;
/**
* Note this implementation only works with JBoss Log Manager. No configuration can be done via this
* API.
*
* Factory for creating {@link Log} instances, with discovery and
* configuration features similar to that employed by standard Java APIs
* such as JAXP.
*
*
* IMPLEMENTATION NOTE - This implementation is heavily
* based on the SAXParserFactory and DocumentBuilderFactory implementations
* (corresponding to the JAXP pluggability APIs) found in Apache Xerces.
*
* @version $Id: LogFactory.java 1606041 2014-06-27 11:56:59Z tn $
*/
@SuppressWarnings({"unused", "WeakerAccess"})
public abstract class LogFactory {
private static class Holder {
static final org.apache.commons.logging.LogFactory LOG_FACTORY = new JBossLogFactory();
}
// Implementation note re AccessController usage
//
// It is important to keep code invoked via an AccessController to small
// auditable blocks. Such code must carefully evaluate all user input
// (parameters, system properties, config file contents, etc). As an
// example, a Log implementation should not write to its logfile
// with an AccessController anywhere in the call stack, otherwise an
// insecure application could configure the log implementation to write
// to a protected file using the privileges granted to JCL rather than
// to the calling application.
//
// Under no circumstance should a non-private method return data that is
// retrieved via an AccessController. That would allow an insecure app
// to invoke that method and obtain data that it is not permitted to have.
//
// Invoking user-supplied code with an AccessController set is not a major
// issue (eg invoking the constructor of the class specified by
// HASHTABLE_IMPLEMENTATION_PROPERTY). That class will be in a different
// trust domain, and therefore must have permissions to do whatever it
// is trying to do regardless of the permissions granted to JCL. There is
// a slight issue in that untrusted code may point that environment var
// to another trusted library, in which case the code runs if both that
// lib and JCL have the necessary permissions even when the untrusted
// caller does not. That's a pretty hard route to exploit though.
// ----------------------------------------------------- Manifest Constants
/**
* The name (priority
) of the key in the config file used to
* specify the priority of that particular config file. The associated value
* is a floating-point number; higher values take priority over lower values.
*/
public static final String PRIORITY_KEY = "priority";
/**
* The name (use_tccl
) of the key in the config file used
* to specify whether org.apache.commons.logging classes should be loaded via the thread
* context class loader (TCCL), or not. By default, the TCCL is used.
*/
public static final String TCCL_KEY = "use_tccl";
/**
* The name (org.apache.commons.logging.LogFactory
) of the property
* used to identify the LogFactory implementation
* class name. This can be used as a system property, or as an entry in a
* configuration properties file.
*/
public static final String FACTORY_PROPERTY = "org.apache.commons.logging.LogFactory";
/**
* The fully qualified class name of the fallback LogFactory
* implementation class to use, if no other can be found.
*/
public static final String FACTORY_DEFAULT = "org.apache.commons.logging.impl.JBossLogFactory";
/**
* The name (commons-org.apache.commons.logging.properties
) of the properties file to search for.
*/
public static final String FACTORY_PROPERTIES = "commons-org.apache.commons.logging.properties";
/**
* JDK1.3+
* 'Service Provider' specification.
*/
protected static final String SERVICE_ID =
"META-INF/services/org.apache.commons.logging.LogFactory";
/**
* The name (org.apache.commons.logging.diagnostics.dest
)
* of the property used to enable internal commons-org.apache.commons.logging
* diagnostic output, in order to get information on what org.apache.commons.logging
* implementations are being discovered, what classloaders they
* are loaded through, etc.
*
* If a system property of this name is set then the value is
* assumed to be the name of a file. The special strings
* STDOUT or STDERR (case-sensitive) indicate output to
* System.out and System.err respectively.
*
* Diagnostic org.apache.commons.logging should be used only to debug problematic
* configurations and should not be set in normal production use.
*/
public static final String DIAGNOSTICS_DEST_PROPERTY =
"org.apache.commons.logging.diagnostics.dest";
/**
* Setting this system property
* (org.apache.commons.logging.LogFactory.HashtableImpl
)
* value allows the Hashtable
used to store
* classloaders to be substituted by an alternative implementation.
*
* Note: LogFactory
will print:
*
* [ERROR] LogFactory: Load of custom hashtable failed
*
* to system error and then continue using a standard Hashtable.
*
* Usage: Set this property when Java is invoked
* and LogFactory
will attempt to load a new instance
* of the given implementation class.
* For example, running the following ant scriplet:
*
* <java classname="${test.runner}" fork="yes" failonerror="${test.failonerror}">
* ...
* <sysproperty
* key="org.apache.commons.logging.LogFactory.HashtableImpl"
* value="org.apache.commons.logging.AltHashtable"/>
* </java>
*
* will mean that LogFactory
will load an instance of
* org.apache.commons.logging.AltHashtable
.
*
* A typical use case is to allow a custom
* Hashtable implementation using weak references to be substituted.
* This will allow classloaders to be garbage collected without
* the need to release them (on 1.3+ JVMs only, of course ;).
*
* This property is not used and only here to provide compatibility.
*
*/
public static final String HASHTABLE_IMPLEMENTATION_PROPERTY =
"org.apache.commons.logging.LogFactory.HashtableImpl";
// ----------------------------------------------------------- Constructors
/**
* Protected constructor that is not available for public use.
*/
protected LogFactory() {
}
// --------------------------------------------------------- Public Methods
/**
* Return the configuration attribute with the specified name (if any),
* or null
if there is no such attribute.
*
* @param name Name of the attribute to return
*/
public abstract Object getAttribute(String name);
/**
* Return an array containing the names of all currently defined
* configuration attributes. If there are no such attributes, a zero
* length array is returned.
*/
public abstract String[] getAttributeNames();
/**
* Convenience method to derive a name from the specified class and
* call getInstance(String)
with it.
*
* @param clazz Class for which a suitable Log name will be derived
*
* @throws LogConfigurationException if a suitable Log
* instance cannot be returned
*/
public abstract Log getInstance(Class clazz)
throws LogConfigurationException;
/**
* Construct (if necessary) and return a Log
instance,
* using the factory's current set of configuration attributes.
*
* NOTE - Depending upon the implementation of
* the LogFactory
you are using, the Log
* instance you are returned may or may not be local to the current
* application, and may or may not be returned again on a subsequent
* call with the same name argument.
*
* @param name Logical name of the Log
instance to be
* returned (the meaning of this name is only known to the underlying
* org.apache.commons.logging implementation that is being wrapped)
*
* @throws LogConfigurationException if a suitable Log
* instance cannot be returned
*/
public abstract Log getInstance(String name)
throws LogConfigurationException;
/**
* Release any internal references to previously created {@link Log}
* instances returned by this factory. This is useful in environments
* like servlet containers, which implement application reloading by
* throwing away a ClassLoader. Dangling references to objects in that
* class loader would prevent garbage collection.
*/
public abstract void release();
/**
* Remove any configuration attribute associated with the specified name.
* If there is no such attribute, no action is taken.
*
* @param name Name of the attribute to remove
*/
public abstract void removeAttribute(String name);
/**
* Set the configuration attribute with the specified name. Calling
* this with a null
value is equivalent to calling
* removeAttribute(name)
.
*
* @param name Name of the attribute to set
* @param value Value of the attribute to set, or null
* to remove any setting for this attribute
*/
public abstract void setAttribute(String name, Object value);
// ------------------------------------------------------- Static Variables
/**
* The previously constructed LogFactory
instances, keyed by
* the ClassLoader
with which it was created.
*
* @deprecated this is not used as other factory types are not supported
*/
@Deprecated
protected static Hashtable factories = null;
/**
* Previously constructed LogFactory
instance as in the
* factories
map, but for the case where
* getClassLoader
returns null
.
* This can happen when:
*
* - using JDK1.1 and the calling code is loaded via the system
* classloader (very common)
* - using JDK1.2+ and the calling code is loaded via the boot
* classloader (only likely for embedded systems work).
*
* Note that factories
is a Hashtable (not a HashMap),
* and hashtables don't allow null as a key.
*
* @deprecated since 1.1.2 - this is not used as other factory types are not supported
*/
@Deprecated
protected static volatile org.apache.commons.logging.LogFactory nullClassLoaderFactory = null;
// --------------------------------------------------------- Static Methods
/**
* Construct (if necessary) and return a LogFactory
* instance, using the following ordered lookup procedure to determine
* the name of the implementation class to be loaded.
*
*
* - The
oorg.apache.commons.logging.LogFactory
system
* property.
* - The JDK 1.3 Service Discovery mechanism
* - Use the properties file
commons-org.apache.commons.logging.properties
* file, if found in the class path of this class. The configuration
* file is in standard java.util.Properties
format and
* contains the fully qualified name of the implementation class
* with the key being the system property defined above.
* - Fall back to a default implementation class
* (
org.apache.commons.logging.impl.LogFactoryImpl
).
*
*
* NOTE - If the properties file method of identifying the
* LogFactory
implementation class is utilized, all of the
* properties defined in this file will be set as configuration attributes
* on the corresponding LogFactory
instance.
*
* NOTE - In a multi-threaded environment it is possible
* that two different instances will be returned for the same
* classloader environment.
*
* @throws LogConfigurationException if the implementation class is not
* available or cannot be instantiated.
*/
public static org.apache.commons.logging.LogFactory getFactory() throws LogConfigurationException {
return Holder.LOG_FACTORY;
}
/**
* Convenience method to return a named logger, without the application
* having to care about factories.
*
* @param clazz Class from which a log name will be derived
*
* @throws LogConfigurationException if a suitable Log
* instance cannot be returned
*/
public static Log getLog(Class clazz) throws LogConfigurationException {
return getFactory().getInstance(clazz);
}
/**
* Convenience method to return a named logger, without the application
* having to care about factories.
*
* @param name Logical name of the Log
instance to be
* returned (the meaning of this name is only known to the underlying
* org.apache.commons.logging implementation that is being wrapped)
*
* @throws LogConfigurationException if a suitable Log
* instance cannot be returned
*/
public static Log getLog(String name) throws LogConfigurationException {
return getFactory().getInstance(name);
}
/**
* Release any internal references to previously created {@link org.apache.commons.logging.LogFactory}
* instances that have been associated with the specified class loader
* (if any), after calling the instance method release()
on
* each of them.
*
* @param classLoader ClassLoader for which to release the LogFactory
*/
public static void release(ClassLoader classLoader) {
getFactory().release();
}
/**
* Release any internal references to previously created {@link org.apache.commons.logging.LogFactory}
* instances, after calling the instance method release()
on
* each of them. This is useful in environments like servlet containers,
* which implement application reloading by throwing away a ClassLoader.
* Dangling references to objects in that class loader would prevent
* garbage collection.
*/
public static void releaseAll() {
getFactory().release();
}
// ------------------------------------------------------ Protected Methods
/**
* Safely get access to the classloader for the specified class.
*
* Theoretically, calling getClassLoader can throw a security exception,
* and so should be done under an AccessController in order to provide
* maximum flexibility. However in practice people don't appear to use
* security policies that forbid getClassLoader calls. So for the moment
* all code is written to call this method rather than Class.getClassLoader,
* so that we could put AccessController stuff in this method without any
* disruption later if we need to.
*
* Even when using an AccessController, however, this method can still
* throw SecurityException. Commons-org.apache.commons.logging basically relies on the
* ability to access classloaders, ie a policy that forbids all
* classloader access will also prevent commons-org.apache.commons.logging from working:
* currently this method will throw an exception preventing the entire app
* from starting up. Maybe it would be good to detect this situation and
* just disable all commons-org.apache.commons.logging? Not high priority though - as stated
* above, security policies that prevent classloader access aren't common.
*
* Note that returning an object fetched via an AccessController would
* technically be a security flaw anyway; untrusted code that has access
* to a trusted JCL library could use it to fetch the classloader for
* a class even when forbidden to do so directly.
*
* @since 1.1
*/
protected static ClassLoader getClassLoader(Class clazz) {
throw new UnsupportedOperationException();
}
/**
* Returns the current context classloader.
*
* In versions prior to 1.1, this method did not use an AccessController.
* In version 1.1, an AccessController wrapper was incorrectly added to
* this method, causing a minor security flaw.
*
* In version 1.1.1 this change was reverted; this method no longer uses
* an AccessController. User code wishing to obtain the context classloader
* must invoke this method via AccessController.doPrivileged if it needs
* support for that.
*
* @return the context classloader associated with the current thread,
* or null if security doesn't allow it.
*
* @throws LogConfigurationException if there was some weird error while
* attempting to get the context classloader.
*/
protected static ClassLoader getContextClassLoader() throws LogConfigurationException {
throw new UnsupportedOperationException();
}
/**
* Return the thread context class loader if available; otherwise return null.
*
* Most/all code should call getContextClassLoaderInternal rather than
* calling this method directly.
*
* The thread context class loader is available for JDK 1.2
* or later, if certain security conditions are met.
*
* Note that no internal org.apache.commons.logging is done within this method because
* this method is called every time LogFactory.getLogger() is called,
* and we don't want too much output generated here.
*
* @return the thread's context classloader or {@code null} if the java security
* policy forbids access to the context classloader from one of the classes
* in the current call stack.
*
* @throws LogConfigurationException if a suitable class loader
* cannot be identified.
* @since 1.1
*/
protected static ClassLoader directGetContextClassLoader() throws LogConfigurationException {
throw new UnsupportedOperationException();
}
/**
* Return a new instance of the specified LogFactory
* implementation class, loaded by the specified class loader.
* If that fails, try the class loader used to load this
* (abstract) LogFactory.
*
ClassLoader conflicts
*
* Note that there can be problems if the specified ClassLoader is not the
* same as the classloader that loaded this class, ie when loading a
* concrete LogFactory subclass via a context classloader.
*
* The problem is the same one that can occur when loading a concrete Log
* subclass via a context classloader.
*
* The problem occurs when code running in the context classloader calls
* class X which was loaded via a parent classloader, and class X then calls
* LogFactory.getFactory (either directly or via LogFactory.getLog). Because
* class X was loaded via the parent, it binds to LogFactory loaded via
* the parent. When the code in this method finds some LogFactoryYYYY
* class in the child (context) classloader, and there also happens to be a
* LogFactory class defined in the child classloader, then LogFactoryYYYY
* will be bound to LogFactory@childloader. It cannot be cast to
* LogFactory@parentloader, ie this method cannot return the object as
* the desired type. Note that it doesn't matter if the LogFactory class
* in the child classloader is identical to the LogFactory class in the
* parent classloader, they are not compatible.
*
* The solution taken here is to simply print out an error message when
* this occurs then throw an exception. The deployer of the application
* must ensure they remove all occurrences of the LogFactory class from
* the child classloader in order to resolve the issue. Note that they
* do not have to move the custom LogFactory subclass; that is ok as
* long as the only LogFactory class it can find to bind to is in the
* parent classloader.
*
* @param factoryClass Fully qualified name of the LogFactory
* implementation class
* @param classLoader ClassLoader from which to load this class
* @param contextClassLoader is the context that this new factory will
* manage org.apache.commons.logging for.
*
* @throws LogConfigurationException if a suitable instance
* cannot be created
* @since 1.1
*/
protected static org.apache.commons.logging.LogFactory newFactory(final String factoryClass,
final ClassLoader classLoader,
final ClassLoader contextClassLoader)
throws LogConfigurationException {
throw new UnsupportedOperationException();
}
/**
* Method provided for backwards compatibility; see newFactory version that
* takes 3 parameters.
*
* This method would only ever be called in some rather odd situation.
* Note that this method is static, so overriding in a subclass doesn't
* have any effect unless this method is called from a method in that
* subclass. However this method only makes sense to use from the
* getFactory method, and as that is almost always invoked via
* LogFactory.getFactory, any custom definition in a subclass would be
* pointless. Only a class with a custom getFactory method, then invoked
* directly via CustomFactoryImpl.getFactory or similar would ever call
* this. Anyway, it's here just in case, though the "managed class loader"
* value output to the diagnostics will not report the correct value.
*/
protected static org.apache.commons.logging.LogFactory newFactory(final String factoryClass,
final ClassLoader classLoader) {
throw new UnsupportedOperationException();
}
/**
* Indicates true if the user has enabled internal org.apache.commons.logging.
*
* By the way, sorry for the incorrect grammar, but calling this method
* areDiagnosticsEnabled just isn't java beans style.
*
* @return true if calls to logDiagnostic will have any effect.
*
* @since 1.1
*/
protected static boolean isDiagnosticsEnabled() {
throw new UnsupportedOperationException();
}
/**
* Write the specified message to the internal org.apache.commons.logging destination.
*
* @param msg is the diagnostic message to be output.
*
* @since 1.1
*/
@SuppressWarnings("FinalStaticMethod")
protected static final void logRawDiagnostic(String msg) {
throw new UnsupportedOperationException();
}
/**
* Returns a string that uniquely identifies the specified object, including
* its class.
*
* The returned string is of form "classname@hashcode", ie is the same as
* the return value of the Object.toString() method, but works even when
* the specified object's class has overidden the toString method.
*
* @param o may be null.
*
* @return a string of form classname@hashcode, or "null" if param o is null.
*
* @since 1.1
*/
public static String objectId(Object o) {
if (o == null) {
return "null";
} else {
return o.getClass().getName() + "@" + System.identityHashCode(o);
}
}
}