2021-06-04 13:37:03 -04:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2021-06-04 13:37:03 -04: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 13:37:03 -04:00
|
|
|
use const DIRECTORY_SEPARATOR;
|
|
|
|
|
|
|
|
class ReadonlyPropertyTest extends TestCase
|
|
|
|
{
|
2021-12-04 21:55:53 +01:00
|
|
|
use InvalidCodeAnalysisTestTrait;
|
|
|
|
use ValidCodeAnalysisTestTrait;
|
2021-06-04 13:37:03 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @return iterable<string,array{string,assertions?:array<string,string>,error_levels?:string[]}>
|
|
|
|
*/
|
|
|
|
public function providerValidCodeParse(): iterable
|
|
|
|
{
|
|
|
|
return [
|
2021-07-21 13:29:07 -04:00
|
|
|
'docblockReadonlyPropertySetInConstructor' => [
|
2021-06-04 13:37:03 -04:00
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
public string $bar;
|
|
|
|
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo (new A)->bar;'
|
|
|
|
],
|
2021-07-21 13:29:07 -04:00
|
|
|
'readonlyPropertySetInConstructor' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
public readonly string $bar;
|
|
|
|
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo (new A)->bar;',
|
|
|
|
[],
|
|
|
|
[],
|
|
|
|
'8.1'
|
|
|
|
],
|
|
|
|
'docblockReadonlyWithPrivateMutationsAllowedPropertySetInAnotherMethod' => [
|
2021-06-04 13:37:03 -04:00
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
* @psalm-allow-private-mutation
|
|
|
|
*/
|
|
|
|
public ?string $bar = null;
|
|
|
|
|
|
|
|
public function setBar(string $s) : void {
|
|
|
|
$this->bar = $s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo (new A)->bar;'
|
|
|
|
],
|
|
|
|
'readonlyPublicPropertySetInAnotherMEthod' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @psalm-readonly-allow-private-mutation
|
|
|
|
*/
|
|
|
|
public ?string $bar = null;
|
|
|
|
|
|
|
|
public function setBar(string $s) : void {
|
|
|
|
$this->bar = $s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo (new A)->bar;'
|
|
|
|
],
|
|
|
|
'readonlyPropertySetChildClass' => [
|
|
|
|
'<?php
|
|
|
|
abstract class A {
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
public string $bar;
|
|
|
|
}
|
|
|
|
|
|
|
|
class B extends A {
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo (new B)->bar;'
|
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return iterable<string,array{string,error_message:string,1?:string[],2?:bool,3?:string}>
|
|
|
|
*/
|
|
|
|
public function providerInvalidCodeParse(): iterable
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
'readonlyPropertySetInConstructorAndAlsoAnotherMethodInsideClass' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
public string $bar;
|
|
|
|
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setBar() : void {
|
|
|
|
$this->bar = "goodbye";
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'error_message' => 'InaccessibleProperty',
|
|
|
|
],
|
|
|
|
'readonlyPropertySetInConstructorAndAlsoAnotherMethodInSublass' => [
|
|
|
|
'<?php
|
|
|
|
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 13:29:07 -04:00
|
|
|
'docblockReadonlyPropertySetInConstructorAndAlsoOutsideClass' => [
|
2021-06-04 13:37:03 -04:00
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/**
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
public string $bar;
|
|
|
|
|
|
|
|
public function __construct() {
|
|
|
|
$this->bar = "hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = new A();
|
|
|
|
$a->bar = "goodbye";',
|
2021-07-21 13:29:07 -04:00
|
|
|
'error_message' => 'InaccessibleProperty - src' . DIRECTORY_SEPARATOR . 'somefile.php:14:21',
|
|
|
|
],
|
|
|
|
'readonlyPropertySetInConstructorAndAlsoOutsideClass' => [
|
|
|
|
'<?php
|
|
|
|
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',
|
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.1',
|
2021-06-04 13:37:03 -04:00
|
|
|
],
|
|
|
|
'readonlyPropertySetInConstructorAndAlsoOutsideClassWithAllowPrivate' => [
|
|
|
|
'<?php
|
|
|
|
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' => [
|
|
|
|
'<?php
|
|
|
|
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 14:39:38 -04:00
|
|
|
'readonlyPropertyAssignOperator' => [
|
|
|
|
'<?php
|
|
|
|
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 13:29:07 -04:00
|
|
|
'readonlyPropertyWithDefault' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
public readonly string $s = "a";
|
|
|
|
}',
|
|
|
|
'error_message' => 'InvalidPropertyAssignment',
|
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.1',
|
|
|
|
],
|
2021-11-13 12:37:08 -06:00
|
|
|
'readonlyPromotedPropertyAssignOperator' => [
|
|
|
|
'<?php
|
|
|
|
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',
|
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.1',
|
|
|
|
],
|
2021-11-13 12:50:02 -06:00
|
|
|
'readonlyPromotedPropertyAccess' => [
|
|
|
|
'<?php
|
|
|
|
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',
|
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.1',
|
|
|
|
],
|
2022-01-16 09:29:16 +01:00
|
|
|
'readonlyPhpDocPromotedPropertyAssignOperator' => [
|
|
|
|
'<?php
|
|
|
|
|
|
|
|
final class A
|
|
|
|
{
|
|
|
|
public function __construct(
|
|
|
|
/**
|
|
|
|
* @psalm-readonly
|
|
|
|
*/
|
|
|
|
private string $string,
|
|
|
|
) {
|
|
|
|
}
|
|
|
|
|
|
|
|
private function mutateString(): void
|
|
|
|
{
|
|
|
|
$this->string = "";
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
'error_message' => 'InaccessibleProperty',
|
|
|
|
[],
|
|
|
|
false,
|
|
|
|
'8.1',
|
|
|
|
],
|
2021-06-04 13:37:03 -04:00
|
|
|
];
|
|
|
|
}
|
|
|
|
}
|