Issue #12643: Respect sys.excepthook in code.InteractiveConsole
This commit is contained in:
parent
c0c03be53e
commit
8bd24fe9ff
19
Lib/code.py
19
Lib/code.py
|
@ -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.
|
||||||
|
|
|
@ -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()
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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)
|
||||||
|
|
Loading…
Reference in New Issue