blob: 6ff53b3be2c6386f40971114ce474ce4b6bca5c2 [file] [log] [blame]
Georg Brandlc6fdec62006-10-28 13:10:17 +00001import unittest
2from test.test_support import check_syntax_error, run_unittest
Jeremy Hylton4588c782001-01-25 20:11:23 +00003
Jeremy Hyltoncd738362001-08-07 16:38:19 +00004import warnings
Georg Brandlc6fdec62006-10-28 13:10:17 +00005warnings.filterwarnings("ignore", r"import \*", SyntaxWarning, "<test string>")
Guido van Rossum796e1e02001-12-15 18:04:10 +00006warnings.filterwarnings("ignore", r"import \*", SyntaxWarning, "<string>")
Jeremy Hyltoncd738362001-08-07 16:38:19 +00007
Georg Brandlc6fdec62006-10-28 13:10:17 +00008class ScopeTests(unittest.TestCase):
Jeremy Hylton4588c782001-01-25 20:11:23 +00009
Georg Brandlc6fdec62006-10-28 13:10:17 +000010 def testSimpleNesting(self):
Tim Petersabd8a332006-11-03 02:32:46 +000011
Georg Brandlc6fdec62006-10-28 13:10:17 +000012 def make_adder(x):
Jeremy Hylton4588c782001-01-25 20:11:23 +000013 def adder(y):
Georg Brandlc6fdec62006-10-28 13:10:17 +000014 return x + y
Jeremy Hylton4588c782001-01-25 20:11:23 +000015 return adder
Jeremy Hylton4588c782001-01-25 20:11:23 +000016
Georg Brandlc6fdec62006-10-28 13:10:17 +000017 inc = make_adder(1)
18 plus10 = make_adder(10)
Jeremy Hylton4588c782001-01-25 20:11:23 +000019
Georg Brandlc6fdec62006-10-28 13:10:17 +000020 self.assertEqual(inc(1), 2)
21 self.assertEqual(plus10(-2), 8)
Jeremy Hylton4588c782001-01-25 20:11:23 +000022
Georg Brandlc6fdec62006-10-28 13:10:17 +000023 def testExtraNesting(self):
Jeremy Hylton4588c782001-01-25 20:11:23 +000024
Georg Brandlc6fdec62006-10-28 13:10:17 +000025 def make_adder2(x):
26 def extra(): # check freevars passing through non-use scopes
27 def adder(y):
28 return x + y
29 return adder
30 return extra()
Jeremy Hylton4588c782001-01-25 20:11:23 +000031
Georg Brandlc6fdec62006-10-28 13:10:17 +000032 inc = make_adder2(1)
33 plus10 = make_adder2(10)
Jeremy Hylton4588c782001-01-25 20:11:23 +000034
Georg Brandlc6fdec62006-10-28 13:10:17 +000035 self.assertEqual(inc(1), 2)
36 self.assertEqual(plus10(-2), 8)
Jeremy Hylton4588c782001-01-25 20:11:23 +000037
Georg Brandlc6fdec62006-10-28 13:10:17 +000038 def testSimpleAndRebinding(self):
Jeremy Hylton4588c782001-01-25 20:11:23 +000039
Georg Brandlc6fdec62006-10-28 13:10:17 +000040 def make_adder3(x):
41 def adder(y):
42 return x + y
43 x = x + 1 # check tracking of assignment to x in defining scope
44 return adder
Jeremy Hylton4588c782001-01-25 20:11:23 +000045
Georg Brandlc6fdec62006-10-28 13:10:17 +000046 inc = make_adder3(0)
47 plus10 = make_adder3(9)
Jeremy Hylton4588c782001-01-25 20:11:23 +000048
Georg Brandlc6fdec62006-10-28 13:10:17 +000049 self.assertEqual(inc(1), 2)
50 self.assertEqual(plus10(-2), 8)
Jeremy Hylton4588c782001-01-25 20:11:23 +000051
Georg Brandlc6fdec62006-10-28 13:10:17 +000052 def testNestingGlobalNoFree(self):
Jeremy Hylton4588c782001-01-25 20:11:23 +000053
Georg Brandlc6fdec62006-10-28 13:10:17 +000054 def make_adder4(): # XXX add exta level of indirection
55 def nest():
56 def nest():
57 def adder(y):
58 return global_x + y # check that plain old globals work
59 return adder
60 return nest()
61 return nest()
Jeremy Hylton4588c782001-01-25 20:11:23 +000062
Georg Brandlc6fdec62006-10-28 13:10:17 +000063 global_x = 1
64 adder = make_adder4()
65 self.assertEqual(adder(1), 2)
Jeremy Hylton4588c782001-01-25 20:11:23 +000066
Georg Brandlc6fdec62006-10-28 13:10:17 +000067 global_x = 10
68 self.assertEqual(adder(-2), 8)
Jeremy Hylton4588c782001-01-25 20:11:23 +000069
Georg Brandlc6fdec62006-10-28 13:10:17 +000070 def testNestingThroughClass(self):
Jeremy Hylton4588c782001-01-25 20:11:23 +000071
Georg Brandlc6fdec62006-10-28 13:10:17 +000072 def make_adder5(x):
73 class Adder:
74 def __call__(self, y):
75 return x + y
76 return Adder()
Jeremy Hylton4588c782001-01-25 20:11:23 +000077
Georg Brandlc6fdec62006-10-28 13:10:17 +000078 inc = make_adder5(1)
79 plus10 = make_adder5(10)
Jeremy Hylton4588c782001-01-25 20:11:23 +000080
Georg Brandlc6fdec62006-10-28 13:10:17 +000081 self.assertEqual(inc(1), 2)
82 self.assertEqual(plus10(-2), 8)
Jeremy Hylton4588c782001-01-25 20:11:23 +000083
Georg Brandlc6fdec62006-10-28 13:10:17 +000084 def testNestingPlusFreeRefToGlobal(self):
Jeremy Hylton4588c782001-01-25 20:11:23 +000085
Georg Brandlc6fdec62006-10-28 13:10:17 +000086 def make_adder6(x):
87 global global_nest_x
88 def adder(y):
89 return global_nest_x + y
90 global_nest_x = x
91 return adder
Jeremy Hylton4588c782001-01-25 20:11:23 +000092
Georg Brandlc6fdec62006-10-28 13:10:17 +000093 inc = make_adder6(1)
94 plus10 = make_adder6(10)
Jeremy Hylton4588c782001-01-25 20:11:23 +000095
Georg Brandlc6fdec62006-10-28 13:10:17 +000096 self.assertEqual(inc(1), 11) # there's only one global
97 self.assertEqual(plus10(-2), 8)
Jeremy Hylton4588c782001-01-25 20:11:23 +000098
Georg Brandlc6fdec62006-10-28 13:10:17 +000099 def testNearestEnclosingScope(self):
Jeremy Hylton4588c782001-01-25 20:11:23 +0000100
Georg Brandlc6fdec62006-10-28 13:10:17 +0000101 def f(x):
102 def g(y):
103 x = 42 # check that this masks binding in f()
104 def h(z):
105 return x + z
106 return h
107 return g(2)
Jeremy Hylton4588c782001-01-25 20:11:23 +0000108
Georg Brandlc6fdec62006-10-28 13:10:17 +0000109 test_func = f(10)
110 self.assertEqual(test_func(5), 47)
111
112 def testMixedFreevarsAndCellvars(self):
113
114 def identity(x):
115 return x
116
117 def f(x, y, z):
118 def g(a, b, c):
119 a = a + x # 3
120 def h():
121 # z * (4 + 9)
122 # 3 * 13
123 return identity(z * (b + y))
124 y = c + z # 9
125 return h
126 return g
127
128 g = f(1, 2, 3)
129 h = g(2, 4, 6)
130 self.assertEqual(h(), 39)
131
132 def testFreeVarInMethod(self):
133
134 def test():
135 method_and_var = "var"
136 class Test:
137 def method_and_var(self):
138 return "method"
139 def test(self):
140 return method_and_var
141 def actual_global(self):
142 return str("global")
143 def str(self):
144 return str(self)
145 return Test()
146
147 t = test()
148 self.assertEqual(t.test(), "var")
149 self.assertEqual(t.method_and_var(), "method")
150 self.assertEqual(t.actual_global(), "global")
151
152 method_and_var = "var"
153 class Test:
154 # this class is not nested, so the rules are different
155 def method_and_var(self):
156 return "method"
157 def test(self):
158 return method_and_var
159 def actual_global(self):
160 return str("global")
161 def str(self):
162 return str(self)
163
164 t = Test()
165 self.assertEqual(t.test(), "var")
166 self.assertEqual(t.method_and_var(), "method")
167 self.assertEqual(t.actual_global(), "global")
168
169 def testRecursion(self):
170
171 def f(x):
172 def fact(n):
173 if n == 0:
174 return 1
175 else:
176 return n * fact(n - 1)
177 if x >= 0:
178 return fact(x)
179 else:
180 raise ValueError, "x must be >= 0"
181
182 self.assertEqual(f(6), 720)
Jeremy Hylton4588c782001-01-25 20:11:23 +0000183
184
Georg Brandlc6fdec62006-10-28 13:10:17 +0000185 def testUnoptimizedNamespaces(self):
Jeremy Hylton4588c782001-01-25 20:11:23 +0000186
Georg Brandlc6fdec62006-10-28 13:10:17 +0000187 check_syntax_error(self, """\
Jeremy Hylton5941d192001-02-27 20:23:58 +0000188def unoptimized_clash1(strip):
Jeremy Hylton4588c782001-01-25 20:11:23 +0000189 def f(s):
190 from string import *
191 return strip(s) # ambiguity: free or local
192 return f
Jeremy Hylton97a01672001-02-09 22:56:46 +0000193""")
Jeremy Hylton4588c782001-01-25 20:11:23 +0000194
Georg Brandlc6fdec62006-10-28 13:10:17 +0000195 check_syntax_error(self, """\
Jeremy Hylton5941d192001-02-27 20:23:58 +0000196def unoptimized_clash2():
Jeremy Hylton4588c782001-01-25 20:11:23 +0000197 from string import *
198 def f(s):
199 return strip(s) # ambiguity: global or local
200 return f
Jeremy Hylton97a01672001-02-09 22:56:46 +0000201""")
Jeremy Hylton4588c782001-01-25 20:11:23 +0000202
Georg Brandlc6fdec62006-10-28 13:10:17 +0000203 check_syntax_error(self, """\
Jeremy Hylton5941d192001-02-27 20:23:58 +0000204def unoptimized_clash2():
Jeremy Hylton97a01672001-02-09 22:56:46 +0000205 from string import *
206 def g():
207 def f(s):
208 return strip(s) # ambiguity: global or local
209 return f
210""")
211
Georg Brandlc6fdec62006-10-28 13:10:17 +0000212 # XXX could allow this for exec with const argument, but what's the point
213 check_syntax_error(self, """\
Jeremy Hylton5941d192001-02-27 20:23:58 +0000214def error(y):
Jeremy Hylton4588c782001-01-25 20:11:23 +0000215 exec "a = 1"
216 def f(x):
217 return x + y
218 return f
Jeremy Hylton97a01672001-02-09 22:56:46 +0000219""")
Jeremy Hylton4588c782001-01-25 20:11:23 +0000220
Georg Brandlc6fdec62006-10-28 13:10:17 +0000221 check_syntax_error(self, """\
Jeremy Hylton5941d192001-02-27 20:23:58 +0000222def f(x):
Jeremy Hylton4588c782001-01-25 20:11:23 +0000223 def g():
224 return x
Jeremy Hylton97a01672001-02-09 22:56:46 +0000225 del x # can't del name
226""")
227
Georg Brandlc6fdec62006-10-28 13:10:17 +0000228 check_syntax_error(self, """\
Jeremy Hylton5941d192001-02-27 20:23:58 +0000229def f():
Jeremy Hylton97a01672001-02-09 22:56:46 +0000230 def g():
Georg Brandlc6fdec62006-10-28 13:10:17 +0000231 from string import *
232 return strip # global or local?
Tim Peters0e6d2132001-02-15 23:56:39 +0000233""")
Jeremy Hylton97a01672001-02-09 22:56:46 +0000234
Georg Brandlc6fdec62006-10-28 13:10:17 +0000235 # and verify a few cases that should work
Jeremy Hylton97a01672001-02-09 22:56:46 +0000236
Georg Brandlc6fdec62006-10-28 13:10:17 +0000237 exec """
Jeremy Hylton97a01672001-02-09 22:56:46 +0000238def noproblem1():
239 from string import *
240 f = lambda x:x
241
242def noproblem2():
243 from string import *
244 def f(x):
245 return x + 1
246
247def noproblem3():
248 from string import *
249 def f(x):
250 global y
251 y = x
Jeremy Hyltoncd738362001-08-07 16:38:19 +0000252"""
Jeremy Hylton4588c782001-01-25 20:11:23 +0000253
Georg Brandlc6fdec62006-10-28 13:10:17 +0000254 def testLambdas(self):
Jeremy Hylton4588c782001-01-25 20:11:23 +0000255
Georg Brandlc6fdec62006-10-28 13:10:17 +0000256 f1 = lambda x: lambda y: x + y
257 inc = f1(1)
258 plus10 = f1(10)
259 self.assertEqual(inc(1), 2)
260 self.assertEqual(plus10(5), 15)
Jeremy Hylton4588c782001-01-25 20:11:23 +0000261
Georg Brandlc6fdec62006-10-28 13:10:17 +0000262 f2 = lambda x: (lambda : lambda y: x + y)()
263 inc = f2(1)
264 plus10 = f2(10)
265 self.assertEqual(inc(1), 2)
266 self.assertEqual(plus10(5), 15)
Jeremy Hylton4588c782001-01-25 20:11:23 +0000267
Georg Brandlc6fdec62006-10-28 13:10:17 +0000268 f3 = lambda x: lambda y: global_x + y
269 global_x = 1
270 inc = f3(None)
271 self.assertEqual(inc(2), 3)
Jeremy Hylton4588c782001-01-25 20:11:23 +0000272
Georg Brandlc6fdec62006-10-28 13:10:17 +0000273 f8 = lambda x, y, z: lambda a, b, c: lambda : z * (b + y)
274 g = f8(1, 2, 3)
275 h = g(2, 4, 6)
276 self.assertEqual(h(), 18)
Jeremy Hyltonde602482001-02-05 17:35:20 +0000277
Georg Brandlc6fdec62006-10-28 13:10:17 +0000278 def testUnboundLocal(self):
Jeremy Hyltonde602482001-02-05 17:35:20 +0000279
Georg Brandlc6fdec62006-10-28 13:10:17 +0000280 def errorInOuter():
281 print y
282 def inner():
283 return y
284 y = 1
Jeremy Hyltonde602482001-02-05 17:35:20 +0000285
Georg Brandlc6fdec62006-10-28 13:10:17 +0000286 def errorInInner():
287 def inner():
288 return y
289 inner()
290 y = 1
Jeremy Hyltonde602482001-02-05 17:35:20 +0000291
Benjamin Peterson50a22522009-07-02 22:56:16 +0000292 self.assertRaises(UnboundLocalError, errorInOuter)
293 self.assertRaises(NameError, errorInInner)
Jeremy Hylton97a01672001-02-09 22:56:46 +0000294
Georg Brandlc6fdec62006-10-28 13:10:17 +0000295 # test for bug #1501934: incorrect LOAD/STORE_GLOBAL generation
296 exec """
Neil Schemenauer0e07b602006-07-09 16:16:34 +0000297global_x = 1
298def f():
299 global_x += 1
300try:
301 f()
302except UnboundLocalError:
303 pass
304else:
Georg Brandlc6fdec62006-10-28 13:10:17 +0000305 fail('scope of global_x not correctly determined')
306""" in {'fail': self.fail}
Neil Schemenauer0e07b602006-07-09 16:16:34 +0000307
Georg Brandlc6fdec62006-10-28 13:10:17 +0000308 def testComplexDefinitions(self):
Jeremy Hylton97a01672001-02-09 22:56:46 +0000309
Georg Brandlc6fdec62006-10-28 13:10:17 +0000310 def makeReturner(*lst):
311 def returner():
312 return lst
313 return returner
Tim Peters0e6d2132001-02-15 23:56:39 +0000314
Georg Brandlc6fdec62006-10-28 13:10:17 +0000315 self.assertEqual(makeReturner(1,2,3)(), (1,2,3))
Tim Peters0e6d2132001-02-15 23:56:39 +0000316
Georg Brandlc6fdec62006-10-28 13:10:17 +0000317 def makeReturner2(**kwargs):
318 def returner():
319 return kwargs
320 return returner
Jeremy Hylton97a01672001-02-09 22:56:46 +0000321
Georg Brandlc6fdec62006-10-28 13:10:17 +0000322 self.assertEqual(makeReturner2(a=11)()['a'], 11)
Jeremy Hylton97a01672001-02-09 22:56:46 +0000323
Senthil Kumaran3ddc4352010-01-08 18:41:40 +0000324 with warnings.catch_warnings():
325 # Silence Py3k warning
326 warnings.filterwarnings("ignore", "tuple parameter unpacking "
327 "has been removed", SyntaxWarning)
328 exec """\
329def makeAddPair((a, b)):
330 def addPair((c, d)):
331 return (a + c, b + d)
332 return addPair
333""" in locals()
Jeremy Hylton97a01672001-02-09 22:56:46 +0000334
Georg Brandlc6fdec62006-10-28 13:10:17 +0000335 self.assertEqual(makeAddPair((1, 2))((100, 200)), (101,202))
Guido van Rossum9aa643c2001-03-01 20:35:45 +0000336
Georg Brandlc6fdec62006-10-28 13:10:17 +0000337 def testScopeOfGlobalStmt(self):
Guido van Rossum9aa643c2001-03-01 20:35:45 +0000338# Examples posted by Samuele Pedroni to python-dev on 3/1/2001
339
Georg Brandlc6fdec62006-10-28 13:10:17 +0000340 exec """\
Guido van Rossum9aa643c2001-03-01 20:35:45 +0000341# I
342x = 7
343def f():
344 x = 1
345 def g():
346 global x
347 def i():
348 def h():
349 return x
350 return h()
351 return i()
352 return g()
Georg Brandlc6fdec62006-10-28 13:10:17 +0000353self.assertEqual(f(), 7)
354self.assertEqual(x, 7)
Guido van Rossum9aa643c2001-03-01 20:35:45 +0000355
356# II
357x = 7
358def f():
359 x = 1
360 def g():
361 x = 2
362 def i():
363 def h():
364 return x
365 return h()
366 return i()
367 return g()
Georg Brandlc6fdec62006-10-28 13:10:17 +0000368self.assertEqual(f(), 2)
369self.assertEqual(x, 7)
Guido van Rossum9aa643c2001-03-01 20:35:45 +0000370
371# III
372x = 7
373def f():
374 x = 1
375 def g():
376 global x
377 x = 2
378 def i():
379 def h():
380 return x
381 return h()
382 return i()
383 return g()
Georg Brandlc6fdec62006-10-28 13:10:17 +0000384self.assertEqual(f(), 2)
385self.assertEqual(x, 2)
Guido van Rossum9aa643c2001-03-01 20:35:45 +0000386
387# IV
388x = 7
389def f():
390 x = 3
391 def g():
392 global x
393 x = 2
394 def i():
395 def h():
396 return x
397 return h()
398 return i()
399 return g()
Georg Brandlc6fdec62006-10-28 13:10:17 +0000400self.assertEqual(f(), 2)
401self.assertEqual(x, 2)
Jeremy Hylton3e0055f2005-10-20 19:59:25 +0000402
403# XXX what about global statements in class blocks?
404# do they affect methods?
405
406x = 12
407class Global:
408 global x
409 x = 13
410 def set(self, val):
411 x = val
412 def get(self):
413 return x
414
415g = Global()
Georg Brandlc6fdec62006-10-28 13:10:17 +0000416self.assertEqual(g.get(), 13)
Jeremy Hylton3e0055f2005-10-20 19:59:25 +0000417g.set(15)
Georg Brandlc6fdec62006-10-28 13:10:17 +0000418self.assertEqual(g.get(), 13)
419"""
Jeremy Hylton5b44a672001-03-13 02:01:12 +0000420
Georg Brandlc6fdec62006-10-28 13:10:17 +0000421 def testLeaks(self):
Jeremy Hylton5b44a672001-03-13 02:01:12 +0000422
Georg Brandlc6fdec62006-10-28 13:10:17 +0000423 class Foo:
424 count = 0
Tim Peters30edd232001-03-16 08:29:48 +0000425
Georg Brandlc6fdec62006-10-28 13:10:17 +0000426 def __init__(self):
427 Foo.count += 1
Jeremy Hylton5b44a672001-03-13 02:01:12 +0000428
Georg Brandlc6fdec62006-10-28 13:10:17 +0000429 def __del__(self):
430 Foo.count -= 1
Jeremy Hylton5b44a672001-03-13 02:01:12 +0000431
Georg Brandlc6fdec62006-10-28 13:10:17 +0000432 def f1():
433 x = Foo()
434 def f2():
435 return x
436 f2()
Tim Peters30edd232001-03-16 08:29:48 +0000437
Georg Brandlc6fdec62006-10-28 13:10:17 +0000438 for i in range(100):
439 f1()
Jeremy Hylton5b44a672001-03-13 02:01:12 +0000440
Georg Brandlc6fdec62006-10-28 13:10:17 +0000441 self.assertEqual(Foo.count, 0)
Jeremy Hylton5c7a2512001-03-21 16:44:39 +0000442
Georg Brandlc6fdec62006-10-28 13:10:17 +0000443 def testClassAndGlobal(self):
Jeremy Hylton5c7a2512001-03-21 16:44:39 +0000444
Georg Brandlc6fdec62006-10-28 13:10:17 +0000445 exec """\
Jeremy Hylton5c7a2512001-03-21 16:44:39 +0000446def test(x):
447 class Foo:
448 global x
449 def __call__(self, y):
450 return x + y
451 return Foo()
452
453x = 0
Georg Brandlc6fdec62006-10-28 13:10:17 +0000454self.assertEqual(test(6)(2), 8)
Jeremy Hylton5c7a2512001-03-21 16:44:39 +0000455x = -1
Georg Brandlc6fdec62006-10-28 13:10:17 +0000456self.assertEqual(test(3)(2), 5)
Jeremy Hylton5c7a2512001-03-21 16:44:39 +0000457
Neil Schemenauerd403c452005-10-23 04:24:49 +0000458looked_up_by_load_name = False
459class X:
460 # Implicit globals inside classes are be looked up by LOAD_NAME, not
461 # LOAD_GLOBAL.
462 locals()['looked_up_by_load_name'] = True
463 passed = looked_up_by_load_name
464
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000465self.assertTrue(X.passed)
Georg Brandlc6fdec62006-10-28 13:10:17 +0000466"""
Neil Schemenauerd403c452005-10-23 04:24:49 +0000467
Georg Brandlc6fdec62006-10-28 13:10:17 +0000468 def testLocalsFunction(self):
Jeremy Hylton5c7a2512001-03-21 16:44:39 +0000469
Georg Brandlc6fdec62006-10-28 13:10:17 +0000470 def f(x):
471 def g(y):
472 def h(z):
473 return y + z
474 w = x + y
475 y += 3
476 return locals()
477 return g
Jeremy Hylton5c7a2512001-03-21 16:44:39 +0000478
Georg Brandlc6fdec62006-10-28 13:10:17 +0000479 d = f(2)(4)
Senthil Kumaran3ddc4352010-01-08 18:41:40 +0000480 self.assertTrue('h' in d)
Georg Brandlc6fdec62006-10-28 13:10:17 +0000481 del d['h']
482 self.assertEqual(d, {'x': 2, 'y': 7, 'w': 6})
Jeremy Hyltonc76770c2001-04-13 16:51:46 +0000483
Jeremy Hylton759410b2007-02-26 18:41:18 +0000484 def testLocalsClass(self):
485 # This test verifies that calling locals() does not pollute
486 # the local namespace of the class with free variables. Old
487 # versions of Python had a bug, where a free variable being
488 # passed through a class namespace would be inserted into
489 # locals() by locals() or exec or a trace function.
490 #
491 # The real bug lies in frame code that copies variables
492 # between fast locals and the locals dict, e.g. when executing
493 # a trace function.
494
495 def f(x):
496 class C:
497 x = 12
498 def m(self):
499 return x
500 locals()
501 return C
502
503 self.assertEqual(f(1).x, 12)
504
505 def f(x):
506 class C:
507 y = x
508 def m(self):
509 return x
510 z = list(locals())
511 return C
512
513 varnames = f(1).z
Benjamin Peterson5c8da862009-06-30 22:57:08 +0000514 self.assertTrue("x" not in varnames)
515 self.assertTrue("y" in varnames)
Jeremy Hylton759410b2007-02-26 18:41:18 +0000516
Amaury Forgeot d'Arce4921fe2008-07-21 22:00:38 +0000517 def testLocalsClass_WithTrace(self):
518 # Issue23728: after the trace function returns, the locals()
519 # dictionary is used to update all variables, this used to
520 # include free variables. But in class statements, free
521 # variables are not inserted...
522 import sys
523 sys.settrace(lambda a,b,c:None)
524 try:
525 x = 12
526
527 class C:
528 def f(self):
529 return x
530
Benjamin Petersonf5574a02008-07-21 22:05:34 +0000531 self.assertEquals(x, 12) # Used to raise UnboundLocalError
Amaury Forgeot d'Arce4921fe2008-07-21 22:00:38 +0000532 finally:
533 sys.settrace(None)
534
Georg Brandlc6fdec62006-10-28 13:10:17 +0000535 def testBoundAndFree(self):
536 # var is bound and free in class
Jeremy Hyltonddc4fd02001-04-27 02:29:40 +0000537
Georg Brandlc6fdec62006-10-28 13:10:17 +0000538 def f(x):
539 class C:
540 def m(self):
541 return x
542 a = x
543 return C
Jeremy Hyltonddc4fd02001-04-27 02:29:40 +0000544
Georg Brandlc6fdec62006-10-28 13:10:17 +0000545 inst = f(3)()
546 self.assertEqual(inst.a, inst.m())
Jeremy Hylton4c889012001-05-08 04:08:59 +0000547
Georg Brandlc6fdec62006-10-28 13:10:17 +0000548 def testInteractionWithTraceFunc(self):
Jeremy Hylton4c889012001-05-08 04:08:59 +0000549
Georg Brandlc6fdec62006-10-28 13:10:17 +0000550 import sys
551 def tracer(a,b,c):
552 return tracer
Jeremy Hylton4c889012001-05-08 04:08:59 +0000553
Georg Brandlc6fdec62006-10-28 13:10:17 +0000554 def adaptgetter(name, klass, getter):
555 kind, des = getter
556 if kind == 1: # AV happens when stepping from this line to next
557 if des == "":
558 des = "_%s__%s" % (klass.__name__, name)
559 return lambda obj: getattr(obj, des)
Jeremy Hylton4c889012001-05-08 04:08:59 +0000560
Georg Brandlc6fdec62006-10-28 13:10:17 +0000561 class TestClass:
562 pass
Jeremy Hylton4c889012001-05-08 04:08:59 +0000563
Georg Brandlc6fdec62006-10-28 13:10:17 +0000564 sys.settrace(tracer)
565 adaptgetter("foo", TestClass, (1, ""))
566 sys.settrace(None)
Jeremy Hylton5121e7d2001-07-30 21:55:29 +0000567
Georg Brandlc6fdec62006-10-28 13:10:17 +0000568 self.assertRaises(TypeError, sys.settrace)
Neal Norwitz290d31e2002-03-03 15:12:58 +0000569
Georg Brandlc6fdec62006-10-28 13:10:17 +0000570 def testEvalExecFreeVars(self):
Jeremy Hylton5121e7d2001-07-30 21:55:29 +0000571
Georg Brandlc6fdec62006-10-28 13:10:17 +0000572 def f(x):
573 return lambda: x + 1
Jeremy Hylton5121e7d2001-07-30 21:55:29 +0000574
Georg Brandlc6fdec62006-10-28 13:10:17 +0000575 g = f(3)
576 self.assertRaises(TypeError, eval, g.func_code)
Jeremy Hyltoncd738362001-08-07 16:38:19 +0000577
Georg Brandlc6fdec62006-10-28 13:10:17 +0000578 try:
579 exec g.func_code in {}
580 except TypeError:
581 pass
582 else:
583 self.fail("exec should have failed, because code contained free vars")
Jeremy Hyltonccae83772001-12-13 19:45:04 +0000584
Georg Brandlc6fdec62006-10-28 13:10:17 +0000585 def testListCompLocalVars(self):
Jeremy Hyltoncf672f12001-10-18 16:23:11 +0000586
Georg Brandlc6fdec62006-10-28 13:10:17 +0000587 try:
588 print bad
589 except NameError:
590 pass
591 else:
592 print "bad should not be defined"
Jeremy Hyltoncf672f12001-10-18 16:23:11 +0000593
Georg Brandlc6fdec62006-10-28 13:10:17 +0000594 def x():
595 [bad for s in 'a b' for bad in s.split()]
Jeremy Hyltoncf672f12001-10-18 16:23:11 +0000596
Georg Brandlc6fdec62006-10-28 13:10:17 +0000597 x()
598 try:
599 print bad
600 except NameError:
601 pass
Jeremy Hylton954aed82002-04-20 04:51:39 +0000602
Georg Brandlc6fdec62006-10-28 13:10:17 +0000603 def testEvalFreeVars(self):
Jeremy Hylton954aed82002-04-20 04:51:39 +0000604
Georg Brandlc6fdec62006-10-28 13:10:17 +0000605 def f(x):
606 def g():
607 x
608 eval("x + 1")
609 return g
Jeremy Hylton954aed82002-04-20 04:51:39 +0000610
Georg Brandlc6fdec62006-10-28 13:10:17 +0000611 f(4)()
612
Amaury Forgeot d'Arc632fad32008-02-16 20:55:24 +0000613 def testFreeingCell(self):
614 # Test what happens when a finalizer accesses
615 # the cell where the object was stored.
616 class Special:
617 def __del__(self):
618 nestedcell_get()
619
620 def f():
621 global nestedcell_get
622 def nestedcell_get():
623 return c
624
625 c = (Special(),)
626 c = 2
627
628 f() # used to crash the interpreter...
629
Jeremy Hylton88f1c042009-03-31 13:48:15 +0000630 def testGlobalInParallelNestedFunctions(self):
631 # A symbol table bug leaked the global statement from one
632 # function to other nested functions in the same block.
633 # This test verifies that a global statement in the first
634 # function does not affect the second function.
635 CODE = """def f():
636 y = 1
637 def g():
638 global y
639 return y
640 def h():
641 return y + 1
642 return g, h
643
644y = 9
645g, h = f()
646result9 = g()
647result2 = h()
648"""
649 local_ns = {}
650 global_ns = {}
651 exec CODE in local_ns, global_ns
652 self.assertEqual(2, global_ns["result2"])
653 self.assertEqual(9, global_ns["result9"])
Amaury Forgeot d'Arc632fad32008-02-16 20:55:24 +0000654
Georg Brandlc6fdec62006-10-28 13:10:17 +0000655
656def test_main():
657 run_unittest(ScopeTests)
658
659if __name__ == '__main__':
660 test_main()