Commit 11ed9479 authored by Damien George's avatar Damien George

py/lexer: Support nested [] and {} characters within f-string params.

Signed-off-by: default avatarDamien George <damien@micropython.org>
parent 196d2684
...@@ -363,9 +363,16 @@ STATIC void parse_string_literal(mp_lexer_t *lex, bool is_raw, bool is_fstring) ...@@ -363,9 +363,16 @@ STATIC void parse_string_literal(mp_lexer_t *lex, bool is_raw, bool is_fstring)
// (MicroPython limitation) note: this is completely unaware of // (MicroPython limitation) note: this is completely unaware of
// Python syntax and will not handle any expression containing '}' or ':'. // Python syntax and will not handle any expression containing '}' or ':'.
// e.g. f'{"}"}' or f'{foo({})}'. // e.g. f'{"}"}' or f'{foo({})}'.
while (!is_end(lex) && !is_char_or(lex, ':', '}')) { unsigned int nested_bracket_level = 0;
while (!is_end(lex) && (nested_bracket_level != 0 || !is_char_or(lex, ':', '}'))) {
unichar c = CUR_CHAR(lex);
if (c == '[' || c == '{') {
nested_bracket_level += 1;
} else if (c == ']' || c == '}') {
nested_bracket_level -= 1;
}
// like the default case at the end of this function, stay 8-bit clean // like the default case at the end of this function, stay 8-bit clean
vstr_add_byte(&lex->fstring_args, CUR_CHAR(lex)); vstr_add_byte(&lex->fstring_args, c);
next_char(lex); next_char(lex);
} }
if (lex->fstring_args.buf[lex->fstring_args.len - 1] == '=') { if (lex->fstring_args.buf[lex->fstring_args.len - 1] == '=') {
......
...@@ -22,6 +22,13 @@ def foo(a, b): ...@@ -22,6 +22,13 @@ def foo(a, b):
return f'{x}{y}{a}{b}' return f'{x}{y}{a}{b}'
print(foo(7, 8)) print(foo(7, 8))
# ':' character within {...} that should not be interpreted as format specifiers.
print(f"a{[0,1,2][0:2]}")
print(f"a{[0,15,2][0:2][-1]:04x}")
# Nested '{' and '}' characters.
print(f"a{ {0,1,2}}")
# PEP-0498 specifies that '\\' and '#' must be disallowed explicitly, whereas # PEP-0498 specifies that '\\' and '#' must be disallowed explicitly, whereas
# MicroPython relies on the syntax error as a result of the substitution. # MicroPython relies on the syntax error as a result of the substitution.
......
""" """
categories: Core categories: Core
description: f-strings cannot support expressions that require parsing to resolve nested braces description: f-strings cannot support expressions that require parsing to resolve unbalanced nested braces and brackets
cause: MicroPython is optimised for code space. cause: MicroPython is optimised for code space.
workaround: Only use simple expressions inside f-strings workaround: Always use balanced braces and brackets in expressions inside f-strings
""" """
f'{"hello {} world"}' print(f'{"hello { world"}')
f"{repr({})}" print(f'{"hello ] world"}')
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