Skip to content

Commit

Permalink
make assigning property via reference impure
Browse files Browse the repository at this point in the history
  • Loading branch information
schlndh committed May 17, 2024
1 parent ca41b7d commit c2bb307
Show file tree
Hide file tree
Showing 3 changed files with 83 additions and 1 deletion.
18 changes: 17 additions & 1 deletion src/Analyser/NodeScopeResolver.php
Original file line number Diff line number Diff line change
Expand Up @@ -2135,7 +2135,23 @@ public function processExprNode(Node\Stmt $stmt, Expr $expr, MutatingScope $scop
$nodeCallback,
$context,
function (MutatingScope $scope) use ($stmt, $expr, $nodeCallback, $context): ExpressionResult {
$impurePoints = [];
if ($expr instanceof AssignRef) {
$referencedExpr = $expr->expr;
while ($referencedExpr instanceof ArrayDimFetch) {
$referencedExpr = $referencedExpr->var;
}

if ($referencedExpr instanceof PropertyFetch || $referencedExpr instanceof StaticPropertyFetch) {
$impurePoints[] = new ImpurePoint(
$scope,
$expr,
'propertyAssign',
'property assignment',
false,
);
}

$scope = $scope->enterExpressionAssign($expr->expr);
}

Expand All @@ -2150,7 +2166,7 @@ function (MutatingScope $scope) use ($stmt, $expr, $nodeCallback, $context): Exp
$result = $this->processExprNode($stmt, $expr->expr, $scope, $nodeCallback, $context->enterDeep());
$hasYield = $result->hasYield();
$throwPoints = $result->getThrowPoints();
$impurePoints = $result->getImpurePoints();
$impurePoints = array_merge($impurePoints, $result->getImpurePoints());
$scope = $result->getScope();

if ($expr instanceof AssignRef) {
Expand Down
5 changes: 5 additions & 0 deletions tests/PHPStan/Rules/Pure/PureMethodRuleTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -157,4 +157,9 @@ public function testPureConstructor(): void
]);
}

public function testImpureAssignRef(): void
{
$this->analyse([__DIR__ . '/data/impure-assign-ref.php'], []);
}

}
61 changes: 61 additions & 0 deletions tests/PHPStan/Rules/Pure/data/impure-assign-ref.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
<?php declare(strict_types = 1); // lint >= 7.4

namespace ImpureAssignRef;

class HelloWorld
{
public int $value = 0;
/** @var array<int, int> */
public array $arr = [];
/** @var array<int, \stdClass> */
public array $objectArr = [];
public static int $staticValue = 0;
/** @var array<int, int> */
public static array $staticArr = [];

private function bar1(): void
{
$value = &$this->value;
$value = 1;
}

private function bar2(): void
{
$value = &$this->arr[0];
$value = 1;
}

private function bar3(): void
{
$value = &self::$staticValue;
$value = 1;
}

private function bar4(): void
{
$value = &self::$staticArr[0];
$value = 1;
}

private function bar5(self $other): void
{
$value = &$other->value;
$value = 1;
}

private function bar6(): void
{
$value = &$this->objectArr[0]->foo;
$value = 1;
}

public function foo(): void
{
$this->bar1();
$this->bar2();
$this->bar3();
$this->bar4();
$this->bar5(new self());
$this->bar6();
}
}

0 comments on commit c2bb307

Please sign in to comment.