From 740b89f49aeddc08b6ff94ca0b5bbac77a6e7bd6 Mon Sep 17 00:00:00 2001 From: Olivier Philippon Date: Tue, 26 Apr 2022 12:40:51 +0100 Subject: [PATCH 1/2] [traceback] Add traceback frame opt-out via a `_rich_traceback_omit = True` machinery --- CHANGELOG.md | 1 + rich/traceback.py | 2 ++ tests/test_traceback.py | 36 ++++++++++++++++++++++++++++++++++++ 3 files changed, 39 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 5001a075c..e4691104b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -11,6 +11,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 - Ability to change terminal window title https://github.com/Textualize/rich/pull/2200 - Added show_speed parameter to progress.track which will show the speed when the total is not known +- Python blocks can now opt out from being rendered in tracebacks's frames, by setting a `_rich_traceback_omit = True` in their local scope https://github.com/Textualize/rich/issues/2207 ### Fixed diff --git a/rich/traceback.py b/rich/traceback.py index 5feefb93b..54b98d504 100644 --- a/rich/traceback.py +++ b/rich/traceback.py @@ -367,6 +367,8 @@ def safe_str(_object: Any) -> str: if filename and not filename.startswith("<"): if not os.path.isabs(filename): filename = os.path.join(_IMPORT_CWD, filename) + if frame_summary.f_locals.get("_rich_traceback_omit", False) is True: + continue frame = Frame( filename=filename or "?", lineno=line_no, diff --git a/tests/test_traceback.py b/tests/test_traceback.py index c4994c87b..895eeebe5 100644 --- a/tests/test_traceback.py +++ b/tests/test_traceback.py @@ -1,6 +1,7 @@ import io import re import sys +from typing import List import pytest @@ -307,6 +308,41 @@ def test_suppress(): assert "foo" in traceback.suppress[1] +@pytest.mark.parametrize( + "rich_traceback_omit_for_level2,expected_frames_length,expected_frame_names", + ( + # fmt: off + [True, 3, ["test_rich_traceback_omit_optional_local_flag", "level1", "level3"]], + [False, 4, ["test_rich_traceback_omit_optional_local_flag", "level1", "level2", "level3"]], + # fmt: on + ), +) +def test_rich_traceback_omit_optional_local_flag( + rich_traceback_omit_for_level2: bool, + expected_frames_length: int, + expected_frame_names: List[str], +): + def level1(): + return level2() + + def level2(): + _rich_traceback_omit = rich_traceback_omit_for_level2 + return level3() + + def level3(): + return 1 / 0 + + try: + level1() + except Exception: + exc_type, exc_value, traceback = sys.exc_info() + trace = Traceback.from_exception(exc_type, exc_value, traceback).trace + frames = trace.stacks[0].frames + assert len(frames) == expected_frames_length + frame_names = [f.name for f in frames] + assert frame_names == expected_frame_names + + if __name__ == "__main__": # pragma: no cover expected = render(get_exception()) From f0ca6ff2bc0dbca2ffcd2a5114d04b92679ecbb3 Mon Sep 17 00:00:00 2001 From: Olivier Philippon Date: Tue, 26 Apr 2022 13:43:41 +0100 Subject: [PATCH 2/2] [traceback] Make `_rich_traceback_guard` more consistent with `_rich_traceback_omit` --- rich/traceback.py | 4 ++-- tests/test_traceback.py | 3 ++- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/rich/traceback.py b/rich/traceback.py index 54b98d504..55acaf070 100644 --- a/rich/traceback.py +++ b/rich/traceback.py @@ -367,7 +367,7 @@ def safe_str(_object: Any) -> str: if filename and not filename.startswith("<"): if not os.path.isabs(filename): filename = os.path.join(_IMPORT_CWD, filename) - if frame_summary.f_locals.get("_rich_traceback_omit", False) is True: + if frame_summary.f_locals.get("_rich_traceback_omit", False): continue frame = Frame( filename=filename or "?", @@ -385,7 +385,7 @@ def safe_str(_object: Any) -> str: else None, ) append(frame) - if "_rich_traceback_guard" in frame_summary.f_locals: + if frame_summary.f_locals.get("_rich_traceback_guard", False): del stack.frames[:] cause = getattr(exc_value, "__cause__", None) diff --git a/tests/test_traceback.py b/tests/test_traceback.py index 895eeebe5..2830e26aa 100644 --- a/tests/test_traceback.py +++ b/tests/test_traceback.py @@ -326,7 +326,8 @@ def level1(): return level2() def level2(): - _rich_traceback_omit = rich_traceback_omit_for_level2 + # true-ish values are enough to trigger the opt-out: + _rich_traceback_omit = 1 if rich_traceback_omit_for_level2 else 0 return level3() def level3():