Skip to content
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
121 changes: 121 additions & 0 deletions Lib/test/_test_atexit.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,121 @@
"""
Tests run by test_atexit in a subprocess since it clears atexit callbacks.
"""
import atexit
import sys
import unittest
from test import support


class GeneralTest(unittest.TestCase):
def setUp(self):
atexit._clear()

def tearDown(self):
atexit._clear()

def assert_raises_unraisable(self, exc_type, func, *args):
with support.catch_unraisable_exception() as cm:
atexit.register(func, *args)
atexit._run_exitfuncs()

self.assertEqual(cm.unraisable.object, func)
self.assertEqual(cm.unraisable.exc_type, exc_type)
self.assertEqual(type(cm.unraisable.exc_value), exc_type)

def test_order(self):
# Check that callbacks are called in reverse order with the expected
# positional and keyword arguments.
calls = []

def func1(*args, **kwargs):
calls.append(('func1', args, kwargs))

def func2(*args, **kwargs):
calls.append(('func2', args, kwargs))

# be sure args are handled properly
atexit.register(func1, 1, 2)
atexit.register(func2)
atexit.register(func2, 3, key="value")
atexit._run_exitfuncs()

self.assertEqual(calls,
[('func2', (3,), {'key': 'value'}),
('func2', (), {}),
('func1', (1, 2), {})])

def test_badargs(self):
def func():
pass

# func() has no parameter, but it's called with 2 parameters
self.assert_raises_unraisable(TypeError, func, 1 ,2)

def test_raise(self):
def raise_type_error():
raise TypeError

self.assert_raises_unraisable(TypeError, raise_type_error)

def test_raise_unnormalized(self):
# bpo-10756: Make sure that an unnormalized exception is handled
# properly.
def div_zero():
1 / 0

self.assert_raises_unraisable(ZeroDivisionError, div_zero)

def test_exit(self):
self.assert_raises_unraisable(SystemExit, sys.exit)

def test_stress(self):
a = [0]
def inc():
a[0] += 1

for i in range(128):
atexit.register(inc)
atexit._run_exitfuncs()

self.assertEqual(a[0], 128)

def test_clear(self):
a = [0]
def inc():
a[0] += 1

atexit.register(inc)
atexit._clear()
atexit._run_exitfuncs()

self.assertEqual(a[0], 0)

def test_unregister(self):
a = [0]
def inc():
a[0] += 1
def dec():
a[0] -= 1

for i in range(4):
atexit.register(inc)
atexit.register(dec)
atexit.unregister(inc)
atexit._run_exitfuncs()

self.assertEqual(a[0], -1)

def test_bound_methods(self):
l = []
atexit.register(l.append, 5)
atexit._run_exitfuncs()
self.assertEqual(l, [5])

atexit.unregister(l.append)
atexit._run_exitfuncs()
self.assertEqual(l, [5])


if __name__ == "__main__":
unittest.main()
151 changes: 5 additions & 146 deletions Lib/test/test_atexit.py
Original file line number Diff line number Diff line change
@@ -1,160 +1,19 @@
import atexit
import io
import os
import sys
import textwrap
import unittest
from test import support
from test.support import script_helper

### helpers
def h1():
print("h1")

def h2():
print("h2")

def h3():
print("h3")

def h4(*args, **kwargs):
print("h4", args, kwargs)

def raise1():
raise TypeError

def raise2():
raise SystemError

def exit():
raise SystemExit


class GeneralTest(unittest.TestCase):
def test_general(self):
# Run _test_atexit.py in a subprocess since it calls atexit._clear()
script = support.findfile("_test_atexit.py")
script_helper.run_test_script(script)

def setUp(self):
self.save_stdout = sys.stdout
self.save_stderr = sys.stderr
self.stream = io.StringIO()
sys.stdout = sys.stderr = self.stream
atexit._clear()

def tearDown(self):
sys.stdout = self.save_stdout
sys.stderr = self.save_stderr
atexit._clear()

def test_args(self):
# be sure args are handled properly
atexit.register(h1)
atexit.register(h4)
atexit.register(h4, 4, kw="abc")
atexit._run_exitfuncs()

self.assertEqual(self.stream.getvalue(),
"h4 (4,) {'kw': 'abc'}\nh4 () {}\nh1\n")

def test_badargs(self):
atexit.register(lambda: 1, 0, 0, (x for x in (1,2)), 0, 0)
self.assertRaises(TypeError, atexit._run_exitfuncs)

def test_order(self):
# be sure handlers are executed in reverse order
atexit.register(h1)
atexit.register(h2)
atexit.register(h3)
atexit._run_exitfuncs()

self.assertEqual(self.stream.getvalue(), "h3\nh2\nh1\n")

def test_raise(self):
# be sure raises are handled properly
atexit.register(raise1)
atexit.register(raise2)

self.assertRaises(TypeError, atexit._run_exitfuncs)

def test_raise_unnormalized(self):
# Issue #10756: Make sure that an unnormalized exception is
# handled properly
atexit.register(lambda: 1 / 0)

self.assertRaises(ZeroDivisionError, atexit._run_exitfuncs)
self.assertIn("ZeroDivisionError", self.stream.getvalue())

def test_exit(self):
# be sure a SystemExit is handled properly
atexit.register(exit)

self.assertRaises(SystemExit, atexit._run_exitfuncs)
self.assertEqual(self.stream.getvalue(), '')

def test_print_tracebacks(self):
# Issue #18776: the tracebacks should be printed when errors occur.
def f():
1/0 # one
def g():
1/0 # two
def h():
1/0 # three
atexit.register(f)
atexit.register(g)
atexit.register(h)

self.assertRaises(ZeroDivisionError, atexit._run_exitfuncs)
stderr = self.stream.getvalue()
self.assertEqual(stderr.count("ZeroDivisionError"), 3)
self.assertIn("# one", stderr)
self.assertIn("# two", stderr)
self.assertIn("# three", stderr)

def test_stress(self):
a = [0]
def inc():
a[0] += 1

for i in range(128):
atexit.register(inc)
atexit._run_exitfuncs()

self.assertEqual(a[0], 128)

def test_clear(self):
a = [0]
def inc():
a[0] += 1

atexit.register(inc)
atexit._clear()
atexit._run_exitfuncs()

self.assertEqual(a[0], 0)

def test_unregister(self):
a = [0]
def inc():
a[0] += 1
def dec():
a[0] -= 1

for i in range(4):
atexit.register(inc)
atexit.register(dec)
atexit.unregister(inc)
atexit._run_exitfuncs()

self.assertEqual(a[0], -1)

def test_bound_methods(self):
l = []
atexit.register(l.append, 5)
atexit._run_exitfuncs()
self.assertEqual(l, [5])

atexit.unregister(l.append)
atexit._run_exitfuncs()
self.assertEqual(l, [5])

class FunctionalTest(unittest.TestCase):
def test_shutdown(self):
# Actually test the shutdown mechanism in a subprocess
code = textwrap.dedent("""
Expand Down
3 changes: 0 additions & 3 deletions Lib/test/test_eintr.py
Original file line number Diff line number Diff line change
@@ -1,9 +1,6 @@
import os
import signal
import subprocess
import sys
import unittest

from test import support
from test.support import script_helper

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
:func:`atexit._run_exitfuncs` now logs callback exceptions using
:data:`sys.unraisablehook`, rather than logging them directly into
:data:`sys.stderr` and raise the last exception.
Loading