Skip to content

Commit 5377f55

Browse files
[3.12] gh-87320: In the code module, handle exceptions raised in sys.excepthook (GH-122456) (GH-122515)
Before, the exception caused by calling non-default sys.excepthook in code.InteractiveInterpreter bubbled up to the caller, ending the REPL. (cherry picked from commit bd3d31f) Co-authored-by: CF Bolz-Tereick <[email protected]>
1 parent ef21e48 commit 5377f55

File tree

3 files changed

+52
-3
lines changed

3 files changed

+52
-3
lines changed

Lib/code.py

Lines changed: 16 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -127,7 +127,7 @@ def showsyntaxerror(self, filename=None):
127127
else:
128128
# If someone has set sys.excepthook, we let that take precedence
129129
# over self.write
130-
sys.excepthook(type, value, tb)
130+
self._call_excepthook(type, value, tb)
131131

132132
def showtraceback(self):
133133
"""Display the exception that just occurred.
@@ -141,16 +141,29 @@ def showtraceback(self):
141141
sys.last_traceback = last_tb
142142
sys.last_exc = ei[1]
143143
try:
144-
lines = traceback.format_exception(ei[0], ei[1], last_tb.tb_next)
145144
if sys.excepthook is sys.__excepthook__:
145+
lines = traceback.format_exception(ei[0], ei[1], last_tb.tb_next)
146146
self.write(''.join(lines))
147147
else:
148148
# If someone has set sys.excepthook, we let that take precedence
149149
# over self.write
150-
sys.excepthook(ei[0], ei[1], last_tb)
150+
self._call_excepthook(ei[0], ei[1], last_tb)
151151
finally:
152152
last_tb = ei = None
153153

154+
def _call_excepthook(self, typ, value, tb):
155+
try:
156+
sys.excepthook(typ, value, tb)
157+
except SystemExit:
158+
raise
159+
except BaseException as e:
160+
e.__context__ = None
161+
print('Error in sys.excepthook:', file=sys.stderr)
162+
sys.__excepthook__(type(e), e, e.__traceback__.tb_next)
163+
print(file=sys.stderr)
164+
print('Original exception was:', file=sys.stderr)
165+
sys.__excepthook__(typ, value, tb)
166+
154167
def write(self, data):
155168
"""Write a string.
156169

Lib/test/test_code_module.py

Lines changed: 33 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -74,6 +74,39 @@ def test_sysexcepthook(self):
7474
self.console.interact()
7575
self.assertTrue(hook.called)
7676

77+
def test_sysexcepthook_crashing_doesnt_close_repl(self):
78+
self.infunc.side_effect = ["1/0", "a = 123", "print(a)", EOFError('Finished')]
79+
self.sysmod.excepthook = 1
80+
self.console.interact()
81+
self.assertEqual(['write', ('123', ), {}], self.stdout.method_calls[0])
82+
error = "".join(call.args[0] for call in self.stderr.method_calls if call[0] == 'write')
83+
self.assertIn("Error in sys.excepthook:", error)
84+
self.assertEqual(error.count("'int' object is not callable"), 1)
85+
self.assertIn("Original exception was:", error)
86+
self.assertIn("division by zero", error)
87+
88+
def test_sysexcepthook_raising_BaseException(self):
89+
self.infunc.side_effect = ["1/0", "a = 123", "print(a)", EOFError('Finished')]
90+
s = "not so fast"
91+
def raise_base(*args, **kwargs):
92+
raise BaseException(s)
93+
self.sysmod.excepthook = raise_base
94+
self.console.interact()
95+
self.assertEqual(['write', ('123', ), {}], self.stdout.method_calls[0])
96+
error = "".join(call.args[0] for call in self.stderr.method_calls if call[0] == 'write')
97+
self.assertIn("Error in sys.excepthook:", error)
98+
self.assertEqual(error.count("not so fast"), 1)
99+
self.assertIn("Original exception was:", error)
100+
self.assertIn("division by zero", error)
101+
102+
def test_sysexcepthook_raising_SystemExit_gets_through(self):
103+
self.infunc.side_effect = ["1/0"]
104+
def raise_base(*args, **kwargs):
105+
raise SystemExit
106+
self.sysmod.excepthook = raise_base
107+
with self.assertRaises(SystemExit):
108+
self.console.interact()
109+
77110
def test_banner(self):
78111
# with banner
79112
self.infunc.side_effect = EOFError('Finished')
Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
In :class:`code.InteractiveInterpreter`, handle exceptions caused by calling a
2+
non-default :func:`sys.excepthook`. Before, the exception bubbled up to the
3+
caller, ending the REPL.

0 commit comments

Comments
 (0)