2022-01-09 21:17:42 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
|
|
|
use Psalm\Tests\Traits\InvalidCodeAnalysisTestTrait;
|
|
|
|
use Psalm\Tests\Traits\ValidCodeAnalysisTestTrait;
|
|
|
|
|
|
|
|
class PropertiesOfTest extends TestCase
|
|
|
|
{
|
|
|
|
use ValidCodeAnalysisTestTrait;
|
|
|
|
use InvalidCodeAnalysisTestTrait;
|
|
|
|
|
|
|
|
/**
|
2022-11-05 22:34:42 +01:00
|
|
|
*
|
2022-01-09 21:17:42 +01:00
|
|
|
*/
|
|
|
|
public function providerValidCodeParse(): iterable
|
|
|
|
{
|
|
|
|
return [
|
2022-10-03 11:28:01 +02:00
|
|
|
'propertiesOfIntersection' => [
|
|
|
|
'code' => '<?php
|
|
|
|
|
|
|
|
interface i {
|
|
|
|
}
|
|
|
|
class b {
|
|
|
|
public int $a = 123;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @psalm-suppress InvalidReturnType
|
|
|
|
* @return properties-of<$a>
|
|
|
|
*/
|
|
|
|
function test1($a) {}
|
|
|
|
/**
|
|
|
|
* @psalm-suppress InvalidReturnType
|
|
|
|
* @return properties-of<i&b>
|
|
|
|
*/
|
|
|
|
function test2() {}
|
|
|
|
/**
|
|
|
|
* @psalm-suppress InvalidReturnType
|
|
|
|
* @return properties-of<b&i>
|
|
|
|
*/
|
|
|
|
function test3() {}
|
|
|
|
|
|
|
|
/** @var i $i */
|
|
|
|
assert($i instanceof b);
|
|
|
|
$result1 = test1($i);
|
|
|
|
$result2 = test2();
|
|
|
|
$result3 = test3();
|
|
|
|
',
|
|
|
|
'assertions' => [
|
2022-11-10 09:18:27 -05:00
|
|
|
'$result1===' => 'array{a: int}<string, mixed>',
|
|
|
|
'$result2===' => 'array{a: int}<string, mixed>',
|
|
|
|
'$result3===' => 'array{a: int}<string, mixed>',
|
2022-10-03 11:28:01 +02:00
|
|
|
]
|
|
|
|
],
|
2022-01-09 21:17:42 +01:00
|
|
|
'publicPropertiesOf' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var bool */
|
|
|
|
public $foo = false;
|
|
|
|
/** @var string */
|
|
|
|
private $bar = "";
|
|
|
|
/** @var int */
|
|
|
|
protected $adams = 42;
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return public-properties-of<A> */
|
|
|
|
function returnPropertyOfA() {
|
|
|
|
return ["foo" => true];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
',
|
|
|
|
],
|
|
|
|
'protectedPropertiesOf' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var bool */
|
|
|
|
public $foo = false;
|
|
|
|
/** @var string */
|
|
|
|
private $bar = "";
|
|
|
|
/** @var int */
|
|
|
|
protected $adams = 42;
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return protected-properties-of<A> */
|
|
|
|
function returnPropertyOfA() {
|
|
|
|
return ["adams" => 42];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
',
|
|
|
|
],
|
|
|
|
'privatePropertiesOf' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var bool */
|
|
|
|
public $foo = false;
|
|
|
|
/** @var string */
|
|
|
|
private $bar = "";
|
|
|
|
/** @var int */
|
|
|
|
protected $adams = 42;
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return private-properties-of<A> */
|
|
|
|
function returnPropertyOfA() {
|
|
|
|
return ["bar" => "foo"];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
',
|
|
|
|
],
|
|
|
|
'allPropertiesOf' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var bool */
|
|
|
|
public $foo = false;
|
|
|
|
/** @var string */
|
|
|
|
private $bar = "";
|
|
|
|
/** @var int */
|
|
|
|
protected $adams = 42;
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return properties-of<A> */
|
|
|
|
function returnPropertyOfA(int $visibility) {
|
|
|
|
return [
|
|
|
|
"foo" => true,
|
|
|
|
"bar" => "foo",
|
|
|
|
"adams" => 1
|
|
|
|
];
|
2022-01-09 21:17:42 +01:00
|
|
|
}
|
|
|
|
',
|
|
|
|
],
|
2022-11-05 22:34:42 +01:00
|
|
|
'finalPropertiesOf' => [
|
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var bool */
|
|
|
|
public $foo = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return properties-of<A> */
|
|
|
|
function returnPropertyOfA() {
|
|
|
|
return ["foo" => true];
|
|
|
|
}
|
|
|
|
',
|
|
|
|
],
|
2022-02-22 19:49:03 +01:00
|
|
|
'allPropertiesOfContainsNoStatic' => [
|
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var bool */
|
|
|
|
public static $imStatic = true;
|
2022-02-21 23:02:27 +01:00
|
|
|
|
2022-02-22 19:49:03 +01:00
|
|
|
/** @var bool */
|
|
|
|
public $foo = false;
|
|
|
|
/** @var string */
|
|
|
|
private $bar = "";
|
|
|
|
/** @var int */
|
|
|
|
protected $adams = 42;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return properties-of<A> */
|
|
|
|
function returnPropertyOfA(int $visibility) {
|
|
|
|
return [
|
|
|
|
"foo" => true,
|
|
|
|
"bar" => "foo",
|
|
|
|
"adams" => 1
|
|
|
|
];
|
|
|
|
}
|
|
|
|
',
|
|
|
|
],
|
2022-01-09 21:17:42 +01:00
|
|
|
'usePropertiesOfSelfAsArrayKey' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var int */
|
|
|
|
public $a = 1;
|
|
|
|
/** @var int */
|
|
|
|
public $b = 2;
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return properties-of<self> */
|
|
|
|
public function asArray() {
|
|
|
|
return [
|
|
|
|
"a" => $this->a,
|
|
|
|
"b" => $this->b
|
|
|
|
];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
}
|
2022-02-21 23:02:27 +01:00
|
|
|
',
|
2022-01-09 21:17:42 +01:00
|
|
|
],
|
|
|
|
'usePropertiesOfStaticAsArrayKey' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var int */
|
|
|
|
public $a = 1;
|
|
|
|
/** @var int */
|
|
|
|
public $b = 2;
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return properties-of<static> */
|
|
|
|
public function asArray() {
|
|
|
|
return [
|
|
|
|
"a" => $this->a,
|
|
|
|
"b" => $this->b
|
|
|
|
];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
}
|
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
class B extends A {
|
|
|
|
/** @var int */
|
|
|
|
public $c = 3;
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
public function asArray() {
|
|
|
|
return [
|
|
|
|
"a" => $this->a,
|
|
|
|
"b" => $this->b,
|
|
|
|
"c" => $this->c,
|
|
|
|
];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
}
|
|
|
|
',
|
|
|
|
],
|
|
|
|
'propertiesOfMultipleInheritanceStaticAsArrayKey' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var int */
|
|
|
|
public $a = 1;
|
|
|
|
/** @var int */
|
|
|
|
public $b = 2;
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return properties-of<static> */
|
|
|
|
public function asArray() {
|
|
|
|
return [
|
|
|
|
"a" => $this->a,
|
|
|
|
"b" => $this->b
|
|
|
|
];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
}
|
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
class B extends A {
|
|
|
|
/** @var int */
|
|
|
|
public $c = 3;
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
class C extends B {
|
|
|
|
/** @var int */
|
|
|
|
public $d = 4;
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
public function asArray() {
|
|
|
|
return [
|
|
|
|
"a" => $this->a,
|
|
|
|
"b" => $this->b,
|
|
|
|
"c" => $this->c,
|
|
|
|
"d" => $this->d,
|
|
|
|
];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
}
|
|
|
|
',
|
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-11-05 22:34:42 +01:00
|
|
|
*
|
2022-01-09 21:17:42 +01:00
|
|
|
*/
|
|
|
|
public function providerInvalidCodeParse(): iterable
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
'onlyOneTemplateParam' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {}
|
|
|
|
class B {}
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @var properties-of<A, B> */
|
|
|
|
$test = "foobar";
|
2022-01-09 21:17:42 +01:00
|
|
|
',
|
|
|
|
'error_message' => 'InvalidDocblock',
|
|
|
|
],
|
2022-02-22 19:49:03 +01:00
|
|
|
'propertiesOfPicksNoStatic' => [
|
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var mixed */
|
|
|
|
public static $foo;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return properties-of<A> */
|
|
|
|
function returnPropertyOfA() {
|
|
|
|
return ["foo" => true];
|
|
|
|
}
|
|
|
|
',
|
|
|
|
'error_message' => 'InvalidReturnStatement'
|
|
|
|
],
|
2022-01-09 21:17:42 +01:00
|
|
|
'publicPropertiesOfPicksNoPrivate' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var mixed */
|
|
|
|
public $foo;
|
|
|
|
/** @var mixed */
|
|
|
|
private $bar;
|
|
|
|
/** @var mixed */
|
|
|
|
protected $adams;
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return public-properties-of<A> */
|
|
|
|
function returnPropertyOfA() {
|
|
|
|
return ["bar" => true];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
',
|
|
|
|
'error_message' => 'InvalidReturnStatement'
|
|
|
|
],
|
|
|
|
'publicPropertiesOfPicksNoProtected' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var mixed */
|
|
|
|
public $foo;
|
|
|
|
/** @var mixed */
|
|
|
|
private $bar;
|
|
|
|
/** @var mixed */
|
|
|
|
protected $adams;
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return public-properties-of<A> */
|
|
|
|
function returnPropertyOfA() {
|
|
|
|
return ["adams" => true];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
',
|
|
|
|
'error_message' => 'InvalidReturnStatement'
|
|
|
|
],
|
|
|
|
'protectedPropertiesOfPicksNoPublic' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var mixed */
|
|
|
|
public $foo;
|
|
|
|
/** @var mixed */
|
|
|
|
private $bar;
|
|
|
|
/** @var mixed */
|
|
|
|
protected $adams;
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return protected-properties-of<A> */
|
|
|
|
function returnPropertyOfA() {
|
|
|
|
return ["foo" => true];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
',
|
|
|
|
'error_message' => 'InvalidReturnStatement'
|
|
|
|
],
|
|
|
|
'protectedPropertiesOfPicksNoPrivate' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var mixed */
|
|
|
|
public $foo;
|
|
|
|
/** @var mixed */
|
|
|
|
private $bar;
|
|
|
|
/** @var mixed */
|
|
|
|
protected $adams;
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return protected-properties-of<A> */
|
|
|
|
function returnPropertyOfA() {
|
|
|
|
return ["bar" => true];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
',
|
|
|
|
'error_message' => 'InvalidReturnStatement'
|
|
|
|
],
|
|
|
|
'privatePropertiesOfPicksNoPublic' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var mixed */
|
|
|
|
public $foo;
|
|
|
|
/** @var mixed */
|
|
|
|
private $bar;
|
|
|
|
/** @var mixed */
|
|
|
|
protected $adams;
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return private-properties-of<A> */
|
|
|
|
function returnPropertyOfA() {
|
|
|
|
return ["foo" => true];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
',
|
|
|
|
'error_message' => 'InvalidReturnStatement'
|
|
|
|
],
|
|
|
|
'privatePropertiesOfPicksNoProtected' => [
|
2022-02-21 23:02:27 +01:00
|
|
|
'code' => '<?php
|
|
|
|
class A {
|
|
|
|
/** @var mixed */
|
|
|
|
public $foo;
|
|
|
|
/** @var mixed */
|
|
|
|
private $bar;
|
|
|
|
/** @var mixed */
|
|
|
|
protected $adams;
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
|
2022-02-21 23:02:27 +01:00
|
|
|
/** @return private-properties-of<A> */
|
|
|
|
function returnPropertyOfA() {
|
|
|
|
return ["adams" => true];
|
|
|
|
}
|
2022-01-09 21:17:42 +01:00
|
|
|
',
|
|
|
|
'error_message' => 'InvalidReturnStatement'
|
|
|
|
],
|
2022-11-05 22:34:42 +01:00
|
|
|
'finalPropertiesOfInexact' => [
|
|
|
|
'code' => '<?php
|
|
|
|
final class A {
|
|
|
|
/** @var bool */
|
|
|
|
public $foo = false;
|
|
|
|
/** @var string */
|
|
|
|
private $bar = "";
|
|
|
|
/** @var int */
|
|
|
|
protected $adams = 42;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @return properties-of<A> */
|
|
|
|
function returnPropertyOfA() {
|
|
|
|
return ["foo" => true];
|
|
|
|
}
|
|
|
|
',
|
|
|
|
'error_message' => 'InvalidReturnStatement'
|
|
|
|
],
|
2022-01-09 21:17:42 +01:00
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|