2018-10-17 17:03:32 +02:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2019-07-05 22:24:00 +02:00
|
|
|
use function define;
|
|
|
|
use function defined;
|
|
|
|
use const DIRECTORY_SEPARATOR;
|
|
|
|
use function get_class;
|
|
|
|
use function getcwd;
|
|
|
|
use function microtime;
|
|
|
|
use function ob_end_clean;
|
|
|
|
use function ob_get_clean;
|
|
|
|
use function ob_start;
|
2019-02-24 19:48:57 +01:00
|
|
|
use Psalm\Codebase;
|
2018-10-17 17:03:32 +02:00
|
|
|
use Psalm\Config;
|
2019-03-23 19:27:54 +01:00
|
|
|
use Psalm\Internal\Analyzer\FileAnalyzer;
|
2020-07-11 23:17:22 +02:00
|
|
|
use Psalm\Internal\IncludeCollector;
|
2020-08-23 16:32:07 +02:00
|
|
|
use Psalm\Internal\RuntimeCaches;
|
2019-02-24 19:48:57 +01:00
|
|
|
use Psalm\Plugin\Hook\AfterCodebasePopulatedInterface;
|
2018-11-12 16:57:05 +01:00
|
|
|
use Psalm\Tests\Internal\Provider;
|
2019-05-30 16:30:41 +02:00
|
|
|
use Psalm\Tests\Progress\EchoProgress;
|
2020-08-23 16:32:07 +02:00
|
|
|
use function realpath;
|
2018-10-17 17:03:32 +02:00
|
|
|
|
2019-03-23 19:27:54 +01:00
|
|
|
class ProjectCheckerTest extends TestCase
|
2018-10-17 17:03:32 +02:00
|
|
|
{
|
|
|
|
/** @var TestConfig */
|
|
|
|
protected static $config;
|
|
|
|
|
2018-11-06 03:57:36 +01:00
|
|
|
/** @var \Psalm\Internal\Analyzer\ProjectAnalyzer */
|
2018-11-11 18:01:14 +01:00
|
|
|
protected $project_analyzer;
|
2018-10-17 17:03:32 +02:00
|
|
|
|
2019-05-17 00:36:36 +02:00
|
|
|
public static function setUpBeforeClass() : void
|
2018-10-17 17:03:32 +02: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');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-17 00:36:36 +02:00
|
|
|
public function setUp() : void
|
2018-10-17 17:03:32 +02:00
|
|
|
{
|
2020-08-23 16:32:07 +02:00
|
|
|
RuntimeCaches::clearAll();
|
2018-10-17 17:03:32 +02:00
|
|
|
$this->file_provider = new Provider\FakeFileProvider();
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
private function getProjectAnalyzerWithConfig(Config $config): \Psalm\Internal\Analyzer\ProjectAnalyzer
|
2018-10-17 17:03:32 +02:00
|
|
|
{
|
2020-07-11 23:17:22 +02:00
|
|
|
$config->setIncludeCollector(new IncludeCollector());
|
2018-11-06 03:57:36 +01:00
|
|
|
return new \Psalm\Internal\Analyzer\ProjectAnalyzer(
|
2018-10-17 17:03:32 +02:00
|
|
|
$config,
|
2018-11-06 03:57:36 +01:00
|
|
|
new \Psalm\Internal\Provider\Providers(
|
2018-10-17 17:03:32 +02:00
|
|
|
$this->file_provider,
|
|
|
|
new Provider\ParserInstanceCacheProvider(),
|
|
|
|
new Provider\FileStorageInstanceCacheProvider(),
|
|
|
|
new Provider\ClassLikeStorageInstanceCacheProvider(),
|
2020-04-12 18:02:19 +02:00
|
|
|
new Provider\FakeFileReferenceCacheProvider(),
|
|
|
|
new Provider\ProjectCacheProvider()
|
2019-05-30 16:30:41 +02:00
|
|
|
),
|
2019-06-09 18:37:28 +02:00
|
|
|
new \Psalm\Report\ReportOptions()
|
2018-10-17 17:03:32 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testCheck(): void
|
2018-10-17 17:03:32 +02:00
|
|
|
{
|
2018-11-11 18:01:14 +01:00
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
2018-10-17 17:03:32 +02:00
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-10 00:23:14 +02:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2018-10-17 17:03:32 +02:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-05-30 16:30:41 +02:00
|
|
|
$this->project_analyzer->progress = new EchoProgress();
|
|
|
|
|
2018-10-17 17:03:32 +02:00
|
|
|
ob_start();
|
2019-05-10 00:23:14 +02:00
|
|
|
$this->project_analyzer->check('tests/fixtures/DummyProject');
|
2018-10-17 17:03:32 +02:00
|
|
|
$output = ob_get_clean();
|
|
|
|
|
2019-05-30 16:47:28 +02:00
|
|
|
$this->assertSame('Scanning files...' . "\n" . 'Analyzing files...' . "\n\n", $output);
|
2018-10-17 17:03:32 +02:00
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
2019-03-23 14:50:47 +01:00
|
|
|
$codebase = $this->project_analyzer->getCodebase();
|
|
|
|
|
2019-10-15 18:09:08 +02:00
|
|
|
$this->assertSame([0, 5], $codebase->analyzer->getTotalTypeCoverage($codebase));
|
2019-03-23 14:50:47 +01:00
|
|
|
|
2018-10-17 17:03:32 +02:00
|
|
|
$this->assertSame(
|
2019-03-23 14:50:47 +01:00
|
|
|
'Psalm was able to infer types for 100% of the codebase',
|
|
|
|
$codebase->analyzer->getTypeInferenceSummary(
|
|
|
|
$codebase
|
|
|
|
)
|
2018-10-17 17:03:32 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testAfterCodebasePopulatedIsInvoked(): void
|
2019-02-24 19:48:57 +01:00
|
|
|
{
|
2019-03-23 19:27:54 +01:00
|
|
|
$hook = new class implements AfterCodebasePopulatedInterface {
|
2019-02-24 19:48:57 +01:00
|
|
|
/** @var bool */
|
|
|
|
public static $called = false;
|
2019-03-23 19:27:54 +01:00
|
|
|
|
2019-02-24 19:48:57 +01: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-10 00:23:14 +02:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2019-02-24 19:48:57 +01:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-03-01 14:45:18 +01:00
|
|
|
$hook_class = get_class($hook);
|
|
|
|
|
|
|
|
$this->project_analyzer->getCodebase()->config->after_codebase_populated[] = $hook_class;
|
2019-02-24 19:48:57 +01:00
|
|
|
|
|
|
|
ob_start();
|
2019-05-10 00:23:14 +02:00
|
|
|
$this->project_analyzer->check('tests/fixtures/DummyProject');
|
2019-02-24 20:14:14 +01:00
|
|
|
ob_end_clean();
|
2019-02-24 19:48:57 +01:00
|
|
|
|
|
|
|
$this->assertTrue($hook::$called);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testCheckAfterNoChange(): void
|
2018-10-17 17:03:32 +02:00
|
|
|
{
|
2018-11-11 18:01:14 +01:00
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
2018-10-17 17:03:32 +02:00
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-10 00:23:14 +02:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2018-10-17 17:03:32 +02:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-06-09 18:37:28 +02:00
|
|
|
$this->assertNotNull($this->project_analyzer->stdout_report_options);
|
|
|
|
|
|
|
|
$this->project_analyzer->stdout_report_options->format = \Psalm\Report::TYPE_JSON;
|
2018-10-17 17:03:32 +02:00
|
|
|
|
2019-05-10 00:23:14 +02:00
|
|
|
$this->project_analyzer->check('tests/fixtures/DummyProject', true);
|
2020-07-20 10:52:01 +02:00
|
|
|
ob_start();
|
2018-11-11 18:01:14 +01:00
|
|
|
\Psalm\IssueBuffer::finish($this->project_analyzer, true, microtime(true));
|
2020-07-20 10:52:01 +02:00
|
|
|
ob_end_clean();
|
2018-10-17 17:03:32 +02:00
|
|
|
|
|
|
|
$this->assertSame(
|
2019-03-23 14:50:47 +01: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 17:03:32 +02:00
|
|
|
);
|
|
|
|
|
2018-11-11 18:01:14 +01:00
|
|
|
$this->project_analyzer->getCodebase()->reloadFiles($this->project_analyzer, []);
|
2018-10-17 17:03:32 +02:00
|
|
|
|
2019-05-10 00:23:14 +02:00
|
|
|
$this->project_analyzer->check('tests/fixtures/DummyProject', true);
|
2018-10-17 17:03:32 +02:00
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
|
|
|
$this->assertSame(
|
2020-06-06 22:57:25 +02:00
|
|
|
'Psalm was able to infer types for 100% of the codebase',
|
2019-03-23 14:50:47 +01:00
|
|
|
$this->project_analyzer->getCodebase()->analyzer->getTypeInferenceSummary(
|
|
|
|
$this->project_analyzer->getCodebase()
|
|
|
|
)
|
2018-10-17 17:03:32 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testCheckAfterFileChange(): void
|
2018-10-17 17:03:32 +02:00
|
|
|
{
|
2018-11-11 18:01:14 +01:00
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
2018-10-17 17:03:32 +02:00
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-10 00:23:14 +02:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2018-10-17 17:03:32 +02:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-06-09 18:37:28 +02:00
|
|
|
$this->assertNotNull($this->project_analyzer->stdout_report_options);
|
|
|
|
|
|
|
|
$this->project_analyzer->stdout_report_options->format = \Psalm\Report::TYPE_JSON;
|
2018-10-17 17:03:32 +02:00
|
|
|
|
2019-05-10 00:23:14 +02:00
|
|
|
$this->project_analyzer->check('tests/fixtures/DummyProject', true);
|
2020-07-20 10:52:01 +02:00
|
|
|
ob_start();
|
2018-11-11 18:01:14 +01:00
|
|
|
\Psalm\IssueBuffer::finish($this->project_analyzer, true, microtime(true));
|
2020-07-20 10:52:01 +02:00
|
|
|
ob_end_clean();
|
2018-10-17 17:03:32 +02:00
|
|
|
|
|
|
|
$this->assertSame(
|
2019-03-23 14:50:47 +01: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 17:03:32 +02:00
|
|
|
);
|
|
|
|
|
2019-05-10 00:23:14 +02:00
|
|
|
$bat_file_path = getcwd()
|
|
|
|
. DIRECTORY_SEPARATOR . 'tests'
|
|
|
|
. DIRECTORY_SEPARATOR . 'fixtures'
|
|
|
|
. DIRECTORY_SEPARATOR . 'DummyProject'
|
|
|
|
. DIRECTORY_SEPARATOR . 'Bat.php';
|
2018-10-17 17:03:32 +02:00
|
|
|
|
|
|
|
$bat_replacement_contents = '<?php
|
|
|
|
|
2019-02-08 23:41:03 +01:00
|
|
|
namespace Vimeo\Test\DummyProject;
|
2018-10-17 17:03:32 +02:00
|
|
|
|
|
|
|
class Bat
|
|
|
|
{
|
|
|
|
public function __construct()
|
|
|
|
{
|
|
|
|
$a = new Bar();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
';
|
|
|
|
|
|
|
|
$this->file_provider->registerFile($bat_file_path, $bat_replacement_contents);
|
|
|
|
|
2018-11-11 18:01:14 +01:00
|
|
|
$this->project_analyzer->getCodebase()->reloadFiles($this->project_analyzer, []);
|
2018-10-17 17:03:32 +02:00
|
|
|
|
2019-05-10 00:23:14 +02:00
|
|
|
$this->project_analyzer->check('tests/fixtures/DummyProject', true);
|
2018-10-17 17:03:32 +02:00
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
|
|
|
$this->assertSame(
|
2019-03-23 14:50:47 +01: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 17:03:32 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testCheckDir(): void
|
2018-10-17 17:03:32 +02:00
|
|
|
{
|
2018-11-11 18:01:14 +01:00
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
2018-10-17 17:03:32 +02:00
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-10 00:23:14 +02:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2018-10-17 17:03:32 +02:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-05-30 16:30:41 +02:00
|
|
|
$this->project_analyzer->progress = new EchoProgress();
|
|
|
|
|
2018-10-17 17:03:32 +02:00
|
|
|
ob_start();
|
2019-05-10 00:23:14 +02:00
|
|
|
$this->project_analyzer->checkDir('tests/fixtures/DummyProject');
|
2018-10-17 17:03:32 +02:00
|
|
|
$output = ob_get_clean();
|
|
|
|
|
2019-05-30 16:47:28 +02:00
|
|
|
$this->assertSame('Scanning files...' . "\n" . 'Analyzing files...' . "\n\n", $output);
|
2018-10-17 17:03:32 +02:00
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
|
|
|
$this->assertSame(
|
2019-03-23 14:50:47 +01: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 17:03:32 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testCheckPaths(): void
|
2018-10-17 17:03:32 +02:00
|
|
|
{
|
2018-11-11 18:01:14 +01:00
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
2018-10-17 17:03:32 +02:00
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-10 00:23:14 +02:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2018-10-17 17:03:32 +02:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-05-30 16:30:41 +02:00
|
|
|
$this->project_analyzer->progress = new EchoProgress();
|
|
|
|
|
2018-10-17 17:03:32 +02:00
|
|
|
ob_start();
|
2020-08-23 16:32:07 +02:00
|
|
|
// checkPaths expects absolute paths,
|
|
|
|
// otherwise it's unable to match them against configured folders
|
2019-10-15 18:09:08 +02:00
|
|
|
$this->project_analyzer->checkPaths([
|
2020-08-23 16:32:07 +02:00
|
|
|
realpath(getcwd() . '/tests/fixtures/DummyProject/Bar.php'),
|
|
|
|
realpath(getcwd() . '/tests/fixtures/DummyProject/SomeTrait.php'),
|
2019-10-15 18:09:08 +02:00
|
|
|
]);
|
2018-10-17 17:03:32 +02:00
|
|
|
$output = ob_get_clean();
|
|
|
|
|
2019-05-30 16:47:28 +02:00
|
|
|
$this->assertSame('Scanning files...' . "\n" . 'Analyzing files...' . "\n\n", $output);
|
2018-10-17 17:03:32 +02:00
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
|
|
|
$this->assertSame(
|
2019-03-23 14:50:47 +01: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 17:03:32 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testCheckFile(): void
|
2018-10-17 17:03:32 +02:00
|
|
|
{
|
2018-11-11 18:01:14 +01:00
|
|
|
$this->project_analyzer = $this->getProjectAnalyzerWithConfig(
|
2018-10-17 17:03:32 +02:00
|
|
|
Config::loadFromXML(
|
|
|
|
(string)getcwd(),
|
|
|
|
'<?xml version="1.0"?>
|
|
|
|
<psalm>
|
|
|
|
<projectFiles>
|
2019-05-10 00:23:14 +02:00
|
|
|
<directory name="tests/fixtures/DummyProject" />
|
2018-10-17 17:03:32 +02:00
|
|
|
</projectFiles>
|
|
|
|
</psalm>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-05-30 16:30:41 +02:00
|
|
|
$this->project_analyzer->progress = new EchoProgress();
|
|
|
|
|
2018-10-17 17:03:32 +02:00
|
|
|
ob_start();
|
2020-08-23 16:32:07 +02:00
|
|
|
// checkPaths expects absolute paths,
|
|
|
|
// otherwise it's unable to match them against configured folders
|
2019-10-15 18:09:08 +02:00
|
|
|
$this->project_analyzer->checkPaths([
|
2020-08-23 16:32:07 +02:00
|
|
|
realpath(getcwd() . '/tests/fixtures/DummyProject/Bar.php'),
|
|
|
|
realpath(getcwd() . '/tests/fixtures/DummyProject/SomeTrait.php'),
|
2019-10-15 18:09:08 +02:00
|
|
|
]);
|
2018-10-17 17:03:32 +02:00
|
|
|
$output = ob_get_clean();
|
|
|
|
|
2019-05-30 16:47:28 +02:00
|
|
|
$this->assertSame('Scanning files...' . "\n" . 'Analyzing files...' . "\n\n", $output);
|
2018-10-17 17:03:32 +02:00
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
|
|
|
$this->assertSame(
|
2019-03-23 14:50:47 +01: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 17:03:32 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|