A Logging System for Python Home

Download

Copyright & License

Recent Changes

API Documentation

"Oh, I'm a lumberjack and I'm okay..." (Monty Python, The Lumberjack
Song)

Table of Contents

Abstract

Motivation

Influences

A Simple Example

Control Flow

Levels

Loggers

Handlers

Formatters

Filters

Configuration

The GUI Configurator

Case Scenarios

Thread Safety

On-The-Fly Reconfiguration

Module-Level Convenience Functions

Performance

Implementation Status

Acknowledgements

Still To Do

Download and Installation

Change History

Copyright and License

Abstract

There was a need for a standard logging system in Python, as comprehensively documented
in PEP 282 and enthusiastically
endorsed by the BDFL in the Parade
of the PEPs. By a happy coincidence, the package described here was already in
development and fairly close in intent and design to the description in the aforementioned
PEP, borrowing as it did heavily from JSR-47 (now JDK 1.4's java.util.logging package) and
log4j. This page describes it
in more detail. As I have tweaked the package to meet comments on PEP 282, I have structured
this page in the same way as the original PEP. This package is now part of Python 2.3, but if
you have an earlier version of Python, you can download the package from here and use it
with Python versions between 1.5.2 and 2.2.x.

Motivation

The Python community has been incredibly helpful to me, a relative newcomer to the
language. Python and its community has certainly saved me much time and effort, and it
seems appropriate to give something back to the community by offering up this package for
people to try. Any feedback will be
gratefully accepted.

Influences

This package owes its greatest debt to Apache log4j. Due notice was also taken
of log4j's comprehensive critique (no longer online) of JSR47. This package bears a close
resemblance to log4j, but is not a close translation. I have attempted to be more minimalist
(and hopefully more Pythonic) in my approach. You be the judge!

A Simple Example

Using the package doesn't get much simpler. It is packaged as a standard Python package
called (unsurprisingly) logging. You just need to import logging
and you're ready to go. Minimal example:

# --- app.py --------------------------------------------------------------------
import logging
logging.warn("Hello")
logging.error("Still here...")
logging.warn("Goodbye")

When you run app.py, the results are:

WARNING:root:Hello
ERROR:root:Still here...
WARNING:root:Goodbye

Don't worry about the format of the output - it's all configurable. Here's a slightly
more involved example; if you've just looked at PEP 282 you will probably get a feeling of
dej� vu. (This is intentional.)

# --- mymodule.py --------------------------------------------------------------------
import logging
log = logging.getLogger("MyModule")
def doIt():
log.debug("doin' stuff")
#do stuff...but suppose an error occurs?
raise TypeError, "bogus type error for testing"
# --- myapp.py -----------------------------------------------------------------------
import logging, mymodule
logging.basicConfig()
log = logging.getLogger("MyApp")
log.setLevel(logging.DEBUG) #set verbosity to show all messages of severity >= DEBUG
log.info("Starting my app")
try:
mymodule.doIt()
except Exception, e:
log.exception("There was a problem.")
log.info("Ending my app")

When you run myapp.py, the results are:

INFO:MyApp:Starting my app
ERROR:MyApp:There was a problem.
Traceback (most recent call last):
File "myapp.py", line 9, in ?
mymodule.doIt()
File "mymodule.py", line 7, in doIt
raise TypeError, "Bogus type error for testing"
TypeError: Bogus type error for testing
INFO:MyApp:Ending my app

But don't worry - the above output is not hardcoded into the package. It's just an
example of what you can do with very little work. As you can see, exceptions are handled
as one would expect.

Control Flow

The package pretty much matches the PEP regarding control flow. The user of the package
makes logging calls on instances of Logger, which are organized into a
hierarchy based on a "dotted name" namespace. This hierarchy is embodied in an
encapsulated singleton Manager instance (which can be ignored by users of the
package, for most purposes). Based on the type of logging call and the logging
configuration (see below), the call may be passed through a set of Filter
instances to decide whether it should be dropped. If not, then the logger consults a set
of Handler instances which are associated with it, and asks each handler
instance to "handle" the logging event. By default, the system moves up the
namespace hierarchy and invokes handlers on all loggers at or above the level of the
logger on which the logging call was made. (You can override this by setting a logger's
"propagate" attribute to 0 - no traversal up the hierarchy is made from such a
logger. But I'm getting ahead of myself...)

Handlers are passed LogRecord instances which (should) contain all the
information we're interested in logging. Handlers, too, can invoke filters to determine
whether a record should be dropped. If not, then the handler takes a handler-specific
action to actually log the record to a file, the console or whatever.

Levels

The following levels are implemented by default:

DEBUG
INFO
WARNING
ERROR
CRITICAL

The CRITICAL level replaces the earlier FATAL level. You can
use either (for now), but CRITICAL is preferred since FATAL
implies that the application is about to terminate. This is not true for many systems
which use logging - for example, a Web server application which encounters a CRITICAL
condition (e.g. running out of resources) will still try to keep going as best it can.

FATAL (and the corresponding fatal() methods) may be removed
in future versions of the package. Currently, CRITICAL is synonymous with FATAL
and critical() methods are synonymous with fatal().

Exceptions logged via exception() use the ERROR level for
logging. If it is desired to log exception information with arbitrary logging levels, this
can be done by passing a keyword argument exc_info with a true value to the
logging methods (see the API documentation for more details).

The levels are not deeply hardcoded into the package - the number of levels, their
numeric values and their textual representation are all configurable. The above levels
represent the experience of the log4j community and so are provided as the default levels
for users who do not have very specific requirements in this area.

The example script log_test4.py shows the use of bespoke logging levels
(as well as filtering by level at logger and handler, as well as use of filter classes).

发表回复