3
3
namespace PHPStan \Rules ;
4
4
5
5
use PhpParser \Node ;
6
+ use PhpParser \Node \ComplexType ;
6
7
use PhpParser \Node \Expr \ConstFetch ;
7
8
use PhpParser \Node \Expr \Variable ;
8
9
use PhpParser \Node \FunctionLike ;
15
16
use PhpParser \Node \Stmt \Function_ ;
16
17
use PhpParser \Node \UnionType ;
17
18
use PHPStan \Analyser \Scope ;
19
+ use PHPStan \Node \Printer \NodeTypePrinter ;
18
20
use PHPStan \Php \PhpVersion ;
19
21
use PHPStan \Reflection \FunctionReflection ;
20
22
use PHPStan \Reflection \ParameterReflection ;
41
43
use function in_array ;
42
44
use function is_string ;
43
45
use function sprintf ;
46
+ use function strtolower ;
44
47
45
48
final class FunctionDefinitionCheck
46
49
{
@@ -103,7 +106,7 @@ public function checkAnonymousFunction(
103
106
{
104
107
$ errors = [];
105
108
$ unionTypeReported = false ;
106
- foreach ($ parameters as $ param ) {
109
+ foreach ($ parameters as $ i => $ param ) {
107
110
if ($ param ->type === null ) {
108
111
continue ;
109
112
}
@@ -123,6 +126,18 @@ public function checkAnonymousFunction(
123
126
if (!$ param ->var instanceof Variable || !is_string ($ param ->var ->name )) {
124
127
throw new ShouldNotHappenException ();
125
128
}
129
+
130
+ $ implicitlyNullableTypeError = $ this ->checkImplicitlyNullableType (
131
+ $ param ->type ,
132
+ $ param ->default ,
133
+ $ i + 1 ,
134
+ $ param ->getStartLine (),
135
+ $ param ->var ->name ,
136
+ );
137
+ if ($ implicitlyNullableTypeError !== null ) {
138
+ $ errors [] = $ implicitlyNullableTypeError ;
139
+ }
140
+
126
141
$ type = $ scope ->getFunctionType ($ param ->type , false , false );
127
142
if ($ type ->isVoid ()->yes ()) {
128
143
$ errors [] = RuleErrorBuilder::message (sprintf ($ parameterMessage , $ param ->var ->name , 'void ' ))
@@ -333,6 +348,18 @@ private function checkParametersAcceptor(
333
348
}
334
349
}
335
350
351
+ foreach ($ parameterNodes as $ i => $ parameterNode ) {
352
+ if (!$ parameterNode ->var instanceof Variable || !is_string ($ parameterNode ->var ->name )) {
353
+ throw new ShouldNotHappenException ();
354
+ }
355
+ $ implicitlyNullableTypeError = $ this ->checkImplicitlyNullableType ($ parameterNode ->type , $ parameterNode ->default , $ i + 1 , $ parameterNode ->getStartLine (), $ parameterNode ->var ->name );
356
+ if ($ implicitlyNullableTypeError === null ) {
357
+ continue ;
358
+ }
359
+
360
+ $ errors [] = $ implicitlyNullableTypeError ;
361
+ }
362
+
336
363
if ($ this ->phpVersion ->deprecatesRequiredParameterAfterOptional ()) {
337
364
$ errors = array_merge ($ errors , $ this ->checkRequiredParameterAfterOptional ($ parameterNodes ));
338
365
}
@@ -654,4 +681,60 @@ private function getReturnTypeReferencedClasses(ParametersAcceptor $parametersAc
654
681
);
655
682
}
656
683
684
+ private function checkImplicitlyNullableType (
685
+ Identifier |Name |ComplexType |null $ type ,
686
+ ?Node \Expr $ default ,
687
+ int $ order ,
688
+ int $ line ,
689
+ string $ name ,
690
+ ): ?IdentifierRuleError
691
+ {
692
+ if (!$ default instanceof ConstFetch) {
693
+ return null ;
694
+ }
695
+
696
+ if ($ default ->name ->toLowerString () !== 'null ' ) {
697
+ return null ;
698
+ }
699
+
700
+ if ($ type === null ) {
701
+ return null ;
702
+ }
703
+
704
+ if ($ type instanceof NullableType || $ type instanceof IntersectionType) {
705
+ return null ;
706
+ }
707
+
708
+ if (!$ this ->phpVersion ->deprecatesImplicitlyNullableParameterTypes ()) {
709
+ return null ;
710
+ }
711
+
712
+ if ($ type instanceof Identifier && strtolower ($ type ->name ) === 'mixed ' ) {
713
+ return null ;
714
+ }
715
+ if ($ type instanceof Name && $ type ->toLowerString () === 'mixed ' ) {
716
+ return null ;
717
+ }
718
+
719
+ if ($ type instanceof UnionType) {
720
+ foreach ($ type ->types as $ innerType ) {
721
+ if ($ innerType instanceof Identifier && strtolower ($ innerType ->name ) === 'null ' ) {
722
+ return null ;
723
+ }
724
+ if ($ innerType instanceof Name && $ innerType ->toLowerString () === 'null ' ) {
725
+ return null ;
726
+ }
727
+ }
728
+ }
729
+
730
+ return RuleErrorBuilder::message (sprintf (
731
+ 'Deprecated in PHP 8.4: Parameter #%d $%s (%s) is implicitly nullable via default value null. ' ,
732
+ $ order ,
733
+ $ name ,
734
+ NodeTypePrinter::printType ($ type ),
735
+ ))->line ($ line )
736
+ ->identifier ('parameter.implicitlyNullable ' )
737
+ ->build ();
738
+ }
739
+
657
740
}
0 commit comments