@@ -93,7 +93,7 @@ assignment[stmt_ty]:
93
93
CHECK_VERSION(6, "Variable annotations syntax is", _Py_AnnAssign(a, b, c, 0, EXTRA)) }
94
94
| a=(z=star_targets '=' { z })+ b=(yield_expr | star_expressions) !'=' tc=[TYPE_COMMENT] {
95
95
_Py_Assign(a, b, NEW_TYPE_COMMENT(p, tc), EXTRA) }
96
- | a=single_target b=augassign c=(yield_expr | star_expressions) {
96
+ | a=single_target b=augassign ~ c=(yield_expr | star_expressions) {
97
97
_Py_AugAssign(a, b->kind, c, EXTRA) }
98
98
| invalid_assignment
99
99
@@ -121,7 +121,9 @@ yield_stmt[stmt_ty]: y=yield_expr { _Py_Expr(y, EXTRA) }
121
121
122
122
assert_stmt[stmt_ty]: 'assert' a=expression b=[',' z=expression { z }] { _Py_Assert(a, b, EXTRA) }
123
123
124
- del_stmt[stmt_ty]: 'del' a=del_targets { _Py_Delete(a, EXTRA) }
124
+ del_stmt[stmt_ty]:
125
+ | 'del' a=del_targets &(';' | NEWLINE) { _Py_Delete(a, EXTRA) }
126
+ | invalid_del_stmt
125
127
126
128
import_stmt[stmt_ty]: import_name | import_from
127
129
import_name[stmt_ty]: 'import' a=dotted_as_names { _Py_Import(a, EXTRA) }
@@ -164,10 +166,11 @@ while_stmt[stmt_ty]:
164
166
| 'while' a=named_expression ':' b=block c=[else_block] { _Py_While(a, b, c, EXTRA) }
165
167
166
168
for_stmt[stmt_ty]:
167
- | 'for' t=star_targets 'in' ex=star_expressions ':' tc=[TYPE_COMMENT] b=block el=[else_block] {
169
+ | 'for' t=star_targets 'in' ~ ex=star_expressions ':' tc=[TYPE_COMMENT] b=block el=[else_block] {
168
170
_Py_For(t, ex, b, el, NEW_TYPE_COMMENT(p, tc), EXTRA) }
169
- | ASYNC 'for' t=star_targets 'in' ex=star_expressions ':' tc=[TYPE_COMMENT] b=block el=[else_block] {
171
+ | ASYNC 'for' t=star_targets 'in' ~ ex=star_expressions ':' tc=[TYPE_COMMENT] b=block el=[else_block] {
170
172
CHECK_VERSION(5, "Async for loops are", _Py_AsyncFor(t, ex, b, el, NEW_TYPE_COMMENT(p, tc), EXTRA)) }
173
+ | invalid_for_target
171
174
172
175
with_stmt[stmt_ty]:
173
176
| 'with' '(' a=','.with_item+ ','? ')' ':' b=block {
@@ -179,7 +182,9 @@ with_stmt[stmt_ty]:
179
182
| ASYNC 'with' a=','.with_item+ ':' tc=[TYPE_COMMENT] b=block {
180
183
CHECK_VERSION(5, "Async with statements are", _Py_AsyncWith(a, b, NEW_TYPE_COMMENT(p, tc), EXTRA)) }
181
184
with_item[withitem_ty]:
182
- | e=expression o=['as' t=target { t }] { _Py_withitem(e, o, p->arena) }
185
+ | e=expression 'as' t=target &(',' | ')' | ':') { _Py_withitem(e, t, p->arena) }
186
+ | invalid_with_item
187
+ | e=expression { _Py_withitem(e, NULL, p->arena) }
183
188
184
189
try_stmt[stmt_ty]:
185
190
| 'try' ':' b=block f=finally_block { _Py_Try(b, NULL, NULL, f, EXTRA) }
@@ -311,7 +316,7 @@ star_named_expression[expr_ty]:
311
316
| '*' a=bitwise_or { _Py_Starred(a, Load, EXTRA) }
312
317
| named_expression
313
318
named_expression[expr_ty]:
314
- | a=NAME ':=' b=expression { _Py_NamedExpr(CHECK(_PyPegen_set_expr_context(p, a, Store)), b, EXTRA) }
319
+ | a=NAME ':=' ~ b=expression { _Py_NamedExpr(CHECK(_PyPegen_set_expr_context(p, a, Store)), b, EXTRA) }
315
320
| expression !':='
316
321
| invalid_named_expression
317
322
@@ -487,18 +492,20 @@ strings[expr_ty] (memo): a=STRING+ { _PyPegen_concatenate_strings(p, a) }
487
492
list[expr_ty]:
488
493
| '[' a=[star_named_expressions] ']' { _Py_List(a, Load, EXTRA) }
489
494
listcomp[expr_ty]:
490
- | '[' a=named_expression b=for_if_clauses ']' { _Py_ListComp(a, b, EXTRA) }
495
+ | '[' a=named_expression ~ b=for_if_clauses ']' { _Py_ListComp(a, b, EXTRA) }
491
496
| invalid_comprehension
492
497
tuple[expr_ty]:
493
498
| '(' a=[y=star_named_expression ',' z=[star_named_expressions] { _PyPegen_seq_insert_in_front(p, y, z) } ] ')' {
494
499
_Py_Tuple(a, Load, EXTRA) }
495
- group[expr_ty]: '(' a=(yield_expr | named_expression) ')' { a }
500
+ group[expr_ty]:
501
+ | '(' a=(yield_expr | named_expression) ')' { a }
502
+ | invalid_group
496
503
genexp[expr_ty]:
497
- | '(' a=expression b=for_if_clauses ')' { _Py_GeneratorExp(a, b, EXTRA) }
504
+ | '(' a=expression ~ b=for_if_clauses ')' { _Py_GeneratorExp(a, b, EXTRA) }
498
505
| invalid_comprehension
499
506
set[expr_ty]: '{' a=expressions_list '}' { _Py_Set(a, EXTRA) }
500
507
setcomp[expr_ty]:
501
- | '{' a=expression b=for_if_clauses '}' { _Py_SetComp(a, b, EXTRA) }
508
+ | '{' a=expression ~ b=for_if_clauses '}' { _Py_SetComp(a, b, EXTRA) }
502
509
| invalid_comprehension
503
510
dict[expr_ty]:
504
511
| '{' a=[double_starred_kvpairs] '}' {
@@ -514,10 +521,11 @@ kvpair[KeyValuePair*]: a=expression ':' b=expression { _PyPegen_key_value_pair(p
514
521
for_if_clauses[asdl_seq*]:
515
522
| for_if_clause+
516
523
for_if_clause[comprehension_ty]:
517
- | ASYNC 'for' a=star_targets 'in' b=disjunction c=('if' z=disjunction { z })* {
524
+ | ASYNC 'for' a=star_targets 'in' ~ b=disjunction c=('if' z=disjunction { z })* {
518
525
CHECK_VERSION(6, "Async comprehensions are", _Py_comprehension(a, b, c, 1, p->arena)) }
519
- | 'for' a=star_targets 'in' b=disjunction c=('if' z=disjunction { z })* {
526
+ | 'for' a=star_targets 'in' ~ b=disjunction c=('if' z=disjunction { z })* {
520
527
_Py_comprehension(a, b, c, 0, p->arena) }
528
+ | invalid_for_target
521
529
522
530
yield_expr[expr_ty]:
523
531
| 'yield' 'from' a=expression { _Py_YieldFrom(a, EXTRA) }
@@ -587,19 +595,15 @@ single_subscript_attribute_target[expr_ty]:
587
595
| a=t_primary '[' b=slices ']' !t_lookahead { _Py_Subscript(a, b, Store, EXTRA) }
588
596
589
597
del_targets[asdl_seq*]: a=','.del_target+ [','] { a }
590
- # The lookaheads to del_target_end ensure that we don't match expressions where a prefix of the
591
- # expression matches our rule, thereby letting these cases fall through to invalid_del_target.
592
598
del_target[expr_ty] (memo):
593
- | a=t_primary '.' b=NAME &del_target_end { _Py_Attribute(a, b->v.Name.id, Del, EXTRA) }
594
- | a=t_primary '[' b=slices ']' &del_target_end { _Py_Subscript(a, b, Del, EXTRA) }
599
+ | a=t_primary '.' b=NAME !t_lookahead { _Py_Attribute(a, b->v.Name.id, Del, EXTRA) }
600
+ | a=t_primary '[' b=slices ']' !t_lookahead { _Py_Subscript(a, b, Del, EXTRA) }
595
601
| del_t_atom
596
602
del_t_atom[expr_ty]:
597
- | a=NAME &del_target_end { _PyPegen_set_expr_context(p, a, Del) }
603
+ | a=NAME { _PyPegen_set_expr_context(p, a, Del) }
598
604
| '(' a=del_target ')' { _PyPegen_set_expr_context(p, a, Del) }
599
605
| '(' a=[del_targets] ')' { _Py_Tuple(a, Del, EXTRA) }
600
606
| '[' a=[del_targets] ']' { _Py_List(a, Del, EXTRA) }
601
- | invalid_del_target
602
- del_target_end: ')' | ']' | ',' | ';' | NEWLINE
603
607
604
608
targets[asdl_seq*]: a=','.target+ [','] { a }
605
609
target[expr_ty] (memo):
@@ -650,16 +654,23 @@ invalid_assignment:
650
654
RAISE_SYNTAX_ERROR_KNOWN_LOCATION(a, "illegal target for annotation") }
651
655
| (star_targets '=')* a=star_expressions '=' {
652
656
RAISE_SYNTAX_ERROR_KNOWN_LOCATION(
653
- _PyPegen_get_invalid_target (a),
654
- "cannot assign to %s", _PyPegen_get_expr_name(_PyPegen_get_invalid_target (a))) }
657
+ GET_INVALID_TARGET (a),
658
+ "cannot assign to %s", _PyPegen_get_expr_name(GET_INVALID_TARGET (a))) }
655
659
| (star_targets '=')* a=yield_expr '=' { RAISE_SYNTAX_ERROR_KNOWN_LOCATION(a, "assignment to yield expression not possible") }
656
660
| a=star_expressions augassign (yield_expr | star_expressions) {
657
661
RAISE_SYNTAX_ERROR_KNOWN_LOCATION(
658
662
a,
659
663
"'%s' is an illegal expression for augmented assignment",
660
664
_PyPegen_get_expr_name(a)
661
665
)}
662
-
666
+ invalid_del_stmt:
667
+ | 'del' a=star_expressions {
668
+ GET_INVALID_DEL_TARGET(a) != NULL ?
669
+ RAISE_SYNTAX_ERROR_KNOWN_LOCATION(
670
+ GET_INVALID_DEL_TARGET(a),
671
+ "cannot delete %s", _PyPegen_get_expr_name(GET_INVALID_DEL_TARGET(a))
672
+ ) :
673
+ RAISE_SYNTAX_ERROR("invalid syntax") }
663
674
invalid_block:
664
675
| NEWLINE !INDENT { RAISE_INDENTATION_ERROR("expected an indented block") }
665
676
invalid_comprehension:
@@ -682,9 +693,25 @@ invalid_lambda_star_etc:
682
693
invalid_double_type_comments:
683
694
| TYPE_COMMENT NEWLINE TYPE_COMMENT NEWLINE INDENT {
684
695
RAISE_SYNTAX_ERROR("Cannot have two type comments on def") }
685
- invalid_del_target:
686
- | a=star_expression &del_target_end {
687
- RAISE_SYNTAX_ERROR_KNOWN_LOCATION(a, "cannot delete %s", _PyPegen_get_expr_name(a)) }
696
+ invalid_with_item:
697
+ | expression 'as' a=expression {
698
+ RAISE_SYNTAX_ERROR_KNOWN_LOCATION(
699
+ GET_INVALID_TARGET(a),
700
+ "cannot assign to %s", _PyPegen_get_expr_name(GET_INVALID_TARGET(a))
701
+ ) }
702
+
703
+ invalid_for_target:
704
+ | ASYNC? 'for' a=star_expressions {
705
+ GET_INVALID_FOR_TARGET(a) != NULL ?
706
+ RAISE_SYNTAX_ERROR_KNOWN_LOCATION(
707
+ GET_INVALID_FOR_TARGET(a),
708
+ "cannot assign to %s", _PyPegen_get_expr_name(GET_INVALID_FOR_TARGET(a))
709
+ ) :
710
+ RAISE_SYNTAX_ERROR("invalid syntax") }
711
+
712
+ invalid_group:
713
+ | '(' a=starred_expression ')' {
714
+ RAISE_SYNTAX_ERROR_KNOWN_LOCATION(a, "can't use starred expression here") }
688
715
invalid_import_from_targets:
689
716
| import_from_as_names ',' {
690
717
RAISE_SYNTAX_ERROR("trailing comma not allowed without surrounding parentheses") }
0 commit comments