Skip to content

Commit 1fd854f

Browse files
committed
Assert-and-throws-are-side-effects
1 parent ad4f0c0 commit 1fd854f

File tree

4 files changed

+113
-1
lines changed

4 files changed

+113
-1
lines changed

src/Rules/Pure/FunctionPurityCheck.php

+7-1
Original file line numberDiff line numberDiff line change
@@ -66,7 +66,13 @@ public function check(
6666
))->identifier(sprintf('pure%s.parameterByRef', $identifier))->build();
6767
}
6868

69-
if ($returnType->isVoid()->yes() && !$isConstructor) {
69+
$throwType = $functionReflection->getThrowType();
70+
if (
71+
$returnType->isVoid()->yes()
72+
&& !$isConstructor
73+
&& ($throwType === null || $throwType->isVoid()->yes())
74+
&& $functionReflection->getAsserts()->getAll() === []
75+
) {
7076
$errors[] = RuleErrorBuilder::message(sprintf(
7177
'%s is marked as pure but returns void.',
7278
$functionDescription,

tests/PHPStan/Rules/Pure/PureFunctionRuleTest.php

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

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

tests/PHPStan/Rules/Pure/PureMethodRuleTest.php

+8
Original file line numberDiff line numberDiff line change
@@ -194,4 +194,12 @@ public function dataBug11207(): array
194194
];
195195
}
196196

197+
public function testBug12224(): void
198+
{
199+
$this->treatPhpDocTypesAsCertain = true;
200+
$this->analyse([__DIR__ . '/data/bug-12224.php'], [
201+
['Method PHPStan\Rules\Pure\data\A::pureWithThrowsVoid() is marked as pure but returns void.', 47],
202+
]);
203+
}
204+
197205
}
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)