-
Notifications
You must be signed in to change notification settings - Fork 98
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add helpers to enable checking of what's logged.
- Loading branch information
Showing
3 changed files
with
82 additions
and
0 deletions.
There are no files selected for viewing
Empty file.
27 changes: 27 additions & 0 deletions
27
metricflow-semantics/metricflow_semantics/formatting/formatting_helpers.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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") |
55 changes: 55 additions & 0 deletions
55
metricflow-semantics/metricflow_semantics/test_helpers/logging_helpers.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
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) |