// -*- C++ -*- // Module: Log4CPLUS // File: hierarchy.h // Created: 6/2001 // Author: Tad E. Smith // // // Copyright 2001-2010 Tad E. Smith // // 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. /** @file */ #ifndef LOG4CPLUS_HIERARCHY_HEADER_ #define LOG4CPLUS_HIERARCHY_HEADER_ #include #if defined (LOG4CPLUS_HAVE_PRAGMA_ONCE) #pragma once #endif #include #include #include #include #include namespace log4cplus { // Forward Declarations class HierarchyLocker; /** * This class is specialized in retrieving loggers by name and * also maintaining the logger hierarchy. * * The casual user should not have to deal with this class * directly. However, if you are in an environment where * multiple applications run in the same process, then read on. * * The structure of the logger hierarchy is maintained by the * {@link #getInstance} method. The hierarchy is such that children * link to their parent but parents do not have any pointers to their * children. Moreover, loggers can be instantiated in any order, in * particular descendant before ancestor. * * In case a descendant is created before a particular ancestor, * then it creates a provision node for the ancestor and adds itself * to the provision node. Other descendants of the same ancestor add * themselves to the previously created provision node. */ class LOG4CPLUS_EXPORT Hierarchy { public: // DISABLE_OFF should be set to a value lower than all possible // priorities. static const LogLevel DISABLE_OFF; static const LogLevel DISABLE_OVERRIDE; // Ctors /** * Create a new Logger hierarchy. */ Hierarchy(); // Dtor virtual ~Hierarchy(); // Methods /** * This call will clear all logger definitions from the internal * hashtable. Invoking this method will irrevocably mess up the * logger hierarchy. * * You should really know what you are doing before * invoking this method. */ virtual void clear(); /** * Returns true if the named logger exists * (in the default hierarchy). * * @param name The name of the logger to search for. */ virtual bool exists(const log4cplus::tstring& name); /** * Similar to {@link #disable(LogLevel)} except that the LogLevel * argument is given as a log4cplus::tstring. */ virtual void disable(const log4cplus::tstring& loglevelStr); /** * Disable all logging requests of LogLevel equal to or * below the ll parameter p, for * all loggers in this hierarchy. Logging requests of * higher LogLevel then p remain unaffected. * * Nevertheless, if the * BasicConfigurator::DISABLE_OVERRIDE_KEY property is set to * true, then logging requests are evaluated as usual. * * The "disable" family of methods are there for speed. They * allow printing methods such as debug, info, etc. to return * immediately after an integer comparison without walking the * logger hierarchy. In most modern computers an integer * comparison is measured in nanoseconds where as a logger walk is * measured in units of microseconds. */ virtual void disable(LogLevel ll); /** * Disable all logging requests regardless of logger and LogLevel. * This method is equivalent to calling {@link #disable} with the * argument FATAL_LOG_LEVEL, the highest possible LogLevel. */ virtual void disableAll(); /** * Disable all Debug logging requests regardless of logger. * This method is equivalent to calling {@link #disable} with the * argument DEBUG_LOG_LEVEL. */ virtual void disableDebug(); /** * Disable all Info logging requests regardless of logger. * This method is equivalent to calling {@link #disable} with the * argument INFO_LOG_LEVEL. */ virtual void disableInfo(); /** * Undoes the effect of calling any of {@link #disable}, {@link * #disableAll}, {@link #disableDebug} and {@link #disableInfo} * methods. More precisely, invoking this method sets the Logger * class internal variable called disable to its * default "off" value. */ virtual void enableAll(); /** * Return a new logger instance named as the first parameter using * the default factory. * * If a logger of that name already exists, then it will be * returned. Otherwise, a new logger will be instantiated and * then linked with its existing ancestors as well as children. * * @param name The name of the logger to retrieve. */ virtual Logger getInstance(const log4cplus::tstring& name); /** * Return a new logger instance named as the first parameter using * factory. * * If a logger of that name already exists, then it will be * returned. Otherwise, a new logger will be instantiated by the * factory parameter and linked with its existing * ancestors as well as children. * * @param name The name of the logger to retrieve. * @param factory The factory that will make the new logger instance. */ virtual Logger getInstance(const log4cplus::tstring& name, spi::LoggerFactory& factory); /** * Returns all the currently defined loggers in this hierarchy. * * The root logger is not included in the returned list. */ virtual LoggerList getCurrentLoggers(); /** * Is the LogLevel specified by level enabled? */ virtual bool isDisabled(LogLevel level); /** * Get the root of this hierarchy. */ virtual Logger getRoot() const; /** * Reset all values contained in this hierarchy instance to their * default. This removes all appenders from all loggers, sets * the LogLevel of all non-root loggers to NOT_SET_LOG_LEVEL, * sets their additivity flag to true and sets the LogLevel * of the root logger to DEBUG_LOG_LEVEL. Moreover, message disabling * is set its default "off" value. * * Existing loggers are not removed. They are just reset. * * This method should be used sparingly and with care as it will * block all logging until it is completed.

