mirror of
https://github.com/danog/psalm.git
synced 2024-12-11 16:59:45 +01:00
16c0496d1c
ISSUE-5962 Fixed wrong line number for @method annotations
2054 lines
67 KiB
PHP
2054 lines
67 KiB
PHP
<?php
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
use Psalm\Config;
|
|
use Psalm\Context;
|
|
use Psalm\Exception\CodeException;
|
|
use Psalm\Tests\Traits\InvalidCodeAnalysisTestTrait;
|
|
use Psalm\Tests\Traits\ValidCodeAnalysisTestTrait;
|
|
|
|
use const DIRECTORY_SEPARATOR;
|
|
|
|
class AnnotationTest extends TestCase
|
|
{
|
|
use InvalidCodeAnalysisTestTrait;
|
|
use ValidCodeAnalysisTestTrait;
|
|
|
|
public function setUp(): void
|
|
{
|
|
parent::setUp();
|
|
$codebase = $this->project_analyzer->getCodebase();
|
|
$codebase->reportUnusedVariables();
|
|
}
|
|
|
|
public function testPhpStormGenericsWithValidArrayIteratorArgument(): void
|
|
{
|
|
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());
|
|
}
|
|
|
|
public function testPhpStormGenericsWithTypeInSignature(): void
|
|
{
|
|
Config::getInstance()->allow_phpstorm_generics = true;
|
|
|
|
$this->addFile(
|
|
'somefile.php',
|
|
'<?php
|
|
function a(array|\ArrayObject $_meta = []): void {}'
|
|
);
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
}
|
|
|
|
public function testPhpStormGenericsWithValidTraversableArgument(): void
|
|
{
|
|
Config::getInstance()->allow_phpstorm_generics = true;
|
|
|
|
$this->addFile(
|
|
'somefile.php',
|
|
'<?php
|
|
function takesString(string $_s): void {}
|
|
|
|
/** @param Traversable|string[] $i */
|
|
function takesTraversableOfString(Traversable $i): void {
|
|
foreach ($i as $s2) {
|
|
takesString($s2);
|
|
}
|
|
}'
|
|
);
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
}
|
|
|
|
public function testPhpStormGenericsWithClassProperty(): void
|
|
{
|
|
Config::getInstance()->allow_phpstorm_generics = true;
|
|
|
|
$this->addFile(
|
|
'somefile.php',
|
|
'<?php
|
|
/** @psalm-suppress MissingConstructor */
|
|
class Foo {
|
|
/** @var \stdClass[]|\ArrayObject */
|
|
public $bar;
|
|
|
|
/**
|
|
* @return \stdClass[]|\ArrayObject
|
|
*/
|
|
public function getBar(): \ArrayObject {
|
|
return $this->bar;
|
|
}
|
|
}'
|
|
);
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
}
|
|
|
|
public function testPhpStormGenericsWithGeneratorArray(): void
|
|
{
|
|
Config::getInstance()->allow_phpstorm_generics = true;
|
|
|
|
$this->addFile(
|
|
'somefile.php',
|
|
'<?php
|
|
class A {
|
|
/**
|
|
* @return stdClass[]|Generator
|
|
*/
|
|
function getCollection(): Generator
|
|
{
|
|
yield new stdClass;
|
|
}
|
|
}'
|
|
);
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
}
|
|
|
|
public function testPhpStormGenericsWithValidIterableArgument(): void
|
|
{
|
|
Config::getInstance()->allow_phpstorm_generics = true;
|
|
|
|
$this->addFile(
|
|
'somefile.php',
|
|
'<?php
|
|
function takesString(string $_s): void {}
|
|
|
|
/** @param iterable|string[] $i */
|
|
function takesArrayIteratorOfString(iterable $i): void {
|
|
foreach ($i as $s2) {
|
|
takesString($s2);
|
|
}
|
|
}'
|
|
);
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
}
|
|
|
|
public function testPhpStormGenericsInvalidArgument(): void
|
|
{
|
|
$this->expectException(CodeException::class);
|
|
$this->expectExceptionMessage('InvalidScalarArgument');
|
|
|
|
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());
|
|
}
|
|
|
|
public function testLessSpecificImplementedReturnTypeWithDocblockOnMultipleLines(): void
|
|
{
|
|
$this->expectException(CodeException::class);
|
|
$this->expectExceptionMessage('LessSpecificImplementedReturnType - somefile.php:5:');
|
|
|
|
$this->addFile(
|
|
'somefile.php',
|
|
'<?php
|
|
|
|
/**
|
|
* @method int test()
|
|
* @method \DateTime modify($modify)
|
|
*/
|
|
class WTF extends \DateTime { }'
|
|
);
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
}
|
|
|
|
public function testLessSpecificImplementedReturnTypeWithDocblockOnMultipleLinesWithMultipleClasses(): void
|
|
{
|
|
$this->expectException(CodeException::class);
|
|
$this->expectExceptionMessage('LessSpecificImplementedReturnType - somefile.php:15:');
|
|
|
|
$this->addFile(
|
|
'somefile.php',
|
|
'<?php
|
|
|
|
class ParentClass
|
|
{
|
|
/**
|
|
* @return $this
|
|
*/
|
|
public function execute()
|
|
{
|
|
return $this;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @method self execute()
|
|
*/
|
|
class BreakingThings extends ParentClass
|
|
{
|
|
}'
|
|
);
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
}
|
|
|
|
public function testLessSpecificImplementedReturnTypeWithDescription(): void
|
|
{
|
|
$this->expectException(CodeException::class);
|
|
$this->expectExceptionMessage('LessSpecificImplementedReturnType - somefile.php:7:');
|
|
|
|
$this->addFile(
|
|
'somefile.php',
|
|
'<?php
|
|
/**
|
|
* test test test
|
|
* test rambling text
|
|
* test test text
|
|
*
|
|
* @method \DateTime modify($modify)
|
|
*/
|
|
class WTF extends \DateTime { }'
|
|
);
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
}
|
|
|
|
public function testPhpStormGenericsNoTypehint(): void
|
|
{
|
|
$this->expectException(CodeException::class);
|
|
$this->expectExceptionMessage('PossiblyInvalidMethodCall');
|
|
|
|
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());
|
|
}
|
|
|
|
public function testInvalidParamDefault(): void
|
|
{
|
|
$this->expectException(CodeException::class);
|
|
$this->expectExceptionMessage('InvalidParamDefault');
|
|
|
|
$this->addFile(
|
|
'somefile.php',
|
|
'<?php
|
|
/**
|
|
* @param array $arr
|
|
* @return void
|
|
*/
|
|
function foo($arr = false) {}'
|
|
);
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
}
|
|
|
|
public function testInvalidParamDefaultButAllowedInConfig(): void
|
|
{
|
|
Config::getInstance()->add_param_default_to_docblock_type = true;
|
|
|
|
$this->addFile(
|
|
'somefile.php',
|
|
'<?php
|
|
/**
|
|
* @param array $_arr
|
|
* @return void
|
|
*/
|
|
function foo($_arr = false) {}
|
|
foo(false);
|
|
foo(["hello"]);'
|
|
);
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
}
|
|
|
|
public function testInvalidTypehintParamDefaultButAllowedInConfig(): void
|
|
{
|
|
$this->expectException(CodeException::class);
|
|
$this->expectExceptionMessage('InvalidParamDefault');
|
|
|
|
Config::getInstance()->add_param_default_to_docblock_type = true;
|
|
|
|
$this->addFile(
|
|
'somefile.php',
|
|
'<?php
|
|
function foo(array $arr = false) : void {}'
|
|
);
|
|
|
|
$this->analyzeFile('somefile.php', new Context());
|
|
}
|
|
|
|
/**
|
|
* @return iterable<string,array{string,assertions?:array<string,string>,error_levels?:string[]}>
|
|
*/
|
|
public function providerValidCodeParse(): iterable
|
|
{
|
|
return [
|
|
'nopType' => [
|
|
'<?php
|
|
$_a = "hello";
|
|
|
|
/** @var int $_a */',
|
|
'assertions' => [
|
|
'$_a' => 'int',
|
|
],
|
|
],
|
|
'validDocblockReturn' => [
|
|
'<?php
|
|
/**
|
|
* @return string
|
|
*/
|
|
function fooFoo(): string {
|
|
return "boop";
|
|
}
|
|
|
|
/**
|
|
* @return array<int, string>
|
|
*/
|
|
function foo2(): array {
|
|
return ["hello"];
|
|
}
|
|
|
|
/**
|
|
* @return array<int, string>
|
|
*/
|
|
function foo3(): array {
|
|
return ["hello"];
|
|
}',
|
|
],
|
|
'reassertWithIs' => [
|
|
'<?php
|
|
/** @param array $a */
|
|
function foo($a): void {
|
|
if (is_array($a)) {
|
|
// do something
|
|
}
|
|
}',
|
|
'assertions' => [],
|
|
'error_level' => ['RedundantConditionGivenDocblockType'],
|
|
],
|
|
'checkArrayWithIs' => [
|
|
'<?php
|
|
/** @param mixed $b */
|
|
function foo($b): void {
|
|
/**
|
|
* @psalm-suppress UnnecessaryVarAnnotation
|
|
* @var array
|
|
*/
|
|
$a = (array)$b;
|
|
if (is_array($a)) {
|
|
// do something
|
|
}
|
|
}',
|
|
'assertions' => [],
|
|
'error_level' => ['RedundantConditionGivenDocblockType'],
|
|
],
|
|
'goodDocblock' => [
|
|
'<?php
|
|
class A {
|
|
/**
|
|
* @param A $a
|
|
* @param bool $b
|
|
*/
|
|
public function g(A $a, $b): void {
|
|
}
|
|
}',
|
|
],
|
|
'goodDocblockInNamespace' => [
|
|
'<?php
|
|
namespace Foo;
|
|
|
|
class A {
|
|
/**
|
|
* @param \Foo\A $a
|
|
* @param bool $b
|
|
*/
|
|
public function g(A $a, $b): void {
|
|
}
|
|
}',
|
|
],
|
|
|
|
'ignoreNullableReturn' => [
|
|
'<?php
|
|
class A {
|
|
/** @var int */
|
|
public $bar = 5;
|
|
public function foo(): void {}
|
|
}
|
|
|
|
/**
|
|
* @return ?A
|
|
* @psalm-ignore-nullable-return
|
|
*/
|
|
function makeA() {
|
|
return rand(0, 1) ? new A() : null;
|
|
}
|
|
|
|
function takeA(A $_a): void { }
|
|
|
|
$a = makeA();
|
|
$a->foo();
|
|
$a->bar = 7;
|
|
takeA($a);',
|
|
],
|
|
'invalidDocblockParamSuppress' => [
|
|
'<?php
|
|
/**
|
|
* @param int $_bar
|
|
* @psalm-suppress MismatchingDocblockParamType
|
|
*/
|
|
function fooFoo(array $_bar): void {
|
|
}',
|
|
],
|
|
'differentDocblockParamClassSuppress' => [
|
|
'<?php
|
|
class A {}
|
|
class B {}
|
|
|
|
/**
|
|
* @param B $_bar
|
|
* @psalm-suppress MismatchingDocblockParamType
|
|
*/
|
|
function fooFoo(A $_bar): void {
|
|
}',
|
|
],
|
|
'varDocblock' => [
|
|
'<?php
|
|
/** @var array<Exception> */
|
|
$a = [];
|
|
|
|
echo $a[0]->getMessage();',
|
|
],
|
|
'ignoreVarDocblock' => [
|
|
'<?php
|
|
/**
|
|
* @var array<Exception>
|
|
* @ignore-var
|
|
*/
|
|
$a = [];
|
|
|
|
$a[0]->getMessage();',
|
|
'assertions' => [],
|
|
'error_level' => ['EmptyArrayAccess', 'MixedMethodCall'],
|
|
],
|
|
'psalmIgnoreVarDocblock' => [
|
|
'<?php
|
|
/**
|
|
* @var array<Exception>
|
|
* @psalm-ignore-var
|
|
*/
|
|
$a = [];
|
|
|
|
$a[0]->getMessage();',
|
|
'assertions' => [],
|
|
'error_level' => ['EmptyArrayAccess', 'MixedMethodCall'],
|
|
],
|
|
'mixedDocblockParamTypeDefinedInParent' => [
|
|
'<?php
|
|
class A {
|
|
/** @param mixed $a */
|
|
public function foo($a): void {}
|
|
}
|
|
|
|
class B extends A {
|
|
public function foo($a): void {}
|
|
}',
|
|
],
|
|
'intDocblockParamTypeDefinedInParent' => [
|
|
'<?php
|
|
class A {
|
|
/** @param int $a */
|
|
public function foo($a): void {}
|
|
}
|
|
|
|
class B extends A {
|
|
public function foo($a): void {}
|
|
}',
|
|
],
|
|
'varSelf' => [
|
|
'<?php
|
|
class A
|
|
{
|
|
public function foo(): void {}
|
|
|
|
public function getMeAgain(): void {
|
|
/** @var self */
|
|
$me = $this;
|
|
$me->foo();
|
|
}
|
|
}',
|
|
],
|
|
'psalmVar' => [
|
|
'<?php
|
|
class A
|
|
{
|
|
/** @psalm-var array<int, string> */
|
|
public $foo = [];
|
|
|
|
public function updateFoo(): void {
|
|
$this->foo[5] = "hello";
|
|
}
|
|
}',
|
|
],
|
|
'psalmParam' => [
|
|
'<?php
|
|
function takesInt(int $_a): void {}
|
|
|
|
/**
|
|
* @psalm-param array<int, string> $a
|
|
* @param string[] $a
|
|
*/
|
|
function foo(array $a): void {
|
|
foreach ($a as $key => $_value) {
|
|
takesInt($key);
|
|
}
|
|
}',
|
|
],
|
|
'returnDocblock' => [
|
|
'<?php
|
|
function foo(int $i): int {
|
|
/** @var int */
|
|
return $i;
|
|
}',
|
|
],
|
|
'doubleVar' => [
|
|
'<?php
|
|
function foo() : array {
|
|
return ["hello" => new stdClass, "goodbye" => new stdClass];
|
|
}
|
|
|
|
$_a = null;
|
|
$_b = null;
|
|
|
|
/**
|
|
* @var string $_key
|
|
* @var stdClass $_value
|
|
*/
|
|
foreach (foo() as $_key => $_value) {
|
|
$_a = $_key;
|
|
$_b = $_value;
|
|
}',
|
|
'assertions' => [
|
|
'$_a' => 'null|string',
|
|
'$_b' => 'null|stdClass',
|
|
],
|
|
],
|
|
'allowOptionalParamsToBeEmptyArray' => [
|
|
'<?php
|
|
/** @param array{b?: int, c?: string} $_a */
|
|
function foo(array $_a = []) : void {}',
|
|
],
|
|
'allowEmptyVarAnnotation' => [
|
|
'<?php
|
|
/**
|
|
* @param $_x
|
|
*/
|
|
function example(array $_x) : void {}',
|
|
],
|
|
'allowCapitalisedNamespacedString' => [
|
|
'<?php
|
|
namespace Foo;
|
|
|
|
/**
|
|
* @param String $_x
|
|
*/
|
|
function example(string $_x) : void {}',
|
|
],
|
|
'megaClosureAnnotationWithoutSpacing' => [
|
|
'<?php
|
|
/** @var array{a:Closure():(array<mixed, mixed>|null), b?:Closure():array<mixed, mixed>, c?:Closure():array<mixed, mixed>, d?:Closure():array<mixed, mixed>, e?:Closure():(array{f:null|string, g:null|string, h:null|string, i:string, j:mixed, k:mixed, l:mixed, m:mixed, n:bool, o?:array{0:string}}|null), p?:Closure():(array{f:null|string, g:null|string, h:null|string, q:string, i:string, j:mixed, k:mixed, l:mixed, m:mixed, n:bool, o?:array{0:string}}|null), r?:Closure():(array<mixed, mixed>|null), s:array<mixed, mixed>} */
|
|
$arr = [];
|
|
|
|
$arr["a"]();',
|
|
],
|
|
'megaClosureAnnotationWithSpacing' => [
|
|
'<?php
|
|
/**
|
|
* @var array{
|
|
* a: Closure() : (array<mixed, mixed>|null),
|
|
* b?: Closure() : array<mixed, mixed>,
|
|
* c?: Closure() : array<mixed, mixed>,
|
|
* d?: Closure() : array<mixed, mixed>,
|
|
* e?: Closure() : (array{
|
|
* f: null|string,
|
|
* g: null|string,
|
|
* h: null|string,
|
|
* i: string,
|
|
* j: mixed,
|
|
* k: mixed,
|
|
* l: mixed,
|
|
* m: mixed,
|
|
* n: bool,
|
|
* o?: array{0:string}
|
|
* }|null),
|
|
* p?: Closure() : (array{
|
|
* f: null|string,
|
|
* g: null|string,
|
|
* h: null|string,
|
|
* q: string,
|
|
* i: string,
|
|
* j: mixed,
|
|
* k: mixed,
|
|
* l: mixed,
|
|
* m: mixed,
|
|
* n: bool,
|
|
* o?: array{0:string}
|
|
* }|null),
|
|
* r?: Closure() : (array<mixed, mixed>|null),
|
|
* s: array<mixed, mixed>
|
|
* }
|
|
*
|
|
* Some text
|
|
*/
|
|
$arr = [];
|
|
|
|
$arr["a"]();',
|
|
],
|
|
'multipeLineGenericArray' => [
|
|
'<?php
|
|
/**
|
|
* @psalm-type MiddlewareArray = array<
|
|
* class-string<\Exception>,
|
|
* array<int, string>
|
|
* >
|
|
*
|
|
* @psalm-type RuleArray = array{
|
|
* rule: string,
|
|
* controller?: class-string<\Exception>,
|
|
* redirect?: string,
|
|
* code?: int,
|
|
* type?: string,
|
|
* middleware?: MiddlewareArray
|
|
* }
|
|
*
|
|
* Foo Bar
|
|
*/
|
|
class A {}',
|
|
],
|
|
'builtInClassInAShape' => [
|
|
'<?php
|
|
/**
|
|
* @return array{d:Exception}
|
|
* @psalm-suppress InvalidReturnType
|
|
*/
|
|
function f() {}'
|
|
],
|
|
'slashAfter?' => [
|
|
'<?php
|
|
namespace ns;
|
|
|
|
/** @param ?\stdClass $_s */
|
|
function foo($_s) : void {
|
|
}
|
|
|
|
foo(null);
|
|
foo(new \stdClass);',
|
|
],
|
|
'returnTypeShouldBeNullable' => [
|
|
'<?php
|
|
/**
|
|
* @return stdClass
|
|
*/
|
|
function foo() : ?stdClass {
|
|
return rand(0, 1) ? new stdClass : null;
|
|
}
|
|
|
|
$f = foo();
|
|
if ($f) {}',
|
|
],
|
|
'spreadOperatorAnnotation' => [
|
|
'<?php
|
|
/** @param string[] $_s */
|
|
function foo(string ...$_s) : void {}
|
|
/** @param string ...$_s */
|
|
function bar(string ...$_s) : void {}
|
|
foo("hello", "goodbye");
|
|
bar("hello", "goodbye");
|
|
foo(...["hello", "goodbye"]);
|
|
bar(...["hello", "goodbye"]);',
|
|
],
|
|
'spreadOperatorByRefAnnotation' => [
|
|
'<?php
|
|
/** @param string &...$s */
|
|
function foo(&...$s) : void {}
|
|
/** @param string ...&$s */
|
|
function bar(&...$s) : void {}
|
|
/** @param string[] &$s */
|
|
function bat(&...$s) : void {}
|
|
|
|
$a = "hello";
|
|
$b = "goodbye";
|
|
$c = "hello again";
|
|
foo($a);
|
|
bar($b);
|
|
bat($c);',
|
|
'assertions' => [
|
|
'$a' => 'string',
|
|
'$b' => 'string',
|
|
'$c' => 'string',
|
|
],
|
|
],
|
|
'valueReturnType' => [
|
|
'<?php
|
|
/**
|
|
* @param "a"|"b" $_p
|
|
*/
|
|
function acceptsLiteral($_p): void {}
|
|
|
|
/**
|
|
* @return "a"|"b"
|
|
*/
|
|
function returnsLiteral(): string {
|
|
return rand(0,1) ? "a" : "b";
|
|
}
|
|
|
|
acceptsLiteral(returnsLiteral());',
|
|
],
|
|
'typeAliasBeforeClass' => [
|
|
'<?php
|
|
/**
|
|
* @psalm-type CoolType = A|B|null
|
|
*/
|
|
|
|
class A {}
|
|
class B {}
|
|
|
|
/** @return CoolType */
|
|
function foo() {
|
|
if (rand(0, 1)) {
|
|
return new A();
|
|
}
|
|
|
|
if (rand(0, 1)) {
|
|
return new B();
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/** @param CoolType $_a **/
|
|
function bar ($_a) : void { }
|
|
|
|
bar(foo());',
|
|
],
|
|
'typeAliasBeforeFunction' => [
|
|
'<?php
|
|
/**
|
|
* @psalm-type A_OR_B = A|B
|
|
* @psalm-type CoolType = A_OR_B|null
|
|
* @return CoolType
|
|
*/
|
|
function foo() {
|
|
if (rand(0, 1)) {
|
|
return new A();
|
|
}
|
|
|
|
if (rand(0, 1)) {
|
|
return new B();
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
class A {}
|
|
class B {}
|
|
|
|
/** @param CoolType $_a **/
|
|
function bar ($_a) : void { }
|
|
|
|
bar(foo());',
|
|
],
|
|
'typeAliasInSeparateBlockBeforeFunction' => [
|
|
'<?php
|
|
/**
|
|
* @psalm-type CoolType = A|B|null
|
|
*/
|
|
/**
|
|
* @return CoolType
|
|
*/
|
|
function foo() {
|
|
if (rand(0, 1)) {
|
|
return new A();
|
|
}
|
|
|
|
if (rand(0, 1)) {
|
|
return new B();
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
class A {}
|
|
class B {}
|
|
|
|
/** @param CoolType $_a **/
|
|
function bar ($_a) : void { }
|
|
|
|
bar(foo());',
|
|
],
|
|
'almostFreeStandingTypeAlias' => [
|
|
'<?php
|
|
/**
|
|
* @psalm-type CoolType = A|B|null
|
|
*/
|
|
|
|
// this breaks up the line
|
|
|
|
class A {}
|
|
class B {}
|
|
|
|
/** @return CoolType */
|
|
function foo() {
|
|
if (rand(0, 1)) {
|
|
return new A();
|
|
}
|
|
|
|
if (rand(0, 1)) {
|
|
return new B();
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
/** @param CoolType $_a **/
|
|
function bar ($_a) : void { }
|
|
|
|
bar(foo());',
|
|
],
|
|
'typeAliasUsedTwice' => [
|
|
'<?php
|
|
/** @psalm-type TA = array<int, string> */
|
|
|
|
class Bar {
|
|
public function foo() : void {
|
|
$bar =
|
|
/** @return TA */
|
|
function() {
|
|
return ["hello"];
|
|
};
|
|
|
|
/** @var array<int, TA> */
|
|
$bat = [$bar(), $bar()];
|
|
|
|
foreach ($bat as $b) {
|
|
echo $b[0];
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @psalm-type _A=array{elt:int}
|
|
* @param _A $p
|
|
* @return _A
|
|
*/
|
|
function f($p) {
|
|
/**
|
|
* @psalm-suppress UnnecessaryVarAnnotation
|
|
* @var _A
|
|
*/
|
|
$r = $p;
|
|
return $r;
|
|
}',
|
|
],
|
|
'listUnpackWithDocblock' => [
|
|
'<?php
|
|
interface I {}
|
|
|
|
class A implements I {
|
|
public function bar() : void {}
|
|
}
|
|
|
|
/** @return I[] */
|
|
function foo() : array {
|
|
return [new A()];
|
|
}
|
|
|
|
/** @var A $a1 */
|
|
[$a1, $_a2] = foo();
|
|
|
|
$a1->bar();',
|
|
],
|
|
'spaceInType' => [
|
|
'<?php
|
|
/** @return string | null */
|
|
function foo(string $s = null) {
|
|
return $s;
|
|
}',
|
|
],
|
|
'missingReturnTypeWithBadDocblockIgnoreBoth' => [
|
|
'<?php
|
|
/**
|
|
* @return [bad]
|
|
*/
|
|
function fooBar() {
|
|
}',
|
|
[],
|
|
[
|
|
'InvalidDocblock' => Config::REPORT_INFO,
|
|
'MissingReturnType' => Config::REPORT_INFO,
|
|
],
|
|
],
|
|
'objectWithPropertiesAnnotation' => [
|
|
'<?php
|
|
/** @param object{foo:string} $o */
|
|
function foo(object $o) : string {
|
|
return $o->foo;
|
|
}
|
|
|
|
$s = new \stdClass();
|
|
$s->foo = "hello";
|
|
foo($s);
|
|
|
|
class A {
|
|
/** @var string */
|
|
public $foo = "hello";
|
|
}
|
|
|
|
foo(new A);',
|
|
],
|
|
'refineTypeInNestedCall' => [
|
|
'<?php
|
|
function foo(array $arr): \Generator {
|
|
/** @var array<string, mixed> $arr */
|
|
foreach (array_filter(array_keys($arr), function (string $key) : bool {
|
|
return strpos($key, "BAR") === 0;
|
|
}) as $envVar) {
|
|
yield $envVar => [getenv($envVar)];
|
|
}
|
|
}',
|
|
],
|
|
'allowAnnotationOnServer' => [
|
|
'<?php
|
|
function foo(): \Generator {
|
|
/** @var array<string, mixed> $_SERVER */
|
|
foreach (array_filter(array_keys($_SERVER), function (string $key) : bool {
|
|
return strpos($key, "BAR") === 0;
|
|
}) as $envVar) {
|
|
yield $envVar => [getenv($envVar)];
|
|
}
|
|
}',
|
|
],
|
|
'annotationOnForeachItems' => [
|
|
'<?php
|
|
function foo(array $arr) : void {
|
|
$item = null;
|
|
|
|
/** @var string $item */
|
|
foreach ($arr as $item) {}
|
|
|
|
if (is_null($item)) {}
|
|
}
|
|
|
|
function bar(array $arr) : void {
|
|
$item = null;
|
|
|
|
/** @var string $item */
|
|
foreach ($arr as $item => $_) {}
|
|
|
|
if (is_null($item)) {}
|
|
}
|
|
|
|
function bat(array $arr) : void {
|
|
$item = null;
|
|
|
|
/**
|
|
* @psalm-suppress MixedArrayAccess
|
|
* @var string $item
|
|
*/
|
|
foreach ($arr as list($item)) {}
|
|
|
|
if (is_null($item)) {}
|
|
}
|
|
|
|
function baz(array $arr) : void {
|
|
$item = null;
|
|
|
|
/**
|
|
* @psalm-suppress MixedArrayAccess
|
|
* @var string $item
|
|
*/
|
|
foreach ($arr as list($item => $_)) {}
|
|
|
|
if (is_null($item)) {}
|
|
}',
|
|
[],
|
|
[
|
|
'MixedAssignment',
|
|
],
|
|
],
|
|
'extraneousDocblockParamName' => [
|
|
'<?php
|
|
/**
|
|
* @param string $_foo
|
|
* @param string[] $bar
|
|
* @param string[] $_barb
|
|
*/
|
|
function f(string $_foo, array $_barb): void {}',
|
|
],
|
|
'nonEmptyArray' => [
|
|
'<?php
|
|
/** @param non-empty-array<string> $arr */
|
|
function foo(array $arr) : void {
|
|
foreach ($arr as $a) {}
|
|
echo $a;
|
|
}
|
|
|
|
foo(["a", "b", "c"]);
|
|
|
|
/** @param array<string> $arr */
|
|
function bar(array $arr) : void {
|
|
if (!$arr) {
|
|
return;
|
|
}
|
|
|
|
foo($arr);
|
|
}',
|
|
],
|
|
'nonEmptyArrayInNamespace' => [
|
|
'<?php
|
|
namespace ns;
|
|
|
|
/** @param non-empty-array<string> $arr */
|
|
function foo(array $arr) : void {
|
|
foreach ($arr as $a) {}
|
|
echo $a;
|
|
}
|
|
|
|
foo(["a", "b", "c"]);
|
|
|
|
/** @param array<string> $arr */
|
|
function bar(array $arr) : void {
|
|
if (!$arr) {
|
|
return;
|
|
}
|
|
|
|
foo($arr);
|
|
}',
|
|
],
|
|
'noExceptionOnIntersection' => [
|
|
'<?php
|
|
class Foo {
|
|
/** @var null|\DateTime&\DateTimeImmutable */
|
|
private $s = null;
|
|
}',
|
|
],
|
|
'intersectionWithSpace' => [
|
|
'<?php
|
|
interface A {
|
|
public function foo() : void;
|
|
}
|
|
interface B {
|
|
public function bar() : void;
|
|
}
|
|
|
|
/** @param A & B $a */
|
|
function f(A $a) : void {
|
|
$a->foo();
|
|
$a->bar();
|
|
}',
|
|
],
|
|
'allowClosingComma' => [
|
|
'<?php
|
|
/**
|
|
* @psalm-type _Alias=array{
|
|
* foo: string,
|
|
* bar: string,
|
|
* baz: array{
|
|
* a: int,
|
|
* },
|
|
* }
|
|
*/
|
|
class Foo { }
|
|
|
|
/**
|
|
* @param array{
|
|
* foo: string,
|
|
* bar: string,
|
|
* baz: array{
|
|
* a: int,
|
|
* },
|
|
* } $foo
|
|
*/
|
|
function foo(array $foo) : int {
|
|
return count($foo);
|
|
}
|
|
|
|
/**
|
|
* @var array{
|
|
* foo:string,
|
|
* bar:string,
|
|
* baz:string,
|
|
* } $_foo
|
|
*/
|
|
$_foo = ["foo" => "", "bar" => "", "baz" => ""];',
|
|
],
|
|
'returnNumber' => [
|
|
'<?php
|
|
class C {
|
|
/**
|
|
* @return 1
|
|
*/
|
|
public static function barBar() {
|
|
return 1;
|
|
}
|
|
}',
|
|
],
|
|
'returnNumberForInterface' => [
|
|
'<?php
|
|
interface I {
|
|
/**
|
|
* @return 1
|
|
*/
|
|
public static function barBar();
|
|
}',
|
|
],
|
|
'psalmTypeAnnotationAboveReturn' => [
|
|
'<?php
|
|
/**
|
|
* @psalm-type Person = array{name: string, age: int}
|
|
*/
|
|
|
|
/**
|
|
* @psalm-return Person
|
|
*/
|
|
function getPerson_error(): array {
|
|
$json = \'{"name": "John", "age": 44}\';
|
|
/** @psalm-var Person */
|
|
return json_decode($json, true);
|
|
}'
|
|
],
|
|
'allowDocblockDefinedTKeyedArrayIntoNonEmpty' => [
|
|
'<?php
|
|
/** @param non-empty-array $_bar */
|
|
function foo(array $_bar) : void { }
|
|
|
|
/** @var array{0:list<string>, 1:list<int>} */
|
|
$bar = [[], []];
|
|
|
|
foo($bar);'
|
|
],
|
|
'allowResourceInList' => [
|
|
'<?php
|
|
/** @param list<scalar|array|object|resource|null> $_s */
|
|
function foo(array $_s) : void { }'
|
|
],
|
|
'possiblyUndefinedObjectProperty' => [
|
|
'<?php
|
|
function consume(string $value): void {
|
|
echo $value;
|
|
}
|
|
|
|
/** @var object{value?: string} $data */
|
|
$data = json_decode("{}", false);
|
|
consume($data->value ?? "");'
|
|
],
|
|
'throwSelf' => [
|
|
'<?php
|
|
namespace Foo;
|
|
|
|
class MyException extends \Exception {
|
|
/**
|
|
* @throws self
|
|
*/
|
|
public static function create(): void {
|
|
throw new self();
|
|
}
|
|
}'
|
|
],
|
|
'parseTrailingCommaInReturn' => [
|
|
'<?php
|
|
/**
|
|
* @psalm-return array{
|
|
* a: int,
|
|
* b: string,
|
|
* }
|
|
*/
|
|
function foo(): array {
|
|
return ["a" => 1, "b" => "two"];
|
|
}'
|
|
],
|
|
'falsableFunctionAllowedWhenBooleanExpected' => [
|
|
'<?php
|
|
|
|
/** @psalm-return bool */
|
|
function alwaysFalse1()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
function alwaysFalse2(): bool
|
|
{
|
|
return false;
|
|
}'
|
|
],
|
|
'dontInheritDocblockReturnWhenRedeclared' => [
|
|
'<?php
|
|
interface Id {}
|
|
|
|
class UserId implements Id {}
|
|
|
|
interface Entity {
|
|
/** @psalm-return Id */
|
|
function id(): Id;
|
|
}
|
|
|
|
class User implements Entity {
|
|
public function id(): UserId {
|
|
return new UserId();
|
|
}
|
|
}',
|
|
[],
|
|
[],
|
|
'7.4'
|
|
],
|
|
'arrayWithKeySlashesAndNewline' => [
|
|
'<?php
|
|
$_arr = ["foo\\bar\nbaz" => "literal"];',
|
|
[
|
|
'$_arr' => 'array{\'foo\\\\bar\nbaz\': string}'
|
|
]
|
|
],
|
|
'doubleSpaceBeforeAt' => [
|
|
'<?php
|
|
/**
|
|
* @param string $_c
|
|
*/
|
|
function foo($_c) : void {}'
|
|
],
|
|
'throwsAnnotationWithBarAndSpace' => [
|
|
'<?php
|
|
/**
|
|
* @throws \Exception| \InvalidArgumentException
|
|
*/
|
|
function bar() : void {}'
|
|
],
|
|
'varDocblockAboveCall' => [
|
|
'<?php
|
|
|
|
function example(string $s): void {
|
|
if (preg_match(\'{foo-(\w+)}\', $s, $m)) {
|
|
/** @var array{string, string} $m */
|
|
takesString($m[1]);
|
|
}
|
|
}
|
|
|
|
function takesString(string $_s): void {}'
|
|
],
|
|
'noCrashWithoutAssignment' => [
|
|
'<?php
|
|
/** @var DateTime $obj */
|
|
echo $obj->format("Y");'
|
|
],
|
|
'intMaskWithClassConstants' => [
|
|
'<?php
|
|
class FileFlag {
|
|
public const OPEN = 1;
|
|
public const MODIFIED = 2;
|
|
public const NEW = 4;
|
|
}
|
|
|
|
/**
|
|
* @param int-mask<FileFlag::OPEN, FileFlag::MODIFIED, FileFlag::NEW> $flags
|
|
*/
|
|
function takesFlags(int $flags) : void {
|
|
echo $flags;
|
|
}
|
|
|
|
takesFlags(FileFlag::MODIFIED | FileFlag::NEW);'
|
|
],
|
|
'intMaskWithZero' => [
|
|
'<?php
|
|
/** @param int-mask<1,2> $_flags */
|
|
function takesFlags(int $_flags): void {}
|
|
|
|
takesFlags(0);
|
|
'
|
|
],
|
|
'intMaskOfWithClassWildcard' => [
|
|
'<?php
|
|
class FileFlag {
|
|
public const OPEN = 1;
|
|
public const MODIFIED = 2;
|
|
public const NEW = 4;
|
|
}
|
|
|
|
/**
|
|
* @param int-mask-of<FileFlag::*> $flags
|
|
*/
|
|
function takesFlags(int $flags) : void {
|
|
echo $flags;
|
|
}
|
|
|
|
takesFlags(FileFlag::MODIFIED | FileFlag::NEW);'
|
|
],
|
|
'intMaskOfWithZero' => [
|
|
'<?php
|
|
class FileFlag {
|
|
public const OPEN = 1;
|
|
public const MODIFIED = 2;
|
|
public const NEW = 4;
|
|
}
|
|
|
|
/** @param int-mask-of<FileFlag::*> $_flags */
|
|
function takesFlags(int $_flags): void {}
|
|
|
|
takesFlags(0);
|
|
'
|
|
],
|
|
'emptyStringFirst' => [
|
|
'<?php
|
|
/**
|
|
* @param \'\'|\'a\'|\'b\' $v
|
|
*/
|
|
function testBad(string $v): void {
|
|
echo $v;
|
|
}'
|
|
],
|
|
'UnnecessaryVarAnnotationSuppress' => [
|
|
'<?php
|
|
/** @psalm-consistent-constructor */
|
|
final class Foo{}
|
|
/**
|
|
* @param class-string $class
|
|
*/
|
|
function foo(string $class): Foo {
|
|
if (!is_subclass_of($class, Foo::class)) {
|
|
throw new \LogicException();
|
|
}
|
|
|
|
/**
|
|
* @psalm-suppress UnnecessaryVarAnnotation
|
|
* @var Foo $instance
|
|
*/
|
|
$instance = new $class();
|
|
|
|
return $instance;
|
|
}',
|
|
],
|
|
'suppressNonInvariantDocblockPropertyType' => [
|
|
'<?php
|
|
class Vendor
|
|
{
|
|
/**
|
|
* @var array
|
|
*/
|
|
public array $config = [];
|
|
public function getConfig(): array {return $this->config;}
|
|
}
|
|
|
|
class A extends Vendor
|
|
{
|
|
/**
|
|
* @var string[]
|
|
* @psalm-suppress NonInvariantDocblockPropertyType
|
|
*/
|
|
public array $config = [];
|
|
}
|
|
$a = new Vendor();
|
|
$_b = new A();
|
|
echo (string)($a->getConfig()[0]??"");'
|
|
],
|
|
'promotedPropertiesDocumentationEitherForParamOrForProperty' => [
|
|
'<?php
|
|
final class UserRole
|
|
{
|
|
/** @psalm-param stdClass $id */
|
|
public function __construct(
|
|
protected $id,
|
|
/** @psalm-var stdClass */
|
|
protected $id2
|
|
) {
|
|
}
|
|
}
|
|
|
|
new UserRole(new stdClass(), new stdClass());
|
|
'
|
|
],
|
|
'promotedPropertiesDocumentationForPropertyAndSignature' => [
|
|
'<?php
|
|
final class A
|
|
{
|
|
public function __construct(
|
|
/**
|
|
* @var iterable<string>
|
|
*/
|
|
private iterable $strings,
|
|
) {
|
|
}
|
|
}'
|
|
],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* @return iterable<string,array{string,error_message:string,1?:string[],2?:bool,3?:string}>
|
|
*/
|
|
public function providerInvalidCodeParse(): iterable
|
|
{
|
|
return [
|
|
'invalidClassMethodReturn' => [
|
|
'<?php
|
|
class C {
|
|
/**
|
|
* @return $thus
|
|
*/
|
|
public function barBar() {
|
|
return $this;
|
|
}
|
|
}',
|
|
'error_message' => 'MissingDocblockType',
|
|
],
|
|
|
|
'invalidClassMethodReturnBrackets' => [
|
|
'<?php
|
|
class C {
|
|
/**
|
|
* @return []
|
|
*/
|
|
public static function barBar() {
|
|
return [];
|
|
}
|
|
}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'invalidInterfaceMethodReturn' => [
|
|
'<?php
|
|
interface I {
|
|
/**
|
|
* @return $thus
|
|
*/
|
|
public static function barBar();
|
|
}',
|
|
'error_message' => 'MissingDocblockType',
|
|
],
|
|
'invalidInterfaceMethodReturnBrackets' => [
|
|
'<?php
|
|
interface I {
|
|
/**
|
|
* @return []
|
|
*/
|
|
public static function barBar();
|
|
}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'invalidPropertyBrackets' => [
|
|
'<?php
|
|
class A {
|
|
/**
|
|
* @var []
|
|
*/
|
|
public $bar;
|
|
}',
|
|
'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',
|
|
],
|
|
'missingParamType' => [
|
|
'<?php
|
|
/**
|
|
* @param string $bar
|
|
*/
|
|
function fooBar(): void {
|
|
}
|
|
|
|
fooBar("hello");',
|
|
'error_message' => 'TooManyArguments - src' . DIRECTORY_SEPARATOR . 'somefile.php:8:21 - Too many arguments for fooBar '
|
|
. '- expecting 0 but saw 1',
|
|
],
|
|
'missingParamVar' => [
|
|
'<?php
|
|
/**
|
|
* @param string
|
|
*/
|
|
function fooBar(): void {
|
|
}',
|
|
'error_message' => 'InvalidDocblock - src' . DIRECTORY_SEPARATOR . 'somefile.php:5:21 - Badly-formatted @param',
|
|
],
|
|
'invalidSlashWithString' => [
|
|
'<?php
|
|
/**
|
|
* @return \?string
|
|
*/
|
|
function foo() {
|
|
return rand(0, 1) ? "hello" : null;
|
|
}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'missingReturnTypeWithBadDocblock' => [
|
|
'<?php
|
|
/**
|
|
* @return [bad]
|
|
*/
|
|
function fooBar() {
|
|
}',
|
|
'error_message' => 'MissingReturnType',
|
|
[
|
|
'InvalidDocblock' => Config::REPORT_INFO,
|
|
],
|
|
],
|
|
'invalidDocblockReturn' => [
|
|
'<?php
|
|
/**
|
|
* @return string
|
|
*/
|
|
function fooFoo(): int {
|
|
return 5;
|
|
}',
|
|
'error_message' => 'MismatchingDocblockReturnType',
|
|
],
|
|
'intParamTypeDefinedInParent' => [
|
|
'<?php
|
|
class A {
|
|
public function foo(int $a): void {}
|
|
}
|
|
|
|
class B extends A {
|
|
public function foo($a): void {}
|
|
}',
|
|
'error_message' => 'MissingParamType',
|
|
'error_levels' => ['MethodSignatureMismatch'],
|
|
],
|
|
'psalmInvalidVar' => [
|
|
'<?php
|
|
class A
|
|
{
|
|
/** @psalm-var array<int, string> */
|
|
public $foo = [];
|
|
|
|
public function updateFoo(): void {
|
|
$this->foo["boof"] = "hello";
|
|
}
|
|
}',
|
|
'error_message' => 'InvalidPropertyAssignmentValue',
|
|
],
|
|
'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 */
|
|
function myvalue($x): void {
|
|
/** @var $myVar MyNS\OtherClass */
|
|
$myVar = $x->conn()->method();
|
|
$myVar->otherMethod();
|
|
}',
|
|
'error_message' => 'MissingDocblockType',
|
|
],
|
|
'dontOverrideSameType' => [
|
|
'<?php
|
|
class A {
|
|
/** @return ?int */
|
|
public function foo(): ?int {
|
|
if (rand(0, 1)) return 5;
|
|
}
|
|
}',
|
|
'error_message' => 'InvalidReturnType',
|
|
],
|
|
'alwaysCheckReturnType' => [
|
|
'<?php
|
|
class A {}
|
|
|
|
/**
|
|
* @return A
|
|
* @psalm-suppress MismatchingDocblockReturnType
|
|
*/
|
|
function foo(): B {
|
|
return new A;
|
|
}',
|
|
'error_message' => 'UndefinedClass',
|
|
],
|
|
'preventBadBoolean' => [
|
|
'<?php
|
|
function foo(): boolean {
|
|
return true;
|
|
}',
|
|
'error_message' => 'UndefinedClass',
|
|
],
|
|
'undefinedDocblockClassCall' => [
|
|
'<?php
|
|
class B {
|
|
/**
|
|
* @return A
|
|
* @psalm-suppress UndefinedDocblockClass
|
|
* @psalm-suppress InvalidReturnStatement
|
|
* @psalm-suppress InvalidReturnType
|
|
*/
|
|
public function foo() {
|
|
return new stdClass();
|
|
}
|
|
|
|
public function bar() {
|
|
$this->foo()->bar();
|
|
}
|
|
}
|
|
',
|
|
'error_message' => 'UndefinedDocblockClass',
|
|
],
|
|
'preventBadTKeyedArrayFormat' => [
|
|
'<?php
|
|
/**
|
|
* @param array{} $arr
|
|
*/
|
|
function bar(array $arr): void {}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'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',
|
|
],
|
|
'doubleBar' => [
|
|
'<?php
|
|
/** @param PDO||Closure|numeric $a */
|
|
function foo($a) : void {}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'badStringVar' => [
|
|
'<?php
|
|
/** @var string; */
|
|
$a = "hello";',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'badCallableVar' => [
|
|
'<?php
|
|
/** @return Closure(int): */
|
|
function foo() : callable {
|
|
return function () : void {};
|
|
}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'hyphenInType' => [
|
|
'<?php
|
|
/**
|
|
* @return - Description
|
|
*/
|
|
function example() {
|
|
return "placeholder";
|
|
}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'badAmpersand' => [
|
|
'<?php
|
|
/** @return &array */
|
|
function foo() : array {
|
|
return [];
|
|
}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'invalidTypeAlias' => [
|
|
'<?php
|
|
/**
|
|
* @psalm-type CoolType = A|B>
|
|
*/
|
|
|
|
class A {}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'typeAliasInTKeyedArray' => [
|
|
'<?php
|
|
/**
|
|
* @psalm-type aType null|"a"|"b"|"c"|"d"
|
|
*/
|
|
|
|
/** @psalm-return array{0:bool,1:aType} */
|
|
function f(): array {
|
|
return [(bool)rand(0,1), rand(0,1) ? "z" : null];
|
|
}',
|
|
'error_message' => 'InvalidReturnStatement',
|
|
],
|
|
'noCrashOnHalfDoneArrayPropertyType' => [
|
|
'<?php
|
|
class A {
|
|
/** @var array< */
|
|
private $foo = [];
|
|
}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'noCrashOnHalfDoneTKeyedArrayPropertyType' => [
|
|
'<?php
|
|
class A {
|
|
/** @var array{ */
|
|
private $foo = [];
|
|
}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'noCrashOnInvalidClassTemplateAsType' => [
|
|
'<?php
|
|
/**
|
|
* @template T as ' . '
|
|
*/
|
|
class A {}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'noCrashOnInvalidFunctionTemplateAsType' => [
|
|
'<?php
|
|
/**
|
|
* @template T as ' . '
|
|
*/
|
|
function foo() : void {}',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'returnTypeNewLineIsIgnored' => [
|
|
'<?php
|
|
/**
|
|
* @return
|
|
* Some text
|
|
*/
|
|
function foo() {}',
|
|
'error_message' => 'MissingReturnType',
|
|
],
|
|
'objectWithPropertiesAnnotationNoMatchingProperty' => [
|
|
'<?php
|
|
/** @param object{foo:string} $o */
|
|
function foo(object $o) : string {
|
|
return $o->foo;
|
|
}
|
|
|
|
class A {}
|
|
|
|
foo(new A);',
|
|
'error_message' => 'InvalidArgument',
|
|
],
|
|
'badVar' => [
|
|
'<?php
|
|
/** @var Foo */
|
|
$a = $_GET["foo"];',
|
|
'error_message' => 'UndefinedDocblockClass',
|
|
],
|
|
'badPsalmType' => [
|
|
'<?php
|
|
/**
|
|
* @psalm-type Foo = array{a:}
|
|
*/',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'mismatchingDocblockParamName' => [
|
|
'<?php
|
|
/** @param string[] $bar */
|
|
function f(array $barb): void {}',
|
|
'error_message' => 'InvalidDocblockParamName - src' . DIRECTORY_SEPARATOR . 'somefile.php:2:41',
|
|
],
|
|
'nonEmptyArrayCalledWithEmpty' => [
|
|
'<?php
|
|
/** @param non-empty-array<string> $arr */
|
|
function foo(array $arr) : void {
|
|
foreach ($arr as $a) {}
|
|
echo $a;
|
|
}
|
|
|
|
foo([]);',
|
|
'error_message' => 'InvalidArgument',
|
|
],
|
|
'nonEmptyArrayCalledWithEmptyInNamespace' => [
|
|
'<?php
|
|
namespace ns;
|
|
|
|
/** @param non-empty-array<string> $arr */
|
|
function foo(array $arr) : void {
|
|
foreach ($arr as $a) {}
|
|
echo $a;
|
|
}
|
|
|
|
foo([]);',
|
|
'error_message' => 'InvalidArgument',
|
|
],
|
|
'nonEmptyArrayCalledWithArray' => [
|
|
'<?php
|
|
/** @param non-empty-array<string> $arr */
|
|
function foo(array $arr) : void {
|
|
foreach ($arr as $a) {}
|
|
echo $a;
|
|
}
|
|
|
|
/** @param array<string> $arr */
|
|
function bar(array $arr) {
|
|
foo($arr);
|
|
}',
|
|
'error_message' => 'ArgumentTypeCoercion',
|
|
],
|
|
'spreadOperatorArrayAnnotationBadArg' => [
|
|
'<?php
|
|
/** @param string[] $_s */
|
|
function foo(string ...$_s) : void {}
|
|
foo(5);',
|
|
'error_message' => 'InvalidScalarArgument',
|
|
],
|
|
'spreadOperatorArrayAnnotationBadSpreadArg' => [
|
|
'<?php
|
|
/** @param string[] $_s */
|
|
function foo(string ...$_s) : void {}
|
|
foo(...[5]);',
|
|
'error_message' => 'InvalidScalarArgument',
|
|
],
|
|
'spreadOperatorByRefAnnotationBadCall1' => [
|
|
'<?php
|
|
/** @param string &...$s */
|
|
function foo(&...$s) : void {}
|
|
|
|
$a = 1;
|
|
foo($a);',
|
|
'error_message' => 'InvalidScalarArgument',
|
|
],
|
|
'spreadOperatorByRefAnnotationBadCall2' => [
|
|
'<?php
|
|
/** @param string ...&$s */
|
|
function foo(&...$s) : void {}
|
|
|
|
$b = 2;
|
|
foo($b);',
|
|
'error_message' => 'InvalidScalarArgument',
|
|
],
|
|
'spreadOperatorByRefAnnotationBadCall3' => [
|
|
'<?php
|
|
/** @param string[] &$s */
|
|
function foo(&...$s) : void {}
|
|
|
|
$c = 3;
|
|
foo($c);',
|
|
'error_message' => 'InvalidScalarArgument',
|
|
],
|
|
'identifyReturnType' => [
|
|
'<?php
|
|
/** @return array{hello: string} */
|
|
function foo() {}',
|
|
'error_message' => 'InvalidReturnType - src' . DIRECTORY_SEPARATOR . 'somefile.php:2:33',
|
|
],
|
|
'invalidParamDocblockAsterisk' => [
|
|
'<?php
|
|
/**
|
|
* @param * $reference
|
|
*/
|
|
function f($reference) {}',
|
|
'error_message' => 'MissingDocblockType',
|
|
],
|
|
'canNeverReturnDeclaredType' => [
|
|
'<?php
|
|
|
|
/** @psalm-return false */
|
|
function alwaysFalse() : bool
|
|
{
|
|
return true;
|
|
}',
|
|
'error_message' => 'InvalidReturnStatement - src' . DIRECTORY_SEPARATOR . 'somefile.php:6:32',
|
|
],
|
|
'falsableWithExpectedTypeTrue' => [
|
|
'<?php
|
|
|
|
/** @psalm-return true */
|
|
function alwaysFalse()
|
|
{
|
|
return false;
|
|
}',
|
|
'error_message' => 'FalsableReturnStatement - src' . DIRECTORY_SEPARATOR . 'somefile.php:6:32',
|
|
],
|
|
'DuplicatedParam' => [
|
|
'<?php
|
|
/**
|
|
* @psalm-param array $arr
|
|
* @psalm-param array $arr
|
|
*/
|
|
function bar(array $arr): void {}',
|
|
'error_message' => 'InvalidDocblock - src' . DIRECTORY_SEPARATOR . 'somefile.php:6:21 - Found duplicated @param or prefixed @param tag in docblock for bar',
|
|
],
|
|
'DuplicatedReturn' => [
|
|
'<?php
|
|
/**
|
|
* @return void
|
|
* @return void
|
|
*/
|
|
function bar(array $arr): void {}',
|
|
'error_message' => 'InvalidDocblock - src' . DIRECTORY_SEPARATOR . 'somefile.php:6:21 - Found duplicated @return or prefixed @return tag in docblock for bar',
|
|
],
|
|
'missingClassForTKeyedArray' => [
|
|
'<?php
|
|
interface I {
|
|
/** @return object{id: int, a: int} */
|
|
public function run();
|
|
}
|
|
|
|
class C implements I {
|
|
/** @return X */
|
|
public function run() {}
|
|
}',
|
|
'error_message' => 'ImplementedReturnTypeMismatch'
|
|
],
|
|
'unexpectedImportType' => [
|
|
'<?php
|
|
/** @psalm-import-type asd */
|
|
function f(): void {}
|
|
',
|
|
'error_message' => 'PossiblyInvalidDocblockTag',
|
|
],
|
|
'unexpectedVarOnFunction' => [
|
|
'<?php
|
|
/** @var int $p */
|
|
function f($p): void {}
|
|
',
|
|
'error_message' => 'PossiblyInvalidDocblockTag',
|
|
],
|
|
'unterminatedParentheses' => [
|
|
'<?php
|
|
/** @return ( */
|
|
function f() {}
|
|
',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'emptyParentheses' => [
|
|
'<?php
|
|
/** @return () */
|
|
function f() {}
|
|
',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'unbalancedParentheses' => [
|
|
"<?php
|
|
/** @return ((string) */
|
|
function f(): string {
|
|
return '';
|
|
}
|
|
",
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'promotedPropertiesDocumentationFailsWhenSendingBadTypeAgainstParam' => [
|
|
'<?php
|
|
final class UserRole
|
|
{
|
|
/** @psalm-param stdClass $id */
|
|
public function __construct(
|
|
protected $id
|
|
) {
|
|
}
|
|
|
|
}
|
|
new UserRole("a");
|
|
',
|
|
'error_message' => 'InvalidArgument',
|
|
],
|
|
'promotedPropertiesDocumentationFailsWhenSendingBadTypeAgainstProperty' => [
|
|
'<?php
|
|
final class UserRole
|
|
{
|
|
public function __construct(
|
|
/** @psalm-var stdClass */
|
|
protected $id2
|
|
) {
|
|
}
|
|
}
|
|
|
|
new UserRole("a");
|
|
',
|
|
'error_message' => 'InvalidArgument',
|
|
],
|
|
'promotedPropertyDuplicateDoc' => [
|
|
'<?php
|
|
final class UserRole
|
|
{
|
|
/** @psalm-param string $id */
|
|
public function __construct(
|
|
/** @psalm-var stdClass */
|
|
protected $id
|
|
) {
|
|
}
|
|
}
|
|
',
|
|
'error_message' => 'InvalidDocblock',
|
|
],
|
|
'promotedPropertyWithParamDocblockAndSignatureType' => [
|
|
'<?php
|
|
|
|
class A
|
|
{
|
|
public function __construct(
|
|
/** @var "cti"|"basic"|"teams"|"" */
|
|
public string $licenseType = "",
|
|
) {
|
|
}
|
|
}
|
|
|
|
$a = new A("ladida");
|
|
$a->licenseType = "dudidu";
|
|
|
|
echo $a->licenseType;',
|
|
'error_message' => 'InvalidArgument',
|
|
],
|
|
];
|
|
}
|
|
}
|