2016-12-11 23:41:11 -05:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2018-02-01 01:10:27 -05:00
|
|
|
use Psalm\Config;
|
|
|
|
use Psalm\Context;
|
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class AnnotationTest extends TestCase
|
2016-12-11 23:41:11 -05:00
|
|
|
{
|
2017-04-24 23:45:02 -04:00
|
|
|
use Traits\FileCheckerInvalidCodeParseTestTrait;
|
|
|
|
use Traits\FileCheckerValidCodeParseTestTrait;
|
2016-12-24 18:23:22 +00:00
|
|
|
|
2018-02-01 01:10:27 -05:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testPhpStormGenericsWithValidArgument()
|
|
|
|
{
|
|
|
|
Config::getInstance()->allow_phpstorm_generics = true;
|
|
|
|
|
|
|
|
$this->addFile(
|
|
|
|
'somefile.php',
|
|
|
|
'<?php
|
|
|
|
function takesString(string $s): void {}
|
|
|
|
|
|
|
|
/** @param ArrayIterator|string[] $i */
|
|
|
|
function takesArrayIteratorOfString(ArrayIterator $i): void {
|
|
|
|
$s = $i->offsetGet("a");
|
|
|
|
takesString($s);
|
|
|
|
|
|
|
|
foreach ($i as $s2) {
|
|
|
|
takesString($s2);
|
|
|
|
}
|
|
|
|
}'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @expectedException \Psalm\Exception\CodeException
|
|
|
|
* @expectedExceptionMessage InvalidScalarArgument
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testPhpStormGenericsInvalidArgument()
|
|
|
|
{
|
|
|
|
Config::getInstance()->allow_phpstorm_generics = true;
|
|
|
|
|
|
|
|
$this->addFile(
|
|
|
|
'somefile.php',
|
|
|
|
'<?php
|
|
|
|
function takesInt(int $s): void {}
|
|
|
|
|
|
|
|
/** @param ArrayIterator|string[] $i */
|
|
|
|
function takesArrayIteratorOfString(ArrayIterator $i): void {
|
|
|
|
$s = $i->offsetGet("a");
|
|
|
|
takesInt($s);
|
|
|
|
}'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @expectedException \Psalm\Exception\CodeException
|
|
|
|
* @expectedExceptionMessage PossiblyInvalidMethodCall
|
|
|
|
*
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testPhpStormGenericsNoTypehint()
|
|
|
|
{
|
|
|
|
Config::getInstance()->allow_phpstorm_generics = true;
|
|
|
|
|
|
|
|
$this->addFile(
|
|
|
|
'somefile.php',
|
|
|
|
'<?php
|
|
|
|
/** @param ArrayIterator|string[] $i */
|
|
|
|
function takesArrayIteratorOfString($i): void {
|
|
|
|
$s = $i->offsetGet("a");
|
|
|
|
}'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
|
|
}
|
|
|
|
|
2017-03-19 14:39:05 -04:00
|
|
|
/**
|
2017-04-24 23:45:02 -04:00
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function providerFileCheckerValidCodeParse()
|
|
|
|
{
|
|
|
|
return [
|
2018-01-21 10:22:04 -05:00
|
|
|
'nopType' => [
|
|
|
|
'<?php
|
|
|
|
$a = "hello";
|
|
|
|
|
|
|
|
/** @var int $a */',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'int',
|
|
|
|
],
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'deprecatedMethod' => [
|
|
|
|
'<?php
|
|
|
|
class Foo {
|
|
|
|
/**
|
|
|
|
* @deprecated
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
public static function barBar(): void {
|
2017-04-24 23:45:02 -04:00
|
|
|
}
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'validDocblockReturn' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(): string {
|
2017-04-24 23:45:02 -04:00
|
|
|
return "boop";
|
|
|
|
}
|
2017-05-04 18:35:05 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
/**
|
|
|
|
* @return array<int, string>
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo2(): array {
|
2017-04-24 23:45:02 -04:00
|
|
|
return ["hello"];
|
|
|
|
}
|
2017-05-04 18:35:05 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
/**
|
|
|
|
* @return array<int, string>
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo3(): array {
|
2017-04-24 23:45:02 -04:00
|
|
|
return ["hello"];
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'reassertWithIs' => [
|
|
|
|
'<?php
|
|
|
|
/** @param array $a */
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo($a): void {
|
2017-04-24 23:45:02 -04:00
|
|
|
if (is_array($a)) {
|
|
|
|
// do something
|
|
|
|
}
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2018-02-06 18:44:53 -05:00
|
|
|
'assertions' => [],
|
|
|
|
'error_level' => ['DocblockTypeContradiction'],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'checkArrayWithIs' => [
|
|
|
|
'<?php
|
|
|
|
/** @param mixed $b */
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo($b): void {
|
2017-04-24 23:45:02 -04:00
|
|
|
/** @var array */
|
|
|
|
$a = (array)$b;
|
|
|
|
if (is_array($a)) {
|
|
|
|
// do something
|
|
|
|
}
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2018-02-06 18:44:53 -05:00
|
|
|
'assertions' => [],
|
|
|
|
'error_level' => ['DocblockTypeContradiction'],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'checkArrayWithIsInsideLoop' => [
|
|
|
|
'<?php
|
|
|
|
/** @param array<mixed, array<mixed, mixed>> $data */
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo($data): void {
|
2017-04-24 23:45:02 -04:00
|
|
|
foreach ($data as $key => $val) {
|
|
|
|
if (!\is_array($data)) {
|
|
|
|
$data = [$key => null];
|
|
|
|
} else {
|
|
|
|
$data[$key] = !empty($val);
|
|
|
|
}
|
|
|
|
}
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-11-19 12:14:02 -05:00
|
|
|
'assertions' => [],
|
2018-02-06 18:44:53 -05:00
|
|
|
'error_level' => ['LoopInvalidation', 'MixedArrayOffset', 'DocblockTypeContradiction'],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'goodDocblock' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @param A $a
|
|
|
|
* @param bool $b
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
public function g(A $a, $b): void {
|
2017-04-24 23:45:02 -04:00
|
|
|
}
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'goodDocblockInNamespace' => [
|
|
|
|
'<?php
|
|
|
|
namespace Foo;
|
2017-05-04 18:35:05 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @param \Foo\A $a
|
|
|
|
* @param bool $b
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
public function g(A $a, $b): void {
|
2017-04-24 23:45:02 -04:00
|
|
|
}
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-05-04 18:35:05 -04:00
|
|
|
],
|
|
|
|
'propertyDocblock' => [
|
|
|
|
'<?php
|
2018-02-03 23:10:22 -05:00
|
|
|
namespace Bar;
|
|
|
|
|
2017-05-04 18:35:05 -04:00
|
|
|
/**
|
|
|
|
* @property string $foo
|
|
|
|
*/
|
|
|
|
class A {
|
2017-09-02 11:18:56 -04:00
|
|
|
/** @param string $name */
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __get($name): ?string {
|
2017-09-02 11:18:56 -04:00
|
|
|
if ($name === "foo") {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
}
|
2017-05-04 18:35:05 -04:00
|
|
|
|
2017-09-02 11:18:56 -04:00
|
|
|
/**
|
|
|
|
* @param string $name
|
|
|
|
* @param mixed $value
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __set($name, $value): void {
|
2017-09-02 11:18:56 -04:00
|
|
|
}
|
2017-05-04 18:35:05 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
2017-11-16 20:47:58 -05:00
|
|
|
$a->foo = "hello";
|
|
|
|
$a->bar = "hello"; // not a property',
|
2017-05-04 18:35:05 -04:00
|
|
|
],
|
2018-02-03 23:10:22 -05:00
|
|
|
'propertyOfTypeClassDocblock' => [
|
|
|
|
'<?php
|
|
|
|
namespace Bar;
|
|
|
|
|
|
|
|
class PropertyType {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @property PropertyType $foo
|
|
|
|
*/
|
|
|
|
class A {
|
|
|
|
/** @param string $name */
|
|
|
|
public function __get($name): ?string {
|
|
|
|
if ($name === "foo") {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $name
|
|
|
|
* @param mixed $value
|
|
|
|
*/
|
|
|
|
public function __set($name, $value): void {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
$a->foo = new PropertyType();',
|
|
|
|
],
|
2017-11-17 13:19:48 -08:00
|
|
|
'propertySealedDocblockDefinedPropertyFetch' => [
|
|
|
|
'<?php
|
2018-02-03 23:10:22 -05:00
|
|
|
namespace Bar;
|
2017-11-17 13:19:48 -08:00
|
|
|
/**
|
|
|
|
* @property string $foo
|
|
|
|
* @psalm-seal-properties
|
|
|
|
*/
|
|
|
|
class A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __get(string $name): ?string {
|
2017-11-17 13:19:48 -08:00
|
|
|
if ($name === "foo") {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @param mixed $value */
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __set(string $name, $value): void {
|
2017-11-17 13:19:48 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
echo $a->foo;',
|
|
|
|
],
|
2017-05-10 12:36:11 -04:00
|
|
|
'ignoreNullableReturn' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/** @var int */
|
|
|
|
public $bar = 5;
|
2018-01-11 15:50:45 -05:00
|
|
|
public function foo(): void {}
|
2017-05-10 12:36:11 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return ?A
|
|
|
|
* @psalm-ignore-nullable-return
|
|
|
|
*/
|
|
|
|
function makeA() {
|
2018-01-11 15:50:45 -05:00
|
|
|
return rand(0, 1) ? new A(): null;
|
2017-05-10 12:36:11 -04:00
|
|
|
}
|
|
|
|
|
2018-01-11 15:50:45 -05:00
|
|
|
function takeA(A $a): void { }
|
2017-05-10 12:36:11 -04:00
|
|
|
|
|
|
|
$a = makeA();
|
|
|
|
$a->foo();
|
|
|
|
$a->bar = 7;
|
2017-05-26 20:05:57 -04:00
|
|
|
takeA($a);',
|
2017-05-10 12:36:11 -04:00
|
|
|
],
|
2017-06-13 14:00:41 -04:00
|
|
|
'invalidDocblockParamSuppress' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param int $bar
|
2018-01-05 12:11:12 -05:00
|
|
|
* @psalm-suppress MismatchingDocblockParamType
|
2017-06-13 14:00:41 -04:00
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(array $bar): void {
|
2017-06-13 14:00:41 -04:00
|
|
|
}',
|
|
|
|
],
|
2017-07-25 16:11:02 -04:00
|
|
|
'differentDocblockParamClassSuppress' => [
|
|
|
|
'<?php
|
|
|
|
class A {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param B $bar
|
2018-01-05 12:11:12 -05:00
|
|
|
* @psalm-suppress MismatchingDocblockParamType
|
2017-07-25 16:11:02 -04:00
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(A $bar): void {
|
2017-07-25 16:11:02 -04:00
|
|
|
}',
|
|
|
|
],
|
|
|
|
'varDocblock' => [
|
|
|
|
'<?php
|
|
|
|
/** @var array<Exception> */
|
|
|
|
$a = [];
|
|
|
|
|
|
|
|
$a[0]->getMessage();',
|
|
|
|
],
|
2017-09-02 11:18:56 -04:00
|
|
|
'mixedDocblockParamTypeDefinedInParent' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/** @param mixed $a */
|
2018-01-11 15:50:45 -05:00
|
|
|
public function foo($a): void {}
|
2017-09-02 11:18:56 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
class B extends A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function foo($a): void {}
|
2017-09-02 11:18:56 -04:00
|
|
|
}',
|
|
|
|
],
|
|
|
|
'intDocblockParamTypeDefinedInParent' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/** @param int $a */
|
2018-01-11 15:50:45 -05:00
|
|
|
public function foo($a): void {}
|
2017-09-02 11:18:56 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
class B extends A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function foo($a): void {}
|
2017-09-02 11:18:56 -04:00
|
|
|
}',
|
|
|
|
],
|
2017-10-07 10:22:52 -04:00
|
|
|
'varSelf' => [
|
|
|
|
'<?php
|
|
|
|
class A
|
|
|
|
{
|
2018-01-11 15:50:45 -05:00
|
|
|
public function foo(): void {}
|
2017-10-07 10:22:52 -04:00
|
|
|
|
2018-01-11 15:50:45 -05:00
|
|
|
public function getMeAgain(): void {
|
2017-10-07 10:22:52 -04:00
|
|
|
/** @var self */
|
|
|
|
$me = $this;
|
|
|
|
$me->foo();
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2017-11-02 21:45:17 -04:00
|
|
|
'psalmVar' => [
|
|
|
|
'<?php
|
|
|
|
class A
|
|
|
|
{
|
|
|
|
/** @psalm-var array<int, string> */
|
|
|
|
public $foo = [];
|
|
|
|
|
2018-01-11 15:50:45 -05:00
|
|
|
public function updateFoo(): void {
|
2017-11-02 21:45:17 -04:00
|
|
|
$this->foo[5] = "hello";
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'psalmParam' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function takesInt(int $a): void {}
|
2017-11-02 21:45:17 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @psalm-param array<int, string> $a
|
|
|
|
* @param string[] $a
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo(array $a): void {
|
2017-11-02 21:45:17 -04:00
|
|
|
foreach ($a as $key => $value) {
|
|
|
|
takesInt($key);
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2017-12-30 10:54:01 -05:00
|
|
|
'returnDocblock' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo(int $i): int {
|
2017-12-30 10:54:01 -05:00
|
|
|
/** @var int */
|
|
|
|
return $i;
|
|
|
|
}',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function providerFileCheckerInvalidCodeParse()
|
|
|
|
{
|
|
|
|
return [
|
2017-05-24 21:11:18 -04:00
|
|
|
'invalidReturn' => [
|
|
|
|
'<?php
|
|
|
|
interface I {
|
|
|
|
/**
|
|
|
|
* @return $thus
|
|
|
|
*/
|
|
|
|
public static function barBar();
|
|
|
|
}',
|
2017-11-14 21:43:31 -05:00
|
|
|
'error_message' => 'MissingDocblockType',
|
2017-05-24 21:11:18 -04:00
|
|
|
],
|
2017-07-09 20:32:35 -04:00
|
|
|
'invalidReturnClass' => [
|
|
|
|
'<?php
|
|
|
|
interface I {
|
|
|
|
/**
|
|
|
|
* @return 1
|
|
|
|
*/
|
|
|
|
public static function barBar();
|
|
|
|
}',
|
|
|
|
'error_message' => 'InvalidDocblock',
|
|
|
|
],
|
|
|
|
'invalidReturnClassWithComma' => [
|
|
|
|
'<?php
|
|
|
|
interface I {
|
|
|
|
/**
|
|
|
|
* @return 1,
|
|
|
|
*/
|
|
|
|
public static function barBar();
|
|
|
|
}',
|
|
|
|
'error_message' => 'InvalidDocblock',
|
|
|
|
],
|
|
|
|
'returnClassWithComma' => [
|
|
|
|
'<?php
|
|
|
|
interface I {
|
|
|
|
/**
|
|
|
|
* @return a,
|
|
|
|
*/
|
|
|
|
public static function barBar();
|
|
|
|
}',
|
|
|
|
'error_message' => 'InvalidDocblock',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'deprecatedMethodWithCall' => [
|
|
|
|
'<?php
|
|
|
|
class Foo {
|
|
|
|
/**
|
|
|
|
* @deprecated
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
public static function barBar(): void {
|
2017-04-24 23:45:02 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Foo::barBar();',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'DeprecatedMethod',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2017-05-25 00:34:39 -04:00
|
|
|
'deprecatedClassWithStaticCall' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @deprecated
|
|
|
|
*/
|
|
|
|
class Foo {
|
2018-01-11 15:50:45 -05:00
|
|
|
public static function barBar(): void {
|
2017-05-25 00:34:39 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Foo::barBar();',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'DeprecatedClass',
|
2017-05-25 00:34:39 -04:00
|
|
|
],
|
|
|
|
'deprecatedClassWithNew' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @deprecated
|
|
|
|
*/
|
|
|
|
class Foo { }
|
|
|
|
|
|
|
|
$a = new Foo();',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'DeprecatedClass',
|
2017-05-25 00:34:39 -04:00
|
|
|
],
|
2018-01-23 09:09:43 -05:00
|
|
|
'deprecatedClassWithExtends' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @deprecated
|
|
|
|
*/
|
|
|
|
class Foo { }
|
|
|
|
|
|
|
|
class Bar extends Foo {}',
|
|
|
|
'error_message' => 'DeprecatedClass',
|
|
|
|
],
|
2017-05-25 01:32:34 -04:00
|
|
|
'deprecatedPropertyGet' => [
|
|
|
|
'<?php
|
|
|
|
class A{
|
|
|
|
/**
|
|
|
|
* @deprecated
|
|
|
|
* @var ?int
|
|
|
|
*/
|
|
|
|
public $foo;
|
|
|
|
}
|
|
|
|
echo (new A)->foo;',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'DeprecatedProperty',
|
2017-05-25 01:32:34 -04:00
|
|
|
],
|
|
|
|
'deprecatedPropertySet' => [
|
|
|
|
'<?php
|
|
|
|
class A{
|
|
|
|
/**
|
|
|
|
* @deprecated
|
|
|
|
* @var ?int
|
|
|
|
*/
|
|
|
|
public $foo;
|
|
|
|
}
|
|
|
|
$a = new A;
|
|
|
|
$a->foo = 5;',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'DeprecatedProperty',
|
2017-05-25 01:32:34 -04:00
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'missingParamType' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
2017-07-25 16:11:02 -04:00
|
|
|
* @param string $bar
|
2017-04-24 23:45:02 -04:00
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooBar(): void {
|
2017-07-25 16:11:02 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
fooBar("hello");',
|
2018-02-02 11:26:55 -05:00
|
|
|
'error_message' => 'TooManyArguments - src/somefile.php:8 - Too many arguments for method fooBar '
|
|
|
|
. '- expecting 0 but saw 1',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'missingParamVar' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param string
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooBar(): void {
|
2017-04-24 23:45:02 -04:00
|
|
|
}',
|
2017-07-26 21:30:01 -04:00
|
|
|
'error_message' => 'InvalidDocblock - src/somefile.php:5 - Badly-formatted @param',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'invalidDocblockReturn' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(): int {
|
2017-07-25 16:11:02 -04:00
|
|
|
return 5;
|
2017-04-24 23:45:02 -04:00
|
|
|
}',
|
2018-01-05 12:11:12 -05:00
|
|
|
'error_message' => 'MismatchingDocblockReturnType',
|
2017-05-04 18:35:05 -04:00
|
|
|
],
|
|
|
|
'propertyDocblockInvalidAssignment' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @property string $foo
|
|
|
|
*/
|
|
|
|
class A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __get(string $name): ?string {
|
2017-05-04 18:35:05 -04:00
|
|
|
if ($name === "foo") {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-03 12:27:01 -04:00
|
|
|
/** @param mixed $value */
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __set(string $name, $value): void {
|
2017-05-04 18:35:05 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
$a->foo = 5;',
|
2018-01-11 14:38:24 -08:00
|
|
|
'error_message' => 'InvalidPropertyAssignmentValue',
|
2017-05-04 18:35:05 -04:00
|
|
|
],
|
2018-02-03 23:10:22 -05:00
|
|
|
'propertyInvalidClassAssignment' => [
|
|
|
|
'<?php
|
|
|
|
namespace Bar;
|
|
|
|
|
|
|
|
class PropertyType {}
|
|
|
|
class SomeOtherPropertyType {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @property PropertyType $foo
|
|
|
|
*/
|
|
|
|
class A {
|
|
|
|
/** @param string $name */
|
|
|
|
public function __get($name): ?string {
|
|
|
|
if ($name === "foo") {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $name
|
|
|
|
* @param mixed $value
|
|
|
|
*/
|
|
|
|
public function __set($name, $value): void {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
$a->foo = new SomeOtherPropertyType();',
|
|
|
|
'error_message' => 'InvalidPropertyAssignmentValue - src/somefile.php:27 - $a->foo with declared type'
|
|
|
|
. ' \'Bar\PropertyType\' cannot',
|
|
|
|
],
|
2017-12-14 11:22:27 -08:00
|
|
|
'propertyWriteDocblockInvalidAssignment' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @property-write string $foo
|
|
|
|
*/
|
|
|
|
class A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __get(string $name): ?string {
|
2017-12-14 11:22:27 -08:00
|
|
|
if ($name === "foo") {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @param mixed $value */
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __set(string $name, $value): void {
|
2017-12-14 11:22:27 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
$a->foo = 5;',
|
2018-01-11 14:38:24 -08:00
|
|
|
'error_message' => 'InvalidPropertyAssignmentValue',
|
2017-12-14 11:22:27 -08:00
|
|
|
],
|
2017-11-16 20:47:58 -05:00
|
|
|
'propertySealedDocblockUndefinedPropertyAssignment' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @property string $foo
|
|
|
|
* @psalm-seal-properties
|
|
|
|
*/
|
|
|
|
class A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __get(string $name): ?string {
|
2017-11-16 20:47:58 -05:00
|
|
|
if ($name === "foo") {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @param mixed $value */
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __set(string $name, $value): void {
|
2017-11-16 20:47:58 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
$a->bar = 5;',
|
|
|
|
'error_message' => 'UndefinedPropertyAssignment',
|
|
|
|
],
|
2017-11-17 13:19:48 -08:00
|
|
|
'propertySealedDocblockDefinedPropertyAssignment' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @property string $foo
|
|
|
|
* @psalm-seal-properties
|
|
|
|
*/
|
|
|
|
class A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __get(string $name): ?string {
|
2017-11-17 13:19:48 -08:00
|
|
|
if ($name === "foo") {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @param mixed $value */
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __set(string $name, $value): void {
|
2017-11-17 13:19:48 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
$a->foo = 5;',
|
2018-01-11 14:38:24 -08:00
|
|
|
'error_message' => 'InvalidPropertyAssignmentValue',
|
2017-11-17 13:19:48 -08:00
|
|
|
],
|
2017-12-14 11:22:27 -08:00
|
|
|
'propertyReadInvalidFetch' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @property-read string $foo
|
|
|
|
*/
|
|
|
|
class A {
|
|
|
|
/** @return mixed */
|
|
|
|
public function __get(string $name) {
|
|
|
|
if ($name === "foo") {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
echo count($a->foo);',
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
],
|
2017-11-16 20:47:58 -05:00
|
|
|
'propertySealedDocblockUndefinedPropertyFetch' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @property string $foo
|
|
|
|
* @psalm-seal-properties
|
|
|
|
*/
|
|
|
|
class A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __get(string $name): ?string {
|
2017-11-16 20:47:58 -05:00
|
|
|
if ($name === "foo") {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @param mixed $value */
|
2018-01-11 15:50:45 -05:00
|
|
|
public function __set(string $name, $value): void {
|
2017-11-16 20:47:58 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
echo $a->bar;',
|
|
|
|
'error_message' => 'UndefinedPropertyFetch',
|
|
|
|
],
|
2017-09-02 18:15:52 -04:00
|
|
|
'noStringParamType' => [
|
2017-09-02 11:18:56 -04:00
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo($a): void {
|
2017-09-02 18:15:52 -04:00
|
|
|
echo substr($a, 4, 2);
|
2017-09-02 11:18:56 -04:00
|
|
|
}',
|
2018-01-28 20:03:47 -05:00
|
|
|
'error_message' => 'MissingParamType - src/somefile.php:2 - Parameter $a has no provided type,'
|
2017-09-02 18:15:52 -04:00
|
|
|
. ' should be string',
|
|
|
|
'error_levels' => ['MixedArgument'],
|
|
|
|
],
|
2017-09-02 19:23:00 -04:00
|
|
|
'noParamTypeButConcat' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo($a): void {
|
2017-09-02 19:23:00 -04:00
|
|
|
echo $a . "foo";
|
|
|
|
}',
|
2018-01-28 20:03:47 -05:00
|
|
|
'error_message' => 'MissingParamType - src/somefile.php:2 - Parameter $a has no provided type,'
|
2017-09-02 19:23:00 -04:00
|
|
|
. ' should be string',
|
|
|
|
'error_levels' => ['MixedOperand'],
|
|
|
|
],
|
2017-09-06 21:44:26 -04:00
|
|
|
'noParamTypeButAddition' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo($a): void {
|
2017-09-06 21:44:26 -04:00
|
|
|
echo $a + 5;
|
|
|
|
}',
|
2018-01-28 20:03:47 -05:00
|
|
|
'error_message' => 'MissingParamType - src/somefile.php:2 - Parameter $a has no provided type,'
|
2017-09-06 21:44:26 -04:00
|
|
|
. ' should be int|float',
|
|
|
|
'error_levels' => ['MixedOperand', 'MixedArgument'],
|
|
|
|
],
|
|
|
|
'noParamTypeButDivision' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo($a): void {
|
2017-09-06 21:44:26 -04:00
|
|
|
echo $a / 5;
|
|
|
|
}',
|
2018-01-28 20:03:47 -05:00
|
|
|
'error_message' => 'MissingParamType - src/somefile.php:2 - Parameter $a has no provided type,'
|
2017-09-06 21:44:26 -04:00
|
|
|
. ' should be int|float',
|
|
|
|
'error_levels' => ['MixedOperand', 'MixedArgument'],
|
|
|
|
],
|
2017-09-02 19:48:59 -04:00
|
|
|
'noParamTypeButTemplatedString' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo($a): void {
|
2017-09-02 19:48:59 -04:00
|
|
|
echo "$a";
|
|
|
|
}',
|
2018-01-28 20:03:47 -05:00
|
|
|
'error_message' => 'MissingParamType - src/somefile.php:2 - Parameter $a has no provided type,'
|
2017-09-02 19:48:59 -04:00
|
|
|
. ' should be string',
|
|
|
|
'error_levels' => ['MixedOperand'],
|
|
|
|
],
|
2017-09-02 18:15:52 -04:00
|
|
|
'noStringIntParamType' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo($a): void {
|
2017-09-02 18:15:52 -04:00
|
|
|
if (is_string($a)) {
|
|
|
|
echo substr($a, 4, 2);
|
|
|
|
} else {
|
|
|
|
echo substr("hello", $a, 2);
|
|
|
|
}
|
|
|
|
}',
|
2018-01-28 20:03:47 -05:00
|
|
|
'error_message' => 'MissingParamType - src/somefile.php:2 - Parameter $a has no provided type,'
|
2017-09-02 18:15:52 -04:00
|
|
|
. ' should be int|string',
|
|
|
|
'error_levels' => ['MixedArgument'],
|
2017-09-02 11:18:56 -04:00
|
|
|
],
|
|
|
|
'intParamTypeDefinedInParent' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function foo(int $a): void {}
|
2017-09-02 11:18:56 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
class B extends A {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function foo($a): void {}
|
2017-09-02 11:18:56 -04:00
|
|
|
}',
|
2018-01-28 20:03:47 -05:00
|
|
|
'error_message' => 'MissingParamType',
|
2017-09-02 11:18:56 -04:00
|
|
|
'error_levels' => ['MethodSignatureMismatch'],
|
|
|
|
],
|
2017-09-02 18:15:52 -04:00
|
|
|
'alreadyHasCheck' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function takesString(string $s): void {}
|
2017-09-02 18:15:52 -04:00
|
|
|
|
2018-01-11 15:50:45 -05:00
|
|
|
function shouldTakeString($s): void {
|
2017-09-02 18:15:52 -04:00
|
|
|
if (is_string($s)) takesString($s);
|
|
|
|
}',
|
2018-01-28 20:03:47 -05:00
|
|
|
'error_message' => 'MissingParamType - src/somefile.php:4 - Parameter $s has no provided type,'
|
2017-09-02 18:15:52 -04:00
|
|
|
. ' could not infer',
|
|
|
|
'error_levels' => ['MixedArgument'],
|
|
|
|
],
|
|
|
|
'isSetBeforeInferrence' => [
|
|
|
|
'input' => '<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function takesString(string $s): void {}
|
2017-09-02 18:15:52 -04:00
|
|
|
|
|
|
|
/** @return mixed */
|
|
|
|
function returnsMixed() {}
|
|
|
|
|
2018-01-11 15:50:45 -05:00
|
|
|
function shouldTakeString($s): void {
|
2017-09-02 18:15:52 -04:00
|
|
|
$s = returnsMixed();
|
|
|
|
takesString($s);
|
|
|
|
}',
|
2018-01-28 20:03:47 -05:00
|
|
|
'error_message' => 'MissingParamType - src/somefile.php:7 - Parameter $s has no provided type,'
|
2017-09-02 18:15:52 -04:00
|
|
|
. ' could not infer',
|
|
|
|
'error_levels' => ['MixedArgument', 'InvalidReturnType', 'MixedAssignment'],
|
|
|
|
],
|
2017-11-02 21:45:17 -04:00
|
|
|
'psalmInvalidVar' => [
|
|
|
|
'<?php
|
|
|
|
class A
|
|
|
|
{
|
|
|
|
/** @psalm-var array<int, string> */
|
|
|
|
public $foo = [];
|
|
|
|
|
2018-01-11 15:50:45 -05:00
|
|
|
public function updateFoo(): void {
|
2017-11-02 21:45:17 -04:00
|
|
|
$this->foo["boof"] = "hello";
|
|
|
|
}
|
|
|
|
}',
|
2018-01-11 14:38:24 -08:00
|
|
|
'error_message' => 'InvalidPropertyAssignmentValue',
|
2017-11-02 21:45:17 -04:00
|
|
|
],
|
2017-11-14 21:43:31 -05:00
|
|
|
'incorrectDocblockOrder' => [
|
|
|
|
'<?php
|
|
|
|
class MyClass {
|
|
|
|
/**
|
|
|
|
* Comment
|
|
|
|
* @var $fooPropTypo string
|
|
|
|
*/
|
|
|
|
public $fooProp = "/tmp/file.txt";
|
|
|
|
}',
|
|
|
|
'error_message' => 'MissingDocblockType',
|
|
|
|
],
|
|
|
|
'badlyFormattedVar' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @return string[]
|
|
|
|
*/
|
|
|
|
function returns_strings() {
|
|
|
|
/** @var array(string) $result */
|
|
|
|
$result = ["example"];
|
|
|
|
return $result;
|
|
|
|
}',
|
|
|
|
'error_message' => 'InvalidDocblock',
|
|
|
|
],
|
|
|
|
'badlyWrittenVar' => [
|
|
|
|
'<?php
|
|
|
|
/** @param mixed $x */
|
2018-01-11 15:50:45 -05:00
|
|
|
function myvalue($x): void {
|
2017-11-14 21:43:31 -05:00
|
|
|
/** @var $myVar MyNS\OtherClass */
|
|
|
|
$myVar = $x->conn()->method();
|
|
|
|
$myVar->otherMethod();
|
|
|
|
}',
|
|
|
|
'error_message' => 'MissingDocblockType',
|
|
|
|
],
|
2018-01-08 17:17:49 -05:00
|
|
|
'dontOverrideSameType' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/** @return ?int */
|
2018-01-11 15:50:45 -05:00
|
|
|
public function foo(): ?int {
|
2018-01-08 17:17:49 -05:00
|
|
|
if (rand(0, 1)) return 5;
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'error_message' => 'InvalidReturnType',
|
|
|
|
],
|
2018-01-09 22:46:55 -05:00
|
|
|
'alwaysCheckReturnType' => [
|
|
|
|
'<?php
|
|
|
|
class A {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return A
|
|
|
|
* @psalm-suppress MismatchingDocblockReturnType
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo(): B {
|
2018-01-09 22:46:55 -05:00
|
|
|
return new A;
|
|
|
|
}',
|
|
|
|
'error_message' => 'UndefinedClass',
|
|
|
|
],
|
2018-01-10 00:07:47 -05:00
|
|
|
'preventBadBoolean' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo(): boolean {
|
2018-01-10 00:07:47 -05:00
|
|
|
return true;
|
|
|
|
}',
|
|
|
|
'error_message' => 'UndefinedClass',
|
|
|
|
],
|
2018-01-19 16:06:30 -05:00
|
|
|
'preventBadObjectLikeFormat' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param array{} $arr
|
|
|
|
*/
|
|
|
|
function bar(array $arr): void {}',
|
|
|
|
'error_message' => 'InvalidDocblock',
|
2018-01-20 11:48:16 -05:00
|
|
|
],
|
2018-02-01 01:10:27 -05:00
|
|
|
'noPhpStormAnnotationsThankYou' => [
|
|
|
|
'<?php
|
|
|
|
/** @param ArrayIterator|string[] $i */
|
|
|
|
function takesArrayIteratorOfString(ArrayIterator $i): void {}',
|
|
|
|
'error_message' => 'MismatchingDocblockParamType',
|
|
|
|
],
|
|
|
|
'noPhpStormAnnotationsPossiblyInvalid' => [
|
|
|
|
'<?php
|
|
|
|
/** @param ArrayIterator|string[] $i */
|
|
|
|
function takesArrayIteratorOfString($i): void {
|
|
|
|
$s = $i->offsetGet("a");
|
|
|
|
}',
|
|
|
|
'error_message' => 'PossiblyInvalidMethodCall',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
];
|
2016-12-31 00:14:00 -05:00
|
|
|
}
|
2016-12-11 23:41:11 -05:00
|
|
|
}
|