diff --git a/src/Reflection/InitializerExprTypeResolver.php b/src/Reflection/InitializerExprTypeResolver.php index f42bf7e6ad..8c9f60b033 100644 --- a/src/Reflection/InitializerExprTypeResolver.php +++ b/src/Reflection/InitializerExprTypeResolver.php @@ -506,7 +506,7 @@ public function getBitwiseAndType(Expr $left, Expr $right, callable $getTypeCall $rightType = $getTypeCallback($right); if ($leftType instanceof NeverType || $rightType instanceof NeverType) { - return new NeverType(); + return $this->getNeverType($leftType, $rightType); } $leftTypes = TypeUtils::getConstantScalars($leftType); @@ -573,7 +573,7 @@ public function getBitwiseOrType(Expr $left, Expr $right, callable $getTypeCallb $rightType = $getTypeCallback($right); if ($leftType instanceof NeverType || $rightType instanceof NeverType) { - return new NeverType(); + return $this->getNeverType($leftType, $rightType); } $leftTypes = TypeUtils::getConstantScalars($leftType); @@ -630,7 +630,7 @@ public function getBitwiseXorType(Expr $left, Expr $right, callable $getTypeCall $rightType = $getTypeCallback($right); if ($leftType instanceof NeverType || $rightType instanceof NeverType) { - return new NeverType(); + return $this->getNeverType($leftType, $rightType); } $leftTypes = TypeUtils::getConstantScalars($leftType); @@ -687,7 +687,7 @@ public function getSpaceshipType(Expr $left, Expr $right, callable $getTypeCallb $callbackRightType = $getTypeCallback($right); if ($callbackLeftType instanceof NeverType || $callbackRightType instanceof NeverType) { - return new NeverType(); + return $this->getNeverType($callbackLeftType, $callbackRightType); } $leftTypes = TypeUtils::getConstantScalars($callbackLeftType); @@ -780,7 +780,7 @@ public function getModType(Expr $left, Expr $right, callable $getTypeCallback): $rightType = $getTypeCallback($right); if ($leftType instanceof NeverType || $rightType instanceof NeverType) { - return new NeverType(); + return $this->getNeverType($leftType, $rightType); } $leftTypes = TypeUtils::getConstantScalars($leftType); @@ -881,7 +881,7 @@ public function getPlusType(Expr $left, Expr $right, callable $getTypeCallback): $rightType = $getTypeCallback($right); if ($leftType instanceof NeverType || $rightType instanceof NeverType) { - return new NeverType(); + return $this->getNeverType($leftType, $rightType); } $leftTypes = TypeUtils::getConstantScalars($leftType); @@ -1200,7 +1200,7 @@ public function getShiftLeftType(Expr $left, Expr $right, callable $getTypeCallb $rightType = $getTypeCallback($right); if ($leftType instanceof NeverType || $rightType instanceof NeverType) { - return new NeverType(); + return $this->getNeverType($leftType, $rightType); } $leftTypes = TypeUtils::getConstantScalars($leftType); @@ -1257,7 +1257,7 @@ public function getShiftRightType(Expr $left, Expr $right, callable $getTypeCall $rightType = $getTypeCallback($right); if ($leftType instanceof NeverType || $rightType instanceof NeverType) { - return new NeverType(); + return $this->getNeverType($leftType, $rightType); } $leftTypes = TypeUtils::getConstantScalars($leftType); @@ -1496,7 +1496,7 @@ private function resolveCommonMath(Expr\BinaryOp $expr, Type $leftType, Type $ri return new ErrorType(); } if ($leftNumberType instanceof NeverType || $rightNumberType instanceof NeverType) { - return new NeverType(); + return $this->getNeverType($leftNumberType, $rightNumberType); } if ( @@ -1990,4 +1990,16 @@ private function getReflectionProvider(): ReflectionProvider return $this->reflectionProviderProvider->getReflectionProvider(); } + private function getNeverType(Type $leftType, Type $rightType): Type + { + // make sure we don't lose the explicit flag in the process + if ($leftType instanceof NeverType && $leftType->isExplicit()) { + return $leftType; + } + if ($rightType instanceof NeverType && $rightType->isExplicit()) { + return $rightType; + } + return new NeverType(); + } + } diff --git a/tests/PHPStan/Analyser/AnalyserIntegrationTest.php b/tests/PHPStan/Analyser/AnalyserIntegrationTest.php index 85fa14a9ca..6d6300e83c 100644 --- a/tests/PHPStan/Analyser/AnalyserIntegrationTest.php +++ b/tests/PHPStan/Analyser/AnalyserIntegrationTest.php @@ -5,14 +5,20 @@ use Bug4288\MyClass; use Bug4713\Service; use ExtendingKnownClassWithCheck\Foo; +use PhpParser\Node\Expr; +use PhpParser\Node\Scalar\LNumber; +use PhpParser\Node\Scalar\String_; use PHPStan\File\FileHelper; use PHPStan\Reflection\InitializerExprContext; use PHPStan\Reflection\InitializerExprTypeResolver; use PHPStan\Reflection\ParametersAcceptorSelector; use PHPStan\Reflection\SignatureMap\SignatureMapProvider; +use PHPStan\ShouldNotHappenException; use PHPStan\Testing\PHPStanTestCase; use PHPStan\Type\Constant\ConstantIntegerType; use PHPStan\Type\Constant\ConstantStringType; +use PHPStan\Type\NeverType; +use PHPStan\Type\Type; use function extension_loaded; use function restore_error_handler; use function sprintf; @@ -1075,6 +1081,118 @@ public function testBug8503(): void $this->assertNoErrors($errors); } + /** + * @dataProvider dataExplicitNever + * + * @param class-string $resultClass + * @param callable(Expr): Type $callback + */ + public function testExplicitNever(Expr $left, Expr $right, callable $callback, string $resultClass, ?bool $resultIsExplicit = null): void + { + $initializerExprTypeResolver = self::getContainer()->getByType(InitializerExprTypeResolver::class); + + $result = $initializerExprTypeResolver->getPlusType( + $left, + $right, + $callback, + ); + $this->assertInstanceOf($resultClass, $result); + + if (!($result instanceof NeverType)) { + return; + } + + if ($resultIsExplicit === null) { + throw new ShouldNotHappenException(); + } + $this->assertSame($resultIsExplicit, $result->isExplicit()); + } + + public function dataExplicitNever(): iterable + { + yield [ + new LNumber(1), + new String_('foo'), + static function (Expr $expr): Type { + if ($expr instanceof LNumber) { + return new ConstantIntegerType(1); + } + return new NeverType(true); + }, + NeverType::class, + true, + ]; + yield [ + new String_('foo'), + new LNumber(1), + static function (Expr $expr): Type { + if ($expr instanceof LNumber) { + return new ConstantIntegerType(1); + } + return new NeverType(true); + }, + NeverType::class, + true, + ]; + + yield [ + new LNumber(1), + new String_('foo'), + static function (Expr $expr): Type { + if ($expr instanceof LNumber) { + return new ConstantIntegerType(1); + } + return new NeverType(false); + }, + NeverType::class, + false, + ]; + yield [ + new String_('foo'), + new LNumber(1), + static function (Expr $expr): Type { + if ($expr instanceof LNumber) { + return new ConstantIntegerType(1); + } + return new NeverType(false); + }, + NeverType::class, + false, + ]; + + yield [ + new String_('foo'), + new LNumber(1), + static function (Expr $expr): Type { + if ($expr instanceof LNumber) { + return new NeverType(true); + } + return new NeverType(false); + }, + NeverType::class, + true, + ]; + yield [ + new LNumber(1), + new String_('foo'), + static function (Expr $expr): Type { + if ($expr instanceof LNumber) { + return new NeverType(true); + } + return new NeverType(false); + }, + NeverType::class, + true, + ]; + + yield [ + new LNumber(1), + new LNumber(1), + static fn (Expr $expr): Type => new ConstantIntegerType(1), + ConstantIntegerType::class, + ]; + } + /** * @param string[]|null $allAnalysedFiles * @return Error[]