2016-10-19 04:02:38 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
|
|
|
use PhpParser;
|
|
|
|
use PhpParser\ParserFactory;
|
|
|
|
use PHPUnit_Framework_TestCase;
|
|
|
|
use Psalm\Context;
|
|
|
|
use Psalm\Checker\TypeChecker;
|
|
|
|
use Psalm\Type;
|
|
|
|
|
|
|
|
class Php70Test extends PHPUnit_Framework_TestCase
|
|
|
|
{
|
|
|
|
protected static $_parser;
|
|
|
|
|
|
|
|
public static function setUpBeforeClass()
|
|
|
|
{
|
|
|
|
self::$_parser = (new ParserFactory)->create(ParserFactory::PREFER_PHP7);
|
|
|
|
|
|
|
|
$config = \Psalm\Config::getInstance();
|
|
|
|
$config->throw_exception = true;
|
|
|
|
$config->use_docblock_types = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setUp()
|
|
|
|
{
|
|
|
|
\Psalm\Checker\FileChecker::clearCache();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testFunctionTypeHints()
|
|
|
|
{
|
|
|
|
$stmts = self::$_parser->parse('<?php
|
|
|
|
function indexof(string $haystack, string $needle) : int
|
|
|
|
{
|
|
|
|
$pos = strpos($haystack, $needle);
|
|
|
|
|
|
|
|
if ($pos === false) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = indexof("arr", "a");
|
|
|
|
');
|
|
|
|
|
|
|
|
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
|
|
|
|
$context = new Context('somefile.php');
|
|
|
|
$file_checker->check(true, true, $context);
|
|
|
|
$this->assertEquals('int', (string) $context->vars_in_scope['$a']);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testMethodTypeHints()
|
|
|
|
{
|
|
|
|
$stmts = self::$_parser->parse('<?php
|
|
|
|
class Foo {
|
|
|
|
public static function indexof(string $haystack, string $needle) : int
|
|
|
|
{
|
|
|
|
$pos = strpos($haystack, $needle);
|
|
|
|
|
|
|
|
if ($pos === false) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $pos;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = Foo::indexof("arr", "a");
|
|
|
|
');
|
|
|
|
|
|
|
|
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
|
|
|
|
$context = new Context('somefile.php');
|
|
|
|
$file_checker->check(true, true, $context);
|
|
|
|
$this->assertEquals('int', (string) $context->vars_in_scope['$a']);
|
|
|
|
}
|
2016-10-20 06:47:10 +02:00
|
|
|
|
|
|
|
public function testNullCoalesce()
|
|
|
|
{
|
|
|
|
$stmts = self::$_parser->parse('<?php
|
|
|
|
$a = $_GET["bar"] ?? "nobody";
|
|
|
|
');
|
|
|
|
|
|
|
|
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
|
|
|
|
$context = new Context('somefile.php');
|
|
|
|
$file_checker->check(true, true, $context);
|
|
|
|
$this->assertEquals('mixed', (string) $context->vars_in_scope['$a']);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSpaceship()
|
|
|
|
{
|
|
|
|
$stmts = self::$_parser->parse('<?php
|
|
|
|
$a = 1 <=> 1;
|
|
|
|
');
|
|
|
|
|
|
|
|
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
|
|
|
|
$context = new Context('somefile.php');
|
|
|
|
$file_checker->check(true, true, $context);
|
|
|
|
$this->assertEquals('int', (string) $context->vars_in_scope['$a']);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testDefineArray()
|
|
|
|
{
|
|
|
|
$stmts = self::$_parser->parse('<?php
|
|
|
|
define("ANIMALS", [
|
|
|
|
"dog",
|
|
|
|
"cat",
|
|
|
|
"bird"
|
|
|
|
]);
|
|
|
|
|
|
|
|
$a = ANIMALS[1];
|
|
|
|
');
|
|
|
|
|
|
|
|
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
|
|
|
|
$context = new Context('somefile.php');
|
|
|
|
$file_checker->check(true, true, $context);
|
|
|
|
$this->assertEquals('string', (string) $context->vars_in_scope['$a']);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testAnonymousClass()
|
|
|
|
{
|
|
|
|
$stmts = self::$_parser->parse('<?php
|
|
|
|
interface Logger {
|
|
|
|
public function log(string $msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
class Application {
|
|
|
|
private $logger;
|
|
|
|
|
|
|
|
public function getLogger(): Logger {
|
|
|
|
return $this->logger;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setLogger(Logger $logger) {
|
|
|
|
$this->logger = $logger;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$app = new Application;
|
|
|
|
$app->setLogger(new class implements Logger {
|
|
|
|
public function log(string $msg) {
|
|
|
|
echo $msg;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
');
|
|
|
|
|
|
|
|
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
|
|
|
|
$context = new Context('somefile.php');
|
|
|
|
$file_checker->check(true, true, $context);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testClosureCall()
|
|
|
|
{
|
|
|
|
$stmts = self::$_parser->parse('<?php
|
|
|
|
class A {private $x = 1;}
|
|
|
|
|
|
|
|
// Pre PHP 7 code
|
|
|
|
$getXCB = function() {return $this->x;};
|
|
|
|
$getX = $getXCB->bindTo(new A, "A"); // intermediate closure
|
|
|
|
$a = $getX();
|
|
|
|
|
|
|
|
// PHP 7+ code
|
|
|
|
$getX = function() {return $this->x;};
|
|
|
|
$b = $getX->call(new A);
|
|
|
|
');
|
|
|
|
|
|
|
|
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
|
|
|
|
$context = new Context('somefile.php');
|
|
|
|
$file_checker->check(true, true, $context);
|
|
|
|
$this->assertEquals('mixed', (string) $context->vars_in_scope['$a']);
|
|
|
|
$this->assertEquals('mixed', (string) $context->vars_in_scope['$b']);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testGeneratorDelegation()
|
|
|
|
{
|
|
|
|
$stmts = self::$_parser->parse('<?php
|
|
|
|
function gen()
|
|
|
|
{
|
|
|
|
yield 1;
|
|
|
|
yield 2;
|
|
|
|
yield from gen2();
|
|
|
|
}
|
|
|
|
|
|
|
|
function gen2()
|
|
|
|
{
|
|
|
|
yield 3;
|
|
|
|
yield 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (gen() as $val)
|
|
|
|
{
|
|
|
|
echo $val, PHP_EOL;
|
|
|
|
}
|
|
|
|
');
|
|
|
|
|
|
|
|
$file_checker = new \Psalm\Checker\FileChecker('somefile.php', $stmts);
|
|
|
|
$context = new Context('somefile.php');
|
|
|
|
$file_checker->check(true, true, $context);
|
|
|
|
$this->assertEquals('mixed', (string) $context->vars_in_scope['$a']);
|
|
|
|
$this->assertEquals('mixed', (string) $context->vars_in_scope['$b']);
|
|
|
|
}
|
2016-10-19 04:02:38 +02:00
|
|
|
}
|