Skip to content

Commit

Permalink
Add helpers to enable checking of what's logged.
Browse files Browse the repository at this point in the history
  • Loading branch information
plypaul committed Jul 19, 2024
1 parent 90add9a commit 048f9e2
Show file tree
Hide file tree
Showing 3 changed files with 82 additions and 0 deletions.
Empty file.
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
from __future__ import annotations

import textwrap


def mf_dedent(text: str) -> str:
"""Remove leading newlines, dedents, and remove tailing newlines.
This function simplifies the commonly used:
text = textwrap.dedent(
[triple quote][backslash]
Line 0
Line 1
[triple quote]
).rstrip()
to:
text = mf_dedent(
[triple quote]
Line 0
Line 1
[triple quote]
)
"""
return textwrap.dedent(text.lstrip("\n")).rstrip("\n")
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
from __future__ import annotations

import logging
from contextlib import contextmanager
from typing import Iterator, List, Optional, Sequence, Tuple

from typing_extensions import override

from metricflow_semantics.mf_logging.logger_configuration import mf_get_logger
from metricflow_semantics.mf_logging.mf_logger import MetricFlowLogger


class RecordingLogHandler(logging.Handler):
"""A log-record handler that stores them so that they can be checked in tests."""

def __init__(self) -> None: # noqa: D107
super().__init__()
self._log_records: List[logging.LogRecord] = []

@override
def emit(self, record: logging.LogRecord) -> None:
self._log_records.append(record)

@property
def log_records(self) -> Sequence[logging.LogRecord]:
"""Return the log records seen by the handler."""
return self._log_records

def get_last_message(self) -> Optional[str]:
"""Return the message in the last log record, or None if this hasn't seen any."""
if len(self._log_records) == 0:
return None

return self._log_records[-1].message


@contextmanager
def recorded_logging_context(logging_level_int: int) -> Iterator[Tuple[MetricFlowLogger, RecordingLogHandler]]:
"""Context with a logger (with the given log level) and associated handler to check what was logged.
The handler records all log records emitted that is appropriate for the given level during this context.
Log propagation could be disabled in this context to clean test log output, but some issues need to be resolved.
"""
mf_logger = mf_get_logger()
standard_logger = mf_logger.standard_library_logger

previous_logging_level = standard_logger.level
handler = RecordingLogHandler()
standard_logger.addHandler(handler)
try:
standard_logger.setLevel(logging_level_int)
yield mf_logger, handler
finally:
standard_logger.removeHandler(handler)
standard_logger.setLevel(previous_logging_level)

0 comments on commit 048f9e2

Please sign in to comment.