blob: ca5f43fa9a73cb0ba3aa300b2c2016304efebc40 [file] [log] [blame]
Fred Drake79ca79d2000-08-21 22:30:53 +00001import parser
Benjamin Petersoneeed0c72008-11-03 15:18:30 +00002import os
Fred Drake58422e52001-06-04 03:56:24 +00003import unittest
Martin v. Löwis66e26632008-03-18 13:16:05 +00004import sys
Barry Warsaw04f357c2002-07-23 19:04:11 +00005from test import test_support
Fred Drake79ca79d2000-08-21 22:30:53 +00006
7#
8# First, we test that we can generate trees from valid source fragments,
9# and that these valid trees are indeed allowed by the tree-loading side
10# of the parser module.
11#
12
Fred Drake58422e52001-06-04 03:56:24 +000013class RoundtripLegalSyntaxTestCase(unittest.TestCase):
Guido van Rossum32c2ae72002-08-22 19:45:32 +000014
Fred Drake58422e52001-06-04 03:56:24 +000015 def roundtrip(self, f, s):
16 st1 = f(s)
17 t = st1.totuple()
18 try:
Fred Drake6e4f2c02001-07-17 19:33:25 +000019 st2 = parser.sequence2st(t)
Anthony Baxterc2a5a632004-08-02 06:10:11 +000020 except parser.ParserError, why:
21 self.fail("could not roundtrip %r: %s" % (s, why))
Fred Drake79ca79d2000-08-21 22:30:53 +000022
Fred Drake58422e52001-06-04 03:56:24 +000023 self.assertEquals(t, st2.totuple(),
24 "could not re-generate syntax tree")
Fred Drake28f739a2000-08-25 22:42:40 +000025
Fred Drake58422e52001-06-04 03:56:24 +000026 def check_expr(self, s):
27 self.roundtrip(parser.expr, s)
Fred Drake28f739a2000-08-25 22:42:40 +000028
Benjamin Peterson084ce7a2008-10-31 02:26:20 +000029 def test_flags_passed(self):
30 # The unicode literals flags has to be passed from the paser to AST
31 # generation.
32 suite = parser.suite("from __future__ import unicode_literals; x = ''")
33 code = suite.compile()
34 scope = {}
35 exec code in scope
36 self.assertTrue(isinstance(scope["x"], unicode))
37
Fred Drake58422e52001-06-04 03:56:24 +000038 def check_suite(self, s):
39 self.roundtrip(parser.suite, s)
Fred Drake28f739a2000-08-25 22:42:40 +000040
Fred Drakecf580c72001-07-17 03:01:29 +000041 def test_yield_statement(self):
Tim Peters496563a2002-04-01 00:28:59 +000042 self.check_suite("def f(): yield 1")
Phillip J. Eby0d6615f2005-08-02 00:46:46 +000043 self.check_suite("def f(): yield")
44 self.check_suite("def f(): x += yield")
45 self.check_suite("def f(): x = yield 1")
46 self.check_suite("def f(): x = y = yield 1")
47 self.check_suite("def f(): x = yield")
48 self.check_suite("def f(): x = y = yield")
49 self.check_suite("def f(): 1 + (yield)*2")
50 self.check_suite("def f(): (yield 1)*2")
Tim Peters496563a2002-04-01 00:28:59 +000051 self.check_suite("def f(): return; yield 1")
52 self.check_suite("def f(): yield 1; return")
53 self.check_suite("def f():\n"
Fred Drakecf580c72001-07-17 03:01:29 +000054 " for x in range(30):\n"
55 " yield x\n")
Phillip J. Eby0d6615f2005-08-02 00:46:46 +000056 self.check_suite("def f():\n"
57 " if (yield):\n"
58 " yield x\n")
Fred Drakecf580c72001-07-17 03:01:29 +000059
Fred Drake58422e52001-06-04 03:56:24 +000060 def test_expressions(self):
61 self.check_expr("foo(1)")
62 self.check_expr("[1, 2, 3]")
63 self.check_expr("[x**3 for x in range(20)]")
64 self.check_expr("[x**3 for x in range(20) if x % 3]")
Neal Norwitzd3a91622006-04-12 05:27:46 +000065 self.check_expr("[x**3 for x in range(20) if x % 2 if x % 3]")
66 self.check_expr("list(x**3 for x in range(20))")
67 self.check_expr("list(x**3 for x in range(20) if x % 3)")
68 self.check_expr("list(x**3 for x in range(20) if x % 2 if x % 3)")
Fred Drake58422e52001-06-04 03:56:24 +000069 self.check_expr("foo(*args)")
70 self.check_expr("foo(*args, **kw)")
71 self.check_expr("foo(**kw)")
72 self.check_expr("foo(key=value)")
73 self.check_expr("foo(key=value, *args)")
74 self.check_expr("foo(key=value, *args, **kw)")
75 self.check_expr("foo(key=value, **kw)")
76 self.check_expr("foo(a, b, c, *args)")
77 self.check_expr("foo(a, b, c, *args, **kw)")
78 self.check_expr("foo(a, b, c, **kw)")
Benjamin Petersonbd6a05f2008-08-19 22:06:11 +000079 self.check_expr("foo(a, *args, keyword=23)")
Fred Drake58422e52001-06-04 03:56:24 +000080 self.check_expr("foo + bar")
Michael W. Hudson5e83b7a2003-01-29 14:20:23 +000081 self.check_expr("foo - bar")
82 self.check_expr("foo * bar")
83 self.check_expr("foo / bar")
84 self.check_expr("foo // bar")
Fred Drake58422e52001-06-04 03:56:24 +000085 self.check_expr("lambda: 0")
86 self.check_expr("lambda x: 0")
87 self.check_expr("lambda *y: 0")
88 self.check_expr("lambda *y, **z: 0")
89 self.check_expr("lambda **z: 0")
90 self.check_expr("lambda x, y: 0")
91 self.check_expr("lambda foo=bar: 0")
92 self.check_expr("lambda foo=bar, spaz=nifty+spit: 0")
93 self.check_expr("lambda foo=bar, **z: 0")
94 self.check_expr("lambda foo=bar, blaz=blat+2, **z: 0")
95 self.check_expr("lambda foo=bar, blaz=blat+2, *y, **z: 0")
96 self.check_expr("lambda x, *y, **z: 0")
Raymond Hettinger354433a2004-05-19 08:20:33 +000097 self.check_expr("(x for x in range(10))")
98 self.check_expr("foo(x for x in range(10))")
Fred Drake79ca79d2000-08-21 22:30:53 +000099
Fred Drake58422e52001-06-04 03:56:24 +0000100 def test_print(self):
101 self.check_suite("print")
102 self.check_suite("print 1")
103 self.check_suite("print 1,")
104 self.check_suite("print >>fp")
105 self.check_suite("print >>fp, 1")
106 self.check_suite("print >>fp, 1,")
Fred Drake79ca79d2000-08-21 22:30:53 +0000107
Fred Drake58422e52001-06-04 03:56:24 +0000108 def test_simple_expression(self):
109 # expr_stmt
110 self.check_suite("a")
Fred Drake79ca79d2000-08-21 22:30:53 +0000111
Fred Drake58422e52001-06-04 03:56:24 +0000112 def test_simple_assignments(self):
113 self.check_suite("a = b")
114 self.check_suite("a = b = c = d = e")
Fred Drake28f739a2000-08-25 22:42:40 +0000115
Fred Drake58422e52001-06-04 03:56:24 +0000116 def test_simple_augmented_assignments(self):
117 self.check_suite("a += b")
118 self.check_suite("a -= b")
119 self.check_suite("a *= b")
120 self.check_suite("a /= b")
Michael W. Hudson5e83b7a2003-01-29 14:20:23 +0000121 self.check_suite("a //= b")
Fred Drake58422e52001-06-04 03:56:24 +0000122 self.check_suite("a %= b")
123 self.check_suite("a &= b")
124 self.check_suite("a |= b")
125 self.check_suite("a ^= b")
126 self.check_suite("a <<= b")
127 self.check_suite("a >>= b")
128 self.check_suite("a **= b")
Fred Drakee3fb18c2001-01-07 06:02:19 +0000129
Fred Drake58422e52001-06-04 03:56:24 +0000130 def test_function_defs(self):
131 self.check_suite("def f(): pass")
132 self.check_suite("def f(*args): pass")
133 self.check_suite("def f(*args, **kw): pass")
134 self.check_suite("def f(**kw): pass")
135 self.check_suite("def f(foo=bar): pass")
136 self.check_suite("def f(foo=bar, *args): pass")
137 self.check_suite("def f(foo=bar, *args, **kw): pass")
138 self.check_suite("def f(foo=bar, **kw): pass")
Fred Drakee3fb18c2001-01-07 06:02:19 +0000139
Fred Drake58422e52001-06-04 03:56:24 +0000140 self.check_suite("def f(a, b): pass")
141 self.check_suite("def f(a, b, *args): pass")
142 self.check_suite("def f(a, b, *args, **kw): pass")
143 self.check_suite("def f(a, b, **kw): pass")
144 self.check_suite("def f(a, b, foo=bar): pass")
145 self.check_suite("def f(a, b, foo=bar, *args): pass")
146 self.check_suite("def f(a, b, foo=bar, *args, **kw): pass")
147 self.check_suite("def f(a, b, foo=bar, **kw): pass")
Fred Drakee3fb18c2001-01-07 06:02:19 +0000148
Anthony Baxterc2a5a632004-08-02 06:10:11 +0000149 self.check_suite("@staticmethod\n"
150 "def f(): pass")
151 self.check_suite("@staticmethod\n"
152 "@funcattrs(x, y)\n"
153 "def f(): pass")
154 self.check_suite("@funcattrs()\n"
155 "def f(): pass")
156
Brett Cannonf4189912005-04-09 02:30:16 +0000157 def test_class_defs(self):
158 self.check_suite("class foo():pass")
Tim Peterse8906822005-04-20 17:45:13 +0000159
Fred Drake58422e52001-06-04 03:56:24 +0000160 def test_import_from_statement(self):
161 self.check_suite("from sys.path import *")
162 self.check_suite("from sys.path import dirname")
Anthony Baxter1a4ddae2004-08-31 10:07:13 +0000163 self.check_suite("from sys.path import (dirname)")
164 self.check_suite("from sys.path import (dirname,)")
Fred Drake58422e52001-06-04 03:56:24 +0000165 self.check_suite("from sys.path import dirname as my_dirname")
Anthony Baxter1a4ddae2004-08-31 10:07:13 +0000166 self.check_suite("from sys.path import (dirname as my_dirname)")
167 self.check_suite("from sys.path import (dirname as my_dirname,)")
Fred Drake58422e52001-06-04 03:56:24 +0000168 self.check_suite("from sys.path import dirname, basename")
Anthony Baxter1a4ddae2004-08-31 10:07:13 +0000169 self.check_suite("from sys.path import (dirname, basename)")
170 self.check_suite("from sys.path import (dirname, basename,)")
Fred Drake58422e52001-06-04 03:56:24 +0000171 self.check_suite(
172 "from sys.path import dirname as my_dirname, basename")
173 self.check_suite(
Anthony Baxter1a4ddae2004-08-31 10:07:13 +0000174 "from sys.path import (dirname as my_dirname, basename)")
175 self.check_suite(
176 "from sys.path import (dirname as my_dirname, basename,)")
177 self.check_suite(
Fred Drake58422e52001-06-04 03:56:24 +0000178 "from sys.path import dirname, basename as my_basename")
Anthony Baxter1a4ddae2004-08-31 10:07:13 +0000179 self.check_suite(
180 "from sys.path import (dirname, basename as my_basename)")
181 self.check_suite(
182 "from sys.path import (dirname, basename as my_basename,)")
Benjamin Petersoneeed0c72008-11-03 15:18:30 +0000183 self.check_suite("from .bogus import x")
Fred Drakee3fb18c2001-01-07 06:02:19 +0000184
Fred Drake58422e52001-06-04 03:56:24 +0000185 def test_basic_import_statement(self):
186 self.check_suite("import sys")
187 self.check_suite("import sys as system")
188 self.check_suite("import sys, math")
189 self.check_suite("import sys as system, math")
190 self.check_suite("import sys, math as my_math")
Fred Drake79ca79d2000-08-21 22:30:53 +0000191
Neal Norwitz9caf9c02003-02-10 01:54:06 +0000192 def test_pep263(self):
193 self.check_suite("# -*- coding: iso-8859-1 -*-\n"
194 "pass\n")
195
196 def test_assert(self):
197 self.check_suite("assert alo < ahi and blo < bhi\n")
198
Benjamin Peterson9e233a52008-11-24 04:19:49 +0000199 def test_with(self):
200 self.check_suite("with open('x'): pass\n")
201 self.check_suite("with open('x') as f: pass\n")
202
Georg Brandlf3a0b862008-12-07 14:47:12 +0000203 def test_try_stmt(self):
204 self.check_suite("try: pass\nexcept: pass\n")
205 self.check_suite("try: pass\nfinally: pass\n")
206 self.check_suite("try: pass\nexcept A: pass\nfinally: pass\n")
207 self.check_suite("try: pass\nexcept A: pass\nexcept: pass\n"
208 "finally: pass\n")
209 self.check_suite("try: pass\nexcept: pass\nelse: pass\n")
210 self.check_suite("try: pass\nexcept: pass\nelse: pass\n"
211 "finally: pass\n")
212
Mark Dickinson9f11f392010-06-30 16:33:23 +0000213 def test_except_clause(self):
214 self.check_suite("try: pass\nexcept: pass\n")
215 self.check_suite("try: pass\nexcept A: pass\n")
216 self.check_suite("try: pass\nexcept A, e: pass\n")
217 self.check_suite("try: pass\nexcept A as e: pass\n")
218
Jeremy Hylton60e96f62006-08-22 20:46:00 +0000219 def test_position(self):
220 # An absolutely minimal test of position information. Better
221 # tests would be a big project.
222 code = "def f(x):\n return x + 1\n"
223 st1 = parser.suite(code)
224 st2 = st1.totuple(line_info=1, col_info=1)
225
226 def walk(tree):
227 node_type = tree[0]
228 next = tree[1]
229 if isinstance(next, tuple):
230 for elt in tree[1:]:
231 for x in walk(elt):
232 yield x
233 else:
234 yield tree
Tim Peters147f9ae2006-08-25 22:05:39 +0000235
Jeremy Hylton60e96f62006-08-22 20:46:00 +0000236 terminals = list(walk(st2))
237 self.assertEqual([
238 (1, 'def', 1, 0),
239 (1, 'f', 1, 4),
240 (7, '(', 1, 5),
241 (1, 'x', 1, 6),
242 (8, ')', 1, 7),
243 (11, ':', 1, 8),
244 (4, '', 1, 9),
245 (5, '', 2, -1),
246 (1, 'return', 2, 4),
247 (1, 'x', 2, 11),
248 (14, '+', 2, 13),
249 (2, '1', 2, 15),
250 (4, '', 2, 16),
251 (6, '', 2, -1),
252 (4, '', 2, -1),
253 (0, '', 2, -1)],
254 terminals)
255
256
Fred Drake79ca79d2000-08-21 22:30:53 +0000257#
258# Second, we take *invalid* trees and make sure we get ParserError
259# rejections for them.
260#
261
Fred Drake58422e52001-06-04 03:56:24 +0000262class IllegalSyntaxTestCase(unittest.TestCase):
Guido van Rossum32c2ae72002-08-22 19:45:32 +0000263
Fred Drake58422e52001-06-04 03:56:24 +0000264 def check_bad_tree(self, tree, label):
265 try:
Fred Drake6e4f2c02001-07-17 19:33:25 +0000266 parser.sequence2st(tree)
Fred Drake58422e52001-06-04 03:56:24 +0000267 except parser.ParserError:
268 pass
269 else:
270 self.fail("did not detect invalid tree for %r" % label)
Fred Drake79ca79d2000-08-21 22:30:53 +0000271
Fred Drake58422e52001-06-04 03:56:24 +0000272 def test_junk(self):
273 # not even remotely valid:
274 self.check_bad_tree((1, 2, 3), "<junk>")
275
Fred Drakecf580c72001-07-17 03:01:29 +0000276 def test_illegal_yield_1(self):
Guido van Rossum32c2ae72002-08-22 19:45:32 +0000277 # Illegal yield statement: def f(): return 1; yield 1
Fred Drakecf580c72001-07-17 03:01:29 +0000278 tree = \
279 (257,
280 (264,
281 (285,
282 (259,
283 (1, 'def'),
284 (1, 'f'),
285 (260, (7, '('), (8, ')')),
286 (11, ':'),
287 (291,
288 (4, ''),
289 (5, ''),
290 (264,
291 (265,
292 (266,
293 (272,
294 (275,
295 (1, 'return'),
296 (313,
297 (292,
298 (293,
299 (294,
300 (295,
301 (297,
302 (298,
303 (299,
304 (300,
305 (301,
306 (302, (303, (304, (305, (2, '1')))))))))))))))))),
307 (264,
308 (265,
309 (266,
310 (272,
311 (276,
312 (1, 'yield'),
313 (313,
314 (292,
315 (293,
316 (294,
317 (295,
318 (297,
319 (298,
320 (299,
321 (300,
322 (301,
323 (302,
324 (303, (304, (305, (2, '1')))))))))))))))))),
325 (4, ''))),
326 (6, ''))))),
327 (4, ''),
328 (0, ''))))
329 self.check_bad_tree(tree, "def f():\n return 1\n yield 1")
330
331 def test_illegal_yield_2(self):
Guido van Rossum32c2ae72002-08-22 19:45:32 +0000332 # Illegal return in generator: def f(): return 1; yield 1
Fred Drakecf580c72001-07-17 03:01:29 +0000333 tree = \
334 (257,
335 (264,
336 (265,
337 (266,
338 (278,
339 (1, 'from'),
340 (281, (1, '__future__')),
341 (1, 'import'),
342 (279, (1, 'generators')))),
343 (4, ''))),
344 (264,
345 (285,
346 (259,
347 (1, 'def'),
348 (1, 'f'),
349 (260, (7, '('), (8, ')')),
350 (11, ':'),
351 (291,
352 (4, ''),
353 (5, ''),
354 (264,
355 (265,
356 (266,
357 (272,
358 (275,
359 (1, 'return'),
360 (313,
361 (292,
362 (293,
363 (294,
364 (295,
365 (297,
366 (298,
367 (299,
368 (300,
369 (301,
370 (302, (303, (304, (305, (2, '1')))))))))))))))))),
371 (264,
372 (265,
373 (266,
374 (272,
375 (276,
376 (1, 'yield'),
377 (313,
378 (292,
379 (293,
380 (294,
381 (295,
382 (297,
383 (298,
384 (299,
385 (300,
386 (301,
387 (302,
388 (303, (304, (305, (2, '1')))))))))))))))))),
389 (4, ''))),
390 (6, ''))))),
391 (4, ''),
392 (0, ''))))
393 self.check_bad_tree(tree, "def f():\n return 1\n yield 1")
394
Fred Drake58422e52001-06-04 03:56:24 +0000395 def test_print_chevron_comma(self):
Guido van Rossum32c2ae72002-08-22 19:45:32 +0000396 # Illegal input: print >>fp,
Fred Drake58422e52001-06-04 03:56:24 +0000397 tree = \
398 (257,
399 (264,
400 (265,
401 (266,
402 (268,
403 (1, 'print'),
404 (35, '>>'),
405 (290,
406 (291,
407 (292,
408 (293,
409 (295,
410 (296,
411 (297,
412 (298, (299, (300, (301, (302, (303, (1, 'fp')))))))))))))),
413 (12, ','))),
414 (4, ''))),
415 (0, ''))
416 self.check_bad_tree(tree, "print >>fp,")
417
418 def test_a_comma_comma_c(self):
Guido van Rossum32c2ae72002-08-22 19:45:32 +0000419 # Illegal input: a,,c
Fred Drake58422e52001-06-04 03:56:24 +0000420 tree = \
421 (258,
422 (311,
423 (290,
424 (291,
425 (292,
426 (293,
427 (295,
428 (296,
429 (297,
430 (298, (299, (300, (301, (302, (303, (1, 'a')))))))))))))),
431 (12, ','),
432 (12, ','),
433 (290,
434 (291,
435 (292,
436 (293,
437 (295,
438 (296,
439 (297,
440 (298, (299, (300, (301, (302, (303, (1, 'c'))))))))))))))),
441 (4, ''),
442 (0, ''))
443 self.check_bad_tree(tree, "a,,c")
444
445 def test_illegal_operator(self):
Guido van Rossum32c2ae72002-08-22 19:45:32 +0000446 # Illegal input: a $= b
Fred Drake58422e52001-06-04 03:56:24 +0000447 tree = \
448 (257,
449 (264,
450 (265,
451 (266,
452 (267,
453 (312,
454 (291,
455 (292,
456 (293,
457 (294,
458 (296,
459 (297,
460 (298,
461 (299,
462 (300, (301, (302, (303, (304, (1, 'a'))))))))))))))),
463 (268, (37, '$=')),
464 (312,
465 (291,
466 (292,
467 (293,
468 (294,
469 (296,
470 (297,
471 (298,
472 (299,
473 (300, (301, (302, (303, (304, (1, 'b'))))))))))))))))),
474 (4, ''))),
475 (0, ''))
476 self.check_bad_tree(tree, "a $= b")
Fred Drake79ca79d2000-08-21 22:30:53 +0000477
Neal Norwitz9caf9c02003-02-10 01:54:06 +0000478 def test_malformed_global(self):
479 #doesn't have global keyword in ast
480 tree = (257,
481 (264,
482 (265,
483 (266,
484 (282, (1, 'foo'))), (4, ''))),
485 (4, ''),
Tim Petersf2715e02003-02-19 02:35:07 +0000486 (0, ''))
Neal Norwitz9caf9c02003-02-10 01:54:06 +0000487 self.check_bad_tree(tree, "malformed global ast")
Fred Drake79ca79d2000-08-21 22:30:53 +0000488
Jeremy Hylton3e0055f2005-10-20 19:59:25 +0000489
490class CompileTestCase(unittest.TestCase):
491
492 # These tests are very minimal. :-(
493
494 def test_compile_expr(self):
495 st = parser.expr('2 + 3')
496 code = parser.compilest(st)
497 self.assertEquals(eval(code), 5)
498
499 def test_compile_suite(self):
500 st = parser.suite('x = 2; y = x + 3')
501 code = parser.compilest(st)
502 globs = {}
503 exec code in globs
504 self.assertEquals(globs['y'], 5)
505
506 def test_compile_error(self):
507 st = parser.suite('1 = 3 + 4')
508 self.assertRaises(SyntaxError, parser.compilest, st)
509
Guido van Rossumb6ac23c2007-07-18 17:19:14 +0000510 def test_compile_badunicode(self):
511 st = parser.suite('a = u"\U12345678"')
512 self.assertRaises(SyntaxError, parser.compilest, st)
513 st = parser.suite('a = u"\u1"')
514 self.assertRaises(SyntaxError, parser.compilest, st)
515
Facundo Batistafc2d0102008-02-23 12:01:13 +0000516class ParserStackLimitTestCase(unittest.TestCase):
517 """try to push the parser to/over it's limits.
518 see http://bugs.python.org/issue1881 for a discussion
519 """
520 def _nested_expression(self, level):
521 return "["*level+"]"*level
522
523 def test_deeply_nested_list(self):
524 e = self._nested_expression(99)
525 st = parser.expr(e)
526 st.compile()
527
528 def test_trigger_memory_error(self):
529 e = self._nested_expression(100)
Martin v. Löwis66e26632008-03-18 13:16:05 +0000530 print >>sys.stderr, "Expecting 's_push: parser stack overflow' in next line"
Facundo Batistafc2d0102008-02-23 12:01:13 +0000531 self.assertRaises(MemoryError, parser.expr, e)
532
Fred Drake2e2be372001-09-20 21:33:42 +0000533def test_main():
Walter Dörwald21d3a322003-05-01 17:45:56 +0000534 test_support.run_unittest(
535 RoundtripLegalSyntaxTestCase,
Jeremy Hylton3e0055f2005-10-20 19:59:25 +0000536 IllegalSyntaxTestCase,
537 CompileTestCase,
Facundo Batistafc2d0102008-02-23 12:01:13 +0000538 ParserStackLimitTestCase,
Walter Dörwald21d3a322003-05-01 17:45:56 +0000539 )
Fred Drake2e2be372001-09-20 21:33:42 +0000540
541
542if __name__ == "__main__":
543 test_main()