From 6959f1d55568f9a743309c6824be32ca48e90192 Mon Sep 17 00:00:00 2001 From: Inada Naoki Date: Wed, 27 Mar 2024 00:32:39 +0900 Subject: [PATCH] optimize CallsiteParameterAdder (#606) * optimize CallsiteParameterAdder CallsiteParameterAdder is slower than logging.Logger.findCaller() because findCaller() doesn't use inspect at all. This commit doesn't stop using inspect.getframeinfo(), but skips heavy code_context calculation. * update CHANGELOG * Do not use inspect in CallsiteParameterAdder * [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci * fix type hint --------- Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> --- CHANGELOG.md | 3 +++ src/structlog/processors.py | 41 ++++++++++++++++--------------------- 2 files changed, 21 insertions(+), 23 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index cf107bc7..9edf242b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -32,6 +32,9 @@ You can find our backwards-compatibility policy [here](https://github.com/hynek/ - `structlog.processors.LogfmtRenderer` now escapes backslashes and double quotes. [#594](https://github.com/hynek/structlog/pull/594) +- `structlog.processors.CallsiteParameterAdder` has been optimized to be about 2x faster. + [#606](https://github.com/hynek/structlog/pull/606) + ## [24.1.0](https://github.com/hynek/structlog/compare/23.3.0...24.1.0) - 2024-01-08 diff --git a/src/structlog/processors.py b/src/structlog/processors.py index ae873ce7..65e6e00e 100644 --- a/src/structlog/processors.py +++ b/src/structlog/processors.py @@ -11,7 +11,6 @@ import datetime import enum -import inspect import json import logging import operator @@ -20,6 +19,7 @@ import threading import time +from types import FrameType from typing import ( Any, Callable, @@ -724,43 +724,39 @@ class CallsiteParameter(enum.Enum): PROCESS_NAME = "process_name" -def _get_callsite_pathname(module: str, frame_info: inspect.Traceback) -> Any: - return frame_info.filename +def _get_callsite_pathname(module: str, frame: FrameType) -> Any: + return frame.f_code.co_filename -def _get_callsite_filename(module: str, frame_info: inspect.Traceback) -> Any: - return os.path.basename(frame_info.filename) +def _get_callsite_filename(module: str, frame: FrameType) -> Any: + return os.path.basename(frame.f_code.co_filename) -def _get_callsite_module(module: str, frame_info: inspect.Traceback) -> Any: - return os.path.splitext(os.path.basename(frame_info.filename))[0] +def _get_callsite_module(module: str, frame: FrameType) -> Any: + return os.path.splitext(os.path.basename(frame.f_code.co_filename))[0] -def _get_callsite_func_name(module: str, frame_info: inspect.Traceback) -> Any: - return frame_info.function +def _get_callsite_func_name(module: str, frame: FrameType) -> Any: + return frame.f_code.co_name -def _get_callsite_lineno(module: str, frame_info: inspect.Traceback) -> Any: - return frame_info.lineno +def _get_callsite_lineno(module: str, frame: FrameType) -> Any: + return frame.f_lineno -def _get_callsite_thread(module: str, frame_info: inspect.Traceback) -> Any: +def _get_callsite_thread(module: str, frame: FrameType) -> Any: return threading.get_ident() -def _get_callsite_thread_name( - module: str, frame_info: inspect.Traceback -) -> Any: +def _get_callsite_thread_name(module: str, frame: FrameType) -> Any: return threading.current_thread().name -def _get_callsite_process(module: str, frame_info: inspect.Traceback) -> Any: +def _get_callsite_process(module: str, frame: FrameType) -> Any: return os.getpid() -def _get_callsite_process_name( - module: str, frame_info: inspect.Traceback -) -> Any: +def _get_callsite_process_name(module: str, frame: FrameType) -> Any: return get_processname() @@ -805,7 +801,7 @@ class CallsiteParameterAdder: """ _handlers: ClassVar[ - dict[CallsiteParameter, Callable[[str, inspect.Traceback], Any]] + dict[CallsiteParameter, Callable[[str, FrameType], Any]] ] = { CallsiteParameter.PATHNAME: _get_callsite_pathname, CallsiteParameter.FILENAME: _get_callsite_filename, @@ -849,7 +845,7 @@ def __init__( # module should not be logging using structlog. self._additional_ignores = ["logging", *additional_ignores] self._active_handlers: list[ - tuple[CallsiteParameter, Callable[[str, inspect.Traceback], Any]] + tuple[CallsiteParameter, Callable[[str, FrameType], Any]] ] = [] self._record_mappings: list[CallsiteParameterAdder._RecordMapping] = [] for parameter in parameters: @@ -879,9 +875,8 @@ def __call__( frame, module = _find_first_app_frame_and_name( additional_ignores=self._additional_ignores ) - frame_info = inspect.getframeinfo(frame) for parameter, handler in self._active_handlers: - event_dict[parameter.value] = handler(module, frame_info) + event_dict[parameter.value] = handler(module, frame) return event_dict