Skip to content

Commit 9ebc315

Browse files
committed
Call GenericObjectTypeCheck from IncompatibleSelfOutTypeRule
1 parent d389ae0 commit 9ebc315

File tree

4 files changed

+101
-4
lines changed

4 files changed

+101
-4
lines changed

src/PhpDoc/StubValidator.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -201,7 +201,7 @@ private function getRuleRegistry(Container $container): RuleRegistry
201201
$container->getParameter('featureToggles')['invalidPhpDocTagLine'],
202202
),
203203
new IncompatibleParamImmediatelyInvokedCallableRule($fileTypeMapper),
204-
new IncompatibleSelfOutTypeRule($unresolvableTypeHelper),
204+
new IncompatibleSelfOutTypeRule($unresolvableTypeHelper, $genericObjectTypeCheck),
205205
new IncompatibleClassConstantPhpDocTypeRule($genericObjectTypeCheck, $unresolvableTypeHelper),
206206
new InvalidThrowsPhpDocValueRule($fileTypeMapper),
207207

src/Rules/PhpDoc/IncompatibleSelfOutTypeRule.php

+36-2
Original file line numberDiff line numberDiff line change
@@ -4,11 +4,14 @@
44

55
use PhpParser\Node;
66
use PHPStan\Analyser\Scope;
7+
use PHPStan\Internal\SprintfHelper;
78
use PHPStan\Node\InClassMethodNode;
9+
use PHPStan\Rules\Generics\GenericObjectTypeCheck;
810
use PHPStan\Rules\Rule;
911
use PHPStan\Rules\RuleErrorBuilder;
1012
use PHPStan\Type\ObjectType;
1113
use PHPStan\Type\VerbosityLevel;
14+
use function array_merge;
1215
use function sprintf;
1316

1417
/**
@@ -17,7 +20,10 @@
1720
final class IncompatibleSelfOutTypeRule implements Rule
1821
{
1922

20-
public function __construct(private UnresolvableTypeHelper $unresolvableTypeHelper)
23+
public function __construct(
24+
private UnresolvableTypeHelper $unresolvableTypeHelper,
25+
private GenericObjectTypeCheck $genericObjectTypeCheck,
26+
)
2127
{
2228
}
2329

@@ -63,7 +69,35 @@ public function processNode(Node $node, Scope $scope): array
6369
))->identifier('parameter.unresolvableType')->build();
6470
}
6571

66-
return $errors;
72+
$escapedTagName = SprintfHelper::escapeFormatString('@phpstan-self-out');
73+
74+
return array_merge($errors, $this->genericObjectTypeCheck->check(
75+
$selfOutType,
76+
sprintf(
77+
'PHPDoc tag %s contains generic type %%s but %%s %%s is not generic.',
78+
$escapedTagName,
79+
),
80+
sprintf(
81+
'Generic type %%s in PHPDoc tag %s does not specify all template types of %%s %%s: %%s',
82+
$escapedTagName,
83+
),
84+
sprintf(
85+
'Generic type %%s in PHPDoc tag %s specifies %%d template types, but %%s %%s supports only %%d: %%s',
86+
$escapedTagName,
87+
),
88+
sprintf(
89+
'Type %%s in generic type %%s in PHPDoc tag %s is not subtype of template type %%s of %%s %%s.',
90+
$escapedTagName,
91+
),
92+
sprintf(
93+
'Call-site variance of %%s in generic type %%s in PHPDoc tag %s is in conflict with %%s template type %%s of %%s %%s.',
94+
$escapedTagName,
95+
),
96+
sprintf(
97+
'Call-site variance of %%s in generic type %%s in PHPDoc tag %s is redundant, template type %%s of %%s %%s has the same variance.',
98+
$escapedTagName,
99+
),
100+
));
67101
}
68102

69103
}

tests/PHPStan/Rules/PhpDoc/IncompatibleSelfOutTypeRuleTest.php

+18-1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
namespace PHPStan\Rules\PhpDoc;
44

5+
use PHPStan\Rules\Generics\GenericObjectTypeCheck;
56
use PHPStan\Rules\Rule;
67
use PHPStan\Testing\RuleTestCase;
78

@@ -13,7 +14,7 @@ class IncompatibleSelfOutTypeRuleTest extends RuleTestCase
1314

1415
protected function getRule(): Rule
1516
{
16-
return new IncompatibleSelfOutTypeRule(new UnresolvableTypeHelper());
17+
return new IncompatibleSelfOutTypeRule(new UnresolvableTypeHelper(), new GenericObjectTypeCheck());
1718
}
1819

1920
public function testRule(): void
@@ -39,6 +40,22 @@ public function testRule(): void
3940
'PHPDoc tag @phpstan-self-out for method IncompatibleSelfOutType\Foo::doBar() contains unresolvable type.',
4041
54,
4142
],
43+
[
44+
'PHPDoc tag @phpstan-self-out contains generic type IncompatibleSelfOutType\GenericCheck<int> but class IncompatibleSelfOutType\GenericCheck is not generic.',
45+
67,
46+
],
47+
[
48+
'Generic type IncompatibleSelfOutType\GenericCheck2<InvalidArgumentException> in PHPDoc tag @phpstan-self-out does not specify all template types of class IncompatibleSelfOutType\GenericCheck2: T, U',
49+
84,
50+
],
51+
[
52+
'Generic type IncompatibleSelfOutType\GenericCheck2<InvalidArgumentException, int<1, max>, string> in PHPDoc tag @phpstan-self-out specifies 3 template types, but class IncompatibleSelfOutType\GenericCheck2 supports only 2: T, U',
53+
92,
54+
],
55+
[
56+
'Type string in generic type IncompatibleSelfOutType\GenericCheck2<InvalidArgumentException, string> in PHPDoc tag @phpstan-self-out is not subtype of template type U of int of class IncompatibleSelfOutType\GenericCheck2.',
57+
100,
58+
],
4259
]);
4360
}
4461

tests/PHPStan/Rules/PhpDoc/data/incompatible-self-out-type.php

+46
Original file line numberDiff line numberDiff line change
@@ -57,3 +57,49 @@ public function doBar(): void
5757
}
5858

5959
}
60+
61+
class GenericCheck
62+
{
63+
64+
/**
65+
* @phpstan-self-out self<int>
66+
*/
67+
public function doFoo(): void
68+
{
69+
70+
}
71+
72+
}
73+
74+
/**
75+
* @template T of \Exception
76+
* @template U of int
77+
*/
78+
class GenericCheck2
79+
{
80+
81+
/**
82+
* @phpstan-self-out self<\InvalidArgumentException>
83+
*/
84+
public function doFoo(): void
85+
{
86+
87+
}
88+
89+
/**
90+
* @phpstan-self-out self<\InvalidArgumentException, positive-int, string>
91+
*/
92+
public function doFoo2(): void
93+
{
94+
95+
}
96+
97+
/**
98+
* @phpstan-self-out self<\InvalidArgumentException, string>
99+
*/
100+
public function doFoo3(): void
101+
{
102+
103+
}
104+
105+
}

0 commit comments

Comments
 (0)