blob: 6585071912a0d0bca4ddb14f6af4b1c839eb97d1 [file] [log] [blame]
Benjamin Peterson55e00f22008-08-17 18:02:44 +00001"""
2Test the API of the symtable module.
3"""
Neal Norwitz89886ab2005-11-25 03:15:49 +00004import symtable
Neal Norwitzcd3e2192006-01-23 07:49:36 +00005import unittest
Jeremy Hylton5e7cb242001-02-02 18:24:26 +00006
Jeremy Hylton5e7cb242001-02-02 18:24:26 +00007
Jeremy Hylton42d90162003-07-15 20:24:27 +00008
Benjamin Peterson55e00f22008-08-17 18:02:44 +00009TEST_CODE = """
10import sys
Jeremy Hyltonbc2a62f2005-10-20 14:27:21 +000011
Benjamin Peterson55e00f22008-08-17 18:02:44 +000012glob = 42
Pablo Galindod5b4f1b2018-10-20 01:46:00 +010013some_var = 12
Pablo Galindo7b7aa942020-10-03 21:23:03 +010014some_non_assigned_global_var = 11
15some_assigned_global_var = 11
Benjamin Peterson55e00f22008-08-17 18:02:44 +000016
17class Mine:
18 instance_var = 24
19 def a_method(p1, p2):
20 pass
21
22def spam(a, b, *var, **kw):
23 global bar
Pablo Galindo7b7aa942020-10-03 21:23:03 +010024 global some_assigned_global_var
25 some_assigned_global_var = 12
Benjamin Peterson55e00f22008-08-17 18:02:44 +000026 bar = 47
Pablo Galindod5b4f1b2018-10-20 01:46:00 +010027 some_var = 10
Benjamin Peterson55e00f22008-08-17 18:02:44 +000028 x = 23
29 glob
30 def internal():
31 return x
Pablo Galindod5b4f1b2018-10-20 01:46:00 +010032 def other_internal():
33 nonlocal some_var
34 some_var = 3
35 return some_var
Benjamin Peterson55e00f22008-08-17 18:02:44 +000036 return internal
37
38def foo():
39 pass
40
41def namespace_test(): pass
42def namespace_test(): pass
43"""
44
45
46def find_block(block, name):
47 for ch in block.get_children():
48 if ch.get_name() == name:
49 return ch
50
Neal Norwitzcd3e2192006-01-23 07:49:36 +000051
52class SymtableTest(unittest.TestCase):
Benjamin Peterson55e00f22008-08-17 18:02:44 +000053
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")
Pablo Galindod5b4f1b2018-10-20 01:46:00 +010060 other_internal = find_block(spam, "other_internal")
Benjamin Peterson55e00f22008-08-17 18:02:44 +000061 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_optimized(self):
71 self.assertFalse(self.top.is_optimized())
Benjamin Peterson55e00f22008-08-17 18:02:44 +000072
73 self.assertTrue(self.spam.is_optimized())
74
75 def test_nested(self):
76 self.assertFalse(self.top.is_nested())
77 self.assertFalse(self.Mine.is_nested())
78 self.assertFalse(self.spam.is_nested())
79 self.assertTrue(self.internal.is_nested())
80
81 def test_children(self):
82 self.assertTrue(self.top.has_children())
83 self.assertTrue(self.Mine.has_children())
84 self.assertFalse(self.foo.has_children())
85
86 def test_lineno(self):
87 self.assertEqual(self.top.get_lineno(), 0)
Pablo Galindo7b7aa942020-10-03 21:23:03 +010088 self.assertEqual(self.spam.get_lineno(), 14)
Benjamin Peterson55e00f22008-08-17 18:02:44 +000089
90 def test_function_info(self):
91 func = self.spam
Benjamin Petersonb462da82012-01-03 16:23:11 -060092 self.assertEqual(sorted(func.get_parameters()), ["a", "b", "kw", "var"])
Pablo Galindod5b4f1b2018-10-20 01:46:00 +010093 expected = ['a', 'b', 'internal', 'kw', 'other_internal', 'some_var', 'var', 'x']
Benjamin Petersonb462da82012-01-03 16:23:11 -060094 self.assertEqual(sorted(func.get_locals()), expected)
Pablo Galindo7b7aa942020-10-03 21:23:03 +010095 self.assertEqual(sorted(func.get_globals()), ["bar", "glob", "some_assigned_global_var"])
Benjamin Petersonb71caf12008-08-20 12:55:31 +000096 self.assertEqual(self.internal.get_frees(), ("x",))
Benjamin Peterson55e00f22008-08-17 18:02:44 +000097
98 def test_globals(self):
99 self.assertTrue(self.spam.lookup("glob").is_global())
Jeremy Hyltonf37708e2009-03-31 15:26:37 +0000100 self.assertFalse(self.spam.lookup("glob").is_declared_global())
Benjamin Peterson55e00f22008-08-17 18:02:44 +0000101 self.assertTrue(self.spam.lookup("bar").is_global())
Jeremy Hyltonf37708e2009-03-31 15:26:37 +0000102 self.assertTrue(self.spam.lookup("bar").is_declared_global())
Benjamin Peterson55e00f22008-08-17 18:02:44 +0000103 self.assertFalse(self.internal.lookup("x").is_global())
104 self.assertFalse(self.Mine.lookup("instance_var").is_global())
Pablo Galindo799d7d62020-04-06 17:05:57 +0100105 self.assertTrue(self.spam.lookup("bar").is_global())
Pablo Galindo7b7aa942020-10-03 21:23:03 +0100106 # Module-scope globals are both global and local
107 self.assertTrue(self.top.lookup("some_non_assigned_global_var").is_global())
108 self.assertTrue(self.top.lookup("some_assigned_global_var").is_global())
Benjamin Peterson55e00f22008-08-17 18:02:44 +0000109
Pablo Galindod5b4f1b2018-10-20 01:46:00 +0100110 def test_nonlocal(self):
111 self.assertFalse(self.spam.lookup("some_var").is_nonlocal())
112 self.assertTrue(self.other_internal.lookup("some_var").is_nonlocal())
113 expected = ("some_var",)
114 self.assertEqual(self.other_internal.get_nonlocals(), expected)
115
Benjamin Peterson55e00f22008-08-17 18:02:44 +0000116 def test_local(self):
117 self.assertTrue(self.spam.lookup("x").is_local())
Pablo Galindo799d7d62020-04-06 17:05:57 +0100118 self.assertFalse(self.spam.lookup("bar").is_local())
Pablo Galindo7b7aa942020-10-03 21:23:03 +0100119 # Module-scope globals are both global and local
120 self.assertTrue(self.top.lookup("some_non_assigned_global_var").is_local())
121 self.assertTrue(self.top.lookup("some_assigned_global_var").is_local())
Pablo Galindo799d7d62020-04-06 17:05:57 +0100122
123 def test_free(self):
124 self.assertTrue(self.internal.lookup("x").is_free())
Benjamin Peterson55e00f22008-08-17 18:02:44 +0000125
126 def test_referenced(self):
127 self.assertTrue(self.internal.lookup("x").is_referenced())
128 self.assertTrue(self.spam.lookup("internal").is_referenced())
129 self.assertFalse(self.spam.lookup("x").is_referenced())
130
131 def test_parameters(self):
132 for sym in ("a", "var", "kw"):
133 self.assertTrue(self.spam.lookup(sym).is_parameter())
134 self.assertFalse(self.spam.lookup("x").is_parameter())
135
136 def test_symbol_lookup(self):
137 self.assertEqual(len(self.top.get_identifiers()),
138 len(self.top.get_symbols()))
139
140 self.assertRaises(KeyError, self.top.lookup, "not_here")
141
142 def test_namespaces(self):
143 self.assertTrue(self.top.lookup("Mine").is_namespace())
144 self.assertTrue(self.Mine.lookup("a_method").is_namespace())
145 self.assertTrue(self.top.lookup("spam").is_namespace())
146 self.assertTrue(self.spam.lookup("internal").is_namespace())
147 self.assertTrue(self.top.lookup("namespace_test").is_namespace())
148 self.assertFalse(self.spam.lookup("x").is_namespace())
149
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000150 self.assertTrue(self.top.lookup("spam").get_namespace() is self.spam)
Benjamin Peterson55e00f22008-08-17 18:02:44 +0000151 ns_test = self.top.lookup("namespace_test")
152 self.assertEqual(len(ns_test.get_namespaces()), 2)
153 self.assertRaises(ValueError, ns_test.get_namespace)
154
155 def test_assigned(self):
156 self.assertTrue(self.spam.lookup("x").is_assigned())
157 self.assertTrue(self.spam.lookup("bar").is_assigned())
158 self.assertTrue(self.top.lookup("spam").is_assigned())
159 self.assertTrue(self.Mine.lookup("a_method").is_assigned())
160 self.assertFalse(self.internal.lookup("x").is_assigned())
161
Yury Selivanovf8cb8a12016-09-08 20:50:03 -0700162 def test_annotated(self):
163 st1 = symtable.symtable('def f():\n x: int\n', 'test', 'exec')
164 st2 = st1.get_children()[0]
165 self.assertTrue(st2.lookup('x').is_local())
166 self.assertTrue(st2.lookup('x').is_annotated())
167 self.assertFalse(st2.lookup('x').is_global())
168 st3 = symtable.symtable('def f():\n x = 1\n', 'test', 'exec')
169 st4 = st3.get_children()[0]
170 self.assertTrue(st4.lookup('x').is_local())
171 self.assertFalse(st4.lookup('x').is_annotated())
172
Pablo Galindode2aea02018-10-14 18:01:03 +0100173 # Test that annotations in the global scope are valid after the
174 # variable is declared as nonlocal.
175 st5 = symtable.symtable('global x\nx: int', 'test', 'exec')
176 self.assertTrue(st5.lookup("x").is_global())
177
178 # Test that annotations for nonlocals are valid after the
179 # variable is declared as nonlocal.
180 st6 = symtable.symtable('def g():\n'
181 ' x = 2\n'
182 ' def f():\n'
183 ' nonlocal x\n'
184 ' x: int',
185 'test', 'exec')
186
Benjamin Peterson55e00f22008-08-17 18:02:44 +0000187 def test_imported(self):
188 self.assertTrue(self.top.lookup("sys").is_imported())
189
190 def test_name(self):
191 self.assertEqual(self.top.get_name(), "top")
192 self.assertEqual(self.spam.get_name(), "spam")
193 self.assertEqual(self.spam.lookup("x").get_name(), "x")
194 self.assertEqual(self.Mine.get_name(), "Mine")
195
196 def test_class_info(self):
Benjamin Petersonb71caf12008-08-20 12:55:31 +0000197 self.assertEqual(self.Mine.get_methods(), ('a_method',))
Benjamin Peterson55e00f22008-08-17 18:02:44 +0000198
199 def test_filename_correct(self):
200 ### Bug tickler: SyntaxError file name correct whether error raised
201 ### while parsing or building symbol table.
Serhiy Storchaka8b583392016-12-11 14:39:01 +0200202 def checkfilename(brokencode, offset):
Benjamin Peterson55e00f22008-08-17 18:02:44 +0000203 try:
204 symtable.symtable(brokencode, "spam", "exec")
205 except SyntaxError as e:
206 self.assertEqual(e.filename, "spam")
Serhiy Storchaka8b583392016-12-11 14:39:01 +0200207 self.assertEqual(e.lineno, 1)
208 self.assertEqual(e.offset, offset)
Benjamin Peterson55e00f22008-08-17 18:02:44 +0000209 else:
210 self.fail("no SyntaxError for %r" % (brokencode,))
Serhiy Storchaka8b583392016-12-11 14:39:01 +0200211 checkfilename("def f(x): foo)(", 14) # parse-time
Ammar Askar025eb982018-09-24 17:12:49 -0400212 checkfilename("def f(x): global x", 11) # symtable-build-time
Serhiy Storchaka9305d832016-06-18 13:53:36 +0300213 symtable.symtable("pass", b"spam", "exec")
Serhiy Storchakafebc3322016-08-06 23:29:29 +0300214 with self.assertWarns(DeprecationWarning), \
215 self.assertRaises(TypeError):
Serhiy Storchaka9305d832016-06-18 13:53:36 +0300216 symtable.symtable("pass", bytearray(b"spam"), "exec")
Serhiy Storchakafebc3322016-08-06 23:29:29 +0300217 with self.assertWarns(DeprecationWarning):
218 symtable.symtable("pass", memoryview(b"spam"), "exec")
Serhiy Storchaka9305d832016-06-18 13:53:36 +0300219 with self.assertRaises(TypeError):
220 symtable.symtable("pass", list(b"spam"), "exec")
Neal Norwitzcd3e2192006-01-23 07:49:36 +0000221
222 def test_eval(self):
223 symbols = symtable.symtable("42", "?", "eval")
224
225 def test_single(self):
226 symbols = symtable.symtable("42", "?", "single")
227
228 def test_exec(self):
229 symbols = symtable.symtable("def f(x): return x", "?", "exec")
230
Dino Viehland41540692019-05-28 16:21:17 -0700231 def test_bytes(self):
232 top = symtable.symtable(TEST_CODE.encode('utf8'), "?", "exec")
233 self.assertIsNotNone(find_block(top, "Mine"))
234
235 code = b'# -*- coding: iso8859-15 -*-\nclass \xb4: pass\n'
236
237 top = symtable.symtable(code, "?", "exec")
238 self.assertIsNotNone(find_block(top, "\u017d"))
239
Pablo Galindo7b7aa942020-10-03 21:23:03 +0100240 def test_symtable_repr(self):
241 self.assertEqual(str(self.top), "<SymbolTable for module ?>")
242 self.assertEqual(str(self.spam), "<Function SymbolTable for spam in ?>")
243
Neal Norwitzcd3e2192006-01-23 07:49:36 +0000244
Neal Norwitzcd3e2192006-01-23 07:49:36 +0000245if __name__ == '__main__':
Zachary Ware38c707e2015-04-13 15:00:43 -0500246 unittest.main()