Skip to content

Commit 5f0b1cc

Browse files
committed
Too-wide return type - do not report void in PHPDoc union type
1 parent b1b7782 commit 5f0b1cc

6 files changed

+128
-8
lines changed

src/Rules/TooWideTypehints/TooWideFunctionReturnTypehintRule.php

+2-4
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
use PHPStan\Type\TypeUtils;
1212
use PHPStan\Type\UnionType;
1313
use PHPStan\Type\VerbosityLevel;
14+
use PHPStan\Type\VoidType;
1415
use function count;
1516
use function sprintf;
1617

@@ -48,16 +49,13 @@ public function processNode(Node $node, Scope $scope): array
4849
foreach ($returnStatements as $returnStatement) {
4950
$returnNode = $returnStatement->getReturnNode();
5051
if ($returnNode->expr === null) {
52+
$returnTypes[] = new VoidType();
5153
continue;
5254
}
5355

5456
$returnTypes[] = $returnStatement->getScope()->getType($returnNode->expr);
5557
}
5658

57-
if (count($returnTypes) === 0) {
58-
return [];
59-
}
60-
6159
$returnType = TypeCombinator::union(...$returnTypes);
6260

6361
$messages = [];

src/Rules/TooWideTypehints/TooWideMethodReturnTypehintRule.php

+2-4
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@
1212
use PHPStan\Type\TypeUtils;
1313
use PHPStan\Type\UnionType;
1414
use PHPStan\Type\VerbosityLevel;
15+
use PHPStan\Type\VoidType;
1516
use function count;
1617
use function sprintf;
1718

@@ -74,16 +75,13 @@ public function processNode(Node $node, Scope $scope): array
7475
foreach ($returnStatements as $returnStatement) {
7576
$returnNode = $returnStatement->getReturnNode();
7677
if ($returnNode->expr === null) {
78+
$returnTypes[] = new VoidType();
7779
continue;
7880
}
7981

8082
$returnTypes[] = $returnStatement->getScope()->getType($returnNode->expr);
8183
}
8284

83-
if (count($returnTypes) === 0) {
84-
return [];
85-
}
86-
8785
$returnType = TypeCombinator::union(...$returnTypes);
8886
if (
8987
!$method->isPrivate()

tests/PHPStan/Rules/TooWideTypehints/TooWideFunctionReturnTypehintRuleTest.php

+10
Original file line numberDiff line numberDiff line change
@@ -51,4 +51,14 @@ public function testRule(): void
5151
]);
5252
}
5353

54+
public function testBug11980(): void
55+
{
56+
$this->analyse([__DIR__ . '/data/bug-11980-function.php'], [
57+
[
58+
'Function Bug11980Function\process2() never returns void so it can be removed from the return type.',
59+
34,
60+
],
61+
]);
62+
}
63+
5464
}

tests/PHPStan/Rules/TooWideTypehints/TooWideMethodReturnTypehintRuleTest.php

+12
Original file line numberDiff line numberDiff line change
@@ -248,4 +248,16 @@ public function testAlwaysCheckFinal(bool $checkProtectedAndPublicMethods, bool
248248
$this->analyse([__DIR__ . '/data/method-too-wide-return-always-check-final.php'], $expectedErrors);
249249
}
250250

251+
public function testBug11980(): void
252+
{
253+
$this->checkProtectedAndPublicMethods = true;
254+
$this->alwaysCheckFinal = true;
255+
$this->analyse([__DIR__ . '/data/bug-11980.php'], [
256+
[
257+
'Method Bug11980\Demo::process2() never returns void so it can be removed from the return type.',
258+
37,
259+
],
260+
]);
261+
}
262+
251263
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
<?php
2+
3+
namespace Bug11980Function;
4+
5+
/**
6+
* @param array<int, array<string, int>> $tokens
7+
* @param int $stackPtr
8+
*
9+
* @return int|void
10+
*/
11+
function process($tokens, $stackPtr)
12+
{
13+
if (empty($tokens[$stackPtr]['nested_parenthesis']) === false) {
14+
// Not a stand-alone statement.
15+
return;
16+
}
17+
18+
$end = 10;
19+
20+
if ($tokens[$end]['code'] !== 10
21+
&& $tokens[$end]['code'] !== 20
22+
) {
23+
// Not a stand-alone statement.
24+
return $end;
25+
}
26+
}
27+
28+
/**
29+
* @param array<int, array<string, int>> $tokens
30+
* @param int $stackPtr
31+
*
32+
* @return int|void
33+
*/
34+
function process2($tokens, $stackPtr)
35+
{
36+
if (empty($tokens[$stackPtr]['nested_parenthesis']) === false) {
37+
// Not a stand-alone statement.
38+
return null;
39+
}
40+
41+
$end = 10;
42+
43+
if ($tokens[$end]['code'] !== 10
44+
&& $tokens[$end]['code'] !== 20
45+
) {
46+
// Not a stand-alone statement.
47+
return $end;
48+
}
49+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
<?php
2+
3+
namespace Bug11980;
4+
5+
final class Demo
6+
{
7+
8+
/**
9+
* @param array<int, array<string, int>> $tokens
10+
* @param int $stackPtr
11+
*
12+
* @return int|void
13+
*/
14+
public function process($tokens, $stackPtr)
15+
{
16+
if (empty($tokens[$stackPtr]['nested_parenthesis']) === false) {
17+
// Not a stand-alone statement.
18+
return;
19+
}
20+
21+
$end = 10;
22+
23+
if ($tokens[$end]['code'] !== 10
24+
&& $tokens[$end]['code'] !== 20
25+
) {
26+
// Not a stand-alone statement.
27+
return $end;
28+
}
29+
}
30+
31+
/**
32+
* @param array<int, array<string, int>> $tokens
33+
* @param int $stackPtr
34+
*
35+
* @return int|void
36+
*/
37+
public function process2($tokens, $stackPtr)
38+
{
39+
if (empty($tokens[$stackPtr]['nested_parenthesis']) === false) {
40+
// Not a stand-alone statement.
41+
return null;
42+
}
43+
44+
$end = 10;
45+
46+
if ($tokens[$end]['code'] !== 10
47+
&& $tokens[$end]['code'] !== 20
48+
) {
49+
// Not a stand-alone statement.
50+
return $end;
51+
}
52+
}
53+
}

0 commit comments

Comments
 (0)