From 3b1c15fe4a49f01a247df92a55162adb1416c46d Mon Sep 17 00:00:00 2001 From: Markus Staab Date: Wed, 4 Jan 2023 12:17:06 +0100 Subject: [PATCH] max()/min() should expect non-empty-array --- resources/functionMap.php | 4 +-- .../Analyser/NodeScopeResolverTest.php | 6 ++++ tests/PHPStan/Analyser/data/bug-7239-php8.php | 36 +++++++++++++++++++ tests/PHPStan/Analyser/data/bug-7239.php | 36 +++++++++++++++++++ .../CallToFunctionParametersRuleTest.php | 30 ++++++++++++++++ 5 files changed, 110 insertions(+), 2 deletions(-) create mode 100644 tests/PHPStan/Analyser/data/bug-7239-php8.php create mode 100644 tests/PHPStan/Analyser/data/bug-7239.php diff --git a/resources/functionMap.php b/resources/functionMap.php index 0b25c6784b..cd38e9c294 100644 --- a/resources/functionMap.php +++ b/resources/functionMap.php @@ -6143,7 +6143,7 @@ 'mapObj::zoomPoint' => ['int', 'nZoomFactor'=>'int', 'oPixelPos'=>'pointObj', 'nImageWidth'=>'int', 'nImageHeight'=>'int', 'oGeorefExt'=>'rectObj'], 'mapObj::zoomRectangle' => ['int', 'oPixelExt'=>'rectObj', 'nImageWidth'=>'int', 'nImageHeight'=>'int', 'oGeorefExt'=>'rectObj'], 'mapObj::zoomScale' => ['int', 'nScaleDenom'=>'float', 'oPixelPos'=>'pointObj', 'nImageWidth'=>'int', 'nImageHeight'=>'int', 'oGeorefExt'=>'rectObj', 'oMaxGeorefExt'=>'rectObj'], -'max' => ['', '...arg1'=>'array'], +'max' => ['', '...arg1'=>'non-empty-array'], 'max\'1' => ['', 'arg1'=>'', 'arg2'=>'', '...args='=>''], 'maxdb::__construct' => ['void', 'host='=>'string', 'username='=>'string', 'passwd='=>'string', 'dbname='=>'string', 'port='=>'int', 'socket='=>'string'], 'maxdb::affected_rows' => ['int', 'link'=>''], @@ -6513,7 +6513,7 @@ 'mhash_keygen_s2k' => ['string|false', 'hash'=>'int', 'input_password'=>'string', 'salt'=>'string', 'bytes'=>'int'], 'microtime' => ['mixed', 'get_as_float='=>'bool'], 'mime_content_type' => ['string|false', 'filename_or_stream'=>'string|resource'], -'min' => ['', '...arg1'=>'array'], +'min' => ['', '...arg1'=>'non-empty-array'], 'min\'1' => ['', 'arg1'=>'', 'arg2'=>'', '...args='=>''], 'ming_keypress' => ['int', 'char'=>'string'], 'ming_setcubicthreshold' => ['void', 'threshold'=>'int'], diff --git a/tests/PHPStan/Analyser/NodeScopeResolverTest.php b/tests/PHPStan/Analyser/NodeScopeResolverTest.php index 43efba4291..e363eea2da 100644 --- a/tests/PHPStan/Analyser/NodeScopeResolverTest.php +++ b/tests/PHPStan/Analyser/NodeScopeResolverTest.php @@ -1169,6 +1169,12 @@ public function dataFileAsserts(): iterable yield from $this->gatherAssertTypes(__DIR__ . '/../Rules/DeadCode/data/bug-8620.php'); yield from $this->gatherAssertTypes(__DIR__ . '/data/bug-8635.php'); yield from $this->gatherAssertTypes(__DIR__ . '/data/bug-8625.php'); + if (PHP_VERSION_ID >= 80000) { + yield from $this->gatherAssertTypes(__DIR__ . '/data/bug-7239-php8.php'); + } + if (PHP_VERSION_ID < 80000) { + yield from $this->gatherAssertTypes(__DIR__ . '/data/bug-7239.php'); + } yield from $this->gatherAssertTypes(__DIR__ . '/data/bug-8621.php'); } diff --git a/tests/PHPStan/Analyser/data/bug-7239-php8.php b/tests/PHPStan/Analyser/data/bug-7239-php8.php new file mode 100644 index 0000000000..11103c7ded --- /dev/null +++ b/tests/PHPStan/Analyser/data/bug-7239-php8.php @@ -0,0 +1,36 @@ + 0) { + assertType('mixed', max($arr)); + assertType('mixed', min($arr)); + } else { + assertType('*ERROR*', max($arr)); + assertType('*ERROR*', min($arr)); + } + + assertType('array', max([], $arr)); + assertType('array', min([], $arr)); + + if (count($strings) > 0) { + assertType('string', max($strings)); + assertType('string', min($strings)); + } else { + assertType('*ERROR*', max($strings)); + assertType('*ERROR*', min($strings)); + } + } +} diff --git a/tests/PHPStan/Analyser/data/bug-7239.php b/tests/PHPStan/Analyser/data/bug-7239.php new file mode 100644 index 0000000000..b31a69e785 --- /dev/null +++ b/tests/PHPStan/Analyser/data/bug-7239.php @@ -0,0 +1,36 @@ + 0) { + assertType('mixed', max($arr)); + assertType('mixed', min($arr)); + } else { + assertType('false', max($arr)); + assertType('false', min($arr)); + } + + assertType('array', max([], $arr)); + assertType('array', min([], $arr)); + + if (count($strings) > 0) { + assertType('string', max($strings)); + assertType('string', min($strings)); + } else { + assertType('false', max($strings)); + assertType('false', min($strings)); + } + } +} diff --git a/tests/PHPStan/Rules/Functions/CallToFunctionParametersRuleTest.php b/tests/PHPStan/Rules/Functions/CallToFunctionParametersRuleTest.php index ad444f2c76..cbdaca0c54 100644 --- a/tests/PHPStan/Rules/Functions/CallToFunctionParametersRuleTest.php +++ b/tests/PHPStan/Rules/Functions/CallToFunctionParametersRuleTest.php @@ -1230,4 +1230,34 @@ public function testBug8389(): void $this->analyse([__DIR__ . '/data/bug-8389.php'], []); } + public function testBug7239(): void + { + $this->analyse([__DIR__ . '/../../Analyser/data/bug-7239.php'], [ + [ + 'Parameter #1 ...$arg1 of function max expects non-empty-array, array{} given.', + 14, + ], + [ + 'Parameter #1 ...$arg1 of function min expects non-empty-array, array{} given.', + 15, + ], + [ + 'Parameter #1 ...$arg1 of function max expects non-empty-array, array{} given.', + 21, + ], + [ + 'Parameter #1 ...$arg1 of function min expects non-empty-array, array{} given.', + 22, + ], + [ + 'Parameter #1 ...$arg1 of function max expects non-empty-array, array{} given.', + 32, + ], + [ + 'Parameter #1 ...$arg1 of function min expects non-empty-array, array{} given.', + 33, + ], + ]); + } + }