Logging convenience routines.
Project description
Logging convenience routines.
Latest release 20250323: setup_logging: default supplant_root_logger based on ttyness of the main log, update supplant logic in LogInfo.apply.
The logging package is very useful, but a little painful to use. This package provides low impact logging setup and some extremely useful if unconventional context hooks for logging.
The default logging verbosity output format has different defaults
based on whether an output log file is a tty
and whether the environment variable $DEBUG is set, and to what.
On terminals warnings and errors get ANSI colouring.
A mode is available that uses cs.upd for certain log levels.
Log messages dispatched via warning and friends from this module
are automatically prefixed with the current cs.pfx prefix string,
providing automatic message context.
Some examples:
Program initialisation:
from cs.logutils import setup_logging
def main(argv):
cmd = os.path.basename(argv.pop(0))
setup_logging(cmd)
Basic logging from anywhere:
from cs.logutils import info, warning, error
[...]
def some_function(...):
[...]
error("nastiness found! bad value=%r", bad_value)
Module contents:
-
add_logfile(filename, logger=None, mode='a', encoding=None, delay=False, format=None, no_prefix=False): Add aFileHandlerlogging to the specifiedfilename; return the chosen logger and the new handler.Parameters:
logger: if supplied and notNone, add theFileHandlerto thatLogger, otherwise to the root Logger. Ifloggeris a string, calllogging.getLogger(logger)to obtain the logger.mode,encodinganddelay: passed to theFileHandlerinitialiser.format: used to override the handler's default format.no_prefix: if true, do not put thePfxcontext onto the front of the message.
-
critical(msg, *args, **kwargs): Emit a log atlogging.CRITICALlevel with the currentPfxprefix. -
D(msg, *args): Print formatted debug string straight tosys.stderrifD_modeis true, bypassing the logging modules entirely. A quick'n'dirty debug tool. -
debug(msg, *args, **kwargs): Emit a log atlogging.DEBUGlevel with the currentPfxprefix. -
error(msg, *args, **kwargs): Emit a log atlogging.ERRORlevel with the currentPfxprefix. -
exception(msg, *args, **kwargs): Emit an exception log with the currentPfxprefix. -
ifverbose(is_verbose, msg, *args, **kwargs): Conditionally log a message.If
is_verboseisNone, log atVERBOSElevel and rely on the logging setup. Otherwise, ifis_verboseis true, log atINFOlevel. -
infer_logging_level(env_debug=None, environ=None, verbose=None): Infer a logging level from theenv_debug, which by default comes from the environment variable$DEBUG.Usually default to
logging.WARNING, but ifsys.stderris a terminal, default tologging.INFO.Parse the environment variable
$DEBUGas a comma separated list of flags.Examine the in sequence flags to affect the logging level:
- numeric < 1:
logging.WARNING - numeric >= 1 and < 2:
logging.INFO - numeric >= 2:
logging.DEBUG "DEBUG":logging.DEBUG"STATUS":STATUS"INFO":logging.INFO"TRACK":TRACK"WARNING":logging.WARNING"ERROR":logging.ERROR
Return an object with the following attributes:
.level: A logging level..flags: All the words from$DEBUGas separated by commas and uppercased.
- numeric < 1:
-
info(msg, *args, **kwargs): Emit a log atlogging.INFOlevel with the currentPfxprefix. -
log(level, msg, *args, **kwargs): Emit a log at the specified level with the currentPfxprefix. -
logException(exc_type, exc_value, exc_tb): Replacement forsys.excepthookthat reports via thecs.logutilslogging wrappers. -
ClassLoggingState(types.SimpleNamespace)`: A logging setup arranged for conventional UNIX command line use.
LoggingState.__init__(self, cmd=None, main_log=None, format=None, level=None, flags=None, upd_mode=None, ansi_mode=None, trace_mode=None, verbose=None, supplant_root_logger=None):
Prepare the LoggingState for conventional UNIX command
line error messaging.
Amongst other things, the default logger now includes
the cs.pfx prefix in the message.
This function runs in two modes:
- if logging has not been set up, it sets up a root logger
- if the root logger already has handlers,
monkey patch the first handler's formatter to prefix the
cs.pfxstate
Parameters:
cmd: program name, default frombasename(sys.argv[0]).main_log: default logging system. IfNone, the main log will go tosys.stderr; ifmain_logis a string, is it used as a filename to open in append mode; otherwisemain_logshould be a stream suitable for use withlogging.StreamHandler(). The resulting log handler is added to theloggingroot logger.format: the message format formain_log. IfNone, useDEFAULT_PFX_FORMAT_TTYwhenmain_logis a tty or FIFO, otherwiseDEFAULT_PFX_FORMAT.level:main_loglogging level. IfNone, infer a level from the environment usinginfer_logging_level().flags: a string containing debugging flags separated by commas. IfNone, infer the flags from the environment usinginfer_logging_level(). The following flags have meaning:D: set cs.logutils.D_mode to True;TDUMP: attach a signal handler to SIGHUP to do a thread stack dump;TRACE: enable various noisy tracing facilities;UPD,NOUPD: set the default forupd_modeto True or False respectively.upd_mode: a Boolean to activate cs.upd as themain_logmethod; ifNone, set it toTrueifflagscontains 'UPD', otherwise toFalseifflagscontains 'NOUPD', otherwise set it frommain_log.isatty(). A true value causes the root logger to usecs.updfor logging.ansi_mode: ifNone, set it frommain_log.isatty() and not cs.colourise.env_no_color(), which thus honours the$NO_COLORenvironment variable (see https://no-color.org/ for the convention). A true value causes the root logger to colour certain logging levels using ANSI terminal sequences (currently only ifcs.updis used).supplant_root_logger: optional mode to replace the root logger'shandlers[0]logger with our own; if not specified, set if the main log is a ttytrace_mode: ifNone, set it according to the presence of 'TRACE' in flags. Otherwise iftrace_modeis true, set the globalloginfo.trace_leveltologinfo.level; otherwise it defaults tologging.DEBUG.verbose: ifNone, then if stderr is a tty then the log level isINFOotherwiseWARNING. Otherwise, ifverboseis true then the log level isINFOotherwiseWARNING.
LoggingState.apply(self):
Apply this LoggingState to the current logging setup.
loginfo = LoggingState(main_log=<_io.TextIOWrapper name='<stderr>' mode='w' encoding='utf-8'>, level=25, verbose=None, trace_level=10, flags=[], cmd='cs-release', upd=<cs.upd.Upd object at 0x105501940>, upd_mode=True, ansi_mode=True, format='%(pfx)s\: %(message)s', supplant_root_logger=True): A logging setup arranged for conventional UNIX command line use.ClassLogTime`: LogTime is a context manager that logs the elapsed time of the enclosed code. After the run, the field .elapsed contains the elapsed time in seconds.
LogTime.__init__(self, tag, *args, **kwargs):
Set up a LogTime.
Parameters:
tag: label included at the start of the log entryargs: optional array; if not emptyargsis applied totagwith%level: keyword argument specifying a log level for a default log entry, defaultlogging.INFOthreshold: keyword argument specifying minimum time to cause a log, default None (no minimum)warning_level: keyword argument specifying the log level for a warning log entry, defaultlogging.WARNINGwarning_threshold: keyword argument specifying a time which raises the log level towarning_level
-
logTo(filename, logger=None, mode='a', encoding=None, delay=False, format=None, no_prefix=False): Add aFileHandlerlogging to the specifiedfilename; return the chosen logger and the new handler.Parameters:
logger: if supplied and notNone, add theFileHandlerto thatLogger, otherwise to the root Logger. Ifloggeris a string, calllogging.getLogger(logger)to obtain the logger.mode,encodinganddelay: passed to theFileHandlerinitialiser.format: used to override the handler's default format.no_prefix: if true, do not put thePfxcontext onto the front of the message.
-
ClassNullHandler(logging.Handler): AHandler` which discards its requests.
NullHandler.emit(self, record):
Discard the log record.
ClassPfxFormatter(logging.Formatter): A Formatter subclass that has access to the program'scmdandPfx` state.
PfxFormatter.__init__(self, fmt=None, datefmt=None, cmd=None):
Initialise the PfxFormatter.
Parameters:
fmt: format template, default fromDEFAULT_PFX_FORMAT'%(asctime)s %(levelname)s %(pfx)s: %(message)s'. Passed through toFormatter.__init__.datefmt: Passed through toFormatter.__init__.cmd: the "command prefix" made available to format strings. If not set,cs.pfx.cmdis presented.
PfxFormatter.format(self, record):
Set record.cmd and record.pfx
to the global cmd and Pfx context prefix respectively,
then call Formatter.format.
PfxFormatter.patch_formatter(formatter):
Monkey patch an existing Formatter instance
with a format method which prepends the current Pfx prefix.
-
quiet(msg, *args, **kwargs): Emit a log atQUIETlevel with the currentPfxprefix. -
setup_logging(cmd_name=None, **kw): Prepare aLoggingStateand return it. It is also available as the globalcs.logutils.loginfo. Side-effect: setscs.pfx.cmdto this value. -
status(msg, *args, **kwargs): Emit a log atSTATUSlevel with the currentPfxprefix. -
trace(msg, *args, **kwargs): Emit a log message atloginfo.trace_levelwith the currentPfxprefix. -
track(msg, *args, **kwargs): Emit a log atTRACKlevel with the currentPfxprefix. -
upd(msg, *args, **kwargs): If we're using anUpdHandler, update the status line otherwise write an info message.Note that this calls
Upd.outdirectly withmsg%argsand thus does not include the currentPfxprefix. You may well want to use thestatus()function instead. -
ClassUpdHandler(logging.StreamHandler): AStreamHandlersubclass whose.emitmethod uses acs.upd.Upd` for transcription.
UpdHandler.__init__(self, strm=None, upd_level=None, ansi_mode=None, over_handler=None):
Initialise the UpdHandler.
Parameters:
strm: the output stream, defaultsys.stderr.upd_level: the magic logging level which updates the status line viaUpd. Default:STATUS.ansi_mode: ifNone, set fromstrm.isatty(). A true value causes the handler to colour certain logging levels using ANSI terminal sequences.
UpdHandler.emit(self, logrec):
Emit a LogRecord logrec.
For the log level self.upd_level update the status line.
For other levels write a distinct line
to the output stream, possibly colourised.
UpdHandler.flush(self):
Flush the update status.
verbose(msg, *args, **kwargs): Emit a log atVERBOSElevel with the currentPfxprefix.warning(msg, *args, **kwargs): Emit a log atlogging.WARNINGlevel with the currentPfxprefix.with_log(filename, **kw): Context manager to add aLoggerto the output logs temporarily.
Release Log
Release 20250323: setup_logging: default supplant_root_logger based on ttyness of the main log, update supplant logic in LogInfo.apply.
Release 20250306: ansi_mode: use white-on-red instead of red, easier to read - really needs some kind of palette from the environment and a light/dark mode.
Release 20241109:
- setup_logging: make it work with no arguments.
- ifdebug: handle loginfo.level=None.
Release 20241007:
- setup_logging: just amend the existing loginfo if already set up.
- Remove the cs.pfx.cmd side effect from LogState.init, now in setup_logging.
Release 20240923:
setup_logging: accept leading cmd_name for backwards compatibility, reported by Lance Cohen.
Release 20240630:
- New LoggingState class for the computed log state, split out setup_logging() as a little stub.
- Drop func_wrap and _ftrace, superceded by cs.debug.trace.
- infer_logging_level: ignore the module.name and module:function_name $DEBUG values, now done by importing cs.debug.
Release 20230212: Late import of cs.upd at need to avoid import loop.
Release 20220531: PfxFormatter.patch_formatter: notice if record.args is not a tuple and do not try to prefix it (for now).
Release 20220530:
- New QUIET log level between TRACK and STATUS, add new quiet() logging call.
- PfxFormatter.patch_formatter: bugfix handling of record.msg,record.args.
Release 20220315: A bit of a hack to prevent double patching a formatter, as when BaseCommand calls a BaseCommand and other circumstances where setup_logging() gets called more than once.
Release 20220227:
- PfxFormatter: new patch_formatter() static method to modify an existing Formatter.
- setup_logging: just use PfxFormatter.patch_formatter on the first handler's formatter if logging is already set up.
Release 20211208: Docstring update.
Release 20210721: UpdHandler.emit: for newline-emitting messages, fall back to new .over_handler if the Upd is disabled.
Release 20210718: setup_logging: new supplant_root_logger=False parameter to pop the existing handler, typical use supplant_root_logger=sys.stderr.isatty().
Release 20210306:
- Default logging level for ttys is now INFO, not STATUS.
- New VERBOSE level below INFO but above DEBUG.
- infer_logging_level: if verbose unspecified, logging=WARNING on a tty and TRACK otherwise, else if verbose, level=VERBOSE, otherwise WARNING.
- Include .verbose in the loginfo.
- New verbose() and ifverbose().
Release 20201021:
- setup_logging: always provide loginfo.upd, being either main_handler.upd if upd_mode otherwise Upd().
- exception(): plumb keyword arguments.
Release 20200729: setup_logging: honour $NO_COLOR if ansi_mode not specified, per https://no-color.org/
Release 20200613:
- LogTime: set .end on exit.
- UpdHandle.emit: fix message colouring logic.
Release 20200521: setup_logging: include the logger in loginfo (presently always the root logger).
Release 20200519: bugfix setup_logging: apparently a LoggingProxy does not have an encoding
Release 20200518:
- Sweeping removal of cs.obj.O, universally supplanted by types.SimpleNamespace.
- Default to logging level TRACK if stderr is a tty instead of logging.INFO.
- New ifverbose function with leading
verboseparameter: if None, log at INFO otherwise if true, log at TRACK, otherwise do not log. - BREAKING: remove global logging_level and trace_level variables, put it all in the global loginfo.
- Make STATUS just below TRACK so that it is above INFO instead of below.
- New status() function for cs.upd messages.
- UpdHandler: treat status_level as special, going directly to Upd.out.
- Improved source line recitation on modern Python.
- Default level if sys.stderr.isatty() now STATUS, not TRACK.
- Some fixes for loginfo initialisation and setting cs.pfx.cmd.
Release 20200229:
- Update for new Upd.without context manager.
- setup_logging: default
upd_modetomain_log.isatty(), was previously False. - Drop UpdHandler.upd method, shadowed by instance attribute, never used.
Release 20190923:
- New
TRACKconstant equal tologging.INFO+5to provide a level higher thanINFO - (which seems unreasonably noisy) and lower than
WARNING - warning for tracking salient events.
- New
track()function to match.
Release 20190220: Improvements to upd_mode.
Release 20190103: Documentation updates.
Release 20190101: Bugfix for @contextmanager usage.
Release 20171030: Assorted fixes from recent module reshuffle. Other small features and cleanups. Drop a couple of unused functions.
Release 20160828: Use "install_requires" instead of "requires" in DISTINFO.
Release 20160827:
- Pfx: import exit handler
- Preliminary per-module and per-function syntax accepted in $DEBUG envvar.
- Improvements to X(), add DP() and XP() prefixed flavours.
- status() function to update terminal status line.
- New X_via_tty global flag: directs X() to tty instead of sys.stderr.
- Assorted other minor improvements.
Release 20150118: metadata updates
Release 20150110: Initial PyPI release.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file cs_logutils-20250323.tar.gz.
File metadata
- Download URL: cs_logutils-20250323.tar.gz
- Upload date:
- Size: 15.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.1
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
720c871ad68f06ff1c1b19ba38d8231068c56b5ae19af3ff0a7df2dba74a299e
|
|
| MD5 |
9025b7588bd81eb30e9d8f9b1069b00e
|
|
| BLAKE2b-256 |
e612d0d0bf3203484426bcef2c8afc9db3635b9af94cf6816e486d5ce6397ec1
|
File details
Details for the file cs_logutils-20250323-py2.py3-none-any.whl.
File metadata
- Download URL: cs_logutils-20250323-py2.py3-none-any.whl
- Upload date:
- Size: 14.9 kB
- Tags: Python 2, Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.1
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
05ade01af232d492ca7680f24dbf90a2b7bbc58a6063f670c7bf67f5f0913fd2
|
|
| MD5 |
73f1acb598c8b051739978f54162f1f7
|
|
| BLAKE2b-256 |
dfbbec227f43dbf396f9b06f33e57f3165907a481c02f8aeecb3ab3f978277f8
|