|
1 | 1 | import unittest
|
| 2 | +import builtins |
| 3 | +from collections import UserDict |
2 | 4 |
|
3 | 5 | from test.support import import_helper
|
| 6 | +from test.support import swap_attr |
4 | 7 |
|
5 | 8 |
|
6 | 9 | # Skip this test if the _testcapi module isn't available.
|
7 | 10 | _testcapi = import_helper.import_module('_testcapi')
|
8 | 11 |
|
| 12 | +NULL = None |
| 13 | + |
9 | 14 |
|
10 | 15 | class PyEval_EvalCodeExTests(unittest.TestCase):
|
11 | 16 |
|
12 | 17 | def test_simple(self):
|
13 | 18 | def f():
|
14 | 19 | return a
|
15 | 20 |
|
16 |
| - self.assertEqual(_testcapi.eval_code_ex(f.__code__, dict(a=1)), 1) |
17 |
| - |
18 |
| - # Need to force the compiler to use LOAD_NAME |
19 |
| - # def test_custom_locals(self): |
20 |
| - # def f(): |
21 |
| - # return |
| 21 | + eval_code_ex = _testcapi.eval_code_ex |
| 22 | + code = f.__code__ |
| 23 | + self.assertEqual(eval_code_ex(code, dict(a=1)), 1) |
| 24 | + |
| 25 | + self.assertRaises(NameError, eval_code_ex, code, {}) |
| 26 | + self.assertRaises(SystemError, eval_code_ex, code, UserDict(a=1)) |
| 27 | + self.assertRaises(SystemError, eval_code_ex, code, []) |
| 28 | + self.assertRaises(SystemError, eval_code_ex, code, 1) |
| 29 | + # CRASHES eval_code_ex(code, NULL) |
| 30 | + # CRASHES eval_code_ex(1, {}) |
| 31 | + # CRASHES eval_code_ex(NULL, {}) |
| 32 | + |
| 33 | + def test_custom_locals(self): |
| 34 | + # Monkey-patch __build_class__ to get a class code object. |
| 35 | + code = None |
| 36 | + def build_class(func, name, /, *bases, **kwds): |
| 37 | + nonlocal code |
| 38 | + code = func.__code__ |
| 39 | + |
| 40 | + with swap_attr(builtins, '__build_class__', build_class): |
| 41 | + class A: |
| 42 | + # Uses LOAD_NAME for a |
| 43 | + r[:] = [a] |
| 44 | + |
| 45 | + eval_code_ex = _testcapi.eval_code_ex |
| 46 | + results = [] |
| 47 | + g = dict(a=1, r=results) |
| 48 | + self.assertIsNone(eval_code_ex(code, g)) |
| 49 | + self.assertEqual(results, [1]) |
| 50 | + self.assertIsNone(eval_code_ex(code, g, dict(a=2))) |
| 51 | + self.assertEqual(results, [2]) |
| 52 | + self.assertIsNone(eval_code_ex(code, g, UserDict(a=3))) |
| 53 | + self.assertEqual(results, [3]) |
| 54 | + self.assertIsNone(eval_code_ex(code, g, {})) |
| 55 | + self.assertEqual(results, [1]) |
| 56 | + self.assertIsNone(eval_code_ex(code, g, NULL)) |
| 57 | + self.assertEqual(results, [1]) |
| 58 | + |
| 59 | + self.assertRaises(TypeError, eval_code_ex, code, g, []) |
| 60 | + self.assertRaises(TypeError, eval_code_ex, code, g, 1) |
| 61 | + self.assertRaises(NameError, eval_code_ex, code, dict(r=results), {}) |
| 62 | + self.assertRaises(NameError, eval_code_ex, code, dict(r=results), NULL) |
| 63 | + self.assertRaises(TypeError, eval_code_ex, code, dict(r=results), []) |
| 64 | + self.assertRaises(TypeError, eval_code_ex, code, dict(r=results), 1) |
22 | 65 |
|
23 | 66 | def test_with_args(self):
|
24 | 67 | def f(a, b, c):
|
25 | 68 | return a
|
26 | 69 |
|
27 |
| - self.assertEqual(_testcapi.eval_code_ex(f.__code__, {}, {}, (1, 2, 3)), 1) |
| 70 | + eval_code_ex = _testcapi.eval_code_ex |
| 71 | + code = f.__code__ |
| 72 | + self.assertEqual(eval_code_ex(code, {}, {}, (1, 2, 3)), 1) |
| 73 | + self.assertRaises(TypeError, eval_code_ex, code, {}, {}, (1, 2)) |
| 74 | + self.assertRaises(TypeError, eval_code_ex, code, {}, {}, (1, 2, 3, 4)) |
28 | 75 |
|
29 | 76 | def test_with_kwargs(self):
|
30 | 77 | def f(a, b, c):
|
31 | 78 | return a
|
32 | 79 |
|
33 |
| - self.assertEqual(_testcapi.eval_code_ex(f.__code__, {}, {}, (), dict(a=1, b=2, c=3)), 1) |
| 80 | + eval_code_ex = _testcapi.eval_code_ex |
| 81 | + code = f.__code__ |
| 82 | + self.assertEqual(eval_code_ex(code, {}, {}, (), dict(a=1, b=2, c=3)), 1) |
| 83 | + self.assertRaises(TypeError, eval_code_ex, code, {}, {}, (), dict(a=1, b=2)) |
| 84 | + self.assertRaises(TypeError, eval_code_ex, code, {}, {}, (), dict(a=1, b=2)) |
| 85 | + self.assertRaises(TypeError, eval_code_ex, code, {}, {}, (), dict(a=1, b=2, c=3, d=4)) |
34 | 86 |
|
35 | 87 | def test_with_default(self):
|
36 | 88 | def f(a):
|
37 | 89 | return a
|
38 | 90 |
|
39 |
| - self.assertEqual(_testcapi.eval_code_ex(f.__code__, {}, {}, (), {}, (1,)), 1) |
| 91 | + eval_code_ex = _testcapi.eval_code_ex |
| 92 | + code = f.__code__ |
| 93 | + self.assertEqual(eval_code_ex(code, {}, {}, (), {}, (1,)), 1) |
| 94 | + self.assertRaises(TypeError, eval_code_ex, code, {}, {}, (), {}, ()) |
40 | 95 |
|
41 | 96 | def test_with_kwarg_default(self):
|
42 | 97 | def f(*, a):
|
43 | 98 | return a
|
44 | 99 |
|
45 |
| - self.assertEqual(_testcapi.eval_code_ex(f.__code__, {}, {}, (), {}, (), dict(a=1)), 1) |
| 100 | + eval_code_ex = _testcapi.eval_code_ex |
| 101 | + code = f.__code__ |
| 102 | + self.assertEqual(eval_code_ex(code, {}, {}, (), {}, (), dict(a=1)), 1) |
| 103 | + self.assertRaises(TypeError, eval_code_ex, code, {}, {}, (), {}, (), {}) |
| 104 | + self.assertRaises(TypeError, eval_code_ex, code, {}, {}, (), {}, (), NULL) |
| 105 | + self.assertRaises(SystemError, eval_code_ex, code, {}, {}, (), {}, (), UserDict(a=1)) |
| 106 | + self.assertRaises(SystemError, eval_code_ex, code, {}, {}, (), {}, (), []) |
| 107 | + self.assertRaises(SystemError, eval_code_ex, code, {}, {}, (), {}, (), 1) |
46 | 108 |
|
47 | 109 | def test_with_closure(self):
|
48 | 110 | a = 1
|
| 111 | + b = 2 |
49 | 112 | def f():
|
| 113 | + b |
50 | 114 | return a
|
51 | 115 |
|
52 |
| - self.assertEqual(_testcapi.eval_code_ex(f.__code__, {}, {}, (), {}, (), {}, f.__closure__), 1) |
| 116 | + eval_code_ex = _testcapi.eval_code_ex |
| 117 | + code = f.__code__ |
| 118 | + self.assertEqual(eval_code_ex(code, {}, {}, (), {}, (), {}, f.__closure__), 1) |
| 119 | + self.assertEqual(eval_code_ex(code, {}, {}, (), {}, (), {}, f.__closure__[::-1]), 2) |
| 120 | + |
| 121 | + # CRASHES eval_code_ex(code, {}, {}, (), {}, (), {}, ()), 1) |
| 122 | + # CRASHES eval_code_ex(code, {}, {}, (), {}, (), {}, NULL), 1) |
53 | 123 |
|
54 | 124 |
|
55 | 125 | if __name__ == "__main__":
|
|
0 commit comments