2016-12-17 04:16:29 +01:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2016-12-17 04:16:29 +01:00
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2021-12-04 21:55:53 +01:00
|
|
|
use Psalm\Tests\Traits\InvalidCodeAnalysisTestTrait;
|
|
|
|
use Psalm\Tests\Traits\ValidCodeAnalysisTestTrait;
|
|
|
|
|
2019-06-26 22:52:29 +02:00
|
|
|
use const DIRECTORY_SEPARATOR;
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class TraitTest extends TestCase
|
2016-12-17 04:16:29 +01:00
|
|
|
{
|
2021-12-04 21:55:53 +01:00
|
|
|
use InvalidCodeAnalysisTestTrait;
|
|
|
|
use ValidCodeAnalysisTestTrait;
|
2016-12-17 04:16:29 +01:00
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function providerValidCodeParse(): iterable
|
2016-12-17 04:16:29 +01:00
|
|
|
{
|
2017-04-25 05:45:02 +02:00
|
|
|
return [
|
|
|
|
'accessiblePrivateMethodFromTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
private function fooFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
public function doFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
$this->fooFoo();
|
|
|
|
}
|
2017-05-27 02:05:57 +02:00
|
|
|
}',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'accessibleProtectedMethodFromTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
protected function fooFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
public function doFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
$this->fooFoo();
|
|
|
|
}
|
2017-05-27 02:05:57 +02:00
|
|
|
}',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'accessiblePublicMethodFromTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function fooFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
public function doFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
$this->fooFoo();
|
|
|
|
}
|
2017-05-27 02:05:57 +02:00
|
|
|
}',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'accessiblePrivatePropertyFromTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
|
|
|
/** @var string */
|
|
|
|
private $fooFoo = "";
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
public function doFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
echo $this->fooFoo;
|
2017-07-25 22:11:02 +02:00
|
|
|
$this->fooFoo = "hello";
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
2017-05-27 02:05:57 +02:00
|
|
|
}',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'accessibleProtectedPropertyFromTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
|
|
|
/** @var string */
|
|
|
|
protected $fooFoo = "";
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
public function doFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
echo $this->fooFoo;
|
2017-07-25 22:11:02 +02:00
|
|
|
$this->fooFoo = "hello";
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
2017-05-27 02:05:57 +02:00
|
|
|
}',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'accessiblePublicPropertyFromTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
|
|
|
/** @var string */
|
|
|
|
public $fooFoo = "";
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
public function doFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
echo $this->fooFoo;
|
2017-07-25 22:11:02 +02:00
|
|
|
$this->fooFoo = "hello";
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
2017-05-27 02:05:57 +02:00
|
|
|
}',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'accessibleProtectedMethodFromInheritedTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
protected function fooFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use T;
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class C extends B {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function doFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
$this->fooFoo();
|
|
|
|
}
|
2017-05-27 02:05:57 +02:00
|
|
|
}',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'accessiblePublicMethodFromInheritedTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function fooFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use T;
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class C extends B {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function doFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
$this->fooFoo();
|
|
|
|
}
|
2017-05-27 02:05:57 +02:00
|
|
|
}',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'staticClassMethodFromWithinTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function fooFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
self::barBar();
|
|
|
|
}
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
public static function barBar(): void {
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
2017-05-27 02:05:57 +02:00
|
|
|
}',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'redefinedTraitMethodWithoutAlias' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function fooFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
public function fooFoo(string $a): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-05-27 02:05:57 +02:00
|
|
|
(new B)->fooFoo("hello");',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'redefinedTraitMethodWithAlias' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function fooFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use T {
|
|
|
|
fooFoo as barBar;
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
public function fooFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
$this->barBar();
|
|
|
|
}
|
2017-05-27 02:05:57 +02:00
|
|
|
}',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'traitSelf' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
|
|
|
public function g(): self
|
|
|
|
{
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class A {
|
|
|
|
use T;
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
$a = (new A)->g();',
|
|
|
|
'assertions' => [
|
2017-06-29 16:22:49 +02:00
|
|
|
'$a' => 'A',
|
2017-05-27 02:05:57 +02:00
|
|
|
],
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'parentTraitSelf' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
|
|
|
public function g(): self
|
|
|
|
{
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class A {
|
|
|
|
use T;
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B extends A {
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class C {
|
|
|
|
use T;
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
$a = (new B)->g();',
|
|
|
|
'assertions' => [
|
2017-06-29 16:22:49 +02:00
|
|
|
'$a' => 'A',
|
2017-05-27 02:05:57 +02:00
|
|
|
],
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'directStaticCall' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
|
|
|
/** @return void */
|
|
|
|
public static function foo() {}
|
|
|
|
}
|
|
|
|
class A {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
/** @return void */
|
|
|
|
public function bar() {
|
|
|
|
T::foo();
|
|
|
|
}
|
2017-05-27 02:05:57 +02:00
|
|
|
}',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'abstractTraitMethod' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
|
|
|
/** @return void */
|
|
|
|
abstract public function foo();
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
abstract class A {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
/** @return void */
|
|
|
|
public function bar() {
|
|
|
|
$this->foo();
|
|
|
|
}
|
2017-05-27 02:05:57 +02:00
|
|
|
}',
|
|
|
|
],
|
2017-06-23 06:39:37 +02:00
|
|
|
'instanceOfTraitUser' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-06-23 06:39:37 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function f(): void {
|
2017-06-23 06:39:37 +02:00
|
|
|
if ($this instanceof A) { }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T;
|
2017-10-24 06:01:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class B {
|
|
|
|
use T;
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'getClassTraitUser' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-10-24 06:01:43 +02:00
|
|
|
trait T {
|
2018-07-18 04:50:30 +02:00
|
|
|
public function f(): void {
|
2018-11-29 05:59:43 +01:00
|
|
|
if (get_class($this) === B::class) {
|
2018-07-18 04:50:30 +02:00
|
|
|
$this->foo();
|
|
|
|
}
|
|
|
|
}
|
2017-10-24 06:01:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
2018-07-18 04:50:30 +02:00
|
|
|
use T;
|
2017-10-24 06:01:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class B {
|
2018-07-18 04:50:30 +02:00
|
|
|
use T;
|
|
|
|
|
|
|
|
public function foo() : void {}
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'staticClassTraitUser' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-07-18 04:50:30 +02:00
|
|
|
trait T {
|
|
|
|
public function f(): void {
|
2018-11-29 05:59:43 +01:00
|
|
|
if (static::class === B::class) {
|
2018-07-18 04:50:30 +02:00
|
|
|
$this->foo();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T;
|
|
|
|
}
|
|
|
|
|
|
|
|
class B {
|
|
|
|
use T;
|
|
|
|
|
|
|
|
public function foo() : void {}
|
|
|
|
}',
|
|
|
|
],
|
2019-02-26 07:03:33 +01:00
|
|
|
'isAClassTraitUserStringClass' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-02-26 07:03:33 +01:00
|
|
|
trait T {
|
|
|
|
public function f(): void {
|
2021-10-12 20:13:04 +02:00
|
|
|
if (is_a(static::class, B::class, true)) { }
|
2019-02-26 07:03:33 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T;
|
|
|
|
}
|
|
|
|
|
|
|
|
class B {
|
|
|
|
use T;
|
|
|
|
|
|
|
|
public function foo() : void {}
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'isAClassTraitUserClassConstant' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-07-18 04:50:30 +02:00
|
|
|
trait T {
|
|
|
|
public function f(): void {
|
2021-10-12 20:13:04 +02:00
|
|
|
if (is_a(static::class, B::class, true)) { }
|
2018-07-18 04:50:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T;
|
|
|
|
}
|
|
|
|
|
|
|
|
class B {
|
|
|
|
use T;
|
|
|
|
|
|
|
|
public function foo() : void {}
|
2017-06-23 06:39:37 +02:00
|
|
|
}',
|
|
|
|
],
|
2017-07-09 21:54:43 +02:00
|
|
|
'useTraitInClassWithAbstractMethod' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-07-09 21:54:43 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
abstract public function foo(): void;
|
2017-07-09 21:54:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function foo(): void {}
|
2017-07-25 22:11:02 +02:00
|
|
|
}',
|
2017-07-09 21:54:43 +02:00
|
|
|
],
|
|
|
|
'useTraitInSubclassWithAbstractMethod' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-07-09 21:54:43 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
abstract public function foo(): void;
|
2017-07-09 21:54:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
abstract class A {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function foo(): void {}
|
2017-07-09 21:54:43 +02:00
|
|
|
}
|
|
|
|
|
2017-07-29 05:38:57 +02:00
|
|
|
class B extends A {
|
|
|
|
use T;
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'useTraitInSubclassWithAbstractMethodInParent' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-07-29 05:38:57 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function foo(): void {}
|
2017-07-29 05:38:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
abstract class A {
|
2018-01-11 21:50:45 +01:00
|
|
|
abstract public function foo(): void {}
|
2017-07-29 05:38:57 +02:00
|
|
|
}
|
|
|
|
|
2017-07-09 21:54:43 +02:00
|
|
|
class B extends A {
|
|
|
|
use T;
|
2017-07-25 22:11:02 +02:00
|
|
|
}',
|
2017-07-09 21:54:43 +02:00
|
|
|
],
|
2017-12-17 07:06:12 +01:00
|
|
|
'differentMethodReturnTypes' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-12-17 07:06:12 +01:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
public static function getSelf(): self {
|
2017-12-17 07:06:12 +01:00
|
|
|
return new self();
|
|
|
|
}
|
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
public static function callGetSelf(): self {
|
2017-12-17 07:06:12 +01:00
|
|
|
return self::getSelf();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T;
|
|
|
|
}
|
|
|
|
|
|
|
|
class B {
|
|
|
|
use T;
|
|
|
|
}',
|
|
|
|
],
|
2018-01-09 15:22:23 +01:00
|
|
|
'parentRefInTraitShouldNotFail' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-01-09 15:22:23 +01:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function foo(): void {
|
2018-01-09 15:22:23 +01:00
|
|
|
parent::foo();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
class A {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function foo(): void {}
|
2018-01-09 15:22:23 +01:00
|
|
|
}
|
|
|
|
class B extends A {
|
|
|
|
use T;
|
|
|
|
}',
|
|
|
|
],
|
2018-01-21 06:32:45 +01:00
|
|
|
'namespacedTraitLookup' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-01-21 06:32:45 +01:00
|
|
|
namespace Classes {
|
|
|
|
use Traits\T;
|
|
|
|
|
|
|
|
class A {}
|
|
|
|
|
|
|
|
class B {
|
|
|
|
use T;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace Traits {
|
|
|
|
use Classes\A;
|
|
|
|
|
|
|
|
trait T {
|
|
|
|
public function getA() : A {
|
|
|
|
return new A;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
$a = (new Classes\B)->getA();
|
|
|
|
}',
|
|
|
|
],
|
2018-03-13 04:37:21 +01:00
|
|
|
'useAndMap' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-03-13 04:37:21 +01:00
|
|
|
class C
|
|
|
|
{
|
|
|
|
use T2;
|
2020-03-03 05:07:21 +01:00
|
|
|
|
2018-03-13 04:37:21 +01:00
|
|
|
use T1 {
|
2020-03-03 05:07:21 +01:00
|
|
|
traitFunc as aliasedTraitFunc;
|
2018-03-13 04:37:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public static function func(): void
|
|
|
|
{
|
2020-03-03 05:07:21 +01:00
|
|
|
static::aliasedTraitFunc();
|
2018-03-13 04:37:21 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
trait T1
|
|
|
|
{
|
|
|
|
public static function traitFunc(): void {}
|
|
|
|
}
|
|
|
|
trait T2 { }',
|
|
|
|
],
|
|
|
|
'mapAndUse' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-03-13 04:37:21 +01:00
|
|
|
class C
|
|
|
|
{
|
|
|
|
use T1 {
|
|
|
|
traitFunc as _func;
|
|
|
|
}
|
|
|
|
use T2;
|
|
|
|
|
|
|
|
public static function func(): void
|
|
|
|
{
|
|
|
|
static::_func();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
trait T1
|
|
|
|
{
|
|
|
|
public static function traitFunc(): void {}
|
|
|
|
}
|
|
|
|
trait T2 { }',
|
|
|
|
],
|
2018-03-14 15:51:13 +01:00
|
|
|
'moreArgsInDefined' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-03-14 15:51:13 +01:00
|
|
|
trait T {
|
|
|
|
abstract public function foo() : void;
|
|
|
|
|
|
|
|
public function callFoo() : void {
|
|
|
|
$this->foo();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T;
|
|
|
|
|
|
|
|
public function foo(string $s = null) : void {
|
|
|
|
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2018-03-21 21:55:31 +01:00
|
|
|
'aliasedMethodInternalCallNoReplacement' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-03-21 21:55:31 +01:00
|
|
|
trait T {
|
|
|
|
public function foo() : int {
|
|
|
|
return $this->bar();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function bar() : int {
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T {
|
|
|
|
bar as bat;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function baz() : int {
|
|
|
|
return $this->bar();
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'aliasedMethodInternalCallWithLocalDefinition' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-03-21 21:55:31 +01:00
|
|
|
trait T {
|
|
|
|
public function bar() : int {
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T {
|
|
|
|
bar as bat;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function bar() : string {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
|
|
|
|
public function baz() : string {
|
|
|
|
return $this->bar();
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2018-06-09 02:07:31 +02:00
|
|
|
'allMethodsReplaced' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-06-09 02:07:31 +02:00
|
|
|
trait T {
|
|
|
|
protected function foo() : void {}
|
|
|
|
|
|
|
|
public function bat() : void {
|
|
|
|
$this->foo();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class C {
|
|
|
|
use T;
|
|
|
|
|
|
|
|
protected function foo(string $s) : void {}
|
|
|
|
|
|
|
|
public function bat() : void {
|
|
|
|
$this->foo("bat");
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2018-03-21 22:39:01 +01:00
|
|
|
'aliasedPrivateMethodInternalCallWithLocalDefinition' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-03-21 22:39:01 +01:00
|
|
|
trait T1 {
|
|
|
|
use T2;
|
|
|
|
|
|
|
|
private function foo() : int {
|
|
|
|
return $this->bar();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
trait T2 {
|
|
|
|
private function bar() : int {
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T1;
|
|
|
|
|
|
|
|
private function baz() : int {
|
|
|
|
return $this->bar();
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2018-05-09 04:32:57 +02:00
|
|
|
'traitClassConst' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-05-09 04:32:57 +02:00
|
|
|
trait A {
|
|
|
|
public function foo(): string {
|
|
|
|
return B::class;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
trait B {}
|
|
|
|
|
|
|
|
class C {
|
|
|
|
use A;
|
2019-03-23 19:27:54 +01:00
|
|
|
}',
|
2018-05-09 04:32:57 +02:00
|
|
|
],
|
2018-05-30 16:13:55 +02:00
|
|
|
'noRedundantConditionForTraitStatic' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-05-30 16:13:55 +02:00
|
|
|
trait Foo {
|
|
|
|
public function bar() : array {
|
|
|
|
$type = static::class;
|
|
|
|
$r = new \ReflectionClass($type);
|
|
|
|
$values = $r->getConstants();
|
|
|
|
$callback =
|
|
|
|
/** @param mixed $v */
|
|
|
|
function ($v) : bool {
|
|
|
|
return \is_int($v) || \is_string($v);
|
|
|
|
};
|
|
|
|
|
|
|
|
if (is_a($type, \Bat::class, true)) {
|
|
|
|
$callback =
|
|
|
|
/** @param mixed $v */
|
|
|
|
function ($v) : bool {
|
|
|
|
return \is_int($v) && 0 === ($v & $v - 1) && $v > 0;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return array_filter($values, $callback);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class Bar {
|
|
|
|
use Foo;
|
|
|
|
}
|
|
|
|
|
|
|
|
class Bat {
|
|
|
|
use Foo;
|
2019-03-23 19:27:54 +01:00
|
|
|
}',
|
2018-05-30 16:13:55 +02:00
|
|
|
],
|
2018-07-03 06:51:57 +02:00
|
|
|
'nonMemoizedAssertions' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-07-03 06:51:57 +02:00
|
|
|
trait T {
|
|
|
|
public function compare(O $other) : void {
|
|
|
|
if ($other instanceof self) {
|
|
|
|
if ($other->value === $this->value) {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class O {}
|
|
|
|
|
|
|
|
class A extends O {
|
|
|
|
use T;
|
|
|
|
|
|
|
|
/** @var string */
|
|
|
|
private $value;
|
|
|
|
|
|
|
|
public function __construct(string $string) {
|
|
|
|
$this->value = $string;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class B extends O {
|
|
|
|
use T;
|
|
|
|
|
|
|
|
/** @var bool */
|
|
|
|
private $value;
|
|
|
|
|
|
|
|
public function __construct(bool $bool) {
|
|
|
|
$this->value = $bool;
|
|
|
|
}
|
2019-03-23 19:27:54 +01:00
|
|
|
}',
|
2018-07-03 06:51:57 +02:00
|
|
|
],
|
2018-11-30 19:45:39 +01:00
|
|
|
'manyTraitAliases' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-11-30 19:45:39 +01:00
|
|
|
trait Foo {
|
|
|
|
public static function staticMethod():void {}
|
|
|
|
public function nonstatic():void {}
|
|
|
|
}
|
|
|
|
|
|
|
|
Class Bar {
|
|
|
|
use Foo {
|
|
|
|
Foo::staticMethod as foo;
|
|
|
|
Foo::staticMethod as foobar;
|
|
|
|
Foo::staticMethod as fine;
|
|
|
|
Foo::nonstatic as bad;
|
|
|
|
Foo::nonstatic as good;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$b = new Bar();
|
|
|
|
|
|
|
|
Bar::fine();
|
|
|
|
$b::fine();
|
|
|
|
$b->fine();
|
|
|
|
|
|
|
|
$b->good();
|
|
|
|
|
|
|
|
Bar::foo();
|
|
|
|
Bar::foobar();
|
|
|
|
|
|
|
|
$b::foo();
|
|
|
|
$b::foobar();
|
|
|
|
|
|
|
|
$b->foo();
|
|
|
|
$b->foobar();
|
|
|
|
|
2019-03-23 19:27:54 +01:00
|
|
|
$b->bad();',
|
2018-11-30 19:45:39 +01:00
|
|
|
],
|
2019-01-07 17:53:22 +01:00
|
|
|
'inheritedProtectedTraitMethodAccess' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-01-07 17:53:22 +01:00
|
|
|
trait T {
|
|
|
|
private function bar() : void {}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T {
|
|
|
|
bar as protected;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class AChild extends A {
|
|
|
|
public function foo() : void {
|
|
|
|
$this->bar();
|
|
|
|
}
|
2019-03-23 19:27:54 +01:00
|
|
|
}',
|
2019-01-07 17:53:22 +01:00
|
|
|
],
|
|
|
|
'inheritedPublicTraitMethodAccess' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-01-07 17:53:22 +01:00
|
|
|
trait T {
|
|
|
|
private function bar() : void {}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T {
|
|
|
|
bar as public;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-23 19:27:54 +01:00
|
|
|
(new A)->bar();',
|
2019-01-07 17:53:22 +01:00
|
|
|
],
|
2019-01-16 16:59:06 +01:00
|
|
|
'allowImplementMethodMadePublicInClass' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-01-16 16:59:06 +01:00
|
|
|
interface I {
|
2019-07-29 01:44:36 +02:00
|
|
|
public function boo() : void;
|
2019-01-16 16:59:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
trait T {
|
|
|
|
private function boo() : void {}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A implements I {
|
|
|
|
use T { boo as public; }
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'allowImplementMethodMadePublicInParent' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-01-16 16:59:06 +01:00
|
|
|
interface I {
|
2019-07-29 01:44:36 +02:00
|
|
|
public function boo() : void;
|
2019-01-16 16:59:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
trait T {
|
|
|
|
private function boo() : void {}
|
|
|
|
}
|
|
|
|
|
|
|
|
class B {
|
|
|
|
use T { boo as public; }
|
|
|
|
}
|
|
|
|
|
|
|
|
class BChild extends B implements I {}',
|
|
|
|
],
|
2019-01-18 16:08:16 +01:00
|
|
|
'allowTraitParentDefinition' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-01-18 16:08:16 +01:00
|
|
|
class A {}
|
|
|
|
|
|
|
|
class C extends A
|
|
|
|
{
|
|
|
|
use T;
|
|
|
|
}
|
|
|
|
|
|
|
|
trait T
|
|
|
|
{
|
|
|
|
public function bar() : ?C
|
|
|
|
{
|
|
|
|
if ($this instanceof A) {
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
2019-03-23 19:27:54 +01:00
|
|
|
}',
|
2019-02-10 22:32:30 +01:00
|
|
|
],
|
|
|
|
'noCrashOnUndefinedIgnoredTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-02-10 22:32:30 +01:00
|
|
|
/** @psalm-suppress UndefinedTrait */
|
|
|
|
class C {
|
|
|
|
use UnknownTrait;
|
2019-03-23 19:27:54 +01:00
|
|
|
}',
|
2019-01-18 16:08:16 +01:00
|
|
|
],
|
2019-02-18 22:52:09 +01:00
|
|
|
'reconcileStaticTraitProperties' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-02-18 22:52:09 +01:00
|
|
|
trait T {
|
|
|
|
/**
|
|
|
|
* @var string|null
|
|
|
|
*/
|
|
|
|
private static $b;
|
|
|
|
|
|
|
|
private static function booA(): string {
|
|
|
|
if (self::$b === null) {
|
|
|
|
return "hello";
|
|
|
|
}
|
|
|
|
return self::$b;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class C {
|
|
|
|
use T;
|
|
|
|
}',
|
|
|
|
],
|
2019-03-02 20:17:26 +01:00
|
|
|
'covariantAbstractReturn' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-03-02 20:17:26 +01:00
|
|
|
trait T {
|
|
|
|
/** @return iterable */
|
|
|
|
abstract public function bar();
|
|
|
|
}
|
|
|
|
|
|
|
|
class C {
|
|
|
|
use T;
|
|
|
|
|
|
|
|
/** @return array */
|
|
|
|
public function bar() { return []; }
|
2019-03-23 19:27:54 +01:00
|
|
|
}',
|
2019-03-02 20:17:26 +01:00
|
|
|
],
|
2019-03-17 16:31:04 +01:00
|
|
|
'traitSelfParam' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-03-17 16:31:04 +01:00
|
|
|
trait T {
|
|
|
|
public function bar(self $object): self {
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class Foo {
|
|
|
|
use T;
|
2019-03-17 18:50:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$f1 = new Foo();
|
|
|
|
$f2 = (new Foo())->bar($f1);',
|
2019-03-17 16:31:04 +01:00
|
|
|
],
|
2019-06-26 06:28:43 +02:00
|
|
|
'traitSelfDocblockReturn' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-06-26 06:28:43 +02:00
|
|
|
trait T {
|
|
|
|
/** @return self */
|
|
|
|
public function getSelf() {
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class C {
|
|
|
|
use T;
|
2019-07-05 22:24:00 +02:00
|
|
|
}',
|
2019-06-26 06:28:43 +02:00
|
|
|
],
|
2019-09-01 21:45:28 +02:00
|
|
|
'abstractThisMethod' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-09-01 21:45:28 +02:00
|
|
|
trait ATrait {
|
|
|
|
/** @return $this */
|
|
|
|
abstract public function bar();
|
|
|
|
}
|
|
|
|
|
|
|
|
class C {
|
|
|
|
use ATrait;
|
|
|
|
|
|
|
|
/** @return $this */
|
|
|
|
public function bar() {
|
|
|
|
return $this;
|
|
|
|
}
|
2022-12-18 17:15:15 +01:00
|
|
|
}',
|
2019-09-01 21:45:28 +02:00
|
|
|
],
|
2019-12-20 18:57:31 +01:00
|
|
|
'classAliasedTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-12-20 18:57:31 +01:00
|
|
|
trait FeatureV1 {}
|
|
|
|
|
|
|
|
class_alias(FeatureV1::class, Feature::class);
|
|
|
|
|
|
|
|
class Application {
|
|
|
|
use Feature;
|
|
|
|
}',
|
|
|
|
],
|
2019-12-20 21:11:27 +01:00
|
|
|
'renameMethodNoCrash' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-12-20 21:11:27 +01:00
|
|
|
|
|
|
|
trait HelloTrait {
|
|
|
|
protected function sayHello() : string {
|
|
|
|
return "Hello";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class Person {
|
|
|
|
use HelloTrait {
|
2020-04-28 15:17:31 +02:00
|
|
|
sayHello as originalSayHello;
|
2019-12-20 21:11:27 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
protected function sayHello() : string {
|
2020-04-28 15:17:31 +02:00
|
|
|
return $this->originalSayHello();
|
2019-12-20 21:11:27 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class BrokenPerson extends Person {
|
2020-04-28 15:17:31 +02:00
|
|
|
protected function originalSayHello() : string {
|
2019-12-20 21:11:27 +01:00
|
|
|
return "bad";
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2019-12-28 21:56:19 +01:00
|
|
|
'instanceofStaticInsideTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-12-28 21:56:19 +01:00
|
|
|
trait T {
|
|
|
|
/**
|
|
|
|
* @param mixed $instance
|
|
|
|
* @return ?static
|
|
|
|
*/
|
|
|
|
public static function filterInstance($instance) {
|
|
|
|
return $instance instanceof static ? $instance : null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T;
|
2022-12-18 17:15:15 +01:00
|
|
|
}',
|
2019-12-28 21:56:19 +01:00
|
|
|
],
|
2020-05-26 23:40:27 +02:00
|
|
|
'propertyNotDefinedInTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2020-05-26 23:40:27 +02:00
|
|
|
class A1 {
|
|
|
|
use A2;
|
|
|
|
|
|
|
|
public static string $titlefield = "blah";
|
|
|
|
}
|
|
|
|
|
|
|
|
trait A2 {
|
|
|
|
public static function test() : string {
|
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
$sortfield = (isset(static::$sortfield)) ?
|
|
|
|
static::$sortfield
|
|
|
|
: static::$titlefield;
|
|
|
|
return $sortfield;
|
|
|
|
}
|
2022-12-18 17:15:15 +01:00
|
|
|
}',
|
2020-05-26 23:40:27 +02:00
|
|
|
],
|
2020-08-05 21:58:26 +02:00
|
|
|
'staticNotBoundInFinal' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2020-08-05 21:58:26 +02:00
|
|
|
trait Foo {
|
|
|
|
/**
|
|
|
|
* @return static
|
|
|
|
*/
|
|
|
|
final public function foo(): self
|
|
|
|
{
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use Foo;
|
2022-12-18 17:15:15 +01:00
|
|
|
}',
|
2020-08-05 21:58:26 +02:00
|
|
|
],
|
2020-10-25 17:20:18 +01:00
|
|
|
'staticReturnWithFinal' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2020-10-25 17:20:18 +01:00
|
|
|
trait T {
|
|
|
|
/** @return static */
|
|
|
|
public function instance() {
|
|
|
|
return new static();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
final class A {
|
|
|
|
use T;
|
2022-12-18 17:15:15 +01:00
|
|
|
}',
|
2020-10-25 17:20:18 +01:00
|
|
|
],
|
2022-02-24 01:50:05 +01:00
|
|
|
'suppressIssueOnTrait' => [
|
2022-02-26 21:28:15 +01:00
|
|
|
'code' => '<?php
|
2022-02-24 01:50:05 +01:00
|
|
|
/** @psalm-suppress InvalidAttribute */
|
|
|
|
#[Attribute]
|
|
|
|
trait Foo {}',
|
|
|
|
],
|
2022-12-03 06:21:10 +01:00
|
|
|
'noCrashOnConditionalTrait' => [
|
|
|
|
'code' => '<?php
|
|
|
|
namespace NS;
|
|
|
|
if (rand(0, 1)) {
|
|
|
|
trait T {}
|
|
|
|
}
|
2022-12-18 17:15:15 +01:00
|
|
|
',
|
2022-12-03 06:21:10 +01:00
|
|
|
],
|
2023-01-17 15:25:45 +01:00
|
|
|
'constant in trait' => [
|
|
|
|
'code' => <<<'PHP'
|
|
|
|
<?php
|
|
|
|
trait TraitA {
|
|
|
|
public const PUBLIC_CONST = 'PUBLIC_CONST';
|
|
|
|
protected const PROTECTED_CONST = 'PROTECTED_CONST';
|
|
|
|
private const PRIVATE_CONST = 'PRIVATE_CONST';
|
|
|
|
}
|
|
|
|
class ClassB {
|
|
|
|
use TraitA;
|
|
|
|
public static function getPublicConst(): string { return self::PUBLIC_CONST; }
|
|
|
|
public static function getProtectedConst(): string { return self::PROTECTED_CONST; }
|
|
|
|
public static function getPrivateConst(): string { return self::PRIVATE_CONST; }
|
|
|
|
}
|
|
|
|
class ClassC extends ClassB {
|
|
|
|
public static function getPublicConst(): string { return self::PUBLIC_CONST; }
|
|
|
|
public static function getProtectedConst(): string { return self::PROTECTED_CONST; }
|
|
|
|
}
|
|
|
|
PHP,
|
|
|
|
'assertions' => [],
|
|
|
|
'ignored_issues' => [],
|
|
|
|
'php_version' => '8.2',
|
|
|
|
],
|
|
|
|
'constant in trait with alias' => [
|
|
|
|
'code' => <<<'PHP'
|
|
|
|
<?php
|
|
|
|
trait TraitA { private const PRIVATE_CONST = 'PRIVATE_CONST'; }
|
|
|
|
class ClassB { use TraitA { PRIVATE_CONST as public PUBLIC_CONST; } }
|
|
|
|
$c = ClassB::PUBLIC_CONST;
|
|
|
|
PHP,
|
|
|
|
'assertions' => ['$c' => 'string'],
|
|
|
|
'ignored_issues' => [],
|
|
|
|
'php_version' => '8.2',
|
|
|
|
],
|
2017-04-25 05:45:02 +02:00
|
|
|
];
|
2016-12-17 04:16:29 +01:00
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function providerInvalidCodeParse(): iterable
|
2016-12-17 04:16:29 +01:00
|
|
|
{
|
2017-04-25 05:45:02 +02:00
|
|
|
return [
|
|
|
|
'inaccessiblePrivateMethodFromInheritedTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
private function fooFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use T;
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
class C extends B {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function doFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
$this->fooFoo();
|
|
|
|
}
|
|
|
|
}',
|
2017-05-27 02:05:57 +02:00
|
|
|
'error_message' => 'InaccessibleMethod',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'undefinedTrait' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
class B {
|
|
|
|
use A;
|
|
|
|
}',
|
2017-05-27 02:05:57 +02:00
|
|
|
'error_message' => 'UndefinedTrait',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'missingPropertyType' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
2019-08-11 06:22:28 +02:00
|
|
|
public $foo = null;
|
2017-04-25 05:45:02 +02:00
|
|
|
}
|
|
|
|
class A {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
public function assignToFoo(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
$this->foo = 5;
|
|
|
|
}
|
|
|
|
}',
|
2019-02-27 22:00:44 +01:00
|
|
|
'error_message' => 'MissingPropertyType - src' . DIRECTORY_SEPARATOR . 'somefile.php:3:32 - Property T::$foo does not have a ' .
|
2018-05-18 17:02:50 +02:00
|
|
|
'declared type - consider int|null',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'missingPropertyTypeWithConstructorInit' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
|
|
|
public $foo;
|
|
|
|
}
|
|
|
|
class A {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-04-21 23:20:21 +02:00
|
|
|
public function __construct() {
|
2017-04-25 05:45:02 +02:00
|
|
|
$this->foo = 5;
|
|
|
|
}
|
|
|
|
}',
|
2019-02-27 22:00:44 +01:00
|
|
|
'error_message' => 'MissingPropertyType - src' . DIRECTORY_SEPARATOR . 'somefile.php:3:32 - Property T::$foo does not have a ' .
|
2017-05-27 02:05:57 +02:00
|
|
|
'declared type - consider int',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'missingPropertyTypeWithConstructorInitAndNull' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
|
|
|
public $foo;
|
|
|
|
}
|
|
|
|
class A {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-04-21 23:20:21 +02:00
|
|
|
public function __construct() {
|
2017-04-25 05:45:02 +02:00
|
|
|
$this->foo = 5;
|
|
|
|
}
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
public function makeNull(): void {
|
2017-04-25 05:45:02 +02:00
|
|
|
$this->foo = null;
|
|
|
|
}
|
|
|
|
}',
|
2019-02-27 22:00:44 +01:00
|
|
|
'error_message' => 'MissingPropertyType - src' . DIRECTORY_SEPARATOR . 'somefile.php:3:32 - Property T::$foo does not have a ' .
|
2018-05-18 17:02:50 +02:00
|
|
|
'declared type - consider int|null',
|
2017-04-25 05:45:02 +02:00
|
|
|
],
|
|
|
|
'missingPropertyTypeWithConstructorInitAndNullDefault' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-04-25 05:45:02 +02:00
|
|
|
trait T {
|
|
|
|
public $foo = null;
|
|
|
|
}
|
|
|
|
class A {
|
|
|
|
use T;
|
2017-06-23 06:39:37 +02:00
|
|
|
|
2018-04-21 23:20:21 +02:00
|
|
|
public function __construct() {
|
2017-04-25 05:45:02 +02:00
|
|
|
$this->foo = 5;
|
|
|
|
}
|
|
|
|
}',
|
2019-02-27 22:00:44 +01:00
|
|
|
'error_message' => 'MissingPropertyType - src' . DIRECTORY_SEPARATOR . 'somefile.php:3:32 - Property T::$foo does not have a ' .
|
2018-01-11 23:35:28 +01:00
|
|
|
'declared type - consider int|null',
|
2017-05-27 02:05:57 +02:00
|
|
|
],
|
2017-07-25 22:11:02 +02:00
|
|
|
'redefinedTraitMethodInSubclass' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2017-07-25 22:11:02 +02:00
|
|
|
trait T {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function fooFoo(): void {
|
2017-07-25 22:11:02 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class B {
|
|
|
|
use T;
|
|
|
|
}
|
|
|
|
|
|
|
|
class C extends B {
|
2018-01-11 21:50:45 +01:00
|
|
|
public function fooFoo(string $a): void {
|
2017-07-25 22:11:02 +02:00
|
|
|
}
|
|
|
|
}',
|
|
|
|
'error_message' => 'MethodSignatureMismatch',
|
|
|
|
],
|
2018-02-12 16:46:45 +01:00
|
|
|
'missingTraitPropertyType' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-02-12 16:46:45 +01:00
|
|
|
trait T {
|
2019-08-11 06:22:28 +02:00
|
|
|
public $foo = 5;
|
2018-02-12 16:46:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
class A {
|
|
|
|
use T;
|
|
|
|
}',
|
|
|
|
'error_message' => 'MissingPropertyType',
|
|
|
|
],
|
2018-05-30 18:23:53 +02:00
|
|
|
'nestedTraitWithBadReturnType' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-05-30 18:23:53 +02:00
|
|
|
trait A {
|
|
|
|
public function foo() : string {
|
|
|
|
return 5;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
trait B {
|
|
|
|
use A;
|
|
|
|
}
|
|
|
|
|
|
|
|
class C {
|
|
|
|
use B;
|
|
|
|
}',
|
|
|
|
'error_message' => 'InvalidReturnType',
|
|
|
|
],
|
2018-06-09 02:06:05 +02:00
|
|
|
'replaceTraitMethod' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2018-06-09 02:06:05 +02:00
|
|
|
trait T {
|
|
|
|
protected function foo() : void {}
|
|
|
|
|
|
|
|
public function bat() : void {
|
|
|
|
$this->foo();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class C {
|
|
|
|
use T;
|
|
|
|
|
|
|
|
protected function foo(string $s) : void {}
|
|
|
|
}',
|
|
|
|
'error_message' => 'TooFewArguments',
|
|
|
|
],
|
2019-01-13 16:19:27 +01:00
|
|
|
'traitMethodMadePrivate' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-01-13 16:19:27 +01:00
|
|
|
trait T {
|
|
|
|
public function foo() : void {
|
|
|
|
echo "here";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class C {
|
|
|
|
use T {
|
|
|
|
foo as private traitFoo;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function bar() : void {
|
|
|
|
$this->traitFoo();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class D extends C {
|
|
|
|
public function bar() : void {
|
|
|
|
$this->traitFoo(); // should fail
|
|
|
|
}
|
|
|
|
}',
|
2019-03-23 19:27:54 +01:00
|
|
|
'error_message' => 'InaccessibleMethod',
|
2019-01-13 16:19:27 +01:00
|
|
|
],
|
2019-03-16 03:12:35 +01:00
|
|
|
'preventTraitPropertyType' => [
|
2022-01-13 19:49:37 +01:00
|
|
|
'code' => '<?php
|
2019-03-16 03:12:35 +01:00
|
|
|
trait T {}
|
|
|
|
|
|
|
|
class X {
|
|
|
|
/** @var T|null */
|
|
|
|
public $hm;
|
|
|
|
}',
|
2019-05-16 00:41:26 +02:00
|
|
|
'error_message' => 'UndefinedDocblockClass',
|
2019-03-16 03:12:35 +01:00
|
|
|
],
|
2023-01-17 15:25:45 +01:00
|
|
|
'constant declaration in trait, php <8.2.0' => [
|
|
|
|
'code' => <<<'PHP'
|
|
|
|
<?php
|
|
|
|
trait A { const B = 0; }
|
|
|
|
PHP,
|
2023-03-03 09:25:27 +01:00
|
|
|
'error_message' => 'ConstantDeclarationInTrait',
|
2023-01-17 15:25:45 +01:00
|
|
|
'ignored_issues' => [],
|
|
|
|
'php_version' => '8.1',
|
|
|
|
],
|
2023-10-08 20:13:33 +02:00
|
|
|
'duplicateTraitProperty' => [
|
|
|
|
'code' => '<?php
|
|
|
|
trait T {
|
|
|
|
public mixed $foo = 5;
|
|
|
|
protected static mixed $foo;
|
|
|
|
}
|
|
|
|
',
|
|
|
|
'error_message' => 'DuplicateProperty',
|
|
|
|
],
|
2017-04-25 05:45:02 +02:00
|
|
|
];
|
2017-04-15 06:04:03 +02:00
|
|
|
}
|
2016-12-17 04:16:29 +01:00
|
|
|
}
|