|
"""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) |
|
self.fail(msg) |
|
elif instr.argval == argval: |
|
msg = '(%s,%r) occurs in bytecode:\n%s' |
|
msg = msg % (opname, argval, disassembly) |
|
self.fail(msg) |
|
|