Skip to main content

No project description provided

Project description

schwarzlog

Library to add some missing functionality in Python's logging module.

$ pip install schwarzlog

Caveat: Most functionality is currently not documented. I'll to write some docs going forward, though.

Motivation / Background

logging is often helpful to find problems in deployed code.

However Python's logging infrastructure is a bit annoying at times. For example if a library starts logging data but the application/unit test did not configure the logging infrastructure Python will emit warnings. If the library supports conditional logging (e.g. passing a flag if it should use logging to avoid the "no logging handler installed" issue mentioned above) this might complicate the library code (due to "is logging enabled" checks).

Also I find it a bit cumbersome to test Python's logging in libraries because one has to install global handlers (and clean up when the test is done!).

This library should solve all these problems with a helper function:

  • It can just return a new logger with a specified name.
  • If logging should be disabled entirely it just returns a fake logger which will discard all messages. The application doesn't have to be aware of this and no global state will be changed.
  • The caller can also pass a pre-configured logger (e.g. to test the emitted log messages easily or to use customized logging mechanisms).

Since its inception this library was extended with a few useful helper functions and specialized logging classes.

CallbackLogger

A Logger-like class which can trigger a additional callback in addition to passing a log message through the logging infrastructure. I'm using this to ensure severe problems logged by lower-level libraries will be displayed in the UI. If you set merge_arguments = True the callback only gets the final message (as str), otherwise it'll get the logging.LogRecord.

Usage:

import logging
from schwarz.log_utils import CallbackLogger

_l = logging.getLogger('foo')
logged_msgs = []
cb = logged_msgs.append
log = CallbackLogger(log=_l, callback=cb, callback_minlevel=logging.ERROR, merge_arguments=True)
log.info('info message')
log.error('error message')
logged_msgs == ['error message']

ForwardingLogger

This logger forwards messages above a certain level (by default: all messages) to a configured parent logger. Optionally it can prepend the configured forward_prefix to all forwarded log messages. forward_suffix works like forward_prefix but appends some string.

This can be helpful if you need to log contextualized messages. For example you could log detailed messages related to a specific file in "imgfile.log" but you want more important messages (e.g. warnings, errors) in another log file used by your application. In that scenario you can quickly spot problems in your main log file while detailed data is available in separate log files.

Python's default logging module can not handle this because:

  • A Logger's log level is only applied for messages emitted directly on that logger (not for propagated log messages), see this blog post by Marius Gedminas.
  • Adding a log prefix only for certain loggers can only by done by duplicating handler configuration. Python's handlers are quite basic so if the duplicated handlers access a shared resource (e.g. a log file) Python will open it twice (which causes data loss if mode='w' is used).

Usage:

import logging
from schwarz.log_utils import get_logger, ForwardingLogger

parent_logger = logging.getLogger('foo')
log = ForwardingLogger(
    forward_to=parent_logger,
    forward_prefix='[ABC] ',
    forward_minlevel=logging.INFO
)
log.info('foo')
# parent_logger sees a log message like "[ABC] foo"

Support for writing tests

The library also contains some helpers to ease writing logging-related tests.

import logging
from schwarz.log_utils.testutils import *

# "lc" works a bit similar to a LogCapture instance
log, lc = build_collecting_logger()
log.info('foo')
log.debug('bar')

assert_did_log_message(lc, 'foo')
# this raises an AssertionError as "foo" was logged with INFO
assert_did_log_message(lc, 'foo', level=logging.DEBUG)

lr = assert_did_log_message(lc, 'foo', level=logging.INFO)
# you can also inspect the actual "LogRecord" instance "lr" if you need to

assert_no_log_messages(lc, min_level=logging.WARN)

Changes

0.7.0 (2024-08-06)

  • drop support for Python 2

0.6.2 (2022-05-25)

  • assert_did_log_message(…) now returns the LogRecord instance which can be used by the caller for more detailled checks.
  • ForwardingLogger now also forwards .exc_info correctly so that the main logger can also log exceptions.

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

schwarzlog-0.7.0.tar.gz (9.3 kB view details)

Uploaded Source

Built Distribution

schwarzlog-0.7.0-py3-none-any.whl (10.6 kB view details)

Uploaded Python 3

File details

Details for the file schwarzlog-0.7.0.tar.gz.

File metadata

  • Download URL: schwarzlog-0.7.0.tar.gz
  • Upload date:
  • Size: 9.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.12.4

File hashes

Hashes for schwarzlog-0.7.0.tar.gz
Algorithm Hash digest
SHA256 69788fca4bed37f7ca398373f799f831d293e8a84b80805c686c63ccf48b517f
MD5 6e9e92bfe77db061c464ec59e0c2a0c9
BLAKE2b-256 49a9a9a2f6cb9b145992e41a5ba6fbe253745447ddc1cc83e7c373e5202794bf

See more details on using hashes here.

File details

Details for the file schwarzlog-0.7.0-py3-none-any.whl.

File metadata

  • Download URL: schwarzlog-0.7.0-py3-none-any.whl
  • Upload date:
  • Size: 10.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.12.4

File hashes

Hashes for schwarzlog-0.7.0-py3-none-any.whl
Algorithm Hash digest
SHA256 25ea01385840223c568aeea6e04b7cef70fe684b369ab0d527293f59bf450983
MD5 e70e635002c58d69f85a920f95acae50
BLAKE2b-256 db036b3a86e7bb68ab3cbd472e7fe2a3fee86afe185ba54c31c74f2c4ffedba8

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