Можно создать пользовательский уровень, но не очень просто.
Вам нужно помнить, чтобы не только изменить structlog.stdlib._NAME_TO_LEVEL
, но и structlog.stdlib._LEVEL_TO_NAME
.
Тогда вынужно добавить метод trace
как минимум к structlog.stdlib._FixedFindCallerLogger
(если вы добавите метод trace
к structlog.stdlib.BoundLogger
, вы также можете вызвать LOG.trace
, что неплохо).
Этот код должен работать:
import logging
import os
import structlog
TRACE = 19
structlog.stdlib.TRACE = TRACE = 5 # this overrides the 19 above with a 5, is that right?
structlog.stdlib._NAME_TO_LEVEL['trace'] = TRACE
structlog.stdlib._LEVEL_TO_NAME[TRACE] = 'trace'
def trace(self, msg, *args, **kw):
return self.log(TRACE, msg, *args, **kw)
structlog.stdlib._FixedFindCallerLogger.trace = trace
structlog.stdlib.BoundLogger.trace = trace
logging.basicConfig(
level=int(os.environ.get("LOGLEVEL", TRACE)),
format=os.environ.get("LOGFORMAT", '%(levelname)-8s= %(asctime)-15s = %(message)s'))
structlog.configure(
processors=[
structlog.stdlib.filter_by_level,
structlog.stdlib.add_logger_name,
structlog.stdlib.add_log_level,
structlog.stdlib.PositionalArgumentsFormatter(),
structlog.processors.StackInfoRenderer(),
structlog.processors.format_exc_info,
structlog.processors.UnicodeDecoder(),
structlog.stdlib.render_to_log_kwargs,
],
context_class=dict,
logger_factory=structlog.stdlib.LoggerFactory(),
wrapper_class=structlog.stdlib.BoundLogger,
cache_logger_on_first_use=True,
)
logging.addLevelName(TRACE, "TRACE")
LOG = structlog.getLogger()
LOG.trace('test')