summaryrefslogtreecommitdiffstats
path: root/Lib
diff options
context:
space:
mode:
authorBatuhan Taskaya <isidentical@gmail.com>2021-05-16 13:33:22 (GMT)
committerGitHub <noreply@github.com>2021-05-16 13:33:22 (GMT)
commit51cef8be8c77dff522bec6f95d6853031bf19038 (patch)
tree0203aa036aa29740dfa17b719cdc854db6cf2522 /Lib
parent1a08c5ac49b748d5e4e4b508d22d3804e3cd4dcc (diff)
downloadcpython-51cef8be8c77dff522bec6f95d6853031bf19038.zip
cpython-51cef8be8c77dff522bec6f95d6853031bf19038.tar.gz
cpython-51cef8be8c77dff522bec6f95d6853031bf19038.tar.bz2
bpo-44142: drop redundant parantheses when unparsing tuples as assignment targets (GH-26156)
Diffstat (limited to 'Lib')
-rw-r--r--Lib/ast.py9
-rw-r--r--Lib/test/test_ast.py3
-rw-r--r--Lib/test/test_unparse.py37
3 files changed, 45 insertions, 4 deletions
diff --git a/Lib/ast.py b/Lib/ast.py
index 0aef172..64487d2 100644
--- a/Lib/ast.py
+++ b/Lib/ast.py
@@ -640,7 +640,8 @@ _INFSTR = "1e" + repr(sys.float_info.max_10_exp + 1)
class _Precedence:
"""Precedence table that originated from python grammar."""
- TUPLE = auto()
+ NAMED_EXPR = auto() # <target> := <expr1>
+ TUPLE = auto() # <expr1>, <expr2>
YIELD = auto() # 'yield', 'yield from'
TEST = auto() # 'if'-'else', 'lambda'
OR = auto() # 'or'
@@ -838,7 +839,7 @@ class _Unparser(NodeVisitor):
self.traverse(node.value)
def visit_NamedExpr(self, node):
- with self.require_parens(_Precedence.TUPLE, node):
+ with self.require_parens(_Precedence.NAMED_EXPR, node):
self.set_precedence(_Precedence.ATOM, node.target, node.value)
self.traverse(node.target)
self.write(" := ")
@@ -859,6 +860,7 @@ class _Unparser(NodeVisitor):
def visit_Assign(self, node):
self.fill()
for target in node.targets:
+ self.set_precedence(_Precedence.TUPLE, target)
self.traverse(target)
self.write(" = ")
self.traverse(node.value)
@@ -1030,6 +1032,7 @@ class _Unparser(NodeVisitor):
def _for_helper(self, fill, node):
self.fill(fill)
+ self.set_precedence(_Precedence.TUPLE, node.target)
self.traverse(node.target)
self.write(" in ")
self.traverse(node.iter)
@@ -1315,7 +1318,7 @@ class _Unparser(NodeVisitor):
)
def visit_Tuple(self, node):
- with self.delimit("(", ")"):
+ with self.require_parens(_Precedence.TUPLE, node):
self.items_view(self.traverse, node.elts)
unop = {"Invert": "~", "Not": "not", "UAdd": "+", "USub": "-"}
diff --git a/Lib/test/test_ast.py b/Lib/test/test_ast.py
index 80d24e9..249e4bf 100644
--- a/Lib/test/test_ast.py
+++ b/Lib/test/test_ast.py
@@ -702,7 +702,8 @@ class AST_Tests(unittest.TestCase):
def test_precedence_enum(self):
class _Precedence(enum.IntEnum):
"""Precedence table that originated from python grammar."""
- TUPLE = enum.auto()
+ NAMED_EXPR = enum.auto() # <target> := <expr1>
+ TUPLE = enum.auto() # <expr1>, <expr2>
YIELD = enum.auto() # 'yield', 'yield from'
TEST = enum.auto() # 'if'-'else', 'lambda'
OR = enum.auto() # 'or'
diff --git a/Lib/test/test_unparse.py b/Lib/test/test_unparse.py
index 4d3340e..63151ec 100644
--- a/Lib/test/test_unparse.py
+++ b/Lib/test/test_unparse.py
@@ -541,6 +541,43 @@ class CosmeticTestCase(ASTTestCase):
self.check_src_roundtrip("lambda x, y, /, z, q, *, u: None")
self.check_src_roundtrip("lambda x, *y, **z: None")
+ def test_star_expr_assign_target(self):
+ for source_type, source in [
+ ("single assignment", "{target} = foo"),
+ ("multiple assignment", "{target} = {target} = bar"),
+ ("for loop", "for {target} in foo:\n pass"),
+ ("async for loop", "async for {target} in foo:\n pass")
+ ]:
+ for target in [
+ "a",
+ "a,",
+ "a, b",
+ "a, *b, c",
+ "a, (b, c), d",
+ "a, (b, c, d), *e",
+ "a, (b, *c, d), e",
+ "a, (b, *c, (d, e), f), g",
+ "[a]",
+ "[a, b]",
+ "[a, *b, c]",
+ "[a, [b, c], d]",
+ "[a, [b, c, d], *e]",
+ "[a, [b, *c, d], e]",
+ "[a, [b, *c, [d, e], f], g]",
+ "a, [b, c], d",
+ "[a, b, (c, d), (e, f)]",
+ "a, b, [*c], d, e"
+ ]:
+ with self.subTest(source_type=source_type, target=target):
+ self.check_src_roundtrip(source.format(target=target))
+
+ def test_star_expr_assign_target_multiple(self):
+ self.check_src_roundtrip("a = b = c = d")
+ self.check_src_roundtrip("a, b = c, d = e, f = g")
+ self.check_src_roundtrip("[a, b] = [c, d] = [e, f] = g")
+ self.check_src_roundtrip("a, b = [c, d] = e, f = g")
+
+
class DirectoryTestCase(ASTTestCase):
"""Test roundtrip behaviour on all files in Lib and Lib/test."""