mirror of
https://github.com/danog/psalm.git
synced 2025-01-07 13:42:11 +01:00
312 lines
10 KiB
PHP
312 lines
10 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Psalm\Tests\Cache;
|
|
|
|
use Psalm\Config;
|
|
use Psalm\Internal\Analyzer\IssueData;
|
|
use Psalm\Internal\Analyzer\ProjectAnalyzer;
|
|
use Psalm\Internal\IncludeCollector;
|
|
use Psalm\Internal\Provider\FakeFileProvider;
|
|
use Psalm\Internal\Provider\Providers;
|
|
use Psalm\Internal\RuntimeCaches;
|
|
use Psalm\IssueBuffer;
|
|
use Psalm\Tests\Internal\Provider\ClassLikeStorageInstanceCacheProvider;
|
|
use Psalm\Tests\Internal\Provider\FakeFileReferenceCacheProvider;
|
|
use Psalm\Tests\Internal\Provider\FileStorageInstanceCacheProvider;
|
|
use Psalm\Tests\Internal\Provider\ParserInstanceCacheProvider;
|
|
use Psalm\Tests\Internal\Provider\ProjectCacheProvider;
|
|
use Psalm\Tests\TestCase;
|
|
|
|
use function microtime;
|
|
use function str_replace;
|
|
|
|
use const DIRECTORY_SEPARATOR;
|
|
|
|
class CacheTest extends TestCase
|
|
{
|
|
public function setUp(): void
|
|
{
|
|
parent::setUp();
|
|
|
|
RuntimeCaches::clearAll();
|
|
}
|
|
|
|
public function tearDown(): void
|
|
{
|
|
RuntimeCaches::clearAll();
|
|
|
|
parent::tearDown();
|
|
}
|
|
|
|
/**
|
|
* @param array<string, list<IssueData>> $issue_data
|
|
* @return array<string, list<string>>
|
|
*/
|
|
private static function normalizeIssueData(array $issue_data): array
|
|
{
|
|
$return = [];
|
|
foreach ($issue_data as $issue_data_per_file) {
|
|
foreach ($issue_data_per_file as $one_issue_data) {
|
|
$file_name = str_replace(DIRECTORY_SEPARATOR, '/', $one_issue_data->file_name);
|
|
$return[$file_name][] = $one_issue_data->type . ': ' . $one_issue_data->message;
|
|
}
|
|
}
|
|
|
|
return $return;
|
|
}
|
|
|
|
/**
|
|
* @param list<array{
|
|
* files: array<string, string|null>,
|
|
* issues?: array<string, list<string>>,
|
|
* }> $interactions
|
|
* @dataProvider provideCacheInteractions
|
|
*/
|
|
public function testCacheInteractions(
|
|
array $interactions,
|
|
): void {
|
|
$config = Config::loadFromXML(
|
|
__DIR__ . DIRECTORY_SEPARATOR . 'test_base_dir',
|
|
<<<'XML'
|
|
<?xml version="1.0"?>
|
|
<psalm>
|
|
<projectFiles>
|
|
<directory name="src" />
|
|
</projectFiles>
|
|
</psalm>
|
|
XML,
|
|
);
|
|
$config->setIncludeCollector(new IncludeCollector());
|
|
|
|
$file_provider = new FakeFileProvider();
|
|
$providers = new Providers(
|
|
$file_provider,
|
|
new ParserInstanceCacheProvider(),
|
|
new FileStorageInstanceCacheProvider(),
|
|
new ClassLikeStorageInstanceCacheProvider(),
|
|
new FakeFileReferenceCacheProvider(),
|
|
new ProjectCacheProvider(),
|
|
);
|
|
|
|
foreach ($interactions as $interaction) {
|
|
foreach ($interaction['files'] as $file_path => $file_contents) {
|
|
$file_path = $config->base_dir . str_replace('/', DIRECTORY_SEPARATOR, $file_path);
|
|
if ($file_contents === null) {
|
|
$file_provider->deleteFile($file_path);
|
|
} else {
|
|
$file_provider->registerFile($file_path, $file_contents);
|
|
}
|
|
}
|
|
|
|
RuntimeCaches::clearAll();
|
|
|
|
$start_time = microtime(true);
|
|
$project_analyzer = new ProjectAnalyzer($config, $providers);
|
|
$project_analyzer->check($config->base_dir, true);
|
|
$project_analyzer->finish($start_time, PSALM_VERSION);
|
|
|
|
$issues = self::normalizeIssueData(IssueBuffer::getIssuesData());
|
|
self::assertSame($interaction['issues'] ?? [], $issues);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return iterable<string, list{
|
|
* list<array{
|
|
* files: array<string, string|null>,
|
|
* issues?: array<string, list<string>>,
|
|
* }>,
|
|
* }>
|
|
*/
|
|
public static function provideCacheInteractions(): iterable
|
|
{
|
|
yield 'deletedFileInvalidatesReferencingMethod' => [
|
|
[
|
|
[
|
|
'files' => [
|
|
'/src/A.php' => <<<'PHP'
|
|
<?php
|
|
class A {
|
|
public function do(B $b): void
|
|
{
|
|
$b->do();
|
|
}
|
|
}
|
|
PHP,
|
|
'/src/B.php' => <<<'PHP'
|
|
<?php
|
|
class B {
|
|
public function do(): void
|
|
{
|
|
echo 'B';
|
|
}
|
|
}
|
|
PHP,
|
|
],
|
|
],
|
|
[
|
|
'files' => [
|
|
'/src/B.php' => null,
|
|
],
|
|
'issues' => [
|
|
'/src/A.php' => [
|
|
'UndefinedClass: Class, interface or enum named B does not exist',
|
|
],
|
|
],
|
|
],
|
|
],
|
|
];
|
|
|
|
yield 'classPropertyTypeChangeInvalidatesReferencingMethod' => [
|
|
[
|
|
[
|
|
'files' => [
|
|
'/src/A.php' => <<<'PHP'
|
|
<?php
|
|
class A {
|
|
public function foo(B $b): int
|
|
{
|
|
return $b->value;
|
|
}
|
|
}
|
|
PHP,
|
|
'/src/B.php' => <<<'PHP'
|
|
<?php
|
|
class B {
|
|
public ?int $value = 0;
|
|
}
|
|
PHP,
|
|
],
|
|
'issues' => [
|
|
'/src/A.php' => [
|
|
"NullableReturnStatement: The declared return type 'int' for A::foo is not nullable, but the function returns 'int|null'",
|
|
"InvalidNullableReturnType: The declared return type 'int' for A::foo is not nullable, but 'int|null' contains null",
|
|
],
|
|
],
|
|
],
|
|
[
|
|
'files' => [
|
|
'/src/B.php' => <<<'PHP'
|
|
<?php
|
|
class B {
|
|
public int $value = 0;
|
|
}
|
|
PHP,
|
|
],
|
|
'issues' => [],
|
|
],
|
|
],
|
|
];
|
|
|
|
yield 'classDocblockChange' => [
|
|
[
|
|
[
|
|
'files' => [
|
|
'/src/A.php' => <<<'PHP'
|
|
<?php
|
|
|
|
/**
|
|
* @template T
|
|
*/
|
|
class A {
|
|
/**
|
|
* @param T $baz
|
|
*/
|
|
public function foo($baz): void
|
|
{
|
|
}
|
|
}
|
|
PHP,
|
|
'/src/B.php' => <<<'PHP'
|
|
<?php
|
|
|
|
class B {
|
|
public function foo(): void
|
|
{
|
|
(new A)->foo(1);
|
|
}
|
|
}
|
|
PHP,
|
|
],
|
|
'issues' => [],
|
|
],
|
|
[
|
|
'files' => [
|
|
'/src/A.php' => <<<'PHP'
|
|
<?php
|
|
|
|
/**
|
|
* @template K
|
|
*/
|
|
class A {
|
|
/**
|
|
* @param T $baz
|
|
*/
|
|
public function foo($baz): void
|
|
{
|
|
}
|
|
}
|
|
PHP,
|
|
],
|
|
'issues' => [
|
|
'/src/A.php' => [
|
|
"UndefinedDocblockClass: Docblock-defined class, interface or enum named T does not exist",
|
|
],
|
|
'/src/B.php' => [
|
|
"InvalidArgument: Argument 1 of A::foo expects T, but 1 provided",
|
|
],
|
|
],
|
|
],
|
|
],
|
|
];
|
|
|
|
yield 'constructorPropertyPromotionChange' => [
|
|
[
|
|
[
|
|
'files' => [
|
|
'/src/A.php' => <<<'PHP'
|
|
<?php
|
|
class A {
|
|
public function __construct(private string $foo)
|
|
{
|
|
}
|
|
public function bar(): string
|
|
{
|
|
return $this->foo;
|
|
}
|
|
}
|
|
PHP,
|
|
],
|
|
'issues' => [],
|
|
],
|
|
[
|
|
'files' => [
|
|
'/src/A.php' => <<<'PHP'
|
|
<?php
|
|
class A
|
|
{
|
|
public function __construct()
|
|
{
|
|
}
|
|
public function bar(): string
|
|
{
|
|
return $this->foo;
|
|
}
|
|
}
|
|
PHP,
|
|
],
|
|
'issues' => [
|
|
'/src/A.php' => [
|
|
"UndefinedThisPropertyFetch: Instance property A::\$foo is not defined",
|
|
"MixedReturnStatement: Could not infer a return type",
|
|
"MixedInferredReturnType: Could not verify return type 'string' for A::bar",
|
|
],
|
|
],
|
|
],
|
|
],
|
|
];
|
|
}
|
|
}
|