*/ virtual void resetConfiguration(); /** * Set the default LoggerFactory instance. */ virtual void setLoggerFactory(std::auto_ptr factory); /** * Returns the default LoggerFactory instance. */ virtual spi::LoggerFactory* getLoggerFactory(); /** * Shutting down a hierarchy will safely close and remove * all appenders in all loggers including the root logger. * * Some appenders such as SocketAppender need to be closed before the * application exits. Otherwise, pending logging events might be * lost. * * The shutdown method is careful to close nested * appenders before closing regular appenders. This is allows * configurations where a regular appender is attached to a logger * and again to a nested appender. */ virtual void shutdown(); private: // Types typedef std::vector ProvisionNode; typedef std::map ProvisionNodeMap; typedef std::map LoggerMap; // Methods /** * This is the implementation of the getInstance() method. * NOTE: This method does not lock the hashtable_mutex. */ LOG4CPLUS_PRIVATE virtual Logger getInstanceImpl(const log4cplus::tstring& name, spi::LoggerFactory& factory); /** * This is the implementation of the getCurrentLoggers(). * NOTE: This method does not lock the hashtable_mutex. */ LOG4CPLUS_PRIVATE virtual void initializeLoggerList(LoggerList& list) const; /** * This method loops through all the *potential* parents of * logger'. There 3 possible cases: * * 1) No entry for the potential parent of 'logger' exists * * We create a ProvisionNode for this potential parent and insert * 'logger' in that provision node. * * 2) There is an entry of type Logger for the potential parent. * * The entry is 'logger's nearest existing parent. We update logger's * parent field with this entry. We also break from the loop * because updating our parent's parent is our parent's * responsibility. * * 3) There entry is of type ProvisionNode for this potential parent. * * We add 'logger' to the list of children for this potential parent. */ LOG4CPLUS_PRIVATE void updateParents(Logger const & logger); /** * We update the links for all the children that placed themselves * in the provision node 'pn'. The second argument 'logger' is a * reference for the newly created Logger, parent of all the * children in 'pn' * * We loop on all the children 'c' in 'pn': * * If the child 'c' has been already linked to a child of * 'logger' then there is no need to update 'c'. * * Otherwise, we set logger's parent field to c's parent and set * c's parent field to logger. */ LOG4CPLUS_PRIVATE void updateChildren(ProvisionNode& pn, Logger const & logger); // Data thread::Mutex hashtable_mutex; std::auto_ptr defaultFactory; ProvisionNodeMap provisionNodes; LoggerMap loggerPtrs; Logger root; int disableValue; bool emittedNoAppenderWarning; // Disallow copying of instances of this class Hierarchy(const Hierarchy&); Hierarchy& operator=(const Hierarchy&); // Friends friend class log4cplus::spi::LoggerImpl; friend class log4cplus::HierarchyLocker; }; LOG4CPLUS_EXPORT Hierarchy & getDefaultHierarchy (); } // end namespace log4cplus #endif // LOG4CPLUS_HIERARCHY_HEADER_