Issue #12643: Respect sys.excepthook in code.InteractiveConsole

This commit is contained in:
Nick Coghlan 2012-08-20 23:02:28 +10:00
parent c0c03be53e
commit 8bd24fe9ff
5 changed files with 91 additions and 5 deletions

View File

@ -105,9 +105,10 @@ class InteractiveInterpreter:
The output is written by self.write(), below. The output is written by self.write(), below.
""" """
type, value, sys.last_traceback = sys.exc_info() type, value, tb = sys.exc_info()
sys.last_type = type sys.last_type = type
sys.last_value = value sys.last_value = value
sys.last_traceback = tb
if filename and type is SyntaxError: if filename and type is SyntaxError:
# Work hard to stuff the correct filename in the exception # Work hard to stuff the correct filename in the exception
try: try:
@ -119,8 +120,13 @@ class InteractiveInterpreter:
# Stuff in the right filename # Stuff in the right filename
value = SyntaxError(msg, (filename, lineno, offset, line)) value = SyntaxError(msg, (filename, lineno, offset, line))
sys.last_value = value sys.last_value = value
lines = traceback.format_exception_only(type, value) if sys.excepthook is sys.__excepthook__:
self.write(''.join(lines)) lines = traceback.format_exception_only(type, value)
self.write(''.join(lines))
else:
# If someone has set sys.excepthook, we let that take precedence
# over self.write
sys.excepthook(type, value, tb)
def showtraceback(self): def showtraceback(self):
"""Display the exception that just occurred. """Display the exception that just occurred.
@ -143,7 +149,12 @@ class InteractiveInterpreter:
lines.extend(traceback.format_exception_only(type, value)) lines.extend(traceback.format_exception_only(type, value))
finally: finally:
tblist = tb = None tblist = tb = None
self.write(''.join(lines)) if sys.excepthook is sys.__excepthook__:
self.write(''.join(lines))
else:
# If someone has set sys.excepthook, we let that take precedence
# over self.write
sys.excepthook(type, value, tb)
def write(self, data): def write(self, data):
"""Write a string. """Write a string.

View File

@ -0,0 +1,72 @@
"Test InteractiveConsole and InteractiveInterpreter from code module"
import sys
import unittest
from contextlib import ExitStack
from unittest import mock
from test import support
code = support.import_module('code')
class TestInteractiveConsole(unittest.TestCase):
def setUp(self):
self.console = code.InteractiveConsole()
self.mock_sys()
def mock_sys(self):
"Mock system environment for InteractiveConsole"
# use exit stack to match patch context managers to addCleanup
stack = ExitStack()
self.addCleanup(stack.close)
self.infunc = stack.enter_context(mock.patch('code.input',
create=True))
self.stdout = stack.enter_context(mock.patch('code.sys.stdout'))
self.stderr = stack.enter_context(mock.patch('code.sys.stderr'))
prepatch = mock.patch('code.sys', wraps=code.sys, spec=code.sys)
self.sysmod = stack.enter_context(prepatch)
if sys.excepthook is sys.__excepthook__:
self.sysmod.excepthook = self.sysmod.__excepthook__
def test_ps1(self):
self.infunc.side_effect = EOFError('Finished')
self.console.interact()
self.assertEqual(self.sysmod.ps1, '>>> ')
def test_ps2(self):
self.infunc.side_effect = EOFError('Finished')
self.console.interact()
self.assertEqual(self.sysmod.ps2, '... ')
def test_console_stderr(self):
self.infunc.side_effect = ["'antioch'", "", EOFError('Finished')]
self.console.interact()
for call in list(self.stdout.method_calls):
if 'antioch' in ''.join(call[1]):
break
else:
raise AssertionError("no console stdout")
def test_syntax_error(self):
self.infunc.side_effect = ["undefined", EOFError('Finished')]
self.console.interact()
for call in self.stderr.method_calls:
if 'NameError:' in ''.join(call[1]):
break
else:
raise AssertionError("No syntax error from console")
def test_sysexcepthook(self):
self.infunc.side_effect = ["raise ValueError('')",
EOFError('Finished')]
hook = mock.Mock()
self.sysmod.excepthook = hook
self.console.interact()
self.assertTrue(hook.called)
def test_main():
support.run_unittest(TestInteractiveConsole)
if __name__ == "__main__":
unittest.main()

View File

@ -9,7 +9,6 @@ class TestUntestedModules(unittest.TestCase):
with support.check_warnings(quiet=True): with support.check_warnings(quiet=True):
import bdb import bdb
import cgitb import cgitb
import code
import distutils.bcppcompiler import distutils.bcppcompiler
import distutils.ccompiler import distutils.ccompiler

View File

@ -487,6 +487,7 @@ Fredrik Håård
Catalin Iacob Catalin Iacob
Mihai Ibanescu Mihai Ibanescu
Ali Ikinci Ali Ikinci
Aaron Iles
Lars Immisch Lars Immisch
Bobby Impollonia Bobby Impollonia
Meador Inge Meador Inge

View File

@ -19,6 +19,9 @@ Core and Builtins
Library Library
------- -------
- Issue #12643: code.InteractiveConsole now respects sys.excepthook when
displaying exceptions (Patch by Aaron Iles)
- Issue #13579: string.Formatter now understands the 'a' conversion specifier. - Issue #13579: string.Formatter now understands the 'a' conversion specifier.
- Issue #15595: Fix subprocess.Popen(universal_newlines=True) - Issue #15595: Fix subprocess.Popen(universal_newlines=True)