| 
									
										
										
										
											2023-10-03 21:03:39 +02:00
										 |  |  | """
 | 
					
						
							|  |  |  | Partial copy of sensai.util.logging | 
					
						
							|  |  |  | """
 | 
					
						
							|  |  |  | # ruff: noqa | 
					
						
							|  |  |  | import atexit | 
					
						
							|  |  |  | import logging as lg | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | from collections.abc import Callable | 
					
						
							|  |  |  | from datetime import datetime | 
					
						
							|  |  |  | from io import StringIO | 
					
						
							|  |  |  | from logging import * | 
					
						
							|  |  |  | from typing import Any | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | log = getLogger(__name__) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LOG_DEFAULT_FORMAT = "%(levelname)-5s %(asctime)-15s %(name)s:%(funcName)s - %(message)s" | 
					
						
							|  |  |  | _logFormat = LOG_DEFAULT_FORMAT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def remove_log_handlers(): | 
					
						
							|  |  |  |     """Removes all current log handlers.""" | 
					
						
							|  |  |  |     logger = getLogger() | 
					
						
							|  |  |  |     while logger.hasHandlers(): | 
					
						
							|  |  |  |         logger.removeHandler(logger.handlers[0]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-12 17:39:11 +02:00
										 |  |  | def remove_log_handler(handler): | 
					
						
							|  |  |  |     getLogger().removeHandler(handler) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-03 21:03:39 +02:00
										 |  |  | def is_log_handler_active(handler): | 
					
						
							|  |  |  |     """Checks whether the given handler is active.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     :param handler: a log handler | 
					
						
							|  |  |  |     :return: True if the handler is active, False otherwise | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     return handler in getLogger().handlers | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # noinspection PyShadowingBuiltins | 
					
						
							|  |  |  | def configure(format=LOG_DEFAULT_FORMAT, level=lg.DEBUG): | 
					
						
							|  |  |  |     """Configures logging to stdout with the given format and log level,
 | 
					
						
							|  |  |  |     also configuring the default log levels of some overly verbose libraries as well as some pandas output options. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     :param format: the log format | 
					
						
							|  |  |  |     :param level: the minimum log level | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     global _logFormat | 
					
						
							|  |  |  |     _logFormat = format | 
					
						
							|  |  |  |     remove_log_handlers() | 
					
						
							|  |  |  |     basicConfig(level=level, format=format, stream=sys.stdout) | 
					
						
							| 
									
										
										
										
											2023-10-03 21:16:53 +02:00
										 |  |  |     # set log levels of third-party libraries | 
					
						
							|  |  |  |     getLogger("numba").setLevel(INFO) | 
					
						
							| 
									
										
										
										
											2023-10-03 21:03:39 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # noinspection PyShadowingBuiltins | 
					
						
							|  |  |  | def run_main(main_fn: Callable[[], Any], format=LOG_DEFAULT_FORMAT, level=lg.DEBUG): | 
					
						
							|  |  |  |     """Configures logging with the given parameters, ensuring that any exceptions that occur during
 | 
					
						
							|  |  |  |     the execution of the given function are logged. | 
					
						
							|  |  |  |     Logs two additional messages, one before the execution of the function, and one upon its completion. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     :param main_fn: the function to be executed | 
					
						
							|  |  |  |     :param format: the log message format | 
					
						
							|  |  |  |     :param level: the minimum log level | 
					
						
							|  |  |  |     :return: the result of `main_fn` | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     configure(format=format, level=level) | 
					
						
							|  |  |  |     log.info("Starting") | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         result = main_fn() | 
					
						
							|  |  |  |         log.info("Done") | 
					
						
							|  |  |  |         return result | 
					
						
							|  |  |  |     except Exception as e: | 
					
						
							|  |  |  |         log.error("Exception during script execution", exc_info=e) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-19 11:40:49 +02:00
										 |  |  | def run_cli(main_fn: Callable[[], Any], format=LOG_DEFAULT_FORMAT, level=lg.DEBUG): | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     Configures logging with the given parameters and runs the given main function as a | 
					
						
							|  |  |  |     CLI using `jsonargparse` (which is configured to also parse attribute docstrings, such | 
					
						
							|  |  |  |     that dataclasses can be used as function arguments). | 
					
						
							|  |  |  |     Using this function requires that `jsonargparse` and `docstring_parser` be available. | 
					
						
							|  |  |  |     Like `run_main`, two additional log messages will be logged (at the beginning and end | 
					
						
							|  |  |  |     of the execution), and it is ensured that all exceptions will be logged. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     :param main_fn: the function to be executed | 
					
						
							|  |  |  |     :param format: the log message format | 
					
						
							|  |  |  |     :param level: the minimum log level | 
					
						
							|  |  |  |     :return: the result of `main_fn` | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     from jsonargparse import set_docstring_parse_options, CLI | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     set_docstring_parse_options(attribute_docstrings=True) | 
					
						
							|  |  |  |     return run_main(lambda: CLI(main_fn), format=format, level=level) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-03 21:03:39 +02:00
										 |  |  | def datetime_tag() -> str: | 
					
						
							|  |  |  |     """:return: a string tag for use in log file names which contains the current date and time (compact but readable)""" | 
					
						
							|  |  |  |     return datetime.now().strftime("%Y%m%d-%H%M%S") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | _fileLoggerPaths: list[str] = [] | 
					
						
							|  |  |  | _isAtExitReportFileLoggerRegistered = False | 
					
						
							|  |  |  | _memoryLogStream: StringIO | None = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _at_exit_report_file_logger(): | 
					
						
							|  |  |  |     for path in _fileLoggerPaths: | 
					
						
							|  |  |  |         print(f"A log file was saved to {path}") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-12 17:39:11 +02:00
										 |  |  | def add_file_logger(path, register_atexit=True): | 
					
						
							| 
									
										
										
										
											2023-10-03 21:03:39 +02:00
										 |  |  |     global _isAtExitReportFileLoggerRegistered | 
					
						
							|  |  |  |     log.info(f"Logging to {path} ...") | 
					
						
							|  |  |  |     handler = FileHandler(path) | 
					
						
							|  |  |  |     handler.setFormatter(Formatter(_logFormat)) | 
					
						
							|  |  |  |     Logger.root.addHandler(handler) | 
					
						
							|  |  |  |     _fileLoggerPaths.append(path) | 
					
						
							| 
									
										
										
										
											2023-10-12 17:39:11 +02:00
										 |  |  |     if not _isAtExitReportFileLoggerRegistered and register_atexit: | 
					
						
							| 
									
										
										
										
											2023-10-03 21:03:39 +02:00
										 |  |  |         atexit.register(_at_exit_report_file_logger) | 
					
						
							|  |  |  |         _isAtExitReportFileLoggerRegistered = True | 
					
						
							| 
									
										
										
										
											2023-10-12 17:39:11 +02:00
										 |  |  |     return handler | 
					
						
							| 
									
										
										
										
											2023-10-03 21:03:39 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def add_memory_logger() -> None: | 
					
						
							|  |  |  |     """Enables in-memory logging (if it is not already enabled), i.e. all log statements are written to a memory buffer and can later be
 | 
					
						
							|  |  |  |     read via function `get_memory_log()`. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     global _memoryLogStream | 
					
						
							|  |  |  |     if _memoryLogStream is not None: | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |     _memoryLogStream = StringIO() | 
					
						
							|  |  |  |     handler = StreamHandler(_memoryLogStream) | 
					
						
							|  |  |  |     handler.setFormatter(Formatter(_logFormat)) | 
					
						
							|  |  |  |     Logger.root.addHandler(handler) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_memory_log(): | 
					
						
							|  |  |  |     """:return: the in-memory log (provided that `add_memory_logger` was called beforehand)""" | 
					
						
							|  |  |  |     return _memoryLogStream.getvalue() | 
					
						
							| 
									
										
										
										
											2023-10-12 17:39:11 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class FileLoggerContext: | 
					
						
							|  |  |  |     def __init__(self, path: str, enabled=True): | 
					
						
							|  |  |  |         self.enabled = enabled | 
					
						
							|  |  |  |         self.path = path | 
					
						
							|  |  |  |         self._log_handler = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __enter__(self): | 
					
						
							|  |  |  |         if self.enabled: | 
					
						
							|  |  |  |             self._log_handler = add_file_logger(self.path, register_atexit=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __exit__(self, exc_type, exc_value, traceback): | 
					
						
							|  |  |  |         if self._log_handler is not None: | 
					
						
							|  |  |  |             remove_log_handler(self._log_handler) |