Skip to content

logging

Custom loggers for Tasks.

Classes:

Name Description
SocketCommunicatorHandler

Logging handler which passes messages via LUTE SocketCommunicator objects.

Functions:

Name Description
get_logger

str) -> logging.Logger: Grab a standard LUTE logger and reduce logging from libraries which output many messages.

LUTE_TASK_LOG_FORMAT: str = f'TASK_LOG -- %(levelname)s:%(name)s: %(message)s' module-attribute

Format specification for the formatter used by the standard LUTE logger.

STD_PYTHON_LOG_FORMAT: str = '%(levelname)s:%(name)s:%(message)s' module-attribute

Default Python logging formatter specification.

ColorFormatter

Bases: Formatter

Provide color text formatting for a logger.

Source code in lute/execution/logging.py
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
class ColorFormatter(logging.Formatter):
    """Provide color text formatting for a logger."""

    WARNING_COLOR_FMT: str = "\x1b[33;20m"
    ERROR_COLOR_FMT: str = "\x1b[31;20m"
    CRITICAL_COLOR_FMT: str = "\x1b[31;1m"
    RESET_COLOR_FMT: str = "\x1b[0m"

    def __init__(
        self,
        fmt: Optional[str] = None,
        datefmt: Optional[str] = None,
        style: str = "%",
        validate: bool = True,
        *,
        is_task: bool = True,
    ) -> None:
        super().__init__(fmt, datefmt, style, validate)
        log_format: str
        if is_task:
            log_format = LUTE_TASK_LOG_FORMAT
        else:
            log_format = STD_PYTHON_LOG_FORMAT
        self.level_formats: Dict[int, str] = {
            logging.DEBUG: f"{log_format}",
            logging.INFO: f"{log_format}",
            logging.WARNING: f"{self.WARNING_COLOR_FMT}{log_format}{self.RESET_COLOR_FMT}",
            logging.ERROR: f"{self.ERROR_COLOR_FMT}{log_format}{self.RESET_COLOR_FMT}",
            logging.CRITICAL: f"{self.CRITICAL_COLOR_FMT}{log_format}{self.RESET_COLOR_FMT}",
        }

    def format(self, record):
        log_fmt: str = self.level_formats[record.levelno]
        formatter: logging.Formatter = logging.Formatter(log_fmt)
        return formatter.format(record)

SocketCommunicatorHandler

Bases: Handler

Logging handler which passes messages via SocketCommunicator objects.

Source code in lute/execution/logging.py
27
28
29
30
31
32
33
34
35
36
37
38
class SocketCommunicatorHandler(logging.Handler):
    """Logging handler which passes messages via SocketCommunicator objects."""

    def __init__(self, *args, **kwargs) -> None:
        super().__init__(*args, **kwargs)
        self._communicator: SocketCommunicator = SocketCommunicator()
        self._communicator.delayed_setup()

    def emit(self, record: logging.LogRecord):
        formatted_message: str = self.format(record)
        msg: Message = Message(contents=formatted_message, signal="TASK_LOG")
        self._communicator.write(msg)

get_logger(name, is_task=True)

Retrieve a logger with LUTE handler and set log levels of other loggers.

This function returns a logger with correct log level set by the debug flag. In addition, it silences (or reduces) logs from commonly imported libraries which produce many messages that make log files hard to parse. This is particularly useful when running in debug mode.

Parameters:

Name Type Description Default
name str

Name of the logger.

required

Returns:

Name Type Description
logger Logger

Custom logger.

Source code in lute/execution/logging.py
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
def get_logger(name: str, is_task: bool = True) -> logging.Logger:
    """Retrieve a logger with LUTE handler and set log levels of other loggers.

    This function returns a logger with correct log level set by the debug flag.
    In addition, it silences (or reduces) logs from commonly imported libraries
    which produce many messages that make log files hard to parse. This is
    particularly useful when running in debug mode.

    Args:
        name (str): Name of the logger.

    Returns:
        logger (logging.Logger): Custom logger.
    """
    for other_name, other_logger in logging.root.manager.loggerDict.items():
        if "matplotlib" in other_name and not isinstance(
            other_logger, logging.PlaceHolder
        ):
            other_logger.disabled = True
        elif "numpy" in other_name and not isinstance(
            other_logger, logging.PlaceHolder
        ):
            other_logger.setLevel(logging.CRITICAL)
    logger: logging.Logger = logging.getLogger(name)
    logger.propagate = False
    handler: logging.Handler
    if is_task:
        handler: SocketCommunicatorHandler = SocketCommunicatorHandler()
    else:
        handler = logging.StreamHandler()
    formatter: ColorFormatter = ColorFormatter(is_task=is_task)
    handler.setFormatter(formatter)
    logger.handlers = []
    logger.addHandler(handler)
    if __debug__:
        logger.setLevel(logging.DEBUG)
    else:
        logger.setLevel(logging.INFO)

    return logger