PageRenderTime 29ms CodeModel.GetById 39ms RepoModel.GetById 0ms app.codeStats 0ms

/teca-env18/lib/python2.7/site-packages/traceback2/tests/test_traceback.py

https://github.com/dpritsos/WEGA
Python | 899 lines | 889 code | 6 blank | 4 comment | 12 complexity | 8fbfe8c39bad20c958c2f64419dc1949 MD5 | raw file
  1. """Test cases for traceback module"""
  2. from collections import namedtuple
  3. import doctest
  4. import io
  5. from io import StringIO
  6. import platform
  7. import sys
  8. import re
  9. import contextlib2 as contextlib
  10. import fixtures
  11. import linecache2 as linecache
  12. import six
  13. from six import b, text_type, u
  14. try:
  15. from six import raise_from
  16. except ImportError:
  17. # support raise_from on 3.x:
  18. # submitted to six: https://bitbucket.org/gutworth/six/issue/102/raise-foo-from-bar-is-a-syntax-error-on-27
  19. if sys.version_info[:2] > (3, 1):
  20. six.exec_("""def raise_from(value, from_value):
  21. raise value from from_value
  22. """)
  23. else:
  24. def raise_from(value, from_value):
  25. raise value
  26. import unittest2 as unittest
  27. import testtools
  28. from testtools.matchers import DocTestMatches, Equals, MatchesAny
  29. import traceback2 as traceback
  30. @contextlib.contextmanager
  31. def captured_output(streamname):
  32. stream = StringIO()
  33. patch = fixtures.MonkeyPatch('sys.%s' % streamname, stream)
  34. with patch:
  35. yield stream
  36. FNAME = __file__
  37. if FNAME.endswith('.pyc'):
  38. FNAME = FNAME[:-1]
  39. class FakeLoader:
  40. def __init__(self, lines):
  41. self._lines = lines
  42. def get_source(self, name):
  43. return self._lines
  44. fake_module = dict(
  45. __name__='fred',
  46. __loader__=FakeLoader(''.join(linecache.getlines(FNAME)))
  47. )
  48. test_code = namedtuple('code', ['co_filename', 'co_name'])
  49. test_frame = namedtuple('frame', ['f_code', 'f_globals', 'f_locals'])
  50. test_tb = namedtuple('tb', ['tb_frame', 'tb_lineno', 'tb_next'])
  51. class SyntaxTracebackCases(testtools.TestCase):
  52. # For now, a very minimal set of tests. I want to be sure that
  53. # formatting of SyntaxErrors works based on changes for 2.1.
  54. def get_exception_format(self, func, exc):
  55. try:
  56. func()
  57. except exc as value:
  58. return traceback.format_exception_only(exc, value)
  59. else:
  60. raise ValueError("call did not raise exception")
  61. def syntax_error_with_caret(self):
  62. compile("def fact(x):\n\treturn x!\n", "?", "exec")
  63. def syntax_error_with_caret_2(self):
  64. compile("1 +\n", "?", "exec")
  65. def syntax_error_bad_indentation(self):
  66. compile("def spam():\n print(1)\n print(2)", "?", "exec")
  67. def syntax_error_with_caret_non_ascii(self):
  68. compile('Python = "\u1e54\xfd\u0163\u0125\xf2\xf1" +', "?", "exec")
  69. def syntax_error_bad_indentation2(self):
  70. compile(" print(2)", "?", "exec")
  71. def test_caret(self):
  72. err = self.get_exception_format(self.syntax_error_with_caret,
  73. SyntaxError)
  74. self.assertEqual(len(err), 4)
  75. self.assertTrue(err[1].strip() == "return x!")
  76. self.assertIn("^", err[2]) # third line has caret
  77. self.assertEqual(err[1].find("!"), err[2].find("^")) # in the right place
  78. err = self.get_exception_format(self.syntax_error_with_caret_2,
  79. SyntaxError)
  80. self.assertIn("^", err[2]) # third line has caret
  81. self.assertEqual(err[2].count('\n'), 1) # and no additional newline
  82. self.assertEqual(err[1].find("+"), err[2].find("^")) # in the right place
  83. err = self.get_exception_format(self.syntax_error_with_caret_non_ascii,
  84. SyntaxError)
  85. self.assertIn("^", err[2]) # third line has caret
  86. self.assertEqual(err[2].count('\n'), 1) # and no additional newline
  87. self.assertEqual(err[1].find("+"), err[2].find("^")) # in the right place
  88. def test_nocaret(self):
  89. exc = SyntaxError("error", ("x.py", 23, None, "bad syntax"))
  90. err = traceback.format_exception_only(SyntaxError, exc)
  91. self.assertEqual(len(err), 3)
  92. self.assertEqual(err[1].strip(), "bad syntax")
  93. def test_bad_indentation(self):
  94. err = self.get_exception_format(self.syntax_error_bad_indentation,
  95. IndentationError)
  96. self.assertEqual(len(err), 4)
  97. self.assertEqual(err[1].strip(), "print(2)")
  98. self.assertIn("^", err[2])
  99. self.assertEqual(err[1].find(")"), err[2].find("^"))
  100. err = self.get_exception_format(self.syntax_error_bad_indentation2,
  101. IndentationError)
  102. self.assertEqual(len(err), 4)
  103. self.assertEqual(err[1].strip(), "print(2)")
  104. self.assertIn("^", err[2])
  105. # pypy has a different offset for its errors.
  106. pos_cpython = err[1].find("p")
  107. pos_pypy = err[1].find(")")
  108. self.assertThat(
  109. err[2].find("^"),
  110. MatchesAny(Equals(pos_cpython), Equals(pos_pypy)))
  111. def test_base_exception(self):
  112. # Test that exceptions derived from BaseException are formatted right
  113. e = KeyboardInterrupt()
  114. lst = traceback.format_exception_only(e.__class__, e)
  115. self.assertThat(lst,
  116. MatchesAny(Equals(['KeyboardInterrupt\n']),
  117. Equals(['exceptions.KeyboardInterrupt\n'])))
  118. def test_format_exception_only_bad__str__(self):
  119. def qualname(X):
  120. return getattr(X, '__qualname__', X.__name__)
  121. class X(Exception):
  122. def __str__(self):
  123. 1/0
  124. err = traceback.format_exception_only(X, X())
  125. self.assertEqual(len(err), 1)
  126. str_value = '<unprintable %s object>' % X.__name__
  127. if X.__module__ in ('__main__', 'builtins'):
  128. str_name = qualname(X)
  129. else:
  130. str_name = '.'.join([X.__module__, qualname(X)])
  131. self.assertEqual(err[0], "%s: %s\n" % (str_name, str_value))
  132. def test_format_exception_only_undecodable__str__(self):
  133. # This won't decode via the ascii codec.
  134. X = Exception(u('\u5341').encode('shift-jis'))
  135. err = traceback.format_exception_only(type(X), X)
  136. self.assertEqual(len(err), 1)
  137. str_value = "b'\\x8f\\\\'"
  138. self.assertEqual(err[0], "Exception: %s\n" % str_value)
  139. def test_without_exception(self):
  140. err = traceback.format_exception_only(None, None)
  141. self.assertEqual(err, ['None\n'])
  142. def test_encoded_file(self):
  143. # Test that tracebacks are correctly printed for encoded source files:
  144. # - correct line number (Issue2384)
  145. # - respect file encoding (Issue3975)
  146. import tempfile, sys, subprocess, os
  147. # The spawned subprocess has its stdout redirected to a PIPE, and its
  148. # encoding may be different from the current interpreter, on Windows
  149. # at least.
  150. process = subprocess.Popen([sys.executable, "-c",
  151. "import sys; print(sys.stdout.encoding)"],
  152. stdout=subprocess.PIPE,
  153. stderr=subprocess.STDOUT)
  154. stdout, stderr = process.communicate()
  155. output_encoding = text_type(stdout, 'ascii').splitlines()[0]
  156. def do_test(firstlines, message, charset, lineno, output_encoding):
  157. # Raise the message in a subprocess, and catch the output
  158. with fixtures.TempDir() as d:
  159. TESTFN = d.path + '/fname'
  160. output = io.open(TESTFN, "w", encoding=charset)
  161. output.write(u("""{0}if 1:
  162. import traceback;
  163. raise RuntimeError('{1}')
  164. """).format(firstlines, message))
  165. output.close()
  166. process = subprocess.Popen([sys.executable, TESTFN],
  167. stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
  168. stdout, stderr = process.communicate()
  169. if output_encoding == 'None':
  170. output_encoding = charset
  171. stdout = stdout.decode(output_encoding).splitlines()
  172. # The source lines are encoded with the 'backslashreplace' handler
  173. encoded_message = message.encode(output_encoding,
  174. 'backslashreplace')
  175. # and we just decoded them with the output_encoding.
  176. message_ascii = encoded_message.decode(output_encoding)
  177. err_line = u("raise RuntimeError('{0}')").format(message_ascii)
  178. err_msg = u("RuntimeError: {0}").format(message_ascii)
  179. if platform.python_implementation() == 'PyPy':
  180. # PyPy includes its own top level app_main.py in the traceback.
  181. del stdout[1]
  182. self.assertIn(("line %s" % lineno), stdout[1],
  183. "Invalid line number: {0!r} instead of {1}".format(
  184. stdout[1], lineno))
  185. self.assertTrue(stdout[2].endswith(err_line),
  186. "Invalid traceback line: {0!r} instead of {1!r}".format(
  187. stdout[2], err_line))
  188. self.assertTrue(stdout[3] == err_msg,
  189. "Invalid error message: {0!r} instead of {1!r}".format(
  190. stdout[3], err_msg))
  191. do_test("", "foo", "ascii", 3, output_encoding)
  192. for charset in ("ascii", "iso-8859-1", "utf-8", "GBK"):
  193. if charset == "ascii":
  194. text = u("foo")
  195. elif charset == "GBK":
  196. text = u("\u4E02\u5100")
  197. else:
  198. text = u("h\xe9 ho")
  199. do_test("# coding: {0}\n".format(charset),
  200. text, charset, 4, output_encoding)
  201. do_test("#!shebang\n# coding: {0}\n".format(charset),
  202. text, charset, 5, output_encoding)
  203. do_test(" \t\f\n# coding: {0}\n".format(charset),
  204. text, charset, 5, output_encoding)
  205. # Issue #18960: coding spec should has no effect
  206. # (Fixed in 3.4)
  207. if sys.version_info[:2] > (3, 3):
  208. do_test(
  209. "0\n# coding: GBK\n", u("h\xe9 ho"), 'utf-8', 5,
  210. output_encoding)
  211. class TracebackFormatTests(unittest.TestCase):
  212. def some_exception(self):
  213. raise KeyError('blah')
  214. def check_traceback_format(self, cleanup_func=None):
  215. try:
  216. if issubclass(six.binary_type, six.string_types):
  217. # Python 2.6 or other platform where the interpreter
  218. # is likely going to be spitting out bytes, which will
  219. # then fail with io.StringIO(), so we skip the cross-
  220. # checks with the C API there. Note that _testcapi
  221. # is included in (at least) Ubuntu CPython packages, which
  222. # makes the import check less effective than desired.
  223. raise ImportError
  224. from _testcapi import traceback_print
  225. except ImportError:
  226. traceback_print = None
  227. try:
  228. self.some_exception()
  229. except KeyError:
  230. type_, value, tb = sys.exc_info()
  231. if cleanup_func is not None:
  232. # Clear the inner frames, not this one
  233. cleanup_func(tb.tb_next)
  234. traceback_fmt = u('Traceback (most recent call last):\n') + \
  235. u('').join(traceback.format_tb(tb))
  236. if traceback_print is not None:
  237. file_ = StringIO()
  238. traceback_print(tb, file_)
  239. python_fmt = file_.getvalue()
  240. # Call all _tb and _exc functions
  241. with captured_output("stderr") as tbstderr:
  242. traceback.print_tb(tb)
  243. tbfile = StringIO()
  244. traceback.print_tb(tb, file=tbfile)
  245. with captured_output("stderr") as excstderr:
  246. traceback.print_exc()
  247. excfmt = traceback.format_exc()
  248. excfile = StringIO()
  249. traceback.print_exc(file=excfile)
  250. else:
  251. self.fail("unable to create test traceback string")
  252. # Make sure that Python and the traceback module format the same thing
  253. if traceback_print is not None:
  254. self.assertEqual(traceback_fmt, python_fmt)
  255. # Now verify the _tb func output
  256. self.assertEqual(tbstderr.getvalue(), tbfile.getvalue())
  257. # Now verify the _exc func output
  258. self.assertEqual(excstderr.getvalue(), excfile.getvalue())
  259. self.assertEqual(excfmt, excfile.getvalue())
  260. # Make sure that the traceback is properly indented.
  261. tb_lines = traceback_fmt.splitlines()
  262. self.assertEqual(len(tb_lines), 5)
  263. banner = tb_lines[0]
  264. location, source_line = tb_lines[-2:]
  265. self.assertTrue(banner.startswith('Traceback'))
  266. self.assertTrue(location.startswith(' File'))
  267. self.assertTrue(source_line.startswith(' raise'))
  268. def test_traceback_format(self):
  269. self.check_traceback_format()
  270. def test_traceback_format_with_cleared_frames(self):
  271. # Check that traceback formatting also works with a clear()ed frame
  272. def cleanup_tb(tb):
  273. if getattr(tb.tb_frame, 'clear_frames', None):
  274. tb.tb_frame.clear()
  275. self.check_traceback_format(cleanup_tb)
  276. def test_stack_format(self):
  277. # Verify _stack functions. Note we have to use _getframe(1) to
  278. # compare them without this frame appearing in the output
  279. with captured_output("stderr") as ststderr:
  280. traceback.print_stack(sys._getframe(1))
  281. stfile = StringIO()
  282. traceback.print_stack(sys._getframe(1), file=stfile)
  283. self.assertEqual(ststderr.getvalue(), stfile.getvalue())
  284. stfmt = traceback.format_stack(sys._getframe(1))
  285. self.assertEqual(ststderr.getvalue(), "".join(stfmt))
  286. cause_message = (
  287. "\nThe above exception was the direct cause "
  288. "of the following exception:\n\n")
  289. context_message = (
  290. "\nDuring handling of the above exception, "
  291. "another exception occurred:\n\n")
  292. boundaries = re.compile(
  293. '(%s|%s)' % (re.escape(cause_message), re.escape(context_message)))
  294. class BaseExceptionReportingTests:
  295. def get_exception(self, exception_or_callable, tb=None):
  296. if isinstance(exception_or_callable, Exception):
  297. return exception_or_callable, tb
  298. try:
  299. exception_or_callable()
  300. except Exception as e:
  301. return e, sys.exc_info()[2]
  302. def zero_div(self):
  303. 1/0 # In zero_div
  304. def check_zero_div(self, msg):
  305. lines = msg.splitlines()
  306. self.assertTrue(lines[-3].startswith(' File'))
  307. self.assertIn('1/0 # In zero_div', lines[-2])
  308. self.assertTrue(lines[-1].startswith('ZeroDivisionError'), lines[-1])
  309. def test_simple(self):
  310. try:
  311. 1/0 # Marker
  312. except ZeroDivisionError as _:
  313. e = _
  314. tb = sys.exc_info()[2]
  315. lines = self.get_report(e, tb=tb).splitlines()
  316. self.assertEqual(len(lines), 4)
  317. self.assertTrue(lines[0].startswith('Traceback'))
  318. self.assertTrue(lines[1].startswith(' File'))
  319. self.assertIn('1/0 # Marker', lines[2])
  320. # < 3 show as exceptions.ZeroDivisionError.
  321. self.assertIn('ZeroDivisionError', lines[3])
  322. @unittest.skipIf(sys.version_info[:2] < (3, 2), "Only applies to 3.2+")
  323. def test_cause(self):
  324. def inner_raise():
  325. try:
  326. self.zero_div()
  327. except ZeroDivisionError as e:
  328. raise_from(KeyError, e)
  329. def outer_raise():
  330. inner_raise() # Marker
  331. blocks = boundaries.split(self.get_report(outer_raise))
  332. self.assertEqual(len(blocks), 3)
  333. self.assertEqual(blocks[1], cause_message)
  334. self.check_zero_div(blocks[0])
  335. self.assertIn('inner_raise() # Marker', blocks[2])
  336. @unittest.skipIf(sys.version_info[:2] < (3, 2), "Only applies to 3.2+")
  337. def test_context(self):
  338. def inner_raise():
  339. try:
  340. self.zero_div()
  341. except ZeroDivisionError:
  342. raise KeyError
  343. def outer_raise():
  344. inner_raise() # Marker
  345. blocks = boundaries.split(self.get_report(outer_raise))
  346. self.assertEqual(len(blocks), 3)
  347. self.assertEqual(blocks[1], context_message)
  348. self.check_zero_div(blocks[0])
  349. self.assertIn('inner_raise() # Marker', blocks[2])
  350. @unittest.skipIf(sys.version_info[:2] < (3, 3), "Only applies to 3.3+")
  351. def test_context_suppression(self):
  352. try:
  353. try:
  354. raise Exception
  355. except:
  356. raise_from(ZeroDivisionError, None)
  357. except ZeroDivisionError as _:
  358. e = _
  359. tb = sys.exc_info()[2]
  360. lines = self.get_report(e, tb)
  361. self.assertThat(lines, DocTestMatches("""\
  362. Traceback (most recent call last):
  363. File "...traceback2/tests/test_traceback.py", line ..., in test_context_suppression
  364. raise_from(ZeroDivisionError, None)
  365. File "<string>", line 2, in raise_from
  366. ZeroDivisionError
  367. """, doctest.ELLIPSIS))
  368. @unittest.skipIf(sys.version_info[:2] < (3, 2), "Only applies to 3.2+")
  369. def test_cause_and_context(self):
  370. # When both a cause and a context are set, only the cause should be
  371. # displayed and the context should be muted.
  372. def inner_raise():
  373. try:
  374. self.zero_div()
  375. except ZeroDivisionError as _e:
  376. e = _e
  377. try:
  378. xyzzy
  379. except NameError:
  380. raise_from(KeyError, e)
  381. def outer_raise():
  382. inner_raise() # Marker
  383. blocks = boundaries.split(self.get_report(outer_raise))
  384. self.assertEqual(len(blocks), 3)
  385. self.assertEqual(blocks[1], cause_message)
  386. self.check_zero_div(blocks[0])
  387. self.assertIn('inner_raise() # Marker', blocks[2])
  388. @unittest.skipIf(sys.version_info[:2] < (3, 2), "Only applies to 3.2+")
  389. def test_cause_recursive(self):
  390. def inner_raise():
  391. try:
  392. try:
  393. self.zero_div()
  394. except ZeroDivisionError as e:
  395. z = e
  396. raise_from(KeyError, e)
  397. except KeyError as e:
  398. raise_from(z, e)
  399. def outer_raise():
  400. inner_raise() # Marker
  401. blocks = boundaries.split(self.get_report(outer_raise))
  402. self.assertEqual(len(blocks), 3)
  403. self.assertEqual(blocks[1], cause_message)
  404. # The first block is the KeyError raised from the ZeroDivisionError
  405. self.assertIn('raise_from(KeyError, e)', blocks[0])
  406. self.assertNotIn('1/0', blocks[0])
  407. # The second block (apart from the boundary) is the ZeroDivisionError
  408. # re-raised from the KeyError
  409. self.assertIn('inner_raise() # Marker', blocks[2])
  410. self.check_zero_div(blocks[2])
  411. def test_syntax_error_offset_at_eol(self):
  412. # See #10186.
  413. def e():
  414. raise SyntaxError('', ('', 0, 5, u('hello')))
  415. msg = self.get_report(e).splitlines()
  416. self.assertEqual(msg[-2], " ^")
  417. def e():
  418. exec("x = 5 | 4 |")
  419. msg = self.get_report(e).splitlines()
  420. self.assertEqual(msg[-2], ' ^')
  421. class PyExcReportingTests(BaseExceptionReportingTests, testtools.TestCase):
  422. #
  423. # This checks reporting through the 'traceback' module, with both
  424. # format_exception() and print_exception().
  425. #
  426. def get_report(self, e, tb=None):
  427. e, tb = self.get_exception(e, tb)
  428. s = ''.join(
  429. traceback.format_exception(type(e), e, tb))
  430. with captured_output("stderr") as sio:
  431. traceback.print_exception(type(e), e, tb)
  432. self.assertEqual(sio.getvalue(), s)
  433. return s
  434. class MiscTracebackCases(unittest.TestCase):
  435. #
  436. # Check non-printing functions in traceback module
  437. #
  438. def test_clear(self):
  439. def outer():
  440. middle()
  441. def middle():
  442. inner()
  443. def inner():
  444. i = 1
  445. 1/0
  446. try:
  447. outer()
  448. except:
  449. type_, value, tb = sys.exc_info()
  450. # Initial assertion: there's one local in the inner frame.
  451. inner_frame = tb.tb_next.tb_next.tb_next.tb_frame
  452. self.assertEqual(len(inner_frame.f_locals), 1)
  453. # Clear traceback frames
  454. traceback.clear_frames(tb)
  455. # Local variable dict should now be empty (on Python 3.4+)
  456. if sys.version_info[:2] > (3, 3):
  457. self.assertEqual({}, inner_frame.f_locals)
  458. class TestFrame(unittest.TestCase):
  459. def test_basics(self):
  460. linecache.clearcache()
  461. linecache.lazycache("f", fake_module)
  462. f = traceback.FrameSummary("f", 1, "dummy")
  463. self.assertEqual(
  464. ("f", 1, "dummy", '"""Test cases for traceback module"""'),
  465. tuple(f))
  466. self.assertEqual(None, f.locals)
  467. def test_lazy_lines(self):
  468. linecache.clearcache()
  469. f = traceback.FrameSummary("f", 1, "dummy", lookup_line=False)
  470. self.assertEqual(None, f._line)
  471. linecache.lazycache("f", fake_module)
  472. self.assertEqual(
  473. '"""Test cases for traceback module"""',
  474. f.line)
  475. def test_explicit_line(self):
  476. f = traceback.FrameSummary("f", 1, "dummy", line="line")
  477. self.assertEqual("line", f.line)
  478. class TestStack(unittest.TestCase):
  479. def test_walk_stack(self):
  480. s = list(traceback.walk_stack(None))
  481. self.assertGreater(len(s), 10)
  482. def test_walk_tb(self):
  483. try:
  484. 1/0
  485. except Exception:
  486. _, _, tb = sys.exc_info()
  487. s = list(traceback.walk_tb(tb))
  488. self.assertEqual(len(s), 1)
  489. def test_extract_stack(self):
  490. s = traceback.StackSummary.extract(traceback.walk_stack(None))
  491. self.assertIsInstance(s, traceback.StackSummary)
  492. def test_extract_stack_limit(self):
  493. s = traceback.StackSummary.extract(traceback.walk_stack(None), limit=5)
  494. self.assertEqual(len(s), 5)
  495. def test_extract_stack_lookup_lines(self):
  496. linecache.clearcache()
  497. linecache.updatecache('/foo.py', fake_module)
  498. c = test_code('/foo.py', 'method')
  499. f = test_frame(c, None, None)
  500. s = traceback.StackSummary.extract(iter([(f, 8)]), lookup_lines=True)
  501. linecache.clearcache()
  502. self.assertEqual(s[0].line, "import sys")
  503. def test_extract_stackup_deferred_lookup_lines(self):
  504. linecache.clearcache()
  505. c = test_code('/foo.py', 'method')
  506. f = test_frame(c, None, None)
  507. s = traceback.StackSummary.extract(iter([(f, 8)]), lookup_lines=False)
  508. self.assertEqual({}, linecache.cache)
  509. linecache.updatecache('/foo.py', fake_module)
  510. self.assertEqual(s[0].line, "import sys")
  511. def test_from_list(self):
  512. s = traceback.StackSummary.from_list([('foo.py', 1, 'fred', 'line')])
  513. self.assertEqual(
  514. [' File "foo.py", line 1, in fred\n line\n'],
  515. s.format())
  516. def test_format_smoke(self):
  517. # For detailed tests see the format_list tests, which consume the same
  518. # code.
  519. s = traceback.StackSummary.from_list([('foo.py', 1, 'fred', 'line')])
  520. self.assertEqual(
  521. [' File "foo.py", line 1, in fred\n line\n'],
  522. s.format())
  523. @unittest.skipIf(sys.getfilesystemencoding()=='ANSI_X3.4-1968',
  524. 'Requires non-ascii fs encoding')
  525. def test_format_unicode_filename(self):
  526. # Filenames in Python2 may be bytestrings that will fail to implicit
  527. # decode.
  528. fname = u('\u5341').encode(sys.getfilesystemencoding())
  529. s = traceback.StackSummary.from_list([(fname, 1, 'fred', 'line')])
  530. self.assertEqual(
  531. [u(' File "\u5341", line 1, in fred\n line\n')],
  532. s.format())
  533. def test_format_bad_filename(self):
  534. # Filenames in Python2 may be bytestrings that will fail to implicit
  535. # decode.
  536. # This won't decode via the implicit(ascii) codec or the default
  537. # fs encoding (unless the encoding is a wildcard encoding).
  538. fname = b('\x8b')
  539. s = traceback.StackSummary.from_list([(fname, 1, 'fred', 'line')])
  540. self.assertEqual(
  541. [' File "b\'\\x8b\'", line 1, in fred\n line\n'],
  542. s.format())
  543. def test_locals(self):
  544. linecache.updatecache('/foo.py', globals())
  545. c = test_code('/foo.py', 'method')
  546. f = test_frame(c, globals(), {'something': 1})
  547. s = traceback.StackSummary.extract(iter([(f, 6)]), capture_locals=True)
  548. self.assertEqual(s[0].locals, {'something': '1'})
  549. def test_no_locals(self):
  550. linecache.updatecache('/foo.py', globals())
  551. c = test_code('/foo.py', 'method')
  552. f = test_frame(c, globals(), {'something': 1})
  553. s = traceback.StackSummary.extract(iter([(f, 6)]))
  554. self.assertEqual(s[0].locals, None)
  555. def test_format_locals(self):
  556. def some_inner(k, v):
  557. a = 1
  558. b = 2
  559. return traceback.StackSummary.extract(
  560. traceback.walk_stack(None), capture_locals=True, limit=1)
  561. s = some_inner(3, 4)
  562. self.assertEqual(
  563. [' File "' + FNAME + '", line 651, '
  564. 'in some_inner\n'
  565. ' traceback.walk_stack(None), capture_locals=True, limit=1)\n'
  566. ' a = 1\n'
  567. ' b = 2\n'
  568. ' k = 3\n'
  569. ' v = 4\n'
  570. ], s.format())
  571. class TestTracebackException(unittest.TestCase):
  572. def test_smoke(self):
  573. try:
  574. 1/0
  575. except Exception:
  576. exc_info = sys.exc_info()
  577. exc = traceback.TracebackException(*exc_info)
  578. expected_stack = traceback.StackSummary.extract(
  579. traceback.walk_tb(exc_info[2]))
  580. self.assertEqual(None, exc.__cause__)
  581. self.assertEqual(None, exc.__context__)
  582. self.assertEqual(False, exc.__suppress_context__)
  583. self.assertEqual(expected_stack, exc.stack)
  584. self.assertEqual(exc_info[0], exc.exc_type)
  585. self.assertEqual(str(exc_info[1]), str(exc))
  586. @unittest.skipIf(sys.version_info[:2] < (3, 0), "Only applies to 3+")
  587. def test_from_exception(self):
  588. # Check all the parameters are accepted.
  589. def foo():
  590. 1/0
  591. try:
  592. foo()
  593. except Exception as e:
  594. exc_info = sys.exc_info()
  595. self.expected_stack = traceback.StackSummary.extract(
  596. traceback.walk_tb(exc_info[2]), limit=1, lookup_lines=False,
  597. capture_locals=True)
  598. self.exc = traceback.TracebackException.from_exception(
  599. e, limit=1, lookup_lines=False, capture_locals=True)
  600. expected_stack = self.expected_stack
  601. exc = self.exc
  602. self.assertEqual(None, exc.__cause__)
  603. self.assertEqual(None, exc.__context__)
  604. self.assertEqual(False, exc.__suppress_context__)
  605. self.assertEqual(expected_stack, exc.stack)
  606. self.assertEqual(exc_info[0], exc.exc_type)
  607. self.assertEqual(str(exc_info[1]), str(exc))
  608. @unittest.skipIf(sys.version_info[:2] < (3, 2), "Only applies to 3.2+")
  609. def test_cause(self):
  610. try:
  611. try:
  612. 1/0
  613. finally:
  614. exc_info_context = sys.exc_info()
  615. exc_context = traceback.TracebackException(*exc_info_context)
  616. cause = Exception("cause")
  617. raise_from(Exception("uh ok"), cause)
  618. except Exception:
  619. exc_info = sys.exc_info()
  620. exc = traceback.TracebackException(*exc_info)
  621. expected_stack = traceback.StackSummary.extract(
  622. traceback.walk_tb(exc_info[2]))
  623. exc_cause = traceback.TracebackException(Exception, cause, None)
  624. self.assertEqual(exc_cause, exc.__cause__)
  625. self.assertEqual(exc_context, exc.__context__)
  626. if hasattr(exc_info[1], '__suppress_context__'):
  627. self.assertEqual(True, exc.__suppress_context__)
  628. self.assertEqual(expected_stack, exc.stack)
  629. self.assertEqual(exc_info[0], exc.exc_type)
  630. self.assertEqual(str(exc_info[1]), str(exc))
  631. @unittest.skipIf(sys.version_info[:2] < (3, 2), "Only applies to 3.2+")
  632. def test_context(self):
  633. try:
  634. try:
  635. 1/0
  636. finally:
  637. exc_info_context = sys.exc_info()
  638. exc_context = traceback.TracebackException(*exc_info_context)
  639. raise Exception("uh oh")
  640. except Exception:
  641. exc_info = sys.exc_info()
  642. exc = traceback.TracebackException(*exc_info)
  643. expected_stack = traceback.StackSummary.extract(
  644. traceback.walk_tb(exc_info[2]))
  645. self.assertEqual(None, exc.__cause__)
  646. self.assertEqual(exc_context, exc.__context__)
  647. self.assertEqual(False, exc.__suppress_context__)
  648. self.assertEqual(expected_stack, exc.stack)
  649. self.assertEqual(exc_info[0], exc.exc_type)
  650. self.assertEqual(str(exc_info[1]), str(exc))
  651. def test_limit(self):
  652. def recurse(n):
  653. if n:
  654. recurse(n-1)
  655. else:
  656. 1/0
  657. try:
  658. recurse(10)
  659. except Exception:
  660. exc_info = sys.exc_info()
  661. exc = traceback.TracebackException(*exc_info, limit=5)
  662. expected_stack = traceback.StackSummary.extract(
  663. traceback.walk_tb(exc_info[2]), limit=5)
  664. self.assertEqual(expected_stack, exc.stack)
  665. def test_lookup_lines(self):
  666. linecache.clearcache()
  667. e = Exception("uh oh")
  668. c = test_code('/foo.py', 'method')
  669. f = test_frame(c, None, None)
  670. tb = test_tb(f, 8, None)
  671. exc = traceback.TracebackException(Exception, e, tb, lookup_lines=False)
  672. self.assertEqual({}, linecache.cache)
  673. linecache.updatecache('/foo.py', fake_module)
  674. self.assertEqual(exc.stack[0].line, "import sys")
  675. def test_locals(self):
  676. linecache.updatecache('/foo.py', fake_module)
  677. e = Exception("uh oh")
  678. c = test_code('/foo.py', 'method')
  679. f = test_frame(c, globals(), {'something': 1, 'other': 'string'})
  680. tb = test_tb(f, 6, None)
  681. exc = traceback.TracebackException(
  682. Exception, e, tb, capture_locals=True)
  683. self.assertEqual(
  684. exc.stack[0].locals, {'something': '1', 'other': "'string'"})
  685. def test_no_locals(self):
  686. linecache.updatecache('/foo.py', fake_module)
  687. e = Exception("uh oh")
  688. c = test_code('/foo.py', 'method')
  689. f = test_frame(c, fake_module, {'something': 1})
  690. tb = test_tb(f, 6, None)
  691. exc = traceback.TracebackException(Exception, e, tb)
  692. self.assertEqual(exc.stack[0].locals, None)
  693. def test_syntax_no_extras(self):
  694. linecache.updatecache('/foo.py', fake_module)
  695. e = SyntaxError("uh oh")
  696. c = test_code('/foo.py', 'method')
  697. f = test_frame(c, fake_module, {'something': 1})
  698. tb = test_tb(f, 6, None)
  699. exc = traceback.TracebackException(SyntaxError, e, tb)
  700. self.assertEqual([
  701. u('Traceback (most recent call last):\n'),
  702. u(' File "/foo.py", line 6, in method\n from io import StringIO\n'),
  703. u(' File "<string>", line None\n'),
  704. u('SyntaxError: uh oh\n')],
  705. list(exc.format()))
  706. def test_syntax_undecoded_lines(self):
  707. # If the interpreter returns bytestrings, we have to decode ourselves.
  708. lines = u("1\n\u5341\n3\n")
  709. fake_module = dict(
  710. __name__='fred',
  711. __loader__=FakeLoader(lines)
  712. )
  713. linecache.updatecache('/foo.py', fake_module)
  714. e = SyntaxError("uh oh")
  715. e.filename = '/foo.py'
  716. e.lineno = 2
  717. e.text = b('something wrong')
  718. e.offset = 1
  719. c = test_code('/foo.py', 'method')
  720. f = test_frame(c, fake_module, {'something': 1})
  721. tb = test_tb(f, 2, None)
  722. exc = traceback.TracebackException(SyntaxError, e, tb)
  723. list(exc.format_exception_only())
  724. self.assertEqual([
  725. u('Traceback (most recent call last):\n'),
  726. u(' File "/foo.py", line 2, in method\n \u5341\n'),
  727. u(' File "/foo.py", line 2\n'),
  728. u(' \u5341\n'),
  729. u(' ^\n'),
  730. u('SyntaxError: uh oh\n')],
  731. list(exc.format()))
  732. @unittest.skipUnless(sys.version_info[0] < 3, "Applies to 2.x only.")
  733. @unittest.skipIf(sys.getfilesystemencoding()=='ANSI_X3.4-1968',
  734. 'Requires non-ascii fs encoding')
  735. def test_format_unicode_filename(self):
  736. # Filenames in Python2 may be bytestrings that will fail to implicit
  737. # decode.
  738. fname = u('\u5341').encode(sys.getfilesystemencoding())
  739. lines = u("1\n2\n3\n")
  740. fake_module = dict(
  741. __name__='fred',
  742. __loader__=FakeLoader(lines)
  743. )
  744. linecache.updatecache(fname, fake_module)
  745. e = SyntaxError("uh oh")
  746. e.filename = fname
  747. e.lineno = 2
  748. e.text = b('something wrong')
  749. e.offset = 1
  750. c = test_code(fname, 'method')
  751. f = test_frame(c, fake_module, {'something': 1})
  752. tb = test_tb(f, 2, None)
  753. exc = traceback.TracebackException(SyntaxError, e, tb)
  754. list(exc.format_exception_only())
  755. self.assertEqual([
  756. u('Traceback (most recent call last):\n'),
  757. u(' File "\u5341", line 2, in method\n 2\n'),
  758. u(' File "\u5341", line 2\n'),
  759. u(' something wrong\n'),
  760. u(' ^\n'),
  761. u('SyntaxError: uh oh\n')],
  762. list(exc.format()))
  763. @unittest.skipUnless(sys.version_info[0] < 3, "Applies to 2.x only.")
  764. def test_format_bad_filename(self):
  765. # Filenames in Python2 may be bytestrings that will fail to implicit
  766. # decode.
  767. # This won't decode via the implicit(ascii) codec or the default
  768. # fs encoding (unless the encoding is a wildcard encoding).
  769. fname = b('\x8b')
  770. lines = u("1\n2\n3\n")
  771. fake_module = dict(
  772. __name__='fred',
  773. __loader__=FakeLoader(lines)
  774. )
  775. linecache.updatecache(fname, fake_module)
  776. e = SyntaxError("uh oh")
  777. e.filename = fname
  778. e.lineno = 2
  779. e.text = b('something wrong')
  780. e.offset = 1
  781. c = test_code(fname, 'method')
  782. f = test_frame(c, fake_module, {'something': 1})
  783. tb = test_tb(f, 2, None)
  784. exc = traceback.TracebackException(SyntaxError, e, tb)
  785. list(exc.format_exception_only())
  786. self.assertEqual([
  787. u('Traceback (most recent call last):\n'),
  788. b(' File "b\'\\x8b\'", line 2, in method\n 2\n').decode(),
  789. b(' File "b\'\\x8b\'", line 2\n').decode(),
  790. u(' something wrong\n'),
  791. u(' ^\n'),
  792. u('SyntaxError: uh oh\n')],
  793. list(exc.format()))