blob: f1fa18ac92af5a5ad085c632d22b4b7397261edc [file] [log] [blame]
Benjamin Peterson7dd85472008-08-17 17:13:26 +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
Benjamin Peterson7dd85472008-08-17 17:13:26 +00006import warnings
7
8from test import test_support
Jeremy Hylton5e7cb242001-02-02 18:24:26 +00009
Jeremy Hylton5e7cb242001-02-02 18:24:26 +000010
Benjamin Peterson7dd85472008-08-17 17:13:26 +000011TEST_CODE = """
12import sys
Jeremy Hylton42d90162003-07-15 20:24:27 +000013
Benjamin Peterson7dd85472008-08-17 17:13:26 +000014glob = 42
Jeremy Hyltonbc2a62f2005-10-20 14:27:21 +000015
Benjamin Peterson7dd85472008-08-17 17:13:26 +000016class Mine:
17 instance_var = 24
18 def a_method(p1, p2):
19 pass
20
21def spam(a, b, *var, **kw):
22 global bar
23 bar = 47
24 x = 23
25 glob
26 def internal():
27 return x
28 return internal
29
30def foo():
31 exec 'm'
32 from sys import *
33
34def namespace_test(): pass
35def namespace_test(): pass
36"""
37
38
39def find_block(block, name):
40 for ch in block.get_children():
41 if ch.get_name() == name:
42 return ch
43
Neal Norwitzcd3e2192006-01-23 07:49:36 +000044
45class SymtableTest(unittest.TestCase):
Benjamin Peterson7dd85472008-08-17 17:13:26 +000046
Brett Cannon672237d2008-09-09 00:49:16 +000047 with warnings.catch_warnings():
Benjamin Peterson7dd85472008-08-17 17:13:26 +000048 # Ignore warnings about "from blank import *"
49 warnings.simplefilter("ignore", SyntaxWarning)
50 top = symtable.symtable(TEST_CODE, "?", "exec")
51 # These correspond to scopes in TEST_CODE
52 Mine = find_block(top, "Mine")
53 a_method = find_block(Mine, "a_method")
54 spam = find_block(top, "spam")
55 internal = find_block(spam, "internal")
56 foo = find_block(top, "foo")
57
Benjamin Petersone977ad42008-08-20 01:42:01 +000058 def test_noops(self):
59 # Check methods that don't work. They should warn and return False.
60 def check(w, msg):
61 self.assertEqual(str(w.message), msg)
62 sym = self.top.lookup("glob")
Nick Coghland2e09382008-09-11 12:11:06 +000063 with test_support.check_warnings() as w:
Benjamin Petersone977ad42008-08-20 01:42:01 +000064 warnings.simplefilter("always", DeprecationWarning)
65 self.assertFalse(sym.is_vararg())
Nick Coghland2e09382008-09-11 12:11:06 +000066 check(w, "is_vararg() is obsolete and will be removed")
67 w.reset()
Benjamin Petersone977ad42008-08-20 01:42:01 +000068 self.assertFalse(sym.is_keywordarg())
Nick Coghland2e09382008-09-11 12:11:06 +000069 check(w, "is_keywordarg() is obsolete and will be removed")
70 w.reset()
Benjamin Petersone977ad42008-08-20 01:42:01 +000071 self.assertFalse(sym.is_in_tuple())
Nick Coghland2e09382008-09-11 12:11:06 +000072 check(w, "is_in_tuple() is obsolete and will be removed")
Benjamin Petersone977ad42008-08-20 01:42:01 +000073
Benjamin Peterson7dd85472008-08-17 17:13:26 +000074 def test_type(self):
75 self.assertEqual(self.top.get_type(), "module")
76 self.assertEqual(self.Mine.get_type(), "class")
77 self.assertEqual(self.a_method.get_type(), "function")
78 self.assertEqual(self.spam.get_type(), "function")
79 self.assertEqual(self.internal.get_type(), "function")
80
81 def test_optimized(self):
82 self.assertFalse(self.top.is_optimized())
83 self.assertFalse(self.top.has_exec())
84 self.assertFalse(self.top.has_import_star())
85
86 self.assertTrue(self.spam.is_optimized())
87
88 self.assertFalse(self.foo.is_optimized())
89 self.assertTrue(self.foo.has_exec())
90 self.assertTrue(self.foo.has_import_star())
91
92 def test_nested(self):
93 self.assertFalse(self.top.is_nested())
94 self.assertFalse(self.Mine.is_nested())
95 self.assertFalse(self.spam.is_nested())
96 self.assertTrue(self.internal.is_nested())
97
98 def test_children(self):
99 self.assertTrue(self.top.has_children())
100 self.assertTrue(self.Mine.has_children())
101 self.assertFalse(self.foo.has_children())
102
103 def test_lineno(self):
104 self.assertEqual(self.top.get_lineno(), 0)
105 self.assertEqual(self.spam.get_lineno(), 11)
106
107 def test_function_info(self):
108 func = self.spam
109 self.assertEqual(func.get_parameters(), ("a", "b", "kw", "var"))
110 self.assertEqual(func.get_locals(),
111 ("a", "b", "bar", "internal", "kw", "var", "x"))
112 self.assertEqual(func.get_globals(), ("bar", "glob"))
113 self.assertEqual(self.internal.get_frees(), ("x",))
114
115 def test_globals(self):
116 self.assertTrue(self.spam.lookup("glob").is_global())
117 self.assertTrue(self.spam.lookup("bar").is_global())
118 self.assertFalse(self.internal.lookup("x").is_global())
119 self.assertFalse(self.Mine.lookup("instance_var").is_global())
120
121 def test_local(self):
122 self.assertTrue(self.spam.lookup("x").is_local())
123 self.assertFalse(self.internal.lookup("x").is_local())
124
125 def test_referenced(self):
126 self.assertTrue(self.internal.lookup("x").is_referenced())
127 self.assertTrue(self.spam.lookup("internal").is_referenced())
128 self.assertFalse(self.spam.lookup("x").is_referenced())
129
130 def test_parameters(self):
131 for sym in ("a", "var", "kw"):
132 self.assertTrue(self.spam.lookup(sym).is_parameter())
133 self.assertFalse(self.spam.lookup("x").is_parameter())
134
135 def test_symbol_lookup(self):
136 self.assertEqual(len(self.top.get_identifiers()),
137 len(self.top.get_symbols()))
138
139 self.assertRaises(KeyError, self.top.lookup, "not_here")
140
141 def test_namespaces(self):
142 self.assertTrue(self.top.lookup("Mine").is_namespace())
143 self.assertTrue(self.Mine.lookup("a_method").is_namespace())
144 self.assertTrue(self.top.lookup("spam").is_namespace())
145 self.assertTrue(self.spam.lookup("internal").is_namespace())
146 self.assertTrue(self.top.lookup("namespace_test").is_namespace())
147 self.assertFalse(self.spam.lookup("x").is_namespace())
148
149 self.assert_(self.top.lookup("spam").get_namespace() is self.spam)
150 ns_test = self.top.lookup("namespace_test")
151 self.assertEqual(len(ns_test.get_namespaces()), 2)
152 self.assertRaises(ValueError, ns_test.get_namespace)
153
154 def test_assigned(self):
155 self.assertTrue(self.spam.lookup("x").is_assigned())
156 self.assertTrue(self.spam.lookup("bar").is_assigned())
157 self.assertTrue(self.top.lookup("spam").is_assigned())
158 self.assertTrue(self.Mine.lookup("a_method").is_assigned())
159 self.assertFalse(self.internal.lookup("x").is_assigned())
160
161 def test_imported(self):
162 self.assertTrue(self.top.lookup("sys").is_imported())
163
164 def test_name(self):
165 self.assertEqual(self.top.get_name(), "top")
166 self.assertEqual(self.spam.get_name(), "spam")
167 self.assertEqual(self.spam.lookup("x").get_name(), "x")
168 self.assertEqual(self.Mine.get_name(), "Mine")
169
170 def test_class_info(self):
171 self.assertEqual(self.Mine.get_methods(), ('a_method',))
172
173 def test_filename_correct(self):
174 ### Bug tickler: SyntaxError file name correct whether error raised
175 ### while parsing or building symbol table.
176 def checkfilename(brokencode):
177 try:
178 symtable.symtable(brokencode, "spam", "exec")
179 except SyntaxError as e:
180 self.assertEqual(e.filename, "spam")
181 else:
182 self.fail("no SyntaxError for %r" % (brokencode,))
183 checkfilename("def f(x): foo)(") # parse-time
184 checkfilename("def f(x): global x") # symtable-build-time
Neal Norwitzcd3e2192006-01-23 07:49:36 +0000185
186 def test_eval(self):
187 symbols = symtable.symtable("42", "?", "eval")
188
189 def test_single(self):
190 symbols = symtable.symtable("42", "?", "single")
191
192 def test_exec(self):
193 symbols = symtable.symtable("def f(x): return x", "?", "exec")
194
195
196def test_main():
197 test_support.run_unittest(SymtableTest)
198
199if __name__ == '__main__':
200 test_main()