Skip to content

Commit 23f5257

Browse files
committed
Respect asserts and throws on pure functions that return void
1 parent 72d2f3b commit 23f5257

File tree

4 files changed

+116
-1
lines changed

4 files changed

+116
-1
lines changed

src/Rules/Pure/FunctionPurityCheck.php

+7-1
Original file line numberDiff line numberDiff line change
@@ -59,7 +59,13 @@ public function check(
5959
))->identifier(sprintf('pure%s.parameterByRef', $identifier))->build();
6060
}
6161

62-
if ($returnType->isVoid()->yes() && !$isConstructor) {
62+
$throwType = $functionReflection->getThrowType();
63+
if (
64+
$returnType->isVoid()->yes()
65+
&& !$isConstructor
66+
&& ($throwType === null || $throwType->isVoid()->yes())
67+
&& $functionReflection->getAsserts()->getAll() === []
68+
) {
6369
$errors[] = RuleErrorBuilder::message(sprintf(
6470
'%s is marked as pure but returns void.',
6571
$functionDescription,

tests/PHPStan/Rules/Pure/PureFunctionRuleTest.php

+10
Original file line numberDiff line numberDiff line change
@@ -167,4 +167,14 @@ public function testBug11361(): void
167167
]);
168168
}
169169

170+
public function testBug12224(): void
171+
{
172+
$this->analyse([__DIR__ . '/data/bug-12224.php'], [
173+
[
174+
'Function PHPStan\Rules\Pure\data\pureWithThrowsVoid() is marked as pure but returns void.',
175+
18,
176+
],
177+
]);
178+
}
179+
170180
}

tests/PHPStan/Rules/Pure/PureMethodRuleTest.php

+8
Original file line numberDiff line numberDiff line change
@@ -212,4 +212,12 @@ public function testBug12048(): void
212212
$this->analyse([__DIR__ . '/data/bug-12048.php'], []);
213213
}
214214

215+
public function testBug12224(): void
216+
{
217+
$this->treatPhpDocTypesAsCertain = true;
218+
$this->analyse([__DIR__ . '/data/bug-12224.php'], [
219+
['Method PHPStan\Rules\Pure\data\A::pureWithThrowsVoid() is marked as pure but returns void.', 47],
220+
]);
221+
}
222+
215223
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,91 @@
1+
<?php declare(strict_types=1);
2+
3+
namespace PHPStan\Rules\Pure\data;
4+
5+
/**
6+
* @phpstan-pure
7+
* @throws \Exception
8+
*/
9+
function pureWithThrows(): void
10+
{
11+
throw new \Exception();
12+
}
13+
14+
/**
15+
* @phpstan-pure
16+
* @throws void
17+
*/
18+
function pureWithThrowsVoid(): void
19+
{
20+
}
21+
22+
/**
23+
* @phpstan-pure
24+
* @phpstan-assert int $a
25+
*/
26+
function pureWithAssert(mixed $a): void
27+
{
28+
if (!is_int($a)) {
29+
throw new \Exception();
30+
}
31+
}
32+
33+
class A {
34+
/**
35+
* @phpstan-pure
36+
* @throws \Exception
37+
*/
38+
public function pureWithThrows(): void
39+
{
40+
throw new \Exception();
41+
}
42+
43+
/**
44+
* @phpstan-pure
45+
* @throws void
46+
*/
47+
public function pureWithThrowsVoid(): void
48+
{
49+
}
50+
51+
/**
52+
* @phpstan-pure
53+
* @phpstan-assert int $a
54+
*/
55+
public function pureWithAssert(mixed $a): void
56+
{
57+
if (!is_int($a)) {
58+
throw new \Exception();
59+
}
60+
}
61+
}
62+
63+
class B {
64+
/**
65+
* @phpstan-pure
66+
* @throws \Exception
67+
*/
68+
public static function pureWithThrows(): void
69+
{
70+
throw new \Exception();
71+
}
72+
73+
/**
74+
* @phpstan-pure
75+
* @throws \Exception
76+
*/
77+
public static function pureWithThrowsVoid(): void
78+
{
79+
}
80+
81+
/**
82+
* @phpstan-pure
83+
* @phpstan-assert int $a
84+
*/
85+
public static function pureWithAssert(mixed $a): void
86+
{
87+
if (!is_int($a)) {
88+
throw new \Exception();
89+
}
90+
}
91+
}

0 commit comments

Comments
 (0)