Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[3.8] bpo-43008: Make IDLE respect sys.excepthook (GH-24302) #24347

Merged
merged 1 commit into from
Jan 27, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion Doc/library/idle.rst
Original file line number Diff line number Diff line change
Expand Up @@ -250,7 +250,7 @@ View Last Restart
Scroll the shell window to the last Shell restart.

Restart Shell
Restart the shell to clean the environment.
Restart the shell to clean the environment and reset display and exception handling.

Previous History
Cycle through earlier commands in history which match the current entry.
Expand Down
3 changes: 3 additions & 0 deletions Lib/idlelib/NEWS.txt
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,9 @@ Released on 2021-02-15?
======================================


bpo-43008: Make IDLE invoke :func:`sys.excepthook` in normal,
2-process mode.

bpo-33065: Fix problem debugging user classes with __repr__ method.

bpo-32631: Finish zzdummy example extension module: make menu entries
Expand Down
43 changes: 39 additions & 4 deletions Lib/idlelib/idle_test/test_run.py
Original file line number Diff line number Diff line change
@@ -1,16 +1,18 @@
"Test run, coverage 49%."

from idlelib import run
import io
import sys
from test.support import captured_output, captured_stderr
import unittest
from unittest import mock
import idlelib
from idlelib.idle_test.mock_idle import Func
from test.support import captured_output, captured_stderr

import io
import sys
idlelib.testing = True # Use {} for executing test user code.


class RunTest(unittest.TestCase):
class PrintExceptionTest(unittest.TestCase):

def test_print_exception_unhashable(self):
class UnhashableException(Exception):
Expand Down Expand Up @@ -351,5 +353,38 @@ def test_fatal_error(self):
self.assertIn('IndexError', msg)
eq(func.called, 2)


class ExecRuncodeTest(unittest.TestCase):

@classmethod
def setUpClass(cls):
cls.addClassCleanup(setattr,run,'print_exception',run.print_exception)
cls.prt = Func() # Need reference.
run.print_exception = cls.prt
mockrpc = mock.Mock()
mockrpc.console.getvar = Func(result=False)
cls.ex = run.Executive(mockrpc)

@classmethod
def tearDownClass(cls):
assert sys.excepthook == sys.__excepthook__

def test_exceptions(self):
ex = self.ex
ex.runcode('1/0')
self.assertIs(ex.user_exc_info[0], ZeroDivisionError)

self.addCleanup(setattr, sys, 'excepthook', sys.__excepthook__)
sys.excepthook = lambda t, e, tb: run.print_exception(t)
ex.runcode('1/0')
self.assertIs(self.prt.args[0], ZeroDivisionError)

sys.excepthook = lambda: None
ex.runcode('1/0')
t, e, tb = ex.user_exc_info
self.assertIs(t, TypeError)
self.assertTrue(isinstance(e.__context__, ZeroDivisionError))


if __name__ == '__main__':
unittest.main(verbosity=2)
27 changes: 19 additions & 8 deletions Lib/idlelib/run.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
import threading
import warnings

import idlelib # testing
from idlelib import autocomplete # AutoComplete, fetch_encodings
from idlelib import calltip # Calltip
from idlelib import debugger_r # start_debugger
Expand Down Expand Up @@ -542,14 +543,17 @@ class Executive:

def __init__(self, rpchandler):
self.rpchandler = rpchandler
self.locals = __main__.__dict__
self.calltip = calltip.Calltip()
self.autocomplete = autocomplete.AutoComplete()
if idlelib.testing is False:
self.locals = __main__.__dict__
self.calltip = calltip.Calltip()
self.autocomplete = autocomplete.AutoComplete()
else:
self.locals = {}

def runcode(self, code):
global interruptable
try:
self.usr_exc_info = None
self.user_exc_info = None
interruptable = True
try:
exec(code, self.locals)
Expand All @@ -562,10 +566,17 @@ def runcode(self, code):
print('SystemExit: ' + str(ob), file=sys.stderr)
# Return to the interactive prompt.
except:
self.usr_exc_info = sys.exc_info()
self.user_exc_info = sys.exc_info() # For testing, hook, viewer.
if quitting:
exit()
print_exception()
if sys.excepthook is sys.__excepthook__:
print_exception()
else:
try:
sys.excepthook(*self.user_exc_info)
except:
self.user_exc_info = sys.exc_info() # For testing.
print_exception()
jit = self.rpchandler.console.getvar("<<toggle-jit-stack-viewer>>")
if jit:
self.rpchandler.interp.open_remote_stack_viewer()
Expand All @@ -590,8 +601,8 @@ def get_the_completion_list(self, what, mode):
return self.autocomplete.fetch_completions(what, mode)

def stackviewer(self, flist_oid=None):
if self.usr_exc_info:
typ, val, tb = self.usr_exc_info
if self.user_exc_info:
typ, val, tb = self.user_exc_info
else:
return None
flist = None
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Make IDLE invoke :func:`sys.excepthook` in normal, 2-process mode.