@@ -53,6 +53,22 @@ public function __construct(
53
53
* @return list<IdentifierRuleError>
54
54
*/
55
55
public function check (ClassReflection $ reflection , ClassLike $ node ): array
56
+ {
57
+ $ errors = [];
58
+ foreach ($ this ->checkInTraitDefinitionContext ($ reflection ) as $ error ) {
59
+ $ errors [] = $ error ;
60
+ }
61
+ foreach ($ this ->checkInTraitUseContext ($ reflection , $ reflection , $ node ) as $ error ) {
62
+ $ errors [] = $ error ;
63
+ }
64
+
65
+ return $ errors ;
66
+ }
67
+
68
+ /**
69
+ * @return list<IdentifierRuleError>
70
+ */
71
+ public function checkInTraitDefinitionContext (ClassReflection $ reflection ): array
56
72
{
57
73
$ phpDoc = $ reflection ->getResolvedPhpDoc ();
58
74
if ($ phpDoc === null ) {
@@ -69,7 +85,7 @@ public function check(ClassReflection $reflection, ClassLike $node): array
69
85
};
70
86
71
87
$ errors = [];
72
- $ className = $ reflection ->getName ();
88
+ $ className = $ reflection ->getDisplayName ();
73
89
74
90
$ importedAliases = [];
75
91
@@ -162,78 +178,86 @@ public function check(ClassReflection $reflection, ClassLike $node): array
162
178
}
163
179
164
180
$ resolvedType = $ typeAliasTag ->getTypeAlias ()->resolve ($ this ->typeNodeResolver );
165
- $ foundError = false ;
166
- TypeTraverser::map ($ resolvedType , static function (Type $ type , callable $ traverse ) use (&$ errors , &$ foundError , $ aliasName ): Type {
167
- if ($ foundError ) {
168
- return $ type ;
169
- }
170
-
171
- if ($ type instanceof CircularTypeAliasErrorType) {
172
- $ errors [] = RuleErrorBuilder::message (sprintf ('Circular definition detected in type alias %s. ' , $ aliasName ))
173
- ->identifier ('typeAlias.circular ' )
174
- ->build ();
175
- $ foundError = true ;
176
- return $ type ;
177
- }
178
-
179
- if ($ type instanceof ErrorType) {
180
- $ errors [] = RuleErrorBuilder::message (sprintf ('Invalid type definition detected in type alias %s. ' , $ aliasName ))
181
- ->identifier ('typeAlias.invalidType ' )
182
- ->build ();
183
- $ foundError = true ;
184
- return $ type ;
185
- }
186
-
187
- return $ traverse ($ type );
188
- });
189
-
190
- if ($ foundError ) {
181
+ if ($ this ->hasErrorType ($ resolvedType , $ aliasName , $ errors )) {
191
182
continue ;
192
183
}
193
184
194
185
if (!$ this ->absentTypeChecks ) {
195
186
continue ;
196
187
}
197
188
198
- if ($ this ->checkMissingTypehints ) {
199
- foreach ($ this ->missingTypehintCheck ->getIterableTypesWithMissingValueTypehint ($ resolvedType ) as $ iterableType ) {
200
- $ iterableTypeDescription = $ iterableType ->describe (VerbosityLevel::typeOnly ());
201
- $ errors [] = RuleErrorBuilder::message (sprintf (
202
- '%s %s has type alias %s with no value type specified in iterable type %s. ' ,
203
- $ reflection ->getClassTypeDescription (),
204
- $ reflection ->getDisplayName (),
205
- $ aliasName ,
206
- $ iterableTypeDescription ,
207
- ))
208
- ->tip (MissingTypehintCheck::MISSING_ITERABLE_VALUE_TYPE_TIP )
209
- ->identifier ('missingType.iterableValue ' )
210
- ->build ();
211
- }
189
+ if (!$ this ->checkMissingTypehints ) {
190
+ continue ;
191
+ }
212
192
213
- foreach ($ this ->missingTypehintCheck ->getNonGenericObjectTypesWithGenericClass ($ resolvedType ) as [$ name , $ genericTypeNames ]) {
214
- $ errors [] = RuleErrorBuilder::message (sprintf (
215
- '%s %s has type alias %s with generic %s but does not specify its types: %s ' ,
216
- $ reflection ->getClassTypeDescription (),
217
- $ reflection ->getDisplayName (),
218
- $ aliasName ,
219
- $ name ,
220
- implode (', ' , $ genericTypeNames ),
221
- ))
222
- ->identifier ('missingType.generics ' )
223
- ->build ();
224
- }
193
+ foreach ($ this ->missingTypehintCheck ->getIterableTypesWithMissingValueTypehint ($ resolvedType ) as $ iterableType ) {
194
+ $ iterableTypeDescription = $ iterableType ->describe (VerbosityLevel::typeOnly ());
195
+ $ errors [] = RuleErrorBuilder::message (sprintf (
196
+ '%s %s has type alias %s with no value type specified in iterable type %s. ' ,
197
+ $ reflection ->getClassTypeDescription (),
198
+ $ reflection ->getDisplayName (),
199
+ $ aliasName ,
200
+ $ iterableTypeDescription ,
201
+ ))
202
+ ->tip (MissingTypehintCheck::MISSING_ITERABLE_VALUE_TYPE_TIP )
203
+ ->identifier ('missingType.iterableValue ' )
204
+ ->build ();
205
+ }
225
206
226
- foreach ($ this ->missingTypehintCheck ->getCallablesWithMissingSignature ($ resolvedType ) as $ callableType ) {
227
- $ errors [] = RuleErrorBuilder::message (sprintf (
228
- '%s %s has type alias %s with no signature specified for %s. ' ,
229
- $ reflection ->getClassTypeDescription (),
230
- $ reflection ->getDisplayName (),
231
- $ aliasName ,
232
- $ callableType ->describe (VerbosityLevel::typeOnly ()),
233
- ))->identifier ('missingType.callable ' )->build ();
234
- }
207
+ foreach ($ this ->missingTypehintCheck ->getNonGenericObjectTypesWithGenericClass ($ resolvedType ) as [$ name , $ genericTypeNames ]) {
208
+ $ errors [] = RuleErrorBuilder::message (sprintf (
209
+ '%s %s has type alias %s with generic %s but does not specify its types: %s ' ,
210
+ $ reflection ->getClassTypeDescription (),
211
+ $ reflection ->getDisplayName (),
212
+ $ aliasName ,
213
+ $ name ,
214
+ implode (', ' , $ genericTypeNames ),
215
+ ))
216
+ ->identifier ('missingType.generics ' )
217
+ ->build ();
218
+ }
219
+
220
+ foreach ($ this ->missingTypehintCheck ->getCallablesWithMissingSignature ($ resolvedType ) as $ callableType ) {
221
+ $ errors [] = RuleErrorBuilder::message (sprintf (
222
+ '%s %s has type alias %s with no signature specified for %s. ' ,
223
+ $ reflection ->getClassTypeDescription (),
224
+ $ reflection ->getDisplayName (),
225
+ $ aliasName ,
226
+ $ callableType ->describe (VerbosityLevel::typeOnly ()),
227
+ ))->identifier ('missingType.callable ' )->build ();
235
228
}
229
+ }
236
230
231
+ return $ errors ;
232
+ }
233
+
234
+ /**
235
+ * @return list<IdentifierRuleError>
236
+ */
237
+ public function checkInTraitUseContext (
238
+ ClassReflection $ reflection ,
239
+ ClassReflection $ implementingClassReflection ,
240
+ ClassLike $ node ,
241
+ ): array
242
+ {
243
+ if ($ reflection ->getNativeReflection ()->getName () === $ implementingClassReflection ->getName ()) {
244
+ $ phpDoc = $ reflection ->getResolvedPhpDoc ();
245
+ } else {
246
+ $ phpDoc = $ reflection ->getTraitContextResolvedPhpDoc ($ implementingClassReflection );
247
+ }
248
+ if ($ phpDoc === null ) {
249
+ return [];
250
+ }
251
+
252
+ $ errors = [];
253
+
254
+ foreach ($ phpDoc ->getTypeAliasTags () as $ typeAliasTag ) {
255
+ $ aliasName = $ typeAliasTag ->getAliasName ();
256
+ $ resolvedType = $ typeAliasTag ->getTypeAlias ()->resolve ($ this ->typeNodeResolver );
257
+ $ throwawayErrors = [];
258
+ if ($ this ->hasErrorType ($ resolvedType , $ aliasName , $ throwawayErrors )) {
259
+ continue ;
260
+ }
237
261
foreach ($ resolvedType ->getReferencedClasses () as $ class ) {
238
262
if (!$ this ->reflectionProvider ->hasClass ($ class )) {
239
263
$ errors [] = RuleErrorBuilder::message (sprintf ('Type alias %s contains unknown class %s. ' , $ aliasName , $ class ))
@@ -304,4 +328,38 @@ private function isAliasNameValid(string $aliasName, ?NameScope $nameScope): boo
304
328
|| $ aliasNameResolvedType instanceof TemplateType; // aliases take precedence over type parameters, this is reported by other rules using TemplateTypeCheck
305
329
}
306
330
331
+ /**
332
+ * @param list<IdentifierRuleError> $errors
333
+ * @param-out list<IdentifierRuleError> $errors
334
+ */
335
+ private function hasErrorType (Type $ type , string $ aliasName , array &$ errors ): bool
336
+ {
337
+ $ foundError = false ;
338
+ TypeTraverser::map ($ type , static function (Type $ type , callable $ traverse ) use (&$ errors , &$ foundError , $ aliasName ): Type {
339
+ if ($ foundError ) {
340
+ return $ type ;
341
+ }
342
+
343
+ if ($ type instanceof CircularTypeAliasErrorType) {
344
+ $ errors [] = RuleErrorBuilder::message (sprintf ('Circular definition detected in type alias %s. ' , $ aliasName ))
345
+ ->identifier ('typeAlias.circular ' )
346
+ ->build ();
347
+ $ foundError = true ;
348
+ return $ type ;
349
+ }
350
+
351
+ if ($ type instanceof ErrorType) {
352
+ $ errors [] = RuleErrorBuilder::message (sprintf ('Invalid type definition detected in type alias %s. ' , $ aliasName ))
353
+ ->identifier ('typeAlias.invalidType ' )
354
+ ->build ();
355
+ $ foundError = true ;
356
+ return $ type ;
357
+ }
358
+
359
+ return $ traverse ($ type );
360
+ });
361
+
362
+ return $ foundError ;
363
+ }
364
+
307
365
}
0 commit comments