2020-10-24 00:10:22 -04:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2020-10-24 00:10:22 -04:00
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2021-12-04 21:55:53 +01:00
|
|
|
use Psalm\Tests\Traits\InvalidCodeAnalysisTestTrait;
|
|
|
|
use Psalm\Tests\Traits\ValidCodeAnalysisTestTrait;
|
|
|
|
|
2022-02-17 23:15:58 -06:00
|
|
|
use const DIRECTORY_SEPARATOR;
|
|
|
|
|
2020-10-24 00:10:22 -04:00
|
|
|
class AttributeTest extends TestCase
|
|
|
|
{
|
2021-12-04 21:55:53 +01:00
|
|
|
use InvalidCodeAnalysisTestTrait;
|
|
|
|
use ValidCodeAnalysisTestTrait;
|
2020-10-24 00:10:22 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @return iterable<string,array{string,assertions?:array<string,string>,error_levels?:string[]}>
|
|
|
|
*/
|
|
|
|
public function providerValidCodeParse(): iterable
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
'classAndPropertyAttributesExists' => [
|
|
|
|
'<?php
|
|
|
|
namespace Foo;
|
|
|
|
|
2020-10-30 13:28:14 -04:00
|
|
|
#[\Attribute(\Attribute::TARGET_CLASS)]
|
2020-10-24 00:10:22 -04:00
|
|
|
class Table {
|
|
|
|
public function __construct(public string $name) {}
|
|
|
|
}
|
|
|
|
|
2020-10-30 13:28:14 -04:00
|
|
|
#[\Attribute(\Attribute::TARGET_PROPERTY)]
|
2020-10-24 00:10:22 -04:00
|
|
|
class Column {
|
|
|
|
public function __construct(public string $name) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Table(name: "videos")]
|
|
|
|
class Video {
|
|
|
|
#[Column(name: "id")]
|
|
|
|
public string $id = "";
|
|
|
|
|
|
|
|
#[Column(name: "title")]
|
|
|
|
public string $name = "";
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Table(name: "users")]
|
|
|
|
class User {
|
|
|
|
public function __construct(
|
|
|
|
#[Column(name: "id")]
|
|
|
|
public string $id,
|
|
|
|
|
|
|
|
#[Column(name: "name")]
|
|
|
|
public string $name = "",
|
|
|
|
) {}
|
|
|
|
}',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'8.0'
|
|
|
|
],
|
|
|
|
'functionAttributeExists' => [
|
|
|
|
'<?php
|
|
|
|
namespace {
|
2020-10-30 13:28:14 -04:00
|
|
|
#[Attribute(Attribute::TARGET_CLASS | Attribute::TARGET_FUNCTION | Attribute::TARGET_PARAMETER)]
|
2020-10-24 00:10:22 -04:00
|
|
|
class Deprecated {}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace Foo\Bar {
|
|
|
|
#[\Deprecated]
|
|
|
|
function foo() : void {}
|
|
|
|
}',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'8.0'
|
|
|
|
],
|
|
|
|
'paramAttributeExists' => [
|
|
|
|
'<?php
|
|
|
|
namespace {
|
2020-10-30 13:28:14 -04:00
|
|
|
#[Attribute(Attribute::TARGET_CLASS | Attribute::TARGET_FUNCTION | Attribute::TARGET_PARAMETER)]
|
2020-10-24 00:10:22 -04:00
|
|
|
class Deprecated {}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace Foo\Bar {
|
|
|
|
function foo(#[\Deprecated] string $foo) : void {}
|
|
|
|
}',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'8.0'
|
|
|
|
],
|
2020-10-30 17:37:16 -04:00
|
|
|
'testReflectingClass' => [
|
|
|
|
'<?php
|
|
|
|
abstract class BaseAttribute {
|
|
|
|
public function __construct(public string $name) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Attribute(Attribute::TARGET_CLASS)]
|
|
|
|
class Table extends BaseAttribute {}
|
|
|
|
|
|
|
|
/** @param class-string $s */
|
|
|
|
function foo(string $s) : void {
|
|
|
|
foreach ((new ReflectionClass($s))->getAttributes(BaseAttribute::class, 2) as $attr) {
|
|
|
|
$attribute = $attr->newInstance();
|
|
|
|
echo $attribute->name;
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'8.0'
|
|
|
|
],
|
2021-12-17 00:12:54 +01:00
|
|
|
'testReflectingAllAttributes' => [
|
|
|
|
'<?php
|
2021-12-20 09:57:06 +01:00
|
|
|
/** @var class-string $a */
|
|
|
|
$cs = stdClass::class;
|
2021-12-17 00:12:54 +01:00
|
|
|
|
2021-12-20 09:57:06 +01:00
|
|
|
$a = new ReflectionClass($cs);
|
|
|
|
$b = $a->getAttributes();
|
2021-12-17 00:12:54 +01:00
|
|
|
',
|
2021-12-20 09:57:06 +01:00
|
|
|
'assertions' => [
|
|
|
|
'$b' => 'array<array-key, ReflectionAttribute<object>>',
|
|
|
|
],
|
2021-12-17 00:12:54 +01:00
|
|
|
[],
|
|
|
|
'8.0'
|
|
|
|
],
|
2020-12-07 14:39:58 -05:00
|
|
|
'convertKeyedArray' => [
|
|
|
|
'<?php
|
|
|
|
#[Attribute(Attribute::TARGET_CLASS)]
|
|
|
|
class Route {
|
|
|
|
private $methods = [];
|
|
|
|
/**
|
|
|
|
* @param string[] $methods
|
|
|
|
*/
|
|
|
|
public function __construct(array $methods = []) {
|
|
|
|
$this->methods = $methods;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#[Route(methods: ["GET"])]
|
|
|
|
class HealthController
|
|
|
|
{}',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'8.0'
|
|
|
|
],
|
2021-01-11 04:55:06 +02:00
|
|
|
'allowsRepeatableFlag' => [
|
|
|
|
'<?php
|
|
|
|
#[Attribute(Attribute::TARGET_ALL|Attribute::IS_REPEATABLE)] // results in int(127)
|
|
|
|
class A {}
|
|
|
|
',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'8.0'
|
|
|
|
],
|
2021-07-29 20:41:08 +02:00
|
|
|
'allowsClassString' => [
|
|
|
|
'<?php
|
|
|
|
|
|
|
|
#[Attribute(Attribute::TARGET_CLASS)]
|
|
|
|
class Foo
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @param class-string<Baz> $_className
|
|
|
|
*/
|
|
|
|
public function __construct(string $_className)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Foo(_className: Baz::class)]
|
|
|
|
class Baz {}',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'8.0'
|
|
|
|
],
|
2021-08-01 02:22:16 +03:00
|
|
|
'allowsClassStringFromDifferentNamespace' => [
|
|
|
|
'<?php
|
|
|
|
|
|
|
|
namespace NamespaceOne {
|
|
|
|
use Attribute;
|
|
|
|
|
|
|
|
#[Attribute(Attribute::TARGET_CLASS)]
|
|
|
|
class FooAttribute
|
|
|
|
{
|
|
|
|
/** @var class-string */
|
|
|
|
private string $className;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param class-string<FoobarInterface> $className
|
|
|
|
*/
|
|
|
|
public function __construct(string $className)
|
|
|
|
{
|
|
|
|
$this->className = $className;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
interface FoobarInterface {}
|
|
|
|
|
|
|
|
class Bar implements FoobarInterface {}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace NamespaceTwo {
|
|
|
|
use NamespaceOne\FooAttribute;
|
|
|
|
use NamespaceOne\Bar as ZZ;
|
|
|
|
|
|
|
|
#[FooAttribute(className: ZZ::class)]
|
|
|
|
class Baz {}
|
|
|
|
}
|
|
|
|
'
|
2021-11-09 19:52:29 +01:00
|
|
|
],
|
|
|
|
'returnTypeWillChange7.1' => [
|
|
|
|
'<?php
|
|
|
|
|
|
|
|
namespace Rabus\PsalmReturnTypeWillChange;
|
|
|
|
|
|
|
|
use EmptyIterator;
|
|
|
|
use IteratorAggregate;
|
|
|
|
use ReturnTypeWillChange;
|
|
|
|
|
|
|
|
final class EmptyCollection implements IteratorAggregate
|
|
|
|
{
|
|
|
|
#[ReturnTypeWillChange]
|
|
|
|
public function getIterator()
|
|
|
|
{
|
|
|
|
return new EmptyIterator();
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'7.1'
|
|
|
|
],
|
|
|
|
'returnTypeWillChange8.1' => [
|
|
|
|
'<?php
|
|
|
|
|
|
|
|
namespace Rabus\PsalmReturnTypeWillChange;
|
|
|
|
|
|
|
|
use EmptyIterator;
|
|
|
|
use IteratorAggregate;
|
|
|
|
use ReturnTypeWillChange;
|
|
|
|
|
|
|
|
final class EmptyCollection implements IteratorAggregate
|
|
|
|
{
|
|
|
|
#[ReturnTypeWillChange]
|
|
|
|
public function getIterator()
|
|
|
|
{
|
|
|
|
return new EmptyIterator();
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'8.1'
|
2022-02-17 23:15:58 -06:00
|
|
|
],
|
|
|
|
'createObjectAsAttributeArg' => [
|
|
|
|
'<?php
|
|
|
|
#[Attribute]
|
|
|
|
class B
|
|
|
|
{
|
|
|
|
public function __construct(?array $listOfB = null) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Attribute(Attribute::TARGET_CLASS)]
|
|
|
|
class A
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @param B[] $listOfB
|
|
|
|
*/
|
|
|
|
public function __construct(?array $listOfB = null) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[A([new B])]
|
|
|
|
class C {}
|
|
|
|
',
|
|
|
|
],
|
2022-02-21 10:44:59 -06:00
|
|
|
'selfInClassAttribute' => [
|
2022-02-21 10:38:50 -06:00
|
|
|
'<?php
|
|
|
|
#[Attribute]
|
|
|
|
class SomeAttr
|
|
|
|
{
|
|
|
|
/** @param class-string $class */
|
|
|
|
public function __construct(string $class) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[SomeAttr(self::class)]
|
2022-02-21 10:44:59 -06:00
|
|
|
class A
|
|
|
|
{
|
|
|
|
#[SomeAttr(self::class)]
|
|
|
|
public const CONST = "const";
|
|
|
|
|
|
|
|
#[SomeAttr(self::class)]
|
|
|
|
public string $foo = "bar";
|
|
|
|
|
|
|
|
#[SomeAttr(self::class)]
|
|
|
|
public function baz(): void {}
|
|
|
|
}
|
|
|
|
',
|
|
|
|
],
|
|
|
|
'parentInClassAttribute' => [
|
|
|
|
'<?php
|
|
|
|
#[Attribute]
|
|
|
|
class SomeAttr
|
|
|
|
{
|
|
|
|
/** @param class-string $class */
|
|
|
|
public function __construct(string $class) {}
|
|
|
|
}
|
|
|
|
|
2022-02-21 10:38:50 -06:00
|
|
|
class A {}
|
|
|
|
|
|
|
|
#[SomeAttr(parent::class)]
|
2022-02-21 10:44:59 -06:00
|
|
|
class B extends A
|
|
|
|
{
|
|
|
|
#[SomeAttr(parent::class)]
|
|
|
|
public const CONST = "const";
|
|
|
|
|
|
|
|
#[SomeAttr(parent::class)]
|
|
|
|
public string $foo = "bar";
|
|
|
|
|
|
|
|
#[SomeAttr(parent::class)]
|
|
|
|
public function baz(): void {}
|
|
|
|
}
|
|
|
|
',
|
|
|
|
],
|
|
|
|
'selfInInterfaceAttribute' => [
|
|
|
|
'<?php
|
|
|
|
#[Attribute]
|
|
|
|
class SomeAttr
|
|
|
|
{
|
|
|
|
/** @param class-string $class */
|
|
|
|
public function __construct(string $class) {}
|
|
|
|
}
|
2022-02-21 10:38:50 -06:00
|
|
|
|
|
|
|
#[SomeAttr(self::class)]
|
2022-02-21 10:44:59 -06:00
|
|
|
interface C
|
|
|
|
{
|
|
|
|
#[SomeAttr(self::class)]
|
|
|
|
public const CONST = "const";
|
|
|
|
|
|
|
|
#[SomeAttr(self::class)]
|
|
|
|
public function baz(): void {}
|
|
|
|
}
|
2022-02-21 10:38:50 -06:00
|
|
|
',
|
|
|
|
],
|
2020-10-24 00:10:22 -04:00
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-03-19 20:44:44 -05:00
|
|
|
* @return iterable<string,array{string,error_message:string,1?:string[],2?:bool,3?:string}>
|
2020-10-24 00:10:22 -04:00
|
|
|
*/
|
|
|
|
public function providerInvalidCodeParse(): iterable
|
|
|
|
{
|
|
|
|
return [
|
2020-11-22 00:52:56 -05:00
|
|
|
'attributeClassHasNoAttributeAnnotation' => [
|
|
|
|
'<?php
|
|
|
|
class A {}
|
|
|
|
|
|
|
|
#[A]
|
|
|
|
class B {}',
|
2022-02-17 23:15:58 -06:00
|
|
|
'error_message' => 'InvalidAttribute - src' . DIRECTORY_SEPARATOR . 'somefile.php:4:23',
|
2020-11-22 00:52:56 -05:00
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.0'
|
|
|
|
],
|
2020-10-24 00:10:22 -04:00
|
|
|
'missingAttributeOnClass' => [
|
|
|
|
'<?php
|
|
|
|
use Foo\Bar\Pure;
|
|
|
|
|
|
|
|
#[Pure]
|
|
|
|
class Video {}',
|
2022-02-17 23:15:58 -06:00
|
|
|
'error_message' => 'UndefinedAttributeClass - src' . DIRECTORY_SEPARATOR . 'somefile.php:4:23',
|
2020-10-24 00:10:22 -04:00
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.0'
|
|
|
|
],
|
|
|
|
'missingAttributeOnFunction' => [
|
|
|
|
'<?php
|
|
|
|
use Foo\Bar\Pure;
|
|
|
|
|
|
|
|
#[Pure]
|
|
|
|
function foo() : void {}',
|
2022-02-17 23:15:58 -06:00
|
|
|
'error_message' => 'UndefinedAttributeClass - src' . DIRECTORY_SEPARATOR . 'somefile.php:4:23',
|
2020-10-24 00:10:22 -04:00
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.0'
|
|
|
|
],
|
|
|
|
'missingAttributeOnParam' => [
|
|
|
|
'<?php
|
|
|
|
use Foo\Bar\Pure;
|
|
|
|
|
|
|
|
function foo(#[Pure] string $str) : void {}',
|
2022-02-17 23:15:58 -06:00
|
|
|
'error_message' => 'UndefinedAttributeClass - src' . DIRECTORY_SEPARATOR . 'somefile.php:4:36',
|
2020-10-24 00:10:22 -04:00
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.0'
|
|
|
|
],
|
|
|
|
'tooFewArgumentsToAttributeConstructor' => [
|
|
|
|
'<?php
|
|
|
|
namespace Foo;
|
|
|
|
|
2020-10-30 13:28:14 -04:00
|
|
|
#[\Attribute(\Attribute::TARGET_CLASS)]
|
2020-10-24 00:10:22 -04:00
|
|
|
class Table {
|
|
|
|
public function __construct(public string $name) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Table()]
|
|
|
|
class Video {}',
|
2022-02-17 23:15:58 -06:00
|
|
|
'error_message' => 'TooFewArguments - src' . DIRECTORY_SEPARATOR . 'somefile.php:9:23',
|
2020-10-24 00:10:22 -04:00
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.0'
|
|
|
|
],
|
2020-11-22 00:44:44 -05:00
|
|
|
'invalidArgument' => [
|
|
|
|
'<?php
|
|
|
|
#[Attribute]
|
|
|
|
class Foo
|
|
|
|
{
|
|
|
|
public function __construct(int $i)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Foo("foo")]
|
|
|
|
class Bar{}',
|
2022-02-17 23:15:58 -06:00
|
|
|
'error_message' => 'InvalidScalarArgument - src' . DIRECTORY_SEPARATOR . 'somefile.php:10:27',
|
2020-11-22 00:44:44 -05:00
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.0'
|
|
|
|
],
|
2020-10-30 13:28:14 -04:00
|
|
|
'classAttributeUsedOnFunction' => [
|
|
|
|
'<?php
|
|
|
|
namespace Foo;
|
|
|
|
|
|
|
|
#[\Attribute(\Attribute::TARGET_CLASS)]
|
|
|
|
class Table {
|
|
|
|
public function __construct(public string $name) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Table("videos")]
|
|
|
|
function foo() : void {}',
|
2022-02-17 23:15:58 -06:00
|
|
|
'error_message' => 'InvalidAttribute - src' . DIRECTORY_SEPARATOR . 'somefile.php:9:23',
|
2020-10-30 13:28:14 -04:00
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.0'
|
|
|
|
],
|
2020-11-22 01:15:52 -05:00
|
|
|
'interfaceCannotBeAttributeClass' => [
|
|
|
|
'<?php
|
|
|
|
#[Attribute]
|
|
|
|
interface Foo {}',
|
2022-02-17 23:15:58 -06:00
|
|
|
'error_message' => 'InvalidAttribute - src' . DIRECTORY_SEPARATOR . 'somefile.php:2:23',
|
2020-11-22 01:15:52 -05:00
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.0'
|
|
|
|
],
|
|
|
|
'traitCannotBeAttributeClass' => [
|
|
|
|
'<?php
|
|
|
|
#[Attribute]
|
|
|
|
interface Foo {}',
|
2022-02-17 23:15:58 -06:00
|
|
|
'error_message' => 'InvalidAttribute - src' . DIRECTORY_SEPARATOR . 'somefile.php:2:23',
|
2020-11-22 01:15:52 -05:00
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.0'
|
|
|
|
],
|
|
|
|
'abstractClassCannotBeAttributeClass' => [
|
|
|
|
'<?php
|
|
|
|
#[Attribute]
|
|
|
|
abstract class Baz {}',
|
2022-02-17 23:15:58 -06:00
|
|
|
'error_message' => 'InvalidAttribute - src' . DIRECTORY_SEPARATOR . 'somefile.php:2:23',
|
2020-11-22 01:15:52 -05:00
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.0'
|
|
|
|
],
|
|
|
|
'abstractClassCannotHavePrivateConstructor' => [
|
|
|
|
'<?php
|
|
|
|
#[Attribute]
|
|
|
|
class Baz {
|
|
|
|
private function __construct() {}
|
|
|
|
}',
|
2022-02-17 23:15:58 -06:00
|
|
|
'error_message' => 'InvalidAttribute - src' . DIRECTORY_SEPARATOR . 'somefile.php:2:23',
|
2020-11-22 01:15:52 -05:00
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.0'
|
|
|
|
],
|
2022-02-21 10:38:50 -06:00
|
|
|
'noParentInAttributeOnClassWithoutParent' => [
|
|
|
|
'<?php
|
|
|
|
#[Attribute]
|
|
|
|
class SomeAttr
|
|
|
|
{
|
|
|
|
/** @param class-string $class */
|
|
|
|
public function __construct(string $class) {}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[SomeAttr(parent::class)]
|
|
|
|
class A {}
|
|
|
|
',
|
|
|
|
'error_message' => 'ParentNotFound',
|
|
|
|
],
|
2020-10-24 00:10:22 -04:00
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|