Skip to main content

Concurrent logging handler (drop-in replacement for RotatingFileHandler) Python 2.6+

Project description

Overview

This module provides an additional log handler for Python’s standard logging package (PEP 282). This handler will write log events to log file which is rotated when the log file reaches a certain size. Multiple processes can safely write to the same log file concurrently.

Details

The ConcurrentRotatingFileHandler class is a drop-in replacement for Python’s standard log handler RotatingFileHandler. This module uses file locking so that multiple processes can concurrently log to a single file without dropping or clobbering log events. This module provides a file rotation scheme like with RotatingFileHanler. Extra care is taken to ensure that logs can be safely rotated before the rotation process is started. (This module works around the file rename issue with RotatingFileHandler on Windows, where a rotation failure means that all subsequent log events are dropped).

This module attempts to preserve log records at all cost. This means that log files will grow larger than the specified maximum (rotation) size. So if disk space is tight, you may want to stick with RotatingFileHandler, which will strictly adhere to the maximum file size.

If you have multiple instances of a script (or multiple scripts) all running at the same time and writing to the same log file, then all of the scripts should be using ConcurrentRotatingFileHandler. You should not attempt to mix and match RotatingFileHandler and ConcurrentRotatingFileHandler.

This package bundles portalocker to deal with file locking. Please be aware that portalocker only supports Unix (posix) an NT platforms at this time, and therefore this package only supports those platforms as well.

Installation

Use the following command to install this package:

pip install ConcurrentLogHandler

If you are installing from source, you can use:

python setup.py install

Examples

Simple Example

Here is a example demonstrating how to use this module directly (from within Python code):

from logging import getLogger, INFO
from cloghandler import ConcurrentRotatingFileHandler
import os

log = getLogger()
# Use an absolute path to prevent file rotation trouble.
logfile = os.path.abspath("mylogfile.log")
# Rotate log after reaching 512K, keep 5 old copies.
rotateHandler = ConcurrentRotatingFileHandler(logfile, "a", 512*1024, 5)
log.addHandler(rotateHandler)
log.setLevel(INFO)

log.info("Here is a very exciting log message, just for you")

Automatic fallback example

If you are distributing your code and you are unsure if the ConcurrentLogHandler package has been installed everywhere your code will run, Python makes it easy to gracefully fallback to the built in RotatingFileHandler, here is an example:

try:
    from cloghandler import ConcurrentRotatingFileHandler as RFHandler
except ImportError:
    # Next 2 lines are optional:  issue a warning to the user
    from warnings import warn
    warn("ConcurrentLogHandler package not installed.  Using builtin log handler")
    from logging.handlers import RotatingFileHandler as RFHandler

log = getLogger()
rotateHandler = RFHandler("/path/to/mylogfile.log", "a", 1048576, 15)
log.addHandler(rotateHandler)

Config file example

This example shows you how to use this log handler with the logging config file parser. This allows you to keep your logging configuration code separate from your application code.

Example config file: logging.ini:

[loggers]
keys=root

[handlers]
keys=hand01

[formatters]
keys=form01

[logger_root]
level=NOTSET
handlers=hand01

[handler_hand01]
class=handlers.ConcurrentRotatingFileHandler
level=NOTSET
formatter=form01
args=("rotating.log", "a", 512*1024, 5)

[formatter_form01]
format=%(asctime)s %(levelname)s %(message)s

Example Python code: app.py:

import logging, logging.config
import cloghandler

logging.config.fileConfig("logging.ini")
log = logging.getLogger()
log.info("Here is a very exciting log message, just for you")

