99 lines
3.1 KiB
Python
99 lines
3.1 KiB
Python
|
import opcode
|
||
|
import re
|
||
|
import sys
|
||
|
import textwrap
|
||
|
import unittest
|
||
|
|
||
|
from test.support import os_helper, verbose
|
||
|
from test.support.script_helper import assert_python_ok
|
||
|
|
||
|
|
||
|
Py_DEBUG = hasattr(sys, 'gettotalrefcount')
|
||
|
|
||
|
@unittest.skipUnless(Py_DEBUG, "lltrace requires Py_DEBUG")
|
||
|
class TestLLTrace(unittest.TestCase):
|
||
|
|
||
|
def test_lltrace_does_not_crash_on_subscript_operator(self):
|
||
|
# If this test fails, it will reproduce a crash reported as
|
||
|
# bpo-34113. The crash happened at the command line console of
|
||
|
# debug Python builds with __ltrace__ enabled (only possible in console),
|
||
|
# when the internal Python stack was negatively adjusted
|
||
|
with open(os_helper.TESTFN, 'w', encoding='utf-8') as fd:
|
||
|
self.addCleanup(os_helper.unlink, os_helper.TESTFN)
|
||
|
fd.write(textwrap.dedent("""\
|
||
|
import code
|
||
|
|
||
|
console = code.InteractiveConsole()
|
||
|
console.push('__ltrace__ = 1')
|
||
|
console.push('a = [1, 2, 3]')
|
||
|
console.push('a[0] = 1')
|
||
|
print('unreachable if bug exists')
|
||
|
"""))
|
||
|
|
||
|
assert_python_ok(os_helper.TESTFN)
|
||
|
|
||
|
def run_code(self, code):
|
||
|
code = textwrap.dedent(code).strip()
|
||
|
with open(os_helper.TESTFN, 'w', encoding='utf-8') as fd:
|
||
|
self.addCleanup(os_helper.unlink, os_helper.TESTFN)
|
||
|
fd.write(code)
|
||
|
status, stdout, stderr = assert_python_ok(os_helper.TESTFN)
|
||
|
self.assertEqual(stderr, b"")
|
||
|
self.assertEqual(status, 0)
|
||
|
result = stdout.decode('utf-8')
|
||
|
if verbose:
|
||
|
print("\n\n--- code ---")
|
||
|
print(code)
|
||
|
print("\n--- stdout ---")
|
||
|
print(result)
|
||
|
print()
|
||
|
return result
|
||
|
|
||
|
def check_op(self, op, stdout, present):
|
||
|
op = opcode.opmap[op]
|
||
|
regex = re.compile(f': {op}($|, )', re.MULTILINE)
|
||
|
if present:
|
||
|
self.assertTrue(regex.search(stdout),
|
||
|
f'": {op}" not found in: {stdout}')
|
||
|
else:
|
||
|
self.assertFalse(regex.search(stdout),
|
||
|
f'": {op}" found in: {stdout}')
|
||
|
|
||
|
def check_op_in(self, op, stdout):
|
||
|
self.check_op(op, stdout, True)
|
||
|
|
||
|
def check_op_not_in(self, op, stdout):
|
||
|
self.check_op(op, stdout, False)
|
||
|
|
||
|
def test_lltrace(self):
|
||
|
stdout = self.run_code("""
|
||
|
def dont_trace_1():
|
||
|
a = "a"
|
||
|
a = 10 * a
|
||
|
def trace_me():
|
||
|
for i in range(3):
|
||
|
+i
|
||
|
def dont_trace_2():
|
||
|
x = 42
|
||
|
y = -x
|
||
|
dont_trace_1()
|
||
|
__ltrace__ = 1
|
||
|
trace_me()
|
||
|
del __ltrace__
|
||
|
dont_trace_2()
|
||
|
""")
|
||
|
self.check_op_in("GET_ITER", stdout)
|
||
|
self.check_op_in("FOR_ITER", stdout)
|
||
|
self.check_op_in("UNARY_POSITIVE", stdout)
|
||
|
self.check_op_in("POP_TOP", stdout)
|
||
|
|
||
|
# before: dont_trace_1() is not traced
|
||
|
self.check_op_not_in("BINARY_MULTIPLY", stdout)
|
||
|
|
||
|
# after: dont_trace_2() is not traced
|
||
|
self.check_op_not_in("UNARY_NEGATIVE", stdout)
|
||
|
|
||
|
|
||
|
if __name__ == "__main__":
|
||
|
unittest.main()
|