2017-02-10 02:35:17 +01:00
|
|
|
|
<?php
|
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
class TemplateTest extends TestCase
|
2017-02-10 02:35:17 +01:00
|
|
|
|
{
|
2018-11-06 03:57:36 +01:00
|
|
|
|
use Traits\InvalidCodeAnalysisTestTrait;
|
|
|
|
|
use Traits\ValidCodeAnalysisTestTrait;
|
2017-02-10 02:35:17 +01:00
|
|
|
|
|
|
|
|
|
/**
|
2017-04-25 05:45:02 +02:00
|
|
|
|
* @return array
|
2017-02-10 02:35:17 +01:00
|
|
|
|
*/
|
2018-11-06 03:57:36 +01:00
|
|
|
|
public function providerValidCodeParse()
|
2017-02-10 02:35:17 +01:00
|
|
|
|
{
|
2017-04-25 05:45:02 +02:00
|
|
|
|
return [
|
|
|
|
|
'classTemplate' => [
|
|
|
|
|
'<?php
|
|
|
|
|
class A {}
|
|
|
|
|
class B {}
|
|
|
|
|
class C {}
|
|
|
|
|
class D {}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @template T as object
|
|
|
|
|
*/
|
|
|
|
|
class Foo {
|
2018-12-18 05:29:27 +01:00
|
|
|
|
/** @var T::class */
|
2017-04-25 05:45:02 +02:00
|
|
|
|
public $T;
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
2018-11-12 18:03:55 +01:00
|
|
|
|
* @param class-string $T
|
2017-04-25 05:45:02 +02:00
|
|
|
|
* @template-typeof T $T
|
|
|
|
|
*/
|
|
|
|
|
public function __construct(string $T) {
|
|
|
|
|
$this->T = $T;
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
public function bar() {
|
|
|
|
|
$t = $this->T;
|
|
|
|
|
return new $t();
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
$at = "A";
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/** @var Foo<A> */
|
|
|
|
|
$afoo = new Foo($at);
|
|
|
|
|
$afoo_bar = $afoo->bar();
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
$bfoo = new Foo(B::class);
|
|
|
|
|
$bfoo_bar = $bfoo->bar();
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-12-03 18:44:08 +01:00
|
|
|
|
// this shouldn’t cause a problem as it’s a docbblock type
|
|
|
|
|
if (!($bfoo_bar instanceof B)) {}
|
|
|
|
|
|
2018-12-13 06:09:01 +01:00
|
|
|
|
$c = C::class;
|
|
|
|
|
$cfoo = new Foo($c);
|
|
|
|
|
$cfoo_bar = $cfoo->bar();',
|
2018-03-06 19:59:59 +01:00
|
|
|
|
'assertions' => [
|
|
|
|
|
'$afoo' => 'Foo<A>',
|
|
|
|
|
'$afoo_bar' => 'A',
|
|
|
|
|
|
|
|
|
|
'$bfoo' => 'Foo<B>',
|
|
|
|
|
'$bfoo_bar' => 'B',
|
|
|
|
|
|
|
|
|
|
'$cfoo' => 'Foo<C>',
|
|
|
|
|
'$cfoo_bar' => 'C',
|
|
|
|
|
],
|
|
|
|
|
'error_levels' => [
|
|
|
|
|
'MixedReturnStatement',
|
|
|
|
|
'LessSpecificReturnStatement',
|
|
|
|
|
'RedundantConditionGivenDocblockType',
|
2018-11-12 18:03:55 +01:00
|
|
|
|
'TypeCoercion'
|
2018-03-06 19:59:59 +01:00
|
|
|
|
],
|
|
|
|
|
],
|
|
|
|
|
'classTemplateSelfs' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template T as object
|
|
|
|
|
*/
|
|
|
|
|
class Foo {
|
2018-12-18 05:29:27 +01:00
|
|
|
|
/** @var T::class */
|
2018-03-06 19:59:59 +01:00
|
|
|
|
public $T;
|
|
|
|
|
|
|
|
|
|
/**
|
2018-11-12 18:03:55 +01:00
|
|
|
|
* @param class-string $T
|
2018-03-06 19:59:59 +01:00
|
|
|
|
* @template-typeof T $T
|
|
|
|
|
*/
|
|
|
|
|
public function __construct(string $T) {
|
|
|
|
|
$this->T = $T;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
public function bar() {
|
|
|
|
|
$t = $this->T;
|
|
|
|
|
return new $t();
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-03-05 23:36:08 +01:00
|
|
|
|
|
|
|
|
|
class E {
|
|
|
|
|
/**
|
|
|
|
|
* @return Foo<self>
|
|
|
|
|
*/
|
|
|
|
|
public static function getFoo() {
|
|
|
|
|
return new Foo(__CLASS__);
|
|
|
|
|
}
|
|
|
|
|
|
2018-03-06 19:59:59 +01:00
|
|
|
|
/**
|
|
|
|
|
* @return Foo<self>
|
|
|
|
|
*/
|
|
|
|
|
public static function getFoo2() {
|
|
|
|
|
return new Foo(self::class);
|
|
|
|
|
}
|
2017-04-25 05:45:02 +02:00
|
|
|
|
|
2018-03-06 19:59:59 +01:00
|
|
|
|
/**
|
|
|
|
|
* @return Foo<static>
|
|
|
|
|
*/
|
|
|
|
|
public static function getFoo3() {
|
|
|
|
|
return new Foo(static::class);
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-04-25 05:45:02 +02:00
|
|
|
|
|
2018-03-06 19:59:59 +01:00
|
|
|
|
class G extends E {}
|
2017-04-25 05:45:02 +02:00
|
|
|
|
|
2018-03-06 19:59:59 +01:00
|
|
|
|
$efoo = E::getFoo();
|
|
|
|
|
$efoo2 = E::getFoo2();
|
|
|
|
|
$efoo3 = E::getFoo3();
|
2017-12-03 18:44:08 +01:00
|
|
|
|
|
2018-03-06 19:59:59 +01:00
|
|
|
|
$gfoo = G::getFoo();
|
|
|
|
|
$gfoo2 = G::getFoo2();
|
|
|
|
|
$gfoo3 = G::getFoo3();',
|
|
|
|
|
'assertions' => [
|
2018-03-05 23:36:08 +01:00
|
|
|
|
'$efoo' => 'Foo<E>',
|
2018-03-06 19:59:59 +01:00
|
|
|
|
'$efoo2' => 'Foo<E>',
|
|
|
|
|
'$efoo3' => 'Foo<E>',
|
|
|
|
|
'$gfoo' => 'Foo<E>',
|
|
|
|
|
'$gfoo2' => 'Foo<E>',
|
|
|
|
|
'$gfoo3' => 'Foo<G>',
|
2017-12-03 18:44:08 +01:00
|
|
|
|
],
|
2018-02-07 00:44:53 +01:00
|
|
|
|
'error_levels' => [
|
|
|
|
|
'LessSpecificReturnStatement',
|
2018-02-07 21:20:47 +01:00
|
|
|
|
'RedundantConditionGivenDocblockType',
|
2018-02-07 00:44:53 +01:00
|
|
|
|
],
|
2017-12-03 18:44:08 +01:00
|
|
|
|
],
|
2017-07-25 22:11:02 +02:00
|
|
|
|
'classTemplateExternalClasses' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template T as object
|
|
|
|
|
*/
|
|
|
|
|
class Foo {
|
2018-12-18 05:29:27 +01:00
|
|
|
|
/** @var T::class */
|
2017-07-25 22:11:02 +02:00
|
|
|
|
public $T;
|
|
|
|
|
|
|
|
|
|
/**
|
2018-11-12 18:03:55 +01:00
|
|
|
|
* @param class-string $T
|
2017-07-25 22:11:02 +02:00
|
|
|
|
* @template-typeof T $T
|
|
|
|
|
*/
|
|
|
|
|
public function __construct(string $T) {
|
|
|
|
|
$this->T = $T;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
public function bar() {
|
|
|
|
|
$t = $this->T;
|
|
|
|
|
return new $t();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-12-18 05:29:27 +01:00
|
|
|
|
$efoo = new Foo(\Exception::class);
|
2017-07-25 22:11:02 +02:00
|
|
|
|
$efoo_bar = $efoo->bar();
|
|
|
|
|
|
2018-12-18 05:29:27 +01:00
|
|
|
|
$ffoo = new Foo(\LogicException::class);
|
2017-07-25 22:11:02 +02:00
|
|
|
|
$ffoo_bar = $ffoo->bar();',
|
|
|
|
|
'assertions' => [
|
|
|
|
|
'$efoo' => 'Foo<Exception>',
|
|
|
|
|
'$efoo_bar' => 'Exception',
|
|
|
|
|
|
|
|
|
|
'$ffoo' => 'Foo<LogicException>',
|
|
|
|
|
'$ffoo_bar' => 'LogicException',
|
|
|
|
|
],
|
2018-12-13 06:09:01 +01:00
|
|
|
|
'error_levels' => ['LessSpecificReturnStatement'],
|
2017-07-25 22:11:02 +02:00
|
|
|
|
],
|
2017-04-25 05:45:02 +02:00
|
|
|
|
'classTemplateContainer' => [
|
|
|
|
|
'<?php
|
|
|
|
|
class A {}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
*/
|
|
|
|
|
class Foo {
|
|
|
|
|
/** @var T */
|
|
|
|
|
public $obj;
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @param T $obj
|
|
|
|
|
*/
|
|
|
|
|
public function __construct($obj) {
|
|
|
|
|
$this->obj = $obj;
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
public function bar() {
|
|
|
|
|
return $this->obj;
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2018-04-20 16:52:23 +02:00
|
|
|
|
/**
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
public function bat() {
|
|
|
|
|
return $this->bar();
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
|
public function __toString(): string {
|
2017-04-25 05:45:02 +02:00
|
|
|
|
return "hello " . $this->obj;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
$afoo = new Foo(new A());
|
|
|
|
|
$afoo_bar = $afoo->bar();',
|
|
|
|
|
'assertions' => [
|
2017-06-29 16:22:49 +02:00
|
|
|
|
'$afoo' => 'Foo<A>',
|
|
|
|
|
'$afoo_bar' => 'A',
|
2017-04-25 05:45:02 +02:00
|
|
|
|
],
|
2018-04-20 16:52:23 +02:00
|
|
|
|
'error_levels' => ['MixedOperand'],
|
2017-04-25 05:45:02 +02:00
|
|
|
|
],
|
|
|
|
|
'phanTuple' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace Phan\Library;
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* An abstract tuple.
|
|
|
|
|
*/
|
|
|
|
|
abstract class Tuple
|
|
|
|
|
{
|
|
|
|
|
const ARITY = 0;
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @return int
|
|
|
|
|
* The arity of this tuple
|
|
|
|
|
*/
|
2018-01-11 21:50:45 +01:00
|
|
|
|
public function arity(): int
|
2017-04-25 05:45:02 +02:00
|
|
|
|
{
|
|
|
|
|
return (int)static::ARITY;
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @return array
|
|
|
|
|
* An array of all elements in this tuple.
|
|
|
|
|
*/
|
2018-01-11 21:50:45 +01:00
|
|
|
|
abstract public function toArray(): array;
|
2017-04-25 05:45:02 +02:00
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* A tuple of 1 element.
|
|
|
|
|
*
|
|
|
|
|
* @template T0
|
|
|
|
|
* The type of element zero
|
|
|
|
|
*/
|
|
|
|
|
class Tuple1 extends Tuple
|
|
|
|
|
{
|
|
|
|
|
/** @var int */
|
|
|
|
|
const ARITY = 1;
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/** @var T0 */
|
|
|
|
|
public $_0;
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @param T0 $_0
|
|
|
|
|
* The 0th element
|
|
|
|
|
*/
|
|
|
|
|
public function __construct($_0) {
|
|
|
|
|
$this->_0 = $_0;
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @return int
|
|
|
|
|
* The arity of this tuple
|
|
|
|
|
*/
|
2018-01-11 21:50:45 +01:00
|
|
|
|
public function arity(): int
|
2017-04-25 05:45:02 +02:00
|
|
|
|
{
|
|
|
|
|
return (int)static::ARITY;
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @return array
|
|
|
|
|
* An array of all elements in this tuple.
|
|
|
|
|
*/
|
2018-01-11 21:50:45 +01:00
|
|
|
|
public function toArray(): array
|
2017-04-25 05:45:02 +02:00
|
|
|
|
{
|
|
|
|
|
return [
|
|
|
|
|
$this->_0,
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* A tuple of 2 elements.
|
|
|
|
|
*
|
|
|
|
|
* @template T0
|
|
|
|
|
* The type of element zero
|
|
|
|
|
*
|
|
|
|
|
* @template T1
|
|
|
|
|
* The type of element one
|
|
|
|
|
*
|
|
|
|
|
* @inherits Tuple1<T0>
|
|
|
|
|
*/
|
|
|
|
|
class Tuple2 extends Tuple1
|
|
|
|
|
{
|
|
|
|
|
/** @var int */
|
|
|
|
|
const ARITY = 2;
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/** @var T1 */
|
|
|
|
|
public $_1;
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @param T0 $_0
|
|
|
|
|
* The 0th element
|
|
|
|
|
*
|
|
|
|
|
* @param T1 $_1
|
|
|
|
|
* The 1st element
|
|
|
|
|
*/
|
|
|
|
|
public function __construct($_0, $_1) {
|
|
|
|
|
parent::__construct($_0);
|
|
|
|
|
$this->_1 = $_1;
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @return array
|
|
|
|
|
* An array of all elements in this tuple.
|
|
|
|
|
*/
|
2018-01-11 21:50:45 +01:00
|
|
|
|
public function toArray(): array
|
2017-04-25 05:45:02 +02:00
|
|
|
|
{
|
|
|
|
|
return [
|
|
|
|
|
$this->_0,
|
|
|
|
|
$this->_1,
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
$a = new Tuple2("cool", 5);
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/** @return void */
|
|
|
|
|
function takes_int(int $i) {}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/** @return void */
|
|
|
|
|
function takes_string(string $s) {}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
takes_string($a->_0);
|
2017-05-27 02:05:57 +02:00
|
|
|
|
takes_int($a->_1);',
|
2017-04-25 05:45:02 +02:00
|
|
|
|
],
|
|
|
|
|
'validTemplatedType' => [
|
|
|
|
|
'<?php
|
2017-07-25 22:11:02 +02:00
|
|
|
|
namespace FooFoo;
|
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @param T $x
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
function foo($x) {
|
|
|
|
|
return $x;
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
|
function bar(string $a): void { }
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-05-27 02:05:57 +02:00
|
|
|
|
bar(foo("string"));',
|
2017-04-25 05:45:02 +02:00
|
|
|
|
],
|
2018-07-14 01:09:35 +02:00
|
|
|
|
'validPsalmTemplatedType' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace FooFoo;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @psalm-template T
|
|
|
|
|
* @psalm-param T $x
|
|
|
|
|
* @psalm-return T
|
|
|
|
|
*/
|
|
|
|
|
function foo($x) {
|
|
|
|
|
return $x;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function bar(string $a): void { }
|
|
|
|
|
|
|
|
|
|
bar(foo("string"));',
|
|
|
|
|
],
|
2017-04-25 05:45:02 +02:00
|
|
|
|
'validTemplatedStaticMethodType' => [
|
|
|
|
|
'<?php
|
2017-07-25 22:11:02 +02:00
|
|
|
|
namespace FooFoo;
|
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
class A {
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @param T $x
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
public static function foo($x) {
|
|
|
|
|
return $x;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
|
function bar(string $a): void { }
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-05-27 02:05:57 +02:00
|
|
|
|
bar(A::foo("string"));',
|
2017-04-25 05:45:02 +02:00
|
|
|
|
],
|
|
|
|
|
'validTemplatedInstanceMethodType' => [
|
|
|
|
|
'<?php
|
2017-07-25 22:11:02 +02:00
|
|
|
|
namespace FooFoo;
|
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
class A {
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @param T $x
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
public function foo($x) {
|
|
|
|
|
return $x;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
|
function bar(string $a): void { }
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-05-27 02:05:57 +02:00
|
|
|
|
bar((new A())->foo("string"));',
|
2017-04-25 05:45:02 +02:00
|
|
|
|
],
|
|
|
|
|
'genericArrayKeys' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
2019-01-05 06:15:53 +01:00
|
|
|
|
* @template T as array-key
|
2017-04-25 05:45:02 +02:00
|
|
|
|
*
|
|
|
|
|
* @param array<T, mixed> $arr
|
|
|
|
|
* @return array<int, T>
|
|
|
|
|
*/
|
|
|
|
|
function my_array_keys($arr) {
|
|
|
|
|
return array_keys($arr);
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-07-25 22:11:02 +02:00
|
|
|
|
$a = my_array_keys(["hello" => 5, "goodbye" => new \Exception()]);',
|
2017-04-25 05:45:02 +02:00
|
|
|
|
'assertions' => [
|
2017-06-29 16:22:49 +02:00
|
|
|
|
'$a' => 'array<int, string>',
|
2017-05-27 02:05:57 +02:00
|
|
|
|
],
|
2017-04-25 05:45:02 +02:00
|
|
|
|
],
|
2017-12-07 21:50:25 +01:00
|
|
|
|
'genericArrayFlip' => [
|
2017-04-25 05:45:02 +02:00
|
|
|
|
'<?php
|
|
|
|
|
/**
|
2019-01-05 06:15:53 +01:00
|
|
|
|
* @template TKey as array-key
|
2017-04-25 05:45:02 +02:00
|
|
|
|
* @template TValue
|
|
|
|
|
*
|
|
|
|
|
* @param array<TKey, TValue> $arr
|
|
|
|
|
* @return array<TValue, TKey>
|
|
|
|
|
*/
|
2017-12-07 21:50:25 +01:00
|
|
|
|
function my_array_flip($arr) {
|
|
|
|
|
return array_flip($arr);
|
2017-04-25 05:45:02 +02:00
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-12-07 21:50:25 +01:00
|
|
|
|
$b = my_array_flip(["hello" => 5, "goodbye" => 6]);',
|
2017-04-25 05:45:02 +02:00
|
|
|
|
'assertions' => [
|
2017-06-29 16:22:49 +02:00
|
|
|
|
'$b' => 'array<int, string>',
|
2017-05-27 02:05:57 +02:00
|
|
|
|
],
|
|
|
|
|
],
|
2018-11-21 18:38:43 +01:00
|
|
|
|
'byRefKeyValueArray' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template TValue
|
2019-01-05 06:15:53 +01:00
|
|
|
|
* @template TKey as array-key
|
2018-11-21 18:38:43 +01:00
|
|
|
|
*
|
|
|
|
|
* @param array<TKey, TValue> $arr
|
|
|
|
|
*/
|
|
|
|
|
function byRef(array &$arr) : void {}
|
|
|
|
|
|
|
|
|
|
$b = ["a" => 5, "c" => 6];
|
|
|
|
|
byRef($b);',
|
|
|
|
|
'assertions' => [
|
|
|
|
|
'$b' => 'array<string, int>',
|
|
|
|
|
],
|
|
|
|
|
],
|
|
|
|
|
'byRefMixedKeyArray' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template TValue
|
|
|
|
|
*
|
|
|
|
|
* @param array<mixed, TValue> $arr
|
|
|
|
|
*/
|
|
|
|
|
function byRef(array &$arr) : void {}
|
|
|
|
|
|
|
|
|
|
$b = ["a" => 5, "c" => 6];
|
|
|
|
|
byRef($b);',
|
|
|
|
|
'assertions' => [
|
|
|
|
|
'$b' => 'array<mixed, int>',
|
|
|
|
|
],
|
|
|
|
|
],
|
|
|
|
|
'mixedArrayPop' => [
|
2017-11-20 06:12:17 +01:00
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template TValue
|
|
|
|
|
*
|
2019-01-05 06:15:53 +01:00
|
|
|
|
* @param array<array-key, TValue> $arr
|
2017-12-07 21:50:25 +01:00
|
|
|
|
* @return TValue|null
|
2017-11-20 06:12:17 +01:00
|
|
|
|
*/
|
|
|
|
|
function my_array_pop(array &$arr) {
|
|
|
|
|
return array_pop($arr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** @var mixed */
|
|
|
|
|
$b = ["a" => 5, "c" => 6];
|
|
|
|
|
$a = my_array_pop($b);',
|
|
|
|
|
'assertions' => [
|
|
|
|
|
'$a' => 'mixed',
|
2019-01-05 06:15:53 +01:00
|
|
|
|
'$b' => 'array<array-key, mixed>',
|
2017-11-20 06:12:17 +01:00
|
|
|
|
],
|
|
|
|
|
'error_levels' => ['MixedAssignment', 'MixedArgument'],
|
|
|
|
|
],
|
2018-11-21 18:38:43 +01:00
|
|
|
|
'genericArrayPop' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template TValue
|
2019-01-05 06:15:53 +01:00
|
|
|
|
* @template TKey as array-key
|
2018-11-21 18:38:43 +01:00
|
|
|
|
*
|
|
|
|
|
* @param array<TKey, TValue> $arr
|
|
|
|
|
* @return TValue|null
|
|
|
|
|
*/
|
|
|
|
|
function my_array_pop(array &$arr) {
|
|
|
|
|
return array_pop($arr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$b = ["a" => 5, "c" => 6];
|
|
|
|
|
$a = my_array_pop($b);',
|
|
|
|
|
'assertions' => [
|
|
|
|
|
'$a' => 'null|int',
|
|
|
|
|
'$b' => 'array<string, int>',
|
|
|
|
|
],
|
|
|
|
|
],
|
2018-05-09 05:17:11 +02:00
|
|
|
|
'intersectionTemplatedTypes' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace NS;
|
|
|
|
|
use Countable;
|
|
|
|
|
|
|
|
|
|
/** @template T */
|
|
|
|
|
class Collection
|
|
|
|
|
{
|
|
|
|
|
/** @psalm-var iterable<T> */
|
|
|
|
|
private $data;
|
|
|
|
|
|
|
|
|
|
/** @psalm-param iterable<T> $data */
|
|
|
|
|
public function __construct(iterable $data) {
|
|
|
|
|
$this->data = $data;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class Item {}
|
|
|
|
|
/** @psalm-param Collection<Item> $c */
|
|
|
|
|
function takesCollectionOfItems(Collection $c): void {}
|
|
|
|
|
|
|
|
|
|
/** @psalm-var iterable<Item> $data2 */
|
|
|
|
|
$data2 = [];
|
|
|
|
|
takesCollectionOfItems(new Collection($data2));
|
|
|
|
|
|
|
|
|
|
/** @psalm-var iterable<Item>&Countable $data */
|
|
|
|
|
$data = [];
|
|
|
|
|
takesCollectionOfItems(new Collection($data));',
|
|
|
|
|
],
|
2018-05-11 22:17:21 +02:00
|
|
|
|
'templateCallableReturnType' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace NS;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @psalm-param callable():T $action
|
|
|
|
|
* @psalm-return T
|
|
|
|
|
*/
|
|
|
|
|
function retry(int $maxRetries, callable $action) {
|
|
|
|
|
return $action();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function takesInt(int $p): void{};
|
|
|
|
|
|
|
|
|
|
takesInt(retry(1, function(): int { return 1; }));',
|
|
|
|
|
],
|
|
|
|
|
'templateClosureReturnType' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace NS;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @psalm-param \Closure():T $action
|
|
|
|
|
* @psalm-return T
|
|
|
|
|
*/
|
|
|
|
|
function retry(int $maxRetries, callable $action) {
|
|
|
|
|
return $action();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function takesInt(int $p): void{};
|
|
|
|
|
|
|
|
|
|
takesInt(retry(1, function(): int { return 1; }));',
|
|
|
|
|
],
|
2018-05-18 23:47:40 +02:00
|
|
|
|
'repeatedCall' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace NS;
|
|
|
|
|
|
|
|
|
|
use Closure;
|
|
|
|
|
|
|
|
|
|
/**
|
2019-01-05 06:15:53 +01:00
|
|
|
|
* @template TKey as array-key
|
2018-05-18 23:47:40 +02:00
|
|
|
|
* @template TValue
|
|
|
|
|
*/
|
|
|
|
|
class ArrayCollection {
|
|
|
|
|
/** @var array<TKey,TValue> */
|
|
|
|
|
private $data;
|
|
|
|
|
|
|
|
|
|
/** @param array<TKey,TValue> $data */
|
|
|
|
|
public function __construct(array $data) {
|
|
|
|
|
$this->data = $data;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @param Closure(TValue):T $func
|
|
|
|
|
* @return ArrayCollection<TKey,T>
|
|
|
|
|
*/
|
|
|
|
|
public function map(Closure $func) {
|
|
|
|
|
return new static(array_map($func, $this->data));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class Item {}
|
|
|
|
|
/**
|
|
|
|
|
* @param ArrayCollection<mixed,Item> $i
|
|
|
|
|
*/
|
|
|
|
|
function takesCollectionOfItems(ArrayCollection $i): void {}
|
|
|
|
|
|
|
|
|
|
$c = new ArrayCollection([ new Item ]);
|
|
|
|
|
takesCollectionOfItems($c);
|
|
|
|
|
takesCollectionOfItems($c->map(function(Item $i): Item { return $i;}));
|
|
|
|
|
takesCollectionOfItems($c->map(function(Item $i): Item { return $i;}));'
|
|
|
|
|
],
|
2018-05-28 23:26:43 +02:00
|
|
|
|
'replaceChildType' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
2019-01-05 06:15:53 +01:00
|
|
|
|
* @template TKey as array-key
|
2018-05-28 23:26:43 +02:00
|
|
|
|
* @template TValue
|
|
|
|
|
* @param Traversable<TKey, TValue> $t
|
|
|
|
|
* @return array<TKey, TValue>
|
|
|
|
|
*/
|
|
|
|
|
function f(Traversable $t): array {
|
|
|
|
|
$ret = [];
|
|
|
|
|
foreach ($t as $k => $v) $ret[$k] = $v;
|
|
|
|
|
return $ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** @return Generator<int, stdClass> */
|
|
|
|
|
function g():Generator { yield new stdClass; }
|
|
|
|
|
|
|
|
|
|
takesArrayOfStdClass(f(g()));
|
|
|
|
|
|
|
|
|
|
/** @param array<stdClass> $p */
|
|
|
|
|
function takesArrayOfStdClass(array $p): void {}',
|
|
|
|
|
],
|
2018-06-13 14:38:07 +02:00
|
|
|
|
'noRepeatedTypeException' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/** @template T */
|
|
|
|
|
class Foo
|
|
|
|
|
{
|
|
|
|
|
/**
|
|
|
|
|
* @psalm-var class-string
|
|
|
|
|
*/
|
|
|
|
|
private $type;
|
|
|
|
|
|
|
|
|
|
/** @var array<T> */
|
|
|
|
|
private $items;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param class-string $type
|
|
|
|
|
* @template-typeof T $type
|
|
|
|
|
*/
|
|
|
|
|
public function __construct(string $type)
|
|
|
|
|
{
|
|
|
|
|
if (!in_array($type, [A::class, B::class], true)) {
|
|
|
|
|
throw new \InvalidArgumentException;
|
|
|
|
|
}
|
|
|
|
|
$this->type = $type;
|
|
|
|
|
$this->items = [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** @param T $item */
|
|
|
|
|
public function add($item): void
|
|
|
|
|
{
|
|
|
|
|
$this->items[] = $item;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class FooFacade
|
|
|
|
|
{
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @param T $item
|
|
|
|
|
*/
|
|
|
|
|
public function add($item): void
|
|
|
|
|
{
|
|
|
|
|
$foo = $this->ensureFoo([$item]);
|
|
|
|
|
$foo->add($item);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @param array<mixed,T> $items
|
|
|
|
|
* @return Foo<T>
|
|
|
|
|
*/
|
|
|
|
|
private function ensureFoo(array $items): EntitySeries
|
|
|
|
|
{
|
|
|
|
|
$type = $items[0] instanceof A ? A::class : B::class;
|
|
|
|
|
return new Foo($type);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class A {}
|
|
|
|
|
class B {}'
|
|
|
|
|
],
|
2018-06-19 19:19:34 +02:00
|
|
|
|
'collectionOfClosure' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template TKey
|
|
|
|
|
* @template TValue
|
|
|
|
|
*/
|
|
|
|
|
class Collection {
|
|
|
|
|
/**
|
|
|
|
|
* @param Closure(TValue):bool $p
|
|
|
|
|
* @return Collection<TKey,TValue>
|
|
|
|
|
*/
|
|
|
|
|
public function filter(Closure $p);
|
|
|
|
|
}
|
|
|
|
|
class I {}
|
|
|
|
|
|
|
|
|
|
/** @var Collection<mixed,Collection<mixed,I>> $c */
|
|
|
|
|
$c = new Collection;
|
|
|
|
|
|
|
|
|
|
$c->filter(
|
|
|
|
|
/** @param Collection<mixed,I> $elt */
|
|
|
|
|
function(Collection $elt): bool { return (bool) rand(0,1); }
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
$c->filter(
|
|
|
|
|
/** @param Collection<mixed,I> $elt */
|
|
|
|
|
function(Collection $elt): bool { return true; }
|
|
|
|
|
);',
|
|
|
|
|
],
|
2018-06-20 16:40:50 +02:00
|
|
|
|
'splatTemplateParam' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
2019-01-05 06:15:53 +01:00
|
|
|
|
* @template TKey as array-key
|
2018-06-20 16:40:50 +02:00
|
|
|
|
* @template TValue
|
|
|
|
|
*
|
|
|
|
|
* @param array<TKey, TValue> $arr
|
|
|
|
|
* @param array $arr2
|
|
|
|
|
* @return array<TKey, TValue>
|
|
|
|
|
*/
|
|
|
|
|
function splat_proof(array $arr, array $arr2) {
|
|
|
|
|
return $arr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$foo = [
|
|
|
|
|
[1, 2, 3],
|
|
|
|
|
[1, 2],
|
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
$a = splat_proof(... $foo);',
|
|
|
|
|
'assertions' => [
|
|
|
|
|
'$a' => 'array<int, int>',
|
|
|
|
|
],
|
|
|
|
|
],
|
2018-11-21 18:38:43 +01:00
|
|
|
|
'passArrayByRef' => [
|
|
|
|
|
'<?php
|
|
|
|
|
function acceptsStdClass(stdClass $_p): void {}
|
|
|
|
|
|
|
|
|
|
$q = [new stdClass];
|
|
|
|
|
acceptsStdClass(fNoRef($q));
|
|
|
|
|
acceptsStdClass(fRef($q));
|
|
|
|
|
acceptsStdClass(fNoRef($q));
|
|
|
|
|
|
|
|
|
|
/**
|
2019-01-05 06:15:53 +01:00
|
|
|
|
* @template TKey as array-key
|
2018-11-21 18:38:43 +01:00
|
|
|
|
* @template TValue
|
|
|
|
|
*
|
|
|
|
|
* @param array<TKey, TValue> $_arr
|
|
|
|
|
* @return null|TValue
|
|
|
|
|
* @psalm-ignore-nullable-return
|
|
|
|
|
*/
|
|
|
|
|
function fRef(array &$_arr) {
|
|
|
|
|
return array_shift($_arr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2019-01-05 06:15:53 +01:00
|
|
|
|
* @template TKey as array-key
|
2018-11-21 18:38:43 +01:00
|
|
|
|
* @template TValue
|
|
|
|
|
*
|
|
|
|
|
* @param array<TKey, TValue> $_arr
|
|
|
|
|
* @return null|TValue
|
|
|
|
|
* @psalm-ignore-nullable-return
|
|
|
|
|
*/
|
|
|
|
|
function fNoRef(array $_arr) {
|
|
|
|
|
return array_shift($_arr);
|
|
|
|
|
}',
|
2018-11-21 23:56:04 +01:00
|
|
|
|
],
|
|
|
|
|
'templatedInterfaceIteration' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace NS;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template TKey
|
|
|
|
|
* @template TValue
|
|
|
|
|
*/
|
|
|
|
|
interface ICollection extends \IteratorAggregate {
|
|
|
|
|
/** @return \Traversable<TKey,TValue> */
|
|
|
|
|
public function getIterator();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class Collection implements ICollection {
|
|
|
|
|
/** @var array */
|
|
|
|
|
private $data;
|
|
|
|
|
public function __construct(array $data) {
|
|
|
|
|
$this->data = $data;
|
|
|
|
|
}
|
|
|
|
|
/** @psalm-suppress LessSpecificImplementedReturnType */
|
|
|
|
|
public function getIterator(): \Traversable {
|
|
|
|
|
return new \ArrayIterator($this->data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** @var ICollection<string, int> */
|
|
|
|
|
$c = new Collection(["a" => 1]);
|
|
|
|
|
|
|
|
|
|
foreach ($c as $k => $v) { atan($v); strlen($k); }'
|
|
|
|
|
],
|
|
|
|
|
'templatedInterfaceGetIteratorIteration' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace NS;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template TKey
|
|
|
|
|
* @template TValue
|
|
|
|
|
*/
|
|
|
|
|
interface ICollection extends \IteratorAggregate {
|
|
|
|
|
/** @return \Traversable<TKey,TValue> */
|
|
|
|
|
public function getIterator();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class Collection implements ICollection {
|
|
|
|
|
/** @var array */
|
|
|
|
|
private $data;
|
|
|
|
|
public function __construct(array $data) {
|
|
|
|
|
$this->data = $data;
|
|
|
|
|
}
|
|
|
|
|
/** @psalm-suppress LessSpecificImplementedReturnType */
|
|
|
|
|
public function getIterator(): \Traversable {
|
|
|
|
|
return new \ArrayIterator($this->data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** @var ICollection<string, int> */
|
|
|
|
|
$c = new Collection(["a" => 1]);
|
|
|
|
|
|
|
|
|
|
foreach ($c->getIterator() as $k => $v) { atan($v); strlen($k); }'
|
|
|
|
|
],
|
2018-11-22 00:38:09 +01:00
|
|
|
|
'implictIteratorTemplating' => [
|
|
|
|
|
'<?php
|
|
|
|
|
class SomeIterator implements IteratorAggregate
|
|
|
|
|
{
|
|
|
|
|
/**
|
|
|
|
|
* @return Generator<int, int>
|
|
|
|
|
*/
|
|
|
|
|
function getIterator()
|
|
|
|
|
{
|
|
|
|
|
yield 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** @param \IteratorAggregate<mixed, int> $i */
|
|
|
|
|
function takesIteratorOfInts(\IteratorAggregate $i) : void {
|
|
|
|
|
foreach ($i as $j) {
|
|
|
|
|
echo $j;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
takesIteratorOfInts(new SomeIterator());'
|
|
|
|
|
],
|
2018-12-08 21:17:43 +01:00
|
|
|
|
'allowTemplatedIntersectionToExtend' => [
|
|
|
|
|
'<?php
|
|
|
|
|
interface Foo {}
|
|
|
|
|
|
|
|
|
|
interface AlmostFoo {
|
|
|
|
|
/**
|
|
|
|
|
* @return Foo
|
|
|
|
|
*/
|
|
|
|
|
public function makeFoo();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
*/
|
|
|
|
|
final class AlmostFooMap implements AlmostFoo {
|
|
|
|
|
/** @var T&Foo */
|
|
|
|
|
private $bar;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param T&Foo $closure
|
|
|
|
|
*/
|
|
|
|
|
public function __construct(Foo $bar)
|
|
|
|
|
{
|
|
|
|
|
$this->bar = $bar;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @return T&Foo
|
|
|
|
|
*/
|
|
|
|
|
public function makeFoo()
|
|
|
|
|
{
|
|
|
|
|
return $this->bar;
|
|
|
|
|
}
|
|
|
|
|
}'
|
2018-12-13 06:09:01 +01:00
|
|
|
|
],
|
|
|
|
|
'restrictTemplateInputWithTClassGoodInput' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace Bar;
|
|
|
|
|
|
|
|
|
|
/** @template T */
|
|
|
|
|
class Foo
|
|
|
|
|
{
|
|
|
|
|
/**
|
|
|
|
|
* @psalm-var T::class
|
|
|
|
|
*/
|
|
|
|
|
private $type;
|
|
|
|
|
|
|
|
|
|
/** @var array<T> */
|
|
|
|
|
private $items;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param T::class $type
|
|
|
|
|
*/
|
|
|
|
|
public function __construct(string $type)
|
|
|
|
|
{
|
|
|
|
|
if (!in_array($type, [A::class, B::class], true)) {
|
|
|
|
|
throw new \InvalidArgumentException;
|
|
|
|
|
}
|
|
|
|
|
$this->type = $type;
|
|
|
|
|
$this->items = [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** @param T $item */
|
|
|
|
|
public function add($item): void
|
|
|
|
|
{
|
|
|
|
|
$this->items[] = $item;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class A {}
|
|
|
|
|
class B {}
|
|
|
|
|
|
|
|
|
|
$foo = new Foo(A::class);
|
|
|
|
|
$foo->add(new A);',
|
|
|
|
|
],
|
2018-12-17 21:49:59 +01:00
|
|
|
|
'classTemplateAsCorrect' => [
|
|
|
|
|
'<?php
|
|
|
|
|
class Foo {}
|
|
|
|
|
class FooChild extends Foo {}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T as Foo
|
|
|
|
|
* @param T $x
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
function bar($x) {
|
|
|
|
|
return $x;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bar(new Foo());
|
|
|
|
|
bar(new FooChild());',
|
|
|
|
|
],
|
|
|
|
|
'classTemplateAsInterface' => [
|
|
|
|
|
'<?php
|
|
|
|
|
interface Foo {}
|
|
|
|
|
interface FooChild extends Foo {}
|
|
|
|
|
class FooImplementer implements Foo {}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T as Foo
|
|
|
|
|
* @param T $x
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
function bar($x) {
|
|
|
|
|
return $x;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function takesFoo(Foo $f) : void {
|
|
|
|
|
bar($f);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function takesFooChild(FooChild $f) : void {
|
|
|
|
|
bar($f);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function takesFooImplementer(FooImplementer $f) : void {
|
|
|
|
|
bar($f);
|
|
|
|
|
}',
|
|
|
|
|
],
|
2018-12-18 05:29:27 +01:00
|
|
|
|
'classTemplateFunctionImplementsInterface' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace A\B;
|
|
|
|
|
|
|
|
|
|
interface Foo {}
|
|
|
|
|
|
|
|
|
|
interface IFooGetter {
|
|
|
|
|
/**
|
|
|
|
|
* @return Foo
|
|
|
|
|
*/
|
|
|
|
|
public function getFoo();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T as Foo
|
|
|
|
|
*/
|
|
|
|
|
class FooGetter implements IFooGetter {
|
|
|
|
|
/** @var T */
|
|
|
|
|
private $t;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param T $t
|
|
|
|
|
*/
|
|
|
|
|
public function __construct(Foo $t)
|
|
|
|
|
{
|
|
|
|
|
$this->t = $t;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
public function getFoo()
|
|
|
|
|
{
|
|
|
|
|
return $this->t;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function passFoo(Foo $f) : Foo {
|
|
|
|
|
return (new FooGetter($f))->getFoo();
|
|
|
|
|
}',
|
|
|
|
|
],
|
|
|
|
|
'templateFunctionVar' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace A\B;
|
|
|
|
|
|
|
|
|
|
class C {
|
|
|
|
|
public function bar() : void {}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
interface D {}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T as C
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
function foo($some_t) : C {
|
|
|
|
|
/** @var T */
|
|
|
|
|
$a = $some_t;
|
|
|
|
|
$a->bar();
|
|
|
|
|
|
|
|
|
|
/** @var T&D */
|
|
|
|
|
$b = $some_t;
|
|
|
|
|
$b->bar();
|
|
|
|
|
|
|
|
|
|
/** @var D&T */
|
|
|
|
|
$b = $some_t;
|
|
|
|
|
$b->bar();
|
|
|
|
|
|
|
|
|
|
return $a;
|
|
|
|
|
}',
|
|
|
|
|
'assertions' => [],
|
|
|
|
|
'error_levels' => ['MixedAssignment', 'MissingParamType'],
|
|
|
|
|
],
|
2019-01-02 12:46:10 +01:00
|
|
|
|
'returnClassString' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @param T::class $s
|
|
|
|
|
* @return T::class
|
|
|
|
|
*/
|
|
|
|
|
function foo(string $s) : string {
|
|
|
|
|
return $s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param A::class $s
|
|
|
|
|
*/
|
|
|
|
|
function bar(string $s) : void {
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class A {}
|
|
|
|
|
|
|
|
|
|
bar(foo(A::class));'
|
|
|
|
|
],
|
2019-01-05 06:15:53 +01:00
|
|
|
|
'callStaticMethodOnTemplatedClassName' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @param class-string $class
|
|
|
|
|
* @template-typeof T $class
|
|
|
|
|
*/
|
|
|
|
|
function foo(string $class, array $args) : void {
|
|
|
|
|
$class::bar($args);
|
|
|
|
|
}',
|
|
|
|
|
'assertions' => [],
|
|
|
|
|
'error_levels' => ['MixedMethodCall'],
|
|
|
|
|
],
|
2019-01-05 16:32:39 +01:00
|
|
|
|
'upcastIterableToTraversable' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template T as iterable
|
|
|
|
|
* @param T::class $class
|
|
|
|
|
*/
|
|
|
|
|
function foo(string $class) : void {
|
|
|
|
|
$a = new $class();
|
|
|
|
|
|
|
|
|
|
foreach ($a as $b) {}
|
|
|
|
|
}',
|
|
|
|
|
'assertions' => [],
|
|
|
|
|
'error_levels' => ['MixedAssignment'],
|
|
|
|
|
],
|
|
|
|
|
'upcastGenericIterableToGenericTraversable' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template T as iterable<int>
|
|
|
|
|
* @param T::class $class
|
|
|
|
|
*/
|
|
|
|
|
function foo(string $class) : void {
|
|
|
|
|
$a = new $class();
|
|
|
|
|
|
|
|
|
|
foreach ($a as $b) {}
|
|
|
|
|
}',
|
|
|
|
|
'assertions' => [],
|
|
|
|
|
'error_levels' => [],
|
|
|
|
|
],
|
2019-01-06 18:16:09 +01:00
|
|
|
|
'bindFirstTemplatedClosureParameter' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
*
|
|
|
|
|
* @param Closure(T):void $t1
|
|
|
|
|
* @param T $t2
|
|
|
|
|
*/
|
|
|
|
|
function apply(Closure $t1, $t2) : void
|
|
|
|
|
{
|
|
|
|
|
$t1($t2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
apply(function(int $_i) : void {}, 5);
|
|
|
|
|
apply(function(string $_i) : void {}, "hello");
|
|
|
|
|
apply(function(stdClass $_i) : void {}, new stdClass);
|
|
|
|
|
|
|
|
|
|
class A {}
|
|
|
|
|
class AChild extends A {}
|
|
|
|
|
|
|
|
|
|
apply(function(A $_i) : void {}, new AChild());',
|
|
|
|
|
],
|
2017-04-25 05:45:02 +02:00
|
|
|
|
];
|
2017-02-10 06:14:44 +01:00
|
|
|
|
}
|
|
|
|
|
|
2017-02-10 02:35:17 +01:00
|
|
|
|
/**
|
2017-04-25 05:45:02 +02:00
|
|
|
|
* @return array
|
2017-02-10 02:35:17 +01:00
|
|
|
|
*/
|
2018-11-06 03:57:36 +01:00
|
|
|
|
public function providerInvalidCodeParse()
|
2017-02-10 02:35:17 +01:00
|
|
|
|
{
|
2017-04-25 05:45:02 +02:00
|
|
|
|
return [
|
|
|
|
|
'invalidTemplatedType' => [
|
|
|
|
|
'<?php
|
2017-07-25 22:11:02 +02:00
|
|
|
|
namespace FooFoo;
|
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @param T $x
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
function foo($x) {
|
|
|
|
|
return $x;
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
|
function bar(string $a): void { }
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
bar(foo(4));',
|
2017-05-27 02:05:57 +02:00
|
|
|
|
'error_message' => 'InvalidScalarArgument',
|
2017-04-25 05:45:02 +02:00
|
|
|
|
],
|
|
|
|
|
'invalidTemplatedStaticMethodType' => [
|
|
|
|
|
'<?php
|
2017-07-25 22:11:02 +02:00
|
|
|
|
namespace FooFoo;
|
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
class A {
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @param T $x
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
public static function foo($x) {
|
|
|
|
|
return $x;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
|
function bar(string $a): void { }
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
bar(A::foo(4));',
|
2017-05-27 02:05:57 +02:00
|
|
|
|
'error_message' => 'InvalidScalarArgument',
|
2017-04-25 05:45:02 +02:00
|
|
|
|
],
|
|
|
|
|
'invalidTemplatedInstanceMethodType' => [
|
|
|
|
|
'<?php
|
2017-07-25 22:11:02 +02:00
|
|
|
|
namespace FooFoo;
|
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
class A {
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @param T $x
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
public function foo($x) {
|
|
|
|
|
return $x;
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2018-01-11 21:50:45 +01:00
|
|
|
|
function bar(string $a): void { }
|
2017-06-29 16:22:49 +02:00
|
|
|
|
|
2017-04-25 05:45:02 +02:00
|
|
|
|
bar((new A())->foo(4));',
|
2017-05-27 02:05:57 +02:00
|
|
|
|
'error_message' => 'InvalidScalarArgument',
|
|
|
|
|
],
|
2018-05-28 23:26:43 +02:00
|
|
|
|
'replaceChildTypeNoHint' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template TKey
|
|
|
|
|
* @template TValue
|
|
|
|
|
* @param Traversable<TKey, TValue> $t
|
|
|
|
|
* @return array<TKey, TValue>
|
|
|
|
|
*/
|
|
|
|
|
function f(Traversable $t): array {
|
|
|
|
|
$ret = [];
|
|
|
|
|
foreach ($t as $k => $v) $ret[$k] = $v;
|
|
|
|
|
return $ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function g():Generator { yield new stdClass; }
|
|
|
|
|
|
|
|
|
|
takesArrayOfStdClass(f(g()));
|
|
|
|
|
|
|
|
|
|
/** @param array<stdClass> $p */
|
|
|
|
|
function takesArrayOfStdClass(array $p): void {}',
|
|
|
|
|
'error_message' => 'MixedTypeCoercion',
|
|
|
|
|
],
|
2018-06-09 16:14:18 +02:00
|
|
|
|
'restrictTemplateInput' => [
|
|
|
|
|
'<?php
|
2019-01-05 06:15:53 +01:00
|
|
|
|
/** @template T as object */
|
2018-06-09 16:14:18 +02:00
|
|
|
|
class Foo
|
|
|
|
|
{
|
|
|
|
|
/**
|
|
|
|
|
* @psalm-var class-string
|
|
|
|
|
*/
|
|
|
|
|
private $type;
|
|
|
|
|
|
|
|
|
|
/** @var array<T> */
|
|
|
|
|
private $items;
|
|
|
|
|
|
|
|
|
|
/**
|
2019-01-05 06:15:53 +01:00
|
|
|
|
* @param T::class $type
|
2018-06-09 16:14:18 +02:00
|
|
|
|
*/
|
|
|
|
|
public function __construct(string $type)
|
|
|
|
|
{
|
|
|
|
|
if (!in_array($type, [A::class, B::class], true)) {
|
|
|
|
|
throw new \InvalidArgumentException;
|
|
|
|
|
}
|
|
|
|
|
$this->type = $type;
|
|
|
|
|
$this->items = [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** @param T $item */
|
|
|
|
|
public function add($item): void
|
|
|
|
|
{
|
|
|
|
|
$this->items[] = $item;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class A {}
|
|
|
|
|
class B {}
|
|
|
|
|
|
|
|
|
|
|
2018-12-13 06:09:01 +01:00
|
|
|
|
$foo = new Foo(A::class);
|
|
|
|
|
$foo->add(new B);',
|
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
|
],
|
|
|
|
|
'restrictTemplateInputWithTClassBadInput' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/** @template T */
|
|
|
|
|
class Foo
|
|
|
|
|
{
|
|
|
|
|
/**
|
|
|
|
|
* @psalm-var class-string
|
|
|
|
|
*/
|
|
|
|
|
private $type;
|
|
|
|
|
|
|
|
|
|
/** @var array<T> */
|
|
|
|
|
private $items;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @param T::class $type
|
|
|
|
|
*/
|
|
|
|
|
public function __construct(string $type)
|
|
|
|
|
{
|
|
|
|
|
if (!in_array($type, [A::class, B::class], true)) {
|
|
|
|
|
throw new \InvalidArgumentException;
|
|
|
|
|
}
|
|
|
|
|
$this->type = $type;
|
|
|
|
|
$this->items = [];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/** @param T $item */
|
|
|
|
|
public function add($item): void
|
|
|
|
|
{
|
|
|
|
|
$this->items[] = $item;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class A {}
|
|
|
|
|
class B {}
|
|
|
|
|
|
2018-06-09 16:14:18 +02:00
|
|
|
|
$foo = new Foo(A::class);
|
|
|
|
|
$foo->add(new B);',
|
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
|
],
|
2018-12-08 20:10:06 +01:00
|
|
|
|
'templatedClosureProperty' => [
|
|
|
|
|
'<?php
|
|
|
|
|
final class State
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
interface Foo
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
function type(string ...$_p): void {}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
*/
|
|
|
|
|
final class AlmostFooMap
|
|
|
|
|
{
|
|
|
|
|
/**
|
|
|
|
|
* @param callable(State):(T&Foo) $closure
|
|
|
|
|
*/
|
|
|
|
|
public function __construct(callable $closure)
|
|
|
|
|
{
|
|
|
|
|
type($closure);
|
|
|
|
|
}
|
|
|
|
|
}',
|
2019-01-05 06:15:53 +01:00
|
|
|
|
'error_message' => 'InvalidArgument - src/somefile.php:20 - Argument 1 of type expects string, callable(State):(T as mixed)&Foo provided',
|
2018-12-17 21:49:59 +01:00
|
|
|
|
],
|
|
|
|
|
'classTemplateAsIncorrectClass' => [
|
|
|
|
|
'<?php
|
|
|
|
|
class Foo {}
|
|
|
|
|
class NotFoo {}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T as Foo
|
|
|
|
|
* @param T $x
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
function bar($x) {
|
|
|
|
|
return $x;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bar(new NotFoo());',
|
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
|
],
|
|
|
|
|
'classTemplateAsIncorrectInterface' => [
|
|
|
|
|
'<?php
|
|
|
|
|
interface Foo {}
|
|
|
|
|
interface NotFoo {}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T as Foo
|
|
|
|
|
* @param T $x
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
function bar($x) {
|
|
|
|
|
return $x;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function takesNotFoo(NotFoo $f) : void {
|
|
|
|
|
bar($f);
|
|
|
|
|
}',
|
|
|
|
|
'error_message' => 'InvalidArgument',
|
|
|
|
|
],
|
2018-12-18 05:29:27 +01:00
|
|
|
|
'templateFunctionMethodCallWithoutMethod' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace A\B;
|
|
|
|
|
|
|
|
|
|
class C {}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T as C
|
|
|
|
|
* @param T $some_t
|
|
|
|
|
*/
|
|
|
|
|
function foo($some_t) : void {
|
|
|
|
|
$some_t->bar();
|
|
|
|
|
}',
|
|
|
|
|
'error_message' => 'PossiblyUndefinedMethod',
|
|
|
|
|
],
|
|
|
|
|
'templateFunctionMethodCallWithoutAsType' => [
|
|
|
|
|
'<?php
|
|
|
|
|
namespace A\B;
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
* @param T $some_t
|
|
|
|
|
* @return T
|
|
|
|
|
*/
|
|
|
|
|
function foo($some_t) : void {
|
|
|
|
|
$some_t->bar();
|
|
|
|
|
}',
|
|
|
|
|
'error_message' => 'MixedMethodCall',
|
|
|
|
|
],
|
2019-01-05 16:32:39 +01:00
|
|
|
|
'forbidLossOfInformationWhenCoercing' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template T as iterable<int>
|
|
|
|
|
* @param T::class $class
|
|
|
|
|
*/
|
|
|
|
|
function foo(string $class) : void {}
|
|
|
|
|
|
|
|
|
|
function bar(Traversable $t) : void {
|
|
|
|
|
foo(get_class($t));
|
|
|
|
|
}',
|
|
|
|
|
'error_message' => 'MixedTypeCoercion',
|
|
|
|
|
],
|
2019-01-06 18:16:09 +01:00
|
|
|
|
'bindFirstTemplatedClosureParameter' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
*
|
|
|
|
|
* @param Closure(T):void $t1
|
|
|
|
|
* @param T $t2
|
|
|
|
|
*/
|
|
|
|
|
function apply(Closure $t1, $t2) : void
|
|
|
|
|
{
|
|
|
|
|
$t1($t2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
apply(function(int $_i) : void {}, "hello");',
|
|
|
|
|
'error_message' => 'InvalidScalarArgument',
|
|
|
|
|
],
|
|
|
|
|
'bindFirstTemplatedClosureParameterTypeCoercion' => [
|
|
|
|
|
'<?php
|
|
|
|
|
/**
|
|
|
|
|
* @template T
|
|
|
|
|
*
|
|
|
|
|
* @param Closure(T):void $t1
|
|
|
|
|
* @param T $t2
|
|
|
|
|
*/
|
|
|
|
|
function apply(Closure $t1, $t2) : void
|
|
|
|
|
{
|
|
|
|
|
$t1($t2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
class A {}
|
|
|
|
|
class AChild extends A {}
|
|
|
|
|
|
|
|
|
|
apply(function(AChild $_i) : void {}, new A());',
|
|
|
|
|
'error_message' => 'TypeCoercion',
|
|
|
|
|
],
|
2017-04-25 05:45:02 +02:00
|
|
|
|
];
|
2017-02-10 04:57:23 +01:00
|
|
|
|
}
|
2017-02-10 02:35:17 +01:00
|
|
|
|
}
|