Change Log

  • 0.9.1: Bug fixes - LP Bug 1199332 and LP Bug 1199333.
    • More gracefully handle out of disk space scenarios. Prevent release() from throwing an exception.

    • Handle logging.shutdown() in Python 2.7+. Close the lock file stream via close().

    • Big thanks to Dan Callaghan for forwarding these issues and patches.

  • 0.9.0: Now requires Python 2.6+
    • Revamp file opening/closing and file-locking internals (inspired by feedback from Vinay Sajip.)

    • Add the ‘delay’ parameter (delayed log file opening) to better match the core logging functionality in more recent version of Python.

    • For anyone still using Python 2.3-2.5, please use the latest 0.8.x release

  • 0.8.6: Fixed packaging bug with test script
    • Fix a small packaging bug from the 0.8.5 release. (Thanks to Björn Häuser for bringing this to my attention.)

    • Updated stresstest.py to always use the correct python version when launching sub-processes instead of the system’s default “python”.

  • 0.8.5: Fixed ValueError: I/O operation on closed file
  • 0.8.4: Fixed lock-file naming issue
    • Resolved a minor issue where lock-files would be improperly named if the log file contained “.log” in the middle of the log name. For example, if you log file was “/var/log/mycompany.logging.mysource.log”, the lock file would be named “/var/log/mycompany.ging.mysource.lock”, which is not correct. Thanks to Dirk Rothe for pointing this out. Since this introduce a slight lock-file behavior difference, make sure all concurrent writers are updated to 0.8.4 at the same time if this issue effects you.

    • Updated ez_setup.py to 0.6c11

  • 0.8.3: Fixed a log file rotation bug and updated docs
    • Fixed a bug that happens after log rotation when multiple processes are witting to the same log file. Each process ends up writing to their own log file (“log.1” or “log.2” instead of “log”). The fix is simply to reopen the log file and check the size again. I do not believe this bug results in data loss; however, this certainly was not the desired behavior. (A big thanks goes to Oliver Tonnhofer for finding, documenting, and providing a patch for this bug.)

    • Cleanup the docs. (aka “the page you are reading right now”) I fixed some silly mistakes and typos… who writes this stuff?

  • 0.8.2: Minor bug fix release (again)
    • Found and resolved another issue with older logging packages that do not support encoding.

  • 0.8.1: Minor bug fix release
    • Now importing “codecs” directly; I found some slight differences in the logging module in different Python 2.4.x releases that caused the module to fail to load.

  • 0.8.0: Minor feature release
    • Add better support for using logging.config.fileConfig(). This class is now available using class=handlers.ConcurrentRotatingFileHandler.

    • Minor changes in how the filename parameter is handled when given a relative path.

  • 0.7.4: Minor bug fix
    • Fixed a typo in the package description (incorrect class name)

    • Added a change log; which you are reading now.

    • Fixed the close() method to no longer assume that stream is still open.

To-do

  • This module has had minimal testing in a multi-threaded process. I see no reason why this should be an issue, but no stress-testing has been done in a threaded situation. If this is important to you, you could always add threading support to the stresstest.py script and send me the patch.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

ConcurrentLogHandler-0.9.1.tar.gz (25.2 kB view details)

Uploaded Source

Built Distributions

ConcurrentLogHandler-0.9.1-py3.2.egg (27.5 kB view details)

Uploaded Source

ConcurrentLogHandler-0.9.1-py2.7.egg (27.3 kB view details)

Uploaded Source

ConcurrentLogHandler-0.9.1-py2.6.egg (27.4 kB view details)

Uploaded Source

File details

Details for the file ConcurrentLogHandler-0.9.1.tar.gz.

File metadata

File hashes

Hashes for ConcurrentLogHandler-0.9.1.tar.gz
Algorithm Hash digest
SHA256 8225a590fd4194c413fa26675bde5f6b80ad79e4182d5876ba3e264f77755918
MD5 9609ecc4c269ac43f0837d89f12554c3
BLAKE2b-256 fde50dc4f256bcc6484d454006b02f33263b20f762a433741b29d53875e0d763

See more details on using hashes here.

File details

Details for the file ConcurrentLogHandler-0.9.1-py3.2.egg.

File metadata

File hashes

Hashes for ConcurrentLogHandler-0.9.1-py3.2.egg
Algorithm Hash digest
SHA256 4ccae08b7f9b3257de35f847e2de8629c00c2075f8ce66db8ed06d7657e2eeae
MD5 a2d7c5db0b038d76ab491763e21d9de1
BLAKE2b-256 e3f675b3cc0fea08fb09d8ffa9901b716c164ebd55c13e301a08e09bd8d5ba81

See more details on using hashes here.

File details

Details for the file ConcurrentLogHandler-0.9.1-py2.7.egg.

File metadata

File hashes

Hashes for ConcurrentLogHandler-0.9.1-py2.7.egg
Algorithm Hash digest
SHA256 aa608aa0ce32d86d2061dec91cd58a2a367f97110851529d2aa6ebf96d9dcd4d
MD5 784d26d945417274a8ba92c5c5a8d273
BLAKE2b-256 80dfac2a81bb6cdf54737708f45f226ed914a9275dbcc4e2cd01010312aa6e86

See more details on using hashes here.

File details

Details for the file ConcurrentLogHandler-0.9.1-py2.6.egg.

File metadata

File hashes

Hashes for ConcurrentLogHandler-0.9.1-py2.6.egg
Algorithm Hash digest
SHA256 5d199eecc23751ab1f705826660f733c1090f62789f3e3c44296e706fc75b547
MD5 b7550fcb21ad24a41e695a7fac4d4110
BLAKE2b-256 9833b5965e6476bb4d1e0c2d07214a7d524dbcf5b56cfd136896afac6b620d83

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page