2016-12-11 23:41:11 -05:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2016-12-11 23:41:11 -05:00
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2021-12-04 21:55:53 +01:00
|
|
|
use Psalm\Tests\Traits\InvalidCodeAnalysisTestTrait;
|
|
|
|
use Psalm\Tests\Traits\ValidCodeAnalysisTestTrait;
|
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class ClassTest extends TestCase
|
2016-12-11 23:41:11 -05:00
|
|
|
{
|
2021-12-04 21:55:53 +01:00
|
|
|
use InvalidCodeAnalysisTestTrait;
|
|
|
|
use ValidCodeAnalysisTestTrait;
|
2017-04-24 23:45:02 -04:00
|
|
|
|
2021-11-26 20:59:41 +01:00
|
|
|
public function testExtendsMysqli(): void
|
2018-07-16 20:52:58 -04:00
|
|
|
{
|
|
|
|
$this->addFile(
|
|
|
|
'somefile.php',
|
|
|
|
'<?php
|
|
|
|
class db extends mysqli {
|
|
|
|
public function close()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function prepare(string $sql)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function commit(?int $flags = null, ?string $name = null)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function real_escape_string(string $string)
|
|
|
|
{
|
|
|
|
return "escaped";
|
|
|
|
}
|
|
|
|
}'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
/**
|
2022-11-05 22:34:42 +01:00
|
|
|
*
|
2017-04-24 23:45:02 -04:00
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
public function providerValidCodeParse(): iterable
|
2017-04-24 23:45:02 -04:00
|
|
|
{
|
|
|
|
return [
|
|
|
|
'overrideProtectedAccessLevelToPublic' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
class A {
|
2018-01-11 15:50:45 -05:00
|
|
|
protected function fooFoo(): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
}
|
2017-06-29 10:22:49 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class B extends A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function fooFoo(): void {}
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'reflectedParents' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
$e = rand(0, 10)
|
|
|
|
? new RuntimeException("m")
|
|
|
|
: null;
|
2017-06-29 10:22:49 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
if ($e instanceof Exception) {
|
|
|
|
echo "good";
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'namespacedAliasedClassCall' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
namespace Aye {
|
|
|
|
class Foo {}
|
|
|
|
}
|
|
|
|
namespace Bee {
|
|
|
|
use Aye as A;
|
2017-06-29 10:22:49 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
new A\Foo();
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'abstractExtendsAbstract' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
abstract class A {
|
|
|
|
/** @return void */
|
|
|
|
abstract public function foo();
|
|
|
|
}
|
2017-06-29 10:22:49 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
abstract class B extends A {
|
|
|
|
/** @return void */
|
|
|
|
public function bar() {
|
|
|
|
$this->foo();
|
|
|
|
}
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'missingParentWithFunction' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
class B extends C {
|
|
|
|
public function fooA() { }
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
2022-01-13 13:49:37 -05:00
|
|
|
'ignored_issues' => [
|
2017-04-24 23:45:02 -04:00
|
|
|
'UndefinedClass',
|
2017-05-26 20:05:57 -04:00
|
|
|
'MissingReturnType',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'subclassWithSimplerArg' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
class A {}
|
|
|
|
class B extends A {}
|
2017-06-29 10:22:49 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class E1 {
|
|
|
|
/**
|
|
|
|
* @param A|B|null $a
|
|
|
|
*/
|
|
|
|
public function __construct($a) {
|
|
|
|
}
|
|
|
|
}
|
2017-06-29 10:22:49 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class E2 extends E1 {
|
|
|
|
/**
|
|
|
|
* @param A|null $a
|
|
|
|
*/
|
|
|
|
public function __construct($a) {
|
|
|
|
parent::__construct($a);
|
|
|
|
}
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2019-02-07 12:25:57 -05:00
|
|
|
'subclassOfInvalidArgumentExceptionWithSimplerArg' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
class A extends InvalidArgumentException {
|
|
|
|
/**
|
|
|
|
* @param string $message
|
|
|
|
* @param int $code
|
|
|
|
* @param Throwable|null $previous_exception
|
|
|
|
*/
|
|
|
|
public function __construct($message, $code, $previous_exception) {
|
|
|
|
parent::__construct($message, $code, $previous_exception);
|
|
|
|
}
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
|
|
|
],
|
2018-04-03 23:14:23 -04:00
|
|
|
'classStringInstantiation' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-04-03 23:14:23 -04:00
|
|
|
class Foo {}
|
|
|
|
class Bar {}
|
|
|
|
$class = mt_rand(0, 1) === 1 ? Foo::class : Bar::class;
|
|
|
|
$object = new $class();',
|
|
|
|
'assertions' => [
|
2019-10-16 22:14:33 -07:00
|
|
|
'$object' => 'Bar|Foo',
|
2018-04-03 23:14:23 -04:00
|
|
|
],
|
|
|
|
],
|
|
|
|
'instantiateClassAndIsA' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-08-06 10:18:55 -04:00
|
|
|
/**
|
|
|
|
* @psalm-consistent-constructor
|
|
|
|
*/
|
2018-04-03 23:14:23 -04:00
|
|
|
class Foo {
|
|
|
|
public function bar() : void{}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string|null
|
|
|
|
*/
|
|
|
|
function getFooClass() {
|
|
|
|
return mt_rand(0, 1) === 1 ? Foo::class : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$foo_class = getFooClass();
|
|
|
|
|
|
|
|
if (is_string($foo_class) && is_a($foo_class, Foo::class, true)) {
|
|
|
|
$foo = new $foo_class();
|
|
|
|
$foo->bar();
|
|
|
|
}',
|
|
|
|
],
|
2018-04-05 12:03:36 -04:00
|
|
|
'returnStringAfterIsACheckWithClassConst' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-04-05 12:03:36 -04:00
|
|
|
class Foo{}
|
|
|
|
function bar(string $maybeBaz) : string {
|
|
|
|
if (!is_a($maybeBaz, Foo::class, true)) {
|
|
|
|
throw new Exception("not Foo");
|
|
|
|
}
|
|
|
|
return $maybeBaz;
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'returnStringAfterIsACheckWithString' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-04-05 12:03:36 -04:00
|
|
|
class Foo{}
|
2021-10-12 20:13:04 +02:00
|
|
|
/** @param class-string $maybeBaz */
|
2018-04-05 12:03:36 -04:00
|
|
|
function bar(string $maybeBaz) : string {
|
2021-10-12 20:13:04 +02:00
|
|
|
if (!is_a($maybeBaz, Foo::class, true)) {
|
2018-04-05 12:03:36 -04:00
|
|
|
throw new Exception("not Foo");
|
|
|
|
}
|
|
|
|
return $maybeBaz;
|
|
|
|
}',
|
|
|
|
],
|
2018-06-06 15:32:03 -04:00
|
|
|
'assignAnonymousClassToArray' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-06-06 15:32:03 -04:00
|
|
|
/**
|
|
|
|
* @param array<string, object> $array
|
|
|
|
*/
|
|
|
|
function foo(array $array, string $key) : void {
|
|
|
|
foreach ($array as $i => $item) {
|
|
|
|
$array[$key] = new class() {};
|
|
|
|
|
|
|
|
if ($array[$i] === $array[$key]) {}
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2018-07-13 09:52:15 -04:00
|
|
|
'getClassSelfClass' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-07-13 09:52:15 -04:00
|
|
|
class C {
|
|
|
|
public function work(object $obj): string {
|
|
|
|
if (get_class($obj) === self::class) {
|
|
|
|
return $obj->baz();
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
public function baz(): string {
|
|
|
|
return "baz";
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2018-07-17 22:50:30 -04:00
|
|
|
'staticClassComparison' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-07-17 22:50:30 -04:00
|
|
|
class C {
|
|
|
|
public function foo1(): string {
|
|
|
|
if (static::class === D::class) {
|
|
|
|
return $this->baz();
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function foo2(): string {
|
|
|
|
if (static::class === D::class) {
|
|
|
|
return static::bat();
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class D extends C {
|
|
|
|
public function baz(): string {
|
|
|
|
return "baz";
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function bat(): string {
|
|
|
|
return "baz";
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'isAStaticClass' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-07-17 22:50:30 -04:00
|
|
|
class C {
|
|
|
|
public function foo1(): string {
|
|
|
|
if (is_a(static::class, D::class, true)) {
|
|
|
|
return $this->baz();
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function foo2(): string {
|
|
|
|
if (is_a(static::class, D::class, true)) {
|
|
|
|
return static::bat();
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class D extends C {
|
|
|
|
public function baz(): string {
|
|
|
|
return "baz";
|
|
|
|
}
|
|
|
|
|
|
|
|
public static function bat(): string {
|
|
|
|
return "baz";
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2018-07-16 22:48:53 -04:00
|
|
|
'typedMagicCall' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-07-16 22:48:53 -04:00
|
|
|
class B {
|
|
|
|
public function __call(string $methodName, array $args) : string {
|
|
|
|
return __METHOD__;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
class A {
|
|
|
|
public function __call(string $methodName, array $args) : B {
|
|
|
|
return new B;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$a = (new A)->zugzug();
|
|
|
|
$b = (new A)->bar()->baz();',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'B',
|
|
|
|
'$b' => 'string',
|
|
|
|
],
|
|
|
|
],
|
2018-11-10 19:05:51 -05:00
|
|
|
'abstractCallToInterfaceMethod' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-11-10 19:05:51 -05:00
|
|
|
interface I {
|
|
|
|
public function fooBar(): array;
|
|
|
|
}
|
|
|
|
|
|
|
|
abstract class A implements I
|
|
|
|
{
|
|
|
|
public function g(): array {
|
|
|
|
return $this->fooBar();
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2018-12-30 18:28:15 +01:00
|
|
|
'noCrashWhenIgnoringUndefinedClass' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-12-30 18:28:15 +01:00
|
|
|
class A extends B {
|
|
|
|
public function foo() {
|
|
|
|
parent::bar();
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
2022-01-13 13:49:37 -05:00
|
|
|
'ignored_issues' => [
|
2019-03-23 14:27:54 -04:00
|
|
|
'UndefinedClass',
|
2018-12-30 18:28:15 +01:00
|
|
|
],
|
|
|
|
],
|
2019-01-05 12:59:12 -05:00
|
|
|
'noCrashWhenIgnoringUndefinedParam' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-01-05 12:59:12 -05:00
|
|
|
function bar(iterable $_i) : void {}
|
|
|
|
function foo(C $c) : void {
|
|
|
|
bar($c);
|
|
|
|
}',
|
|
|
|
'assertions' => [],
|
2022-01-13 13:49:37 -05:00
|
|
|
'ignored_issues' => [
|
2019-01-05 12:59:12 -05:00
|
|
|
'UndefinedClass',
|
|
|
|
'InvalidArgument',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'noCrashWhenIgnoringUndefinedReturnIterableArg' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-01-05 12:59:12 -05:00
|
|
|
function bar(iterable $_i) : void {}
|
|
|
|
function foo() : D {
|
|
|
|
return new D();
|
|
|
|
}
|
|
|
|
bar(foo());',
|
|
|
|
'assertions' => [],
|
2022-01-13 13:49:37 -05:00
|
|
|
'ignored_issues' => [
|
2019-01-05 12:59:12 -05:00
|
|
|
'UndefinedClass',
|
|
|
|
'MixedInferredReturnType',
|
|
|
|
'InvalidArgument',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'noCrashWhenIgnoringUndefinedReturnClassArg' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-01-05 12:59:12 -05:00
|
|
|
class Exists {}
|
|
|
|
function bar(Exists $_i) : void {}
|
|
|
|
function foo() : D {
|
|
|
|
return new D();
|
|
|
|
}
|
|
|
|
bar(foo());',
|
|
|
|
'assertions' => [],
|
2022-01-13 13:49:37 -05:00
|
|
|
'ignored_issues' => [
|
2019-01-05 12:59:12 -05:00
|
|
|
'UndefinedClass',
|
|
|
|
'MixedInferredReturnType',
|
|
|
|
'InvalidArgument',
|
|
|
|
],
|
|
|
|
],
|
2019-01-04 12:28:00 -05:00
|
|
|
'allowAbstractInstantiationOnPossibleChild' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-08-06 10:18:55 -04:00
|
|
|
/**
|
|
|
|
* @psalm-consistent-constructor
|
|
|
|
*/
|
2019-01-04 12:28:00 -05:00
|
|
|
abstract class A {}
|
|
|
|
|
|
|
|
function foo(string $a_class) : void {
|
|
|
|
if (is_a($a_class, A::class, true)) {
|
|
|
|
new $a_class();
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2022-11-10 00:12:40 -04:00
|
|
|
'markInferredMutationFreeDuringPropertyTypeInferenceAsActuallyInferred' => [
|
|
|
|
'code' => '<?php
|
|
|
|
class A {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @psalm-consistent-constructor
|
|
|
|
*/
|
|
|
|
abstract class AbstractClass
|
|
|
|
{
|
|
|
|
protected $renderer;
|
|
|
|
|
|
|
|
public function __construct(A $r)
|
|
|
|
{
|
|
|
|
$this->renderer = $r;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class ConcreteClass extends AbstractClass
|
|
|
|
{
|
|
|
|
public function __construct(A $r)
|
|
|
|
{
|
|
|
|
parent::__construct($r);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
',
|
|
|
|
],
|
2019-05-19 15:56:04 -04:00
|
|
|
'interfaceExistsCreatesClassString' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-05-19 15:56:04 -04:00
|
|
|
function funB(string $className) : ?ReflectionClass {
|
|
|
|
if (class_exists($className)) {
|
|
|
|
return new ReflectionClass($className);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (interface_exists($className)) {
|
|
|
|
return new ReflectionClass($className);
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}',
|
|
|
|
],
|
2019-05-20 20:57:59 -04:00
|
|
|
'allowClassExistsAndInterfaceExists' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-05-20 20:57:59 -04:00
|
|
|
function foo(string $s) : void {
|
|
|
|
if (class_exists($s) || interface_exists($s)) {}
|
2019-07-05 16:24:00 -04:00
|
|
|
}',
|
2019-05-20 20:57:59 -04:00
|
|
|
],
|
2022-01-20 17:33:06 -05:00
|
|
|
'classExistsWithFalseArgRefinedAsString' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-05-26 13:16:44 -04:00
|
|
|
/**
|
|
|
|
* @param class-string $class
|
2019-05-26 13:34:07 -04:00
|
|
|
* @return string
|
2019-05-26 13:16:44 -04:00
|
|
|
*/
|
|
|
|
function autoload(string $class) : string {
|
|
|
|
if (class_exists($class, false)) {
|
|
|
|
return $class;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $class;
|
2019-07-05 16:24:00 -04:00
|
|
|
}',
|
2019-05-26 13:16:44 -04:00
|
|
|
],
|
2020-11-01 13:14:17 -05:00
|
|
|
'allowNegatingClassExistsWithoutAutloading' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-11-01 13:14:17 -05:00
|
|
|
function specifyString(string $className): void{
|
|
|
|
if (!class_exists($className, false)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
new ReflectionClass($className);
|
|
|
|
}'
|
|
|
|
],
|
2019-05-28 13:16:09 -04:00
|
|
|
'classExistsWithFalseArgInside' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-05-28 13:16:09 -04:00
|
|
|
function foo(string $s) : void {
|
|
|
|
if (class_exists($s, false)) {
|
|
|
|
/** @psalm-suppress MixedMethodCall */
|
|
|
|
new $s();
|
|
|
|
}
|
2019-07-05 16:24:00 -04:00
|
|
|
}',
|
2019-05-28 13:16:09 -04:00
|
|
|
],
|
2019-07-04 15:46:11 -04:00
|
|
|
'classAliasOnNonexistantClass' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-07-04 15:46:11 -04:00
|
|
|
if (!class_exists(\PHPUnit\Framework\TestCase::class)) {
|
|
|
|
/** @psalm-suppress UndefinedClass */
|
|
|
|
class_alias(\PHPUnit_Framework_TestCase::class, \PHPUnit\Framework\TestCase::class);
|
|
|
|
}
|
|
|
|
|
|
|
|
class T extends \PHPUnit\Framework\TestCase {
|
|
|
|
|
|
|
|
}',
|
2022-01-13 13:49:37 -05:00
|
|
|
'assertions' => [],
|
|
|
|
'ignored_issues' => ['PropertyNotSetInConstructor'],
|
2019-07-04 15:46:11 -04:00
|
|
|
],
|
2019-08-15 10:17:27 -04:00
|
|
|
'classAliasNoException' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-08-15 10:17:27 -04:00
|
|
|
class_alias("Bar\F1", "Bar\F2");
|
|
|
|
|
|
|
|
namespace Bar {
|
|
|
|
class F1 {
|
|
|
|
public static function baz() : void {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
Bar\F2::baz();
|
|
|
|
}',
|
|
|
|
],
|
2019-09-30 22:15:48 -04:00
|
|
|
'classAliasEcho' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-09-30 22:15:48 -04:00
|
|
|
class A { }
|
|
|
|
class_alias("A", "A_A");
|
|
|
|
|
|
|
|
echo A_A::class;'
|
|
|
|
],
|
2019-11-29 08:21:00 +02:00
|
|
|
'classAliasTrait' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-11-29 08:21:00 +02:00
|
|
|
trait FeatureV1 {}
|
|
|
|
class_alias(FeatureV1::class, Feature::class);
|
|
|
|
class App { use Feature; }
|
|
|
|
'
|
2020-01-01 17:23:13 -05:00
|
|
|
],
|
|
|
|
'classAliasParent' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-01-01 17:23:13 -05:00
|
|
|
class NewA {}
|
|
|
|
class_alias(NewA::class, OldA::class);
|
|
|
|
function action(NewA $_m): void {}
|
|
|
|
|
|
|
|
class OldAChild extends OldA {}
|
|
|
|
action(new OldA());
|
|
|
|
action(new OldAChild());'
|
|
|
|
],
|
2021-01-22 02:38:17 +02:00
|
|
|
'classAliasStaticProperty' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2021-01-22 02:38:17 +02:00
|
|
|
class A {
|
|
|
|
/** @var int */
|
|
|
|
public static $prop = 1;
|
|
|
|
}
|
|
|
|
class_alias(A::class, B::class);
|
|
|
|
B::$prop = 123;'
|
|
|
|
],
|
2020-02-11 22:34:34 -05:00
|
|
|
'resourceAndNumericSoftlyReserved' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-11-04 12:30:02 -05:00
|
|
|
namespace {
|
|
|
|
class Numeric {}
|
|
|
|
}
|
|
|
|
|
2020-02-11 22:34:34 -05:00
|
|
|
namespace Foo {
|
|
|
|
class Resource {}
|
|
|
|
class Numeric {}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace Bar {
|
|
|
|
use \Foo\Resource;
|
|
|
|
use \Foo\Numeric;
|
|
|
|
|
|
|
|
new \Foo\Resource();
|
|
|
|
new \Foo\Numeric();
|
|
|
|
|
|
|
|
new Resource();
|
|
|
|
new Numeric();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Resource $r
|
|
|
|
* @param Numeric $n
|
|
|
|
* @return void
|
|
|
|
*/
|
2021-05-03 16:59:24 -04:00
|
|
|
function foo(Resource $r, Numeric $n) : void {}
|
2020-02-11 22:34:34 -05:00
|
|
|
}'
|
|
|
|
],
|
2020-02-13 17:44:16 -05:00
|
|
|
'inheritInterfaceFromParent' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-02-13 17:44:16 -05:00
|
|
|
class A {}
|
|
|
|
class AChild extends A {}
|
|
|
|
|
|
|
|
interface IParent {
|
|
|
|
public function get(): A;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface IChild extends IParent {
|
|
|
|
/**
|
|
|
|
* @psalm-return AChild
|
|
|
|
*/
|
|
|
|
public function get(): A;
|
|
|
|
}
|
|
|
|
|
|
|
|
class Concrete implements IChild {
|
|
|
|
public function get(): A {
|
|
|
|
return new AChild;
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2020-02-18 19:46:05 -05:00
|
|
|
'noErrorsAfterClassExists' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-02-18 19:46:05 -05:00
|
|
|
if (class_exists(A::class)) {
|
|
|
|
if (method_exists(A::class, "method")) {
|
2020-11-08 11:04:39 -05:00
|
|
|
/** @psalm-suppress MixedArgument */
|
2020-02-18 19:46:05 -05:00
|
|
|
echo A::method();
|
|
|
|
}
|
|
|
|
|
|
|
|
echo A::class;
|
|
|
|
/** @psalm-suppress MixedArgument */
|
|
|
|
echo A::SOME_CONST;
|
|
|
|
}'
|
|
|
|
],
|
2020-03-09 14:24:19 -04:00
|
|
|
'noCrashOnClassExists' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-03-09 14:24:19 -04:00
|
|
|
if (!class_exists(ReflectionGenerator::class)) {
|
|
|
|
class ReflectionGenerator {
|
|
|
|
private $prop;
|
|
|
|
}
|
|
|
|
}',
|
2020-03-11 17:41:05 -04:00
|
|
|
],
|
2021-03-29 07:11:45 +03:00
|
|
|
'instanceofWithPhantomClass' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2021-03-29 07:11:45 +03:00
|
|
|
if (class_exists(NS\UnknonwClass::class)) {
|
|
|
|
null instanceof NS\UnknonwClass;
|
|
|
|
}
|
|
|
|
',
|
|
|
|
],
|
2020-03-11 17:41:05 -04:00
|
|
|
'extendException' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-03-11 17:41:05 -04:00
|
|
|
class ME extends Exception {
|
|
|
|
protected $message = "hello";
|
|
|
|
}',
|
2020-03-24 18:00:20 -04:00
|
|
|
],
|
2020-03-24 18:14:10 -04:00
|
|
|
'allowFinalReturnerForStatic' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-08-05 19:39:27 -04:00
|
|
|
/**
|
|
|
|
* @psalm-consistent-constructor
|
|
|
|
*/
|
2020-03-24 18:14:10 -04:00
|
|
|
class A {
|
|
|
|
/** @return static */
|
|
|
|
public static function getInstance() {
|
|
|
|
return new static();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
final class AChild extends A {
|
|
|
|
public static function getInstance() {
|
|
|
|
return new AChild();
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2020-03-27 19:17:44 -04:00
|
|
|
'intersectWithStatic' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-03-27 19:17:44 -04:00
|
|
|
interface M1 {
|
|
|
|
/** @return M2&static */
|
|
|
|
function mock();
|
|
|
|
}
|
|
|
|
|
|
|
|
interface M2 {}
|
|
|
|
|
|
|
|
class A {}
|
|
|
|
|
|
|
|
/** @return A&M1 */
|
|
|
|
function intersect(A $a) {
|
|
|
|
assert($a instanceof M1);
|
|
|
|
|
|
|
|
if (rand(0, 1)) {
|
|
|
|
return $a;
|
|
|
|
}
|
|
|
|
|
|
|
|
$b = $a->mock();
|
|
|
|
|
|
|
|
return $b;
|
|
|
|
}'
|
|
|
|
],
|
2022-10-14 13:30:57 +02:00
|
|
|
'preventDoubleStaticResolution1' => [
|
|
|
|
'code' => '<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @template TTKey
|
|
|
|
* @template TTValue
|
|
|
|
*
|
|
|
|
* @extends ArrayObject<TTKey, TTValue>
|
|
|
|
*/
|
|
|
|
class iter extends ArrayObject {
|
|
|
|
/**
|
|
|
|
* @return self<TTKey, TTValue>
|
|
|
|
*/
|
|
|
|
public function stabilize(): self {
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
}
|
2022-11-05 22:34:42 +01:00
|
|
|
|
2022-10-14 13:30:57 +02:00
|
|
|
class a {
|
|
|
|
/**
|
|
|
|
* @return iter<int, static>
|
|
|
|
*/
|
|
|
|
public function ret(): iter {
|
|
|
|
return new iter([$this]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
class b extends a {
|
|
|
|
}
|
2022-11-05 22:34:42 +01:00
|
|
|
|
2022-10-14 13:30:57 +02:00
|
|
|
$a = new b;
|
|
|
|
$a = $a->ret();
|
|
|
|
$a = $a->stabilize();',
|
|
|
|
'assertions' => [
|
|
|
|
'$a===' => 'iter<int, b&static>'
|
|
|
|
]
|
|
|
|
],
|
|
|
|
'preventDoubleStaticResolution2' => [
|
|
|
|
'code' => '<?php
|
|
|
|
/**
|
|
|
|
* @template TTKey
|
|
|
|
* @template TTValue
|
|
|
|
*
|
|
|
|
* @extends ArrayObject<TTKey, TTValue>
|
|
|
|
*/
|
|
|
|
class iter extends ArrayObject {
|
|
|
|
/**
|
|
|
|
* @return self<TTKey, TTValue>
|
|
|
|
*/
|
|
|
|
public function stabilize(): self {
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
}
|
2022-11-05 22:34:42 +01:00
|
|
|
|
2022-10-14 13:30:57 +02:00
|
|
|
interface a {
|
|
|
|
/**
|
|
|
|
* @return iter<int, static>
|
|
|
|
*/
|
|
|
|
public function ret(): iter;
|
|
|
|
}
|
|
|
|
class b implements a {
|
|
|
|
public function ret(): iter {
|
|
|
|
return new iter([$this]);
|
|
|
|
}
|
|
|
|
}
|
2022-11-05 22:34:42 +01:00
|
|
|
|
2022-10-14 13:30:57 +02:00
|
|
|
/** @var a */
|
|
|
|
$a = new b;
|
|
|
|
$a = $a->ret();
|
|
|
|
$a = $a->stabilize();',
|
|
|
|
'assertions' => [
|
|
|
|
'$a===' => 'iter<int, a&static>'
|
|
|
|
]
|
|
|
|
],
|
|
|
|
'preventDoubleStaticResolution3' => [
|
|
|
|
'code' => '<?php
|
|
|
|
/**
|
|
|
|
* @template TTKey
|
|
|
|
* @template TTValue
|
|
|
|
*
|
|
|
|
* @extends ArrayObject<TTKey, TTValue>
|
|
|
|
*/
|
|
|
|
class iter extends ArrayObject {
|
|
|
|
/**
|
|
|
|
* @return self<TTKey, TTValue>
|
|
|
|
*/
|
|
|
|
public function stabilize(): self {
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
}
|
2022-11-05 22:34:42 +01:00
|
|
|
|
2022-10-14 13:30:57 +02:00
|
|
|
interface a {
|
|
|
|
/**
|
|
|
|
* @return iter<int, a&static>
|
|
|
|
*/
|
|
|
|
public function ret(): iter;
|
|
|
|
}
|
|
|
|
class b implements a {
|
|
|
|
public function ret(): iter {
|
|
|
|
return new iter([$this]);
|
|
|
|
}
|
|
|
|
}
|
2022-11-05 22:34:42 +01:00
|
|
|
|
2022-10-14 13:30:57 +02:00
|
|
|
/** @var a */
|
|
|
|
$a = new b;
|
|
|
|
$a = $a->ret();
|
|
|
|
$a = $a->stabilize();',
|
|
|
|
'assertions' => [
|
|
|
|
'$a===' => 'iter<int, a&static>'
|
|
|
|
]
|
|
|
|
],
|
2021-02-07 05:10:05 +02:00
|
|
|
'allowTraversableImplementationAlongWithIteratorAggregate' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2022-01-26 18:46:02 +01:00
|
|
|
/**
|
|
|
|
* @implements Traversable<int, 1>
|
|
|
|
* @implements IteratorAggregate<int, 1>
|
|
|
|
*/
|
2021-02-07 05:10:05 +02:00
|
|
|
final class C implements Traversable, IteratorAggregate {
|
|
|
|
public function getIterator() {
|
|
|
|
yield 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
',
|
|
|
|
],
|
|
|
|
'allowTraversableImplementationAlongWithIterator' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2022-01-26 18:46:02 +01:00
|
|
|
/**
|
|
|
|
* @implements Traversable<1, 1>
|
|
|
|
* @implements Iterator<1, 1>
|
|
|
|
*/
|
2021-02-07 05:10:05 +02:00
|
|
|
final class C implements Traversable, Iterator {
|
|
|
|
public function current() { return 1; }
|
|
|
|
public function key() { return 1; }
|
|
|
|
public function next() { }
|
|
|
|
public function rewind() { }
|
|
|
|
public function valid() { return false; }
|
|
|
|
}
|
|
|
|
',
|
|
|
|
],
|
|
|
|
'allowTraversableImplementationOnAbstractClass' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2022-01-26 18:46:02 +01:00
|
|
|
/**
|
|
|
|
* @template TKey
|
|
|
|
* @template TValue
|
2022-11-05 22:34:42 +01:00
|
|
|
*
|
2022-01-26 18:46:02 +01:00
|
|
|
* @implements Traversable<TKey, TValue>
|
|
|
|
*/
|
2021-02-07 05:10:05 +02:00
|
|
|
abstract class C implements Traversable {}
|
|
|
|
',
|
|
|
|
],
|
|
|
|
'allowIndirectTraversableImplementationOnAbstractClass' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2022-01-26 18:46:02 +01:00
|
|
|
/**
|
|
|
|
* @extends Traversable<int, int>
|
|
|
|
*/
|
2021-02-07 05:10:05 +02:00
|
|
|
interface I extends Traversable {}
|
|
|
|
abstract class C implements I {}
|
|
|
|
',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-11-05 22:34:42 +01:00
|
|
|
*
|
2017-04-24 23:45:02 -04:00
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
public function providerInvalidCodeParse(): iterable
|
2017-04-24 23:45:02 -04:00
|
|
|
{
|
|
|
|
return [
|
|
|
|
'undefinedClass' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
(new Foo());',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'UndefinedClass',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'wrongCaseClass' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
class Foo {}
|
|
|
|
(new foo());',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'InvalidClass',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2017-10-12 19:46:00 -04:00
|
|
|
'wrongCaseClassWithCall' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-10-12 19:46:00 -04:00
|
|
|
class A {}
|
|
|
|
needsA(new A);
|
2018-01-11 15:50:45 -05:00
|
|
|
function needsA(a $x): void {}',
|
2017-10-12 19:46:00 -04:00
|
|
|
'error_message' => 'InvalidClass',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'invalidThisFetch' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
echo $this;',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'InvalidScope',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'invalidThisArgument' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
$this = "hello";',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'InvalidScope',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'undefinedConstant' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
echo HELLO;',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'UndefinedConstant',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'undefinedClassConstant' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
class A {}
|
|
|
|
echo A::HELLO;',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'UndefinedConstant',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2018-03-04 12:24:50 -05:00
|
|
|
'overridePublicAccessLevelToPrivate' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
class A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function fooFoo(): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
}
|
2017-06-29 10:22:49 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class B extends A {
|
2018-01-11 15:50:45 -05:00
|
|
|
private function fooFoo(): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
}',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'OverriddenMethodAccess',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'overridePublicAccessLevelToProtected' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
class A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function fooFoo(): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
}
|
2017-06-29 10:22:49 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class B extends A {
|
2018-01-11 15:50:45 -05:00
|
|
|
protected function fooFoo(): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
}',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'OverriddenMethodAccess',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'overrideProtectedAccessLevelToPrivate' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
class A {
|
2018-01-11 15:50:45 -05:00
|
|
|
protected function fooFoo(): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
}
|
2017-06-29 10:22:49 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class B extends A {
|
2018-01-11 15:50:45 -05:00
|
|
|
private function fooFoo(): void {}
|
2017-04-24 23:45:02 -04:00
|
|
|
}',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'OverriddenMethodAccess',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2018-03-04 12:24:50 -05:00
|
|
|
'overridePublicPropertyAccessLevelToPrivate' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-03-04 12:24:50 -05:00
|
|
|
class A {
|
|
|
|
/** @var string|null */
|
|
|
|
public $foo;
|
|
|
|
}
|
|
|
|
|
|
|
|
class B extends A {
|
|
|
|
/** @var string|null */
|
|
|
|
private $foo;
|
|
|
|
}',
|
|
|
|
'error_message' => 'OverriddenPropertyAccess',
|
|
|
|
],
|
|
|
|
'overridePublicPropertyAccessLevelToProtected' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-03-04 12:24:50 -05:00
|
|
|
class A {
|
|
|
|
/** @var string|null */
|
|
|
|
public $foo;
|
|
|
|
}
|
|
|
|
|
|
|
|
class B extends A {
|
|
|
|
/** @var string|null */
|
|
|
|
protected $foo;
|
|
|
|
}',
|
|
|
|
'error_message' => 'OverriddenPropertyAccess',
|
|
|
|
],
|
|
|
|
'overrideProtectedPropertyAccessLevelToPrivate' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-03-04 12:24:50 -05:00
|
|
|
class A {
|
|
|
|
/** @var string|null */
|
|
|
|
protected $foo;
|
|
|
|
}
|
|
|
|
|
|
|
|
class B extends A {
|
|
|
|
/** @var string|null */
|
|
|
|
private $foo;
|
|
|
|
}',
|
|
|
|
'error_message' => 'OverriddenPropertyAccess',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'classRedefinition' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
class Foo {}
|
|
|
|
class Foo {}',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'DuplicateClass',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'classRedefinitionInNamespace' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
namespace Aye {
|
|
|
|
class Foo {}
|
|
|
|
class Foo {}
|
|
|
|
}',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'DuplicateClass',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'classRedefinitionInSeparateNamespace' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
namespace Aye {
|
|
|
|
class Foo {}
|
|
|
|
}
|
|
|
|
namespace Aye {
|
|
|
|
class Foo {}
|
|
|
|
}',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'DuplicateClass',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'abstractClassInstantiation' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
abstract class A {}
|
|
|
|
new A();',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'AbstractInstantiation',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2017-06-30 10:24:47 -04:00
|
|
|
'abstractClassMethod' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2018-07-21 20:50:42 -04:00
|
|
|
abstract class A {
|
2019-07-28 19:44:36 -04:00
|
|
|
abstract public function foo() : void;
|
2018-07-21 20:50:42 -04:00
|
|
|
}
|
2017-06-30 10:24:47 -04:00
|
|
|
|
2018-07-21 20:50:42 -04:00
|
|
|
class B extends A { }',
|
|
|
|
'error_message' => 'UnimplementedAbstractMethod',
|
|
|
|
],
|
|
|
|
'abstractReflectedClassMethod' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2022-01-26 18:46:02 +01:00
|
|
|
/**
|
|
|
|
* @template TKey
|
|
|
|
* @template TValue
|
2022-01-31 10:28:47 +01:00
|
|
|
* @extends FilterIterator<TKey, TValue, Iterator<TKey, TValue>>
|
2022-01-26 18:46:02 +01:00
|
|
|
*/
|
2018-07-21 20:50:42 -04:00
|
|
|
class DedupeIterator extends FilterIterator {
|
2022-01-31 10:28:47 +01:00
|
|
|
/**
|
|
|
|
* @param Iterator<TKey, TValue> $i
|
|
|
|
*/
|
2018-07-21 20:50:42 -04:00
|
|
|
public function __construct(Iterator $i) {
|
|
|
|
parent::__construct($i);
|
|
|
|
}
|
|
|
|
}',
|
2017-06-30 10:24:47 -04:00
|
|
|
'error_message' => 'UnimplementedAbstractMethod',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'missingParent' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
class A extends B { }',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'UndefinedClass',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2017-12-07 15:50:25 -05:00
|
|
|
'lessSpecificReturnStatement' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-04-24 23:45:02 -04:00
|
|
|
class A {}
|
|
|
|
class B extends A {}
|
2017-06-29 10:22:49 -04:00
|
|
|
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo(A $a): B {
|
2017-04-24 23:45:02 -04:00
|
|
|
return $a;
|
|
|
|
}',
|
2017-12-07 15:50:25 -05:00
|
|
|
'error_message' => 'LessSpecificReturnStatement',
|
2017-05-26 20:05:57 -04:00
|
|
|
],
|
2017-12-22 18:56:59 +01:00
|
|
|
'circularReference' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2017-12-22 18:56:59 +01:00
|
|
|
class A extends A {}',
|
|
|
|
'error_message' => 'CircularReference',
|
|
|
|
],
|
2019-01-04 12:28:00 -05:00
|
|
|
'preventAbstractInstantiationDefiniteClasss' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-01-04 12:28:00 -05:00
|
|
|
abstract class A {}
|
|
|
|
|
|
|
|
function foo(string $a_class) : void {
|
|
|
|
if ($a_class === A::class) {
|
|
|
|
new $a_class();
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'error_message' => 'AbstractInstantiation',
|
|
|
|
],
|
2019-03-07 11:16:40 -05:00
|
|
|
'preventExtendingInterface' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-03-07 11:16:40 -05:00
|
|
|
interface Foo {}
|
|
|
|
|
|
|
|
class Bar extends Foo {}',
|
|
|
|
'error_message' => 'UndefinedClass',
|
|
|
|
],
|
|
|
|
'preventImplementingClass' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-03-07 11:16:40 -05:00
|
|
|
class Foo {}
|
|
|
|
|
|
|
|
class Bar implements Foo {}',
|
|
|
|
'error_message' => 'UndefinedInterface',
|
|
|
|
],
|
2019-07-17 20:33:44 -04:00
|
|
|
'classAliasAlreadyDefinedClass' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2019-07-17 20:33:44 -04:00
|
|
|
class A {}
|
|
|
|
|
|
|
|
class B {}
|
|
|
|
|
|
|
|
if (false) {
|
|
|
|
class_alias(A::class, B::class);
|
|
|
|
}
|
|
|
|
|
|
|
|
function foo(A $a, B $b) : void {
|
|
|
|
if ($a === $b) {}
|
|
|
|
}',
|
|
|
|
'error_message' => 'TypeDoesNotContainType',
|
|
|
|
],
|
2020-03-10 21:26:02 -04:00
|
|
|
'cannotOverrideFinalType' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-03-10 21:26:02 -04:00
|
|
|
class P {
|
|
|
|
public final function f() : void {}
|
|
|
|
}
|
|
|
|
|
|
|
|
class C extends P {
|
|
|
|
public function f() : void {}
|
|
|
|
}',
|
|
|
|
'error_message' => 'MethodSignatureMismatch',
|
|
|
|
],
|
2020-03-24 18:00:20 -04:00
|
|
|
'preventFinalOverriding' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2020-08-05 19:39:27 -04:00
|
|
|
/**
|
|
|
|
* @psalm-consistent-constructor
|
|
|
|
*/
|
2020-03-24 18:00:20 -04:00
|
|
|
class A {
|
|
|
|
/** @return static */
|
|
|
|
public static function getInstance() {
|
|
|
|
return new static();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class AChild extends A {
|
|
|
|
public static function getInstance() {
|
|
|
|
return new AChild();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class AGrandChild extends AChild {
|
|
|
|
public function foo() : void {}
|
|
|
|
}
|
|
|
|
|
|
|
|
AGrandChild::getInstance()->foo();',
|
|
|
|
'error_message' => 'LessSpecificReturnStatement',
|
|
|
|
],
|
2021-02-07 05:10:05 +02:00
|
|
|
'preventTraversableImplementation' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2022-01-26 18:46:02 +01:00
|
|
|
/**
|
|
|
|
* @implements Traversable<int, int>
|
|
|
|
*/
|
2021-02-07 05:10:05 +02:00
|
|
|
final class C implements Traversable {}
|
|
|
|
',
|
|
|
|
'error_message' => 'InvalidTraversableImplementation',
|
|
|
|
],
|
|
|
|
'preventIndirectTraversableImplementation' => [
|
2022-01-13 13:49:37 -05:00
|
|
|
'code' => '<?php
|
2022-01-26 18:46:02 +01:00
|
|
|
/**
|
|
|
|
* @extends Traversable<int, int>
|
|
|
|
*/
|
2021-02-07 05:10:05 +02:00
|
|
|
interface I extends Traversable {}
|
|
|
|
final class C implements I {}
|
|
|
|
',
|
|
|
|
'error_message' => 'InvalidTraversableImplementation',
|
|
|
|
],
|
2022-01-26 18:46:02 +01:00
|
|
|
'detectMissingTemplateExtends' => [
|
2022-01-26 18:56:35 +01:00
|
|
|
'code' => '<?php
|
2022-01-26 18:46:02 +01:00
|
|
|
/** @template T */
|
|
|
|
abstract class A {}
|
|
|
|
final class B extends A {}
|
|
|
|
',
|
|
|
|
'error_message' => 'MissingTemplateParam',
|
|
|
|
],
|
|
|
|
'detectMissingTemplateImplements' => [
|
2022-01-26 18:56:35 +01:00
|
|
|
'code' => '<?php
|
2022-01-26 18:46:02 +01:00
|
|
|
/** @template T */
|
|
|
|
interface A {}
|
|
|
|
final class B implements A {}
|
|
|
|
',
|
|
|
|
'error_message' => 'MissingTemplateParam',
|
|
|
|
],
|
|
|
|
'detectMissingTemplateUse' => [
|
2022-01-26 18:56:35 +01:00
|
|
|
'code' => '<?php
|
2022-01-26 18:46:02 +01:00
|
|
|
/** @template T */
|
|
|
|
trait A {}
|
|
|
|
final class B {
|
|
|
|
use A;
|
|
|
|
}
|
|
|
|
',
|
|
|
|
'error_message' => 'MissingTemplateParam',
|
|
|
|
],
|
|
|
|
|
|
|
|
'detectMissingTemplateExtendsNative' => [
|
2022-01-26 18:56:35 +01:00
|
|
|
'code' => '<?php
|
2022-01-26 18:46:02 +01:00
|
|
|
final class C extends ArrayObject {}
|
|
|
|
',
|
|
|
|
'error_message' => 'MissingTemplateParam',
|
|
|
|
],
|
|
|
|
|
|
|
|
'detectMissingTemplateImplementsNative' => [
|
2022-01-26 18:56:35 +01:00
|
|
|
'code' => '<?php
|
2022-01-26 18:46:02 +01:00
|
|
|
final class C implements Iterator {
|
|
|
|
public function current(): mixed {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
public function key(): mixed {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
public function next(): void {
|
|
|
|
}
|
|
|
|
public function rewind(): void {
|
|
|
|
}
|
|
|
|
public function valid(): bool {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
',
|
|
|
|
'error_message' => 'MissingTemplateParam',
|
|
|
|
],
|
2022-10-07 09:44:10 +01:00
|
|
|
'cannotNameClassConstantClass' => [
|
2022-10-16 13:59:15 +02:00
|
|
|
'code' => '<?php
|
2022-10-07 09:44:10 +01:00
|
|
|
class Foo
|
|
|
|
{
|
|
|
|
/** @var class-string<Bar> */
|
|
|
|
protected const CLASS = Bar::class;
|
|
|
|
}
|
2022-11-05 22:34:42 +01:00
|
|
|
|
2022-10-07 09:44:10 +01:00
|
|
|
class Bar {}
|
|
|
|
',
|
|
|
|
'error_message' => 'ReservedWord',
|
|
|
|
]
|
2017-04-24 23:45:02 -04:00
|
|
|
];
|
2017-03-30 11:44:38 -04:00
|
|
|
}
|
2016-12-11 23:41:11 -05:00
|
|
|
}
|