2016-10-18 22:02:38 -04:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
|
|
|
use PhpParser\ParserFactory;
|
|
|
|
use PHPUnit_Framework_TestCase;
|
2016-11-02 02:29:00 -04:00
|
|
|
use Psalm\Checker\FileChecker;
|
|
|
|
use Psalm\Config;
|
2016-10-18 22:02:38 -04:00
|
|
|
use Psalm\Context;
|
|
|
|
|
|
|
|
class Php70Test extends PHPUnit_Framework_TestCase
|
|
|
|
{
|
2016-12-14 12:55:23 -05:00
|
|
|
/** @var \PhpParser\Parser */
|
2016-11-02 02:29:00 -04:00
|
|
|
protected static $parser;
|
2016-10-18 22:02:38 -04:00
|
|
|
|
2017-01-02 15:31:18 -05:00
|
|
|
/** @var \Psalm\Checker\ProjectChecker */
|
|
|
|
protected $project_checker;
|
|
|
|
|
2016-10-18 22:02:38 -04:00
|
|
|
public static function setUpBeforeClass()
|
|
|
|
{
|
2016-11-02 02:29:00 -04:00
|
|
|
self::$parser = (new ParserFactory)->create(ParserFactory::PREFER_PHP7);
|
2016-10-18 22:02:38 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function setUp()
|
|
|
|
{
|
2016-12-16 18:56:23 -05:00
|
|
|
$config = new TestConfig();
|
2016-11-02 02:29:00 -04:00
|
|
|
FileChecker::clearCache();
|
2017-01-02 15:31:18 -05:00
|
|
|
$this->project_checker = new \Psalm\Checker\ProjectChecker();
|
2016-10-18 22:02:38 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testFunctionTypeHints()
|
|
|
|
{
|
2016-11-02 02:29:00 -04:00
|
|
|
$stmts = self::$parser->parse('<?php
|
2016-10-18 22:02:38 -04:00
|
|
|
function indexof(string $haystack, string $needle) : int
|
|
|
|
{
|
|
|
|
$pos = strpos($haystack, $needle);
|
|
|
|
|
|
|
|
if ($pos === false) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
$a = indexof("arr", "a");
|
|
|
|
');
|
|
|
|
|
2017-01-02 15:31:18 -05:00
|
|
|
$file_checker = new FileChecker('somefile.php', $this->project_checker, $stmts);
|
2016-10-18 22:02:38 -04:00
|
|
|
$context = new Context('somefile.php');
|
2017-01-07 15:09:47 -05:00
|
|
|
$file_checker->visitAndAnalyzeMethods($context);
|
2016-10-18 22:02:38 -04:00
|
|
|
$this->assertEquals('int', (string) $context->vars_in_scope['$a']);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testMethodTypeHints()
|
|
|
|
{
|
2016-11-02 02:29:00 -04:00
|
|
|
$stmts = self::$parser->parse('<?php
|
2016-10-18 22:02:38 -04:00
|
|
|
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");
|
|
|
|
');
|
|
|
|
|
2017-01-02 15:31:18 -05:00
|
|
|
$file_checker = new FileChecker('somefile.php', $this->project_checker, $stmts);
|
2016-10-18 22:02:38 -04:00
|
|
|
$context = new Context('somefile.php');
|
2017-01-07 15:09:47 -05:00
|
|
|
$file_checker->visitAndAnalyzeMethods($context);
|
2016-10-18 22:02:38 -04:00
|
|
|
$this->assertEquals('int', (string) $context->vars_in_scope['$a']);
|
|
|
|
}
|
2016-10-20 00:47:10 -04:00
|
|
|
|
|
|
|
public function testNullCoalesce()
|
|
|
|
{
|
2016-12-29 20:07:42 -05:00
|
|
|
Config::getInstance()->setCustomErrorLevel('MixedAssignment', Config::REPORT_SUPPRESS);
|
2016-12-16 18:56:23 -05:00
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
$stmts = self::$parser->parse('<?php
|
2016-10-20 00:47:10 -04:00
|
|
|
$a = $_GET["bar"] ?? "nobody";
|
|
|
|
');
|
|
|
|
|
2017-01-02 15:31:18 -05:00
|
|
|
$file_checker = new FileChecker('somefile.php', $this->project_checker, $stmts);
|
2016-10-20 00:47:10 -04:00
|
|
|
$context = new Context('somefile.php');
|
2017-01-07 15:09:47 -05:00
|
|
|
$file_checker->visitAndAnalyzeMethods($context);
|
2016-10-20 00:47:10 -04:00
|
|
|
$this->assertEquals('mixed', (string) $context->vars_in_scope['$a']);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testSpaceship()
|
|
|
|
{
|
2016-11-02 02:29:00 -04:00
|
|
|
$stmts = self::$parser->parse('<?php
|
2016-10-20 00:47:10 -04:00
|
|
|
$a = 1 <=> 1;
|
|
|
|
');
|
|
|
|
|
2017-01-02 15:31:18 -05:00
|
|
|
$file_checker = new FileChecker('somefile.php', $this->project_checker, $stmts);
|
2016-10-20 00:47:10 -04:00
|
|
|
$context = new Context('somefile.php');
|
2017-01-07 15:09:47 -05:00
|
|
|
$file_checker->visitAndAnalyzeMethods($context);
|
2016-10-20 00:47:10 -04:00
|
|
|
$this->assertEquals('int', (string) $context->vars_in_scope['$a']);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testDefineArray()
|
|
|
|
{
|
2016-11-02 02:29:00 -04:00
|
|
|
$stmts = self::$parser->parse('<?php
|
2016-10-20 00:47:10 -04:00
|
|
|
define("ANIMALS", [
|
|
|
|
"dog",
|
|
|
|
"cat",
|
|
|
|
"bird"
|
|
|
|
]);
|
|
|
|
|
|
|
|
$a = ANIMALS[1];
|
|
|
|
');
|
|
|
|
|
2017-01-02 15:31:18 -05:00
|
|
|
$file_checker = new FileChecker('somefile.php', $this->project_checker, $stmts);
|
2016-10-20 00:47:10 -04:00
|
|
|
$context = new Context('somefile.php');
|
2017-01-07 15:09:47 -05:00
|
|
|
$file_checker->visitAndAnalyzeMethods($context);
|
2016-10-20 00:47:10 -04:00
|
|
|
$this->assertEquals('string', (string) $context->vars_in_scope['$a']);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testAnonymousClass()
|
|
|
|
{
|
2016-11-02 02:29:00 -04:00
|
|
|
$stmts = self::$parser->parse('<?php
|
2016-10-20 00:47:10 -04:00
|
|
|
interface Logger {
|
2016-11-11 17:13:24 -05:00
|
|
|
/** @return void */
|
2016-10-20 00:47:10 -04:00
|
|
|
public function log(string $msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
class Application {
|
2016-11-01 14:32:19 -04:00
|
|
|
/** @var Logger|null */
|
2016-10-20 00:47:10 -04:00
|
|
|
private $logger;
|
|
|
|
|
2016-11-11 17:13:24 -05:00
|
|
|
/** @return void */
|
2016-10-20 00:47:10 -04:00
|
|
|
public function setLogger(Logger $logger) {
|
|
|
|
$this->logger = $logger;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$app = new Application;
|
|
|
|
$app->setLogger(new class implements Logger {
|
|
|
|
public function log(string $msg) {
|
|
|
|
echo $msg;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
');
|
|
|
|
|
2017-01-02 15:31:18 -05:00
|
|
|
$file_checker = new FileChecker('somefile.php', $this->project_checker, $stmts);
|
2016-10-20 00:47:10 -04:00
|
|
|
$context = new Context('somefile.php');
|
2017-01-07 15:09:47 -05:00
|
|
|
$file_checker->visitAndAnalyzeMethods($context);
|
2016-10-20 00:47:10 -04:00
|
|
|
}
|
|
|
|
|
2017-01-13 10:44:04 -05:00
|
|
|
public function testAnonymousClassFunctionReturnType()
|
|
|
|
{
|
|
|
|
$stmts = self::$parser->parse('<?php
|
|
|
|
$class = new class {
|
|
|
|
public function f() : int {
|
|
|
|
return 42;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
function g(int $i) : int {
|
|
|
|
return $i;
|
|
|
|
}
|
|
|
|
|
|
|
|
$x = g($class->f());
|
|
|
|
');
|
|
|
|
|
|
|
|
$file_checker = new FileChecker('somefile.php', $this->project_checker, $stmts);
|
|
|
|
$context = new Context('somefile.php');
|
|
|
|
$file_checker->visitAndAnalyzeMethods($context);
|
|
|
|
}
|
|
|
|
|
2016-10-20 14:26:03 -04:00
|
|
|
public function testGeneratorWithReturn()
|
|
|
|
{
|
2016-11-02 02:29:00 -04:00
|
|
|
$stmts = self::$parser->parse('<?php
|
2016-10-20 14:26:03 -04:00
|
|
|
/**
|
2016-10-20 18:05:28 -04:00
|
|
|
* @return Generator<int,int>
|
|
|
|
* @psalm-generator-return string
|
2016-10-20 14:26:03 -04:00
|
|
|
*/
|
2016-12-30 13:09:00 -05:00
|
|
|
function fooFoo(int $i) : Generator {
|
2016-10-20 14:26:03 -04:00
|
|
|
if ($i === 1) {
|
|
|
|
return "bash";
|
|
|
|
}
|
|
|
|
|
|
|
|
yield 1;
|
|
|
|
}
|
|
|
|
');
|
|
|
|
|
2017-01-02 15:31:18 -05:00
|
|
|
$file_checker = new FileChecker('somefile.php', $this->project_checker, $stmts);
|
2016-10-20 14:26:03 -04:00
|
|
|
$context = new Context('somefile.php');
|
2017-01-07 15:09:47 -05:00
|
|
|
$file_checker->visitAndAnalyzeMethods($context);
|
2016-10-20 00:47:10 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testGeneratorDelegation()
|
|
|
|
{
|
2016-12-29 20:07:42 -05:00
|
|
|
Config::getInstance()->setCustomErrorLevel('MixedAssignment', Config::REPORT_SUPPRESS);
|
2016-12-16 18:56:23 -05:00
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
$stmts = self::$parser->parse('<?php
|
2016-10-20 11:31:19 -04:00
|
|
|
/**
|
2016-10-20 14:26:03 -04:00
|
|
|
* @return Generator<int,int>
|
2016-10-20 18:05:28 -04:00
|
|
|
* @psalm-generator-return int
|
2016-10-20 11:31:19 -04:00
|
|
|
*/
|
2016-10-20 14:26:03 -04:00
|
|
|
function count_to_ten() : Generator {
|
2016-10-20 00:47:10 -04:00
|
|
|
yield 1;
|
|
|
|
yield 2;
|
2016-10-20 11:31:19 -04:00
|
|
|
yield from [3, 4];
|
|
|
|
yield from new ArrayIterator([5, 6]);
|
|
|
|
yield from seven_eight();
|
|
|
|
return yield from nine_ten();
|
2016-10-20 00:47:10 -04:00
|
|
|
}
|
|
|
|
|
2016-10-20 11:31:19 -04:00
|
|
|
/**
|
2016-10-20 14:26:03 -04:00
|
|
|
* @return Generator<int,int>
|
2016-10-20 11:31:19 -04:00
|
|
|
*/
|
2016-10-20 14:26:03 -04:00
|
|
|
function seven_eight() : Generator {
|
2016-10-20 11:31:19 -04:00
|
|
|
yield 7;
|
|
|
|
yield from eight();
|
2016-10-20 00:47:10 -04:00
|
|
|
}
|
|
|
|
|
2016-10-20 11:31:19 -04:00
|
|
|
/**
|
2016-10-20 14:26:03 -04:00
|
|
|
* @return Generator<int,int>
|
2016-10-20 11:31:19 -04:00
|
|
|
*/
|
2016-10-20 14:26:03 -04:00
|
|
|
function eight() : Generator {
|
2016-10-20 11:31:19 -04:00
|
|
|
yield 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-10-20 14:26:03 -04:00
|
|
|
* @return Generator<int,int>
|
2016-10-20 18:05:28 -04:00
|
|
|
* @psalm-generator-return int
|
2016-10-20 11:31:19 -04:00
|
|
|
*/
|
2016-10-20 14:26:03 -04:00
|
|
|
function nine_ten() : Generator {
|
2016-10-20 11:31:19 -04:00
|
|
|
yield 9;
|
|
|
|
return 10;
|
2016-10-20 00:47:10 -04:00
|
|
|
}
|
2016-10-20 11:31:19 -04:00
|
|
|
|
|
|
|
$gen = count_to_ten();
|
|
|
|
foreach ($gen as $num) {
|
|
|
|
echo "$num ";
|
|
|
|
}
|
|
|
|
$gen2 = $gen->getReturn();
|
2016-10-20 00:47:10 -04:00
|
|
|
');
|
|
|
|
|
2017-01-02 15:31:18 -05:00
|
|
|
$file_checker = new FileChecker('somefile.php', $this->project_checker, $stmts);
|
2016-10-20 00:47:10 -04:00
|
|
|
$context = new Context('somefile.php');
|
2017-01-07 15:09:47 -05:00
|
|
|
$file_checker->visitAndAnalyzeMethods($context);
|
2016-11-13 11:24:25 -05:00
|
|
|
$this->assertEquals('Generator<int, int>', (string) $context->vars_in_scope['$gen']);
|
2016-10-20 14:26:03 -04:00
|
|
|
$this->assertEquals('mixed', (string) $context->vars_in_scope['$gen2']);
|
2016-10-20 00:47:10 -04:00
|
|
|
}
|
2016-11-20 22:40:19 -05:00
|
|
|
|
|
|
|
public function testMultipleUse()
|
|
|
|
{
|
|
|
|
$stmts = self::$parser->parse('<?php
|
|
|
|
namespace Name\Space {
|
|
|
|
class A {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
class B {
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace Noom\Spice {
|
|
|
|
use Name\Space\{
|
|
|
|
A,
|
|
|
|
B
|
|
|
|
};
|
|
|
|
|
|
|
|
new A();
|
|
|
|
new B();
|
|
|
|
}
|
|
|
|
');
|
|
|
|
|
2017-01-02 15:31:18 -05:00
|
|
|
$file_checker = new FileChecker('somefile.php', $this->project_checker, $stmts);
|
2016-11-20 22:40:19 -05:00
|
|
|
$context = new Context('somefile.php');
|
2017-01-07 15:09:47 -05:00
|
|
|
$file_checker->visitAndAnalyzeMethods($context);
|
2016-11-20 22:40:19 -05:00
|
|
|
}
|
2016-10-18 22:02:38 -04:00
|
|
|
}
|