1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-10 06:58:41 +01:00
psalm/src/Psalm/Codebase.php

2381 lines
76 KiB
PHP
Raw Normal View History

<?php
namespace Psalm;
2021-12-03 21:40:18 +01:00
use Exception;
2022-02-15 01:18:49 +01:00
use InvalidArgumentException;
2019-07-05 22:24:00 +02:00
use LanguageServerProtocol\Command;
2021-12-03 20:11:20 +01:00
use LanguageServerProtocol\CompletionItem;
use LanguageServerProtocol\CompletionItemKind;
use LanguageServerProtocol\InsertTextFormat;
use LanguageServerProtocol\ParameterInformation;
2019-07-05 22:24:00 +02:00
use LanguageServerProtocol\Position;
use LanguageServerProtocol\Range;
2021-12-03 20:11:20 +01:00
use LanguageServerProtocol\SignatureInformation;
use LanguageServerProtocol\TextEdit;
use PhpParser;
2021-12-04 03:37:19 +01:00
use PhpParser\Node\Arg;
use Psalm\CodeLocation\Raw;
2021-12-03 20:29:06 +01:00
use Psalm\Exception\UnanalyzedFileException;
2021-12-04 03:37:19 +01:00
use Psalm\Exception\UnpopulatedClasslikeException;
2022-02-15 01:18:49 +01:00
use Psalm\Internal\Analyzer\FunctionLikeAnalyzer;
use Psalm\Internal\Analyzer\NamespaceAnalyzer;
2022-02-15 01:18:49 +01:00
use Psalm\Internal\Analyzer\ProjectAnalyzer;
2019-07-05 22:24:00 +02:00
use Psalm\Internal\Analyzer\Statements\Block\ForeachAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\Fetch\ConstFetchAnalyzer;
use Psalm\Internal\Analyzer\Statements\Expression\Fetch\VariableFetchAnalyzer;
2021-06-08 04:55:21 +02:00
use Psalm\Internal\Analyzer\StatementsAnalyzer;
2021-12-04 03:37:19 +01:00
use Psalm\Internal\Codebase\Analyzer;
use Psalm\Internal\Codebase\ClassLikes;
use Psalm\Internal\Codebase\Functions;
use Psalm\Internal\Codebase\InternalCallMapHandler;
2021-12-04 03:37:19 +01:00
use Psalm\Internal\Codebase\Methods;
use Psalm\Internal\Codebase\Populator;
use Psalm\Internal\Codebase\Properties;
2021-12-03 20:11:20 +01:00
use Psalm\Internal\Codebase\Reflection;
2021-12-04 03:37:19 +01:00
use Psalm\Internal\Codebase\Scanner;
use Psalm\Internal\Codebase\TaintFlowGraph;
2021-12-03 20:11:20 +01:00
use Psalm\Internal\DataFlow\TaintSink;
use Psalm\Internal\DataFlow\TaintSource;
2022-12-20 21:52:25 +01:00
use Psalm\Internal\LanguageServer\PHPMarkdownContent;
use Psalm\Internal\LanguageServer\Reference;
2021-12-03 20:11:20 +01:00
use Psalm\Internal\MethodIdentifier;
2018-11-06 03:57:36 +01:00
use Psalm\Internal\Provider\ClassLikeStorageProvider;
use Psalm\Internal\Provider\FileProvider;
use Psalm\Internal\Provider\FileReferenceProvider;
use Psalm\Internal\Provider\FileStorageProvider;
use Psalm\Internal\Provider\Providers;
use Psalm\Internal\Provider\StatementsProvider;
2021-06-08 04:55:21 +02:00
use Psalm\Internal\Type\Comparator\UnionTypeComparator;
use Psalm\Progress\Progress;
use Psalm\Progress\VoidProgress;
use Psalm\Storage\ClassLikeStorage;
use Psalm\Storage\FileStorage;
2021-12-04 03:37:19 +01:00
use Psalm\Storage\FunctionLikeParameter;
use Psalm\Storage\FunctionLikeStorage;
use Psalm\Storage\FunctionStorage;
use Psalm\Storage\MethodStorage;
2021-12-13 16:28:14 +01:00
use Psalm\Type\Atomic;
2021-12-13 04:45:57 +01:00
use Psalm\Type\Atomic\TBool;
use Psalm\Type\Atomic\TClassConstant;
use Psalm\Type\Atomic\TKeyedArray;
use Psalm\Type\Atomic\TLiteralInt;
use Psalm\Type\Atomic\TLiteralString;
use Psalm\Type\Atomic\TNamedObject;
2021-12-03 20:11:20 +01:00
use Psalm\Type\TaintKindGroup;
2021-12-13 16:28:14 +01:00
use Psalm\Type\Union;
2021-12-03 21:40:18 +01:00
use ReflectionProperty;
use ReflectionType;
use UnexpectedValueException;
2021-06-08 04:55:21 +02:00
2022-02-15 01:18:49 +01:00
use function array_combine;
2021-06-08 04:55:21 +02:00
use function array_merge;
use function array_pop;
use function array_reverse;
2022-12-20 21:52:25 +01:00
use function array_values;
2021-06-08 04:55:21 +02:00
use function count;
use function dirname;
use function error_log;
use function explode;
use function implode;
2022-02-15 01:18:49 +01:00
use function in_array;
use function intdiv;
2021-12-03 21:07:25 +01:00
use function is_numeric;
use function is_string;
2021-06-08 04:55:21 +02:00
use function krsort;
use function ksort;
use function preg_match;
2021-12-03 21:07:25 +01:00
use function preg_replace;
2022-12-20 21:52:25 +01:00
use function str_replace;
2019-07-05 22:24:00 +02:00
use function strlen;
use function strpos;
2019-07-05 22:24:00 +02:00
use function strrpos;
use function strtolower;
use function substr;
use function substr_count;
2021-06-08 04:55:21 +02:00
use const PHP_VERSION_ID;
final class Codebase
{
/**
2018-04-05 20:40:41 +02:00
* @var Config
*/
public $config;
/**
* A map of fully-qualified use declarations to the files
* that reference them (keyed by filename)
*
2021-12-04 03:37:19 +01:00
* @var array<lowercase-string, array<int, CodeLocation>>
*/
public $use_referencing_locations = [];
/**
* @var FileStorageProvider
*/
public $file_storage_provider;
/**
* @var ClassLikeStorageProvider
*/
public $classlike_storage_provider;
/**
* @var bool
*/
public $collect_references = false;
/**
* @var bool
*/
public $collect_locations = false;
/**
* @var null|'always'|'auto'
*/
2021-09-26 23:24:07 +02:00
public $find_unused_code;
/**
* @var FileProvider
*/
public $file_provider;
/**
* @var FileReferenceProvider
*/
public $file_reference_provider;
/**
* @var StatementsProvider
*/
public $statements_provider;
2022-12-11 23:26:05 +01:00
private Progress $progress;
/**
2021-12-13 16:28:14 +01:00
* @var array<string, Union>
*/
2018-02-04 00:52:35 +01:00
private static $stubbed_constants = [];
/**
* Whether to register autoloaded information
*
* @var bool
*/
public $register_autoload_files = false;
/**
2018-02-04 00:52:35 +01:00
* Whether to log functions just at the file level or globally (for stubs)
*
* @var bool
*/
public $register_stub_files = false;
/**
* @var bool
*/
public $find_unused_variables = false;
/**
2021-12-04 03:37:19 +01:00
* @var Scanner
*/
2018-02-04 00:52:35 +01:00
public $scanner;
/**
2021-12-04 03:37:19 +01:00
* @var Analyzer
*/
2018-02-04 00:52:35 +01:00
public $analyzer;
/**
2021-12-04 03:37:19 +01:00
* @var Functions
*/
2018-02-04 00:52:35 +01:00
public $functions;
/**
2021-12-04 03:37:19 +01:00
* @var ClassLikes
*/
2018-02-04 00:52:35 +01:00
public $classlikes;
/**
2021-12-04 03:37:19 +01:00
* @var Methods
*/
2018-02-04 00:52:35 +01:00
public $methods;
/**
2021-12-04 03:37:19 +01:00
* @var Properties
*/
public $properties;
/**
2021-12-04 03:37:19 +01:00
* @var Populator
*/
2018-02-04 00:52:35 +01:00
public $populator;
/**
2021-12-04 03:37:19 +01:00
* @var ?TaintFlowGraph
*/
2021-09-26 23:24:07 +02:00
public $taint_flow_graph;
/**
* @var bool
*/
public $server_mode = false;
2019-02-24 07:33:25 +01:00
/**
* @var bool
*/
public $store_node_types = false;
2018-11-06 03:57:36 +01:00
/**
* Whether or not to infer types from usage. Computationally expensive, so turned off by default
*
* @var bool
*/
public $infer_types_from_usage = false;
/**
* @var bool
*/
public $alter_code = false;
/**
* @var bool
*/
public $diff_methods = false;
/**
* @var array<lowercase-string, string>
*/
2019-06-02 18:02:32 +02:00
public $methods_to_move = [];
/**
* @var array<lowercase-string, string>
2019-06-02 18:02:32 +02:00
*/
public $methods_to_rename = [];
2019-06-04 06:32:19 +02:00
/**
* @var array<string, string>
*/
public $properties_to_move = [];
/**
* @var array<string, string>
*/
public $properties_to_rename = [];
/**
* @var array<string, string>
*/
public $class_constants_to_move = [];
/**
* @var array<string, string>
*/
public $class_constants_to_rename = [];
2019-06-04 22:36:32 +02:00
/**
* @var array<lowercase-string, string>
2019-06-04 22:36:32 +02:00
*/
public $classes_to_move = [];
/**
* @var array<lowercase-string, string>
*/
public $call_transforms = [];
2019-06-04 06:32:19 +02:00
/**
* @var array<string, string>
*/
public $property_transforms = [];
/**
* @var array<string, string>
*/
public $class_constant_transforms = [];
2019-06-04 22:36:32 +02:00
/**
2020-03-15 04:54:42 +01:00
* @var array<lowercase-string, string>
2019-06-04 22:36:32 +02:00
*/
public $class_transforms = [];
/**
* @var bool
*/
public $allow_backwards_incompatible_changes = true;
/** @var int */
public $analysis_php_version_id = PHP_VERSION_ID;
/** @var 'cli'|'config'|'composer'|'tests'|'runtime' */
public $php_version_source = 'runtime';
2019-08-18 20:27:50 +02:00
/**
* @var bool
*/
public $track_unused_suppressions = false;
/** @internal */
public function __construct(
Config $config,
Providers $providers,
?Progress $progress = null
) {
if ($progress === null) {
$progress = new VoidProgress();
}
$this->config = $config;
$this->file_storage_provider = $providers->file_storage_provider;
$this->classlike_storage_provider = $providers->classlike_storage_provider;
$this->progress = $progress;
$this->file_provider = $providers->file_provider;
$this->file_reference_provider = $providers->file_reference_provider;
$this->statements_provider = $providers->statements_provider;
self::$stubbed_constants = [];
2021-12-14 01:54:11 +01:00
$reflection = new Reflection($providers->classlike_storage_provider, $this);
2021-12-04 03:37:19 +01:00
$this->scanner = new Scanner(
2018-02-04 00:52:35 +01:00
$this,
$config,
$providers->file_storage_provider,
$providers->file_provider,
2020-01-03 19:10:24 +01:00
$reflection,
$providers->file_reference_provider,
2022-12-18 17:15:15 +01:00
$progress,
2018-02-04 00:52:35 +01:00
);
$this->loadAnalyzer();
2021-12-04 03:37:19 +01:00
$this->functions = new Functions($providers->file_storage_provider, $reflection);
2021-12-04 03:37:19 +01:00
$this->classlikes = new ClassLikes(
2018-10-07 02:11:19 +02:00
$this->config,
$providers->classlike_storage_provider,
$providers->file_reference_provider,
2020-03-03 04:27:54 +01:00
$providers->statements_provider,
2022-12-18 17:15:15 +01:00
$this->scanner,
2018-02-04 00:52:35 +01:00
);
2021-12-04 03:37:19 +01:00
$this->properties = new Properties(
$providers->classlike_storage_provider,
$providers->file_reference_provider,
2022-12-18 17:15:15 +01:00
$this->classlikes,
);
2021-12-04 03:37:19 +01:00
$this->methods = new Methods(
$providers->classlike_storage_provider,
$providers->file_reference_provider,
2022-12-18 17:15:15 +01:00
$this->classlikes,
);
2021-12-04 03:37:19 +01:00
$this->populator = new Populator(
$providers->classlike_storage_provider,
$providers->file_storage_provider,
2018-02-04 00:52:35 +01:00
$this->classlikes,
$providers->file_reference_provider,
2022-12-18 17:15:15 +01:00
$progress,
2018-02-04 00:52:35 +01:00
);
2018-10-07 02:11:19 +02:00
$this->loadAnalyzer();
}
private function loadAnalyzer(): void
{
2021-12-04 03:37:19 +01:00
$this->analyzer = new Analyzer(
$this->config,
$this->file_provider,
$this->file_storage_provider,
2022-12-18 17:15:15 +01:00
$this->progress,
);
}
/**
* @param array<string> $candidate_files
*/
2022-12-20 21:52:25 +01:00
public function reloadFiles(ProjectAnalyzer $project_analyzer, array $candidate_files, bool $force = false): void
{
$this->loadAnalyzer();
2022-12-20 21:52:25 +01:00
if ($force) {
FileReferenceProvider::clearCache();
}
$this->file_reference_provider->loadReferenceCache(false);
FunctionLikeAnalyzer::clearCache();
2022-12-20 21:52:25 +01:00
if ($force || !$this->statements_provider->parser_cache_provider) {
$diff_files = $candidate_files;
} else {
$diff_files = [];
$parser_cache_provider = $this->statements_provider->parser_cache_provider;
foreach ($candidate_files as $candidate_file_path) {
if ($parser_cache_provider->loadExistingFileContentsFromCache($candidate_file_path)
!== $this->file_provider->getContents($candidate_file_path)
) {
$diff_files[] = $candidate_file_path;
}
}
}
2018-11-11 18:01:14 +01:00
$referenced_files = $project_analyzer->getReferencedFilesFromDiff($diff_files, false);
foreach ($diff_files as $diff_file_path) {
$this->invalidateInformationForFile($diff_file_path);
}
foreach ($referenced_files as $referenced_file_path) {
2019-07-05 22:24:00 +02:00
if (in_array($referenced_file_path, $diff_files, true)) {
continue;
}
$file_storage = $this->file_storage_provider->get($referenced_file_path);
foreach ($file_storage->classlikes_in_file as $fq_classlike_name) {
$this->classlike_storage_provider->remove($fq_classlike_name);
$this->classlikes->removeClassLike($fq_classlike_name);
}
$this->file_storage_provider->remove($referenced_file_path);
$this->scanner->removeFile($referenced_file_path);
}
$referenced_files = array_combine($referenced_files, $referenced_files);
$this->scanner->addFilesToDeepScan($referenced_files);
2019-01-02 18:10:52 +01:00
$this->addFilesToAnalyze(array_combine($candidate_files, $candidate_files));
$this->scanner->scanFiles($this->classlikes);
$this->file_reference_provider->updateReferenceCache($this, $referenced_files);
$this->populator->populateCodebase();
}
public function enterServerMode(): void
{
$this->server_mode = true;
2019-02-24 07:33:25 +01:00
$this->store_node_types = true;
}
public function collectLocations(): void
{
$this->collect_locations = true;
$this->classlikes->collect_locations = true;
$this->methods->collect_locations = true;
$this->properties->collect_locations = true;
}
/**
* @param 'always'|'auto' $find_unused_code
*/
public function reportUnusedCode(string $find_unused_code = 'auto'): void
{
$this->collect_references = true;
$this->classlikes->collect_references = true;
$this->find_unused_code = $find_unused_code;
$this->find_unused_variables = true;
}
public function reportUnusedVariables(): void
{
$this->collect_references = true;
$this->find_unused_variables = true;
}
/**
2018-02-04 00:52:35 +01:00
* @param array<string, string> $files_to_analyze
*/
public function addFilesToAnalyze(array $files_to_analyze): void
{
2018-02-04 00:52:35 +01:00
$this->scanner->addFilesToDeepScan($files_to_analyze);
$this->analyzer->addFilesToAnalyze($files_to_analyze);
}
/**
2018-02-04 00:52:35 +01:00
* Scans all files their related files
*/
public function scanFiles(int $threads = 1): void
{
$has_changes = $this->scanner->scanFiles($this->classlikes, $threads);
2018-02-04 00:52:35 +01:00
if ($has_changes) {
$this->populator->populateCodebase();
}
}
public function getFileContents(string $file_path): string
{
2018-02-04 00:52:35 +01:00
return $this->file_provider->getContents($file_path);
}
/**
2019-10-09 01:06:16 +02:00
* @return list<PhpParser\Node\Stmt>
*/
public function getStatementsForFile(string $file_path): array
{
2018-02-04 00:52:35 +01:00
return $this->statements_provider->getStatementsForFile(
$file_path,
$this->analysis_php_version_id,
2022-12-18 17:15:15 +01:00
$this->progress,
2018-02-04 00:52:35 +01:00
);
}
public function createClassLikeStorage(string $fq_classlike_name): ClassLikeStorage
{
2018-02-04 00:52:35 +01:00
return $this->classlike_storage_provider->create($fq_classlike_name);
}
public function cacheClassLikeStorage(ClassLikeStorage $classlike_storage, string $file_path): void
2018-02-19 06:27:39 +01:00
{
if (!$this->classlike_storage_provider->cache) {
return;
}
$file_contents = $this->file_provider->getContents($file_path);
$this->classlike_storage_provider->cache->writeToCache($classlike_storage, $file_path, $file_contents);
2018-02-19 06:27:39 +01:00
}
public function exhumeClassLikeStorage(string $fq_classlike_name, string $file_path): void
2018-02-19 06:27:39 +01:00
{
$file_contents = $this->file_provider->getContents($file_path);
$storage = $this->classlike_storage_provider->exhume(
$fq_classlike_name,
$file_path,
2022-12-18 17:15:15 +01:00
$file_contents,
);
2018-02-19 06:27:39 +01:00
if ($storage->is_trait) {
$this->classlikes->addFullyQualifiedTraitName($storage->name, $file_path);
2018-02-19 06:27:39 +01:00
} elseif ($storage->is_interface) {
$this->classlikes->addFullyQualifiedInterfaceName($storage->name, $file_path);
2018-02-19 06:27:39 +01:00
} else {
$this->classlikes->addFullyQualifiedClassName($storage->name, $file_path);
2018-02-19 06:27:39 +01:00
}
}
2021-12-13 16:28:14 +01:00
public static function getPsalmTypeFromReflection(?ReflectionType $type): Union
{
2021-12-03 20:11:20 +01:00
return Reflection::getPsalmTypeFromReflectionType($type);
}
public function createFileStorageForPath(string $file_path): FileStorage
{
2018-02-04 00:52:35 +01:00
return $this->file_storage_provider->create($file_path);
}
/**
2020-10-17 18:36:44 +02:00
* @return array<int, CodeLocation>
*/
public function findReferencesToSymbol(string $symbol): array
{
if (!$this->collect_locations) {
2021-12-03 21:40:18 +01:00
throw new UnexpectedValueException('Should not be checking references');
2018-02-04 00:52:35 +01:00
}
2019-01-22 17:10:37 +01:00
if (strpos($symbol, '::$') !== false) {
return $this->findReferencesToProperty($symbol);
}
2018-02-04 00:52:35 +01:00
if (strpos($symbol, '::') !== false) {
return $this->findReferencesToMethod($symbol);
}
return $this->findReferencesToClassLike($symbol);
}
/**
2020-10-17 18:36:44 +02:00
* @return array<int, CodeLocation>
*/
public function findReferencesToMethod(string $method_id): array
{
return $this->file_reference_provider->getClassMethodLocations(strtolower($method_id));
}
2019-01-22 17:10:37 +01:00
/**
2020-10-17 18:36:44 +02:00
* @return array<int, CodeLocation>
2019-01-22 17:10:37 +01:00
*/
public function findReferencesToProperty(string $property_id): array
2019-01-22 17:10:37 +01:00
{
2022-12-20 21:52:25 +01:00
/** @psalm-suppress PossiblyUndefinedIntArrayOffset */
[$fq_class_name, $property_name] = explode('::', $property_id);
2019-07-05 22:24:00 +02:00
return $this->file_reference_provider->getClassPropertyLocations(
2022-12-18 17:15:15 +01:00
strtolower($fq_class_name) . '::' . $property_name,
);
2019-01-22 17:10:37 +01:00
}
/**
2020-10-17 18:36:44 +02:00
* @return CodeLocation[]
* @psalm-return array<int, CodeLocation>
*/
public function findReferencesToClassLike(string $fq_class_name): array
{
2018-02-04 00:52:35 +01:00
$fq_class_name_lc = strtolower($fq_class_name);
$locations = $this->file_reference_provider->getClassLocations($fq_class_name_lc);
2018-02-04 00:52:35 +01:00
if (isset($this->use_referencing_locations[$fq_class_name_lc])) {
$locations = [...$locations, ...$this->use_referencing_locations[$fq_class_name_lc]];
}
return $locations;
}
public function getClosureStorage(string $file_path, string $closure_id): FunctionStorage
2018-01-31 23:09:09 +01:00
{
$file_storage = $this->file_storage_provider->get($file_path);
// closures can be returned here
if (isset($file_storage->functions[$closure_id])) {
return $file_storage->functions[$closure_id];
}
2021-12-03 21:40:18 +01:00
throw new UnexpectedValueException(
2022-12-18 17:15:15 +01:00
'Expecting ' . $closure_id . ' to have storage in ' . $file_path,
2018-01-31 23:09:09 +01:00
);
}
2021-12-13 16:28:14 +01:00
public function addGlobalConstantType(string $const_id, Union $type): void
{
self::$stubbed_constants[$const_id] = $type;
}
2021-12-13 16:28:14 +01:00
public function getStubbedConstantType(string $const_id): ?Union
{
2021-09-26 22:06:24 +02:00
return self::$stubbed_constants[$const_id] ?? null;
}
2020-01-09 05:48:42 +01:00
/**
2021-12-13 16:28:14 +01:00
* @return array<string, Union>
2020-01-09 05:48:42 +01:00
*/
public function getAllStubbedConstants(): array
2020-01-09 05:48:42 +01:00
{
return self::$stubbed_constants;
}
public function fileExists(string $file_path): bool
{
2018-02-04 00:52:35 +01:00
return $this->file_provider->fileExists($file_path);
}
/**
2018-02-04 00:52:35 +01:00
* Check whether a class/interface exists
*/
public function classOrInterfaceExists(
string $fq_class_name,
?CodeLocation $code_location = null,
?string $calling_fq_class_name = null,
?string $calling_method_id = null
): bool {
return $this->classlikes->classOrInterfaceExists(
$fq_class_name,
$code_location,
$calling_fq_class_name,
2022-12-18 17:15:15 +01:00
$calling_method_id,
);
}
/**
* Check whether a class/interface exists
*/
public function classOrInterfaceOrEnumExists(
string $fq_class_name,
?CodeLocation $code_location = null,
?string $calling_fq_class_name = null,
?string $calling_method_id = null
): bool {
return $this->classlikes->classOrInterfaceOrEnumExists(
$fq_class_name,
$code_location,
$calling_fq_class_name,
2022-12-18 17:15:15 +01:00
$calling_method_id,
);
}
public function classExtendsOrImplements(string $fq_class_name, string $possible_parent): bool
{
2018-02-04 00:52:35 +01:00
return $this->classlikes->classExtends($fq_class_name, $possible_parent)
|| $this->classlikes->classImplements($fq_class_name, $possible_parent);
}
/**
2018-02-04 00:52:35 +01:00
* Determine whether or not a given class exists
*/
public function classExists(
string $fq_class_name,
?CodeLocation $code_location = null,
?string $calling_fq_class_name = null,
?string $calling_method_id = null
): bool {
return $this->classlikes->classExists(
$fq_class_name,
$code_location,
$calling_fq_class_name,
2022-12-18 17:15:15 +01:00
$calling_method_id,
);
}
/**
2018-02-04 00:52:35 +01:00
* Determine whether or not a class extends a parent
*
2021-12-04 03:37:19 +01:00
* @throws UnpopulatedClasslikeException when called on unpopulated class
* @throws InvalidArgumentException when class does not exist
*/
public function classExtends(string $fq_class_name, string $possible_parent): bool
{
return $this->classlikes->classExtends($fq_class_name, $possible_parent, true);
}
/**
2018-02-04 00:52:35 +01:00
* Check whether a class implements an interface
*/
public function classImplements(string $fq_class_name, string $interface): bool
{
2018-02-04 00:52:35 +01:00
return $this->classlikes->classImplements($fq_class_name, $interface);
}
2020-09-21 00:27:02 +02:00
public function interfaceExists(
string $fq_interface_name,
?CodeLocation $code_location = null,
?string $calling_fq_class_name = null,
?string $calling_method_id = null
): bool {
return $this->classlikes->interfaceExists(
$fq_interface_name,
$code_location,
$calling_fq_class_name,
2022-12-18 17:15:15 +01:00
$calling_method_id,
);
}
public function interfaceExtends(string $interface_name, string $possible_parent): bool
{
2018-02-04 00:52:35 +01:00
return $this->classlikes->interfaceExtends($interface_name, $possible_parent);
}
2018-02-04 00:52:35 +01:00
/**
2020-10-17 18:36:44 +02:00
* @return array<string, string> all interfaces extended by $interface_name
2018-02-04 00:52:35 +01:00
*/
public function getParentInterfaces(string $fq_interface_name): array
2018-02-04 00:52:35 +01:00
{
return $this->classlikes->getParentInterfaces(
2022-12-18 17:15:15 +01:00
$this->classlikes->getUnAliasedName($fq_interface_name),
);
2018-02-04 00:52:35 +01:00
}
2018-02-04 00:52:35 +01:00
/**
* Determine whether or not a class has the correct casing
*/
public function classHasCorrectCasing(string $fq_class_name): bool
2018-02-04 00:52:35 +01:00
{
return $this->classlikes->classHasCorrectCasing($fq_class_name);
}
public function interfaceHasCorrectCasing(string $fq_interface_name): bool
{
2018-02-04 00:52:35 +01:00
return $this->classlikes->interfaceHasCorrectCasing($fq_interface_name);
}
2022-09-08 14:33:20 +02:00
public function traitHasCorrectCasing(string $fq_trait_name): bool
2018-02-04 00:52:35 +01:00
{
2022-09-08 14:33:20 +02:00
return $this->classlikes->traitHasCorrectCasing($fq_trait_name);
2018-02-04 00:52:35 +01:00
}
/**
* Given a function id, return the function like storage for
* a method, closure, or function.
2020-05-15 16:18:05 +02:00
*
* @param non-empty-string $function_id
* @return FunctionStorage|MethodStorage
*/
public function getFunctionLikeStorage(
StatementsAnalyzer $statements_analyzer,
string $function_id
): FunctionLikeStorage {
$doesMethodExist =
2021-12-03 20:11:20 +01:00
MethodIdentifier::isValidMethodIdReference($function_id)
&& $this->methodExists($function_id);
if ($doesMethodExist) {
2021-12-03 20:11:20 +01:00
$method_id = MethodIdentifier::wrap($function_id);
$declaring_method_id = $this->methods->getDeclaringMethodId($method_id);
if (!$declaring_method_id) {
2021-12-03 21:40:18 +01:00
throw new UnexpectedValueException('Declaring method for ' . $method_id . ' cannot be found');
}
return $this->methods->getStorage($declaring_method_id);
}
return $this->functions->getStorage($statements_analyzer, strtolower($function_id));
}
2018-02-04 00:52:35 +01:00
/**
* Whether or not a given method exists
*
2021-12-04 03:37:19 +01:00
* @param string|MethodIdentifier $method_id
* @param string|MethodIdentifier|null $calling_method_id
2018-02-04 00:52:35 +01:00
*/
public function methodExists(
$method_id,
?CodeLocation $code_location = null,
$calling_method_id = null,
?string $file_path = null,
bool $is_used = true
): bool {
return $this->methods->methodExists(
2021-12-14 01:54:11 +01:00
MethodIdentifier::wrap($method_id),
is_string($calling_method_id) ? strtolower($calling_method_id) : strtolower((string) $calling_method_id),
$code_location,
null,
$file_path,
true,
2022-12-18 17:15:15 +01:00
$is_used,
);
}
/**
2021-12-04 03:37:19 +01:00
* @param string|MethodIdentifier $method_id
* @return array<int, FunctionLikeParameter>
*/
public function getMethodParams($method_id): array
{
2021-12-14 01:54:11 +01:00
return $this->methods->getMethodParams(MethodIdentifier::wrap($method_id));
}
/**
2021-12-04 03:37:19 +01:00
* @param string|MethodIdentifier $method_id
*/
public function isVariadic($method_id): bool
{
2021-12-14 01:54:11 +01:00
return $this->methods->isVariadic(MethodIdentifier::wrap($method_id));
2018-02-04 00:52:35 +01:00
}
2018-02-04 00:52:35 +01:00
/**
2021-12-04 03:37:19 +01:00
* @param string|MethodIdentifier $method_id
* @param list<Arg> $call_args
2018-02-04 00:52:35 +01:00
*/
2021-12-13 16:28:14 +01:00
public function getMethodReturnType($method_id, ?string &$self_class, array $call_args = []): ?Union
2018-02-04 00:52:35 +01:00
{
return $this->methods->getMethodReturnType(
2021-12-14 01:54:11 +01:00
MethodIdentifier::wrap($method_id),
$self_class,
null,
2022-12-18 17:15:15 +01:00
$call_args,
);
2018-02-04 00:52:35 +01:00
}
2018-02-04 00:52:35 +01:00
/**
2021-12-04 03:37:19 +01:00
* @param string|MethodIdentifier $method_id
2018-02-04 00:52:35 +01:00
*/
public function getMethodReturnsByRef($method_id): bool
2018-02-04 00:52:35 +01:00
{
2021-12-14 01:54:11 +01:00
return $this->methods->getMethodReturnsByRef(MethodIdentifier::wrap($method_id));
2018-02-04 00:52:35 +01:00
}
2018-02-04 00:52:35 +01:00
/**
2021-12-04 03:37:19 +01:00
* @param string|MethodIdentifier $method_id
2018-02-04 00:52:35 +01:00
*/
public function getMethodReturnTypeLocation(
$method_id,
CodeLocation &$defined_location = null
): ?CodeLocation {
return $this->methods->getMethodReturnTypeLocation(
2021-12-14 01:54:11 +01:00
MethodIdentifier::wrap($method_id),
2022-12-18 17:15:15 +01:00
$defined_location,
);
}
2018-01-31 23:09:09 +01:00
/**
2021-12-04 03:37:19 +01:00
* @param string|MethodIdentifier $method_id
2018-01-31 23:09:09 +01:00
*/
public function getDeclaringMethodId($method_id): ?string
2018-01-31 23:09:09 +01:00
{
2021-12-14 01:54:11 +01:00
$new_method_id = $this->methods->getDeclaringMethodId(MethodIdentifier::wrap($method_id));
return $new_method_id ? (string) $new_method_id : null;
2018-02-04 00:52:35 +01:00
}
2018-01-31 23:09:09 +01:00
2018-02-04 00:52:35 +01:00
/**
* Get the class this method appears in (vs is declared in, which could give a trait)
*
2021-12-04 03:37:19 +01:00
* @param string|MethodIdentifier $method_id
2018-02-04 00:52:35 +01:00
*/
public function getAppearingMethodId($method_id): ?string
2018-02-04 00:52:35 +01:00
{
2021-12-14 01:54:11 +01:00
$new_method_id = $this->methods->getAppearingMethodId(MethodIdentifier::wrap($method_id));
return $new_method_id ? (string) $new_method_id : null;
2018-01-31 23:09:09 +01:00
}
/**
2021-12-04 03:37:19 +01:00
* @param string|MethodIdentifier $method_id
2021-12-14 01:54:11 +01:00
* @return array<string, MethodIdentifier>
*/
public function getOverriddenMethodIds($method_id): array
{
2021-12-14 01:54:11 +01:00
return $this->methods->getOverriddenMethodIds(MethodIdentifier::wrap($method_id));
}
/**
2021-12-04 03:37:19 +01:00
* @param string|MethodIdentifier $method_id
*/
public function getCasedMethodId($method_id): string
{
2021-12-14 01:54:11 +01:00
return $this->methods->getCasedMethodId(MethodIdentifier::wrap($method_id));
}
public function invalidateInformationForFile(string $file_path): void
{
$this->scanner->removeFile($file_path);
try {
$file_storage = $this->file_storage_provider->get($file_path);
2021-12-03 21:40:18 +01:00
} catch (InvalidArgumentException $e) {
return;
}
foreach ($file_storage->classlikes_in_file as $fq_classlike_name) {
$this->classlike_storage_provider->remove($fq_classlike_name);
$this->classlikes->removeClassLike($fq_classlike_name);
}
$this->file_storage_provider->remove($file_path);
}
2020-09-21 00:27:02 +02:00
public function getFunctionStorageForSymbol(string $file_path, string $symbol): ?FunctionLikeStorage
{
if (strpos($symbol, '::')) {
$symbol = substr($symbol, 0, -2);
/** @psalm-suppress ArgumentTypeCoercion */
2021-12-03 20:11:20 +01:00
$method_id = new MethodIdentifier(...explode('::', $symbol));
$declaring_method_id = $this->methods->getDeclaringMethodId($method_id);
if (!$declaring_method_id) {
return null;
}
2021-09-25 02:34:21 +02:00
return $this->methods->getStorage($declaring_method_id);
}
$function_id = strtolower(substr($symbol, 0, -2));
$file_storage = $this->file_storage_provider->get($file_path);
if (isset($file_storage->functions[$function_id])) {
2021-09-25 02:34:21 +02:00
return $file_storage->functions[$function_id];
}
if (!$function_id) {
return null;
}
2021-09-25 02:34:21 +02:00
return $this->functions->getStorage(null, $function_id);
}
/**
2022-12-20 21:52:25 +01:00
* Get Markup content from Reference
*/
2023-03-02 19:02:23 +01:00
public function getMarkupContentForSymbolByReference(
2022-12-20 21:52:25 +01:00
Reference $reference
): ?PHPMarkdownContent {
//Direct Assignment
if (is_numeric($reference->symbol[0])) {
return new PHPMarkdownContent(
preg_replace(
'/^[^:]*:/',
'',
$reference->symbol,
),
);
}
2022-12-20 21:52:25 +01:00
//Class
if (strpos($reference->symbol, '::')) {
//Class Method
if (strpos($reference->symbol, '()')) {
$symbol = substr($reference->symbol, 0, -2);
2022-12-20 21:52:25 +01:00
/** @psalm-suppress ArgumentTypeCoercion */
$method_id = new MethodIdentifier(...explode('::', $symbol));
2022-12-20 21:52:25 +01:00
$declaring_method_id = $this->methods->getDeclaringMethodId(
$method_id,
);
2022-12-20 21:52:25 +01:00
if (!$declaring_method_id) {
return null;
}
2022-12-20 21:52:25 +01:00
$storage = $this->methods->getStorage($declaring_method_id);
2022-12-20 21:52:25 +01:00
return new PHPMarkdownContent(
$storage->getHoverMarkdown(),
"{$storage->defining_fqcln}::{$storage->cased_name}",
$storage->description,
);
}
2022-12-20 21:52:25 +01:00
/** @psalm-suppress PossiblyUndefinedIntArrayOffset */
[, $symbol_name] = explode('::', $reference->symbol);
//Class Property
if (strpos($reference->symbol, '$') !== false) {
$property_id = preg_replace('/^\\\\/', '', $reference->symbol);
/** @psalm-suppress PossiblyUndefinedIntArrayOffset */
[$fq_class_name, $property_name] = explode('::$', $property_id);
$class_storage = $this->classlikes->getStorageFor($fq_class_name);
//Get Real Properties
if (isset($class_storage->declaring_property_ids[$property_name])) {
$declaring_property_class = $class_storage->declaring_property_ids[$property_name];
$declaring_class_storage = $this->classlike_storage_provider->get($declaring_property_class);
if (isset($declaring_class_storage->properties[$property_name])) {
$storage = $declaring_class_storage->properties[$property_name];
return new PHPMarkdownContent(
"{$storage->getInfo()} {$symbol_name}",
$reference->symbol,
$storage->description,
);
}
}
2022-12-20 21:52:25 +01:00
//Get Docblock properties
if (isset($class_storage->pseudo_property_set_types['$'.$property_name])) {
return new PHPMarkdownContent(
'public '.
(string) $class_storage->pseudo_property_set_types['$'.$property_name].' $'.$property_name,
$reference->symbol,
);
}
2022-12-20 21:52:25 +01:00
//Get Docblock properties
if (isset($class_storage->pseudo_property_get_types['$'.$property_name])) {
return new PHPMarkdownContent(
'public '.
(string) $class_storage->pseudo_property_get_types['$'.$property_name].' $'.$property_name,
$reference->symbol,
);
}
2022-12-20 21:52:25 +01:00
return null;
}
2022-12-20 21:52:25 +01:00
/** @psalm-suppress PossiblyUndefinedIntArrayOffset */
[$fq_classlike_name, $const_name] = explode(
'::',
$reference->symbol,
);
2022-12-20 21:52:25 +01:00
$class_constants = $this->classlikes->getConstantsForClass(
$fq_classlike_name,
ReflectionProperty::IS_PRIVATE,
);
2022-12-20 21:52:25 +01:00
if (!isset($class_constants[$const_name])) {
return null;
}
2022-12-20 21:52:25 +01:00
//Class Constant
return new PHPMarkdownContent(
$class_constants[$const_name]->getHoverMarkdown($const_name),
$fq_classlike_name . '::' . $const_name,
$class_constants[$const_name]->description,
);
}
2022-12-20 21:52:25 +01:00
//Procedural Function
if (strpos($reference->symbol, '()')) {
$function_id = strtolower(substr($reference->symbol, 0, -2));
$file_storage = $this->file_storage_provider->get(
$reference->file_path,
);
2022-12-20 21:52:25 +01:00
if (isset($file_storage->functions[$function_id])) {
$function_storage = $file_storage->functions[$function_id];
return new PHPMarkdownContent(
$function_storage->getHoverMarkdown(),
$function_id,
$function_storage->description,
);
}
2022-12-20 21:52:25 +01:00
if (!$function_id) {
return null;
}
2022-12-20 21:52:25 +01:00
$function = $this->functions->getStorage(null, $function_id);
return new PHPMarkdownContent(
$function->getHoverMarkdown(),
$function_id,
$function->description,
);
}
2022-12-20 21:52:25 +01:00
//Procedural Variable
if (strpos($reference->symbol, '$') === 0) {
$type = VariableFetchAnalyzer::getGlobalType($reference->symbol, $this->analysis_php_version_id);
if (!$type->isMixed()) {
return new PHPMarkdownContent(
(string) $type,
$reference->symbol,
);
2022-12-20 21:52:25 +01:00
}
}
2022-12-20 21:52:25 +01:00
try {
$storage = $this->classlike_storage_provider->get(
$reference->symbol,
);
return new PHPMarkdownContent(
($storage->abstract ? 'abstract ' : '') .
'class ' .
$storage->name,
$storage->name,
$storage->description,
);
} catch (InvalidArgumentException $e) {
//continue on as normal
}
if (strpos($reference->symbol, '\\')) {
$const_name_parts = explode('\\', $reference->symbol);
$const_name = array_pop($const_name_parts);
$namespace_name = implode('\\', $const_name_parts);
$namespace_constants = NamespaceAnalyzer::getConstantsForNamespace(
$namespace_name,
ReflectionProperty::IS_PUBLIC,
);
//Namespace Constant
if (isset($namespace_constants[$const_name])) {
$type = $namespace_constants[$const_name];
return new PHPMarkdownContent(
$reference->symbol . ' ' . $type,
$reference->symbol,
);
}
2022-12-20 21:52:25 +01:00
} else {
$file_storage = $this->file_storage_provider->get(
$reference->file_path,
);
// ?
if (isset($file_storage->constants[$reference->symbol])) {
return new PHPMarkdownContent(
'const ' .
$reference->symbol .
' ' .
$file_storage->constants[$reference->symbol],
$reference->symbol,
);
}
$type = ConstFetchAnalyzer::getGlobalConstType(
$this,
$reference->symbol,
$reference->symbol,
);
2019-07-05 22:24:00 +02:00
2022-12-20 21:52:25 +01:00
//Global Constant
if ($type) {
return new PHPMarkdownContent(
'const ' . $reference->symbol . ' ' . $type,
$reference->symbol,
);
}
}
2022-12-20 21:52:25 +01:00
return new PHPMarkdownContent($reference->symbol);
}
2023-03-02 19:02:23 +01:00
/**
* @psalm-suppress PossiblyUnusedMethod
* @deprecated will be removed in Psalm 6. use {@see Codebase::getSymbolLocationByReference()} instead
*/
public function getSymbolInformation(string $file_path, string $symbol): ?array
{
if (is_numeric($symbol[0])) {
return ['type' => preg_replace('/^[^:]*:/', '', $symbol)];
}
try {
if (strpos($symbol, '::')) {
if (strpos($symbol, '()')) {
$symbol = substr($symbol, 0, -2);
/** @psalm-suppress ArgumentTypeCoercion */
$method_id = new MethodIdentifier(...explode('::', $symbol));
$declaring_method_id = $this->methods->getDeclaringMethodId($method_id);
if (!$declaring_method_id) {
return null;
}
$storage = $this->methods->getStorage($declaring_method_id);
return [
'type' => '<?php ' . $storage->getCompletionSignature(),
'description' => $storage->description,
];
}
[, $symbol_name] = explode('::', $symbol);
if (strpos($symbol, '$') !== false) {
$storage = $this->properties->getStorage($symbol);
return [
'type' => '<?php ' . $storage->getInfo() . ' ' . $symbol_name,
'description' => $storage->description,
];
}
[$fq_classlike_name, $const_name] = explode('::', $symbol);
$class_constants = $this->classlikes->getConstantsForClass(
$fq_classlike_name,
ReflectionProperty::IS_PRIVATE,
);
if (!isset($class_constants[$const_name])) {
return null;
}
return [
'type' => '<?php ' . $const_name,
'description' => $class_constants[$const_name]->description,
];
}
if (strpos($symbol, '()')) {
$function_id = strtolower(substr($symbol, 0, -2));
$file_storage = $this->file_storage_provider->get($file_path);
if (isset($file_storage->functions[$function_id])) {
$function_storage = $file_storage->functions[$function_id];
return [
'type' => '<?php ' . $function_storage->getCompletionSignature(),
'description' => $function_storage->description,
];
}
if (!$function_id) {
return null;
}
$function = $this->functions->getStorage(null, $function_id);
return [
'type' => '<?php ' . $function->getCompletionSignature(),
'description' => $function->description,
];
}
if (strpos($symbol, '$') === 0) {
$type = VariableFetchAnalyzer::getGlobalType($symbol, $this->analysis_php_version_id);
if (!$type->isMixed()) {
return ['type' => '<?php ' . $type];
}
}
try {
$storage = $this->classlike_storage_provider->get($symbol);
return [
'type' => '<?php ' . ($storage->abstract ? 'abstract ' : '') . 'class ' . $storage->name,
'description' => $storage->description,
];
} catch (InvalidArgumentException $e) {
}
if (strpos($symbol, '\\')) {
$const_name_parts = explode('\\', $symbol);
$const_name = array_pop($const_name_parts);
$namespace_name = implode('\\', $const_name_parts);
$namespace_constants = NamespaceAnalyzer::getConstantsForNamespace(
$namespace_name,
ReflectionProperty::IS_PUBLIC,
);
if (isset($namespace_constants[$const_name])) {
$type = $namespace_constants[$const_name];
return ['type' => '<?php const ' . $symbol . ' ' . $type];
}
} else {
$file_storage = $this->file_storage_provider->get($file_path);
if (isset($file_storage->constants[$symbol])) {
return ['type' => '<?php const ' . $symbol . ' ' . $file_storage->constants[$symbol]];
}
$constant = ConstFetchAnalyzer::getGlobalConstType($this, $symbol, $symbol);
if ($constant) {
return ['type' => '<?php const ' . $symbol . ' ' . $constant];
}
}
return null;
} catch (Exception $e) {
error_log($e->getMessage());
return null;
}
}
/**
* @psalm-suppress PossiblyUnusedMethod
* @deprecated will be removed in Psalm 6. use {@see Codebase::getSymbolLocationByReference()} instead
*/
public function getSymbolLocation(string $file_path, string $symbol): ?CodeLocation
{
if (is_numeric($symbol[0])) {
$symbol = preg_replace('/:.*/', '', $symbol);
$symbol_parts = explode('-', $symbol);
$file_contents = $this->getFileContents($file_path);
return new Raw(
$file_contents,
$file_path,
$this->config->shortenFileName($file_path),
(int) $symbol_parts[0],
(int) $symbol_parts[1],
);
}
try {
if (strpos($symbol, '::')) {
if (strpos($symbol, '()')) {
$symbol = substr($symbol, 0, -2);
/** @psalm-suppress ArgumentTypeCoercion */
$method_id = new MethodIdentifier(...explode('::', $symbol));
$declaring_method_id = $this->methods->getDeclaringMethodId($method_id);
if (!$declaring_method_id) {
return null;
}
$storage = $this->methods->getStorage($declaring_method_id);
return $storage->location;
}
if (strpos($symbol, '$') !== false) {
$storage = $this->properties->getStorage($symbol);
return $storage->location;
}
[$fq_classlike_name, $const_name] = explode('::', $symbol);
$class_constants = $this->classlikes->getConstantsForClass(
$fq_classlike_name,
ReflectionProperty::IS_PRIVATE,
);
if (!isset($class_constants[$const_name])) {
return null;
}
return $class_constants[$const_name]->location;
}
if (strpos($symbol, '()')) {
$file_storage = $this->file_storage_provider->get($file_path);
$function_id = strtolower(substr($symbol, 0, -2));
if (isset($file_storage->functions[$function_id])) {
return $file_storage->functions[$function_id]->location;
}
if (!$function_id) {
return null;
}
return $this->functions->getStorage(null, $function_id)->location;
}
return $this->classlike_storage_provider->get($symbol)->location;
} catch (UnexpectedValueException $e) {
error_log($e->getMessage());
return null;
} catch (InvalidArgumentException $e) {
return null;
}
}
public function getSymbolLocationByReference(Reference $reference): ?CodeLocation
{
2022-12-20 21:52:25 +01:00
if (is_numeric($reference->symbol[0])) {
$symbol = preg_replace('/:.*/', '', $reference->symbol);
2019-07-02 00:48:33 +02:00
$symbol_parts = explode('-', $symbol);
2022-12-20 21:52:25 +01:00
if (!isset($symbol_parts[0]) || !isset($symbol_parts[1])) {
return null;
}
$file_contents = $this->getFileContents($reference->file_path);
2019-07-02 00:48:33 +02:00
return new Raw(
2019-08-18 20:27:50 +02:00
$file_contents,
2022-12-20 21:52:25 +01:00
$reference->file_path,
$this->config->shortenFileName($reference->file_path),
2019-08-18 20:27:50 +02:00
(int) $symbol_parts[0],
2022-12-18 17:15:15 +01:00
(int) $symbol_parts[1],
2019-08-18 20:27:50 +02:00
);
2019-07-02 00:48:33 +02:00
}
try {
2022-12-20 21:52:25 +01:00
if (strpos($reference->symbol, '::')) {
if (strpos($reference->symbol, '()')) {
$symbol = substr($reference->symbol, 0, -2);
2020-10-15 00:51:15 +02:00
/** @psalm-suppress ArgumentTypeCoercion */
2022-12-20 21:52:25 +01:00
$method_id = new MethodIdentifier(
...explode('::', $symbol),
);
2022-12-20 21:52:25 +01:00
$declaring_method_id = $this->methods->getDeclaringMethodId(
$method_id,
);
if (!$declaring_method_id) {
return null;
}
$storage = $this->methods->getStorage($declaring_method_id);
return $storage->location;
}
2022-12-20 21:52:25 +01:00
if (strpos($reference->symbol, '$') !== false) {
$storage = $this->properties->getStorage(
$reference->symbol,
);
return $storage->location;
}
2022-12-20 21:52:25 +01:00
/** @psalm-suppress PossiblyUndefinedIntArrayOffset */
[$fq_classlike_name, $const_name] = explode(
'::',
$reference->symbol,
);
$class_constants = $this->classlikes->getConstantsForClass(
$fq_classlike_name,
2022-12-18 17:15:15 +01:00
ReflectionProperty::IS_PRIVATE,
);
if (!isset($class_constants[$const_name])) {
return null;
}
return $class_constants[$const_name]->location;
}
2022-12-20 21:52:25 +01:00
if (strpos($reference->symbol, '()')) {
$file_storage = $this->file_storage_provider->get(
$reference->file_path,
);
2022-12-20 21:52:25 +01:00
$function_id = strtolower(substr($reference->symbol, 0, -2));
if (isset($file_storage->functions[$function_id])) {
return $file_storage->functions[$function_id]->location;
}
if (!$function_id) {
return null;
}
2022-12-20 21:52:25 +01:00
return $this->functions->getStorage(null, $function_id)
->location;
}
2022-12-20 21:52:25 +01:00
return $this->classlike_storage_provider->get(
$reference->symbol,
)->location;
2021-12-03 21:40:18 +01:00
} catch (UnexpectedValueException $e) {
error_log($e->getMessage());
2019-07-05 22:24:00 +02:00
return null;
2021-12-03 21:40:18 +01:00
} catch (InvalidArgumentException $e) {
return null;
}
}
2023-03-02 19:02:23 +01:00
/**
* @psalm-suppress PossiblyUnusedMethod
* @return array{0: string, 1: Range}|null
*/
public function getReferenceAtPosition(string $file_path, Position $position): ?array
{
$ref = $this->getReferenceAtPositionAsReference($file_path, $position);
2023-03-02 19:07:50 +01:00
if ($ref === null) {
2023-03-02 19:02:23 +01:00
return null;
}
return [$ref->symbol, $ref->range];
}
/**
2022-12-20 21:52:25 +01:00
* Get Reference from Position
*/
2023-03-02 19:02:23 +01:00
public function getReferenceAtPositionAsReference(
2022-12-20 21:52:25 +01:00
string $file_path,
Position $position
): ?Reference {
$is_open = $this->file_provider->isOpen($file_path);
if (!$is_open) {
2021-12-03 20:29:06 +01:00
throw new UnanalyzedFileException($file_path . ' is not open');
}
$file_contents = $this->getFileContents($file_path);
$offset = $position->toOffset($file_contents);
2022-12-20 21:52:25 +01:00
$reference_maps = $this->analyzer->getMapsForFile($file_path);
$reference_start_pos = null;
$reference_end_pos = null;
2022-12-20 21:52:25 +01:00
$symbol = null;
2022-12-20 21:52:25 +01:00
foreach ($reference_maps as $reference_map) {
ksort($reference_map);
2022-12-20 21:52:25 +01:00
foreach ($reference_map as $start_pos => [$end_pos, $possible_reference]) {
if ($offset < $start_pos) {
break;
}
if ($offset > $end_pos) {
continue;
}
$reference_start_pos = $start_pos;
$reference_end_pos = $end_pos;
$symbol = $possible_reference;
}
2022-12-20 21:52:25 +01:00
if ($symbol !== null &&
$reference_start_pos !== null &&
$reference_end_pos !== null
) {
break;
}
}
2022-12-20 21:52:25 +01:00
if ($symbol === null || $reference_start_pos === null || $reference_end_pos === null) {
2019-07-02 00:48:33 +02:00
return null;
}
$range = new Range(
self::getPositionFromOffset($reference_start_pos, $file_contents),
2022-12-18 17:15:15 +01:00
self::getPositionFromOffset($reference_end_pos, $file_contents),
);
2022-12-20 21:52:25 +01:00
return new Reference($file_path, $symbol, $range);
}
/**
* @return array{0: non-empty-string, 1: int, 2: Range}|null
*/
public function getFunctionArgumentAtPosition(string $file_path, Position $position): ?array
{
$is_open = $this->file_provider->isOpen($file_path);
if (!$is_open) {
2021-12-03 20:29:06 +01:00
throw new UnanalyzedFileException($file_path . ' is not open');
}
$file_contents = $this->getFileContents($file_path);
$offset = $position->toOffset($file_contents);
[, , $argument_map] = $this->analyzer->getMapsForFile($file_path);
$reference = null;
$argument_number = null;
if (!$argument_map) {
return null;
}
$start_pos = null;
$end_pos = null;
ksort($argument_map);
foreach ($argument_map as $start_pos => [$end_pos, $possible_reference, $possible_argument_number]) {
if ($offset < $start_pos) {
break;
}
if ($offset > $end_pos) {
continue;
}
$reference = $possible_reference;
$argument_number = $possible_argument_number;
}
if ($reference === null || $start_pos === null || $end_pos === null || $argument_number === null) {
return null;
}
$range = new Range(
self::getPositionFromOffset($start_pos, $file_contents),
2022-12-18 17:15:15 +01:00
self::getPositionFromOffset($end_pos, $file_contents),
);
return [$reference, $argument_number, $range];
}
2019-07-19 05:08:54 +02:00
/**
2020-05-15 16:18:05 +02:00
* @param non-empty-string $function_symbol
2019-07-19 05:08:54 +02:00
*/
public function getSignatureInformation(
string $function_symbol,
string $file_path = null
2021-12-03 20:11:20 +01:00
): ?SignatureInformation {
$signature_label = '';
$signature_documentation = null;
2019-07-19 05:08:54 +02:00
if (strpos($function_symbol, '::') !== false) {
2020-10-15 00:51:15 +02:00
/** @psalm-suppress ArgumentTypeCoercion */
2021-12-03 20:11:20 +01:00
$method_id = new MethodIdentifier(...explode('::', $function_symbol));
$declaring_method_id = $this->methods->getDeclaringMethodId($method_id);
2019-07-19 05:08:54 +02:00
if ($declaring_method_id === null) {
return null;
}
$method_storage = $this->methods->getStorage($declaring_method_id);
$params = $method_storage->params;
$signature_label = $method_storage->cased_name;
$signature_documentation = $method_storage->description;
2019-07-19 05:08:54 +02:00
} else {
try {
if ($file_path) {
$function_storage = $this->functions->getStorage(
null,
strtolower($function_symbol),
dirname($file_path),
2022-12-18 17:15:15 +01:00
$file_path,
);
} else {
$function_storage = $this->functions->getStorage(null, strtolower($function_symbol));
}
2019-07-19 05:30:44 +02:00
$params = $function_storage->params;
$signature_label = $function_storage->cased_name;
$signature_documentation = $function_storage->description;
2021-12-03 21:40:18 +01:00
} catch (Exception $exception) {
if (InternalCallMapHandler::inCallMap($function_symbol)) {
$callables = InternalCallMapHandler::getCallablesFromCallMap($function_symbol);
2019-07-19 05:08:54 +02:00
2019-07-19 05:30:44 +02:00
if (!$callables || !$callables[0]->params) {
return null;
}
$params = $callables[0]->params;
} else {
return null;
}
}
2019-07-19 05:08:54 +02:00
}
$signature_label .= '(';
2019-07-19 05:08:54 +02:00
$parameters = [];
foreach ($params as $i => $param) {
$parameter_label = ($param->type ?: 'mixed') . ' $' . $param->name;
2021-12-03 20:11:20 +01:00
$parameters[] = new ParameterInformation(
[
strlen($signature_label),
strlen($signature_label) + strlen($parameter_label),
],
2022-12-18 17:15:15 +01:00
$param->description ?? null,
);
2019-07-19 05:08:54 +02:00
$signature_label .= $parameter_label;
if ($i < (count($params) - 1)) {
$signature_label .= ', ';
}
}
$signature_label .= ')';
2021-12-03 20:11:20 +01:00
return new SignatureInformation(
2019-07-19 05:08:54 +02:00
$signature_label,
$parameters,
2022-12-18 17:15:15 +01:00
$signature_documentation,
2019-07-19 05:08:54 +02:00
);
}
/**
* @return array{0: string, 1: '->'|'::'|'['|'symbol', 2: int}|null
*/
public function getCompletionDataAtPosition(string $file_path, Position $position): ?array
{
$is_open = $this->file_provider->isOpen($file_path);
if (!$is_open) {
2021-12-03 20:29:06 +01:00
throw new UnanalyzedFileException($file_path . ' is not open');
}
$file_contents = $this->getFileContents($file_path);
$offset = $position->toOffset($file_contents);
[$reference_map, $type_map] = $this->analyzer->getMapsForFile($file_path);
if (!$reference_map && !$type_map) {
return null;
}
krsort($type_map);
foreach ($type_map as $start_pos => [$end_pos_excluding_whitespace, $possible_type]) {
if ($offset < $start_pos) {
continue;
}
2022-12-20 21:52:25 +01:00
/** @psalm-suppress PossiblyUndefinedIntArrayOffset */
$num_whitespace_bytes = preg_match('/\G\s+/', $file_contents, $matches, 0, $end_pos_excluding_whitespace)
? strlen($matches[0])
: 0;
$end_pos = $end_pos_excluding_whitespace + $num_whitespace_bytes;
if ($offset - $end_pos === 1) {
$candidate_gap = substr($file_contents, $end_pos, 1);
2021-10-04 00:03:06 +02:00
if ($candidate_gap === '[') {
$gap = $candidate_gap;
$recent_type = $possible_type;
if ($recent_type === 'mixed') {
return null;
}
return [$recent_type, $gap, $offset];
}
}
2019-02-24 07:33:25 +01:00
if ($offset - $end_pos === 2 || $offset - $end_pos === 3) {
$candidate_gap = substr($file_contents, $end_pos, 2);
if ($candidate_gap === '->' || $candidate_gap === '::') {
$gap = $candidate_gap;
$recent_type = $possible_type;
if ($recent_type === 'mixed') {
return null;
}
return [$recent_type, $gap, $offset];
}
}
}
foreach ($reference_map as $start_pos => [$end_pos, $possible_reference]) {
if ($offset < $start_pos) {
continue;
}
// If the reference precedes a "::" then treat it as a class reference.
if ($offset - $end_pos === 2 && substr($file_contents, $end_pos, 2) === '::') {
return [$possible_reference, '::', $offset];
}
// Only continue for references that are partial / don't exist.
if ($possible_reference[0] !== '*') {
continue;
}
if ($offset - $end_pos === 0) {
$recent_type = $possible_reference;
return [$recent_type, 'symbol', $offset];
}
}
return null;
}
2021-12-13 16:28:14 +01:00
public function getTypeContextAtPosition(string $file_path, Position $position): ?Union
{
$file_contents = $this->getFileContents($file_path);
$offset = $position->toOffset($file_contents);
[$reference_map, $type_map, $argument_map] = $this->analyzer->getMapsForFile($file_path);
if (!$reference_map && !$type_map && !$argument_map) {
return null;
}
foreach ($argument_map as $start_pos => [$end_pos, $function, $argument_num]) {
if ($offset < $start_pos || $offset > $end_pos) {
continue;
}
// First parameter to a function-like
$function_storage = $this->getFunctionStorageForSymbol($file_path, $function . '()');
if (!$function_storage || !$function_storage->params || !isset($function_storage->params[$argument_num])) {
return null;
}
2021-09-25 02:34:21 +02:00
return $function_storage->params[$argument_num]->type;
}
return null;
}
/**
2021-12-04 03:37:19 +01:00
* @return list<CompletionItem>
*/
2022-12-21 00:31:56 +01:00
public function getCompletionItemsForClassishThing(
string $type_string,
string $gap,
bool $snippets_supported = false
): array {
$completion_items = [];
$type = Type::parseString($type_string);
foreach ($type->getAtomicTypes() as $atomic_type) {
2021-12-13 04:45:57 +01:00
if ($atomic_type instanceof TNamedObject) {
try {
$class_storage = $this->classlike_storage_provider->get($atomic_type->value);
foreach ($class_storage->appearing_method_ids as $declaring_method_id) {
$method_storage = $this->methods->getStorage($declaring_method_id);
if ($method_storage->is_static || $gap === '->') {
2021-12-03 20:11:20 +01:00
$completion_item = new CompletionItem(
$method_storage->cased_name,
2021-12-03 20:11:20 +01:00
CompletionItemKind::METHOD,
2022-12-20 21:52:25 +01:00
$method_storage->getCompletionSignature(),
$method_storage->description,
(string)$method_storage->visibility,
$method_storage->cased_name,
2023-03-03 08:20:41 +01:00
$method_storage->cased_name,
null,
null,
new Command('Trigger parameter hints', 'editor.action.triggerParameterHints'),
null,
2022-12-18 17:15:15 +01:00
2,
);
2022-12-20 21:52:25 +01:00
if ($snippets_supported && count($method_storage->params) > 0) {
$completion_item->insertText .= '($0)';
$completion_item->insertTextFormat =
InsertTextFormat::SNIPPET;
} else {
$completion_item->insertText .= '()';
}
$completion_items[] = $completion_item;
}
}
2022-12-20 21:52:25 +01:00
$pseudo_property_types = [];
foreach ($class_storage->pseudo_property_get_types as $property_name => $type) {
$pseudo_property_types[$property_name] = new CompletionItem(
str_replace('$', '', $property_name),
CompletionItemKind::PROPERTY,
$type->__toString(),
null,
'1', //sort text
str_replace('$', '', $property_name),
($gap === '::' ? '$' : '') .
str_replace('$', '', $property_name),
);
}
foreach ($class_storage->pseudo_property_set_types as $property_name => $type) {
$pseudo_property_types[$property_name] = new CompletionItem(
str_replace('$', '', $property_name),
CompletionItemKind::PROPERTY,
$type->__toString(),
null,
'1',
str_replace('$', '', $property_name),
($gap === '::' ? '$' : '') .
str_replace('$', '', $property_name),
);
}
$completion_items = array_merge($completion_items, array_values($pseudo_property_types));
foreach ($class_storage->declaring_property_ids as $property_name => $declaring_class) {
$property_storage = $this->properties->getStorage(
2022-12-18 17:15:15 +01:00
$declaring_class . '::$' . $property_name,
);
if ($property_storage->is_static || $gap === '->') {
2021-12-03 20:11:20 +01:00
$completion_items[] = new CompletionItem(
'$' . $property_name,
2021-12-03 20:11:20 +01:00
CompletionItemKind::PROPERTY,
$property_storage->getInfo(),
$property_storage->description,
(string)$property_storage->visibility,
$property_name,
2022-12-18 17:15:15 +01:00
($gap === '::' ? '$' : '') . $property_name,
);
}
}
foreach ($class_storage->constants as $const_name => $const) {
2021-12-03 20:11:20 +01:00
$completion_items[] = new CompletionItem(
$const_name,
2021-12-03 20:11:20 +01:00
CompletionItemKind::VARIABLE,
'const ' . $const_name,
$const->description,
null,
$const_name,
2022-12-18 17:15:15 +01:00
$const_name,
);
}
2021-12-03 21:40:18 +01:00
} catch (Exception $e) {
error_log($e->getMessage());
continue;
}
}
2019-06-13 15:42:38 +02:00
}
return $completion_items;
}
/**
2021-12-04 03:37:19 +01:00
* @return list<CompletionItem>
*/
public function getCompletionItemsForPartialSymbol(
string $type_string,
int $offset,
string $file_path
): array {
$fq_suggestion = false;
if (($type_string[1] ?? '') === '\\') {
$fq_suggestion = true;
}
$matching_classlike_names = $this->classlikes->getMatchingClassLikeNames($type_string);
$completion_items = [];
$file_storage = $this->file_storage_provider->get($file_path);
$aliases = null;
foreach ($file_storage->classlikes_in_file as $fq_class_name => $_) {
try {
$class_storage = $this->classlike_storage_provider->get($fq_class_name);
2021-12-03 21:40:18 +01:00
} catch (Exception $e) {
continue;
}
if (!$class_storage->stmt_location) {
continue;
}
if ($offset > $class_storage->stmt_location->raw_file_start
&& $offset < $class_storage->stmt_location->raw_file_end
) {
$aliases = $class_storage->aliases;
break;
}
}
if (!$aliases) {
foreach ($file_storage->namespace_aliases as $namespace_start => $namespace_aliases) {
if ($namespace_start < $offset) {
$aliases = $namespace_aliases;
break;
}
}
if (!$aliases) {
$aliases = $file_storage->aliases;
}
}
foreach ($matching_classlike_names as $fq_class_name) {
$extra_edits = [];
$insertion_text = Type::getStringFromFQCLN(
$fq_class_name,
$aliases && $aliases->namespace ? $aliases->namespace : null,
2021-09-26 22:57:04 +02:00
$aliases->uses_flipped ?? [],
2022-12-18 17:15:15 +01:00
null,
);
if ($aliases
&& !$fq_suggestion
&& $aliases->namespace
&& $insertion_text === '\\' . $fq_class_name
&& $aliases->namespace_first_stmt_start
) {
$file_contents = $this->getFileContents($file_path);
$class_name = preg_replace('/^.*\\\/', '', $fq_class_name, 1);
if ($aliases->uses_end) {
$position = self::getPositionFromOffset($aliases->uses_end, $file_contents);
2021-12-03 20:11:20 +01:00
$extra_edits[] = new TextEdit(
new Range(
$position,
2022-12-18 17:15:15 +01:00
$position,
),
2022-12-18 17:15:15 +01:00
"\n" . 'use ' . $fq_class_name . ';',
);
} else {
$position = self::getPositionFromOffset($aliases->namespace_first_stmt_start, $file_contents);
2021-12-03 20:11:20 +01:00
$extra_edits[] = new TextEdit(
new Range(
$position,
2022-12-18 17:15:15 +01:00
$position,
),
2022-12-18 17:15:15 +01:00
'use ' . $fq_class_name . ';' . "\n" . "\n",
);
}
$insertion_text = $class_name;
}
try {
$class_storage = $this->classlike_storage_provider->get($fq_class_name);
$description = $class_storage->description;
2021-12-03 21:40:18 +01:00
} catch (Exception $e) {
$description = null;
}
2021-12-03 20:11:20 +01:00
$completion_items[] = new CompletionItem(
$fq_class_name,
2021-12-03 20:11:20 +01:00
CompletionItemKind::CLASS_,
null,
$description,
null,
$fq_class_name,
$insertion_text,
null,
2022-12-18 17:15:15 +01:00
$extra_edits,
);
}
$functions = $this->functions->getMatchingFunctionNames($type_string, $offset, $file_path, $this);
$namespace_map = [];
if ($aliases) {
$namespace_map += $aliases->uses_flipped;
if ($aliases->namespace) {
$namespace_map[$aliases->namespace] = '';
}
}
// Sort the map by longest first, so we replace most specific
// used namespaces first.
ksort($namespace_map);
$namespace_map = array_reverse($namespace_map);
foreach ($functions as $function_lowercase => $function) {
// Transform FQFN relative to all uses namespaces
$function_name = $function->cased_name;
if (!$function_name) {
continue;
}
$in_namespace_map = false;
foreach ($namespace_map as $namespace_name => $namespace_alias) {
if (strpos($function_lowercase, $namespace_name . '\\') === 0) {
$function_name = $namespace_alias . '\\' . substr($function_name, strlen($namespace_name) + 1);
$in_namespace_map = true;
}
}
// If the function is not use'd, and it's not a global function
// prepend it with a backslash.
if (!$in_namespace_map && strpos($function_name, '\\') !== false) {
$function_name = '\\' . $function_name;
}
2021-12-03 20:11:20 +01:00
$completion_items[] = new CompletionItem(
$function_name,
2021-12-03 20:11:20 +01:00
CompletionItemKind::FUNCTION,
2022-12-20 21:52:25 +01:00
$function->getCompletionSignature(),
$function->description,
null,
$function_name,
2021-02-15 19:21:05 +01:00
$function_name . (count($function->params) !== 0 ? '($0)' : '()'),
2021-02-15 20:05:54 +01:00
null,
null,
new Command('Trigger parameter hints', 'editor.action.triggerParameterHints'),
2021-02-15 20:05:54 +01:00
null,
2022-12-18 17:15:15 +01:00
2,
);
}
return $completion_items;
}
/**
2021-12-04 03:37:19 +01:00
* @return list<CompletionItem>
*/
2021-12-13 16:28:14 +01:00
public function getCompletionItemsForType(Union $type): array
{
$completion_items = [];
foreach ($type->getAtomicTypes() as $atomic_type) {
2021-12-13 04:45:57 +01:00
if ($atomic_type instanceof TBool) {
$bools = (string) $atomic_type === 'bool' ? ['true', 'false'] : [(string) $atomic_type];
foreach ($bools as $property_name) {
2021-12-03 20:11:20 +01:00
$completion_items[] = new CompletionItem(
$property_name,
2021-12-03 20:11:20 +01:00
CompletionItemKind::VALUE,
'bool',
null,
null,
null,
2022-12-18 17:15:15 +01:00
$property_name,
);
}
2021-12-13 04:45:57 +01:00
} elseif ($atomic_type instanceof TLiteralString) {
2021-12-03 20:11:20 +01:00
$completion_items[] = new CompletionItem(
$atomic_type->value,
2021-12-03 20:11:20 +01:00
CompletionItemKind::VALUE,
$atomic_type->getId(),
null,
null,
null,
2022-12-18 17:15:15 +01:00
"'$atomic_type->value'",
);
2021-12-13 04:45:57 +01:00
} elseif ($atomic_type instanceof TLiteralInt) {
2021-12-03 20:11:20 +01:00
$completion_items[] = new CompletionItem(
(string) $atomic_type->value,
2021-12-03 20:11:20 +01:00
CompletionItemKind::VALUE,
$atomic_type->getId(),
null,
null,
null,
2022-12-18 17:15:15 +01:00
(string) $atomic_type->value,
);
2021-12-13 04:45:57 +01:00
} elseif ($atomic_type instanceof TClassConstant) {
$const = $atomic_type->fq_classlike_name . '::' . $atomic_type->const_name;
2021-12-03 20:11:20 +01:00
$completion_items[] = new CompletionItem(
$const,
2021-12-03 20:11:20 +01:00
CompletionItemKind::VALUE,
$atomic_type->getId(),
null,
null,
null,
2022-12-18 17:15:15 +01:00
$const,
);
}
}
return $completion_items;
}
/**
2021-12-04 03:37:19 +01:00
* @return list<CompletionItem>
*/
public function getCompletionItemsForArrayKeys(
string $type_string
): array {
$completion_items = [];
$type = Type::parseString($type_string);
foreach ($type->getAtomicTypes() as $atomic_type) {
2021-12-13 04:45:57 +01:00
if ($atomic_type instanceof TKeyedArray) {
foreach ($atomic_type->properties as $property_name => $property) {
2021-12-03 20:11:20 +01:00
$completion_items[] = new CompletionItem(
(string) $property_name,
2021-12-03 20:11:20 +01:00
CompletionItemKind::PROPERTY,
(string) $property,
null,
null,
null,
2022-12-18 17:15:15 +01:00
"'$property_name'",
);
}
}
}
return $completion_items;
}
private static function getPositionFromOffset(int $offset, string $file_contents): Position
{
$file_contents = substr($file_contents, 0, $offset);
$offsetLength = $offset - strlen($file_contents);
//PHP 8.0: Argument #3 ($offset) must be contained in argument #1 ($haystack)
if (($textlen = strlen($file_contents)) < $offsetLength) {
$offsetLength = $textlen;
}
$before_newline_count = strrpos($file_contents, "\n", $offsetLength);
return new Position(
substr_count($file_contents, "\n"),
2022-12-18 17:15:15 +01:00
$offset - (int)$before_newline_count - 1,
);
}
2022-12-20 21:52:25 +01:00
public function addTemporaryFileChanges(string $file_path, string $new_content, ?int $version = null): void
{
2022-12-20 21:52:25 +01:00
$this->file_provider->addTemporaryFileChanges($file_path, $new_content, $version);
}
2018-11-02 02:52:39 +01:00
public function removeTemporaryFileChanges(string $file_path): void
{
$this->file_provider->removeTemporaryFileChanges($file_path);
}
2019-02-18 21:54:23 +01:00
/**
* Checks if type is a subtype of other
*
* Given two types, checks if `$input_type` is a subtype of `$container_type`.
2021-12-13 16:28:14 +01:00
* If you consider `Union` as a set of types, this will tell you
* if `$input_type` is fully contained in `$container_type`,
*
* $input_type $container_type
*
* Useful for emitting issues like InvalidArgument, where argument at the call site
* should be a subset of the function parameter type.
2019-02-18 21:54:23 +01:00
*/
public function isTypeContainedByType(
2021-12-13 16:28:14 +01:00
Union $input_type,
Union $container_type
): bool {
2020-07-22 01:40:35 +02:00
return UnionTypeComparator::isContainedBy($this, $input_type, $container_type);
}
/**
* Checks if type has any part that is a subtype of other
*
* Given two types, checks if *any part* of `$input_type` is a subtype of `$container_type`.
2021-12-13 16:28:14 +01:00
* If you consider `Union` as a set of types, this will tell you if intersection
* of `$input_type` with `$container_type` is not empty.
*
2019-02-20 17:06:51 +01:00
* $input_type $container_type , e.g. they are not disjoint.
*
* Useful for emitting issues like PossiblyInvalidArgument, where argument at the call
* site should be a subtype of the function parameter type, but it's has some types that are
* not a subtype of the required type.
*/
public function canTypeBeContainedByType(
2021-12-13 16:28:14 +01:00
Union $input_type,
Union $container_type
): bool {
2020-07-22 01:40:35 +02:00
return UnionTypeComparator::canBeContainedBy($this, $input_type, $container_type);
}
/**
* Extracts key and value types from a traversable object (or iterable)
*
* Given an iterable type (*but not TArray*) returns a tuple of it's key/value types.
* First element of the tuple holds key type, second has the value type.
*
* Example:
* ```php
* $codebase->getKeyValueParamsForTraversableObject(Type::parseString('iterable<int,string>'))
* // returns [Union(TInt), Union(TString)]
* ```
*
2021-12-13 16:28:14 +01:00
* @return array{Union, Union}
*/
2021-12-13 16:28:14 +01:00
public function getKeyValueParamsForTraversableObject(Atomic $type): array
{
$key_type = null;
$value_type = null;
ForeachAnalyzer::getKeyValueParamsForTraversableObject($type, $this, $key_type, $value_type);
return [
$key_type ?? Type::getMixed(),
$value_type ?? Type::getMixed(),
];
}
/**
* @param array<string, mixed> $phantom_classes
* @psalm-suppress PossiblyUnusedMethod part of the public API
*/
public function queueClassLikeForScanning(
string $fq_classlike_name,
bool $analyze_too = false,
bool $store_failure = true,
array $phantom_classes = []
): void {
$this->scanner->queueClassLikeForScanning($fq_classlike_name, $analyze_too, $store_failure, $phantom_classes);
}
/**
* @param array<string> $taints
* @psalm-suppress PossiblyUnusedMethod
*/
public function addTaintSource(
2021-12-13 16:28:14 +01:00
Union $expr_type,
string $taint_id,
2021-12-03 20:11:20 +01:00
array $taints = TaintKindGroup::ALL_INPUT,
?CodeLocation $code_location = null
): Union {
if (!$this->taint_flow_graph) {
return $expr_type;
}
2021-12-03 20:11:20 +01:00
$source = new TaintSource(
$taint_id,
$taint_id,
$code_location,
null,
2022-12-18 17:15:15 +01:00
$taints,
);
$this->taint_flow_graph->addSource($source);
return $expr_type->addParentNodes([$source->id => $source]);
}
2020-07-16 22:02:26 +02:00
/**
* @param array<string> $taints
* @psalm-suppress PossiblyUnusedMethod
*/
public function addTaintSink(
string $taint_id,
2021-12-03 20:11:20 +01:00
array $taints = TaintKindGroup::ALL_INPUT,
2020-07-16 22:02:26 +02:00
?CodeLocation $code_location = null
): void {
if (!$this->taint_flow_graph) {
2020-07-16 22:02:26 +02:00
return;
}
2021-12-03 20:11:20 +01:00
$sink = new TaintSink(
2020-07-16 22:02:26 +02:00
$taint_id,
$taint_id,
$code_location,
null,
2022-12-18 17:15:15 +01:00
$taints,
2020-07-16 22:02:26 +02:00
);
$this->taint_flow_graph->addSink($sink);
2020-07-16 22:02:26 +02:00
}
2022-02-15 01:18:49 +01:00
public function getMinorAnalysisPhpVersion(): int
2022-02-15 01:18:49 +01:00
{
return self::transformPhpVersionId($this->analysis_php_version_id % 10_000, 100);
}
2022-02-15 01:18:49 +01:00
public function getMajorAnalysisPhpVersion(): int
{
return self::transformPhpVersionId($this->analysis_php_version_id, 10_000);
}
2022-02-15 01:18:49 +01:00
public static function transformPhpVersionId(int $php_version_id, int $div): int
{
return intdiv($php_version_id, $div);
2022-02-15 01:18:49 +01:00
}
}