Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
4549f7f
Add `setShouldNotImplyOppositeCase()` on `SpecifiedTypes` to replace …
VincentLanglet May 26, 2026
e8a3593
Add comment explaining why shouldNotImplyOppositeCase causes early re…
phpstan-bot May 26, 2026
3ddd2ae
Rename `shouldNotImplyOppositeCase` to `shouldNotDetermineCheckResult`
phpstan-bot May 26, 2026
024909e
Rename `shouldNotDetermineCheckResult` to `specifyOnly`
phpstan-bot May 26, 2026
458cf86
Keep rootExpr for equality assertions, move specifyOnly after rootExp…
phpstan-bot May 27, 2026
53505ba
Rework
VincentLanglet May 27, 2026
643309d
Remove unused specifyOnly flag, document setRootExpr
phpstan-bot May 27, 2026
9e103a4
Add duplicate call detection for rootExpr-based type specifying
phpstan-bot May 27, 2026
7bee117
Remove duplicate array_key_exists check
phpstan-bot May 27, 2026
cd00997
Add setSideEffectOnly() flag on SpecifiedTypes, replace rootExpr work…
phpstan-bot May 28, 2026
6f52428
Split bug-14705 test into PHP 7.4-compatible and PHP 8.0+ parts
phpstan-bot May 28, 2026
65defc2
Rename sideEffectOnly to specifyOnly on SpecifiedTypes
phpstan-bot May 28, 2026
a8a3398
Revert unrelated duplicate array_key_exists removal
phpstan-bot May 28, 2026
334ba20
Document setSpecifyOnly() for third-party extension migration
phpstan-bot May 28, 2026
aca0f8e
Correct setSpecifyOnly() PHPDoc: clarify filterByTruthyValue scope
phpstan-bot May 28, 2026
f4e6bdb
Store specifyOnly boolean marker via overwrite to fix duplicate detec…
phpstan-bot May 29, 2026
db5c57d
Fix specifyOnly boolean marker overwriting function return types
phpstan-bot May 29, 2026
3ac709a
Explain specifyOnly expression-statement handling in NodeScopeResolver
phpstan-bot May 29, 2026
7049739
Move bug-14705 test into nsrt with assertType, merge realpath elvis c…
phpstan-bot May 29, 2026
2d92654
Rename SpecifiedTypes getter to shouldSpecifyOnly() for consistency
phpstan-bot May 29, 2026
b8ea1e9
Add array_key_exists duplicate-in-loop assertType case to bug-14705
phpstan-bot May 29, 2026
a6314d6
Improvement
VincentLanglet May 30, 2026
13f5bbd
Revert "Improvement"
VincentLanglet May 30, 2026
4c61516
Remove duplicate-detection paragraph from setSpecifyOnly() PHPDoc
phpstan-bot May 30, 2026
f3fd443
Shorten specifyOnly expression-statement comment
phpstan-bot May 30, 2026
32bc5c9
Annotate duplicate array_key_exists assertType with '// could be true'
phpstan-bot May 30, 2026
18e2767
Avoid setAlwaysOverwriteTypes() in specifyOnly statement path
phpstan-bot Jun 4, 2026
69a9e98
Rename specifyOnly to equality on SpecifiedTypes
phpstan-bot Jun 4, 2026
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
10 changes: 10 additions & 0 deletions src/Analyser/MutatingScope.php
Original file line number Diff line number Diff line change
Expand Up @@ -3302,6 +3302,11 @@ public function filterByTruthyValue(Expr $expr): self
}

$specifiedTypes = $this->typeSpecifier->specifyTypesInCondition($this, $expr, TypeSpecifierContext::createTruthy());
if ($specifiedTypes->isEquality()) {
$specifiedTypes = $specifiedTypes->unionWith(
$this->typeSpecifier->create($expr, new ConstantBooleanType(true), TypeSpecifierContext::createTrue(), $this),
);
}
$scope = $this->filterBySpecifiedTypes($specifiedTypes);
$this->truthyScopes[$exprString] = $scope;

Expand All @@ -3319,6 +3324,11 @@ public function filterByFalseyValue(Expr $expr): self
}

