2021-06-04 19:37:03 +02:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2021-06-04 19:37:03 +02:00
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2021-12-04 21:55:53 +01:00
|
|
|
use Psalm\Tests\Traits\InvalidCodeAnalysisTestTrait;
|
|
|
|
use Psalm\Tests\Traits\ValidCodeAnalysisTestTrait;
|
|
|
|
|
2021-06-04 19:37:03 +02:00
|
|
|
use const DIRECTORY_SEPARATOR;
|
|
|
|
|
|
|
|
class ReadonlyPropertyTest extends TestCase
|
|
|
|
{
|
2021-12-04 21:55:53 +01:00
|
|
|
use InvalidCodeAnalysisTestTrait;
|
|
|
|
use ValidCodeAnalysisTestTrait;
|
2021-06-04 19:37:03 +02:00
|
|
|
|
|
|
|
public function providerValidCodeParse(): iterable
|
|
|
|
{
|
|
|
|
return [
|
2021-07-21 19:29:07 +02:00
|
|
|
'docblockReadonlyPropertySetInConstructor' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-06-04 19:37:03 +02:00
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
public string $bar;
|
|
|
|
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo (new A)->bar;'
|
|
|
|
],
|
2021-07-21 19:29:07 +02:00
|
|
|
'readonlyPropertySetInConstructor' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-07-21 19:29:07 +02:00
|
|
|
class A {
|
|
|
|
public readonly string $bar;
|
|
|
|
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo (new A)->bar;',
|
2022-01-13 19:49:37 +01:00
|
|
|
'assertions' => [],
|
|
|
|
'ignored_issues' => [],
|
|
|
|
'php_version' => '8.1'
|
2021-07-21 19:29:07 +02:00
|
|
|
],
|
|
|
|
'docblockReadonlyWithPrivateMutationsAllowedPropertySetInAnotherMethod' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-06-04 19:37:03 +02:00
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
* @psalm-allow-private-mutation
|
|
|
|
*/
|
|
|
|
public ?string $bar = null;
|
|
|
|
|
|
|
|
public function setBar(string $s) : void {
|
|
|
|
$this->bar = $s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo (new A)->bar;'
|
|
|
|
],
|
2022-07-29 15:50:56 +02:00
|
|
|
'readonlyPublicPropertySetInAnotherMethod' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-06-04 19:37:03 +02:00
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @psalm-readonly-allow-private-mutation
|
|
|
|
*/
|
|
|
|
public ?string $bar = null;
|
|
|
|
|
|
|
|
public function setBar(string $s) : void {
|
|
|
|
$this->bar = $s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo (new A)->bar;'
|
|
|
|
],
|
2022-07-29 15:50:56 +02:00
|
|
|
'docblockReadonlyWithPrivateMutationsAllowedConstructorPropertySetInAnotherMethod' => [
|
2022-10-16 13:59:15 +02:00
|
|
|
'code' => '<?php
|
2022-07-29 15:50:56 +02:00
|
|
|
class A {
|
|
|
|
public function __construct(
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
* @psalm-allow-private-mutation
|
|
|
|
*/
|
|
|
|
public ?string $bar = null,
|
|
|
|
) {}
|
|
|
|
|
|
|
|
public function setBar(string $s) : void {
|
|
|
|
$this->bar = $s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo (new A)->bar;'
|
|
|
|
],
|
|
|
|
'readonlyPublicConstructorPropertySetInAnotherMethod' => [
|
2022-10-16 13:59:15 +02:00
|
|
|
'code' => '<?php
|
2022-07-29 15:50:56 +02:00
|
|
|
class A {
|
|
|
|
public function __construct(
|
|
|
|
/**
|
|
|
|
* @psalm-readonly-allow-private-mutation
|
|
|
|
*/
|
|
|
|
public ?string $bar = null,
|
|
|
|
) {}
|
|
|
|
|
|
|
|
public function setBar(string $s) : void {
|
|
|
|
$this->bar = $s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo (new A)->bar;'
|
|
|
|
],
|
2021-06-04 19:37:03 +02:00
|
|
|
'readonlyPropertySetChildClass' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-06-04 19:37:03 +02:00
|
|
|
abstract class A {
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
public string $bar;
|
|
|
|
}
|
|
|
|
|
|
|
|
class B extends A {
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo (new B)->bar;'
|
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
public function providerInvalidCodeParse(): iterable
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
'readonlyPropertySetInConstructorAndAlsoAnotherMethodInsideClass' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-06-04 19:37:03 +02:00
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
public string $bar;
|
|
|
|
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setBar() : void {
|
|
|
|
$this->bar = "goodbye";
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'error_message' => 'InaccessibleProperty',
|
|
|
|
],
|
|
|
|
'readonlyPropertySetInConstructorAndAlsoAnotherMethodInSublass' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-06-04 19:37:03 +02:00
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
public string $bar;
|
|
|
|
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class B extends A {
|
|
|
|
public function setBar() : void {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'error_message' => 'InaccessibleProperty',
|
|
|
|
],
|
2021-07-21 19:29:07 +02:00
|
|
|
'docblockReadonlyPropertySetInConstructorAndAlsoOutsideClass' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-06-04 19:37:03 +02:00
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
public string $bar;
|
|
|
|
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
$a->bar = "goodbye";',
|
2021-07-21 19:29:07 +02:00
|
|
|
'error_message' => 'InaccessibleProperty - src' . DIRECTORY_SEPARATOR . 'somefile.php:14:21',
|
|
|
|
],
|
|
|
|
'readonlyPropertySetInConstructorAndAlsoOutsideClass' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-07-21 19:29:07 +02:00
|
|
|
class A {
|
|
|
|
public readonly string $bar;
|
|
|
|
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
$a->bar = "goodbye";',
|
|
|
|
'error_message' => 'InaccessibleProperty - src' . DIRECTORY_SEPARATOR . 'somefile.php:11:21',
|
2022-01-13 19:49:37 +01:00
|
|
|
'ignored_issues' => [],
|
|
|
|
'php_version' => '8.1',
|
2021-06-04 19:37:03 +02:00
|
|
|
],
|
|
|
|
'readonlyPropertySetInConstructorAndAlsoOutsideClassWithAllowPrivate' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-06-04 19:37:03 +02:00
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
* @psalm-allow-private-mutation
|
|
|
|
*/
|
|
|
|
public string $bar;
|
|
|
|
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setAgain() : void {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
$a->bar = "goodbye";',
|
|
|
|
'error_message' => 'InaccessibleProperty - src' . DIRECTORY_SEPARATOR . 'somefile.php:19:21',
|
|
|
|
],
|
|
|
|
'readonlyPublicPropertySetInConstructorAndAlsoOutsideClass' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-06-04 19:37:03 +02:00
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @psalm-readonly-allow-private-mutation
|
|
|
|
*/
|
|
|
|
public string $bar;
|
|
|
|
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setAgain() : void {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
$a->bar = "goodbye";',
|
|
|
|
'error_message' => 'InaccessibleProperty - src' . DIRECTORY_SEPARATOR . 'somefile.php:18:21',
|
|
|
|
],
|
2021-06-04 20:39:38 +02:00
|
|
|
'readonlyPropertyAssignOperator' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-06-04 20:39:38 +02:00
|
|
|
class Test {
|
|
|
|
/** @readonly */
|
|
|
|
public int $prop;
|
|
|
|
|
|
|
|
public function __construct(int $prop) {
|
|
|
|
// Legal initialization.
|
|
|
|
$this->prop = $prop;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$test = new Test(5);
|
|
|
|
|
|
|
|
$test->prop += 1;',
|
|
|
|
'error_message' => 'InaccessibleProperty'
|
|
|
|
],
|
2021-07-21 19:29:07 +02:00
|
|
|
'readonlyPropertyWithDefault' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-07-21 19:29:07 +02:00
|
|
|
class A {
|
|
|
|
public readonly string $s = "a";
|
|
|
|
}',
|
|
|
|
'error_message' => 'InvalidPropertyAssignment',
|
2022-01-13 19:49:37 +01:00
|
|
|
'ignored_issues' => [],
|
|
|
|
'php_version' => '8.1',
|
2021-07-21 19:29:07 +02:00
|
|
|
],
|
2021-11-13 19:37:08 +01:00
|
|
|
'readonlyPromotedPropertyAssignOperator' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-11-13 19:37:08 +01:00
|
|
|
class A {
|
|
|
|
public function __construct(public readonly string $bar) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A("hello");
|
|
|
|
$a->bar = "goodbye";',
|
|
|
|
'error_message' => 'InaccessibleProperty - src' . DIRECTORY_SEPARATOR . 'somefile.php:8:21',
|
2022-01-13 19:49:37 +01:00
|
|
|
'ignored_issues' => [],
|
|
|
|
'php_version' => '8.1',
|
2021-11-13 19:37:08 +01:00
|
|
|
],
|
2021-11-13 19:50:02 +01:00
|
|
|
'readonlyPromotedPropertyAccess' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2021-11-13 19:50:02 +01:00
|
|
|
class A {
|
|
|
|
public function __construct(private readonly string $bar) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A("hello");
|
|
|
|
$b = $a->bar;',
|
|
|
|
'error_message' => 'InaccessibleProperty - src' . DIRECTORY_SEPARATOR . 'somefile.php:8:26',
|
2022-01-13 19:49:37 +01:00
|
|
|
'ignored_issues' => [],
|
|
|
|
'php_version' => '8.1',
|
2021-11-13 19:50:02 +01:00
|
|
|
],
|
2022-01-16 09:29:16 +01:00
|
|
|
'readonlyPhpDocPromotedPropertyAssignOperator' => [
|
2022-01-18 23:07:35 +01:00
|
|
|
'code' => '<?php
|
2022-01-16 09:29:16 +01:00
|
|
|
|
|
|
|
final class A
|
|
|
|
{
|
|
|
|
public function __construct(
|
|
|
|
/**
|
|
|
|
* @psalm-readonly
|
|
|
|
*/
|
|
|
|
private string $string,
|
|
|
|
) {
|
|
|
|
}
|
|
|
|
|
|
|
|
private function mutateString(): void
|
|
|
|
{
|
|
|
|
$this->string = "";
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'error_message' => 'InaccessibleProperty',
|
2022-01-18 23:07:35 +01:00
|
|
|
'ignored_issues' => [],
|
|
|
|
'php_version' => '8.1',
|
2022-01-16 09:29:16 +01:00
|
|
|
],
|
2021-06-04 19:37:03 +02:00
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|