1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-11 07:31:45 +01:00
psalm/src/Psalm/Internal/Provider/FileReferenceProvider.php

1101 lines
32 KiB
PHP
Raw Normal View History

<?php
2018-11-06 03:57:36 +01:00
namespace Psalm\Internal\Provider;
use function array_filter;
use function array_keys;
use function array_merge;
use function array_unique;
2019-07-05 22:24:00 +02:00
use function file_exists;
use Psalm\Codebase;
use Psalm\CodeLocation;
2020-02-17 00:24:40 +01:00
use Psalm\Internal\Analyzer\IssueData;
2019-07-05 22:24:00 +02:00
use Psalm\Internal\Analyzer\ClassLikeAnalyzer;
/**
* @psalm-import-type FileMapType from \Psalm\Internal\Codebase\Analyzer
*
2017-02-18 23:49:34 +01:00
* Used to determine which files reference other files, necessary for using the --diff
* option from the command line.
*/
class FileReferenceProvider
{
2018-09-30 05:51:06 +02:00
/**
* @var bool
*/
private $loaded_from_cache = false;
/**
* A lookup table used for getting all the references to a class not inside a method
* indexed by file
*
* @var array<string, array<string,bool>>
*/
private static $nonmethod_references_to_classes = [];
/**
* A lookup table used for getting all the methods that reference a class
*
* @var array<string, array<string,bool>>
*/
private static $method_references_to_classes = [];
/**
* A lookup table used for getting all the files that reference a class member
*
* @var array<string, array<string,bool>>
*/
private static $file_references_to_class_members = [];
/**
* A lookup table used for getting all the files that reference a missing class member
*
* @var array<string, array<string,bool>>
*/
private static $file_references_to_missing_class_members = [];
/**
* @var array<string, array<string, true>>
*/
private static $files_inheriting_classes = [];
/**
* A list of all files deleted since the last successful run
*
* @var array<int, string>|null
*/
private static $deleted_files = null;
/**
* A lookup table used for getting all the files referenced by a file
*
* @var array<string, array{a:array<int, string>, i:array<int, string>}>
*/
private static $file_references = [];
/**
* @var array<string, array<string, bool>>
*/
private static $method_references_to_class_members = [];
/**
* @var array<string, array<string, bool>>
*/
private static $method_references_to_missing_class_members = [];
2019-04-27 23:38:24 +02:00
/**
* @var array<string, array<string, bool>>
*/
private static $references_to_mixed_member_names = [];
/**
* @var array<string, array<int, CodeLocation>>
*/
private static $class_method_locations = [];
/**
* @var array<string, array<int, CodeLocation>>
*/
private static $class_property_locations = [];
/**
* @var array<string, array<int, CodeLocation>>
*/
private static $class_locations = [];
/**
* @var array<string, string>
*/
private static $classlike_files = [];
2018-09-30 05:51:06 +02:00
/**
* @var array<string, array<string, int>>
2018-09-30 05:51:06 +02:00
*/
2018-11-02 02:52:39 +01:00
private static $analyzed_methods = [];
2018-09-30 05:51:06 +02:00
/**
* @var array<string, array<int, IssueData>>
*/
private static $issues = [];
/**
* @var array<string, FileMapType>
*/
private static $file_maps = [];
/**
* @var array<string, array{int, int}>
*/
private static $mixed_counts = [];
/**
* @var array<string, array<int, array<string, bool>>>
*/
private static $method_param_uses = [];
/**
* @var ?FileReferenceCacheProvider
*/
public $cache;
public function __construct(FileReferenceCacheProvider $cache = null)
{
$this->cache = $cache;
}
/**
* @return array<string>
*/
public function getDeletedReferencedFiles()
{
if (self::$deleted_files === null) {
self::$deleted_files = array_filter(
array_keys(self::$file_references),
/**
* @param string $file_name
2017-05-27 02:16:18 +02:00
*
* @return bool
*/
function ($file_name) {
return !file_exists($file_name);
}
);
}
return self::$deleted_files;
}
/**
* @param lowercase-string $fq_class_name_lc
* @return void
*/
public function addNonMethodReferenceToClass(string $source_file, string $fq_class_name_lc)
{
self::$nonmethod_references_to_classes[$fq_class_name_lc][$source_file] = true;
}
/**
* @return array<string, array<string,bool>>
*/
public function getAllNonMethodReferencesToClasses()
{
return self::$nonmethod_references_to_classes;
}
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function addNonMethodReferencesToClasses(array $references)
{
foreach ($references as $key => $reference) {
if (isset(self::$nonmethod_references_to_classes[$key])) {
self::$nonmethod_references_to_classes[$key] = array_merge(
$reference,
self::$nonmethod_references_to_classes[$key]
);
} else {
self::$nonmethod_references_to_classes[$key] = $reference;
}
}
}
/**
* @param array<string, string> $map
*/
public function addClassLikeFiles(array $map) : void
{
self::$classlike_files += $map;
}
/**
* @return void
*/
public function addFileReferenceToClassMember(string $source_file, string $referenced_member_id)
{
self::$file_references_to_class_members[$referenced_member_id][$source_file] = true;
}
/**
* @return void
*/
public function addFileReferenceToMissingClassMember(string $source_file, string $referenced_member_id)
{
self::$file_references_to_missing_class_members[$referenced_member_id][$source_file] = true;
}
/**
* @return array<string, array<string,bool>>
*/
public function getAllFileReferencesToClassMembers()
{
return self::$file_references_to_class_members;
}
/**
* @return array<string, array<string,bool>>
*/
public function getAllFileReferencesToMissingClassMembers()
{
return self::$file_references_to_missing_class_members;
}
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function addFileReferencesToClassMembers(array $references)
{
foreach ($references as $key => $reference) {
if (isset(self::$file_references_to_class_members[$key])) {
self::$file_references_to_class_members[$key] = array_merge(
$reference,
self::$file_references_to_class_members[$key]
);
} else {
self::$file_references_to_class_members[$key] = $reference;
}
}
}
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function addFileReferencesToMissingClassMembers(array $references)
{
foreach ($references as $key => $reference) {
if (isset(self::$file_references_to_missing_class_members[$key])) {
self::$file_references_to_missing_class_members[$key] = array_merge(
$reference,
self::$file_references_to_missing_class_members[$key]
);
} else {
self::$file_references_to_missing_class_members[$key] = $reference;
}
}
}
/**
* @param string $source_file
Refactor scanning and analysis, introducing multithreading (#191) * Add failing test * Add visitor to soup up classlike references * Move a whole bunch of code into the visitor * Move some methods back, move onto analysis stage * Use the getAliases method everywhere * Fix refs * Fix more refs * Fix some tests * Fix more tests * Fix include tests * Shift config class finding to project checker and fix bugs * Fix a few more tests * transition test to new syntax * Remove var_dump * Delete a bunch of code and fix mutation test * Remove unnecessary visitation * Transition to better mocked out file provider, breaking some cached statement loading * Use different scheme for naming anonymous classes * Fix anonymous class issues * Refactor file/statement loading * Add specific property types * Fix mapped property assignment * Improve how we deal with traits * Fix trait checking * Pass Psalm checks * Add multi-process support * Delay console output until the end * Remove PHP 7 syntax * Update file storage with classes * Fix scanning individual files and add reflection return types * Always turn XDebug off * Add quicker method of getting method mutations * Queue return types for crawling * Interpret all strings as possible classes once we see a `get_class` call * Check invalid return types again * Fix template namespacing issues * Default to class-insensitive file names for includes * Don’t overwrite existing issues data * Add var docblocks for scanning * Add null check * Fix loading of external classes in templates * Only try to populate class when we haven’t yet seen it’s not a class * Fix trait property accessibility * Only ever improve docblock param type * Make param replacement more robust * Fix static const missing inferred type * Fix a few more tests * Register constant definitions * Fix trait aliasing * Skip constant type tests for now * Fix linting issues * Make sure caching is off for tests * Remove unnecessary return * Use emulative parser if on PHP 5.6 * Cache parser for faster first-time parse * Fix constant resolution when scanning classes * Remove test that’s beyond a practical scope * Add back --diff support * Add --help for --threads * Remove unused vars
2017-07-25 22:11:02 +02:00
* @param string $fq_class_name_lc
2017-05-27 02:16:18 +02:00
*
* @return void
*/
public function addFileInheritanceToClass($source_file, $fq_class_name_lc)
{
Refactor scanning and analysis, introducing multithreading (#191) * Add failing test * Add visitor to soup up classlike references * Move a whole bunch of code into the visitor * Move some methods back, move onto analysis stage * Use the getAliases method everywhere * Fix refs * Fix more refs * Fix some tests * Fix more tests * Fix include tests * Shift config class finding to project checker and fix bugs * Fix a few more tests * transition test to new syntax * Remove var_dump * Delete a bunch of code and fix mutation test * Remove unnecessary visitation * Transition to better mocked out file provider, breaking some cached statement loading * Use different scheme for naming anonymous classes * Fix anonymous class issues * Refactor file/statement loading * Add specific property types * Fix mapped property assignment * Improve how we deal with traits * Fix trait checking * Pass Psalm checks * Add multi-process support * Delay console output until the end * Remove PHP 7 syntax * Update file storage with classes * Fix scanning individual files and add reflection return types * Always turn XDebug off * Add quicker method of getting method mutations * Queue return types for crawling * Interpret all strings as possible classes once we see a `get_class` call * Check invalid return types again * Fix template namespacing issues * Default to class-insensitive file names for includes * Don’t overwrite existing issues data * Add var docblocks for scanning * Add null check * Fix loading of external classes in templates * Only try to populate class when we haven’t yet seen it’s not a class * Fix trait property accessibility * Only ever improve docblock param type * Make param replacement more robust * Fix static const missing inferred type * Fix a few more tests * Register constant definitions * Fix trait aliasing * Skip constant type tests for now * Fix linting issues * Make sure caching is off for tests * Remove unnecessary return * Use emulative parser if on PHP 5.6 * Cache parser for faster first-time parse * Fix constant resolution when scanning classes * Remove test that’s beyond a practical scope * Add back --diff support * Add --help for --threads * Remove unused vars
2017-07-25 22:11:02 +02:00
self::$files_inheriting_classes[$fq_class_name_lc][$source_file] = true;
}
/**
* @return void
*/
public function addMethodParamUse(string $method_id, int $offset, string $referencing_method_id)
{
self::$method_param_uses[$method_id][$offset][$referencing_method_id] = true;
}
/**
* @param string $file
2017-05-27 02:16:18 +02:00
*
* @return array<int, string>
*/
2018-10-07 02:11:19 +02:00
private function calculateFilesReferencingFile(Codebase $codebase, $file)
{
$referenced_files = [];
2018-11-06 03:57:36 +01:00
$file_classes = ClassLikeAnalyzer::getClassesForFile($codebase, $file);
foreach ($file_classes as $file_class_lc => $_) {
if (isset(self::$nonmethod_references_to_classes[$file_class_lc])) {
$new_files = array_keys(self::$nonmethod_references_to_classes[$file_class_lc]);
$referenced_files = array_merge(
$referenced_files,
$new_files
);
}
if (isset(self::$method_references_to_classes[$file_class_lc])) {
$new_referencing_methods = array_keys(self::$method_references_to_classes[$file_class_lc]);
foreach ($new_referencing_methods as $new_referencing_method_id) {
$fq_class_name_lc = \explode('::', $new_referencing_method_id)[0];
try {
$referenced_files[] = $codebase->scanner->getClassLikeFilePath($fq_class_name_lc);
} catch (\UnexpectedValueException $e) {
if (isset(self::$classlike_files[$fq_class_name_lc])) {
$referenced_files[] = self::$classlike_files[$fq_class_name_lc];
}
}
}
}
}
return array_unique($referenced_files);
}
/**
* @param string $file
2017-05-27 02:16:18 +02:00
*
* @return array<int, string>
*/
2018-10-07 02:11:19 +02:00
private function calculateFilesInheritingFile(Codebase $codebase, $file)
{
$referenced_files = [];
2018-11-06 03:57:36 +01:00
$file_classes = ClassLikeAnalyzer::getClassesForFile($codebase, $file);
foreach ($file_classes as $file_class_lc => $_) {
if (isset(self::$files_inheriting_classes[$file_class_lc])) {
$referenced_files = array_merge(
$referenced_files,
array_keys(self::$files_inheriting_classes[$file_class_lc])
);
}
}
return array_unique($referenced_files);
}
/**
* @return void
*/
public function removeDeletedFilesFromReferences()
{
$deleted_files = self::getDeletedReferencedFiles();
if ($deleted_files) {
foreach ($deleted_files as $file) {
unset(self::$file_references[$file]);
}
if ($this->cache) {
$this->cache->setCachedFileReferences(self::$file_references);
}
}
}
/**
* @param string $file
2017-05-27 02:16:18 +02:00
*
* @return array<string>
*/
public function getFilesReferencingFile($file)
{
return isset(self::$file_references[$file]['a']) ? self::$file_references[$file]['a'] : [];
}
/**
* @param string $file
2017-05-27 02:16:18 +02:00
*
* @return array<string>
*/
public function getFilesInheritingFromFile($file)
{
return isset(self::$file_references[$file]['i']) ? self::$file_references[$file]['i'] : [];
}
/**
* @return array<string, array<string, bool>>
*/
public function getAllMethodReferencesToClassMembers()
{
return self::$method_references_to_class_members;
}
/**
* @return array<string, array<string, bool>>
*/
public function getAllMethodReferencesToClasses()
{
return self::$method_references_to_classes;
}
/**
* @return array<string, array<string, bool>>
*/
public function getAllMethodReferencesToMissingClassMembers()
{
return self::$method_references_to_missing_class_members;
}
2019-04-27 23:38:24 +02:00
/**
* @return array<string, array<string,bool>>
*/
public function getAllReferencesToMixedMemberNames()
{
return self::$references_to_mixed_member_names;
}
/**
* @return array<string, array<int, array<string, bool>>>
*/
public function getAllMethodParamUses()
{
return self::$method_param_uses;
}
/**
* @param bool $force_reload
2019-07-05 22:24:00 +02:00
*
* @return bool
* @psalm-suppress MixedPropertyTypeCoercion
*/
public function loadReferenceCache($force_reload = true)
{
if ($this->cache && (!$this->loaded_from_cache || $force_reload)) {
2018-09-30 05:51:06 +02:00
$this->loaded_from_cache = true;
$file_references = $this->cache->getCachedFileReferences();
if ($file_references === null) {
return false;
}
self::$file_references = $file_references;
$nonmethod_references_to_classes = $this->cache->getCachedNonMethodClassReferences();
if ($nonmethod_references_to_classes === null) {
return false;
}
self::$nonmethod_references_to_classes = $nonmethod_references_to_classes;
$method_references_to_classes = $this->cache->getCachedMethodClassReferences();
if ($method_references_to_classes === null) {
return false;
}
self::$method_references_to_classes = $method_references_to_classes;
$method_references_to_class_members = $this->cache->getCachedMethodMemberReferences();
if ($method_references_to_class_members === null) {
return false;
}
self::$method_references_to_class_members = $method_references_to_class_members;
$method_references_to_missing_class_members = $this->cache->getCachedMethodMissingMemberReferences();
if ($method_references_to_missing_class_members === null) {
return false;
}
self::$method_references_to_missing_class_members = $method_references_to_missing_class_members;
$file_references_to_class_members = $this->cache->getCachedFileMemberReferences();
2017-05-25 04:07:49 +02:00
if ($file_references_to_class_members === null) {
return false;
}
self::$file_references_to_class_members = $file_references_to_class_members;
$file_references_to_missing_class_members = $this->cache->getCachedFileMissingMemberReferences();
if ($file_references_to_missing_class_members === null) {
return false;
}
self::$file_references_to_missing_class_members = $file_references_to_missing_class_members;
2019-04-27 23:38:24 +02:00
$references_to_mixed_member_names = $this->cache->getCachedMixedMemberNameReferences();
if ($references_to_mixed_member_names === null) {
return false;
}
self::$references_to_mixed_member_names = $references_to_mixed_member_names;
2018-11-02 02:52:39 +01:00
$analyzed_methods = $this->cache->getAnalyzedMethodCache();
2018-10-07 02:11:19 +02:00
2018-11-02 02:52:39 +01:00
if ($analyzed_methods === false) {
2018-10-07 02:11:19 +02:00
return false;
}
2018-11-02 02:52:39 +01:00
self::$analyzed_methods = $analyzed_methods;
2018-10-07 02:11:19 +02:00
$issues = $this->cache->getCachedIssues();
if ($issues === null) {
return false;
}
self::$issues = $issues;
$method_param_uses = $this->cache->getCachedMethodParamUses();
if ($method_param_uses === null) {
return false;
}
self::$method_param_uses = $method_param_uses;
$mixed_counts = $this->cache->getTypeCoverage();
if ($mixed_counts === false) {
return false;
}
self::$mixed_counts = $mixed_counts;
$classlike_files = $this->cache->getCachedClassLikeFiles();
if ($classlike_files === null) {
return false;
}
self::$classlike_files = $classlike_files;
self::$file_maps = $this->cache->getFileMapCache() ?: [];
return true;
}
return false;
}
/**
* @param array<string, string|bool> $visited_files
2017-05-27 02:16:18 +02:00
*
* @return void
*/
2018-10-07 02:11:19 +02:00
public function updateReferenceCache(Codebase $codebase, array $visited_files)
{
foreach ($visited_files as $file => $_) {
$all_file_references = array_unique(
array_merge(
isset(self::$file_references[$file]['a']) ? self::$file_references[$file]['a'] : [],
2018-10-07 02:11:19 +02:00
$this->calculateFilesReferencingFile($codebase, $file)
)
);
$inheritance_references = array_unique(
array_merge(
isset(self::$file_references[$file]['i']) ? self::$file_references[$file]['i'] : [],
2018-10-07 02:11:19 +02:00
$this->calculateFilesInheritingFile($codebase, $file)
)
);
self::$file_references[$file] = [
'a' => $all_file_references,
'i' => $inheritance_references,
];
}
if ($this->cache) {
$this->cache->setCachedFileReferences(self::$file_references);
$this->cache->setCachedMethodClassReferences(self::$method_references_to_classes);
$this->cache->setCachedNonMethodClassReferences(self::$nonmethod_references_to_classes);
$this->cache->setCachedMethodMemberReferences(self::$method_references_to_class_members);
$this->cache->setCachedFileMemberReferences(self::$file_references_to_class_members);
$this->cache->setCachedMethodMissingMemberReferences(self::$method_references_to_missing_class_members);
$this->cache->setCachedFileMissingMemberReferences(self::$file_references_to_missing_class_members);
2019-04-27 23:38:24 +02:00
$this->cache->setCachedMixedMemberNameReferences(self::$references_to_mixed_member_names);
$this->cache->setCachedMethodParamUses(self::$method_param_uses);
$this->cache->setCachedIssues(self::$issues);
$this->cache->setCachedClassLikeFiles(self::$classlike_files);
$this->cache->setFileMapCache(self::$file_maps);
$this->cache->setTypeCoverage(self::$mixed_counts);
2018-11-02 02:52:39 +01:00
$this->cache->setAnalyzedMethodCache(self::$analyzed_methods);
}
}
/**
* @param lowercase-string $fq_class_name_lc
* @return void
*/
public function addMethodReferenceToClass(string $calling_function_id, string $fq_class_name_lc)
{
if (!isset(self::$method_references_to_classes[$fq_class_name_lc])) {
self::$method_references_to_classes[$fq_class_name_lc] = [$calling_function_id => true];
} else {
self::$method_references_to_classes[$fq_class_name_lc][$calling_function_id] = true;
}
}
/**
* @return void
*/
public function addMethodReferenceToClassMember(string $calling_function_id, string $referenced_member_id)
{
if (!isset(self::$method_references_to_class_members[$referenced_member_id])) {
self::$method_references_to_class_members[$referenced_member_id] = [$calling_function_id => true];
} else {
self::$method_references_to_class_members[$referenced_member_id][$calling_function_id] = true;
}
}
/**
* @return void
*/
public function addMethodReferenceToMissingClassMember(string $calling_function_id, string $referenced_member_id)
{
if (!isset(self::$method_references_to_missing_class_members[$referenced_member_id])) {
self::$method_references_to_missing_class_members[$referenced_member_id] = [$calling_function_id => true];
} else {
self::$method_references_to_missing_class_members[$referenced_member_id][$calling_function_id] = true;
}
}
/**
* @return void
*/
public function addCallingLocationForClassMethod(CodeLocation $code_location, string $referenced_member_id)
{
if (!isset(self::$class_method_locations[$referenced_member_id])) {
self::$class_method_locations[$referenced_member_id] = [$code_location];
} else {
self::$class_method_locations[$referenced_member_id][] = $code_location;
}
}
/**
* @return void
*/
public function addCallingLocationForClassProperty(CodeLocation $code_location, string $referenced_property_id)
{
if (!isset(self::$class_property_locations[$referenced_property_id])) {
self::$class_property_locations[$referenced_property_id] = [$code_location];
} else {
self::$class_property_locations[$referenced_property_id][] = $code_location;
}
}
/**
* @return void
*/
public function addCallingLocationForClass(CodeLocation $code_location, string $referenced_class)
{
if (!isset(self::$class_locations[$referenced_class])) {
self::$class_locations[$referenced_class] = [$code_location];
} else {
self::$class_locations[$referenced_class][] = $code_location;
}
}
public function isClassMethodReferenced(string $method_id) : bool
{
return !empty(self::$file_references_to_class_members[$method_id])
|| !empty(self::$method_references_to_class_members[$method_id]);
}
public function isClassPropertyReferenced(string $property_id) : bool
{
return !empty(self::$file_references_to_class_members[$property_id])
|| !empty(self::$method_references_to_class_members[$property_id]);
}
public function isClassReferenced(string $fq_class_name_lc) : bool
{
return isset(self::$method_references_to_classes[$fq_class_name_lc])
|| isset(self::$nonmethod_references_to_classes[$fq_class_name_lc]);
}
public function isMethodParamUsed(string $method_id, int $offset) : bool
{
return !empty(self::$method_param_uses[$method_id][$offset]);
}
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function setNonMethodReferencesToClasses(array $references)
{
self::$nonmethod_references_to_classes = $references;
}
/**
* @return array<string, array<int, CodeLocation>>
*/
public function getAllClassMethodLocations() : array
{
return self::$class_method_locations;
}
/**
* @return array<string, array<int, CodeLocation>>
*/
public function getAllClassPropertyLocations() : array
{
return self::$class_property_locations;
}
/**
* @return array<string, array<int, CodeLocation>>
*/
public function getAllClassLocations() : array
{
return self::$class_locations;
}
/**
* @return array<int, CodeLocation>
*/
public function getClassMethodLocations(string $method_id) : array
{
return self::$class_method_locations[$method_id] ?? [];
}
/**
* @return array<int, CodeLocation>
*/
public function getClassPropertyLocations(string $property_id) : array
{
return self::$class_property_locations[$property_id] ?? [];
}
/**
* @return array<int, CodeLocation>
*/
public function getClassLocations(string $fq_class_name_lc) : array
{
return self::$class_locations[$fq_class_name_lc] ?? [];
}
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function addMethodReferencesToClassMembers(array $references)
{
foreach ($references as $key => $reference) {
if (isset(self::$method_references_to_class_members[$key])) {
self::$method_references_to_class_members[$key] = array_merge(
$reference,
self::$method_references_to_class_members[$key]
);
} else {
self::$method_references_to_class_members[$key] = $reference;
}
}
}
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function addMethodReferencesToClasses(array $references)
{
foreach ($references as $key => $reference) {
if (isset(self::$method_references_to_classes[$key])) {
self::$method_references_to_classes[$key] = array_merge(
$reference,
self::$method_references_to_classes[$key]
);
} else {
self::$method_references_to_classes[$key] = $reference;
}
}
}
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function addMethodReferencesToMissingClassMembers(array $references)
{
foreach ($references as $key => $reference) {
if (isset(self::$method_references_to_missing_class_members[$key])) {
self::$method_references_to_missing_class_members[$key] = array_merge(
$reference,
self::$method_references_to_missing_class_members[$key]
);
} else {
self::$method_references_to_missing_class_members[$key] = $reference;
}
}
}
/**
* @param array<string, array<int, array<string, bool>>> $references
*
* @return void
*/
public function addMethodParamUses(array $references)
{
foreach ($references as $method_id => $method_param_uses) {
if (isset(self::$method_param_uses[$method_id])) {
foreach ($method_param_uses as $offset => $reference_map) {
if (isset(self::$method_param_uses[$method_id][$offset])) {
self::$method_param_uses[$method_id][$offset] = array_merge(
self::$method_param_uses[$method_id][$offset],
$reference_map
);
} else {
self::$method_param_uses[$method_id][$offset] = $reference_map;
}
}
} else {
self::$method_param_uses[$method_id] = $method_param_uses;
}
}
}
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function setCallingMethodReferencesToClasses(array $references)
{
self::$method_references_to_classes = $references;
}
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function setCallingMethodReferencesToClassMembers(array $references)
{
self::$method_references_to_class_members = $references;
}
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function setCallingMethodReferencesToMissingClassMembers(array $references)
{
self::$method_references_to_missing_class_members = $references;
}
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function setFileReferencesToClassMembers(array $references)
{
self::$file_references_to_class_members = $references;
}
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function setFileReferencesToMissingClassMembers(array $references)
{
self::$file_references_to_missing_class_members = $references;
}
2019-04-27 23:38:24 +02:00
/**
* @param array<string, array<string,bool>> $references
*
* @return void
*/
public function setReferencesToMixedMemberNames(array $references)
{
self::$references_to_mixed_member_names = $references;
}
/**
* @param array<string, array<int, array<string, bool>>> $references
*
* @return void
*/
public function setMethodParamUses(array $references)
{
self::$method_param_uses = $references;
}
/**
* @param array<string, array<int, CodeLocation>> $references
*
* @return void
*/
public function addClassMethodLocations(array $references)
{
foreach ($references as $referenced_member_id => $locations) {
if (isset(self::$class_method_locations[$referenced_member_id])) {
self::$class_method_locations[$referenced_member_id] = array_merge(
self::$class_method_locations[$referenced_member_id],
$locations
);
} else {
self::$class_method_locations[$referenced_member_id] = $locations;
}
}
}
/**
* @param array<string, array<int, CodeLocation>> $references
*
* @return void
*/
public function addClassPropertyLocations(array $references)
{
foreach ($references as $referenced_member_id => $locations) {
if (isset(self::$class_property_locations[$referenced_member_id])) {
self::$class_property_locations[$referenced_member_id] = array_merge(
self::$class_property_locations[$referenced_member_id],
$locations
);
} else {
self::$class_property_locations[$referenced_member_id] = $locations;
}
}
}
/**
* @param array<string, array<int, CodeLocation>> $references
*
* @return void
*/
public function addClassLocations(array $references)
{
foreach ($references as $referenced_member_id => $locations) {
if (isset(self::$class_locations[$referenced_member_id])) {
self::$class_locations[$referenced_member_id] = array_merge(
self::$class_locations[$referenced_member_id],
$locations
);
} else {
self::$class_locations[$referenced_member_id] = $locations;
}
}
}
/**
* @return array<string, array<int, IssueData>>
*/
public function getExistingIssues() : array
{
return self::$issues;
}
/**
2018-09-30 05:51:06 +02:00
* @param string $file_path
2019-07-05 22:24:00 +02:00
*
* @return void
*/
2018-09-30 05:51:06 +02:00
public function clearExistingIssuesForFile($file_path)
{
2018-09-30 05:51:06 +02:00
unset(self::$issues[$file_path]);
}
/**
2018-09-27 19:32:08 +02:00
* @param string $file_path
* @param IssueData $issue
2019-07-05 22:24:00 +02:00
*
* @return void
*/
public function clearExistingFileMapsForFile($file_path)
{
unset(self::$file_maps[$file_path]);
}
/**
* @param string $file_path
* @param IssueData $issue
2019-07-05 22:24:00 +02:00
*
* @return void
*/
2020-02-17 00:24:40 +01:00
public function addIssue($file_path, IssueData $issue)
{
// dont save parse errors ever, as they're not responsive to AST diffing
2020-02-17 00:24:40 +01:00
if ($issue->type === 'ParseError') {
return;
}
if (!isset(self::$issues[$file_path])) {
self::$issues[$file_path] = [$issue];
} else {
self::$issues[$file_path][] = $issue;
}
}
2018-09-30 05:51:06 +02:00
/**
2018-11-02 02:52:39 +01:00
* @param array<string, array<string, int>> $analyzed_methods
2019-07-05 22:24:00 +02:00
*
2018-10-07 02:11:19 +02:00
* @return void
2018-09-30 05:51:06 +02:00
*/
2018-11-02 02:52:39 +01:00
public function setAnalyzedMethods(array $analyzed_methods)
2018-09-30 05:51:06 +02:00
{
2018-11-02 02:52:39 +01:00
self::$analyzed_methods = $analyzed_methods;
2018-09-30 05:51:06 +02:00
}
/**
* @param array<string, FileMapType> $file_maps
*/
public function setFileMaps(array $file_maps) : void
{
self::$file_maps = $file_maps;
}
/**
* @return array<string, array{int, int}>
*/
public function getTypeCoverage()
{
return self::$mixed_counts;
}
/**
* @param array<string, array{int, int}> $mixed_counts
2019-07-05 22:24:00 +02:00
*
* @return void
*/
public function setTypeCoverage(array $mixed_counts)
{
self::$mixed_counts = array_merge(self::$mixed_counts, $mixed_counts);
}
2018-09-30 05:51:06 +02:00
/**
* @return array<string, array<string, int>>
2018-09-30 05:51:06 +02:00
*/
2018-11-02 02:52:39 +01:00
public function getAnalyzedMethods()
2018-09-30 05:51:06 +02:00
{
2018-11-02 02:52:39 +01:00
return self::$analyzed_methods;
2018-09-30 05:51:06 +02:00
}
/**
* @return array<string, FileMapType>
*/
public function getFileMaps()
{
return self::$file_maps;
}
/**
* @return void
*/
public static function clearCache()
{
self::$files_inheriting_classes = [];
self::$deleted_files = null;
self::$file_references = [];
self::$file_references_to_class_members = [];
self::$method_references_to_class_members = [];
self::$method_references_to_classes = [];
self::$nonmethod_references_to_classes = [];
self::$file_references_to_missing_class_members = [];
self::$method_references_to_missing_class_members = [];
2019-04-27 23:38:24 +02:00
self::$references_to_mixed_member_names = [];
self::$class_method_locations = [];
self::$class_property_locations = [];
Test parallelization (#4045) * Run tests in random order Being able to run tests in any order is a pre-requisite for being able to run them in parallel. * Reset type coverage between tests, fix affected tests * Reset parser and lexer between test runs and on php version change Previously lexer was reset, but parser kept the reference to the old one, and reference to the parser was kept by StatementsProvider. This resulted in order-dependent tests - if the parser was first initialized with phpVersion set to 7.4 then arrow functions worked fine, but were failing when the parser was initially constructed with settings for 7.3 This can be demonstrated on current master by upgrading to nikic/php-parser:4.9 and running: ``` vendor/bin/phpunit --no-coverage --filter="inferredArgArrowFunction" tests/ClosureTest.php ``` Now all tests using PHP 7.4 features must set the PHP version accordingly. * Marked more tests using 7.4 syntax * Reset newline-between-annotation flag between tests * Resolve real paths before passing them to checkPaths When checkPaths is called from psalm.php the paths are resolved, so we just mimicking SUT behaviour here. * Restore newline-between-annotations in DocCommentTest * Tweak Appveyor caches * Tweak TravisCI caches * Tweak CircleCI caches * Run tests in parallel Use `vendor/bin/paratest` instead of `vendor/bin/phpunit` * Use default paratest runner on Windows WrapperRunner is not supported on Windows. * TRAVIS_TAG could be empty * Restore appveyor conditional caching
2020-08-23 16:32:07 +02:00
self::$class_locations = [];
2018-11-02 02:52:39 +01:00
self::$analyzed_methods = [];
self::$issues = [];
self::$file_maps = [];
self::$method_param_uses = [];
self::$classlike_files = [];
Test parallelization (#4045) * Run tests in random order Being able to run tests in any order is a pre-requisite for being able to run them in parallel. * Reset type coverage between tests, fix affected tests * Reset parser and lexer between test runs and on php version change Previously lexer was reset, but parser kept the reference to the old one, and reference to the parser was kept by StatementsProvider. This resulted in order-dependent tests - if the parser was first initialized with phpVersion set to 7.4 then arrow functions worked fine, but were failing when the parser was initially constructed with settings for 7.3 This can be demonstrated on current master by upgrading to nikic/php-parser:4.9 and running: ``` vendor/bin/phpunit --no-coverage --filter="inferredArgArrowFunction" tests/ClosureTest.php ``` Now all tests using PHP 7.4 features must set the PHP version accordingly. * Marked more tests using 7.4 syntax * Reset newline-between-annotation flag between tests * Resolve real paths before passing them to checkPaths When checkPaths is called from psalm.php the paths are resolved, so we just mimicking SUT behaviour here. * Restore newline-between-annotations in DocCommentTest * Tweak Appveyor caches * Tweak TravisCI caches * Tweak CircleCI caches * Run tests in parallel Use `vendor/bin/paratest` instead of `vendor/bin/phpunit` * Use default paratest runner on Windows WrapperRunner is not supported on Windows. * TRAVIS_TAG could be empty * Restore appveyor conditional caching
2020-08-23 16:32:07 +02:00
self::$mixed_counts = [];
}
}