$specifiedTypes = $this->typeSpecifier->specifyTypesInCondition($this, $expr, TypeSpecifierContext::createFalsey());
if ($specifiedTypes->isEquality()) {
$specifiedTypes = $specifiedTypes->unionWith(
$this->typeSpecifier->create($expr, new ConstantBooleanType(false), TypeSpecifierContext::createTrue(), $this),
);
}
$scope = $this->filterBySpecifiedTypes($specifiedTypes);
$this->falseyScopes[$exprString] = $scope;

Expand Down
13 changes: 11 additions & 2 deletions src/Analyser/NodeScopeResolver.php
Original file line number Diff line number Diff line change
Expand Up @@ -143,6 +143,7 @@
use PHPStan\ShouldNotHappenException;
use PHPStan\TrinaryLogic;
use PHPStan\Type\ClosureType;
use PHPStan\Type\Constant\ConstantBooleanType;
use PHPStan\Type\Constant\ConstantIntegerType;
use PHPStan\Type\Constant\ConstantStringType;
use PHPStan\Type\FileTypeMapper;
Expand Down Expand Up @@ -1142,11 +1143,19 @@ public function processStmtNode(
$this->callNodeCallback($nodeCallback, new NoopExpressionNode($stmt->expr, $hasAssign), $scope, $storage);
}
$scope = $result->getScope();
$scope = $scope->filterBySpecifiedTypes($this->typeSpecifier->specifyTypesInCondition(
$specifiedTypes = $this->typeSpecifier->specifyTypesInCondition(
$scope,
$stmt->expr,
TypeSpecifierContext::createNull(),
));
);
$scope = $scope->filterBySpecifiedTypes($specifiedTypes);
if ($specifiedTypes->isEquality()) {
// Statement counterpart of the equality handling in filterByTruthyValue():
// store the call's true result so a duplicate void assertion statement is
// reported as always-true. We assign directly because void calls have no
// return value to protect, and intersecting true with void would produce never.
$scope = $scope->assignExpression($stmt->expr, new ConstantBooleanType(true), new ConstantBooleanType(true));
}
$hasYield = $result->hasYield();
$throwPoints = $result->getThrowPoints();
$impurePoints = $result->getImpurePoints();
Expand Down
40 changes: 40 additions & 0 deletions src/Analyser/SpecifiedTypes.php
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,8 @@ final class SpecifiedTypes

private bool $overwrite = false;

private bool $equality = false;

/** @var array<string, ConditionalExpressionHolder[]> */
private array $newConditionalExpressionHolders = [];

Expand Down Expand Up @@ -51,19 +53,48 @@ public function setAlwaysOverwriteTypes(): self
{
$self = new self($this->sureTypes, $this->sureNotTypes);
$self->overwrite = true;
$self->equality = $this->equality;
$self->newConditionalExpressionHolders = $this->newConditionalExpressionHolders;
$self->rootExpr = $this->rootExpr;

return $self;
}

/**
* Marks these types as coming from an equality check, the same concept as
* the "=Type" equality assertions documented at
* https://phpstan.org/writing-php-code/narrowing-types#equality-assertions
*
* The narrowed types are only applied; they do not determine the check
* outcome, so ImpossibleCheckTypeHelper will not use them to report
* always-true/false for the check expression.
*
* @api
*/
public function setEquality(): self
{
$self = new self($this->sureTypes, $this->sureNotTypes);
$self->overwrite = $this->overwrite;
$self->equality = true;
$self->newConditionalExpressionHolders = $this->newConditionalExpressionHolders;
$self->rootExpr = $this->rootExpr;

return $self;
}

public function isEquality(): bool
{
return $this->equality;
}

