From 36f70057a5e1616e916b5ad38f4ca726acfc59c8 Mon Sep 17 00:00:00 2001 From: Anthony Sottile Date: Mon, 16 Dec 2024 20:31:16 -0500 Subject: [PATCH] fix search for .encode(...) call with 3.12+ f-string tokens --- pyupgrade/_plugins/default_encoding.py | 4 ++-- pyupgrade/_token_helpers.py | 14 ++++++++++++++ tests/features/default_encoding_test.py | 5 +++++ 3 files changed, 21 insertions(+), 2 deletions(-) diff --git a/pyupgrade/_plugins/default_encoding.py b/pyupgrade/_plugins/default_encoding.py index f681510d..67c0a2c0 100644 --- a/pyupgrade/_plugins/default_encoding.py +++ b/pyupgrade/_plugins/default_encoding.py @@ -12,12 +12,12 @@ from pyupgrade._data import State from pyupgrade._data import TokenFunc from pyupgrade._string_helpers import is_codec +from pyupgrade._token_helpers import find_call from pyupgrade._token_helpers import find_closing_bracket -from pyupgrade._token_helpers import find_op def _fix_default_encoding(i: int, tokens: list[Token]) -> None: - i = find_op(tokens, i + 1, '(') + i = find_call(tokens, i + 1) j = find_closing_bracket(tokens, i) del tokens[i + 1:j] diff --git a/pyupgrade/_token_helpers.py b/pyupgrade/_token_helpers.py index f55722d6..fa1935d6 100644 --- a/pyupgrade/_token_helpers.py +++ b/pyupgrade/_token_helpers.py @@ -48,6 +48,20 @@ def find_op(tokens: list[Token], i: int, src: str) -> int: return _find_token(tokens, i, 'OP', src) +def find_call(tokens: list[Token], i: int) -> int: + depth = 0 + while depth or not tokens[i].matches(name='OP', src='('): + if is_open(tokens[i]): + depth += 1 + elif is_close(tokens[i]): + # why max(...)? -- + # ("something").method(...) + # ^--start target--^ + depth = max(depth - 1, 0) + i += 1 + return i + + def find_end(tokens: list[Token], i: int) -> int: while tokens[i].name != 'NEWLINE': i += 1 diff --git a/tests/features/default_encoding_test.py b/tests/features/default_encoding_test.py index f039fdca..6fb1277b 100644 --- a/tests/features/default_encoding_test.py +++ b/tests/features/default_encoding_test.py @@ -38,6 +38,11 @@ 'f"{x}(".encode()', id='3.12+ handle open brace in fstring', ), + pytest.param( + 'f"{foo(bar)}(".encode("utf-8")', + 'f"{foo(bar)}(".encode()', + id='f-string with function call', + ), ), ) def test_fix_encode(s, expected):