2016-10-18 22:02:38 -04:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class Php70Test extends TestCase
|
2016-10-18 22:02:38 -04:00
|
|
|
{
|
2018-11-05 21:57:36 -05:00
|
|
|
use Traits\InvalidCodeAnalysisTestTrait;
|
|
|
|
use Traits\ValidCodeAnalysisTestTrait;
|
2016-10-20 00:47:10 -04:00
|
|
|
|
2017-01-13 14:07:23 -05:00
|
|
|
/**
|
2019-03-01 22:55:20 +02:00
|
|
|
* @return iterable<string,array{string,assertions?:array<string,string>,error_levels?:string[]}>
|
2017-01-13 14:07:23 -05:00
|
|
|
*/
|
2018-11-05 21:57:36 -05:00
|
|
|
public function providerValidCodeParse()
|
2016-10-20 00:47:10 -04:00
|
|
|
{
|
2017-04-24 23:45:02 -04:00
|
|
|
return [
|
|
|
|
'functionTypeHints' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function indexof(string $haystack, string $needle): int
|
2017-04-24 23:45:02 -04:00
|
|
|
{
|
|
|
|
$pos = strpos($haystack, $needle);
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
if ($pos === false) {
|
|
|
|
return -1;
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
return $pos;
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
$a = indexof("arr", "a");',
|
|
|
|
'assertions' => [
|
2017-06-29 10:22:49 -04:00
|
|
|
'$a' => 'int',
|
2017-05-26 20:05:57 -04:00
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'methodTypeHints' => [
|
|
|
|
'<?php
|
|
|
|
class Foo {
|
2018-01-11 15:50:45 -05:00
|
|
|
public static function indexof(string $haystack, string $needle): int
|
2017-04-24 23:45:02 -04:00
|
|
|
{
|
|
|
|
$pos = strpos($haystack, $needle);
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
if ($pos === false) {
|
|
|
|
return -1;
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
return $pos;
|
|
|
|
}
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
$a = Foo::indexof("arr", "a");',
|
|
|
|
'assertions' => [
|
2017-06-29 10:22:49 -04:00
|
|
|
'$a' => 'int',
|
2017-05-26 20:05:57 -04:00
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'nullCoalesce' => [
|
|
|
|
'<?php
|
2017-06-20 14:38:13 -04:00
|
|
|
$arr = ["hello", "goodbye"];
|
|
|
|
$a = $arr[rand(0, 10)] ?? null;',
|
2017-04-24 23:45:02 -04:00
|
|
|
'assertions' => [
|
2019-10-16 22:14:33 -07:00
|
|
|
'$a' => 'null|string',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
],
|
2017-09-13 14:35:42 -04:00
|
|
|
'nullCoalesceWithNullableOnLeft' => [
|
|
|
|
'<?php
|
|
|
|
/** @return ?string */
|
|
|
|
function foo() {
|
|
|
|
return rand(0, 10) > 5 ? "hello" : null;
|
|
|
|
}
|
|
|
|
$a = foo() ?? "goodbye";',
|
|
|
|
'assertions' => [
|
|
|
|
'$a' => 'string',
|
|
|
|
],
|
|
|
|
],
|
2017-06-20 14:38:58 -04:00
|
|
|
'nullCoalesceWithReference' => [
|
|
|
|
'<?php
|
2017-06-20 15:38:32 -04:00
|
|
|
$var = 0;
|
|
|
|
($a =& $var) ?? "hello";',
|
2017-06-20 14:38:58 -04:00
|
|
|
'assertions' => [
|
2019-08-27 20:19:10 -07:00
|
|
|
'$a' => 'int',
|
2017-06-20 14:38:58 -04:00
|
|
|
],
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'spaceship' => [
|
|
|
|
'<?php
|
|
|
|
$a = 1 <=> 1;',
|
|
|
|
'assertions' => [
|
2017-06-29 10:22:49 -04:00
|
|
|
'$a' => 'int',
|
2017-05-26 20:05:57 -04:00
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'defineArray' => [
|
|
|
|
'<?php
|
|
|
|
define("ANIMALS", [
|
|
|
|
"dog",
|
|
|
|
"cat",
|
|
|
|
"bird"
|
|
|
|
]);
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
$a = ANIMALS[1];',
|
|
|
|
'assertions' => [
|
2017-06-29 10:22:49 -04:00
|
|
|
'$a' => 'string',
|
2017-05-26 20:05:57 -04:00
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'anonymousClassLogger' => [
|
|
|
|
'<?php
|
|
|
|
interface Logger {
|
|
|
|
/** @return void */
|
|
|
|
public function log(string $msg);
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class Application {
|
|
|
|
/** @var Logger|null */
|
|
|
|
private $logger;
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
/** @return void */
|
|
|
|
public function setLogger(Logger $logger) {
|
|
|
|
$this->logger = $logger;
|
|
|
|
}
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
$app = new Application;
|
|
|
|
$app->setLogger(new class implements Logger {
|
2017-11-26 16:03:17 -05:00
|
|
|
/** @return void */
|
2017-04-24 23:45:02 -04:00
|
|
|
public function log(string $msg) {
|
|
|
|
echo $msg;
|
|
|
|
}
|
2017-05-26 20:05:57 -04:00
|
|
|
});',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'anonymousClassFunctionReturnType' => [
|
|
|
|
'<?php
|
|
|
|
$class = new class {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function f(): int {
|
2017-04-24 23:45:02 -04:00
|
|
|
return 42;
|
|
|
|
}
|
|
|
|
};
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2018-01-11 15:50:45 -05:00
|
|
|
function g(int $i): int {
|
2017-04-24 23:45:02 -04:00
|
|
|
return $i;
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-05-26 20:05:57 -04:00
|
|
|
$x = g($class->f());',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2017-08-14 15:46:01 -04:00
|
|
|
'anonymousClassStatement' => [
|
|
|
|
'<?php
|
|
|
|
new class {};',
|
|
|
|
],
|
2017-10-19 14:40:38 -04:00
|
|
|
'anonymousClassTwoFunctions' => [
|
|
|
|
'<?php
|
|
|
|
interface I {}
|
|
|
|
|
|
|
|
class A
|
|
|
|
{
|
|
|
|
/** @var ?I */
|
|
|
|
protected $i;
|
|
|
|
|
|
|
|
public function foo(): void
|
|
|
|
{
|
|
|
|
$this->i = new class implements I {};
|
|
|
|
}
|
|
|
|
|
2018-01-11 15:50:45 -05:00
|
|
|
public function foo2(): void {} // commenting this line out fixes
|
2017-10-19 14:40:38 -04:00
|
|
|
}',
|
|
|
|
],
|
2018-03-18 01:07:14 -04:00
|
|
|
'anonymousClassExtendsWithThis' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
public function foo() : void {}
|
|
|
|
}
|
|
|
|
$class = new class extends A {
|
|
|
|
public function f(): int {
|
|
|
|
$this->foo();
|
|
|
|
return 42;
|
|
|
|
}
|
|
|
|
};',
|
|
|
|
],
|
2017-08-14 15:46:01 -04:00
|
|
|
'returnAnonymousClass' => [
|
|
|
|
'<?php
|
|
|
|
/** @return object */
|
|
|
|
function getNewAnonymousClass() {
|
|
|
|
return new class {};
|
|
|
|
}',
|
|
|
|
],
|
|
|
|
'returnAnonymousClassInClass' => [
|
|
|
|
'<?php
|
|
|
|
class A {
|
|
|
|
/** @return object */
|
|
|
|
public function getNewAnonymousClass() {
|
|
|
|
return new class {};
|
|
|
|
}
|
|
|
|
}',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'generatorWithReturn' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @return Generator<int,int>
|
|
|
|
* @psalm-generator-return string
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function fooFoo(int $i): Generator {
|
2017-04-24 23:45:02 -04:00
|
|
|
if ($i === 1) {
|
|
|
|
return "bash";
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
yield 1;
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'generatorDelegation' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
2018-07-13 17:34:44 -04:00
|
|
|
* @return Generator<int, int, mixed, int>
|
2017-04-24 23:45:02 -04:00
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function count_to_ten(): Generator {
|
2017-04-24 23:45:02 -04:00
|
|
|
yield 1;
|
|
|
|
yield 2;
|
|
|
|
yield from [3, 4];
|
|
|
|
yield from new ArrayIterator([5, 6]);
|
|
|
|
yield from seven_eight();
|
|
|
|
return yield from nine_ten();
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
/**
|
2018-07-13 17:34:44 -04:00
|
|
|
* @return Generator<int, int>
|
2017-04-24 23:45:02 -04:00
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function seven_eight(): Generator {
|
2017-04-24 23:45:02 -04:00
|
|
|
yield 7;
|
|
|
|
yield from eight();
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
/**
|
|
|
|
* @return Generator<int,int>
|
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function eight(): Generator {
|
2017-04-24 23:45:02 -04:00
|
|
|
yield 8;
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
/**
|
2018-07-13 17:34:44 -04:00
|
|
|
* @return Generator<int,int, mixed, int>
|
2017-04-24 23:45:02 -04:00
|
|
|
*/
|
2018-01-11 15:50:45 -05:00
|
|
|
function nine_ten(): Generator {
|
2017-04-24 23:45:02 -04:00
|
|
|
yield 9;
|
|
|
|
return 10;
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
$gen = count_to_ten();
|
|
|
|
foreach ($gen as $num) {
|
|
|
|
echo "$num ";
|
|
|
|
}
|
|
|
|
$gen2 = $gen->getReturn();',
|
|
|
|
'assertions' => [
|
2018-07-13 17:34:44 -04:00
|
|
|
'$gen' => 'Generator<int, int, mixed, int>',
|
|
|
|
'$gen2' => 'int',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_levels' => ['MixedAssignment'],
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
2019-02-09 15:48:15 -05:00
|
|
|
'yieldFromArray' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @return Generator<int, int, mixed, void>
|
|
|
|
*/
|
|
|
|
function Bar() : Generator {
|
|
|
|
yield from [1];
|
2019-03-23 14:27:54 -04:00
|
|
|
}',
|
2019-02-09 15:48:15 -05:00
|
|
|
],
|
2017-11-14 22:08:15 -05:00
|
|
|
'generatorWithNestedYield' => [
|
|
|
|
'<?php
|
2018-01-11 15:50:45 -05:00
|
|
|
function other_generator(): Generator {
|
2017-11-14 22:08:15 -05:00
|
|
|
yield "traffic";
|
|
|
|
return 1;
|
|
|
|
}
|
2018-01-11 15:50:45 -05:00
|
|
|
function foo(): Generator {
|
2017-11-14 22:29:15 -05:00
|
|
|
/** @var int */
|
2017-11-14 22:08:15 -05:00
|
|
|
$value = yield from other_generator();
|
|
|
|
var_export($value);
|
|
|
|
}',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
'multipleUse' => [
|
|
|
|
'<?php
|
|
|
|
namespace Name\Space {
|
|
|
|
class A {
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
class B {
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
}
|
|
|
|
}
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
namespace Noom\Spice {
|
|
|
|
use Name\Space\{
|
|
|
|
A,
|
|
|
|
B
|
|
|
|
};
|
2017-06-20 14:38:13 -04:00
|
|
|
|
2017-04-24 23:45:02 -04:00
|
|
|
new A();
|
|
|
|
new B();
|
2017-05-26 20:05:57 -04:00
|
|
|
}',
|
|
|
|
],
|
2018-02-07 22:29:32 -05:00
|
|
|
'generatorVoidReturn' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @return Generator
|
|
|
|
*/
|
|
|
|
function generator2() : Generator {
|
|
|
|
if (rand(0,1)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
yield 2;
|
|
|
|
}',
|
2018-02-07 23:33:31 -05:00
|
|
|
],
|
2018-07-13 17:34:44 -04:00
|
|
|
'returnType' => [
|
|
|
|
'<?php
|
|
|
|
function takesInt(int $i) : void {
|
|
|
|
echo $i;
|
|
|
|
}
|
|
|
|
|
|
|
|
function takesString(string $s) : void {
|
|
|
|
echo $s;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Generator<int, string, mixed, int>
|
|
|
|
*/
|
|
|
|
function other_generator() : Generator {
|
|
|
|
yield "traffic";
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return Generator<int, string>
|
|
|
|
*/
|
|
|
|
function foo() : Generator {
|
|
|
|
$a = yield from other_generator();
|
|
|
|
takesInt($a);
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (foo() as $s) {
|
|
|
|
takesString($s);
|
|
|
|
}',
|
|
|
|
],
|
2018-09-04 13:29:35 -04:00
|
|
|
'expectNonNullableTypeWithYield' => [
|
|
|
|
'<?php
|
|
|
|
function example() : Generator {
|
|
|
|
yield from [2];
|
|
|
|
return null;
|
|
|
|
}',
|
|
|
|
],
|
2019-01-31 17:58:53 -05:00
|
|
|
'yieldFromIterable' => [
|
|
|
|
'<?php
|
|
|
|
/**
|
|
|
|
* @param iterable<int, string> $s
|
|
|
|
* @return Generator<int, string>
|
|
|
|
*/
|
|
|
|
function foo(iterable $s) : Traversable {
|
|
|
|
yield from $s;
|
|
|
|
}',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
];
|
2017-01-13 10:44:04 -05:00
|
|
|
}
|
|
|
|
|
2017-01-13 14:07:23 -05:00
|
|
|
/**
|
2019-03-01 22:55:20 +02:00
|
|
|
* @return iterable<string,array{string,error_message:string,2?:string[],3?:bool,4?:string}>
|
2017-01-13 14:07:23 -05:00
|
|
|
*/
|
2018-11-05 21:57:36 -05:00
|
|
|
public function providerInvalidCodeParse()
|
2016-10-20 14:26:03 -04:00
|
|
|
{
|
2017-04-24 23:45:02 -04:00
|
|
|
return [
|
|
|
|
'anonymousClassWithBadStatement' => [
|
|
|
|
'<?php
|
|
|
|
$foo = new class {
|
|
|
|
public function a() {
|
|
|
|
new B();
|
|
|
|
}
|
|
|
|
};',
|
2017-05-26 20:05:57 -04:00
|
|
|
'error_message' => 'UndefinedClass',
|
2017-04-24 23:45:02 -04:00
|
|
|
],
|
|
|
|
'anonymousClassWithInvalidFunctionReturnType' => [
|
|
|
|
'<?php
|
|
|
|
$foo = new class {
|
2018-01-11 15:50:45 -05:00
|
|
|
public function a(): string {
|
2017-04-24 23:45:02 -04:00
|
|
|
return 5;
|
|
|
|
}
|
|
|
|
};',
|
2017-12-07 15:50:25 -05:00
|
|
|
'error_message' => 'InvalidReturnStatement',
|
2017-05-26 20:05:57 -04:00
|
|
|
],
|
2018-09-04 13:29:35 -04:00
|
|
|
'expectNonNullableTypeWithNullReturn' => [
|
|
|
|
'<?php
|
|
|
|
function example() : Generator {
|
|
|
|
yield from [2];
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
function example2() : Generator {
|
|
|
|
if (rand(0, 1)) {
|
|
|
|
return example();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}',
|
|
|
|
'error_message' => 'NullableReturnStatement',
|
|
|
|
],
|
2017-04-24 23:45:02 -04:00
|
|
|
];
|
2016-11-20 22:40:19 -05:00
|
|
|
}
|
2016-10-18 22:02:38 -04:00
|
|
|
}
|