View Javadoc
1   /**
2    * Copyright (c) 2004-2011 QOS.ch
3    * All rights reserved.
4    *
5    * Permission is hereby granted, free  of charge, to any person obtaining
6    * a  copy  of this  software  and  associated  documentation files  (the
7    * "Software"), to  deal in  the Software without  restriction, including
8    * without limitation  the rights to  use, copy, modify,  merge, publish,
9    * distribute,  sublicense, and/or sell  copies of  the Software,  and to
10   * permit persons to whom the Software  is furnished to do so, subject to
11   * the following conditions:
12   *
13   * The  above  copyright  notice  and  this permission  notice  shall  be
14   * included in all copies or substantial portions of the Software.
15   *
16   * THE  SOFTWARE IS  PROVIDED  "AS  IS", WITHOUT  WARRANTY  OF ANY  KIND,
17   * EXPRESS OR  IMPLIED, INCLUDING  BUT NOT LIMITED  TO THE  WARRANTIES OF
18   * MERCHANTABILITY,    FITNESS    FOR    A   PARTICULAR    PURPOSE    AND
19   * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20   * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21   * OF CONTRACT, TORT OR OTHERWISE,  ARISING FROM, OUT OF OR IN CONNECTION
22   * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23   *
24   */
25  package org.slf4j.log4j12;
26  
27  import static org.slf4j.event.EventConstants.NA_SUBST;
28  
29  import java.io.Serializable;
30  
31  import org.apache.log4j.Level;
32  import org.apache.log4j.spi.LocationInfo;
33  import org.apache.log4j.spi.ThrowableInformation;
34  import org.slf4j.Logger;
35  import org.slf4j.Marker;
36  import org.slf4j.event.LoggingEvent;
37  import org.slf4j.event.SubstituteLoggingEvent;
38  import org.slf4j.helpers.LegacyAbstractLogger;
39  import org.slf4j.helpers.MessageFormatter;
40  import org.slf4j.helpers.NormalizedParameters;
41  import org.slf4j.helpers.SubstituteLogger;
42  import org.slf4j.spi.LocationAwareLogger;
43  import org.slf4j.spi.LoggingEventAware;
44  import org.slf4j.spi.LoggingEventBuilder;
45  
46  /**
47   * A wrapper over {@link org.apache.log4j.Logger org.apache.log4j.Logger} in
48   * conforming to the {@link Logger} interface.
49   * 
50   * <p>
51   * Note that the logging levels mentioned in this class refer to those defined
52   * in the <a href=
53   * "http://logging.apache.org/log4j/docs/api/org/apache/log4j/Level.html">
54   * <code>org.apache.log4j.Level</code></a> class.
55   * 
56   * <p>
57   * The TRACE level was introduced in log4j version 1.2.12. In order to avoid
58   * crashing the host application, in the case the log4j version in use predates
59   * 1.2.12, the TRACE level will be mapped as DEBUG. See also
60   * <a href="http://jira.qos.ch/browse/SLF4J-59">SLF4J-59</a>.
61   * 
62   * @author Ceki G&uuml;lc&uuml;
63   */
64  public final class Log4jLoggerAdapter extends LegacyAbstractLogger implements LocationAwareLogger, LoggingEventAware, Serializable {
65  
66      private static final long serialVersionUID = 6182834493563598289L;
67  
68      final transient org.apache.log4j.Logger logger;
69  
70      /**
71       * Following the pattern discussed in pages 162 through 168 of "The complete
72       * log4j manual".
73       */
74  
75      final static String FQCN_NOMINAL = org.slf4j.helpers.AbstractLogger.class.getName();
76      final static String FQCN_SUBSTITUE = FQCN_NOMINAL;
77      final static String FQCN_FLUENT = org.slf4j.spi.DefaultLoggingEventBuilder.class.getName();
78  
79      // Does the log4j version in use recognize the TRACE level?
80      // The trace level was introduced in log4j 1.2.12.
81      final boolean traceCapable;
82  
83      // WARN: Log4jLoggerAdapter constructor should have only package access so
84      // that only Log4jLoggerFactory be able to create one.
85      Log4jLoggerAdapter(org.apache.log4j.Logger logger) {
86          this.logger = logger;
87          this.name = logger.getName();
88          traceCapable = isTraceCapable();
89      }
90  
91      private boolean isTraceCapable() {
92          try {
93              logger.isTraceEnabled();
94              return true;
95          } catch (NoSuchMethodError e) {
96              return false;
97          }
98      }
99  
100     /**
101      * Is this logger instance enabled for the TRACE level?
102      * 
103      * @return True if this Logger is enabled for level TRACE, false otherwise.
104      */
105     public boolean isTraceEnabled() {
106         if (traceCapable) {
107             return logger.isTraceEnabled();
108         } else {
109             return logger.isDebugEnabled();
110         }
111     }
112 
113     /**
114      * Is this logger instance enabled for the DEBUG level?
115      * 
116      * @return True if this Logger is enabled for level DEBUG, false otherwise.
117      */
118     public boolean isDebugEnabled() {
119         return logger.isDebugEnabled();
120     }
121 
122     /**
123      * Is this logger instance enabled for the INFO level?
124      * 
125      * @return True if this Logger is enabled for the INFO level, false otherwise.
126      */
127     public boolean isInfoEnabled() {
128         return logger.isInfoEnabled();
129     }
130 
131     /**
132      * Is this logger instance enabled for the WARN level?
133      * 
134      * @return True if this Logger is enabled for the WARN level, false otherwise.
135      */
136     public boolean isWarnEnabled() {
137         return logger.isEnabledFor(Level.WARN);
138     }
139 
140     /**
141      * Is this logger instance enabled for level ERROR?
142      * 
143      * @return True if this Logger is enabled for level ERROR, false otherwise.
144      */
145     public boolean isErrorEnabled() {
146         return logger.isEnabledFor(Level.ERROR);
147     }
148 
149     @Override
150     public void log(Marker marker, String callerFQCN, int level, String msg, Object[] arguments, Throwable t) {
151         Level log4jLevel = toLog4jLevel(level);
152         NormalizedParameters np = NormalizedParameters.normalize(msg, arguments, t);
153         String formattedMessage = MessageFormatter.basicArrayFormat(np.getMessage(), np.getArguments());
154         logger.log(callerFQCN, log4jLevel, formattedMessage, np.getThrowable());
155     }
156 
157     @Override
158     protected void handleNormalizedLoggingCall(org.slf4j.event.Level level, Marker marker, String msg, Object[] arguments, Throwable throwable) {
159         Level log4jLevel = toLog4jLevel(level.toInt());
160         String formattedMessage = MessageFormatter.basicArrayFormat(msg, arguments);
161         logger.log(getFullyQualifiedCallerName(), log4jLevel, formattedMessage, throwable);
162     }
163 
164     /**
165      * Called by {@link SubstituteLogger} or by {@link LoggingEventBuilder} instances
166      * @param event
167      */
168     public void log(LoggingEvent event) {
169         Level log4jLevel = toLog4jLevel(event.getLevel().toInt());
170         if (!logger.isEnabledFor(log4jLevel))
171             return;
172 
173         org.apache.log4j.spi.LoggingEvent log4jevent = event2Log4jEvent(event, log4jLevel);
174         logger.callAppenders(log4jevent);
175 
176     }
177 
178     private org.apache.log4j.spi.LoggingEvent event2Log4jEvent(LoggingEvent event, Level log4jLevel) {
179 
180         String formattedMessage = MessageFormatter.basicArrayFormat(event.getMessage(), event.getArgumentArray());
181 
182         LocationInfo locationInfo = null;
183         String fcqn = null;
184 
185         if (event instanceof SubstituteLoggingEvent) {
186             locationInfo = new LocationInfo(NA_SUBST, NA_SUBST, NA_SUBST, "0");
187             fcqn = FQCN_SUBSTITUE;
188         } else {
189             fcqn = FQCN_FLUENT;
190         }
191 
192         ThrowableInformation ti = null;
193         Throwable t = event.getThrowable();
194         if (t != null)
195             ti = new ThrowableInformation(t);
196 
197         org.apache.log4j.spi.LoggingEvent log4jEvent = new org.apache.log4j.spi.LoggingEvent(fcqn, logger, event.getTimeStamp(), log4jLevel, formattedMessage,
198                         event.getThreadName(), ti, null, locationInfo, null);
199 
200         return log4jEvent;
201     }
202 
203     private Level toLog4jLevel(int slf4jLevelInt) {
204         Level log4jLevel;
205         switch (slf4jLevelInt) {
206         case LocationAwareLogger.TRACE_INT:
207             log4jLevel = traceCapable ? Level.TRACE : Level.DEBUG;
208             break;
209         case LocationAwareLogger.DEBUG_INT:
210             log4jLevel = Level.DEBUG;
211             break;
212         case LocationAwareLogger.INFO_INT:
213             log4jLevel = Level.INFO;
214             break;
215         case LocationAwareLogger.WARN_INT:
216             log4jLevel = Level.WARN;
217             break;
218         case LocationAwareLogger.ERROR_INT:
219             log4jLevel = Level.ERROR;
220             break;
221         default:
222             throw new IllegalStateException("Level number " + slf4jLevelInt + " is not recognized.");
223         }
224         return log4jLevel;
225     }
226 
227     @Override
228     protected String getFullyQualifiedCallerName() {
229         return FQCN_NOMINAL;
230     }
231 
232 }