2019-02-08 10:14:50 +00:00
|
|
|
"""Helpers for logging allowing more advanced logging styles to be used."""
|
|
|
|
import inspect
|
|
|
|
import logging
|
|
|
|
|
|
|
|
|
2019-07-21 16:59:02 +00:00
|
|
|
# mypy: allow-untyped-defs, no-check-untyped-defs
|
|
|
|
|
2019-02-08 10:14:50 +00:00
|
|
|
class KeywordMessage:
|
|
|
|
"""
|
|
|
|
Represents a logging message with keyword arguments.
|
|
|
|
|
|
|
|
Adapted from: https://stackoverflow.com/a/24683360/2267718
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(self, fmt, args, kwargs):
|
|
|
|
"""Initialize a new BraceMessage object."""
|
|
|
|
self._fmt = fmt
|
|
|
|
self._args = args
|
|
|
|
self._kwargs = kwargs
|
|
|
|
|
|
|
|
def __str__(self):
|
|
|
|
"""Convert the object to a string for logging."""
|
|
|
|
return str(self._fmt).format(*self._args, **self._kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
class KeywordStyleAdapter(logging.LoggerAdapter):
|
|
|
|
"""Represents an adapter wrapping the logger allowing KeywordMessages."""
|
|
|
|
|
|
|
|
def __init__(self, logger, extra=None):
|
|
|
|
"""Initialize a new StyleAdapter for the provided logger."""
|
|
|
|
super(KeywordStyleAdapter, self).__init__(logger, extra or {})
|
|
|
|
|
|
|
|
def log(self, level, msg, *args, **kwargs):
|
|
|
|
"""Log the message provided at the appropriate level."""
|
|
|
|
if self.isEnabledFor(level):
|
|
|
|
msg, log_kwargs = self.process(msg, kwargs)
|
|
|
|
self.logger._log( # pylint: disable=protected-access
|
|
|
|
level, KeywordMessage(msg, args, kwargs), (), **log_kwargs
|
|
|
|
)
|
|
|
|
|
|
|
|
def process(self, msg, kwargs):
|
|
|
|
"""Process the keyward args in preparation for logging."""
|
|
|
|
return (
|
|
|
|
msg,
|
|
|
|
{
|
|
|
|
k: kwargs[k]
|
|
|
|
for k in inspect.getfullargspec(
|
|
|
|
self.logger._log # pylint: disable=protected-access
|
|
|
|
).args[1:] if k in kwargs
|
|
|
|
}
|
|
|
|
)
|