Skip to content

Commit 640a0ba

Browse files
committed
Assert-and-throws-are-side-effects
1 parent 33dc757 commit 640a0ba

File tree

4 files changed

+109
-3
lines changed

4 files changed

+109
-3
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

+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

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

197-
public function testBug12048(): void
197+
public function testBug12224(): void
198198
{
199199
$this->treatPhpDocTypesAsCertain = true;
200-
$this->analyse([__DIR__ . '/data/bug-12048.php'], []);
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+
]);
201203
}
202204

203205
}
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)