2022-09-17 15:26:13 +03:00
|
|
|
"""bytecode_helper - support tools for testing correct bytecode generation"""
|
|
|
|
|
|
|
|
import unittest
|
|
|
|
import dis
|
|
|
|
import io
|
|
|
|
|
|
|
|
_UNSPECIFIED = object()
|
|
|
|
|
|
|
|
class BytecodeTestCase(unittest.TestCase):
|
|
|
|
"""Custom assertion methods for inspecting bytecode."""
|
|
|
|
|
|
|
|
def get_disassembly_as_string(self, co):
|
|
|
|
s = io.StringIO()
|
|
|
|
dis.dis(co, file=s)
|
|
|
|
return s.getvalue()
|
|
|
|
|
|
|
|
def assertInBytecode(self, x, opname, argval=_UNSPECIFIED):
|
|
|
|
"""Returns instr if opname is found, otherwise throws AssertionError"""
|
|
|
|
for instr in dis.get_instructions(x):
|
|
|
|
if instr.opname == opname:
|
|
|
|
if argval is _UNSPECIFIED or instr.argval == argval:
|
|
|
|
return instr
|
|
|
|
disassembly = self.get_disassembly_as_string(x)
|
|
|
|
if argval is _UNSPECIFIED:
|
|
|
|
msg = '%s not found in bytecode:\n%s' % (opname, disassembly)
|
|
|
|
else:
|
|
|
|
msg = '(%s,%r) not found in bytecode:\n%s'
|
|
|
|
msg = msg % (opname, argval, disassembly)
|
|
|
|
self.fail(msg)
|
|
|
|
|
|
|
|
def assertNotInBytecode(self, x, opname, argval=_UNSPECIFIED):
|
|
|
|
"""Throws AssertionError if opname is found"""
|
|
|
|
for instr in dis.get_instructions(x):
|
|
|
|
if instr.opname == opname:
|
|
|
|
disassembly = self.get_disassembly_as_string(x)
|
|
|
|
if argval is _UNSPECIFIED:
|
|
|
|
msg = '%s occurs in bytecode:\n%s' % (opname, disassembly)
|
2022-10-09 16:27:10 +03:00
|
|
|
self.fail(msg)
|
2022-09-17 15:26:13 +03:00
|
|
|
elif instr.argval == argval:
|
|
|
|
msg = '(%s,%r) occurs in bytecode:\n%s'
|
|
|
|
msg = msg % (opname, argval, disassembly)
|
2022-10-09 16:27:10 +03:00
|
|
|
self.fail(msg)
|