Source code for ocrd_utils.logging

"""
Logging setup

By default: Log with lastResort logger, usually STDERR.

Logging can be overridden either programmatically in code using the library or by creating one or more of

- /etc/ocrd_logging.py
- $HOME/ocrd_logging.py
- $PWD/ocrd_logging.py

These files will be executed in the context of ocrd/ocrd_logging.py, with `logging` global set.
"""
# pylint: disable=no-member

from __future__ import absolute_import
from traceback import format_stack

import logging
import logging.config
import os
import sys

from .constants import LOG_FORMAT, LOG_TIMEFMT

__all__ = [
    'disableLogging',
    'getLevelName',
    'getLogger',
    'initLogging',
    'logging',
    'setOverrideLogLevel',
]

_initialized_flag = False
_overrideLogLevel = None

_ocrdLevel2pythonLevel = {
    'TRACE': 'DEBUG',
    'OFF': 'CRITICAL',
    'FATAL': 'ERROR',
}

class PropagationShyLogger(logging.Logger):

    def addHandler(self, hdlr):
        super().addHandler(hdlr)
        self.propagate = not self.handlers

    def removeHandler(self, hdlr):
        super().removeHandler(hdlr)
        self.propagate = not self.handlers

logging.setLoggerClass(PropagationShyLogger)
logging.getLogger().propagate = False

[docs]def getLevelName(lvl): """ Get (string) python logging level for (string) spec-defined log level name. """ lvl = _ocrdLevel2pythonLevel.get(lvl, lvl) return logging.getLevelName(lvl)
[docs]def setOverrideLogLevel(lvl, silent=False): """ Override all logger filter levels to include lvl and above. - Set root logger level - iterates all existing loggers and sets their log level to ``NOTSET``. Args: lvl (string): Log level name. silent (boolean): Whether to log the override call """ if lvl is None: return root_logger = logging.getLogger('') if not silent: root_logger.info('Overriding log level globally to %s', lvl) lvl = getLevelName(lvl) global _overrideLogLevel # pylint: disable=global-statement _overrideLogLevel = lvl for loggerName in logging.Logger.manager.loggerDict: logger = logging.Logger.manager.loggerDict[loggerName] if isinstance(logger, logging.PlaceHolder): continue logger.setLevel(logging.NOTSET) root_logger.setLevel(lvl)
[docs]def getLogger(*args, **kwargs): """ Wrapper around ``logging.getLogger`` that respects `overrideLogLevel <#setOverrideLogLevel>`_. """ if not _initialized_flag: initLogging() logging.getLogger('').critical('getLogger was called before initLogging. Source of the call:') for line in [x for x in format_stack(limit=2)[0].split('\n') if x]: logging.getLogger('').critical(line) name = args[0] logger = logging.getLogger(*args, **kwargs) if _overrideLogLevel and name: logger.setLevel(logging.NOTSET) return logger
[docs]def initLogging(): """ Reset root logger, read logging configuration if exists, otherwise use basicConfig """ global _initialized_flag # pylint: disable=global-statement if _initialized_flag: logging.getLogger('').critical('initLogging was called multiple times. Source of latest call:') for line in [x for x in format_stack(limit=2)[0].split('\n') if x]: logging.getLogger('').critical(line) logging.disable(logging.NOTSET) for handler in logging.root.handlers[:]: logging.root.removeHandler(handler) CONFIG_PATHS = [ os.path.curdir, os.path.join(os.path.expanduser('~')), '/etc', ] config_file = next((f for f \ in [os.path.join(p, 'ocrd_logging.conf') for p in CONFIG_PATHS] \ if os.path.exists(f)), None) if config_file: logging.config.fileConfig(config_file) logging.getLogger('ocrd.logging').debug("Picked up logging config at %s" % config_file) else: # Default logging config logging.basicConfig(level=logging.INFO, format=LOG_FORMAT, datefmt=LOG_TIMEFMT, stream=sys.stderr) logging.getLogger('').setLevel(logging.INFO) # logging.getLogger('ocrd.resolver').setLevel(logging.INFO) # logging.getLogger('ocrd.resolver.download_to_directory').setLevel(logging.INFO) # logging.getLogger('ocrd.resolver.add_files_to_mets').setLevel(logging.INFO) logging.getLogger('PIL').setLevel(logging.INFO) # To cut back on the `Self-intersection at or near point` INFO messages logging.getLogger('shapely.geos').setLevel(logging.ERROR) logging.getLogger('tensorflow').setLevel(logging.ERROR) if _overrideLogLevel: # for existing loggers that won't have getLogger do this # (because they were instantiated through logging.getLogger # instead of ours, or come from logging.config.fileConfig), # unset log levels so the global override can apply: for loggerName in logging.Logger.manager.loggerDict: logger = logging.Logger.manager.loggerDict[loggerName] if isinstance(logger, logging.PlaceHolder): continue logger.setLevel(logging.NOTSET) logging.getLogger('').setLevel(_overrideLogLevel) _initialized_flag = True
[docs]def disableLogging(): global _initialized_flag # pylint: disable=global-statement _initialized_flag = False global _overrideLogLevel # pylint: disable=global-statement _overrideLogLevel = None logging.basicConfig(level=logging.CRITICAL) logging.disable(logging.ERROR)
# Initializing stream handlers at module level # would cause message output in all runtime contexts, # including those which are already run for std output # (--dump-json, --version, ocrd-tool, bashlib etc). # So this needs to be an opt-in from the CLIs/decorators: #initLogging() # Also, we even have to block log output for libraries # (like matplotlib/tensorflow) which set up logging # themselves already: disableLogging()