diff options
Diffstat (limited to 'Lib/test/test_dis.py')
| -rw-r--r-- | Lib/test/test_dis.py | 111 | 
1 files changed, 100 insertions, 11 deletions
| diff --git a/Lib/test/test_dis.py b/Lib/test/test_dis.py index 5c59eaa..1bcd693 100644 --- a/Lib/test/test_dis.py +++ b/Lib/test/test_dis.py @@ -1,11 +1,35 @@  # Minimal tests for dis module  from test.support import run_unittest, captured_stdout +import difflib  import unittest  import sys  import dis  import io +class _C: +    def __init__(self, x): +        self.x = x == 1 + +dis_c_instance_method = """\ + %-4d         0 LOAD_FAST                1 (x) +              3 LOAD_CONST               1 (1) +              6 COMPARE_OP               2 (==) +              9 LOAD_FAST                0 (self) +             12 STORE_ATTR               0 (x) +             15 LOAD_CONST               0 (None) +             18 RETURN_VALUE +""" % (_C.__init__.__code__.co_firstlineno + 1,) + +dis_c_instance_method_bytes = """\ +          0 LOAD_FAST           1 (1) +          3 LOAD_CONST          1 (1) +          6 COMPARE_OP          2 (==) +          9 LOAD_FAST           0 (0) +         12 STORE_ATTR          0 (0) +         15 LOAD_CONST          0 (0) +         18 RETURN_VALUE +"""  def _f(a):      print(a) @@ -14,7 +38,7 @@ def _f(a):  dis_f = """\   %-4d         0 LOAD_GLOBAL              0 (print)                3 LOAD_FAST                0 (a) -              6 CALL_FUNCTION            1 +              6 CALL_FUNCTION            1 (1 positional, 0 keyword pair)                9 POP_TOP   %-4d        10 LOAD_CONST               1 (1) @@ -23,6 +47,16 @@ dis_f = """\         _f.__code__.co_firstlineno + 2) +dis_f_co_code = """\ +          0 LOAD_GLOBAL         0 (0) +          3 LOAD_FAST           0 (0) +          6 CALL_FUNCTION       1 (1 positional, 0 keyword pair) +          9 POP_TOP +         10 LOAD_CONST          1 (1) +         13 RETURN_VALUE +""" + +  def bug708901():      for res in range(1,                       10): @@ -34,7 +68,7 @@ dis_bug708901 = """\                6 LOAD_CONST               1 (1)   %-4d         9 LOAD_CONST               2 (10) -             12 CALL_FUNCTION            2 +             12 CALL_FUNCTION            2 (2 positional, 0 keyword pair)               15 GET_ITER          >>   16 FOR_ITER                 6 (to 25)               19 STORE_FAST               0 (res) @@ -138,18 +172,27 @@ dis_compound_stmt_str = """\  """  class DisTests(unittest.TestCase): -    def do_disassembly_test(self, func, expected): + +    def get_disassembly(self, func, lasti=-1, wrapper=True):          s = io.StringIO()          save_stdout = sys.stdout          sys.stdout = s -        dis.dis(func) -        sys.stdout = save_stdout -        got = s.getvalue() +        try: +            if wrapper: +                dis.dis(func) +            else: +                dis.disassemble(func, lasti) +        finally: +            sys.stdout = save_stdout          # Trim trailing blanks (if any). -        lines = got.split('\n') -        lines = [line.rstrip() for line in lines] -        expected = expected.split("\n") -        import difflib +        return [line.rstrip() for line in s.getvalue().splitlines()] + +    def get_disassemble_as_string(self, func, lasti=-1): +        return '\n'.join(self.get_disassembly(func, lasti, False)) + +    def do_disassembly_test(self, func, expected): +        lines = self.get_disassembly(func) +        expected = expected.splitlines()          if expected != lines:              self.fail(                  "events did not match expectation:\n" + @@ -157,7 +200,7 @@ class DisTests(unittest.TestCase):                                          lines)))      def test_opmap(self): -        self.assertEqual(dis.opmap["STOP_CODE"], 0) +        self.assertEqual(dis.opmap["NOP"], 9)          self.assertIn(dis.opmap["LOAD_CONST"], dis.hasconst)          self.assertIn(dis.opmap["STORE_NAME"], dis.hasname) @@ -211,6 +254,44 @@ class DisTests(unittest.TestCase):          self.do_disassembly_test(simple_stmt_str, dis_simple_stmt_str)          self.do_disassembly_test(compound_stmt_str, dis_compound_stmt_str) +    def test_disassemble_bytes(self): +        self.do_disassembly_test(_f.__code__.co_code, dis_f_co_code) + +    def test_disassemble_method(self): +        self.do_disassembly_test(_C(1).__init__, dis_c_instance_method) + +    def test_disassemble_method_bytes(self): +        method_bytecode = _C(1).__init__.__code__.co_code +        self.do_disassembly_test(method_bytecode, dis_c_instance_method_bytes) + +    def test_dis_none(self): +        try: +            del sys.last_traceback +        except AttributeError: +            pass +        self.assertRaises(RuntimeError, dis.dis, None) + +    def test_dis_object(self): +        self.assertRaises(TypeError, dis.dis, object()) + +    def test_dis_traceback(self): +        try: +            del sys.last_traceback +        except AttributeError: +            pass + +        try: +            1/0 +        except Exception as e: +            tb = e.__traceback__ +            sys.last_traceback = tb + +        tb_dis = self.get_disassemble_as_string(tb.tb_frame.f_code, tb.tb_lasti) +        self.do_disassembly_test(None, tb_dis) + +    def test_dis_object(self): +        self.assertRaises(TypeError, dis.dis, object()) +  code_info_code_info = """\  Name:              code_info  Filename:          (.*) @@ -258,6 +339,7 @@ Flags:             OPTIMIZED, NEWLOCALS, VARARGS, VARKEYWORDS, GENERATOR  Constants:     0: None     1: <code object f at (.*), file "(.*)", line (.*)> +   2: 'tricky.<locals>.f'  Variable names:     0: x     1: y @@ -364,6 +446,13 @@ class CodeInfoTests(unittest.TestCase):                  dis.show_code(x)              self.assertRegex(output.getvalue(), expected+"\n") +    def test_code_info_object(self): +        self.assertRaises(TypeError, dis.code_info, object()) + +    def test_pretty_flags_no_flags(self): +        self.assertEqual(dis.pretty_flags(0), '0x0') + +  def test_main():      run_unittest(DisTests, CodeInfoTests) | 
