Commit 407b3bd8 authored by Mark Dickinson's avatar Mark Dickinson

Issue #14696: Fix parser module to understand 'nonlocal' declarations.

parent 2420d831
...@@ -57,6 +57,16 @@ class RoundtripLegalSyntaxTestCase(unittest.TestCase): ...@@ -57,6 +57,16 @@ class RoundtripLegalSyntaxTestCase(unittest.TestCase):
" if (yield):\n" " if (yield):\n"
" yield x\n") " yield x\n")
def test_nonlocal_statement(self):
self.check_suite("def f():\n"
" x = 0\n"
" def g():\n"
" nonlocal x\n")
self.check_suite("def f():\n"
" x = y = 0\n"
" def g():\n"
" nonlocal x, y\n")
def test_expressions(self): def test_expressions(self):
self.check_expr("foo(1)") self.check_expr("foo(1)")
self.check_expr("[1, 2, 3]") self.check_expr("[1, 2, 3]")
......
...@@ -56,6 +56,8 @@ Core and Builtins ...@@ -56,6 +56,8 @@ Core and Builtins
Library Library
------- -------
- Issue #14696: Fix parser module to understand 'nonlocal' declarations.
- Issue #10941: Fix imaplib.Internaldate2tuple to produce correct result near - Issue #10941: Fix imaplib.Internaldate2tuple to produce correct result near
the DST transition. Patch by Joe Peterson. the DST transition. Patch by Joe Peterson.
......
...@@ -954,7 +954,8 @@ VALIDATER(del_stmt); ...@@ -954,7 +954,8 @@ VALIDATER(del_stmt);
VALIDATER(return_stmt); VALIDATER(raise_stmt); VALIDATER(return_stmt); VALIDATER(raise_stmt);
VALIDATER(import_stmt); VALIDATER(import_stmt); VALIDATER(import_stmt); VALIDATER(import_stmt);
VALIDATER(import_name); VALIDATER(yield_stmt); VALIDATER(import_name); VALIDATER(yield_stmt);
VALIDATER(global_stmt); VALIDATER(assert_stmt); VALIDATER(global_stmt); VALIDATER(nonlocal_stmt);
VALIDATER(assert_stmt);
VALIDATER(compound_stmt); VALIDATER(test_or_star_expr); VALIDATER(compound_stmt); VALIDATER(test_or_star_expr);
VALIDATER(while); VALIDATER(for); VALIDATER(while); VALIDATER(for);
VALIDATER(try); VALIDATER(except_clause); VALIDATER(try); VALIDATER(except_clause);
...@@ -1477,6 +1478,7 @@ validate_small_stmt(node *tree) ...@@ -1477,6 +1478,7 @@ validate_small_stmt(node *tree)
|| (ntype == flow_stmt) || (ntype == flow_stmt)
|| (ntype == import_stmt) || (ntype == import_stmt)
|| (ntype == global_stmt) || (ntype == global_stmt)
|| (ntype == nonlocal_stmt)
|| (ntype == assert_stmt)) || (ntype == assert_stmt))
res = validate_node(CHILD(tree, 0)); res = validate_node(CHILD(tree, 0));
else { else {
...@@ -1834,8 +1836,10 @@ validate_import_stmt(node *tree) ...@@ -1834,8 +1836,10 @@ validate_import_stmt(node *tree)
} }
/* global_stmt:
*
* 'global' NAME (',' NAME)*
*/
static int static int
validate_global_stmt(node *tree) validate_global_stmt(node *tree)
{ {
...@@ -1857,6 +1861,30 @@ validate_global_stmt(node *tree) ...@@ -1857,6 +1861,30 @@ validate_global_stmt(node *tree)
return (res); return (res);
} }
/* nonlocal_stmt:
*
* 'nonlocal' NAME (',' NAME)*
*/
static int
validate_nonlocal_stmt(node *tree)
{
int j;
int nch = NCH(tree);
int res = (validate_ntype(tree, nonlocal_stmt)
&& is_even(nch) && (nch >= 2));
if (!res && !PyErr_Occurred())
err_string("illegal nonlocal statement");
if (res)
res = (validate_name(CHILD(tree, 0), "nonlocal")
&& validate_ntype(CHILD(tree, 1), NAME));
for (j = 2; res && (j < nch); j += 2)
res = (validate_comma(CHILD(tree, j))
&& validate_ntype(CHILD(tree, j + 1), NAME));
return res;
}
/* assert_stmt: /* assert_stmt:
* *
...@@ -2921,8 +2949,8 @@ validate_node(node *tree) ...@@ -2921,8 +2949,8 @@ validate_node(node *tree)
break; break;
case small_stmt: case small_stmt:
/* /*
* expr_stmt | del_stmt | pass_stmt | flow_stmt * expr_stmt | del_stmt | pass_stmt | flow_stmt |
* | import_stmt | global_stmt | assert_stmt * import_stmt | global_stmt | nonlocal_stmt | assert_stmt
*/ */
res = validate_small_stmt(tree); res = validate_small_stmt(tree);
break; break;
...@@ -2989,6 +3017,9 @@ validate_node(node *tree) ...@@ -2989,6 +3017,9 @@ validate_node(node *tree)
case global_stmt: case global_stmt:
res = validate_global_stmt(tree); res = validate_global_stmt(tree);
break; break;
case nonlocal_stmt:
res = validate_nonlocal_stmt(tree);
break;
case assert_stmt: case assert_stmt:
res = validate_assert_stmt(tree); res = validate_assert_stmt(tree);
break; break;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment