aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSerhiy Storchaka <storchaka@gmail.com>2024-07-31 14:54:06 +0300
committerGitHub <noreply@github.com>2024-07-31 11:54:06 +0000
commit5377f55b4e022041b7b57b5489c66c9b3c046c7e (patch)
tree8a1fa137edf1c87bdf82d1476f5ededdc8c3e1a1
parent[3.12] Docs: bump Sphinx to 8.0 and update constraints (GH-122496) (#122500) (diff)
downloadcpython-5377f55b4e022041b7b57b5489c66c9b3c046c7e.tar.gz
cpython-5377f55b4e022041b7b57b5489c66c9b3c046c7e.tar.bz2
cpython-5377f55b4e022041b7b57b5489c66c9b3c046c7e.zip
[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 bd3d31f380cd451a4ab6da5fbfde463fed95b5b5) Co-authored-by: CF Bolz-Tereick <cfbolz@gmx.de>
-rw-r--r--Lib/code.py19
-rw-r--r--Lib/test/test_code_module.py33
-rw-r--r--Misc/NEWS.d/next/Library/2024-07-30-14-46-16.gh-issue-87320.-Yk1wb.rst3
3 files changed, 52 insertions, 3 deletions
diff --git a/Lib/code.py b/Lib/code.py
index 2bd5fa3e795..b4b1ef3b8b9 100644
--- a/Lib/code.py
+++ b/Lib/code.py
@@ -127,7 +127,7 @@ class InteractiveInterpreter:
else:
# If someone has set sys.excepthook, we let that take precedence
# over self.write
- sys.excepthook(type, value, tb)
+ self._call_excepthook(type, value, tb)
def showtraceback(self):
"""Display the exception that just occurred.
@@ -141,16 +141,29 @@ class InteractiveInterpreter:
sys.last_traceback = last_tb
sys.last_exc = ei[1]
try:
- lines = traceback.format_exception(ei[0], ei[1], last_tb.tb_next)
if sys.excepthook is sys.__excepthook__:
+ lines = traceback.format_exception(ei[0], ei[1], last_tb.tb_next)
self.write(''.join(lines))
else:
# If someone has set sys.excepthook, we let that take precedence
# over self.write
- sys.excepthook(ei[0], ei[1], last_tb)
+ self._call_excepthook(ei[0], ei[1], last_tb)
finally:
last_tb = ei = None
+ def _call_excepthook(self, typ, value, tb):
+ try:
+ sys.excepthook(typ, value, tb)
+ except SystemExit:
+ raise
+ except BaseException as e:
+ e.__context__ = None
+ print('Error in sys.excepthook:', file=sys.stderr)
+ sys.__excepthook__(type(e), e, e.__traceback__.tb_next)
+ print(file=sys.stderr)
+ print('Original exception was:', file=sys.stderr)
+ sys.__excepthook__(typ, value, tb)
+
def write(self, data):
"""Write a string.
diff --git a/Lib/test/test_code_module.py b/Lib/test/test_code_module.py
index 226bc3a853b..6082abe8986 100644
--- a/Lib/test/test_code_module.py
+++ b/Lib/test/test_code_module.py
@@ -74,6 +74,39 @@ class TestInteractiveConsole(unittest.TestCase):
self.console.interact()
self.assertTrue(hook.called)
+ def test_sysexcepthook_crashing_doesnt_close_repl(self):
+ self.infunc.side_effect = ["1/0", "a = 123", "print(a)", EOFError('Finished')]
+ self.sysmod.excepthook = 1
+ self.console.interact()
+ self.assertEqual(['write', ('123', ), {}], self.stdout.method_calls[0])
+ error = "".join(call.args[0] for call in self.stderr.method_calls if call[0] == 'write')
+ self.assertIn("Error in sys.excepthook:", error)
+ self.assertEqual(error.count("'int' object is not callable"), 1)
+ self.assertIn("Original exception was:", error)
+ self.assertIn("division by zero", error)
+
+ def test_sysexcepthook_raising_BaseException(self):
+ self.infunc.side_effect = ["1/0", "a = 123", "print(a)", EOFError('Finished')]
+ s = "not so fast"
+ def raise_base(*args, **kwargs):
+ raise BaseException(s)
+ self.sysmod.excepthook = raise_base
+ self.console.interact()
+ self.assertEqual(['write', ('123', ), {}], self.stdout.method_calls[0])
+ error = "".join(call.args[0] for call in self.stderr.method_calls if call[0] == 'write')
+ self.assertIn("Error in sys.excepthook:", error)
+ self.assertEqual(error.count("not so fast"), 1)
+ self.assertIn("Original exception was:", error)
+ self.assertIn("division by zero", error)
+
+ def test_sysexcepthook_raising_SystemExit_gets_through(self):
+ self.infunc.side_effect = ["1/0"]
+ def raise_base(*args, **kwargs):
+ raise SystemExit
+ self.sysmod.excepthook = raise_base
+ with self.assertRaises(SystemExit):
+ self.console.interact()
+
def test_banner(self):
# with banner
self.infunc.side_effect = EOFError('Finished')
diff --git a/Misc/NEWS.d/next/Library/2024-07-30-14-46-16.gh-issue-87320.-Yk1wb.rst b/Misc/NEWS.d/next/Library/2024-07-30-14-46-16.gh-issue-87320.-Yk1wb.rst
new file mode 100644
index 00000000000..adce11e8c88
--- /dev/null
+++ b/Misc/NEWS.d/next/Library/2024-07-30-14-46-16.gh-issue-87320.-Yk1wb.rst
@@ -0,0 +1,3 @@
+In :class:`code.InteractiveInterpreter`, handle exceptions caused by calling a
+non-default :func:`sys.excepthook`. Before, the exception bubbled up to the
+caller, ending the REPL.