mirror of https://github.com/python/cpython
gh-123562: Improve `SyntaxError` message for `case ... as a.b` (#123563)
This commit is contained in:
parent
c3ed775899
commit
23f159ae71
|
@ -1375,7 +1375,9 @@ invalid_case_block:
|
||||||
RAISE_INDENTATION_ERROR("expected an indented block after 'case' statement on line %d", a->lineno) }
|
RAISE_INDENTATION_ERROR("expected an indented block after 'case' statement on line %d", a->lineno) }
|
||||||
invalid_as_pattern:
|
invalid_as_pattern:
|
||||||
| or_pattern 'as' a="_" { RAISE_SYNTAX_ERROR_KNOWN_LOCATION(a, "cannot use '_' as a target") }
|
| or_pattern 'as' a="_" { RAISE_SYNTAX_ERROR_KNOWN_LOCATION(a, "cannot use '_' as a target") }
|
||||||
| or_pattern 'as' !NAME a=expression { RAISE_SYNTAX_ERROR_KNOWN_LOCATION(a, "invalid pattern target") }
|
| or_pattern 'as' a=expression {
|
||||||
|
RAISE_SYNTAX_ERROR_KNOWN_LOCATION(
|
||||||
|
a, "cannot use %s as pattern target", _PyPegen_get_expr_name(a)) }
|
||||||
invalid_class_pattern:
|
invalid_class_pattern:
|
||||||
| name_or_attr '(' a=invalid_class_argument_pattern { RAISE_SYNTAX_ERROR_KNOWN_RANGE(
|
| name_or_attr '(' a=invalid_class_argument_pattern { RAISE_SYNTAX_ERROR_KNOWN_RANGE(
|
||||||
PyPegen_first_item(a, pattern_ty),
|
PyPegen_first_item(a, pattern_ty),
|
||||||
|
|
|
@ -3015,6 +3015,13 @@ class TestSyntaxErrors(unittest.TestCase):
|
||||||
pass
|
pass
|
||||||
""")
|
""")
|
||||||
|
|
||||||
|
def test_multiple_assignments_to_name_in_pattern_6(self):
|
||||||
|
self.assert_syntax_error("""
|
||||||
|
match ...:
|
||||||
|
case a as a + 1: # NAME and expression with no ()
|
||||||
|
pass
|
||||||
|
""")
|
||||||
|
|
||||||
def test_multiple_starred_names_in_sequence_pattern_0(self):
|
def test_multiple_starred_names_in_sequence_pattern_0(self):
|
||||||
self.assert_syntax_error("""
|
self.assert_syntax_error("""
|
||||||
match ...:
|
match ...:
|
||||||
|
|
|
@ -1932,7 +1932,31 @@ Corner-cases that used to crash:
|
||||||
... case 42 as 1+2+4:
|
... case 42 as 1+2+4:
|
||||||
... ...
|
... ...
|
||||||
Traceback (most recent call last):
|
Traceback (most recent call last):
|
||||||
SyntaxError: invalid pattern target
|
SyntaxError: cannot use expression as pattern target
|
||||||
|
|
||||||
|
>>> match ...:
|
||||||
|
... case 42 as a.b:
|
||||||
|
... ...
|
||||||
|
Traceback (most recent call last):
|
||||||
|
SyntaxError: cannot use attribute as pattern target
|
||||||
|
|
||||||
|
>>> match ...:
|
||||||
|
... case 42 as (a, b):
|
||||||
|
... ...
|
||||||
|
Traceback (most recent call last):
|
||||||
|
SyntaxError: cannot use tuple as pattern target
|
||||||
|
|
||||||
|
>>> match ...:
|
||||||
|
... case 42 as (a + 1):
|
||||||
|
... ...
|
||||||
|
Traceback (most recent call last):
|
||||||
|
SyntaxError: cannot use expression as pattern target
|
||||||
|
|
||||||
|
>>> match ...:
|
||||||
|
... case (32 as x) | (42 as a()):
|
||||||
|
... ...
|
||||||
|
Traceback (most recent call last):
|
||||||
|
SyntaxError: cannot use function call as pattern target
|
||||||
|
|
||||||
>>> match ...:
|
>>> match ...:
|
||||||
... case Foo(z=1, y=2, x):
|
... case Foo(z=1, y=2, x):
|
||||||
|
@ -2817,6 +2841,22 @@ while 1:
|
||||||
end_offset=22 + len("obj.attr"),
|
end_offset=22 + len("obj.attr"),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def test_match_stmt_invalid_as_expr(self):
|
||||||
|
self._check_error(
|
||||||
|
textwrap.dedent(
|
||||||
|
"""
|
||||||
|
match 1:
|
||||||
|
case x as obj.attr:
|
||||||
|
...
|
||||||
|
"""
|
||||||
|
),
|
||||||
|
errtext="cannot use attribute as pattern target",
|
||||||
|
lineno=3,
|
||||||
|
end_lineno=3,
|
||||||
|
offset=15,
|
||||||
|
end_offset=15 + len("obj.attr"),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def load_tests(loader, tests, pattern):
|
def load_tests(loader, tests, pattern):
|
||||||
tests.addTest(doctest.DocTestSuite())
|
tests.addTest(doctest.DocTestSuite())
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
Improve :exc:`SyntaxError` message for using ``case ... as ...`` with not a
|
||||||
|
name.
|
|
@ -24005,7 +24005,7 @@ invalid_case_block_rule(Parser *p)
|
||||||
return _res;
|
return _res;
|
||||||
}
|
}
|
||||||
|
|
||||||
// invalid_as_pattern: or_pattern 'as' "_" | or_pattern 'as' !NAME expression
|
// invalid_as_pattern: or_pattern 'as' "_" | or_pattern 'as' expression
|
||||||
static void *
|
static void *
|
||||||
invalid_as_pattern_rule(Parser *p)
|
invalid_as_pattern_rule(Parser *p)
|
||||||
{
|
{
|
||||||
|
@ -24048,12 +24048,12 @@ invalid_as_pattern_rule(Parser *p)
|
||||||
D(fprintf(stderr, "%*c%s invalid_as_pattern[%d-%d]: %s failed!\n", p->level, ' ',
|
D(fprintf(stderr, "%*c%s invalid_as_pattern[%d-%d]: %s failed!\n", p->level, ' ',
|
||||||
p->error_indicator ? "ERROR!" : "-", _mark, p->mark, "or_pattern 'as' \"_\""));
|
p->error_indicator ? "ERROR!" : "-", _mark, p->mark, "or_pattern 'as' \"_\""));
|
||||||
}
|
}
|
||||||
{ // or_pattern 'as' !NAME expression
|
{ // or_pattern 'as' expression
|
||||||
if (p->error_indicator) {
|
if (p->error_indicator) {
|
||||||
p->level--;
|
p->level--;
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
D(fprintf(stderr, "%*c> invalid_as_pattern[%d-%d]: %s\n", p->level, ' ', _mark, p->mark, "or_pattern 'as' !NAME expression"));
|
D(fprintf(stderr, "%*c> invalid_as_pattern[%d-%d]: %s\n", p->level, ' ', _mark, p->mark, "or_pattern 'as' expression"));
|
||||||
Token * _keyword;
|
Token * _keyword;
|
||||||
expr_ty a;
|
expr_ty a;
|
||||||
pattern_ty or_pattern_var;
|
pattern_ty or_pattern_var;
|
||||||
|
@ -24062,13 +24062,11 @@ invalid_as_pattern_rule(Parser *p)
|
||||||
&&
|
&&
|
||||||
(_keyword = _PyPegen_expect_token(p, 666)) // token='as'
|
(_keyword = _PyPegen_expect_token(p, 666)) // token='as'
|
||||||
&&
|
&&
|
||||||
_PyPegen_lookahead_with_name(0, _PyPegen_name_token, p)
|
|
||||||
&&
|
|
||||||
(a = expression_rule(p)) // expression
|
(a = expression_rule(p)) // expression
|
||||||
)
|
)
|
||||||
{
|
{
|
||||||
D(fprintf(stderr, "%*c+ invalid_as_pattern[%d-%d]: %s succeeded!\n", p->level, ' ', _mark, p->mark, "or_pattern 'as' !NAME expression"));
|
D(fprintf(stderr, "%*c+ invalid_as_pattern[%d-%d]: %s succeeded!\n", p->level, ' ', _mark, p->mark, "or_pattern 'as' expression"));
|
||||||
_res = RAISE_SYNTAX_ERROR_KNOWN_LOCATION ( a , "invalid pattern target" );
|
_res = RAISE_SYNTAX_ERROR_KNOWN_LOCATION ( a , "cannot use %s as pattern target" , _PyPegen_get_expr_name ( a ) );
|
||||||
if (_res == NULL && PyErr_Occurred()) {
|
if (_res == NULL && PyErr_Occurred()) {
|
||||||
p->error_indicator = 1;
|
p->error_indicator = 1;
|
||||||
p->level--;
|
p->level--;
|
||||||
|
@ -24078,7 +24076,7 @@ invalid_as_pattern_rule(Parser *p)
|
||||||
}
|
}
|
||||||
p->mark = _mark;
|
p->mark = _mark;
|
||||||
D(fprintf(stderr, "%*c%s invalid_as_pattern[%d-%d]: %s failed!\n", p->level, ' ',
|
D(fprintf(stderr, "%*c%s invalid_as_pattern[%d-%d]: %s failed!\n", p->level, ' ',
|
||||||
p->error_indicator ? "ERROR!" : "-", _mark, p->mark, "or_pattern 'as' !NAME expression"));
|
p->error_indicator ? "ERROR!" : "-", _mark, p->mark, "or_pattern 'as' expression"));
|
||||||
}
|
}
|
||||||
_res = NULL;
|
_res = NULL;
|
||||||
done:
|
done:
|
||||||
|
|
Loading…
Reference in New Issue