1/**
2 * Copyright (c) 2004-2012 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 */
25package org.slf4j.impl;
26
27import java.io.FileNotFoundException;
28import java.io.FileOutputStream;
29import java.io.InputStream;
30import java.io.PrintStream;
31import java.security.AccessController;
32import java.security.PrivilegedAction;
33import java.text.DateFormat;
34import java.text.SimpleDateFormat;
35import java.util.Date;
36import java.util.Properties;
37
38import org.slf4j.Logger;
39import org.slf4j.helpers.FormattingTuple;
40import org.slf4j.helpers.MarkerIgnoringBase;
41import org.slf4j.helpers.MessageFormatter;
42import org.slf4j.helpers.Util;
43import org.slf4j.spi.LocationAwareLogger;
44
45/**
46 * <p>Simple implementation of {@link Logger} that sends all enabled log messages,
47 * for all defined loggers, to the console ({@code System.err}).
48 * The following system properties are supported to configure the behavior of this logger:</p>
49 *
50 * <ul>
51 * <li><code>org.slf4j.simpleLogger.logFile</code> - The output target which can be the <em>path</em> to a file, or
52 * the special values "System.out" and "System.err". Default is "System.err".
53 *
54 * <li><code>org.slf4j.simpleLogger.defaultLogLevel</code> - Default log level for all instances of SimpleLogger.
55 * Must be one of ("trace", "debug", "info", "warn", or "error"). If not specified, defaults to "info". </li>
56 *
57 * <li><code>org.slf4j.simpleLogger.log.<em>a.b.c</em></code> - Logging detail level for a SimpleLogger instance
58 * named "a.b.c". Right-side value must be one of "trace", "debug", "info", "warn", or "error". When a SimpleLogger
59 * named "a.b.c" is initialized, its level is assigned from this property. If unspecified, the level of nearest parent
60 * logger will be used, and if none is set, then the value specified by
61 * <code>org.slf4j.simpleLogger.defaultLogLevel</code> will be used.</li>
62 *
63 * <li><code>org.slf4j.simpleLogger.showDateTime</code> - Set to <code>true</code> if you want the current date and
64 * time to be included in output messages. Default is <code>false</code></li>
65 *
66 * <li><code>org.slf4j.simpleLogger.dateTimeFormat</code> - The date and time format to be used in the output messages.
67 * The pattern describing the date and time format is defined by
68 * <a href="http://docs.oracle.com/javase/1.5.0/docs/api/java/text/SimpleDateFormat.html"><code>SimpleDateFormat</code></a>.
69 * If the format is not specified or is invalid, the number of milliseconds since start up will be output. </li>
70 *
71 * <li><code>org.slf4j.simpleLogger.showThreadName</code> -Set to <code>true</code> if you want to output the current
72 * thread name. Defaults to <code>true</code>.</li>
73 *
74 * <li><code>org.slf4j.simpleLogger.showLogName</code> - Set to <code>true</code> if you want the Logger instance name
75 * to be included in output messages. Defaults to <code>true</code>.</li>
76 *
77 * <li><code>org.slf4j.simpleLogger.showShortLogName</code> - Set to <code>true</code> if you want the last component
78 * of the name to be included in output messages. Defaults to <code>false</code>.</li>
79 *
80 * <li><code>org.slf4j.simpleLogger.levelInBrackets</code> - Should the level string be output in brackets? Defaults
81 * to <code>false</code>.</li>
82 *
83 * <li><code>org.slf4j.simpleLogger.warnLevelString</code> - The string value output for the warn level. Defaults
84 * to <code>WARN</code>.</li>
85
86 * </ul>
87 *
88 * <p>In addition to looking for system properties with the names specified above, this implementation also checks for
89 * a class loader resource named <code>"simplelogger.properties"</code>, and includes any matching definitions
90 * from this resource (if it exists).</p>
91 *
92 * <p>With no configuration, the default output includes the relative time in milliseconds, thread name, the level,
93 * logger name, and the message followed by the line separator for the host.  In log4j terms it amounts to the "%r [%t]
94 * %level %logger - %m%n" pattern. </p>
95 * <p>Sample output follows.</p>
96 * <pre>
97 * 176 [main] INFO examples.Sort - Populating an array of 2 elements in reverse order.
98 * 225 [main] INFO examples.SortAlgo - Entered the sort method.
99 * 304 [main] INFO examples.SortAlgo - Dump of integer array:
100 * 317 [main] INFO examples.SortAlgo - Element [0] = 0
101 * 331 [main] INFO examples.SortAlgo - Element [1] = 1
102 * 343 [main] INFO examples.Sort - The next log statement should be an error message.
103 * 346 [main] ERROR examples.SortAlgo - Tried to dump an uninitialized array.
104 *   at org.log4j.examples.SortAlgo.dump(SortAlgo.java:58)
105 *   at org.log4j.examples.Sort.main(Sort.java:64)
106 * 467 [main] INFO  examples.Sort - Exiting main method.
107 * </pre>
108 *
109 * <p>This implementation is heavily inspired by
110 * <a href="http://commons.apache.org/logging/">Apache Commons Logging</a>'s SimpleLog.</p>
111 *
112 * @author Ceki G&uuml;lc&uuml;
113 * @author <a href="mailto:sanders@apache.org">Scott Sanders</a>
114 * @author Rod Waldhoff
115 * @author Robert Burrell Donkin
116 * @author C&eacute;drik LIME
117 */
118public class SimpleLogger extends MarkerIgnoringBase {
119
120    private static final long serialVersionUID = -632788891211436180L;
121    private static final String CONFIGURATION_FILE = "simplelogger.properties";
122
123    private static long START_TIME = System.currentTimeMillis();
124    private static final Properties SIMPLE_LOGGER_PROPS = new Properties();
125
126    private static final int LOG_LEVEL_TRACE = LocationAwareLogger.TRACE_INT;
127    private static final int LOG_LEVEL_DEBUG = LocationAwareLogger.DEBUG_INT;
128    private static final int LOG_LEVEL_INFO = LocationAwareLogger.INFO_INT;
129    private static final int LOG_LEVEL_WARN = LocationAwareLogger.WARN_INT;
130    private static final int LOG_LEVEL_ERROR = LocationAwareLogger.ERROR_INT;
131
132    private static boolean INITIALIZED = false;
133
134    private static int DEFAULT_LOG_LEVEL = LOG_LEVEL_INFO;
135    private static boolean SHOW_DATE_TIME = false;
136    private static String DATE_TIME_FORMAT_STR = null;
137    private static DateFormat DATE_FORMATTER = null;
138    private static boolean SHOW_THREAD_NAME = true;
139    private static boolean SHOW_LOG_NAME = true;
140    private static boolean SHOW_SHORT_LOG_NAME = false;
141    private static String LOG_FILE = "System.err";
142    private static PrintStream TARGET_STREAM = null;
143    private static boolean LEVEL_IN_BRACKETS = false;
144    private static String WARN_LEVEL_STRING = "WARN";
145
146    /** All system properties used by <code>SimpleLogger</code> start with this prefix */
147    public static final String SYSTEM_PREFIX = "org.slf4j.simpleLogger.";
148
149    public static final String DEFAULT_LOG_LEVEL_KEY = SYSTEM_PREFIX + "defaultLogLevel";
150    public static final String SHOW_DATE_TIME_KEY = SYSTEM_PREFIX + "showDateTime";
151    public static final String DATE_TIME_FORMAT_KEY = SYSTEM_PREFIX + "dateTimeFormat";
152    public static final String SHOW_THREAD_NAME_KEY = SYSTEM_PREFIX + "showThreadName";
153    public static final String SHOW_LOG_NAME_KEY = SYSTEM_PREFIX + "showLogName";
154    public static final String SHOW_SHORT_LOG_NAME_KEY = SYSTEM_PREFIX + "showShortLogName";
155    public static final String LOG_FILE_KEY = SYSTEM_PREFIX + "logFile";
156    public static final String LEVEL_IN_BRACKETS_KEY = SYSTEM_PREFIX + "levelInBrackets";
157    public static final String WARN_LEVEL_STRING_KEY = SYSTEM_PREFIX + "warnLevelString";
158
159    public static final String LOG_KEY_PREFIX = SYSTEM_PREFIX + "log.";
160
161    private static String getStringProperty(String name) {
162        String prop = null;
163        try {
164            prop = System.getProperty(name);
165        } catch (SecurityException e) {
166            ; // Ignore
167        }
168        return (prop == null) ? SIMPLE_LOGGER_PROPS.getProperty(name) : prop;
169    }
170
171    private static String getStringProperty(String name, String defaultValue) {
172        String prop = getStringProperty(name);
173        return (prop == null) ? defaultValue : prop;
174    }
175
176    private static boolean getBooleanProperty(String name, boolean defaultValue) {
177        String prop = getStringProperty(name);
178        return (prop == null) ? defaultValue : "true".equalsIgnoreCase(prop);
179    }
180
181    // Initialize class attributes.
182    // Load properties file, if found.
183    // Override with system properties.
184    static void init() {
185        INITIALIZED = true;
186        loadProperties();
187
188        String defaultLogLevelString = getStringProperty(DEFAULT_LOG_LEVEL_KEY, null);
189        if (defaultLogLevelString != null)
190            DEFAULT_LOG_LEVEL = stringToLevel(defaultLogLevelString);
191
192        SHOW_LOG_NAME = getBooleanProperty(SHOW_LOG_NAME_KEY, SHOW_LOG_NAME);
193        SHOW_SHORT_LOG_NAME = getBooleanProperty(SHOW_SHORT_LOG_NAME_KEY, SHOW_SHORT_LOG_NAME);
194        SHOW_DATE_TIME = getBooleanProperty(SHOW_DATE_TIME_KEY, SHOW_DATE_TIME);
195        SHOW_THREAD_NAME = getBooleanProperty(SHOW_THREAD_NAME_KEY, SHOW_THREAD_NAME);
196        DATE_TIME_FORMAT_STR = getStringProperty(DATE_TIME_FORMAT_KEY, DATE_TIME_FORMAT_STR);
197        LEVEL_IN_BRACKETS = getBooleanProperty(LEVEL_IN_BRACKETS_KEY, LEVEL_IN_BRACKETS);
198        WARN_LEVEL_STRING = getStringProperty(WARN_LEVEL_STRING_KEY, WARN_LEVEL_STRING);
199
200        LOG_FILE = getStringProperty(LOG_FILE_KEY, LOG_FILE);
201        TARGET_STREAM = computeTargetStream(LOG_FILE);
202
203        if (DATE_TIME_FORMAT_STR != null) {
204            try {
205                DATE_FORMATTER = new SimpleDateFormat(DATE_TIME_FORMAT_STR);
206            } catch (IllegalArgumentException e) {
207                Util.report("Bad date format in " + CONFIGURATION_FILE + "; will output relative time", e);
208            }
209        }
210    }
211
212    private static PrintStream computeTargetStream(String logFile) {
213        if ("System.err".equalsIgnoreCase(logFile))
214            return System.err;
215        else if ("System.out".equalsIgnoreCase(logFile)) {
216            return System.out;
217        } else {
218            try {
219                FileOutputStream fos = new FileOutputStream(logFile);
220                PrintStream printStream = new PrintStream(fos);
221                return printStream;
222            } catch (FileNotFoundException e) {
223                Util.report("Could not open [" + logFile + "]. Defaulting to System.err", e);
224                return System.err;
225            }
226        }
227    }
228
229    private static void loadProperties() {
230        // Add props from the resource simplelogger.properties
231        InputStream in = AccessController.doPrivileged(new PrivilegedAction<InputStream>() {
232            public InputStream run() {
233                ClassLoader threadCL = Thread.currentThread().getContextClassLoader();
234                if (threadCL != null) {
235                    return threadCL.getResourceAsStream(CONFIGURATION_FILE);
236                } else {
237                    return ClassLoader.getSystemResourceAsStream(CONFIGURATION_FILE);
238                }
239            }
240        });
241        if (null != in) {
242            try {
243                SIMPLE_LOGGER_PROPS.load(in);
244                in.close();
245            } catch (java.io.IOException e) {
246                // ignored
247            }
248        }
249    }
250
251    /** The current log level */
252    protected int currentLogLevel = LOG_LEVEL_INFO;
253    /** The short name of this simple log instance */
254    private transient String shortLogName = null;
255
256    /**
257     * Package access allows only {@link SimpleLoggerFactory} to instantiate
258     * SimpleLogger instances.
259     */
260    SimpleLogger(String name) {
261        if (!INITIALIZED) {
262            init();
263        }
264        this.name = name;
265
266        String levelString = recursivelyComputeLevelString();
267        if (levelString != null) {
268            this.currentLogLevel = stringToLevel(levelString);
269        } else {
270            this.currentLogLevel = DEFAULT_LOG_LEVEL;
271        }
272    }
273
274    String recursivelyComputeLevelString() {
275        String tempName = name;
276        String levelString = null;
277        int indexOfLastDot = tempName.length();
278        while ((levelString == null) && (indexOfLastDot > -1)) {
279            tempName = tempName.substring(0, indexOfLastDot);
280            levelString = getStringProperty(LOG_KEY_PREFIX + tempName, null);
281            indexOfLastDot = String.valueOf(tempName).lastIndexOf(".");
282        }
283        return levelString;
284    }
285
286    private static int stringToLevel(String levelStr) {
287        if ("trace".equalsIgnoreCase(levelStr)) {
288            return LOG_LEVEL_TRACE;
289        } else if ("debug".equalsIgnoreCase(levelStr)) {
290            return LOG_LEVEL_DEBUG;
291        } else if ("info".equalsIgnoreCase(levelStr)) {
292            return LOG_LEVEL_INFO;
293        } else if ("warn".equalsIgnoreCase(levelStr)) {
294            return LOG_LEVEL_WARN;
295        } else if ("error".equalsIgnoreCase(levelStr)) {
296            return LOG_LEVEL_ERROR;
297        }
298        // assume INFO by default
299        return LOG_LEVEL_INFO;
300    }
301
302    /**
303     * This is our internal implementation for logging regular (non-parameterized)
304     * log messages.
305     *
306     * @param level   One of the LOG_LEVEL_XXX constants defining the log level
307     * @param message The message itself
308     * @param t       The exception whose stack trace should be logged
309     */
310    private void log(int level, String message, Throwable t) {
311        if (!isLevelEnabled(level)) {
312            return;
313        }
314
315        StringBuilder buf = new StringBuilder(32);
316
317        // Append date-time if so configured
318        if (SHOW_DATE_TIME) {
319            if (DATE_FORMATTER != null) {
320                buf.append(getFormattedDate());
321                buf.append(' ');
322            } else {
323                buf.append(System.currentTimeMillis() - START_TIME);
324                buf.append(' ');
325            }
326        }
327
328        // Append current thread name if so configured
329        if (SHOW_THREAD_NAME) {
330            buf.append('[');
331            buf.append(Thread.currentThread().getName());
332            buf.append("] ");
333        }
334
335        if (LEVEL_IN_BRACKETS)
336            buf.append('[');
337
338        // Append a readable representation of the log level
339        switch (level) {
340        case LOG_LEVEL_TRACE:
341            buf.append("TRACE");
342            break;
343        case LOG_LEVEL_DEBUG:
344            buf.append("DEBUG");
345            break;
346        case LOG_LEVEL_INFO:
347            buf.append("INFO");
348            break;
349        case LOG_LEVEL_WARN:
350            buf.append(WARN_LEVEL_STRING);
351            break;
352        case LOG_LEVEL_ERROR:
353            buf.append("ERROR");
354            break;
355        }
356        if (LEVEL_IN_BRACKETS)
357            buf.append(']');
358        buf.append(' ');
359
360        // Append the name of the log instance if so configured
361        if (SHOW_SHORT_LOG_NAME) {
362            if (shortLogName == null)
363                shortLogName = computeShortName();
364            buf.append(String.valueOf(shortLogName)).append(" - ");
365        } else if (SHOW_LOG_NAME) {
366            buf.append(String.valueOf(name)).append(" - ");
367        }
368
369        // Append the message
370        buf.append(message);
371
372        write(buf, t);
373
374    }
375
376    void write(StringBuilder buf, Throwable t) {
377        TARGET_STREAM.println(buf.toString());
378        if (t != null) {
379            t.printStackTrace(TARGET_STREAM);
380        }
381        TARGET_STREAM.flush();
382    }
383
384    private String getFormattedDate() {
385        Date now = new Date();
386        String dateText;
387        synchronized (DATE_FORMATTER) {
388            dateText = DATE_FORMATTER.format(now);
389        }
390        return dateText;
391    }
392
393    private String computeShortName() {
394        return name.substring(name.lastIndexOf(".") + 1);
395    }
396
397    /**
398     * For formatted messages, first substitute arguments and then log.
399     *
400     * @param level
401     * @param format
402     * @param arg1
403     * @param arg2
404     */
405    private void formatAndLog(int level, String format, Object arg1, Object arg2) {
406        if (!isLevelEnabled(level)) {
407            return;
408        }
409        FormattingTuple tp = MessageFormatter.format(format, arg1, arg2);
410        log(level, tp.getMessage(), tp.getThrowable());
411    }
412
413    /**
414     * For formatted messages, first substitute arguments and then log.
415     *
416     * @param level
417     * @param format
418     * @param arguments a list of 3 ore more arguments
419     */
420    private void formatAndLog(int level, String format, Object... arguments) {
421        if (!isLevelEnabled(level)) {
422            return;
423        }
424        FormattingTuple tp = MessageFormatter.arrayFormat(format, arguments);
425        log(level, tp.getMessage(), tp.getThrowable());
426    }
427
428    /**
429     * Is the given log level currently enabled?
430     *
431     * @param logLevel is this level enabled?
432     */
433    protected boolean isLevelEnabled(int logLevel) {
434        // log level are numerically ordered so can use simple numeric
435        // comparison
436        return (logLevel >= currentLogLevel);
437    }
438
439    /** Are {@code trace} messages currently enabled? */
440    public boolean isTraceEnabled() {
441        return isLevelEnabled(LOG_LEVEL_TRACE);
442    }
443
444    /**
445     * A simple implementation which logs messages of level TRACE according
446     * to the format outlined above.
447     */
448    public void trace(String msg) {
449        log(LOG_LEVEL_TRACE, msg, null);
450    }
451
452    /**
453     * Perform single parameter substitution before logging the message of level
454     * TRACE according to the format outlined above.
455     */
456    public void trace(String format, Object param1) {
457        formatAndLog(LOG_LEVEL_TRACE, format, param1, null);
458    }
459
460    /**
461     * Perform double parameter substitution before logging the message of level
462     * TRACE according to the format outlined above.
463     */
464    public void trace(String format, Object param1, Object param2) {
465        formatAndLog(LOG_LEVEL_TRACE, format, param1, param2);
466    }
467
468    /**
469     * Perform double parameter substitution before logging the message of level
470     * TRACE according to the format outlined above.
471     */
472    public void trace(String format, Object... argArray) {
473        formatAndLog(LOG_LEVEL_TRACE, format, argArray);
474    }
475
476    /** Log a message of level TRACE, including an exception. */
477    public void trace(String msg, Throwable t) {
478        log(LOG_LEVEL_TRACE, msg, t);
479    }
480
481    /** Are {@code debug} messages currently enabled? */
482    public boolean isDebugEnabled() {
483        return isLevelEnabled(LOG_LEVEL_DEBUG);
484    }
485
486    /**
487     * A simple implementation which logs messages of level DEBUG according
488     * to the format outlined above.
489     */
490    public void debug(String msg) {
491        log(LOG_LEVEL_DEBUG, msg, null);
492    }
493
494    /**
495     * Perform single parameter substitution before logging the message of level
496     * DEBUG according to the format outlined above.
497     */
498    public void debug(String format, Object param1) {
499        formatAndLog(LOG_LEVEL_DEBUG, format, param1, null);
500    }
501
502    /**
503     * Perform double parameter substitution before logging the message of level
504     * DEBUG according to the format outlined above.
505     */
506    public void debug(String format, Object param1, Object param2) {
507        formatAndLog(LOG_LEVEL_DEBUG, format, param1, param2);
508    }
509
510    /**
511     * Perform double parameter substitution before logging the message of level
512     * DEBUG according to the format outlined above.
513     */
514    public void debug(String format, Object... argArray) {
515        formatAndLog(LOG_LEVEL_DEBUG, format, argArray);
516    }
517
518    /** Log a message of level DEBUG, including an exception. */
519    public void debug(String msg, Throwable t) {
520        log(LOG_LEVEL_DEBUG, msg, t);
521    }
522
523    /** Are {@code info} messages currently enabled? */
524    public boolean isInfoEnabled() {
525        return isLevelEnabled(LOG_LEVEL_INFO);
526    }
527
528    /**
529     * A simple implementation which logs messages of level INFO according
530     * to the format outlined above.
531     */
532    public void info(String msg) {
533        log(LOG_LEVEL_INFO, msg, null);
534    }
535
536    /**
537     * Perform single parameter substitution before logging the message of level
538     * INFO according to the format outlined above.
539     */
540    public void info(String format, Object arg) {
541        formatAndLog(LOG_LEVEL_INFO, format, arg, null);
542    }
543
544    /**
545     * Perform double parameter substitution before logging the message of level
546     * INFO according to the format outlined above.
547     */
548    public void info(String format, Object arg1, Object arg2) {
549        formatAndLog(LOG_LEVEL_INFO, format, arg1, arg2);
550    }
551
552    /**
553     * Perform double parameter substitution before logging the message of level
554     * INFO according to the format outlined above.
555     */
556    public void info(String format, Object... argArray) {
557        formatAndLog(LOG_LEVEL_INFO, format, argArray);
558    }
559
560    /** Log a message of level INFO, including an exception. */
561    public void info(String msg, Throwable t) {
562        log(LOG_LEVEL_INFO, msg, t);
563    }
564
565    /** Are {@code warn} messages currently enabled? */
566    public boolean isWarnEnabled() {
567        return isLevelEnabled(LOG_LEVEL_WARN);
568    }
569
570    /**
571     * A simple implementation which always logs messages of level WARN according
572     * to the format outlined above.
573     */
574    public void warn(String msg) {
575        log(LOG_LEVEL_WARN, msg, null);
576    }
577
578    /**
579     * Perform single parameter substitution before logging the message of level
580     * WARN according to the format outlined above.
581     */
582    public void warn(String format, Object arg) {
583        formatAndLog(LOG_LEVEL_WARN, format, arg, null);
584    }
585
586    /**
587     * Perform double parameter substitution before logging the message of level
588     * WARN according to the format outlined above.
589     */
590    public void warn(String format, Object arg1, Object arg2) {
591        formatAndLog(LOG_LEVEL_WARN, format, arg1, arg2);
592    }
593
594    /**
595     * Perform double parameter substitution before logging the message of level
596     * WARN according to the format outlined above.
597     */
598    public void warn(String format, Object... argArray) {
599        formatAndLog(LOG_LEVEL_WARN, format, argArray);
600    }
601
602    /** Log a message of level WARN, including an exception. */
603    public void warn(String msg, Throwable t) {
604        log(LOG_LEVEL_WARN, msg, t);
605    }
606
607    /** Are {@code error} messages currently enabled? */
608    public boolean isErrorEnabled() {
609        return isLevelEnabled(LOG_LEVEL_ERROR);
610    }
611
612    /**
613     * A simple implementation which always logs messages of level ERROR according
614     * to the format outlined above.
615     */
616    public void error(String msg) {
617        log(LOG_LEVEL_ERROR, msg, null);
618    }
619
620    /**
621     * Perform single parameter substitution before logging the message of level
622     * ERROR according to the format outlined above.
623     */
624    public void error(String format, Object arg) {
625        formatAndLog(LOG_LEVEL_ERROR, format, arg, null);
626    }
627
628    /**
629     * Perform double parameter substitution before logging the message of level
630     * ERROR according to the format outlined above.
631     */
632    public void error(String format, Object arg1, Object arg2) {
633        formatAndLog(LOG_LEVEL_ERROR, format, arg1, arg2);
634    }
635
636    /**
637     * Perform double parameter substitution before logging the message of level
638     * ERROR according to the format outlined above.
639     */
640    public void error(String format, Object... argArray) {
641        formatAndLog(LOG_LEVEL_ERROR, format, argArray);
642    }
643
644    /** Log a message of level ERROR, including an exception. */
645    public void error(String msg, Throwable t) {
646        log(LOG_LEVEL_ERROR, msg, t);
647    }
648}
649