From c27af7a7ad384f91d6d7f13682b00a8ba8a9e765 Mon Sep 17 00:00:00 2001 From: risto Date: Fri, 6 Dec 2024 12:56:38 +0100 Subject: [PATCH] feat: enhance logging configuration and stack trace management - Add `log_stack_trace_as` and `log_stack_trace_for` env vars - Modify `stack_trace` function logic in `log_handler.py` - Introduce `tracardi/logging.py` for central log config - Update log condition checks and default bulk size handling - Change `_get_context_object` to sync in `context.py` --- tracardi/config.py | 2 ++ tracardi/exceptions/log_handler.py | 20 ++++++++++++++++---- tracardi/logging.py | 7 +++++++ 3 files changed, 25 insertions(+), 4 deletions(-) create mode 100644 tracardi/logging.py diff --git a/tracardi/config.py b/tracardi/config.py index b5aa7ad5..831df255 100644 --- a/tracardi/config.py +++ b/tracardi/config.py @@ -284,6 +284,8 @@ def __init__(self, env): self.server_logging_level = _get_logging_level( env['SERVER_LOGGING_LEVEL']) if 'SERVER_LOGGING_LEVEL' in env else logging.WARNING self.skip_errors_on_profile_mapping = get_env_as_bool('SKIP_ERRORS_ON_PROFILE_MAPPING', 'no') + self.log_stack_trace_as = env.get('LOG_STACK_TRACE_AS', 'string') + self.log_stack_trace_for = env.get('LOG_STACK_TRACE_AS', 'CRITICAL,ERROR').split(',') self._config = None self._unset_secrets() diff --git a/tracardi/exceptions/log_handler.py b/tracardi/exceptions/log_handler.py index 3bba7197..3d6d0b6f 100644 --- a/tracardi/exceptions/log_handler.py +++ b/tracardi/exceptions/log_handler.py @@ -7,6 +7,7 @@ import sys from tracardi.context import get_context, ContextError +from tracardi.logging import log_stack_trace_as, log_stack_trace_for, log_bulk_size from tracardi.service.adapter.logger.logger_adapter import log_format_adapter from tracardi.service.logging.tools import _get_logging_level from tracardi.service.utils.date import now_in_utc @@ -18,8 +19,8 @@ def stack_trace(level): - if level not in ["ERROR", "CRITICAL", "WARNING"]: - return [] + if level not in log_stack_trace_for: + return {} # Extract the traceback object tb = sys.exc_info()[2] @@ -121,6 +122,15 @@ def emit(self, record: LogRecord): if record.levelno <= 25: return + _trace = stack_trace(record.levelname) + if log_stack_trace_as == 'json': + if _trace: + stack_trace_str = f"JSON:{json.dumps(_trace)}" + else: + stack_trace_str = None + else: + stack_trace_str = record.stack_info + log = { # Maps to tracardi-log index "date": now_in_utc(), "message": record.msg, @@ -128,7 +138,7 @@ def emit(self, record: LogRecord): "file": record.filename, "line": record.lineno, "level": record.levelname, - "stack_info": json.dumps(stack_trace(record.levelname)), + "stack_info": stack_trace_str, # "exc_info": record.exc_info # Can not save this to TrackerPayload "module": self._get(record, "package", record.module), "class_name": self._get(record, "class_name", record.funcName), @@ -142,7 +152,9 @@ def emit(self, record: LogRecord): self.collection.append(log) - def has_logs(self, min_log_size=500): + def has_logs(self, min_log_size=None): + if min_log_size is None: + min_log_size = log_bulk_size if not isinstance(self.collection, list): return False return len(self.collection) >= min_log_size or (time() - self.last_save) > 60 diff --git a/tracardi/logging.py b/tracardi/logging.py new file mode 100644 index 00000000..b2c3f3a6 --- /dev/null +++ b/tracardi/logging.py @@ -0,0 +1,7 @@ +import os + +from tracardi.service.utils.environment import get_env_as_int + +log_stack_trace_as = os.environ.get('LOG_STACK_TRACE_AS', 'string') +log_stack_trace_for = os.environ.get('LOG_STACK_TRACE_AS', 'CRITICAL,ERROR').split(',') +log_bulk_size = get_env_as_int('LOG_BULK_SIZE', 500) \ No newline at end of file