diff --git a/CHANGELOG.md b/CHANGELOG.md index 9348d7f..e700f85 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -3,6 +3,7 @@ ## 2.1.0 (unreleased) - Add support for `service.environment` from APM log correlation ([#96](https://github.com/elastic/ecs-logging-python/pull/96)) +- Fix stack trace handling in StructLog for ECS compliance ([#97](https://github.com/elastic/ecs-logging-python/pull/97)) ## 2.0.2 (2023-05-17) diff --git a/ecs_logging/_structlog.py b/ecs_logging/_structlog.py index a34e2c9..70a83ca 100644 --- a/ecs_logging/_structlog.py +++ b/ecs_logging/_structlog.py @@ -47,5 +47,13 @@ def format_to_ecs(self, event_dict): )[:-3] + "Z" ) + + if "exception" in event_dict: + stack_trace = event_dict.pop("exception") + if "error" in event_dict: + event_dict["error"]["stack_trace"] = stack_trace + else: + event_dict["error"] = {"stack_trace": stack_trace} + event_dict.setdefault("ecs", {}).setdefault("version", ECS_VERSION) return event_dict diff --git a/tests/test_structlog_formatter.py b/tests/test_structlog_formatter.py index 75b6f22..3338a4c 100644 --- a/tests/test_structlog_formatter.py +++ b/tests/test_structlog_formatter.py @@ -15,12 +15,14 @@ # specific language governing permissions and limitations # under the License. -import ecs_logging -import structlog -from unittest import mock +import json from io import StringIO +from unittest import mock import pytest +import structlog + +import ecs_logging class NotSerializable: @@ -28,7 +30,8 @@ def __repr__(self): return "" -def make_event_dict(): +@pytest.fixture +def event_dict(): return { "event": "test message", "log.logger": "logger-name", @@ -37,20 +40,29 @@ def make_event_dict(): } -def test_conflicting_event_dict(): +@pytest.fixture +def event_dict_with_exception(): + return { + "event": "test message", + "log.logger": "logger-name", + "foo": "bar", + "exception": "", + } + + +def test_conflicting_event_dict(event_dict): formatter = ecs_logging.StructlogFormatter() - event_dict = make_event_dict() event_dict["foo.bar"] = "baz" with pytest.raises(TypeError): formatter(None, "debug", event_dict) @mock.patch("time.time") -def test_event_dict_formatted(time, spec_validator): +def test_event_dict_formatted(time, spec_validator, event_dict): time.return_value = 1584720997.187709 formatter = ecs_logging.StructlogFormatter() - assert spec_validator(formatter(None, "debug", make_event_dict())) == ( + assert spec_validator(formatter(None, "debug", event_dict)) == ( '{"@timestamp":"2020-03-20T16:16:37.187Z","log.level":"debug",' '"message":"test message",' '"baz":"",' @@ -80,3 +92,19 @@ def test_can_be_set_as_processor(time, spec_validator): '"message":"test message","custom":"key","dot":{"ted":1},' '"ecs":{"version":"1.6.0"}}\n' ) + + +def test_exception_log_is_ecs_compliant_when_used_with_format_exc_info( + event_dict_with_exception, +): + formatter = ecs_logging.StructlogFormatter() + formatted_event_dict = json.loads( + formatter(None, "debug", event_dict_with_exception) + ) + + assert ( + "exception" not in formatted_event_dict + ), "The key 'exception' at the root of a log is not ECS-compliant" + assert "error" in formatted_event_dict + assert "stack_trace" in formatted_event_dict["error"] + assert "" in formatted_event_dict["error"]["stack_trace"]