Skip to content

Commit

Permalink
Make Trio's excepthook play nicely with Ubuntu's excepthook
Browse files Browse the repository at this point in the history
  • Loading branch information
njsmith committed May 17, 2020
1 parent 51dd435 commit 0e56e28
Show file tree
Hide file tree
Showing 5 changed files with 88 additions and 12 deletions.
5 changes: 5 additions & 0 deletions ci.sh
Original file line number Diff line number Diff line change
Expand Up @@ -407,6 +407,11 @@ else
# Actual tests
python -m pip install -r test-requirements.txt

# So we can run the test for our apport/excepthook interaction working
if [ -e /etc/lsb-release ] && grep -q Ubuntu /etc/lsb-release; then
sudo apt install -q python3-apport
fi

# If we're testing with a LSP installed, then it might break network
# stuff, so wait until after we've finished setting everything else
# up.
Expand Down
8 changes: 8 additions & 0 deletions newsfragments/1065.bugfix.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
On Ubuntu systems, the system Python includes a custom
unhandled-exception hook to perform `crash reporting
<https://wiki.ubuntu.com/Apport>`__. Unfortunately, Trio wants to use
the same hook to print nice `MultiError` tracebacks, causing a
conflict. Previously, Trio would detect the conflict, print a warning,
and you just wouldn't get nice `MultiError` tracebacks. Now, Trio has
gotten clever enough to integrate its hook with Ubuntu's, so the two
systems should Just Work together.
53 changes: 41 additions & 12 deletions trio/_core/_multierror.py
Original file line number Diff line number Diff line change
Expand Up @@ -434,8 +434,8 @@ def trio_excepthook(etype, value, tb):
sys.stderr.write(chunk)


IPython_handler_installed = False
warning_given = False
monkeypatched_or_warned = False

if "IPython" in sys.modules:
import IPython
ip = IPython.get_ipython()
Expand All @@ -448,7 +448,7 @@ def trio_excepthook(etype, value, tb):
"tracebacks.",
category=RuntimeWarning
)
warning_given = True
monkeypatched_or_warned = True
else:

def trio_show_traceback(self, etype, value, tb, tb_offset=None):
Expand All @@ -457,15 +457,44 @@ def trio_show_traceback(self, etype, value, tb, tb_offset=None):
trio_excepthook(etype, value, tb)

ip.set_custom_exc((MultiError,), trio_show_traceback)
IPython_handler_installed = True
monkeypatched_or_warned = True

if sys.excepthook is sys.__excepthook__:
sys.excepthook = trio_excepthook
else:
if not IPython_handler_installed and not warning_given:
warnings.warn(
"You seem to already have a custom sys.excepthook handler "
"installed. I'll skip installing Trio's custom handler, but this "
"means MultiErrors will not show full tracebacks.",
category=RuntimeWarning
)
monkeypatched_or_warned = True

# Ubuntu's system Python has a sitecustomize.py file that import
# apport_python_hook and replaces sys.excepthook.
#
# The custom hook captures the error for crash reporting, and then calls
# sys.__excepthook__ to actually print the error.
#
# We don't mind it capturing the error for crash reporting, but we want to
# take over printing the error. So we monkeypatch the apport_python_hook
# module so that instead of calling sys.__excepthook__, it calls our custom
# hook.
#
# More details: https://github.com/python-trio/trio/issues/1065
if sys.excepthook.__name__ == "apport_excepthook":
import apport_python_hook
assert sys.excepthook is apport_python_hook.apport_excepthook

# Give it a descriptive name as a hint for anyone who's stuck trying to
# debug this mess later.
class TrioFakeSysModuleForApport:
pass

fake_sys = TrioFakeSysModuleForApport()
fake_sys.__dict__.update(sys.__dict__)
fake_sys.__excepthook__ = trio_excepthook
apport_python_hook.sys = fake_sys

monkeypatched_or_warned = True

if not monkeypatched_or_warned:
warnings.warn(
"You seem to already have a custom sys.excepthook handler "
"installed. I'll skip installing Trio's custom handler, but this "
"means MultiErrors will not show full tracebacks.",
category=RuntimeWarning
)
23 changes: 23 additions & 0 deletions trio/_core/tests/test_multierror.py
Original file line number Diff line number Diff line change
Expand Up @@ -709,3 +709,26 @@ def test_ipython_custom_exc_handler():
)
# Make sure our other warning doesn't show up
assert "custom sys.excepthook" not in completed.stdout.decode("utf-8")


@slow
@pytest.mark.skipif(
not Path("/usr/lib/python3/dist-packages/apport_python_hook.py").exists(),
reason="need Ubuntu with python3-apport installed"
)
def test_apport_excepthook_monkeypatch_interaction():
completed = run_script("apport_excepthook.py")
stdout = completed.stdout.decode("utf-8")

# No warning
assert "custom sys.excepthook" not in stdout

# Proper traceback
assert_match_in_seq(
[
"Details of embedded",
"KeyError",
"Details of embedded",
"ValueError",
], stdout
)
11 changes: 11 additions & 0 deletions trio/_core/tests/test_multierror_scripts/apport_excepthook.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
# The apport_python_hook package is only installed as part of Ubuntu's system
# python, and not available in venvs. So before we can import it we have to
# make sure it's on sys.path.
import sys
sys.path.append("/usr/lib/python3/dist-packages")
import apport_python_hook
apport_python_hook.install()

import trio

raise trio.MultiError([KeyError("key_error"), ValueError("value_error")])

0 comments on commit 0e56e28

Please sign in to comment.