Skip to content

Commit 682fc81

Browse files
authored
Merge pull request #11057 from pytest-dev/backport-11041-to-7.3.x
[7.3.x] 11028 - Fix warlus operator behavior when called by a function
2 parents 69689c6 + 331bc1b commit 682fc81

File tree

3 files changed

+113
-3
lines changed

3 files changed

+113
-3
lines changed

changelog/11028.bugfix.rst

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
Fixed bug in assertion rewriting where a variable assigned with the walrus operator could not be used later in a function call.

src/_pytest/assertion/rewrite.py

+22-3
Original file line numberDiff line numberDiff line change
@@ -996,7 +996,9 @@ def visit_BoolOp(self, boolop: ast.BoolOp) -> Tuple[ast.Name, str]:
996996
]
997997
):
998998
pytest_temp = self.variable()
999-
self.variables_overwrite[v.left.target.id] = pytest_temp
999+
self.variables_overwrite[
1000+
v.left.target.id
1001+
] = v.left # type:ignore[assignment]
10001002
v.left.target.id = pytest_temp
10011003
self.push_format_context()
10021004
res, expl = self.visit(v)
@@ -1037,10 +1039,19 @@ def visit_Call(self, call: ast.Call) -> Tuple[ast.Name, str]:
10371039
new_args = []
10381040
new_kwargs = []
10391041
for arg in call.args:
1042+
if isinstance(arg, ast.Name) and arg.id in self.variables_overwrite:
1043+
arg = self.variables_overwrite[arg.id] # type:ignore[assignment]
10401044
res, expl = self.visit(arg)
10411045
arg_expls.append(expl)
10421046
new_args.append(res)
10431047
for keyword in call.keywords:
1048+
if (
1049+
isinstance(keyword.value, ast.Name)
1050+
and keyword.value.id in self.variables_overwrite
1051+
):
1052+
keyword.value = self.variables_overwrite[
1053+
keyword.value.id
1054+
] # type:ignore[assignment]
10441055
res, expl = self.visit(keyword.value)
10451056
new_kwargs.append(ast.keyword(keyword.arg, res))
10461057
if keyword.arg:
@@ -1075,7 +1086,13 @@ def visit_Compare(self, comp: ast.Compare) -> Tuple[ast.expr, str]:
10751086
self.push_format_context()
10761087
# We first check if we have overwritten a variable in the previous assert
10771088
if isinstance(comp.left, ast.Name) and comp.left.id in self.variables_overwrite:
1078-
comp.left.id = self.variables_overwrite[comp.left.id]
1089+
comp.left = self.variables_overwrite[
1090+
comp.left.id
1091+
] # type:ignore[assignment]
1092+
if isinstance(comp.left, namedExpr):
1093+
self.variables_overwrite[
1094+
comp.left.target.id
1095+
] = comp.left # type:ignore[assignment]
10791096
left_res, left_expl = self.visit(comp.left)
10801097
if isinstance(comp.left, (ast.Compare, ast.BoolOp)):
10811098
left_expl = f"({left_expl})"
@@ -1093,7 +1110,9 @@ def visit_Compare(self, comp: ast.Compare) -> Tuple[ast.expr, str]:
10931110
and next_operand.target.id == left_res.id
10941111
):
10951112
next_operand.target.id = self.variable()
1096-
self.variables_overwrite[left_res.id] = next_operand.target.id
1113+
self.variables_overwrite[
1114+
left_res.id
1115+
] = next_operand # type:ignore[assignment]
10971116
next_res, next_expl = self.visit(next_operand)
10981117
if isinstance(next_operand, (ast.Compare, ast.BoolOp)):
10991118
next_expl = f"({next_expl})"

testing/test_assertrewrite.py

+90
Original file line numberDiff line numberDiff line change
@@ -1436,6 +1436,96 @@ def test_walrus_operator_not_override_value():
14361436
assert result.ret == 0
14371437

14381438

1439+
@pytest.mark.skipif(
1440+
sys.version_info < (3, 8), reason="walrus operator not available in py<38"
1441+
)
1442+
class TestIssue11028:
1443+
def test_assertion_walrus_operator_in_operand(self, pytester: Pytester) -> None:
1444+
pytester.makepyfile(
1445+
"""
1446+
def test_in_string():
1447+
assert (obj := "foo") in obj
1448+
"""
1449+
)
1450+
result = pytester.runpytest()
1451+
assert result.ret == 0
1452+
1453+
def test_assertion_walrus_operator_in_operand_json_dumps(
1454+
self, pytester: Pytester
1455+
) -> None:
1456+
pytester.makepyfile(
1457+
"""
1458+
import json
1459+
1460+
def test_json_encoder():
1461+
assert (obj := "foo") in json.dumps(obj)
1462+
"""
1463+
)
1464+
result = pytester.runpytest()
1465+
assert result.ret == 0
1466+
1467+
def test_assertion_walrus_operator_equals_operand_function(
1468+
self, pytester: Pytester
1469+
) -> None:
1470+
pytester.makepyfile(
1471+
"""
1472+
def f(a):
1473+
return a
1474+
1475+
def test_call_other_function_arg():
1476+
assert (obj := "foo") == f(obj)
1477+
"""
1478+
)
1479+
result = pytester.runpytest()
1480+
assert result.ret == 0
1481+
1482+
def test_assertion_walrus_operator_equals_operand_function_keyword_arg(
1483+
self, pytester: Pytester
1484+
) -> None:
1485+
pytester.makepyfile(
1486+
"""
1487+
def f(a='test'):
1488+
return a
1489+
1490+
def test_call_other_function_k_arg():
1491+
assert (obj := "foo") == f(a=obj)
1492+
"""
1493+
)
1494+
result = pytester.runpytest()
1495+
assert result.ret == 0
1496+
1497+
def test_assertion_walrus_operator_equals_operand_function_arg_as_function(
1498+
self, pytester: Pytester
1499+
) -> None:
1500+
pytester.makepyfile(
1501+
"""
1502+
def f(a='test'):
1503+
return a
1504+
1505+
def test_function_of_function():
1506+
assert (obj := "foo") == f(f(obj))
1507+
"""
1508+
)
1509+
result = pytester.runpytest()
1510+
assert result.ret == 0
1511+
1512+
def test_assertion_walrus_operator_gt_operand_function(
1513+
self, pytester: Pytester
1514+
) -> None:
1515+
pytester.makepyfile(
1516+
"""
1517+
def add_one(a):
1518+
return a + 1
1519+
1520+
def test_gt():
1521+
assert (obj := 4) > add_one(obj)
1522+
"""
1523+
)
1524+
result = pytester.runpytest()
1525+
assert result.ret == 1
1526+
result.stdout.fnmatch_lines(["*assert 4 > 5", "*where 5 = add_one(4)"])
1527+
1528+
14391529
@pytest.mark.skipif(
14401530
sys.maxsize <= (2**31 - 1), reason="Causes OverflowError on 32bit systems"
14411531
)

0 commit comments

Comments
 (0)