2018-10-17 11:03:32 -04:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2019-02-24 20:48:57 +02:00
|
|
|
use Psalm\Codebase;
|
2018-10-17 11:03:32 -04:00
|
|
|
use Psalm\Config;
|
2019-03-23 14:27:54 -04:00
|
|
|
use Psalm\Internal\Analyzer\FileAnalyzer;
|
2019-02-24 20:48:57 +02:00
|
|
|
use Psalm\Plugin\Hook\AfterCodebasePopulatedInterface;
|
2018-11-12 10:57:05 -05:00
|
|
|
use Psalm\Tests\Internal\Provider;
|
2018-10-17 11:03:32 -04:00
|
|
|
|
2019-03-23 14:27:54 -04:00
|
|
|
class ProjectCheckerTest extends TestCase
|
2018-10-17 11:03:32 -04:00
|
|
|
{
|
|
|
|
/** @var TestConfig */
|
|
|
|
protected static $config;
|
|
|
|
|
2018-11-05 21:57:36 -05:00
|
|
|
/** @var \Psalm\Internal\Analyzer\ProjectAnalyzer */
|
2018-11-11 12:01:14 -05:00
|
|
|
protected $project_analyzer;
|
2018-10-17 11:03:32 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2019-05-16 18:36:36 -04:00
|
|
|
public static function setUpBeforeClass() : void
|
2018-10-17 11:03:32 -04:00
|
|
|
{
|
|
|
|
self::$config = new TestConfig();
|
|
|
|
|
|
|
|
if (!defined('PSALM_VERSION')) {
|
|
|
|
define('PSALM_VERSION', '2.0.0');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!defined('PHP_PARSER_VERSION')) {
|
|
|
|
define('PHP_PARSER_VERSION', '4.0.0');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
2019-05-16 18:36:36 -04:00
|
|
|
public function setUp() : void
|
2018-10-17 11:03:32 -04:00
|
|
|
{
|
2018-11-05 21:57:36 -05:00
|
|
|
FileAnalyzer::clearCache();
|
2018-10-17 11:03:32 -04:00
|
|
|
$this->file_provider = new Provider\FakeFileProvider();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param Config $config
|
|
|
|
*
|
2018-11-05 21:57:36 -05:00
|
|
|
* @return \Psalm\Internal\Analyzer\ProjectAnalyzer
|
2018-10-17 11:03:32 -04:00
|
|
|
*/
|
2018-11-05 21:57:36 -05:00
|
|
|
private function getProjectAnalyzerWithConfig(Config $config)
|
2018-10-17 11:03:32 -04:00
|
|
|
{
|
2018-11-05 21:57:36 -05:00
|
|
|
return new \Psalm\Internal\Analyzer\ProjectAnalyzer(
|
2018-10-17 11:03:32 -04:00
|
|
|
$config,
|
2018-11-05 21:57:36 -05:00
|
|
|
new \Psalm\Internal\Provider\Providers(
|
2018-10-17 11:03:32 -04:00
|
|
|
$this->file_provider,
|
|
|
|
new Provider\ParserInstanceCacheProvider(),
|
|
|
|
new Provider\FileStorageInstanceCacheProvider(),
|
|
|
|
new Provider\ClassLikeStorageInstanceCacheProvider(),
|
|
|
|
new Provider\FakeFileReferenceCacheProvider()
|
2019-05-27 13:07:02 -04:00
|
|
|
)
|
2018-10-17 11:03:32 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCheck()
|
|
|
|
{
|
2018-11-11 12:01:14 -05:00
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
2018-10-17 11:03:32 -04:00
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-09 18:23:14 -04:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2018-10-17 11:03:32 -04:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
ob_start();
|
2019-05-09 18:23:14 -04:00
|
|
|
$this->project_analyzer->check('tests/fixtures/DummyProject');
|
2018-10-17 11:03:32 -04:00
|
|
|
$output = ob_get_clean();
|
|
|
|
|
|
|
|
$this->assertSame('Scanning files...' . "\n" . 'Analyzing files...' . "\n", $output);
|
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
2019-03-23 09:50:47 -04:00
|
|
|
$codebase = $this->project_analyzer->getCodebase();
|
|
|
|
|
|
|
|
$this->assertSame([0, 4], $codebase->analyzer->getTotalTypeCoverage($codebase));
|
|
|
|
|
2018-10-17 11:03:32 -04:00
|
|
|
$this->assertSame(
|
2019-03-23 09:50:47 -04:00
|
|
|
'Psalm was able to infer types for 100% of the codebase',
|
|
|
|
$codebase->analyzer->getTypeInferenceSummary(
|
|
|
|
$codebase
|
|
|
|
)
|
2018-10-17 11:03:32 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-02-24 20:48:57 +02:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testAfterCodebasePopulatedIsInvoked()
|
|
|
|
{
|
2019-03-23 14:27:54 -04:00
|
|
|
$hook = new class implements AfterCodebasePopulatedInterface {
|
2019-02-24 20:48:57 +02:00
|
|
|
/** @var bool */
|
|
|
|
public static $called = false;
|
2019-03-23 14:27:54 -04:00
|
|
|
|
2019-02-24 20:48:57 +02:00
|
|
|
/** @return void */
|
|
|
|
public static function afterCodebasePopulated(Codebase $codebase)
|
|
|
|
{
|
|
|
|
self::$called = true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-09 18:23:14 -04:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2019-02-24 20:48:57 +02:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-03-01 15:45:18 +02:00
|
|
|
$hook_class = get_class($hook);
|
|
|
|
|
|
|
|
/** @psalm-suppress TypeCoercion see vimeo/psalm#1397 */
|
|
|
|
$this->project_analyzer->getCodebase()->config->after_codebase_populated[] = $hook_class;
|
2019-02-24 20:48:57 +02:00
|
|
|
|
|
|
|
ob_start();
|
2019-05-09 18:23:14 -04:00
|
|
|
$this->project_analyzer->check('tests/fixtures/DummyProject');
|
2019-02-24 21:14:14 +02:00
|
|
|
ob_end_clean();
|
2019-02-24 20:48:57 +02:00
|
|
|
|
|
|
|
$this->assertTrue($hook::$called);
|
|
|
|
}
|
|
|
|
|
2018-10-17 11:03:32 -04:00
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCheckAfterNoChange()
|
|
|
|
{
|
2018-11-11 12:01:14 -05:00
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
2018-10-17 11:03:32 -04:00
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-09 18:23:14 -04:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2018-10-17 11:03:32 -04:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2018-11-11 12:01:14 -05:00
|
|
|
$this->project_analyzer->output_format = \Psalm\Internal\Analyzer\ProjectAnalyzer::TYPE_JSON;
|
2018-10-17 11:03:32 -04:00
|
|
|
|
2019-05-09 18:23:14 -04:00
|
|
|
$this->project_analyzer->check('tests/fixtures/DummyProject', true);
|
2018-11-11 12:01:14 -05:00
|
|
|
\Psalm\IssueBuffer::finish($this->project_analyzer, true, microtime(true));
|
2018-10-17 11:03:32 -04:00
|
|
|
|
|
|
|
$this->assertSame(
|
2019-03-23 09:50:47 -04:00
|
|
|
'Psalm was able to infer types for 100% of the codebase',
|
|
|
|
$this->project_analyzer->getCodebase()->analyzer->getTypeInferenceSummary(
|
|
|
|
$this->project_analyzer->getCodebase()
|
|
|
|
)
|
2018-10-17 11:03:32 -04:00
|
|
|
);
|
|
|
|
|
2018-11-11 12:01:14 -05:00
|
|
|
$this->project_analyzer->getCodebase()->reloadFiles($this->project_analyzer, []);
|
2018-10-17 11:03:32 -04:00
|
|
|
|
2019-05-09 18:23:14 -04:00
|
|
|
$this->project_analyzer->check('tests/fixtures/DummyProject', true);
|
2018-10-17 11:03:32 -04:00
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
|
|
|
$this->assertSame(
|
|
|
|
'No files analyzed',
|
2019-03-23 09:50:47 -04:00
|
|
|
$this->project_analyzer->getCodebase()->analyzer->getTypeInferenceSummary(
|
|
|
|
$this->project_analyzer->getCodebase()
|
|
|
|
)
|
2018-10-17 11:03:32 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCheckAfterFileChange()
|
|
|
|
{
|
2018-11-11 12:01:14 -05:00
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
2018-10-17 11:03:32 -04:00
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-09 18:23:14 -04:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2018-10-17 11:03:32 -04:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2018-11-11 12:01:14 -05:00
|
|
|
$this->project_analyzer->output_format = \Psalm\Internal\Analyzer\ProjectAnalyzer::TYPE_JSON;
|
2018-10-17 11:03:32 -04:00
|
|
|
|
2019-05-09 18:23:14 -04:00
|
|
|
$this->project_analyzer->check('tests/fixtures/DummyProject', true);
|
2018-11-11 12:01:14 -05:00
|
|
|
\Psalm\IssueBuffer::finish($this->project_analyzer, true, microtime(true));
|
2018-10-17 11:03:32 -04:00
|
|
|
|
|
|
|
$this->assertSame(
|
2019-03-23 09:50:47 -04:00
|
|
|
'Psalm was able to infer types for 100% of the codebase',
|
|
|
|
$this->project_analyzer->getCodebase()->analyzer->getTypeInferenceSummary(
|
|
|
|
$this->project_analyzer->getCodebase()
|
|
|
|
)
|
2018-10-17 11:03:32 -04:00
|
|
|
);
|
|
|
|
|
2019-05-09 18:23:14 -04:00
|
|
|
$bat_file_path = getcwd()
|
|
|
|
. DIRECTORY_SEPARATOR . 'tests'
|
|
|
|
. DIRECTORY_SEPARATOR . 'fixtures'
|
|
|
|
. DIRECTORY_SEPARATOR . 'DummyProject'
|
|
|
|
. DIRECTORY_SEPARATOR . 'Bat.php';
|
2018-10-17 11:03:32 -04:00
|
|
|
|
|
|
|
$bat_replacement_contents = '<?php
|
|
|
|
|
2019-02-08 17:41:03 -05:00
|
|
|
namespace Vimeo\Test\DummyProject;
|
2018-10-17 11:03:32 -04:00
|
|
|
|
|
|
|
class Bat
|
|
|
|
{
|
|
|
|
public function __construct()
|
|
|
|
{
|
|
|
|
$a = new Bar();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
';
|
|
|
|
|
|
|
|
$this->file_provider->registerFile($bat_file_path, $bat_replacement_contents);
|
|
|
|
|
2018-11-11 12:01:14 -05:00
|
|
|
$this->project_analyzer->getCodebase()->reloadFiles($this->project_analyzer, []);
|
2018-10-17 11:03:32 -04:00
|
|
|
|
2019-05-09 18:23:14 -04:00
|
|
|
$this->project_analyzer->check('tests/fixtures/DummyProject', true);
|
2018-10-17 11:03:32 -04:00
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
|
|
|
$this->assertSame(
|
2019-03-23 09:50:47 -04:00
|
|
|
'Psalm was able to infer types for 100% of the codebase',
|
|
|
|
$this->project_analyzer->getCodebase()->analyzer->getTypeInferenceSummary(
|
|
|
|
$this->project_analyzer->getCodebase()
|
|
|
|
)
|
2018-10-17 11:03:32 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCheckDir()
|
|
|
|
{
|
2018-11-11 12:01:14 -05:00
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
2018-10-17 11:03:32 -04:00
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-09 18:23:14 -04:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2018-10-17 11:03:32 -04:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
ob_start();
|
2019-05-09 18:23:14 -04:00
|
|
|
$this->project_analyzer->checkDir('tests/fixtures/DummyProject');
|
2018-10-17 11:03:32 -04:00
|
|
|
$output = ob_get_clean();
|
|
|
|
|
|
|
|
$this->assertSame('Scanning files...' . "\n" . 'Analyzing files...' . "\n", $output);
|
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
|
|
|
$this->assertSame(
|
2019-03-23 09:50:47 -04:00
|
|
|
'Psalm was able to infer types for 100% of the codebase',
|
|
|
|
$this->project_analyzer->getCodebase()->analyzer->getTypeInferenceSummary(
|
|
|
|
$this->project_analyzer->getCodebase()
|
|
|
|
)
|
2018-10-17 11:03:32 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCheckPaths()
|
|
|
|
{
|
2018-11-11 12:01:14 -05:00
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
2018-10-17 11:03:32 -04:00
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-09 18:23:14 -04:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2018-10-17 11:03:32 -04:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
ob_start();
|
2019-05-09 18:23:14 -04:00
|
|
|
$this->project_analyzer->checkPaths(['tests/fixtures/DummyProject/Bar.php']);
|
2018-10-17 11:03:32 -04:00
|
|
|
$output = ob_get_clean();
|
|
|
|
|
|
|
|
$this->assertSame('Scanning files...' . "\n" . 'Analyzing files...' . "\n", $output);
|
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
|
|
|
$this->assertSame(
|
2019-03-23 09:50:47 -04:00
|
|
|
'Psalm was able to infer types for 100% of the codebase',
|
|
|
|
$this->project_analyzer->getCodebase()->analyzer->getTypeInferenceSummary(
|
|
|
|
$this->project_analyzer->getCodebase()
|
|
|
|
)
|
2018-10-17 11:03:32 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function testCheckFile()
|
|
|
|
{
|
2018-11-11 12:01:14 -05:00
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
2018-10-17 11:03:32 -04:00
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-09 18:23:14 -04:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2018-10-17 11:03:32 -04:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
ob_start();
|
2019-05-09 18:23:14 -04:00
|
|
|
$this->project_analyzer->checkFile('tests/fixtures/DummyProject/Bar.php');
|
2018-10-17 11:03:32 -04:00
|
|
|
$output = ob_get_clean();
|
|
|
|
|
|
|
|
$this->assertSame('Scanning files...' . "\n" . 'Analyzing files...' . "\n", $output);
|
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
|
|
|
$this->assertSame(
|
2019-03-23 09:50:47 -04:00
|
|
|
'Psalm was able to infer types for 100% of the codebase',
|
|
|
|
$this->project_analyzer->getCodebase()->analyzer->getTypeInferenceSummary(
|
|
|
|
$this->project_analyzer->getCodebase()
|
|
|
|
)
|
2018-10-17 11:03:32 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|