| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  | import os | 
					
						
							|  |  |  | import sys | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  | from test.support import TESTFN, TESTFN_UNICODE, FS_NONASCII, rmtree, unlink, captured_stdout | 
					
						
							| 
									
										
										
										
											2016-01-13 07:46:54 -08:00
										 |  |  | from test.support.script_helper import assert_python_ok, assert_python_failure | 
					
						
							| 
									
										
										
										
											2018-04-30 20:46:52 -07:00
										 |  |  | import textwrap | 
					
						
							| 
									
										
										
										
											2010-08-02 12:48:46 +00:00
										 |  |  | import unittest | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-02 12:48:46 +00:00
										 |  |  | import trace | 
					
						
							| 
									
										
										
										
											2016-01-11 07:09:42 -08:00
										 |  |  | from trace import Trace | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | from test.tracedmodules import testmod | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #------------------------------- Utilities -----------------------------------# | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def fix_ext_py(filename): | 
					
						
							| 
									
										
										
										
											2015-04-13 14:21:02 -04:00
										 |  |  |     """Given a .pyc filename converts it to the appropriate .py""" | 
					
						
							|  |  |  |     if filename.endswith('.pyc'): | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  |         filename = filename[:-1] | 
					
						
							|  |  |  |     return filename | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def my_file_and_modname(): | 
					
						
							|  |  |  |     """The .py file and module name of this file (__file__)""" | 
					
						
							|  |  |  |     modname = os.path.splitext(os.path.basename(__file__))[0] | 
					
						
							|  |  |  |     return fix_ext_py(__file__), modname | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_firstlineno(func): | 
					
						
							|  |  |  |     return func.__code__.co_firstlineno | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #-------------------- Target functions for tracing ---------------------------# | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # The relative line numbers of lines in these functions matter for verifying | 
					
						
							|  |  |  | # tracing. Please modify the appropriate tests if you change one of the | 
					
						
							|  |  |  | # functions. Absolute line numbers don't matter. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def traced_func_linear(x, y): | 
					
						
							|  |  |  |     a = x | 
					
						
							|  |  |  |     b = y | 
					
						
							|  |  |  |     c = a + b | 
					
						
							|  |  |  |     return c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def traced_func_loop(x, y): | 
					
						
							|  |  |  |     c = x | 
					
						
							|  |  |  |     for i in range(5): | 
					
						
							|  |  |  |         c += y | 
					
						
							|  |  |  |     return c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def traced_func_importing(x, y): | 
					
						
							|  |  |  |     return x + y + testmod.func(1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def traced_func_simple_caller(x): | 
					
						
							|  |  |  |     c = traced_func_linear(x, x) | 
					
						
							|  |  |  |     return c + x | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def traced_func_importing_caller(x): | 
					
						
							|  |  |  |     k = traced_func_simple_caller(x) | 
					
						
							|  |  |  |     k += traced_func_importing(k, x) | 
					
						
							|  |  |  |     return k | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def traced_func_generator(num): | 
					
						
							|  |  |  |     c = 5       # executed once | 
					
						
							|  |  |  |     for i in range(num): | 
					
						
							|  |  |  |         yield i + c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def traced_func_calling_generator(): | 
					
						
							|  |  |  |     k = 0 | 
					
						
							|  |  |  |     for i in traced_func_generator(10): | 
					
						
							|  |  |  |         k += i | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def traced_doubler(num): | 
					
						
							|  |  |  |     return num * 2 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-01 09:16:35 +03:00
										 |  |  | def traced_capturer(*args, **kwargs): | 
					
						
							|  |  |  |     return args, kwargs | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  | def traced_caller_list_comprehension(): | 
					
						
							|  |  |  |     k = 10 | 
					
						
							|  |  |  |     mylist = [traced_doubler(i) for i in range(k)] | 
					
						
							|  |  |  |     return mylist | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 13:16:02 +02:00
										 |  |  | def traced_decorated_function(): | 
					
						
							|  |  |  |     def decorator1(f): | 
					
						
							|  |  |  |         return f | 
					
						
							|  |  |  |     def decorator_fabric(): | 
					
						
							|  |  |  |         def decorator2(f): | 
					
						
							|  |  |  |             return f | 
					
						
							|  |  |  |         return decorator2 | 
					
						
							|  |  |  |     @decorator1 | 
					
						
							|  |  |  |     @decorator_fabric() | 
					
						
							|  |  |  |     def func(): | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  |     func() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class TracedClass(object): | 
					
						
							|  |  |  |     def __init__(self, x): | 
					
						
							|  |  |  |         self.a = x | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def inst_method_linear(self, y): | 
					
						
							|  |  |  |         return self.a + y | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def inst_method_calling(self, x): | 
					
						
							|  |  |  |         c = self.inst_method_linear(x) | 
					
						
							|  |  |  |         return c + traced_func_linear(x, c) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     def class_method_linear(cls, y): | 
					
						
							|  |  |  |         return y * 2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @staticmethod | 
					
						
							|  |  |  |     def static_method_linear(y): | 
					
						
							|  |  |  |         return y * 2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #------------------------------ Test cases -----------------------------------# | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TestLineCounts(unittest.TestCase): | 
					
						
							|  |  |  |     """White-box testing of line-counting, via runfunc""" | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							| 
									
										
										
										
											2011-02-21 19:29:56 +00:00
										 |  |  |         self.addCleanup(sys.settrace, sys.gettrace()) | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  |         self.tracer = Trace(count=1, trace=0, countfuncs=0, countcallers=0) | 
					
						
							|  |  |  |         self.my_py_filename = fix_ext_py(__file__) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_traced_func_linear(self): | 
					
						
							|  |  |  |         result = self.tracer.runfunc(traced_func_linear, 2, 5) | 
					
						
							|  |  |  |         self.assertEqual(result, 7) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # all lines are executed once | 
					
						
							|  |  |  |         expected = {} | 
					
						
							|  |  |  |         firstlineno = get_firstlineno(traced_func_linear) | 
					
						
							|  |  |  |         for i in range(1, 5): | 
					
						
							|  |  |  |             expected[(self.my_py_filename, firstlineno +  i)] = 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(self.tracer.results().counts, expected) | 
					
						
							| 
									
										
										
										
											2010-08-02 12:48:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  |     def test_traced_func_loop(self): | 
					
						
							|  |  |  |         self.tracer.runfunc(traced_func_loop, 2, 3) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         firstlineno = get_firstlineno(traced_func_loop) | 
					
						
							|  |  |  |         expected = { | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 1): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 2): 6, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 3): 5, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 4): 1, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         self.assertEqual(self.tracer.results().counts, expected) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_traced_func_importing(self): | 
					
						
							|  |  |  |         self.tracer.runfunc(traced_func_importing, 2, 5) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         firstlineno = get_firstlineno(traced_func_importing) | 
					
						
							|  |  |  |         expected = { | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 1): 1, | 
					
						
							|  |  |  |             (fix_ext_py(testmod.__file__), 2): 1, | 
					
						
							|  |  |  |             (fix_ext_py(testmod.__file__), 3): 1, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.assertEqual(self.tracer.results().counts, expected) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_trace_func_generator(self): | 
					
						
							|  |  |  |         self.tracer.runfunc(traced_func_calling_generator) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         firstlineno_calling = get_firstlineno(traced_func_calling_generator) | 
					
						
							|  |  |  |         firstlineno_gen = get_firstlineno(traced_func_generator) | 
					
						
							|  |  |  |         expected = { | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno_calling + 1): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno_calling + 2): 11, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno_calling + 3): 10, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno_gen + 1): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno_gen + 2): 11, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno_gen + 3): 10, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         self.assertEqual(self.tracer.results().counts, expected) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_trace_list_comprehension(self): | 
					
						
							|  |  |  |         self.tracer.runfunc(traced_caller_list_comprehension) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         firstlineno_calling = get_firstlineno(traced_caller_list_comprehension) | 
					
						
							|  |  |  |         firstlineno_called = get_firstlineno(traced_doubler) | 
					
						
							|  |  |  |         expected = { | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno_calling + 1): 1, | 
					
						
							| 
									
										
										
										
											2019-07-31 08:16:13 +10:00
										 |  |  |             # List comprehensions work differently in 3.x, so the count | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  |             # below changed compared to 2.x. | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno_calling + 2): 12, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno_calling + 3): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno_called + 1): 10, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         self.assertEqual(self.tracer.results().counts, expected) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 13:16:02 +02:00
										 |  |  |     def test_traced_decorated_function(self): | 
					
						
							|  |  |  |         self.tracer.runfunc(traced_decorated_function) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         firstlineno = get_firstlineno(traced_decorated_function) | 
					
						
							|  |  |  |         expected = { | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 1): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 2): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 3): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 4): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 5): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 6): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 7): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 8): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 9): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 10): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 11): 1, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         self.assertEqual(self.tracer.results().counts, expected) | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_linear_methods(self): | 
					
						
							|  |  |  |         # XXX todo: later add 'static_method_linear' and 'class_method_linear' | 
					
						
							|  |  |  |         # here, once issue1764286 is resolved | 
					
						
							|  |  |  |         # | 
					
						
							|  |  |  |         for methname in ['inst_method_linear',]: | 
					
						
							|  |  |  |             tracer = Trace(count=1, trace=0, countfuncs=0, countcallers=0) | 
					
						
							|  |  |  |             traced_obj = TracedClass(25) | 
					
						
							|  |  |  |             method = getattr(traced_obj, methname) | 
					
						
							|  |  |  |             tracer.runfunc(method, 20) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             firstlineno = get_firstlineno(method) | 
					
						
							|  |  |  |             expected = { | 
					
						
							|  |  |  |                 (self.my_py_filename, firstlineno + 1): 1, | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             self.assertEqual(tracer.results().counts, expected) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 13:16:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  | class TestRunExecCounts(unittest.TestCase): | 
					
						
							|  |  |  |     """A simple sanity test of line-counting, via runctx (exec)""" | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							|  |  |  |         self.my_py_filename = fix_ext_py(__file__) | 
					
						
							| 
									
										
										
										
											2011-02-21 19:29:56 +00:00
										 |  |  |         self.addCleanup(sys.settrace, sys.gettrace()) | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_exec_counts(self): | 
					
						
							|  |  |  |         self.tracer = Trace(count=1, trace=0, countfuncs=0, countcallers=0) | 
					
						
							|  |  |  |         code = r'''traced_func_loop(2, 5)''' | 
					
						
							|  |  |  |         code = compile(code, __file__, 'exec') | 
					
						
							|  |  |  |         self.tracer.runctx(code, globals(), vars()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         firstlineno = get_firstlineno(traced_func_loop) | 
					
						
							|  |  |  |         expected = { | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 1): 1, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 2): 6, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 3): 5, | 
					
						
							|  |  |  |             (self.my_py_filename, firstlineno + 4): 1, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-16 11:05:33 +02:00
										 |  |  |         # When used through 'run', some other spurious counts are produced, like | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  |         # the settrace of threading, which we ignore, just making sure that the | 
					
						
							|  |  |  |         # counts fo traced_func_loop were right. | 
					
						
							|  |  |  |         # | 
					
						
							|  |  |  |         for k in expected.keys(): | 
					
						
							|  |  |  |             self.assertEqual(self.tracer.results().counts[k], expected[k]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TestFuncs(unittest.TestCase): | 
					
						
							|  |  |  |     """White-box testing of funcs tracing""" | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							| 
									
										
										
										
											2011-02-21 19:29:56 +00:00
										 |  |  |         self.addCleanup(sys.settrace, sys.gettrace()) | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  |         self.tracer = Trace(count=0, trace=0, countfuncs=1) | 
					
						
							|  |  |  |         self.filemod = my_file_and_modname() | 
					
						
							| 
									
										
										
										
											2014-06-29 17:44:05 -04:00
										 |  |  |         self._saved_tracefunc = sys.gettrace() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def tearDown(self): | 
					
						
							|  |  |  |         if self._saved_tracefunc is not None: | 
					
						
							|  |  |  |             sys.settrace(self._saved_tracefunc) | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_simple_caller(self): | 
					
						
							|  |  |  |         self.tracer.runfunc(traced_func_simple_caller, 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         expected = { | 
					
						
							|  |  |  |             self.filemod + ('traced_func_simple_caller',): 1, | 
					
						
							|  |  |  |             self.filemod + ('traced_func_linear',): 1, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         self.assertEqual(self.tracer.results().calledfuncs, expected) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-01 09:16:35 +03:00
										 |  |  |     def test_arg_errors(self): | 
					
						
							|  |  |  |         res = self.tracer.runfunc(traced_capturer, 1, 2, self=3, func=4) | 
					
						
							|  |  |  |         self.assertEqual(res, ((1, 2), {'self': 3, 'func': 4})) | 
					
						
							| 
									
										
										
										
											2019-06-05 18:22:31 +03:00
										 |  |  |         with self.assertRaises(TypeError): | 
					
						
							|  |  |  |             self.tracer.runfunc(func=traced_capturer, arg=1) | 
					
						
							| 
									
										
										
										
											2019-04-01 09:16:35 +03:00
										 |  |  |         with self.assertRaises(TypeError): | 
					
						
							|  |  |  |             self.tracer.runfunc() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  |     def test_loop_caller_importing(self): | 
					
						
							|  |  |  |         self.tracer.runfunc(traced_func_importing_caller, 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         expected = { | 
					
						
							|  |  |  |             self.filemod + ('traced_func_simple_caller',): 1, | 
					
						
							|  |  |  |             self.filemod + ('traced_func_linear',): 1, | 
					
						
							|  |  |  |             self.filemod + ('traced_func_importing_caller',): 1, | 
					
						
							|  |  |  |             self.filemod + ('traced_func_importing',): 1, | 
					
						
							|  |  |  |             (fix_ext_py(testmod.__file__), 'testmod', 'func'): 1, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         self.assertEqual(self.tracer.results().calledfuncs, expected) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-22 03:04:06 +00:00
										 |  |  |     @unittest.skipIf(hasattr(sys, 'gettrace') and sys.gettrace(), | 
					
						
							|  |  |  |                      'pre-existing trace function throws off measurements') | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  |     def test_inst_method_calling(self): | 
					
						
							|  |  |  |         obj = TracedClass(20) | 
					
						
							|  |  |  |         self.tracer.runfunc(obj.inst_method_calling, 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         expected = { | 
					
						
							|  |  |  |             self.filemod + ('TracedClass.inst_method_calling',): 1, | 
					
						
							|  |  |  |             self.filemod + ('TracedClass.inst_method_linear',): 1, | 
					
						
							|  |  |  |             self.filemod + ('traced_func_linear',): 1, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         self.assertEqual(self.tracer.results().calledfuncs, expected) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 13:16:02 +02:00
										 |  |  |     def test_traced_decorated_function(self): | 
					
						
							|  |  |  |         self.tracer.runfunc(traced_decorated_function) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         expected = { | 
					
						
							|  |  |  |             self.filemod + ('traced_decorated_function',): 1, | 
					
						
							|  |  |  |             self.filemod + ('decorator_fabric',): 1, | 
					
						
							|  |  |  |             self.filemod + ('decorator2',): 1, | 
					
						
							|  |  |  |             self.filemod + ('decorator1',): 1, | 
					
						
							|  |  |  |             self.filemod + ('func',): 1, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         self.assertEqual(self.tracer.results().calledfuncs, expected) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class TestCallers(unittest.TestCase): | 
					
						
							|  |  |  |     """White-box testing of callers tracing""" | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							| 
									
										
										
										
											2011-02-21 19:29:56 +00:00
										 |  |  |         self.addCleanup(sys.settrace, sys.gettrace()) | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  |         self.tracer = Trace(count=0, trace=0, countcallers=1) | 
					
						
							|  |  |  |         self.filemod = my_file_and_modname() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-22 03:04:06 +00:00
										 |  |  |     @unittest.skipIf(hasattr(sys, 'gettrace') and sys.gettrace(), | 
					
						
							|  |  |  |                      'pre-existing trace function throws off measurements') | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  |     def test_loop_caller_importing(self): | 
					
						
							|  |  |  |         self.tracer.runfunc(traced_func_importing_caller, 1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         expected = { | 
					
						
							|  |  |  |             ((os.path.splitext(trace.__file__)[0] + '.py', 'trace', 'Trace.runfunc'), | 
					
						
							|  |  |  |                 (self.filemod + ('traced_func_importing_caller',))): 1, | 
					
						
							|  |  |  |             ((self.filemod + ('traced_func_simple_caller',)), | 
					
						
							|  |  |  |                 (self.filemod + ('traced_func_linear',))): 1, | 
					
						
							|  |  |  |             ((self.filemod + ('traced_func_importing_caller',)), | 
					
						
							|  |  |  |                 (self.filemod + ('traced_func_simple_caller',))): 1, | 
					
						
							|  |  |  |             ((self.filemod + ('traced_func_importing_caller',)), | 
					
						
							|  |  |  |                 (self.filemod + ('traced_func_importing',))): 1, | 
					
						
							|  |  |  |             ((self.filemod + ('traced_func_importing',)), | 
					
						
							|  |  |  |                 (fix_ext_py(testmod.__file__), 'testmod', 'func')): 1, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         self.assertEqual(self.tracer.results().callers, expected) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Created separately for issue #3821 | 
					
						
							| 
									
										
										
										
											2010-08-02 12:48:46 +00:00
										 |  |  | class TestCoverage(unittest.TestCase): | 
					
						
							| 
									
										
										
										
											2011-02-21 19:29:56 +00:00
										 |  |  |     def setUp(self): | 
					
						
							|  |  |  |         self.addCleanup(sys.settrace, sys.gettrace()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-02 12:48:46 +00:00
										 |  |  |     def tearDown(self): | 
					
						
							|  |  |  |         rmtree(TESTFN) | 
					
						
							|  |  |  |         unlink(TESTFN) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-24 18:03:12 +00:00
										 |  |  |     def _coverage(self, tracer, | 
					
						
							| 
									
										
										
										
											2015-05-20 19:37:10 +03:00
										 |  |  |                   cmd='import test.support, test.test_pprint;' | 
					
						
							|  |  |  |                       'test.support.run_unittest(test.test_pprint.QueryTestCase)'): | 
					
						
							| 
									
										
										
										
											2010-09-24 18:03:12 +00:00
										 |  |  |         tracer.run(cmd) | 
					
						
							| 
									
										
										
										
											2010-08-02 12:48:46 +00:00
										 |  |  |         r = tracer.results() | 
					
						
							|  |  |  |         r.write_results(show_missing=True, summary=True, coverdir=TESTFN) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_coverage(self): | 
					
						
							|  |  |  |         tracer = trace.Trace(trace=0, count=1) | 
					
						
							|  |  |  |         with captured_stdout() as stdout: | 
					
						
							|  |  |  |             self._coverage(tracer) | 
					
						
							|  |  |  |         stdout = stdout.getvalue() | 
					
						
							| 
									
										
										
										
											2016-11-11 12:05:01 +02:00
										 |  |  |         self.assertIn("pprint.py", stdout) | 
					
						
							|  |  |  |         self.assertIn("case.py", stdout)   # from unittest | 
					
						
							| 
									
										
										
										
											2010-08-02 12:48:46 +00:00
										 |  |  |         files = os.listdir(TESTFN) | 
					
						
							| 
									
										
										
										
											2016-11-11 12:05:01 +02:00
										 |  |  |         self.assertIn("pprint.cover", files) | 
					
						
							|  |  |  |         self.assertIn("unittest.case.cover", files) | 
					
						
							| 
									
										
										
										
											2010-08-02 12:48:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def test_coverage_ignore(self): | 
					
						
							|  |  |  |         # Ignore all files, nothing should be traced nor printed | 
					
						
							|  |  |  |         libpath = os.path.normpath(os.path.dirname(os.__file__)) | 
					
						
							|  |  |  |         # sys.prefix does not work when running from a checkout | 
					
						
							| 
									
										
										
										
											2012-05-26 03:45:29 +01:00
										 |  |  |         tracer = trace.Trace(ignoredirs=[sys.base_prefix, sys.base_exec_prefix, | 
					
						
							|  |  |  |                              libpath], trace=0, count=1) | 
					
						
							| 
									
										
										
										
											2010-08-02 12:48:46 +00:00
										 |  |  |         with captured_stdout() as stdout: | 
					
						
							|  |  |  |             self._coverage(tracer) | 
					
						
							|  |  |  |         if os.path.exists(TESTFN): | 
					
						
							|  |  |  |             files = os.listdir(TESTFN) | 
					
						
							| 
									
										
										
										
											2012-04-14 14:10:13 -04:00
										 |  |  |             self.assertEqual(files, ['_importlib.cover'])  # Ignore __import__ | 
					
						
							| 
									
										
										
										
											2010-08-02 12:48:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-24 18:03:12 +00:00
										 |  |  |     def test_issue9936(self): | 
					
						
							|  |  |  |         tracer = trace.Trace(trace=0, count=1) | 
					
						
							|  |  |  |         modname = 'test.tracedmodules.testmod' | 
					
						
							|  |  |  |         # Ensure that the module is executed in import | 
					
						
							|  |  |  |         if modname in sys.modules: | 
					
						
							|  |  |  |             del sys.modules[modname] | 
					
						
							|  |  |  |         cmd = ("import test.tracedmodules.testmod as t;" | 
					
						
							|  |  |  |                "t.func(0); t.func2();") | 
					
						
							|  |  |  |         with captured_stdout() as stdout: | 
					
						
							|  |  |  |             self._coverage(tracer, cmd) | 
					
						
							|  |  |  |         stdout.seek(0) | 
					
						
							|  |  |  |         stdout.readline() | 
					
						
							|  |  |  |         coverage = {} | 
					
						
							|  |  |  |         for line in stdout: | 
					
						
							|  |  |  |             lines, cov, module = line.split()[:3] | 
					
						
							|  |  |  |             coverage[module] = (int(lines), int(cov[:-1])) | 
					
						
							| 
									
										
										
										
											2010-09-24 22:04:22 +00:00
										 |  |  |         # XXX This is needed to run regrtest.py as a script | 
					
						
							| 
									
										
										
										
											2010-11-26 18:51:39 +00:00
										 |  |  |         modname = trace._fullmodname(sys.modules[modname].__file__) | 
					
						
							| 
									
										
										
										
											2010-09-24 18:03:12 +00:00
										 |  |  |         self.assertIn(modname, coverage) | 
					
						
							|  |  |  |         self.assertEqual(coverage[modname], (5, 100)) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-08 18:32:40 +00:00
										 |  |  | ### Tests that don't mess with sys.settrace and can be traced | 
					
						
							|  |  |  | ### themselves TODO: Skip tests that do mess with sys.settrace when | 
					
						
							|  |  |  | ### regrtest is invoked with -T option. | 
					
						
							|  |  |  | class Test_Ignore(unittest.TestCase): | 
					
						
							|  |  |  |     def test_ignored(self): | 
					
						
							| 
									
										
										
										
											2010-11-08 23:10:20 +00:00
										 |  |  |         jn = os.path.join | 
					
						
							| 
									
										
										
										
											2010-11-26 18:51:39 +00:00
										 |  |  |         ignore = trace._Ignore(['x', 'y.z'], [jn('foo', 'bar')]) | 
					
						
							| 
									
										
										
										
											2010-11-08 18:32:40 +00:00
										 |  |  |         self.assertTrue(ignore.names('x.py', 'x')) | 
					
						
							|  |  |  |         self.assertFalse(ignore.names('xy.py', 'xy')) | 
					
						
							|  |  |  |         self.assertFalse(ignore.names('y.py', 'y')) | 
					
						
							| 
									
										
										
										
											2010-11-08 23:10:20 +00:00
										 |  |  |         self.assertTrue(ignore.names(jn('foo', 'bar', 'baz.py'), 'baz')) | 
					
						
							|  |  |  |         self.assertFalse(ignore.names(jn('bar', 'z.py'), 'z')) | 
					
						
							| 
									
										
										
										
											2010-11-08 18:32:40 +00:00
										 |  |  |         # Matched before. | 
					
						
							| 
									
										
										
										
											2010-11-08 23:10:20 +00:00
										 |  |  |         self.assertTrue(ignore.names(jn('bar', 'baz.py'), 'baz')) | 
					
						
							| 
									
										
										
										
											2010-11-08 18:32:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-30 20:46:52 -07:00
										 |  |  | # Created for Issue 31908 -- CLI utility not writing cover files | 
					
						
							|  |  |  | class TestCoverageCommandLineOutput(unittest.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     codefile = 'tmp.py' | 
					
						
							|  |  |  |     coverfile = 'tmp.cover' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def setUp(self): | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  |         with open(self.codefile, 'w', encoding='iso-8859-15') as f: | 
					
						
							| 
									
										
										
										
											2018-04-30 20:46:52 -07:00
										 |  |  |             f.write(textwrap.dedent('''\
 | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  |                 # coding: iso-8859-15 | 
					
						
							|  |  |  |                 x = 'spœm' | 
					
						
							| 
									
										
										
										
											2018-04-30 20:46:52 -07:00
										 |  |  |                 if []: | 
					
						
							|  |  |  |                     print('unreachable') | 
					
						
							|  |  |  |             '''))
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def tearDown(self): | 
					
						
							|  |  |  |         unlink(self.codefile) | 
					
						
							|  |  |  |         unlink(self.coverfile) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_cover_files_written_no_highlight(self): | 
					
						
							| 
									
										
										
										
											2018-08-27 13:10:36 +03:00
										 |  |  |         # Test also that the cover file for the trace module is not created | 
					
						
							|  |  |  |         # (issue #34171). | 
					
						
							|  |  |  |         tracedir = os.path.dirname(os.path.abspath(trace.__file__)) | 
					
						
							|  |  |  |         tracecoverpath = os.path.join(tracedir, 'trace.cover') | 
					
						
							|  |  |  |         unlink(tracecoverpath) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-30 20:46:52 -07:00
										 |  |  |         argv = '-m trace --count'.split() + [self.codefile] | 
					
						
							|  |  |  |         status, stdout, stderr = assert_python_ok(*argv) | 
					
						
							| 
									
										
										
										
											2018-08-25 10:27:55 +03:00
										 |  |  |         self.assertEqual(stderr, b'') | 
					
						
							|  |  |  |         self.assertFalse(os.path.exists(tracecoverpath)) | 
					
						
							| 
									
										
										
										
											2018-04-30 20:46:52 -07:00
										 |  |  |         self.assertTrue(os.path.exists(self.coverfile)) | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  |         with open(self.coverfile, encoding='iso-8859-15') as f: | 
					
						
							| 
									
										
										
										
											2018-04-30 20:46:52 -07:00
										 |  |  |             self.assertEqual(f.read(), | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  |                 "       # coding: iso-8859-15\n" | 
					
						
							|  |  |  |                 "    1: x = 'spœm'\n" | 
					
						
							| 
									
										
										
										
											2018-04-30 20:46:52 -07:00
										 |  |  |                 "    1: if []:\n" | 
					
						
							|  |  |  |                 "           print('unreachable')\n" | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_cover_files_written_with_highlight(self): | 
					
						
							|  |  |  |         argv = '-m trace --count --missing'.split() + [self.codefile] | 
					
						
							|  |  |  |         status, stdout, stderr = assert_python_ok(*argv) | 
					
						
							|  |  |  |         self.assertTrue(os.path.exists(self.coverfile)) | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  |         with open(self.coverfile, encoding='iso-8859-15') as f: | 
					
						
							| 
									
										
										
										
											2018-04-30 20:46:52 -07:00
										 |  |  |             self.assertEqual(f.read(), textwrap.dedent('''\
 | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  |                        # coding: iso-8859-15 | 
					
						
							|  |  |  |                     1: x = 'spœm' | 
					
						
							| 
									
										
										
										
											2018-04-30 20:46:52 -07:00
										 |  |  |                     1: if []: | 
					
						
							|  |  |  |                 >>>>>>     print('unreachable') | 
					
						
							|  |  |  |             '''))
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-13 07:46:54 -08:00
										 |  |  | class TestCommandLine(unittest.TestCase): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_failures(self): | 
					
						
							|  |  |  |         _errors = ( | 
					
						
							| 
									
										
										
										
											2019-06-01 05:49:10 +01:00
										 |  |  |             (b'progname is missing: required with the main options', '-l', '-T'), | 
					
						
							| 
									
										
										
										
											2016-01-13 07:46:54 -08:00
										 |  |  |             (b'cannot specify both --listfuncs and (--trace or --count)', '-lc'), | 
					
						
							|  |  |  |             (b'argument -R/--no-report: not allowed with argument -r/--report', '-rR'), | 
					
						
							|  |  |  |             (b'must specify one of --trace, --count, --report, --listfuncs, or --trackcalls', '-g'), | 
					
						
							|  |  |  |             (b'-r/--report requires -f/--file', '-r'), | 
					
						
							|  |  |  |             (b'--summary can only be used with --count or --report', '-sT'), | 
					
						
							|  |  |  |             (b'unrecognized arguments: -y', '-y')) | 
					
						
							|  |  |  |         for message, *args in _errors: | 
					
						
							|  |  |  |             *_, stderr = assert_python_failure('-m', 'trace', *args) | 
					
						
							|  |  |  |             self.assertIn(message, stderr) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def test_listfuncs_flag_success(self): | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  |         filename = TESTFN + '.py' | 
					
						
							|  |  |  |         modulename = os.path.basename(TESTFN) | 
					
						
							|  |  |  |         with open(filename, 'w', encoding='utf-8') as fd: | 
					
						
							|  |  |  |             self.addCleanup(unlink, filename) | 
					
						
							| 
									
										
										
										
											2016-01-13 07:46:54 -08:00
										 |  |  |             fd.write("a = 1\n") | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  |             status, stdout, stderr = assert_python_ok('-m', 'trace', '-l', filename, | 
					
						
							| 
									
										
										
										
											2020-06-25 20:39:12 +03:00
										 |  |  |                                                       PYTHONIOENCODING='utf-8') | 
					
						
							| 
									
										
										
										
											2016-01-13 07:46:54 -08:00
										 |  |  |             self.assertIn(b'functions called:', stdout) | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  |             expected = f'filename: {filename}, modulename: {modulename}, funcname: <module>' | 
					
						
							|  |  |  |             self.assertIn(expected.encode(), stdout) | 
					
						
							| 
									
										
										
										
											2010-08-02 12:48:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-16 22:32:37 -08:00
										 |  |  |     def test_sys_argv_list(self): | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  |         with open(TESTFN, 'w', encoding='utf-8') as fd: | 
					
						
							| 
									
										
										
										
											2018-02-16 22:32:37 -08:00
										 |  |  |             self.addCleanup(unlink, TESTFN) | 
					
						
							|  |  |  |             fd.write("import sys\n") | 
					
						
							|  |  |  |             fd.write("print(type(sys.argv))\n") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         status, direct_stdout, stderr = assert_python_ok(TESTFN) | 
					
						
							| 
									
										
										
										
											2020-06-25 20:39:12 +03:00
										 |  |  |         status, trace_stdout, stderr = assert_python_ok('-m', 'trace', '-l', TESTFN, | 
					
						
							|  |  |  |                                                         PYTHONIOENCODING='utf-8') | 
					
						
							| 
									
										
										
										
											2018-02-16 22:32:37 -08:00
										 |  |  |         self.assertIn(direct_stdout.strip(), trace_stdout) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 09:15:43 +03:00
										 |  |  |     def test_count_and_summary(self): | 
					
						
							|  |  |  |         filename = f'{TESTFN}.py' | 
					
						
							|  |  |  |         coverfilename = f'{TESTFN}.cover' | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  |         modulename = os.path.basename(TESTFN) | 
					
						
							|  |  |  |         with open(filename, 'w', encoding='utf-8') as fd: | 
					
						
							| 
									
										
										
										
											2018-08-11 09:15:43 +03:00
										 |  |  |             self.addCleanup(unlink, filename) | 
					
						
							|  |  |  |             self.addCleanup(unlink, coverfilename) | 
					
						
							|  |  |  |             fd.write(textwrap.dedent("""\
 | 
					
						
							|  |  |  |                 x = 1 | 
					
						
							|  |  |  |                 y = 2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 def f(): | 
					
						
							|  |  |  |                     return x + y | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 for i in range(10): | 
					
						
							|  |  |  |                     f() | 
					
						
							|  |  |  |             """))
 | 
					
						
							| 
									
										
										
										
											2020-06-25 20:39:12 +03:00
										 |  |  |         status, stdout, _ = assert_python_ok('-m', 'trace', '-cs', filename, | 
					
						
							|  |  |  |                                              PYTHONIOENCODING='utf-8') | 
					
						
							| 
									
										
										
										
											2018-08-11 09:15:43 +03:00
										 |  |  |         stdout = stdout.decode() | 
					
						
							|  |  |  |         self.assertEqual(status, 0) | 
					
						
							|  |  |  |         self.assertIn('lines   cov%   module   (path)', stdout) | 
					
						
							| 
									
										
										
										
											2020-06-28 03:52:26 -07:00
										 |  |  |         self.assertIn(f'6   100%   {modulename}   ({filename})', stdout) | 
					
						
							| 
									
										
										
										
											2018-08-11 09:15:43 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-01 05:49:10 +01:00
										 |  |  |     def test_run_as_module(self): | 
					
						
							|  |  |  |         assert_python_ok('-m', 'trace', '-l', '--module', 'timeit', '-n', '1') | 
					
						
							|  |  |  |         assert_python_failure('-m', 'trace', '-l', '--module', 'not_a_module_zzz') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-13 18:14:34 +00:00
										 |  |  | if __name__ == '__main__': | 
					
						
							| 
									
										
										
										
											2015-05-20 19:37:10 +03:00
										 |  |  |     unittest.main() |