blob: d6739f18660a39f2763baeeaaedc6c59e0ce9998 [file] [log] [blame]
Guido van Rossum3bead091992-01-27 17:00:37 +00001# Python test set -- part 5, built-in exceptions
2
Serhiy Storchakab7853962017-04-08 09:55:07 +03003import copy
Thomas Wouters73e5a5b2006-06-08 15:35:45 +00004import os
5import sys
Thomas Wouters4d70c3d2006-06-08 14:42:34 +00006import unittest
Guido van Rossumbf12cdb2006-08-17 20:24:18 +00007import pickle
Barry Warsaw8d109cb2008-05-08 04:26:35 +00008import weakref
Antoine Pitroua7622852011-09-01 21:37:43 +02009import errno
Thomas Wouters73e5a5b2006-06-08 15:35:45 +000010
Martin Panter3263f682016-02-28 03:16:11 +000011from test.support import (TESTFN, captured_stderr, check_impl_detail,
Victor Stinner8f4ef3b2019-07-01 18:28:25 +020012 check_warnings, cpython_only, gc_collect,
xdegaye56d1f5c2017-10-26 15:09:06 +020013 no_tracing, unlink, import_module, script_helper,
14 SuppressCrashReport)
Victor Stinnere4d300e2019-05-22 23:44:02 +020015from test import support
16
17
Richard Oudkerk5562d9d2012-07-28 17:45:28 +010018class NaiveException(Exception):
19 def __init__(self, x):
20 self.x = x
21
22class SlottedNaiveException(Exception):
23 __slots__ = ('x',)
24 def __init__(self, x):
25 self.x = x
26
Martin Panter3263f682016-02-28 03:16:11 +000027class BrokenStrException(Exception):
28 def __str__(self):
29 raise Exception("str() is broken")
30
Guido van Rossum3bead091992-01-27 17:00:37 +000031# XXX This is not really enough, each *operation* should be tested!
32
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000033class ExceptionTests(unittest.TestCase):
Barry Warsawb9c1d3d2001-08-13 23:07:00 +000034
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000035 def raise_catch(self, exc, excname):
36 try:
Collin Winter828f04a2007-08-31 00:04:24 +000037 raise exc("spam")
Guido van Rossumb940e112007-01-10 16:19:56 +000038 except exc as err:
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000039 buf1 = str(err)
40 try:
41 raise exc("spam")
Guido van Rossumb940e112007-01-10 16:19:56 +000042 except exc as err:
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000043 buf2 = str(err)
Ezio Melottib3aedd42010-11-20 19:04:17 +000044 self.assertEqual(buf1, buf2)
45 self.assertEqual(exc.__name__, excname)
Guido van Rossum3bead091992-01-27 17:00:37 +000046
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000047 def testRaising(self):
48 self.raise_catch(AttributeError, "AttributeError")
49 self.assertRaises(AttributeError, getattr, sys, "undefined_attribute")
Guido van Rossum3bead091992-01-27 17:00:37 +000050
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000051 self.raise_catch(EOFError, "EOFError")
52 fp = open(TESTFN, 'w')
53 fp.close()
54 fp = open(TESTFN, 'r')
55 savestdin = sys.stdin
56 try:
57 try:
58 import marshal
Antoine Pitrou4a90ef02012-03-03 02:35:32 +010059 marshal.loads(b'')
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000060 except EOFError:
61 pass
62 finally:
63 sys.stdin = savestdin
64 fp.close()
65 unlink(TESTFN)
Guido van Rossum3bead091992-01-27 17:00:37 +000066
Antoine Pitrou6b4883d2011-10-12 02:54:14 +020067 self.raise_catch(OSError, "OSError")
68 self.assertRaises(OSError, open, 'this file does not exist', 'r')
Guido van Rossum3bead091992-01-27 17:00:37 +000069
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000070 self.raise_catch(ImportError, "ImportError")
71 self.assertRaises(ImportError, __import__, "undefined_module")
Guido van Rossum3bead091992-01-27 17:00:37 +000072
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000073 self.raise_catch(IndexError, "IndexError")
74 x = []
75 self.assertRaises(IndexError, x.__getitem__, 10)
Guido van Rossum3bead091992-01-27 17:00:37 +000076
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000077 self.raise_catch(KeyError, "KeyError")
78 x = {}
79 self.assertRaises(KeyError, x.__getitem__, 'key')
Guido van Rossum3bead091992-01-27 17:00:37 +000080
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000081 self.raise_catch(KeyboardInterrupt, "KeyboardInterrupt")
Guido van Rossum3bead091992-01-27 17:00:37 +000082
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000083 self.raise_catch(MemoryError, "MemoryError")
Guido van Rossum3bead091992-01-27 17:00:37 +000084
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000085 self.raise_catch(NameError, "NameError")
86 try: x = undefined_variable
87 except NameError: pass
Guido van Rossum3bead091992-01-27 17:00:37 +000088
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000089 self.raise_catch(OverflowError, "OverflowError")
90 x = 1
91 for dummy in range(128):
92 x += x # this simply shouldn't blow up
Guido van Rossum3bead091992-01-27 17:00:37 +000093
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000094 self.raise_catch(RuntimeError, "RuntimeError")
Yury Selivanovf488fb42015-07-03 01:04:23 -040095 self.raise_catch(RecursionError, "RecursionError")
Guido van Rossum3bead091992-01-27 17:00:37 +000096
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000097 self.raise_catch(SyntaxError, "SyntaxError")
Georg Brandl7cae87c2006-09-06 06:51:57 +000098 try: exec('/\n')
Thomas Wouters4d70c3d2006-06-08 14:42:34 +000099 except SyntaxError: pass
Guido van Rossum3bead091992-01-27 17:00:37 +0000100
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000101 self.raise_catch(IndentationError, "IndentationError")
Fred Drake72e48bd2000-09-08 16:32:34 +0000102
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000103 self.raise_catch(TabError, "TabError")
Georg Brandle1b5ac62008-06-04 13:06:58 +0000104 try: compile("try:\n\t1/0\n \t1/0\nfinally:\n pass\n",
105 '<string>', 'exec')
106 except TabError: pass
107 else: self.fail("TabError not raised")
Fred Drake72e48bd2000-09-08 16:32:34 +0000108
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000109 self.raise_catch(SystemError, "SystemError")
Fred Drake72e48bd2000-09-08 16:32:34 +0000110
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000111 self.raise_catch(SystemExit, "SystemExit")
112 self.assertRaises(SystemExit, sys.exit, 0)
Fred Drake85f36392000-07-11 17:53:00 +0000113
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000114 self.raise_catch(TypeError, "TypeError")
115 try: [] + ()
116 except TypeError: pass
Fred Drake85f36392000-07-11 17:53:00 +0000117
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000118 self.raise_catch(ValueError, "ValueError")
Guido van Rossume63bae62007-07-17 00:34:25 +0000119 self.assertRaises(ValueError, chr, 17<<16)
Guido van Rossum3bead091992-01-27 17:00:37 +0000120
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000121 self.raise_catch(ZeroDivisionError, "ZeroDivisionError")
122 try: x = 1/0
123 except ZeroDivisionError: pass
Guido van Rossum3bead091992-01-27 17:00:37 +0000124
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000125 self.raise_catch(Exception, "Exception")
126 try: x = 1/0
Guido van Rossumb940e112007-01-10 16:19:56 +0000127 except Exception as e: pass
Guido van Rossum3bead091992-01-27 17:00:37 +0000128
Yury Selivanovccc897f2015-07-03 01:16:04 -0400129 self.raise_catch(StopAsyncIteration, "StopAsyncIteration")
130
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000131 def testSyntaxErrorMessage(self):
132 # make sure the right exception message is raised for each of
133 # these code fragments
Guido van Rossum3bead091992-01-27 17:00:37 +0000134
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000135 def ckmsg(src, msg):
136 try:
137 compile(src, '<fragment>', 'exec')
Guido van Rossumb940e112007-01-10 16:19:56 +0000138 except SyntaxError as e:
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000139 if e.msg != msg:
140 self.fail("expected %s, got %s" % (msg, e.msg))
141 else:
142 self.fail("failed to get expected SyntaxError")
Guido van Rossum3bead091992-01-27 17:00:37 +0000143
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000144 s = '''if 1:
145 try:
146 continue
147 except:
148 pass'''
Jeremy Hyltonede049b2001-09-26 20:01:13 +0000149
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000150 ckmsg(s, "'continue' not properly in loop")
151 ckmsg("continue\n", "'continue' not properly in loop")
Thomas Wouters303de6a2006-04-20 22:42:37 +0000152
Martijn Pieters772d8092017-08-22 21:16:23 +0100153 def testSyntaxErrorMissingParens(self):
154 def ckmsg(src, msg, exception=SyntaxError):
155 try:
156 compile(src, '<fragment>', 'exec')
157 except exception as e:
158 if e.msg != msg:
159 self.fail("expected %s, got %s" % (msg, e.msg))
160 else:
161 self.fail("failed to get expected SyntaxError")
162
163 s = '''print "old style"'''
164 ckmsg(s, "Missing parentheses in call to 'print'. "
165 "Did you mean print(\"old style\")?")
166
167 s = '''print "old style",'''
168 ckmsg(s, "Missing parentheses in call to 'print'. "
169 "Did you mean print(\"old style\", end=\" \")?")
170
171 s = '''exec "old style"'''
172 ckmsg(s, "Missing parentheses in call to 'exec'")
173
174 # should not apply to subclasses, see issue #31161
175 s = '''if True:\nprint "No indent"'''
176 ckmsg(s, "expected an indented block", IndentationError)
177
178 s = '''if True:\n print()\n\texec "mixed tabs and spaces"'''
179 ckmsg(s, "inconsistent use of tabs and spaces in indentation", TabError)
180
Serhiy Storchaka65fd0592014-01-21 22:26:52 +0200181 def testSyntaxErrorOffset(self):
Serhiy Storchaka0cc6b5e2020-02-12 12:17:00 +0200182 def check(src, lineno, offset, encoding='utf-8'):
Serhiy Storchaka65fd0592014-01-21 22:26:52 +0200183 with self.assertRaises(SyntaxError) as cm:
184 compile(src, '<fragment>', 'exec')
185 self.assertEqual(cm.exception.lineno, lineno)
186 self.assertEqual(cm.exception.offset, offset)
Serhiy Storchaka0cc6b5e2020-02-12 12:17:00 +0200187 if cm.exception.text is not None:
188 if not isinstance(src, str):
189 src = src.decode(encoding, 'replace')
190 line = src.split('\n')[lineno-1]
191 self.assertEqual(cm.exception.text.rstrip('\n'), line)
Serhiy Storchaka65fd0592014-01-21 22:26:52 +0200192
193 check('def fact(x):\n\treturn x!\n', 2, 10)
194 check('1 +\n', 1, 4)
195 check('def spam():\n print(1)\n print(2)', 3, 10)
196 check('Python = "Python" +', 1, 20)
197 check('Python = "\u1e54\xfd\u0163\u0125\xf2\xf1" +', 1, 20)
Serhiy Storchaka0cc6b5e2020-02-12 12:17:00 +0200198 check(b'# -*- coding: cp1251 -*-\nPython = "\xcf\xb3\xf2\xee\xed" +',
199 2, 19, encoding='cp1251')
200 check(b'Python = "\xcf\xb3\xf2\xee\xed" +', 1, 18)
Ammar Askar025eb982018-09-24 17:12:49 -0400201 check('x = "a', 1, 7)
202 check('lambda x: x = 2', 1, 1)
203
204 # Errors thrown by compile.c
205 check('class foo:return 1', 1, 11)
206 check('def f():\n continue', 2, 3)
207 check('def f():\n break', 2, 3)
208 check('try:\n pass\nexcept:\n pass\nexcept ValueError:\n pass', 2, 3)
209
210 # Errors thrown by tokenizer.c
211 check('(0x+1)', 1, 3)
212 check('x = 0xI', 1, 6)
213 check('0010 + 2', 1, 4)
214 check('x = 32e-+4', 1, 8)
215 check('x = 0o9', 1, 6)
Serhiy Storchaka0cc6b5e2020-02-12 12:17:00 +0200216 check('\u03b1 = 0xI', 1, 6)
217 check(b'\xce\xb1 = 0xI', 1, 6)
218 check(b'# -*- coding: iso8859-7 -*-\n\xe1 = 0xI', 2, 6,
219 encoding='iso8859-7')
Ammar Askar025eb982018-09-24 17:12:49 -0400220
221 # Errors thrown by symtable.c
Serhiy Storchakab619b092018-11-27 09:40:29 +0200222 check('x = [(yield i) for i in range(3)]', 1, 5)
Ammar Askar025eb982018-09-24 17:12:49 -0400223 check('def f():\n from _ import *', 1, 1)
224 check('def f(x, x):\n pass', 1, 1)
225 check('def f(x):\n nonlocal x', 2, 3)
226 check('def f(x):\n x = 1\n global x', 3, 3)
227 check('nonlocal x', 1, 1)
228 check('def f():\n global x\n nonlocal x', 2, 3)
229
230 # Errors thrown by ast.c
231 check('for 1 in []: pass', 1, 5)
232 check('def f(*):\n pass', 1, 7)
233 check('[*x for x in xs]', 1, 2)
234 check('def f():\n x, y: int', 2, 3)
235 check('(yield i) = 2', 1, 1)
236 check('foo(x for x in range(10), 100)', 1, 5)
237 check('foo(1=2)', 1, 5)
238
239 # Errors thrown by future.c
240 check('from __future__ import doesnt_exist', 1, 1)
241 check('from __future__ import braces', 1, 1)
242 check('x=1\nfrom __future__ import division', 2, 1)
243
Serhiy Storchaka65fd0592014-01-21 22:26:52 +0200244
Benjamin Peterson17e0bbc2010-06-28 15:39:55 +0000245 @cpython_only
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000246 def testSettingException(self):
247 # test that setting an exception at the C level works even if the
248 # exception object can't be constructed.
Jeremy Hyltonede049b2001-09-26 20:01:13 +0000249
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000250 class BadException(Exception):
251 def __init__(self_):
Collin Winter828f04a2007-08-31 00:04:24 +0000252 raise RuntimeError("can't instantiate BadException")
Finn Bockaa3dc452001-12-08 10:15:48 +0000253
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000254 class InvalidException:
255 pass
Thomas Wouters303de6a2006-04-20 22:42:37 +0000256
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000257 def test_capi1():
258 import _testcapi
259 try:
260 _testcapi.raise_exception(BadException, 1)
Guido van Rossumb940e112007-01-10 16:19:56 +0000261 except TypeError as err:
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000262 exc, err, tb = sys.exc_info()
263 co = tb.tb_frame.f_code
Ezio Melottib3aedd42010-11-20 19:04:17 +0000264 self.assertEqual(co.co_name, "test_capi1")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000265 self.assertTrue(co.co_filename.endswith('test_exceptions.py'))
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000266 else:
267 self.fail("Expected exception")
Jeremy Hyltonede049b2001-09-26 20:01:13 +0000268
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000269 def test_capi2():
270 import _testcapi
271 try:
272 _testcapi.raise_exception(BadException, 0)
Guido van Rossumb940e112007-01-10 16:19:56 +0000273 except RuntimeError as err:
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000274 exc, err, tb = sys.exc_info()
275 co = tb.tb_frame.f_code
Ezio Melottib3aedd42010-11-20 19:04:17 +0000276 self.assertEqual(co.co_name, "__init__")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000277 self.assertTrue(co.co_filename.endswith('test_exceptions.py'))
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000278 co2 = tb.tb_frame.f_back.f_code
Ezio Melottib3aedd42010-11-20 19:04:17 +0000279 self.assertEqual(co2.co_name, "test_capi2")
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000280 else:
281 self.fail("Expected exception")
Thomas Wouters477c8d52006-05-27 19:21:47 +0000282
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000283 def test_capi3():
284 import _testcapi
285 self.assertRaises(SystemError, _testcapi.raise_exception,
286 InvalidException, 1)
287
288 if not sys.platform.startswith('java'):
289 test_capi1()
290 test_capi2()
291 test_capi3()
292
Thomas Wouters89f507f2006-12-13 04:49:30 +0000293 def test_WindowsError(self):
294 try:
295 WindowsError
296 except NameError:
297 pass
298 else:
Antoine Pitrou6b4883d2011-10-12 02:54:14 +0200299 self.assertIs(WindowsError, OSError)
300 self.assertEqual(str(OSError(1001)), "1001")
301 self.assertEqual(str(OSError(1001, "message")),
302 "[Errno 1001] message")
303 # POSIX errno (9 aka EBADF) is untranslated
304 w = OSError(9, 'foo', 'bar')
305 self.assertEqual(w.errno, 9)
306 self.assertEqual(w.winerror, None)
307 self.assertEqual(str(w), "[Errno 9] foo: 'bar'")
308 # ERROR_PATH_NOT_FOUND (win error 3) becomes ENOENT (2)
309 w = OSError(0, 'foo', 'bar', 3)
310 self.assertEqual(w.errno, 2)
311 self.assertEqual(w.winerror, 3)
312 self.assertEqual(w.strerror, 'foo')
313 self.assertEqual(w.filename, 'bar')
Martin Panter5487c132015-10-26 11:05:42 +0000314 self.assertEqual(w.filename2, None)
Richard Oudkerk30147712012-08-28 19:33:26 +0100315 self.assertEqual(str(w), "[WinError 3] foo: 'bar'")
Antoine Pitrou6b4883d2011-10-12 02:54:14 +0200316 # Unknown win error becomes EINVAL (22)
317 w = OSError(0, 'foo', None, 1001)
318 self.assertEqual(w.errno, 22)
319 self.assertEqual(w.winerror, 1001)
320 self.assertEqual(w.strerror, 'foo')
321 self.assertEqual(w.filename, None)
Martin Panter5487c132015-10-26 11:05:42 +0000322 self.assertEqual(w.filename2, None)
Richard Oudkerk30147712012-08-28 19:33:26 +0100323 self.assertEqual(str(w), "[WinError 1001] foo")
Antoine Pitrou6b4883d2011-10-12 02:54:14 +0200324 # Non-numeric "errno"
325 w = OSError('bar', 'foo')
326 self.assertEqual(w.errno, 'bar')
327 self.assertEqual(w.winerror, None)
328 self.assertEqual(w.strerror, 'foo')
329 self.assertEqual(w.filename, None)
Martin Panter5487c132015-10-26 11:05:42 +0000330 self.assertEqual(w.filename2, None)
Thomas Wouters89f507f2006-12-13 04:49:30 +0000331
Victor Stinnerd223fa62015-04-02 14:17:38 +0200332 @unittest.skipUnless(sys.platform == 'win32',
333 'test specific to Windows')
Serhiy Storchakaf41f8f92015-04-02 09:47:27 +0300334 def test_windows_message(self):
335 """Should fill in unknown error code in Windows error message"""
Victor Stinnerd223fa62015-04-02 14:17:38 +0200336 ctypes = import_module('ctypes')
337 # this error code has no message, Python formats it as hexadecimal
338 code = 3765269347
339 with self.assertRaisesRegex(OSError, 'Windows Error 0x%x' % code):
340 ctypes.pythonapi.PyErr_SetFromWindowsErr(code)
Serhiy Storchakaf41f8f92015-04-02 09:47:27 +0300341
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000342 def testAttributes(self):
343 # test that exception attributes are happy
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000344
345 exceptionList = [
Guido van Rossumebe3e162007-05-17 18:20:34 +0000346 (BaseException, (), {'args' : ()}),
347 (BaseException, (1, ), {'args' : (1,)}),
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000348 (BaseException, ('foo',),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000349 {'args' : ('foo',)}),
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000350 (BaseException, ('foo', 1),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000351 {'args' : ('foo', 1)}),
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000352 (SystemExit, ('foo',),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000353 {'args' : ('foo',), 'code' : 'foo'}),
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200354 (OSError, ('foo',),
Martin Panter5487c132015-10-26 11:05:42 +0000355 {'args' : ('foo',), 'filename' : None, 'filename2' : None,
Thomas Wouters89f507f2006-12-13 04:49:30 +0000356 'errno' : None, 'strerror' : None}),
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200357 (OSError, ('foo', 'bar'),
Martin Panter5487c132015-10-26 11:05:42 +0000358 {'args' : ('foo', 'bar'),
359 'filename' : None, 'filename2' : None,
Thomas Wouters89f507f2006-12-13 04:49:30 +0000360 'errno' : 'foo', 'strerror' : 'bar'}),
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200361 (OSError, ('foo', 'bar', 'baz'),
Martin Panter5487c132015-10-26 11:05:42 +0000362 {'args' : ('foo', 'bar'),
363 'filename' : 'baz', 'filename2' : None,
Thomas Wouters89f507f2006-12-13 04:49:30 +0000364 'errno' : 'foo', 'strerror' : 'bar'}),
Larry Hastingsb0827312014-02-09 22:05:19 -0800365 (OSError, ('foo', 'bar', 'baz', None, 'quux'),
366 {'args' : ('foo', 'bar'), 'filename' : 'baz', 'filename2': 'quux'}),
Andrew Svetlov3438fa42012-12-17 23:35:18 +0200367 (OSError, ('errnoStr', 'strErrorStr', 'filenameStr'),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000368 {'args' : ('errnoStr', 'strErrorStr'),
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000369 'strerror' : 'strErrorStr', 'errno' : 'errnoStr',
370 'filename' : 'filenameStr'}),
Andrew Svetlov3438fa42012-12-17 23:35:18 +0200371 (OSError, (1, 'strErrorStr', 'filenameStr'),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000372 {'args' : (1, 'strErrorStr'), 'errno' : 1,
Martin Panter5487c132015-10-26 11:05:42 +0000373 'strerror' : 'strErrorStr',
374 'filename' : 'filenameStr', 'filename2' : None}),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000375 (SyntaxError, (), {'msg' : None, 'text' : None,
Guido van Rossumd8faa362007-04-27 19:54:29 +0000376 'filename' : None, 'lineno' : None, 'offset' : None,
377 'print_file_and_line' : None}),
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000378 (SyntaxError, ('msgStr',),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000379 {'args' : ('msgStr',), 'text' : None,
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000380 'print_file_and_line' : None, 'msg' : 'msgStr',
381 'filename' : None, 'lineno' : None, 'offset' : None}),
382 (SyntaxError, ('msgStr', ('filenameStr', 'linenoStr', 'offsetStr',
383 'textStr')),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000384 {'offset' : 'offsetStr', 'text' : 'textStr',
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000385 'args' : ('msgStr', ('filenameStr', 'linenoStr',
386 'offsetStr', 'textStr')),
387 'print_file_and_line' : None, 'msg' : 'msgStr',
388 'filename' : 'filenameStr', 'lineno' : 'linenoStr'}),
389 (SyntaxError, ('msgStr', 'filenameStr', 'linenoStr', 'offsetStr',
390 'textStr', 'print_file_and_lineStr'),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000391 {'text' : None,
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000392 'args' : ('msgStr', 'filenameStr', 'linenoStr', 'offsetStr',
393 'textStr', 'print_file_and_lineStr'),
394 'print_file_and_line' : None, 'msg' : 'msgStr',
395 'filename' : None, 'lineno' : None, 'offset' : None}),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000396 (UnicodeError, (), {'args' : (),}),
Walter Dörwaldeceb0fb2007-05-24 17:49:56 +0000397 (UnicodeEncodeError, ('ascii', 'a', 0, 1,
398 'ordinal not in range'),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000399 {'args' : ('ascii', 'a', 0, 1,
Thomas Wouters89f507f2006-12-13 04:49:30 +0000400 'ordinal not in range'),
Guido van Rossumef87d6e2007-05-02 19:09:54 +0000401 'encoding' : 'ascii', 'object' : 'a',
Thomas Wouters89f507f2006-12-13 04:49:30 +0000402 'start' : 0, 'reason' : 'ordinal not in range'}),
Guido van Rossum254348e2007-11-21 19:29:53 +0000403 (UnicodeDecodeError, ('ascii', bytearray(b'\xff'), 0, 1,
Guido van Rossum98297ee2007-11-06 21:34:58 +0000404 'ordinal not in range'),
Guido van Rossum254348e2007-11-21 19:29:53 +0000405 {'args' : ('ascii', bytearray(b'\xff'), 0, 1,
Guido van Rossum98297ee2007-11-06 21:34:58 +0000406 'ordinal not in range'),
407 'encoding' : 'ascii', 'object' : b'\xff',
408 'start' : 0, 'reason' : 'ordinal not in range'}),
Walter Dörwaldeceb0fb2007-05-24 17:49:56 +0000409 (UnicodeDecodeError, ('ascii', b'\xff', 0, 1,
410 'ordinal not in range'),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000411 {'args' : ('ascii', b'\xff', 0, 1,
Thomas Wouters89f507f2006-12-13 04:49:30 +0000412 'ordinal not in range'),
Guido van Rossumb8142c32007-05-08 17:49:10 +0000413 'encoding' : 'ascii', 'object' : b'\xff',
Thomas Wouters89f507f2006-12-13 04:49:30 +0000414 'start' : 0, 'reason' : 'ordinal not in range'}),
Walter Dörwaldeceb0fb2007-05-24 17:49:56 +0000415 (UnicodeTranslateError, ("\u3042", 0, 1, "ouch"),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000416 {'args' : ('\u3042', 0, 1, 'ouch'),
Guido van Rossumef87d6e2007-05-02 19:09:54 +0000417 'object' : '\u3042', 'reason' : 'ouch',
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000418 'start' : 0, 'end' : 1}),
Richard Oudkerk5562d9d2012-07-28 17:45:28 +0100419 (NaiveException, ('foo',),
420 {'args': ('foo',), 'x': 'foo'}),
421 (SlottedNaiveException, ('foo',),
422 {'args': ('foo',), 'x': 'foo'}),
Thomas Wouters477c8d52006-05-27 19:21:47 +0000423 ]
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000424 try:
Antoine Pitrou6b4883d2011-10-12 02:54:14 +0200425 # More tests are in test_WindowsError
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000426 exceptionList.append(
427 (WindowsError, (1, 'strErrorStr', 'filenameStr'),
Guido van Rossumebe3e162007-05-17 18:20:34 +0000428 {'args' : (1, 'strErrorStr'),
Antoine Pitrou6b4883d2011-10-12 02:54:14 +0200429 'strerror' : 'strErrorStr', 'winerror' : None,
Martin Panter5487c132015-10-26 11:05:42 +0000430 'errno' : 1,
431 'filename' : 'filenameStr', 'filename2' : None})
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000432 )
Thomas Wouters73e5a5b2006-06-08 15:35:45 +0000433 except NameError:
434 pass
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000435
Guido van Rossumebe3e162007-05-17 18:20:34 +0000436 for exc, args, expected in exceptionList:
437 try:
438 e = exc(*args)
439 except:
Guido van Rossum98297ee2007-11-06 21:34:58 +0000440 print("\nexc=%r, args=%r" % (exc, args), file=sys.stderr)
Guido van Rossumebe3e162007-05-17 18:20:34 +0000441 raise
442 else:
443 # Verify module name
Richard Oudkerk5562d9d2012-07-28 17:45:28 +0100444 if not type(e).__name__.endswith('NaiveException'):
445 self.assertEqual(type(e).__module__, 'builtins')
Guido van Rossumebe3e162007-05-17 18:20:34 +0000446 # Verify no ref leaks in Exc_str()
447 s = str(e)
448 for checkArgName in expected:
449 value = getattr(e, checkArgName)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000450 self.assertEqual(repr(value),
451 repr(expected[checkArgName]),
452 '%r.%s == %r, expected %r' % (
453 e, checkArgName,
454 value, expected[checkArgName]))
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000455
Guido van Rossumebe3e162007-05-17 18:20:34 +0000456 # test for pickling support
Guido van Rossum99603b02007-07-20 00:22:32 +0000457 for p in [pickle]:
Guido van Rossumebe3e162007-05-17 18:20:34 +0000458 for protocol in range(p.HIGHEST_PROTOCOL + 1):
459 s = p.dumps(e, protocol)
460 new = p.loads(s)
461 for checkArgName in expected:
462 got = repr(getattr(new, checkArgName))
463 want = repr(expected[checkArgName])
Ezio Melottib3aedd42010-11-20 19:04:17 +0000464 self.assertEqual(got, want,
465 'pickled "%r", attribute "%s' %
466 (e, checkArgName))
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000467
Collin Winter828f04a2007-08-31 00:04:24 +0000468 def testWithTraceback(self):
469 try:
470 raise IndexError(4)
471 except:
472 tb = sys.exc_info()[2]
473
474 e = BaseException().with_traceback(tb)
Ezio Melottie9615932010-01-24 19:26:24 +0000475 self.assertIsInstance(e, BaseException)
Collin Winter828f04a2007-08-31 00:04:24 +0000476 self.assertEqual(e.__traceback__, tb)
477
478 e = IndexError(5).with_traceback(tb)
Ezio Melottie9615932010-01-24 19:26:24 +0000479 self.assertIsInstance(e, IndexError)
Collin Winter828f04a2007-08-31 00:04:24 +0000480 self.assertEqual(e.__traceback__, tb)
481
482 class MyException(Exception):
483 pass
484
485 e = MyException().with_traceback(tb)
Ezio Melottie9615932010-01-24 19:26:24 +0000486 self.assertIsInstance(e, MyException)
Collin Winter828f04a2007-08-31 00:04:24 +0000487 self.assertEqual(e.__traceback__, tb)
488
489 def testInvalidTraceback(self):
490 try:
491 Exception().__traceback__ = 5
492 except TypeError as e:
Benjamin Peterson577473f2010-01-19 00:09:57 +0000493 self.assertIn("__traceback__ must be a traceback", str(e))
Collin Winter828f04a2007-08-31 00:04:24 +0000494 else:
495 self.fail("No exception raised")
496
Georg Brandlab6f2f62009-03-31 04:16:10 +0000497 def testInvalidAttrs(self):
498 self.assertRaises(TypeError, setattr, Exception(), '__cause__', 1)
499 self.assertRaises(TypeError, delattr, Exception(), '__cause__')
500 self.assertRaises(TypeError, setattr, Exception(), '__context__', 1)
501 self.assertRaises(TypeError, delattr, Exception(), '__context__')
502
Collin Winter828f04a2007-08-31 00:04:24 +0000503 def testNoneClearsTracebackAttr(self):
504 try:
505 raise IndexError(4)
506 except:
507 tb = sys.exc_info()[2]
508
509 e = Exception()
510 e.__traceback__ = tb
511 e.__traceback__ = None
512 self.assertEqual(e.__traceback__, None)
513
514 def testChainingAttrs(self):
515 e = Exception()
Nick Coghlanab7bf212012-02-26 17:49:52 +1000516 self.assertIsNone(e.__context__)
Benjamin Petersond5a1c442012-05-14 22:09:31 -0700517 self.assertIsNone(e.__cause__)
Collin Winter828f04a2007-08-31 00:04:24 +0000518
519 e = TypeError()
Nick Coghlanab7bf212012-02-26 17:49:52 +1000520 self.assertIsNone(e.__context__)
Benjamin Petersond5a1c442012-05-14 22:09:31 -0700521 self.assertIsNone(e.__cause__)
Collin Winter828f04a2007-08-31 00:04:24 +0000522
Andrew Svetlov3438fa42012-12-17 23:35:18 +0200523 class MyException(OSError):
Collin Winter828f04a2007-08-31 00:04:24 +0000524 pass
525
526 e = MyException()
Nick Coghlanab7bf212012-02-26 17:49:52 +1000527 self.assertIsNone(e.__context__)
Benjamin Petersond5a1c442012-05-14 22:09:31 -0700528 self.assertIsNone(e.__cause__)
Nick Coghlanab7bf212012-02-26 17:49:52 +1000529
530 def testChainingDescriptors(self):
531 try:
532 raise Exception()
533 except Exception as exc:
534 e = exc
535
536 self.assertIsNone(e.__context__)
Benjamin Petersond5a1c442012-05-14 22:09:31 -0700537 self.assertIsNone(e.__cause__)
538 self.assertFalse(e.__suppress_context__)
Nick Coghlanab7bf212012-02-26 17:49:52 +1000539
540 e.__context__ = NameError()
541 e.__cause__ = None
542 self.assertIsInstance(e.__context__, NameError)
543 self.assertIsNone(e.__cause__)
Benjamin Petersond5a1c442012-05-14 22:09:31 -0700544 self.assertTrue(e.__suppress_context__)
545 e.__suppress_context__ = False
546 self.assertFalse(e.__suppress_context__)
Collin Winter828f04a2007-08-31 00:04:24 +0000547
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000548 def testKeywordArgs(self):
549 # test that builtin exception don't take keyword args,
550 # but user-defined subclasses can if they want
551 self.assertRaises(TypeError, BaseException, a=1)
552
553 class DerivedException(BaseException):
554 def __init__(self, fancy_arg):
555 BaseException.__init__(self)
556 self.fancy_arg = fancy_arg
557
558 x = DerivedException(fancy_arg=42)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000559 self.assertEqual(x.fancy_arg, 42)
Thomas Wouters4d70c3d2006-06-08 14:42:34 +0000560
Brett Cannon31f59292011-02-21 19:29:56 +0000561 @no_tracing
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000562 def testInfiniteRecursion(self):
563 def f():
564 return f()
Yury Selivanovf488fb42015-07-03 01:04:23 -0400565 self.assertRaises(RecursionError, f)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000566
567 def g():
568 try:
569 return g()
570 except ValueError:
571 return -1
Yury Selivanovf488fb42015-07-03 01:04:23 -0400572 self.assertRaises(RecursionError, g)
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000573
Ezio Melotti2f5a78c2009-12-24 22:54:06 +0000574 def test_str(self):
575 # Make sure both instances and classes have a str representation.
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000576 self.assertTrue(str(Exception))
577 self.assertTrue(str(Exception('a')))
Ezio Melotti2f5a78c2009-12-24 22:54:06 +0000578 self.assertTrue(str(Exception('a', 'b')))
Thomas Wouters89f507f2006-12-13 04:49:30 +0000579
Barry Warsaw8d109cb2008-05-08 04:26:35 +0000580 def testExceptionCleanupNames(self):
581 # Make sure the local variable bound to the exception instance by
582 # an "except" statement is only visible inside the except block.
Guido van Rossumb940e112007-01-10 16:19:56 +0000583 try:
584 raise Exception()
585 except Exception as e:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000586 self.assertTrue(e)
Guido van Rossumb940e112007-01-10 16:19:56 +0000587 del e
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000588 self.assertNotIn('e', locals())
Guido van Rossumb940e112007-01-10 16:19:56 +0000589
Barry Warsaw8d109cb2008-05-08 04:26:35 +0000590 def testExceptionCleanupState(self):
591 # Make sure exception state is cleaned up as soon as the except
592 # block is left. See #2507
593
594 class MyException(Exception):
595 def __init__(self, obj):
596 self.obj = obj
597 class MyObj:
598 pass
599
600 def inner_raising_func():
601 # Create some references in exception value and traceback
602 local_ref = obj
603 raise MyException(obj)
604
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000605 # Qualified "except" with "as"
Barry Warsaw8d109cb2008-05-08 04:26:35 +0000606 obj = MyObj()
607 wr = weakref.ref(obj)
608 try:
609 inner_raising_func()
610 except MyException as e:
611 pass
612 obj = None
613 obj = wr()
Serhiy Storchakaf15c4d32017-03-30 18:05:08 +0300614 self.assertIsNone(obj)
Barry Warsaw8d109cb2008-05-08 04:26:35 +0000615
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000616 # Qualified "except" without "as"
617 obj = MyObj()
618 wr = weakref.ref(obj)
619 try:
620 inner_raising_func()
621 except MyException:
622 pass
623 obj = None
624 obj = wr()
Serhiy Storchakaf15c4d32017-03-30 18:05:08 +0300625 self.assertIsNone(obj)
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000626
627 # Bare "except"
628 obj = MyObj()
629 wr = weakref.ref(obj)
630 try:
631 inner_raising_func()
632 except:
633 pass
634 obj = None
635 obj = wr()
Serhiy Storchakaf15c4d32017-03-30 18:05:08 +0300636 self.assertIsNone(obj)
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000637
638 # "except" with premature block leave
639 obj = MyObj()
640 wr = weakref.ref(obj)
641 for i in [0]:
642 try:
643 inner_raising_func()
644 except:
645 break
646 obj = None
647 obj = wr()
Serhiy Storchakaf15c4d32017-03-30 18:05:08 +0300648 self.assertIsNone(obj)
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000649
650 # "except" block raising another exception
651 obj = MyObj()
652 wr = weakref.ref(obj)
653 try:
654 try:
655 inner_raising_func()
656 except:
657 raise KeyError
Guido van Rossumb4fb6e42008-06-14 20:20:24 +0000658 except KeyError as e:
659 # We want to test that the except block above got rid of
660 # the exception raised in inner_raising_func(), but it
661 # also ends up in the __context__ of the KeyError, so we
662 # must clear the latter manually for our test to succeed.
663 e.__context__ = None
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000664 obj = None
665 obj = wr()
Philip Jenveyb37ac8e2012-11-14 14:37:24 -0800666 # guarantee no ref cycles on CPython (don't gc_collect)
667 if check_impl_detail(cpython=False):
668 gc_collect()
Serhiy Storchakaf15c4d32017-03-30 18:05:08 +0300669 self.assertIsNone(obj)
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000670
671 # Some complicated construct
672 obj = MyObj()
673 wr = weakref.ref(obj)
674 try:
675 inner_raising_func()
676 except MyException:
677 try:
678 try:
679 raise
680 finally:
681 raise
682 except MyException:
683 pass
684 obj = None
Philip Jenveyb37ac8e2012-11-14 14:37:24 -0800685 if check_impl_detail(cpython=False):
686 gc_collect()
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000687 obj = wr()
Serhiy Storchakaf15c4d32017-03-30 18:05:08 +0300688 self.assertIsNone(obj)
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000689
690 # Inside an exception-silencing "with" block
691 class Context:
692 def __enter__(self):
693 return self
694 def __exit__ (self, exc_type, exc_value, exc_tb):
695 return True
696 obj = MyObj()
697 wr = weakref.ref(obj)
698 with Context():
699 inner_raising_func()
700 obj = None
Philip Jenveyb37ac8e2012-11-14 14:37:24 -0800701 if check_impl_detail(cpython=False):
702 gc_collect()
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000703 obj = wr()
Serhiy Storchakaf15c4d32017-03-30 18:05:08 +0300704 self.assertIsNone(obj)
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000705
Amaury Forgeot d'Arcba117ef2010-09-10 21:39:53 +0000706 def test_exception_target_in_nested_scope(self):
707 # issue 4617: This used to raise a SyntaxError
708 # "can not delete variable 'e' referenced in nested scope"
709 def print_error():
710 e
711 try:
712 something
713 except Exception as e:
714 print_error()
715 # implicit "del e" here
716
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000717 def test_generator_leaking(self):
718 # Test that generator exception state doesn't leak into the calling
719 # frame
720 def yield_raise():
721 try:
722 raise KeyError("caught")
723 except KeyError:
724 yield sys.exc_info()[0]
725 yield sys.exc_info()[0]
726 yield sys.exc_info()[0]
727 g = yield_raise()
Ezio Melottib3aedd42010-11-20 19:04:17 +0000728 self.assertEqual(next(g), KeyError)
729 self.assertEqual(sys.exc_info()[0], None)
730 self.assertEqual(next(g), KeyError)
731 self.assertEqual(sys.exc_info()[0], None)
732 self.assertEqual(next(g), None)
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000733
734 # Same test, but inside an exception handler
735 try:
736 raise TypeError("foo")
737 except TypeError:
738 g = yield_raise()
Ezio Melottib3aedd42010-11-20 19:04:17 +0000739 self.assertEqual(next(g), KeyError)
740 self.assertEqual(sys.exc_info()[0], TypeError)
741 self.assertEqual(next(g), KeyError)
742 self.assertEqual(sys.exc_info()[0], TypeError)
743 self.assertEqual(next(g), TypeError)
Benjamin Petersoneec3d712008-06-11 15:59:43 +0000744 del g
Ezio Melottib3aedd42010-11-20 19:04:17 +0000745 self.assertEqual(sys.exc_info()[0], TypeError)
Thomas Wouters89f507f2006-12-13 04:49:30 +0000746
Benjamin Peterson83195c32011-07-03 13:44:00 -0500747 def test_generator_leaking2(self):
748 # See issue 12475.
749 def g():
750 yield
751 try:
752 raise RuntimeError
753 except RuntimeError:
754 it = g()
755 next(it)
756 try:
757 next(it)
758 except StopIteration:
759 pass
760 self.assertEqual(sys.exc_info(), (None, None, None))
761
Antoine Pitrouc4c19b32015-03-18 22:22:46 +0100762 def test_generator_leaking3(self):
763 # See issue #23353. When gen.throw() is called, the caller's
764 # exception state should be save and restored.
765 def g():
766 try:
767 yield
768 except ZeroDivisionError:
769 yield sys.exc_info()[1]
770 it = g()
771 next(it)
772 try:
773 1/0
774 except ZeroDivisionError as e:
775 self.assertIs(sys.exc_info()[1], e)
776 gen_exc = it.throw(e)
777 self.assertIs(sys.exc_info()[1], e)
778 self.assertIs(gen_exc, e)
779 self.assertEqual(sys.exc_info(), (None, None, None))
780
781 def test_generator_leaking4(self):
782 # See issue #23353. When an exception is raised by a generator,
783 # the caller's exception state should still be restored.
784 def g():
785 try:
786 1/0
787 except ZeroDivisionError:
788 yield sys.exc_info()[0]
789 raise
790 it = g()
791 try:
792 raise TypeError
793 except TypeError:
794 # The caller's exception state (TypeError) is temporarily
795 # saved in the generator.
796 tp = next(it)
797 self.assertIs(tp, ZeroDivisionError)
798 try:
799 next(it)
800 # We can't check it immediately, but while next() returns
801 # with an exception, it shouldn't have restored the old
802 # exception state (TypeError).
803 except ZeroDivisionError as e:
804 self.assertIs(sys.exc_info()[1], e)
805 # We used to find TypeError here.
806 self.assertEqual(sys.exc_info(), (None, None, None))
807
Benjamin Petersonac913412011-07-03 16:25:11 -0500808 def test_generator_doesnt_retain_old_exc(self):
809 def g():
810 self.assertIsInstance(sys.exc_info()[1], RuntimeError)
811 yield
812 self.assertEqual(sys.exc_info(), (None, None, None))
813 it = g()
814 try:
815 raise RuntimeError
816 except RuntimeError:
817 next(it)
818 self.assertRaises(StopIteration, next, it)
819
Benjamin Petersonae5f2f42010-03-07 17:10:51 +0000820 def test_generator_finalizing_and_exc_info(self):
821 # See #7173
822 def simple_gen():
823 yield 1
824 def run_gen():
825 gen = simple_gen()
826 try:
827 raise RuntimeError
828 except RuntimeError:
829 return next(gen)
830 run_gen()
831 gc_collect()
832 self.assertEqual(sys.exc_info(), (None, None, None))
833
Antoine Pitroua370fcf2011-08-20 14:15:03 +0200834 def _check_generator_cleanup_exc_state(self, testfunc):
835 # Issue #12791: exception state is cleaned up as soon as a generator
836 # is closed (reference cycles are broken).
837 class MyException(Exception):
838 def __init__(self, obj):
839 self.obj = obj
840 class MyObj:
841 pass
842
843 def raising_gen():
844 try:
845 raise MyException(obj)
846 except MyException:
847 yield
848
849 obj = MyObj()
850 wr = weakref.ref(obj)
851 g = raising_gen()
852 next(g)
853 testfunc(g)
854 g = obj = None
855 obj = wr()
Serhiy Storchakaf15c4d32017-03-30 18:05:08 +0300856 self.assertIsNone(obj)
Antoine Pitroua370fcf2011-08-20 14:15:03 +0200857
858 def test_generator_throw_cleanup_exc_state(self):
859 def do_throw(g):
860 try:
861 g.throw(RuntimeError())
862 except RuntimeError:
863 pass
864 self._check_generator_cleanup_exc_state(do_throw)
865
866 def test_generator_close_cleanup_exc_state(self):
867 def do_close(g):
868 g.close()
869 self._check_generator_cleanup_exc_state(do_close)
870
871 def test_generator_del_cleanup_exc_state(self):
872 def do_del(g):
873 g = None
874 self._check_generator_cleanup_exc_state(do_del)
875
876 def test_generator_next_cleanup_exc_state(self):
877 def do_next(g):
878 try:
879 next(g)
880 except StopIteration:
881 pass
882 else:
883 self.fail("should have raised StopIteration")
884 self._check_generator_cleanup_exc_state(do_next)
885
886 def test_generator_send_cleanup_exc_state(self):
887 def do_send(g):
888 try:
889 g.send(None)
890 except StopIteration:
891 pass
892 else:
893 self.fail("should have raised StopIteration")
894 self._check_generator_cleanup_exc_state(do_send)
895
Benjamin Peterson27d63672008-06-15 20:09:12 +0000896 def test_3114(self):
897 # Bug #3114: in its destructor, MyObject retrieves a pointer to
898 # obsolete and/or deallocated objects.
Benjamin Peterson979f3112008-06-15 00:05:44 +0000899 class MyObject:
900 def __del__(self):
901 nonlocal e
902 e = sys.exc_info()
903 e = ()
904 try:
905 raise Exception(MyObject())
906 except:
907 pass
Ezio Melottib3aedd42010-11-20 19:04:17 +0000908 self.assertEqual(e, (None, None, None))
Benjamin Peterson979f3112008-06-15 00:05:44 +0000909
Benjamin Peterson24dfb052014-04-02 12:05:35 -0400910 def test_unicode_change_attributes(self):
Eric Smith0facd772010-02-24 15:42:29 +0000911 # See issue 7309. This was a crasher.
912
913 u = UnicodeEncodeError('baz', 'xxxxx', 1, 5, 'foo')
914 self.assertEqual(str(u), "'baz' codec can't encode characters in position 1-4: foo")
915 u.end = 2
916 self.assertEqual(str(u), "'baz' codec can't encode character '\\x78' in position 1: foo")
917 u.end = 5
918 u.reason = 0x345345345345345345
919 self.assertEqual(str(u), "'baz' codec can't encode characters in position 1-4: 965230951443685724997")
920 u.encoding = 4000
921 self.assertEqual(str(u), "'4000' codec can't encode characters in position 1-4: 965230951443685724997")
922 u.start = 1000
923 self.assertEqual(str(u), "'4000' codec can't encode characters in position 1000-4: 965230951443685724997")
924
925 u = UnicodeDecodeError('baz', b'xxxxx', 1, 5, 'foo')
926 self.assertEqual(str(u), "'baz' codec can't decode bytes in position 1-4: foo")
927 u.end = 2
928 self.assertEqual(str(u), "'baz' codec can't decode byte 0x78 in position 1: foo")
929 u.end = 5
930 u.reason = 0x345345345345345345
931 self.assertEqual(str(u), "'baz' codec can't decode bytes in position 1-4: 965230951443685724997")
932 u.encoding = 4000
933 self.assertEqual(str(u), "'4000' codec can't decode bytes in position 1-4: 965230951443685724997")
934 u.start = 1000
935 self.assertEqual(str(u), "'4000' codec can't decode bytes in position 1000-4: 965230951443685724997")
936
937 u = UnicodeTranslateError('xxxx', 1, 5, 'foo')
938 self.assertEqual(str(u), "can't translate characters in position 1-4: foo")
939 u.end = 2
940 self.assertEqual(str(u), "can't translate character '\\x78' in position 1: foo")
941 u.end = 5
942 u.reason = 0x345345345345345345
943 self.assertEqual(str(u), "can't translate characters in position 1-4: 965230951443685724997")
944 u.start = 1000
945 self.assertEqual(str(u), "can't translate characters in position 1000-4: 965230951443685724997")
Benjamin Peterson6e7740c2008-08-20 23:23:34 +0000946
Benjamin Peterson9b09ba12014-04-02 12:15:06 -0400947 def test_unicode_errors_no_object(self):
948 # See issue #21134.
Benjamin Petersone3311212014-04-02 15:51:38 -0400949 klasses = UnicodeEncodeError, UnicodeDecodeError, UnicodeTranslateError
Benjamin Peterson9b09ba12014-04-02 12:15:06 -0400950 for klass in klasses:
951 self.assertEqual(str(klass.__new__(klass)), "")
952
Brett Cannon31f59292011-02-21 19:29:56 +0000953 @no_tracing
Benjamin Peterson69c88f72008-07-31 01:47:08 +0000954 def test_badisinstance(self):
955 # Bug #2542: if issubclass(e, MyException) raises an exception,
956 # it should be ignored
957 class Meta(type):
958 def __subclasscheck__(cls, subclass):
959 raise ValueError()
960 class MyException(Exception, metaclass=Meta):
961 pass
962
Martin Panter3263f682016-02-28 03:16:11 +0000963 with captured_stderr() as stderr:
Benjamin Peterson69c88f72008-07-31 01:47:08 +0000964 try:
965 raise KeyError()
966 except MyException as e:
967 self.fail("exception should not be a MyException")
968 except KeyError:
969 pass
970 except:
Antoine Pitrouec569b72008-08-26 22:40:48 +0000971 self.fail("Should have raised KeyError")
Benjamin Peterson69c88f72008-07-31 01:47:08 +0000972 else:
Antoine Pitrouec569b72008-08-26 22:40:48 +0000973 self.fail("Should have raised KeyError")
974
975 def g():
976 try:
977 return g()
Yury Selivanovf488fb42015-07-03 01:04:23 -0400978 except RecursionError:
Antoine Pitrouec569b72008-08-26 22:40:48 +0000979 return sys.exc_info()
980 e, v, tb = g()
Serhiy Storchakaf15c4d32017-03-30 18:05:08 +0300981 self.assertIsInstance(v, RecursionError, type(v))
Benjamin Peterson577473f2010-01-19 00:09:57 +0000982 self.assertIn("maximum recursion depth exceeded", str(v))
Benjamin Peterson69c88f72008-07-31 01:47:08 +0000983
xdegaye56d1f5c2017-10-26 15:09:06 +0200984 @cpython_only
985 def test_recursion_normalizing_exception(self):
986 # Issue #22898.
987 # Test that a RecursionError is raised when tstate->recursion_depth is
988 # equal to recursion_limit in PyErr_NormalizeException() and check
989 # that a ResourceWarning is printed.
990 # Prior to #22898, the recursivity of PyErr_NormalizeException() was
luzpaza5293b42017-11-05 07:37:50 -0600991 # controlled by tstate->recursion_depth and a PyExc_RecursionErrorInst
xdegaye56d1f5c2017-10-26 15:09:06 +0200992 # singleton was being used in that case, that held traceback data and
993 # locals indefinitely and would cause a segfault in _PyExc_Fini() upon
994 # finalization of these locals.
995 code = """if 1:
996 import sys
Victor Stinner3f2f4fe2020-03-13 13:07:31 +0100997 from _testinternalcapi import get_recursion_depth
xdegaye56d1f5c2017-10-26 15:09:06 +0200998
999 class MyException(Exception): pass
1000
1001 def setrecursionlimit(depth):
1002 while 1:
1003 try:
1004 sys.setrecursionlimit(depth)
1005 return depth
1006 except RecursionError:
1007 # sys.setrecursionlimit() raises a RecursionError if
1008 # the new recursion limit is too low (issue #25274).
1009 depth += 1
1010
1011 def recurse(cnt):
1012 cnt -= 1
1013 if cnt:
1014 recurse(cnt)
1015 else:
1016 generator.throw(MyException)
1017
1018 def gen():
1019 f = open(%a, mode='rb', buffering=0)
1020 yield
1021
1022 generator = gen()
1023 next(generator)
1024 recursionlimit = sys.getrecursionlimit()
1025 depth = get_recursion_depth()
1026 try:
1027 # Upon the last recursive invocation of recurse(),
1028 # tstate->recursion_depth is equal to (recursion_limit - 1)
1029 # and is equal to recursion_limit when _gen_throw() calls
1030 # PyErr_NormalizeException().
1031 recurse(setrecursionlimit(depth + 2) - depth - 1)
1032 finally:
1033 sys.setrecursionlimit(recursionlimit)
1034 print('Done.')
1035 """ % __file__
1036 rc, out, err = script_helper.assert_python_failure("-Wd", "-c", code)
1037 # Check that the program does not fail with SIGABRT.
1038 self.assertEqual(rc, 1)
1039 self.assertIn(b'RecursionError', err)
1040 self.assertIn(b'ResourceWarning', err)
1041 self.assertIn(b'Done.', out)
1042
1043 @cpython_only
1044 def test_recursion_normalizing_infinite_exception(self):
1045 # Issue #30697. Test that a RecursionError is raised when
1046 # PyErr_NormalizeException() maximum recursion depth has been
1047 # exceeded.
1048 code = """if 1:
1049 import _testcapi
1050 try:
1051 raise _testcapi.RecursingInfinitelyError
1052 finally:
1053 print('Done.')
1054 """
1055 rc, out, err = script_helper.assert_python_failure("-c", code)
1056 self.assertEqual(rc, 1)
1057 self.assertIn(b'RecursionError: maximum recursion depth exceeded '
1058 b'while normalizing an exception', err)
1059 self.assertIn(b'Done.', out)
1060
1061 @cpython_only
1062 def test_recursion_normalizing_with_no_memory(self):
1063 # Issue #30697. Test that in the abort that occurs when there is no
1064 # memory left and the size of the Python frames stack is greater than
1065 # the size of the list of preallocated MemoryError instances, the
1066 # Fatal Python error message mentions MemoryError.
1067 code = """if 1:
1068 import _testcapi
1069 class C(): pass
1070 def recurse(cnt):
1071 cnt -= 1
1072 if cnt:
1073 recurse(cnt)
1074 else:
1075 _testcapi.set_nomemory(0)
1076 C()
1077 recurse(16)
1078 """
1079 with SuppressCrashReport():
1080 rc, out, err = script_helper.assert_python_failure("-c", code)
Victor Stinner9e5d30c2020-03-07 00:54:20 +01001081 self.assertIn(b'Fatal Python error: _PyErr_NormalizeException: '
1082 b'Cannot recover from MemoryErrors while '
1083 b'normalizing exceptions.', err)
Amaury Forgeot d'Arce19cadb2008-07-31 22:56:02 +00001084
Serhiy Storchaka5cfc79d2014-02-07 10:06:39 +02001085 @cpython_only
Amaury Forgeot d'Arce19cadb2008-07-31 22:56:02 +00001086 def test_MemoryError(self):
1087 # PyErr_NoMemory always raises the same exception instance.
1088 # Check that the traceback is not doubled.
1089 import traceback
Benjamin Peterson0067bd62008-08-16 16:11:03 +00001090 from _testcapi import raise_memoryerror
Amaury Forgeot d'Arce19cadb2008-07-31 22:56:02 +00001091 def raiseMemError():
1092 try:
Benjamin Peterson0067bd62008-08-16 16:11:03 +00001093 raise_memoryerror()
Amaury Forgeot d'Arce19cadb2008-07-31 22:56:02 +00001094 except MemoryError as e:
1095 tb = e.__traceback__
1096 else:
1097 self.fail("Should have raises a MemoryError")
1098 return traceback.format_tb(tb)
1099
1100 tb1 = raiseMemError()
1101 tb2 = raiseMemError()
1102 self.assertEqual(tb1, tb2)
1103
Benjamin Peterson17e0bbc2010-06-28 15:39:55 +00001104 @cpython_only
Georg Brandl1e28a272009-12-28 08:41:01 +00001105 def test_exception_with_doc(self):
1106 import _testcapi
1107 doc2 = "This is a test docstring."
1108 doc4 = "This is another test docstring."
1109
1110 self.assertRaises(SystemError, _testcapi.make_exception_with_doc,
1111 "error1")
1112
1113 # test basic usage of PyErr_NewException
1114 error1 = _testcapi.make_exception_with_doc("_testcapi.error1")
1115 self.assertIs(type(error1), type)
1116 self.assertTrue(issubclass(error1, Exception))
1117 self.assertIsNone(error1.__doc__)
1118
1119 # test with given docstring
1120 error2 = _testcapi.make_exception_with_doc("_testcapi.error2", doc2)
1121 self.assertEqual(error2.__doc__, doc2)
1122
1123 # test with explicit base (without docstring)
1124 error3 = _testcapi.make_exception_with_doc("_testcapi.error3",
1125 base=error2)
1126 self.assertTrue(issubclass(error3, error2))
1127
1128 # test with explicit base tuple
1129 class C(object):
1130 pass
1131 error4 = _testcapi.make_exception_with_doc("_testcapi.error4", doc4,
1132 (error3, C))
1133 self.assertTrue(issubclass(error4, error3))
1134 self.assertTrue(issubclass(error4, C))
1135 self.assertEqual(error4.__doc__, doc4)
1136
1137 # test with explicit dictionary
1138 error5 = _testcapi.make_exception_with_doc("_testcapi.error5", "",
1139 error4, {'a': 1})
1140 self.assertTrue(issubclass(error5, error4))
1141 self.assertEqual(error5.a, 1)
1142 self.assertEqual(error5.__doc__, "")
1143
Serhiy Storchaka5cfc79d2014-02-07 10:06:39 +02001144 @cpython_only
Antoine Pitrou07e20ef2010-10-28 22:56:58 +00001145 def test_memory_error_cleanup(self):
1146 # Issue #5437: preallocated MemoryError instances should not keep
1147 # traceback objects alive.
1148 from _testcapi import raise_memoryerror
1149 class C:
1150 pass
1151 wr = None
1152 def inner():
1153 nonlocal wr
1154 c = C()
1155 wr = weakref.ref(c)
1156 raise_memoryerror()
1157 # We cannot use assertRaises since it manually deletes the traceback
1158 try:
1159 inner()
1160 except MemoryError as e:
1161 self.assertNotEqual(wr(), None)
1162 else:
1163 self.fail("MemoryError not raised")
1164 self.assertEqual(wr(), None)
1165
Brett Cannon31f59292011-02-21 19:29:56 +00001166 @no_tracing
Antoine Pitrou07e20ef2010-10-28 22:56:58 +00001167 def test_recursion_error_cleanup(self):
1168 # Same test as above, but with "recursion exceeded" errors
1169 class C:
1170 pass
1171 wr = None
1172 def inner():
1173 nonlocal wr
1174 c = C()
1175 wr = weakref.ref(c)
1176 inner()
1177 # We cannot use assertRaises since it manually deletes the traceback
1178 try:
1179 inner()
Yury Selivanovf488fb42015-07-03 01:04:23 -04001180 except RecursionError as e:
Antoine Pitrou07e20ef2010-10-28 22:56:58 +00001181 self.assertNotEqual(wr(), None)
1182 else:
Yury Selivanovf488fb42015-07-03 01:04:23 -04001183 self.fail("RecursionError not raised")
Antoine Pitrou07e20ef2010-10-28 22:56:58 +00001184 self.assertEqual(wr(), None)
Georg Brandl1e28a272009-12-28 08:41:01 +00001185
Antoine Pitroua7622852011-09-01 21:37:43 +02001186 def test_errno_ENOTDIR(self):
1187 # Issue #12802: "not a directory" errors are ENOTDIR even on Windows
1188 with self.assertRaises(OSError) as cm:
1189 os.listdir(__file__)
1190 self.assertEqual(cm.exception.errno, errno.ENOTDIR, cm.exception)
1191
Martin Panter3263f682016-02-28 03:16:11 +00001192 def test_unraisable(self):
1193 # Issue #22836: PyErr_WriteUnraisable() should give sensible reports
1194 class BrokenDel:
1195 def __del__(self):
1196 exc = ValueError("del is broken")
1197 # The following line is included in the traceback report:
1198 raise exc
1199
Victor Stinnere4d300e2019-05-22 23:44:02 +02001200 obj = BrokenDel()
1201 with support.catch_unraisable_exception() as cm:
1202 del obj
Martin Panter3263f682016-02-28 03:16:11 +00001203
Victor Stinnere4d300e2019-05-22 23:44:02 +02001204 self.assertEqual(cm.unraisable.object, BrokenDel.__del__)
1205 self.assertIsNotNone(cm.unraisable.exc_traceback)
Martin Panter3263f682016-02-28 03:16:11 +00001206
1207 def test_unhandled(self):
1208 # Check for sensible reporting of unhandled exceptions
1209 for exc_type in (ValueError, BrokenStrException):
1210 with self.subTest(exc_type):
1211 try:
1212 exc = exc_type("test message")
1213 # The following line is included in the traceback report:
1214 raise exc
1215 except exc_type:
1216 with captured_stderr() as stderr:
1217 sys.__excepthook__(*sys.exc_info())
1218 report = stderr.getvalue()
1219 self.assertIn("test_exceptions.py", report)
1220 self.assertIn("raise exc", report)
1221 self.assertIn(exc_type.__name__, report)
1222 if exc_type is BrokenStrException:
1223 self.assertIn("<exception str() failed>", report)
1224 else:
1225 self.assertIn("test message", report)
1226 self.assertTrue(report.endswith("\n"))
1227
xdegaye66caacf2017-10-23 18:08:41 +02001228 @cpython_only
1229 def test_memory_error_in_PyErr_PrintEx(self):
1230 code = """if 1:
1231 import _testcapi
1232 class C(): pass
1233 _testcapi.set_nomemory(0, %d)
1234 C()
1235 """
1236
1237 # Issue #30817: Abort in PyErr_PrintEx() when no memory.
1238 # Span a large range of tests as the CPython code always evolves with
1239 # changes that add or remove memory allocations.
1240 for i in range(1, 20):
1241 rc, out, err = script_helper.assert_python_failure("-c", code % i)
1242 self.assertIn(rc, (1, 120))
1243 self.assertIn(b'MemoryError', err)
1244
Mark Shannonae3087c2017-10-22 22:41:51 +01001245 def test_yield_in_nested_try_excepts(self):
1246 #Issue #25612
1247 class MainError(Exception):
1248 pass
1249
1250 class SubError(Exception):
1251 pass
1252
1253 def main():
1254 try:
1255 raise MainError()
1256 except MainError:
1257 try:
1258 yield
1259 except SubError:
1260 pass
1261 raise
1262
1263 coro = main()
1264 coro.send(None)
1265 with self.assertRaises(MainError):
1266 coro.throw(SubError())
1267
1268 def test_generator_doesnt_retain_old_exc2(self):
1269 #Issue 28884#msg282532
1270 def g():
1271 try:
1272 raise ValueError
1273 except ValueError:
1274 yield 1
1275 self.assertEqual(sys.exc_info(), (None, None, None))
1276 yield 2
1277
1278 gen = g()
1279
1280 try:
1281 raise IndexError
1282 except IndexError:
1283 self.assertEqual(next(gen), 1)
1284 self.assertEqual(next(gen), 2)
1285
1286 def test_raise_in_generator(self):
1287 #Issue 25612#msg304117
1288 def g():
1289 yield 1
1290 raise
1291 yield 2
1292
1293 with self.assertRaises(ZeroDivisionError):
1294 i = g()
1295 try:
1296 1/0
1297 except:
1298 next(i)
1299 next(i)
1300
Zackery Spytzce6a0702019-08-25 03:44:09 -06001301 @unittest.skipUnless(__debug__, "Won't work if __debug__ is False")
1302 def test_assert_shadowing(self):
1303 # Shadowing AssertionError would cause the assert statement to
1304 # misbehave.
1305 global AssertionError
1306 AssertionError = TypeError
1307 try:
1308 assert False, 'hello'
1309 except BaseException as e:
1310 del AssertionError
1311 self.assertIsInstance(e, AssertionError)
1312 self.assertEqual(str(e), 'hello')
1313 else:
1314 del AssertionError
1315 self.fail('Expected exception')
1316
Antoine Pitroua7622852011-09-01 21:37:43 +02001317
Brett Cannon79ec55e2012-04-12 20:24:54 -04001318class ImportErrorTests(unittest.TestCase):
1319
1320 def test_attributes(self):
1321 # Setting 'name' and 'path' should not be a problem.
1322 exc = ImportError('test')
1323 self.assertIsNone(exc.name)
1324 self.assertIsNone(exc.path)
1325
1326 exc = ImportError('test', name='somemodule')
1327 self.assertEqual(exc.name, 'somemodule')
1328 self.assertIsNone(exc.path)
1329
1330 exc = ImportError('test', path='somepath')
1331 self.assertEqual(exc.path, 'somepath')
1332 self.assertIsNone(exc.name)
1333
1334 exc = ImportError('test', path='somepath', name='somename')
1335 self.assertEqual(exc.name, 'somename')
1336 self.assertEqual(exc.path, 'somepath')
1337
Michael Seifert64c8f702017-04-09 09:47:12 +02001338 msg = "'invalid' is an invalid keyword argument for ImportError"
Serhiy Storchaka47dee112016-09-27 20:45:35 +03001339 with self.assertRaisesRegex(TypeError, msg):
1340 ImportError('test', invalid='keyword')
1341
1342 with self.assertRaisesRegex(TypeError, msg):
1343 ImportError('test', name='name', invalid='keyword')
1344
1345 with self.assertRaisesRegex(TypeError, msg):
1346 ImportError('test', path='path', invalid='keyword')
1347
1348 with self.assertRaisesRegex(TypeError, msg):
1349 ImportError(invalid='keyword')
1350
Serhiy Storchaka47dee112016-09-27 20:45:35 +03001351 with self.assertRaisesRegex(TypeError, msg):
1352 ImportError('test', invalid='keyword', another=True)
1353
Serhiy Storchakae9e44482016-09-28 07:53:32 +03001354 def test_reset_attributes(self):
1355 exc = ImportError('test', name='name', path='path')
1356 self.assertEqual(exc.args, ('test',))
1357 self.assertEqual(exc.msg, 'test')
1358 self.assertEqual(exc.name, 'name')
1359 self.assertEqual(exc.path, 'path')
1360
1361 # Reset not specified attributes
1362 exc.__init__()
1363 self.assertEqual(exc.args, ())
1364 self.assertEqual(exc.msg, None)
1365 self.assertEqual(exc.name, None)
1366 self.assertEqual(exc.path, None)
1367
Brett Cannon07c6e712012-08-24 13:05:09 -04001368 def test_non_str_argument(self):
1369 # Issue #15778
Nadeem Vawda6d708702012-10-14 01:42:32 +02001370 with check_warnings(('', BytesWarning), quiet=True):
1371 arg = b'abc'
1372 exc = ImportError(arg)
1373 self.assertEqual(str(arg), str(exc))
Brett Cannon79ec55e2012-04-12 20:24:54 -04001374
Serhiy Storchakab7853962017-04-08 09:55:07 +03001375 def test_copy_pickle(self):
1376 for kwargs in (dict(),
1377 dict(name='somename'),
1378 dict(path='somepath'),
1379 dict(name='somename', path='somepath')):
1380 orig = ImportError('test', **kwargs)
1381 for proto in range(pickle.HIGHEST_PROTOCOL + 1):
1382 exc = pickle.loads(pickle.dumps(orig, proto))
1383 self.assertEqual(exc.args, ('test',))
1384 self.assertEqual(exc.msg, 'test')
1385 self.assertEqual(exc.name, orig.name)
1386 self.assertEqual(exc.path, orig.path)
1387 for c in copy.copy, copy.deepcopy:
1388 exc = c(orig)
1389 self.assertEqual(exc.args, ('test',))
1390 self.assertEqual(exc.msg, 'test')
1391 self.assertEqual(exc.name, orig.name)
1392 self.assertEqual(exc.path, orig.path)
1393
Brett Cannon79ec55e2012-04-12 20:24:54 -04001394
Thomas Wouters4d70c3d2006-06-08 14:42:34 +00001395if __name__ == '__main__':
Guido van Rossumb8142c32007-05-08 17:49:10 +00001396 unittest.main()