python (3.11.7)
1 """
2 Test the API of the symtable module.
3 """
4 import symtable
5 import unittest
6
7
8
9 TEST_CODE = """
10 import sys
11
12 glob = 42
13 some_var = 12
14 some_non_assigned_global_var = 11
15 some_assigned_global_var = 11
16
17 class Mine:
18 instance_var = 24
19 def a_method(p1, p2):
20 pass
21
22 def spam(a, b, *var, **kw):
23 global bar
24 global some_assigned_global_var
25 some_assigned_global_var = 12
26 bar = 47
27 some_var = 10
28 x = 23
29 glob
30 def internal():
31 return x
32 def other_internal():
33 nonlocal some_var
34 some_var = 3
35 return some_var
36 return internal
37
38 def foo():
39 pass
40
41 def namespace_test(): pass
42 def namespace_test(): pass
43 """
44
45
46 def find_block(block, name):
47 for ch in block.get_children():
48 if ch.get_name() == name:
49 return ch
50
51
52 class ESC[4;38;5;81mSymtableTest(ESC[4;38;5;149munittestESC[4;38;5;149m.ESC[4;38;5;149mTestCase):
53
54 top = symtable.symtable(TEST_CODE, "?", "exec")
55 # These correspond to scopes in TEST_CODE
56 Mine = find_block(top, "Mine")
57 a_method = find_block(Mine, "a_method")
58 spam = find_block(top, "spam")
59 internal = find_block(spam, "internal")
60 other_internal = find_block(spam, "other_internal")
61 foo = find_block(top, "foo")
62
63 def test_type(self):
64 self.assertEqual(self.top.get_type(), "module")
65 self.assertEqual(self.Mine.get_type(), "class")
66 self.assertEqual(self.a_method.get_type(), "function")
67 self.assertEqual(self.spam.get_type(), "function")
68 self.assertEqual(self.internal.get_type(), "function")
69
70 def test_id(self):
71 self.assertGreater(self.top.get_id(), 0)
72 self.assertGreater(self.Mine.get_id(), 0)
73 self.assertGreater(self.a_method.get_id(), 0)
74 self.assertGreater(self.spam.get_id(), 0)
75 self.assertGreater(self.internal.get_id(), 0)
76
77 def test_optimized(self):
78 self.assertFalse(self.top.is_optimized())
79
80 self.assertTrue(self.spam.is_optimized())
81
82 def test_nested(self):
83 self.assertFalse(self.top.is_nested())
84 self.assertFalse(self.Mine.is_nested())
85 self.assertFalse(self.spam.is_nested())
86 self.assertTrue(self.internal.is_nested())
87
88 def test_children(self):
89 self.assertTrue(self.top.has_children())
90 self.assertTrue(self.Mine.has_children())
91 self.assertFalse(self.foo.has_children())
92
93 def test_lineno(self):
94 self.assertEqual(self.top.get_lineno(), 0)
95 self.assertEqual(self.spam.get_lineno(), 14)
96
97 def test_function_info(self):
98 func = self.spam
99 self.assertEqual(sorted(func.get_parameters()), ["a", "b", "kw", "var"])
100 expected = ['a', 'b', 'internal', 'kw', 'other_internal', 'some_var', 'var', 'x']
101 self.assertEqual(sorted(func.get_locals()), expected)
102 self.assertEqual(sorted(func.get_globals()), ["bar", "glob", "some_assigned_global_var"])
103 self.assertEqual(self.internal.get_frees(), ("x",))
104
105 def test_globals(self):
106 self.assertTrue(self.spam.lookup("glob").is_global())
107 self.assertFalse(self.spam.lookup("glob").is_declared_global())
108 self.assertTrue(self.spam.lookup("bar").is_global())
109 self.assertTrue(self.spam.lookup("bar").is_declared_global())
110 self.assertFalse(self.internal.lookup("x").is_global())
111 self.assertFalse(self.Mine.lookup("instance_var").is_global())
112 self.assertTrue(self.spam.lookup("bar").is_global())
113 # Module-scope globals are both global and local
114 self.assertTrue(self.top.lookup("some_non_assigned_global_var").is_global())
115 self.assertTrue(self.top.lookup("some_assigned_global_var").is_global())
116
117 def test_nonlocal(self):
118 self.assertFalse(self.spam.lookup("some_var").is_nonlocal())
119 self.assertTrue(self.other_internal.lookup("some_var").is_nonlocal())
120 expected = ("some_var",)
121 self.assertEqual(self.other_internal.get_nonlocals(), expected)
122
123 def test_local(self):
124 self.assertTrue(self.spam.lookup("x").is_local())
125 self.assertFalse(self.spam.lookup("bar").is_local())
126 # Module-scope globals are both global and local
127 self.assertTrue(self.top.lookup("some_non_assigned_global_var").is_local())
128 self.assertTrue(self.top.lookup("some_assigned_global_var").is_local())
129
130 def test_free(self):
131 self.assertTrue(self.internal.lookup("x").is_free())
132
133 def test_referenced(self):
134 self.assertTrue(self.internal.lookup("x").is_referenced())
135 self.assertTrue(self.spam.lookup("internal").is_referenced())
136 self.assertFalse(self.spam.lookup("x").is_referenced())
137
138 def test_parameters(self):
139 for sym in ("a", "var", "kw"):
140 self.assertTrue(self.spam.lookup(sym).is_parameter())
141 self.assertFalse(self.spam.lookup("x").is_parameter())
142
143 def test_symbol_lookup(self):
144 self.assertEqual(len(self.top.get_identifiers()),
145 len(self.top.get_symbols()))
146
147 self.assertRaises(KeyError, self.top.lookup, "not_here")
148
149 def test_namespaces(self):
150 self.assertTrue(self.top.lookup("Mine").is_namespace())
151 self.assertTrue(self.Mine.lookup("a_method").is_namespace())
152 self.assertTrue(self.top.lookup("spam").is_namespace())
153 self.assertTrue(self.spam.lookup("internal").is_namespace())
154 self.assertTrue(self.top.lookup("namespace_test").is_namespace())
155 self.assertFalse(self.spam.lookup("x").is_namespace())
156
157 self.assertTrue(self.top.lookup("spam").get_namespace() is self.spam)
158 ns_test = self.top.lookup("namespace_test")
159 self.assertEqual(len(ns_test.get_namespaces()), 2)
160 self.assertRaises(ValueError, ns_test.get_namespace)
161
162 ns_test_2 = self.top.lookup("glob")
163 self.assertEqual(len(ns_test_2.get_namespaces()), 0)
164 self.assertRaises(ValueError, ns_test_2.get_namespace)
165
166 def test_assigned(self):
167 self.assertTrue(self.spam.lookup("x").is_assigned())
168 self.assertTrue(self.spam.lookup("bar").is_assigned())
169 self.assertTrue(self.top.lookup("spam").is_assigned())
170 self.assertTrue(self.Mine.lookup("a_method").is_assigned())
171 self.assertFalse(self.internal.lookup("x").is_assigned())
172
173 def test_annotated(self):
174 st1 = symtable.symtable('def f():\n x: int\n', 'test', 'exec')
175 st2 = st1.get_children()[0]
176 self.assertTrue(st2.lookup('x').is_local())
177 self.assertTrue(st2.lookup('x').is_annotated())
178 self.assertFalse(st2.lookup('x').is_global())
179 st3 = symtable.symtable('def f():\n x = 1\n', 'test', 'exec')
180 st4 = st3.get_children()[0]
181 self.assertTrue(st4.lookup('x').is_local())
182 self.assertFalse(st4.lookup('x').is_annotated())
183
184 # Test that annotations in the global scope are valid after the
185 # variable is declared as nonlocal.
186 st5 = symtable.symtable('global x\nx: int', 'test', 'exec')
187 self.assertTrue(st5.lookup("x").is_global())
188
189 # Test that annotations for nonlocals are valid after the
190 # variable is declared as nonlocal.
191 st6 = symtable.symtable('def g():\n'
192 ' x = 2\n'
193 ' def f():\n'
194 ' nonlocal x\n'
195 ' x: int',
196 'test', 'exec')
197
198 def test_imported(self):
199 self.assertTrue(self.top.lookup("sys").is_imported())
200
201 def test_name(self):
202 self.assertEqual(self.top.get_name(), "top")
203 self.assertEqual(self.spam.get_name(), "spam")
204 self.assertEqual(self.spam.lookup("x").get_name(), "x")
205 self.assertEqual(self.Mine.get_name(), "Mine")
206
207 def test_class_info(self):
208 self.assertEqual(self.Mine.get_methods(), ('a_method',))
209
210 def test_filename_correct(self):
211 ### Bug tickler: SyntaxError file name correct whether error raised
212 ### while parsing or building symbol table.
213 def checkfilename(brokencode, offset):
214 try:
215 symtable.symtable(brokencode, "spam", "exec")
216 except SyntaxError as e:
217 self.assertEqual(e.filename, "spam")
218 self.assertEqual(e.lineno, 1)
219 self.assertEqual(e.offset, offset)
220 else:
221 self.fail("no SyntaxError for %r" % (brokencode,))
222 checkfilename("def f(x): foo)(", 14) # parse-time
223 checkfilename("def f(x): global x", 11) # symtable-build-time
224 symtable.symtable("pass", b"spam", "exec")
225 with self.assertWarns(DeprecationWarning), \
226 self.assertRaises(TypeError):
227 symtable.symtable("pass", bytearray(b"spam"), "exec")
228 with self.assertWarns(DeprecationWarning):
229 symtable.symtable("pass", memoryview(b"spam"), "exec")
230 with self.assertRaises(TypeError):
231 symtable.symtable("pass", list(b"spam"), "exec")
232
233 def test_eval(self):
234 symbols = symtable.symtable("42", "?", "eval")
235
236 def test_single(self):
237 symbols = symtable.symtable("42", "?", "single")
238
239 def test_exec(self):
240 symbols = symtable.symtable("def f(x): return x", "?", "exec")
241
242 def test_bytes(self):
243 top = symtable.symtable(TEST_CODE.encode('utf8'), "?", "exec")
244 self.assertIsNotNone(find_block(top, "Mine"))
245
246 code = b'# -*- coding: iso8859-15 -*-\nclass \xb4: pass\n'
247
248 top = symtable.symtable(code, "?", "exec")
249 self.assertIsNotNone(find_block(top, "\u017d"))
250
251 def test_symtable_repr(self):
252 self.assertEqual(str(self.top), "<SymbolTable for module ?>")
253 self.assertEqual(str(self.spam), "<Function SymbolTable for spam in ?>")
254
255 def test_symtable_entry_repr(self):
256 expected = f"<symtable entry top({self.top.get_id()}), line {self.top.get_lineno()}>"
257 self.assertEqual(repr(self.top._table), expected)
258
259
260 if __name__ == '__main__':
261 unittest.main()