/**
* @api
*/
public function setRootExpr(?Expr $rootExpr): self
{
$self = new self($this->sureTypes, $this->sureNotTypes);
$self->overwrite = $this->overwrite;
$self->equality = $this->equality;
$self->newConditionalExpressionHolders = $this->newConditionalExpressionHolders;
$self->rootExpr = $rootExpr;

Expand All @@ -77,6 +108,7 @@ public function setNewConditionalExpressionHolders(array $newConditionalExpressi
{
$self = new self($this->sureTypes, $this->sureNotTypes);
$self->overwrite = $this->overwrite;
$self->equality = $this->equality;
$self->newConditionalExpressionHolders = $newConditionalExpressionHolders;
$self->rootExpr = $this->rootExpr;

Expand Down Expand Up @@ -128,6 +160,7 @@ public function removeExpr(string $exprString): self

$self = new self($sureTypes, $sureNotTypes);
$self->overwrite = $this->overwrite;
$self->equality = $this->equality;
$self->newConditionalExpressionHolders = $this->newConditionalExpressionHolders;
$self->rootExpr = $this->rootExpr;

Expand Down Expand Up @@ -167,6 +200,9 @@ public function intersectWith(SpecifiedTypes $other): self
if ($this->overwrite && $other->overwrite) {
$result = $result->setAlwaysOverwriteTypes();
}
if ($this->equality || $other->equality) {
$result->equality = true;
}

return $result->setRootExpr($rootExpr);
}
Expand Down Expand Up @@ -204,6 +240,9 @@ public function unionWith(SpecifiedTypes $other): self
if ($this->overwrite || $other->overwrite) {
$result = $result->setAlwaysOverwriteTypes();
}
if ($this->equality || $other->equality) {
$result->equality = true;
}

$conditionalExpressionHolders = $this->newConditionalExpressionHolders;
foreach ($other->newConditionalExpressionHolders as $exprString => $holders) {
Expand Down Expand Up @@ -235,6 +274,7 @@ public function normalize(Scope $scope): self
if ($this->overwrite) {
$result = $result->setAlwaysOverwriteTypes();
}
$result->equality = $this->equality;

return $result->setRootExpr($this->rootExpr);
}
Expand Down
5 changes: 4 additions & 1 deletion src/Analyser/TypeSpecifier.php
Original file line number Diff line number Diff line change
Expand Up @@ -473,7 +473,10 @@ static function (Type $type, callable $traverse) use ($templateTypeMap, &$contai
$assertedType,
$assert->isNegated() ? TypeSpecifierContext::createFalse() : TypeSpecifierContext::createTrue(),
$scope,
)->setRootExpr($containsUnresolvedTemplate || $assert->isEquality() ? $call : null);
);
if ($containsUnresolvedTemplate || $assert->isEquality()) {
$newTypes = $newTypes->setEquality();
}
$types = $types !== null ? $types->unionWith($newTypes) : $newTypes;

if (!$context->null() || !$assertedType instanceof ConstantBooleanType) {
Expand Down
14 changes: 14 additions & 0 deletions src/Rules/Comparison/ImpossibleCheckTypeHelper.php
Original file line number Diff line number Diff line change
Expand Up @@ -309,6 +309,20 @@
return null;
}

if ($specifiedTypes->isEquality()) {
if ($scope->hasExpressionType($node)->yes()) {

Check warning on line 313 in src/Rules/Comparison/ImpossibleCheckTypeHelper.php

View workflow job for this annotation

GitHub Actions / Mutation Testing (8.4, ubuntu-latest)

Escaped Mutant for Mutator "PHPStan\Infection\TrinaryLogicMutator": @@ @@ } if ($specifiedTypes->isEquality()) { - if ($scope->hasExpressionType($node)->yes()) { + if (!$scope->hasExpressionType($node)->no()) { $nodeType = $this->treatPhpDocTypesAsCertain ? $scope->getType($node) : $scope->getNativeType($node); if ($nodeType->isTrue()->yes()) { return true;

Check warning on line 313 in src/Rules/Comparison/ImpossibleCheckTypeHelper.php

View workflow job for this annotation

GitHub Actions / Mutation Testing (8.3, ubuntu-latest)

Escaped Mutant for Mutator "PHPStan\Infection\TrinaryLogicMutator": @@ @@ } if ($specifiedTypes->isEquality()) { - if ($scope->hasExpressionType($node)->yes()) { + if (!$scope->hasExpressionType($node)->no()) { $nodeType = $this->treatPhpDocTypesAsCertain ? $scope->getType($node) : $scope->getNativeType($node); if ($nodeType->isTrue()->yes()) { return true;
$nodeType = $this->treatPhpDocTypesAsCertain ? $scope->getType($node) : $scope->getNativeType($node);
if ($nodeType->isTrue()->yes()) {

Check warning on line 315 in src/Rules/Comparison/ImpossibleCheckTypeHelper.php

View workflow job for this annotation

GitHub Actions / Mutation Testing (8.4, ubuntu-latest)

Escaped Mutant for Mutator "PHPStan\Infection\LooseBooleanMutator": @@ @@ if ($specifiedTypes->isEquality()) { if ($scope->hasExpressionType($node)->yes()) { $nodeType = $this->treatPhpDocTypesAsCertain ? $scope->getType($node) : $scope->getNativeType($node); - if ($nodeType->isTrue()->yes()) { + if ($nodeType->toBoolean()->isTrue()->yes()) { return true; } if ($nodeType->isFalse()->yes()) {

Check warning on line 315 in src/Rules/Comparison/ImpossibleCheckTypeHelper.php

View workflow job for this annotation

GitHub Actions / Mutation Testing (8.4, ubuntu-latest)

Escaped Mutant for Mutator "PHPStan\Infection\TrinaryLogicMutator": @@ @@ if ($specifiedTypes->isEquality()) { if ($scope->hasExpressionType($node)->yes()) { $nodeType = $this->treatPhpDocTypesAsCertain ? $scope->getType($node) : $scope->getNativeType($node); - if ($nodeType->isTrue()->yes()) { + if (!$nodeType->toBoolean()->isTrue()->no()) { return true; } if ($nodeType->isFalse()->yes()) {

Check warning on line 315 in src/Rules/Comparison/ImpossibleCheckTypeHelper.php

View workflow job for this annotation

GitHub Actions / Mutation Testing (8.3, ubuntu-latest)

Escaped Mutant for Mutator "PHPStan\Infection\LooseBooleanMutator": @@ @@ if ($specifiedTypes->isEquality()) { if ($scope->hasExpressionType($node)->yes()) { $nodeType = $this->treatPhpDocTypesAsCertain ? $scope->getType($node) : $scope->getNativeType($node); - if ($nodeType->isTrue()->yes()) { + if ($nodeType->toBoolean()->isTrue()->yes()) { return true; } if ($nodeType->isFalse()->yes()) {
return true;
}
if ($nodeType->isFalse()->yes()) {
return false;
}
}

return null;
}

$sureTypes = $specifiedTypes->getSureTypes();
$sureNotTypes = $specifiedTypes->getSureNotTypes();

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,7 @@
namespace PHPStan\Type\Php;

use PhpParser\Node\Expr\ArrayDimFetch;
use PhpParser\Node\Expr\BinaryOp\Identical;
use PhpParser\Node\Expr\ConstFetch;
use PhpParser\Node\Expr\FuncCall;
use PhpParser\Node\Name;
use PHPStan\Analyser\Scope;
use PHPStan\Analyser\SpecifiedTypes;
use PHPStan\Analyser\TypeSpecifier;
Expand Down Expand Up @@ -115,7 +112,7 @@ public function specifyTypes(
$arrayType->getIterableValueType(),
$context,
$scope,
))->setRootExpr(new Identical($arrayDimFetch, new ConstFetch(new Name('__PHPSTAN_FAUX_CONSTANT'))));
))->setEquality();
}

return new SpecifiedTypes();
Expand Down
2 changes: 1 addition & 1 deletion src/Type/Php/PregMatchTypeSpecifyingExtension.php
Original file line number Diff line number Diff line change
Expand Up @@ -106,7 +106,7 @@ public function specifyTypes(FunctionReflection $functionReflection, FuncCall $n
$matchedType,
$context,
$scope,
)->setRootExpr($node);
)->setEquality();
if ($overwrite) {
$types = $types->setAlwaysOverwriteTypes();
}
Expand Down
15 changes: 1 addition & 14 deletions src/Type/Php/StrContainingTypeSpecifyingExtension.php
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,7 @@

namespace PHPStan\Type\Php;

use PhpParser\Node\Arg;
use PhpParser\Node\Expr\BinaryOp\BooleanAnd;
use PhpParser\Node\Expr\BinaryOp\NotIdentical;
use PhpParser\Node\Expr\FuncCall;
use PhpParser\Node\Name;
use PhpParser\Node\Scalar\String_;
use PHPStan\Analyser\Scope;
use PHPStan\Analyser\SpecifiedTypes;
use PHPStan\Analyser\TypeSpecifier;
Expand Down Expand Up @@ -89,15 +84,7 @@ public function specifyTypes(FunctionReflection $functionReflection, FuncCall $n
new IntersectionType($accessories),
$context,
$scope,
)->setRootExpr(new BooleanAnd(
new NotIdentical(
$args[$needleArg]->value,
new String_(''),
),
new FuncCall(new Name('FAUX_FUNCTION'), [
new Arg($args[$needleArg]->value),
]),
));
)->setEquality();
}
}

Expand Down
149 changes: 149 additions & 0 deletions tests/PHPStan/Analyser/nsrt/bug-14705.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,149 @@
<?php // lint >= 8.0

namespace Bug14705;

use function PHPStan\Testing\assertType;

class Foo
{

/**
* strpos with non-empty-string haystack should not report always-true.
*
* @param non-empty-string $haystack
* @param non-empty-string $needle
*/
public function strposNonEmpty(string $haystack, string $needle): void
{
if (strpos($haystack, $needle) !== false) {
assertType('non-empty-string', $haystack);
assertType('non-empty-string', $needle);
}
}

/**
* str_contains with non-empty-string haystack should not report always-true.
*
* @param non-empty-string $haystack
*/
public function strContainsNonEmpty(string $haystack, string $needle): void
{
if (str_contains($haystack, $needle)) {
assertType('non-empty-string', $haystack);
assertType('string', $needle);
}
}

/**
* str_starts_with with non-empty-string haystack should not report always-true.
*
* @param non-empty-string $haystack
*/
public function strStartsWithNonEmpty(string $haystack, string $needle): void
{
if (str_starts_with($haystack, $needle)) {
assertType('non-empty-string', $haystack);
assertType('string', $needle);
}
}

/**
* str_ends_with with non-empty-string haystack should not report always-true.
*
* @param non-empty-string $haystack
*/
public function strEndsWithNonEmpty(string $haystack, string $needle): void
{
if (str_ends_with($haystack, $needle)) {
assertType('non-empty-string', $haystack);
assertType('string', $needle);
}
}

/**
* array_key_exists with non-constant key on a non-empty-array should not report always-true.
*
* @param non-empty-array<string, int> $array
*/
public function arrayKeyExistsNonEmpty(array $array, string $key): void
{
if (array_key_exists($key, $array)) {
assertType('non-empty-array<string, int>', $array);
}
}

/**
* @phpstan-assert-if-true =non-empty-string $foo
*/
public function isValid(string $foo): bool
{
return $foo !== '';
}

public function equalityAssertDuplicate(string $task): void
{
if ($this->isValid($task)) {
assertType('non-empty-string', $task);
if ($this->isValid($task)) { // reported as always-true
assertType('non-empty-string', $task);
}
}
}

/**
* @phpstan-assert =non-empty-string $foo
*/
public function assertValid(string $foo): void
{
if ($foo === '') {
throw new \Exception();
}
}

public function voidAssertDuplicate(string $task): void
{
$this->assertValid($task);
assertType('non-empty-string', $task);
$this->assertValid($task); // reported as always-true
assertType('non-empty-string', $task);
}

public function realpathElvis(string $fileName): void
{
$fileName = realpath($fileName) ?: $fileName;
assertType('string', $fileName);
}

/** @param list<string> $paths */
public function realpathElvisWithLoop(string $fileName, array $paths): void
{
$fileName = realpath($fileName) ?: $fileName;
assertType('string', $fileName);

foreach ($paths as $path) {
if (str_starts_with($fileName, $path)) {
assertType('string', $fileName);
}
}
}

/**
* Duplicate array_key_exists after an early-continue narrows the negated
* call to false, while the non-negated call stays bool.
*
* @param array<string,string|array<int,string>> $theInput
* @phpstan-param array{'name':string,'owners':array<int,string>} $theInput
* @param array<int,string> $theTags
*/
public function arrayKeyExistsDuplicateInLoop(array $theInput, array $theTags): void
{
foreach ($theTags as $tag) {
if (!array_key_exists($tag, $theInput)) {
continue;
}
assertType('false', !array_key_exists($tag, $theInput));
assertType('bool', array_key_exists($tag, $theInput)); // could be true
}
}

}
Loading
Loading