2018-10-17 11:03:32 -04:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
2021-01-06 15:05:53 +01:00
|
|
|
use Psalm\Plugin\EventHandler\Event\AfterCodebasePopulatedEvent;
|
2019-07-05 16:24:00 -04: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;
|
2018-10-17 11:03:32 -04:00
|
|
|
use Psalm\Config;
|
2020-07-12 00:17:22 +03:00
|
|
|
use Psalm\Internal\IncludeCollector;
|
2020-08-23 17:32:07 +03:00
|
|
|
use Psalm\Internal\RuntimeCaches;
|
2021-01-06 15:05:53 +01:00
|
|
|
use Psalm\Plugin\EventHandler\AfterCodebasePopulatedInterface;
|
2018-11-12 10:57:05 -05:00
|
|
|
use Psalm\Tests\Internal\Provider;
|
2019-05-30 16:30:41 +02:00
|
|
|
use Psalm\Tests\Progress\EchoProgress;
|
2020-08-23 17:32:07 +03:00
|
|
|
use function realpath;
|
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
|
|
|
|
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');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-16 18:36:36 -04:00
|
|
|
public function setUp() : void
|
2018-10-17 11:03:32 -04:00
|
|
|
{
|
2020-08-23 17:32:07 +03:00
|
|
|
RuntimeCaches::clearAll();
|
2018-10-17 11:03:32 -04: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 11:03:32 -04:00
|
|
|
{
|
2020-07-12 00:17:22 +03:00
|
|
|
$config->setIncludeCollector(new IncludeCollector());
|
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(),
|
2020-04-12 12:02:19 -04:00
|
|
|
new Provider\FakeFileReferenceCacheProvider(),
|
|
|
|
new Provider\ProjectCacheProvider()
|
2019-05-30 16:30:41 +02:00
|
|
|
),
|
2019-06-09 12:37:28 -04:00
|
|
|
new \Psalm\Report\ReportOptions()
|
2018-10-17 11:03:32 -04:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testCheck(): void
|
2018-10-17 11:03:32 -04:00
|
|
|
{
|
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>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-05-30 16:30:41 +02:00
|
|
|
$this->project_analyzer->progress = new EchoProgress();
|
|
|
|
|
2018-10-17 11:03:32 -04:00
|
|
|
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();
|
|
|
|
|
2019-05-30 10:47:28 -04:00
|
|
|
$this->assertSame('Scanning files...' . "\n" . 'Analyzing files...' . "\n\n", $output);
|
2018-10-17 11:03:32 -04:00
|
|
|
|
|
|
|
$this->assertSame(0, \Psalm\IssueBuffer::getErrorCount());
|
|
|
|
|
2019-03-23 09:50:47 -04:00
|
|
|
$codebase = $this->project_analyzer->getCodebase();
|
|
|
|
|
2019-10-15 12:09:08 -04:00
|
|
|
$this->assertSame([0, 5], $codebase->analyzer->getTotalTypeCoverage($codebase));
|
2019-03-23 09:50:47 -04:00
|
|
|
|
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
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testAfterCodebasePopulatedIsInvoked(): void
|
2019-02-24 20:48:57 +02:00
|
|
|
{
|
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 */
|
2021-01-06 15:05:53 +01:00
|
|
|
public static function afterCodebasePopulated(AfterCodebasePopulatedEvent $event)
|
2019-02-24 20:48:57 +02:00
|
|
|
{
|
|
|
|
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);
|
|
|
|
|
2021-01-06 15:05:53 +01:00
|
|
|
$this->project_analyzer->getCodebase()->config->eventDispatcher->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);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testCheckAfterNoChange(): void
|
2018-10-17 11:03:32 -04:00
|
|
|
{
|
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>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-06-09 12:37:28 -04:00
|
|
|
$this->assertNotNull($this->project_analyzer->stdout_report_options);
|
|
|
|
|
|
|
|
$this->project_analyzer->stdout_report_options->format = \Psalm\Report::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);
|
2020-07-20 11:52:01 +03:00
|
|
|
ob_start();
|
2021-03-17 01:28:18 +02:00
|
|
|
\Psalm\IssueBuffer::finish($this->project_analyzer, true, microtime(true));
|
2020-07-20 11:52:01 +03:00
|
|
|
ob_end_clean();
|
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(
|
2020-06-06 16:57:25 -04:00
|
|
|
'Psalm was able to infer types for 100% of the codebase',
|
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
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testCheckAfterFileChange(): void
|
2018-10-17 11:03:32 -04:00
|
|
|
{
|
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>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-06-09 12:37:28 -04:00
|
|
|
$this->assertNotNull($this->project_analyzer->stdout_report_options);
|
|
|
|
|
|
|
|
$this->project_analyzer->stdout_report_options->format = \Psalm\Report::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);
|
2020-07-20 11:52:01 +03:00
|
|
|
ob_start();
|
2021-03-17 01:28:18 +02:00
|
|
|
\Psalm\IssueBuffer::finish($this->project_analyzer, true, microtime(true));
|
2020-07-20 11:52:01 +03:00
|
|
|
ob_end_clean();
|
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
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testCheckDir(): void
|
2018-10-17 11:03:32 -04:00
|
|
|
{
|
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>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-05-30 16:30:41 +02:00
|
|
|
$this->project_analyzer->progress = new EchoProgress();
|
|
|
|
|
2018-10-17 11:03:32 -04:00
|
|
|
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();
|
|
|
|
|
2019-05-30 10:47:28 -04:00
|
|
|
$this->assertSame('Scanning files...' . "\n" . 'Analyzing files...' . "\n\n", $output);
|
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
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testCheckPaths(): void
|
2018-10-17 11:03:32 -04:00
|
|
|
{
|
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>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-05-30 16:30:41 +02:00
|
|
|
$this->project_analyzer->progress = new EchoProgress();
|
|
|
|
|
2018-10-17 11:03:32 -04:00
|
|
|
ob_start();
|
2020-08-23 17:32:07 +03:00
|
|
|
// checkPaths expects absolute paths,
|
|
|
|
// otherwise it's unable to match them against configured folders
|
2019-10-15 12:09:08 -04:00
|
|
|
$this->project_analyzer->checkPaths([
|
2020-08-23 17:32:07 +03:00
|
|
|
realpath(getcwd() . '/tests/fixtures/DummyProject/Bar.php'),
|
|
|
|
realpath(getcwd() . '/tests/fixtures/DummyProject/SomeTrait.php'),
|
2019-10-15 12:09:08 -04:00
|
|
|
]);
|
2018-10-17 11:03:32 -04:00
|
|
|
$output = ob_get_clean();
|
|
|
|
|
2019-05-30 10:47:28 -04:00
|
|
|
$this->assertSame('Scanning files...' . "\n" . 'Analyzing files...' . "\n\n", $output);
|
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
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function testCheckFile(): void
|
2018-10-17 11:03:32 -04:00
|
|
|
{
|
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>'
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
2019-05-30 16:30:41 +02:00
|
|
|
$this->project_analyzer->progress = new EchoProgress();
|
|
|
|
|
2018-10-17 11:03:32 -04:00
|
|
|
ob_start();
|
2020-08-23 17:32:07 +03:00
|
|
|
// checkPaths expects absolute paths,
|
|
|
|
// otherwise it's unable to match them against configured folders
|
2019-10-15 12:09:08 -04:00
|
|
|
$this->project_analyzer->checkPaths([
|
2020-08-23 17:32:07 +03:00
|
|
|
realpath(getcwd() . '/tests/fixtures/DummyProject/Bar.php'),
|
|
|
|
realpath(getcwd() . '/tests/fixtures/DummyProject/SomeTrait.php'),
|
2019-10-15 12:09:08 -04:00
|
|
|
]);
|
2018-10-17 11:03:32 -04:00
|
|
|
$output = ob_get_clean();
|
|
|
|
|
2019-05-30 10:47:28 -04:00
|
|
|
$this->assertSame('Scanning files...' . "\n" . 'Analyzing files...' . "\n\n", $output);
|
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
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|