2018-01-21 19:38:51 +01:00
|
|
|
<?php
|
|
|
|
namespace Psalm;
|
|
|
|
|
2019-07-05 22:24:00 +02:00
|
|
|
use LanguageServerProtocol\Command;
|
|
|
|
use LanguageServerProtocol\Position;
|
|
|
|
use LanguageServerProtocol\Range;
|
2018-01-21 19:38:51 +01:00
|
|
|
use PhpParser;
|
2021-01-28 17:18:28 +01:00
|
|
|
use Psalm\Internal\Analyzer\NamespaceAnalyzer;
|
2021-06-08 04:55:21 +02:00
|
|
|
use Psalm\Internal\Analyzer\ProjectAnalyzer;
|
2019-07-05 22:24:00 +02:00
|
|
|
use Psalm\Internal\Analyzer\Statements\Block\ForeachAnalyzer;
|
2021-01-28 17:18:28 +01:00
|
|
|
use Psalm\Internal\Analyzer\Statements\Expression\Fetch\ConstFetchAnalyzer;
|
2021-01-29 00:58:02 +01:00
|
|
|
use Psalm\Internal\Analyzer\Statements\Expression\Fetch\VariableFetchAnalyzer;
|
2021-06-08 04:55:21 +02:00
|
|
|
use Psalm\Internal\Analyzer\StatementsAnalyzer;
|
2020-05-25 19:10:06 +02:00
|
|
|
use Psalm\Internal\Codebase\InternalCallMapHandler;
|
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;
|
2019-05-30 16:30:41 +02:00
|
|
|
use Psalm\Progress\Progress;
|
|
|
|
use Psalm\Progress\VoidProgress;
|
2018-01-21 19:38:51 +01:00
|
|
|
use Psalm\Storage\ClassLikeStorage;
|
|
|
|
use Psalm\Storage\FileStorage;
|
|
|
|
use Psalm\Storage\FunctionLikeStorage;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
|
|
|
use function array_combine;
|
|
|
|
use function array_merge;
|
|
|
|
use function array_pop;
|
|
|
|
use function array_reverse;
|
|
|
|
use function count;
|
|
|
|
use function dirname;
|
|
|
|
use function error_log;
|
|
|
|
use function explode;
|
|
|
|
use function implode;
|
|
|
|
use function in_array;
|
2020-02-15 02:54:26 +01:00
|
|
|
use function is_string;
|
2021-06-08 04:55:21 +02:00
|
|
|
use function krsort;
|
|
|
|
use function ksort;
|
|
|
|
use function preg_match;
|
2019-07-05 22:24:00 +02:00
|
|
|
use function strlen;
|
2019-06-26 22:52:29 +02:00
|
|
|
use function strpos;
|
2019-07-05 22:24:00 +02:00
|
|
|
use function strrpos;
|
2019-06-26 22:52:29 +02:00
|
|
|
use function strtolower;
|
|
|
|
use function substr;
|
|
|
|
use function substr_count;
|
2021-06-08 04:55:21 +02:00
|
|
|
|
|
|
|
use const PHP_MAJOR_VERSION;
|
|
|
|
use const PHP_MINOR_VERSION;
|
2018-01-21 19:38:51 +01:00
|
|
|
|
|
|
|
class Codebase
|
|
|
|
{
|
|
|
|
/**
|
2018-04-05 20:40:41 +02:00
|
|
|
* @var Config
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
|
|
|
public $config;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A map of fully-qualified use declarations to the files
|
|
|
|
* that reference them (keyed by filename)
|
|
|
|
*
|
2020-12-29 12:42:12 +01:00
|
|
|
* @var array<lowercase-string, array<int, \Psalm\CodeLocation>>
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
|
|
|
public $use_referencing_locations = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A map of file names to the classes that they contain explicit references to
|
|
|
|
* used in collaboration with use_referencing_locations
|
|
|
|
*
|
2020-12-29 12:42:12 +01:00
|
|
|
* @var array<string, array<lowercase-string, bool>>
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
|
|
|
public $use_referencing_files = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var FileStorageProvider
|
|
|
|
*/
|
|
|
|
public $file_storage_provider;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var ClassLikeStorageProvider
|
|
|
|
*/
|
|
|
|
public $classlike_storage_provider;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $collect_references = false;
|
|
|
|
|
2019-04-13 00:28:07 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $collect_locations = false;
|
|
|
|
|
2019-04-17 19:15:06 +02:00
|
|
|
/**
|
|
|
|
* @var null|'always'|'auto'
|
|
|
|
*/
|
|
|
|
public $find_unused_code = null;
|
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
/**
|
|
|
|
* @var FileProvider
|
|
|
|
*/
|
2018-10-26 06:59:14 +02:00
|
|
|
public $file_provider;
|
2018-01-21 19:38:51 +01:00
|
|
|
|
2018-10-07 04:58:21 +02:00
|
|
|
/**
|
|
|
|
* @var FileReferenceProvider
|
|
|
|
*/
|
|
|
|
public $file_reference_provider;
|
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
/**
|
|
|
|
* @var StatementsProvider
|
|
|
|
*/
|
2018-02-19 17:53:30 +01:00
|
|
|
public $statements_provider;
|
2018-01-21 19:38:51 +01:00
|
|
|
|
|
|
|
/**
|
2019-05-30 16:30:41 +02:00
|
|
|
* @var Progress
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2019-05-30 16:30:41 +02:00
|
|
|
private $progress;
|
2018-01-21 19:38:51 +01:00
|
|
|
|
|
|
|
/**
|
2018-02-04 00:52:35 +01:00
|
|
|
* @var array<string, Type\Union>
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2018-02-04 00:52:35 +01:00
|
|
|
private static $stubbed_constants = [];
|
2018-01-21 19:38:51 +01:00
|
|
|
|
2018-06-30 21:29:37 +02:00
|
|
|
/**
|
|
|
|
* Whether to register autoloaded information
|
|
|
|
*
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $register_autoload_files = false;
|
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
/**
|
2018-02-04 00:52:35 +01:00
|
|
|
* Whether to log functions just at the file level or globally (for stubs)
|
|
|
|
*
|
|
|
|
* @var bool
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2018-06-30 21:29:37 +02:00
|
|
|
public $register_stub_files = false;
|
2018-01-21 19:38:51 +01:00
|
|
|
|
2019-03-05 21:45:09 +01:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $find_unused_variables = false;
|
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
/**
|
2018-11-12 16:46:55 +01:00
|
|
|
* @var Internal\Codebase\Scanner
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2018-02-04 00:52:35 +01:00
|
|
|
public $scanner;
|
2018-01-21 19:38:51 +01:00
|
|
|
|
|
|
|
/**
|
2018-11-12 16:46:55 +01:00
|
|
|
* @var Internal\Codebase\Analyzer
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2018-02-04 00:52:35 +01:00
|
|
|
public $analyzer;
|
2018-01-21 19:38:51 +01:00
|
|
|
|
|
|
|
/**
|
2018-11-12 16:46:55 +01:00
|
|
|
* @var Internal\Codebase\Functions
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2018-02-04 00:52:35 +01:00
|
|
|
public $functions;
|
2018-01-21 19:38:51 +01:00
|
|
|
|
|
|
|
/**
|
2018-11-12 16:46:55 +01:00
|
|
|
* @var Internal\Codebase\ClassLikes
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2018-02-04 00:52:35 +01:00
|
|
|
public $classlikes;
|
2018-01-21 19:38:51 +01:00
|
|
|
|
2018-01-31 22:08:52 +01:00
|
|
|
/**
|
2018-11-12 16:46:55 +01:00
|
|
|
* @var Internal\Codebase\Methods
|
2018-01-31 22:08:52 +01:00
|
|
|
*/
|
2018-02-04 00:52:35 +01:00
|
|
|
public $methods;
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2018-02-09 00:14:28 +01:00
|
|
|
/**
|
2018-11-12 16:46:55 +01:00
|
|
|
* @var Internal\Codebase\Properties
|
2018-02-09 00:14:28 +01:00
|
|
|
*/
|
|
|
|
public $properties;
|
|
|
|
|
2018-01-31 22:08:52 +01:00
|
|
|
/**
|
2018-11-12 16:46:55 +01:00
|
|
|
* @var Internal\Codebase\Populator
|
2018-01-31 22:08:52 +01:00
|
|
|
*/
|
2018-02-04 00:52:35 +01:00
|
|
|
public $populator;
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2019-08-04 16:37:36 +02:00
|
|
|
/**
|
2020-09-25 06:37:40 +02:00
|
|
|
* @var ?Internal\Codebase\TaintFlowGraph
|
2019-08-04 16:37:36 +02:00
|
|
|
*/
|
2020-09-25 06:37:40 +02:00
|
|
|
public $taint_flow_graph = null;
|
2019-08-04 16:37:36 +02:00
|
|
|
|
2018-10-17 21:52:26 +02:00
|
|
|
/**
|
|
|
|
* @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;
|
|
|
|
|
2019-06-01 06:56:54 +02:00
|
|
|
/**
|
2020-02-15 02:54:26 +01:00
|
|
|
* @var array<lowercase-string, string>
|
2019-06-01 06:56:54 +02:00
|
|
|
*/
|
2019-06-02 18:02:32 +02:00
|
|
|
public $methods_to_move = [];
|
|
|
|
|
|
|
|
/**
|
2020-02-15 02:54:26 +01:00
|
|
|
* @var array<lowercase-string, string>
|
2019-06-02 18:02:32 +02:00
|
|
|
*/
|
|
|
|
public $methods_to_rename = [];
|
2019-06-01 06:56:54 +02:00
|
|
|
|
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
|
|
|
/**
|
2020-02-15 02:54:26 +01:00
|
|
|
* @var array<lowercase-string, string>
|
2019-06-04 22:36:32 +02:00
|
|
|
*/
|
|
|
|
public $classes_to_move = [];
|
|
|
|
|
2019-06-01 06:56:54 +02:00
|
|
|
/**
|
2020-12-29 12:42:12 +01:00
|
|
|
* @var array<lowercase-string, string>
|
2019-06-01 06:56:54 +02:00
|
|
|
*/
|
|
|
|
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 = [];
|
|
|
|
|
2019-05-13 20:38:18 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $allow_backwards_incompatible_changes = true;
|
|
|
|
|
2019-02-07 18:25:57 +01:00
|
|
|
/**
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
public $php_major_version = PHP_MAJOR_VERSION;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
public $php_minor_version = PHP_MINOR_VERSION;
|
|
|
|
|
2019-08-18 20:27:50 +02:00
|
|
|
/**
|
|
|
|
* @var bool
|
|
|
|
*/
|
|
|
|
public $track_unused_suppressions = false;
|
2019-08-04 16:37:36 +02:00
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
public function __construct(
|
|
|
|
Config $config,
|
2018-09-28 22:18:45 +02:00
|
|
|
Providers $providers,
|
2020-09-07 01:36:47 +02:00
|
|
|
?Progress $progress = null
|
2018-01-21 19:38:51 +01:00
|
|
|
) {
|
2019-05-30 16:30:41 +02:00
|
|
|
if ($progress === null) {
|
|
|
|
$progress = new VoidProgress();
|
|
|
|
}
|
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
$this->config = $config;
|
2018-09-28 22:18:45 +02:00
|
|
|
$this->file_storage_provider = $providers->file_storage_provider;
|
|
|
|
$this->classlike_storage_provider = $providers->classlike_storage_provider;
|
2019-05-30 16:30:41 +02:00
|
|
|
$this->progress = $progress;
|
2018-09-28 22:18:45 +02:00
|
|
|
$this->file_provider = $providers->file_provider;
|
2018-10-07 04:58:21 +02:00
|
|
|
$this->file_reference_provider = $providers->file_reference_provider;
|
2018-09-28 22:18:45 +02:00
|
|
|
$this->statements_provider = $providers->statements_provider;
|
2018-01-21 19:38:51 +01:00
|
|
|
|
|
|
|
self::$stubbed_constants = [];
|
|
|
|
|
2020-01-03 19:10:24 +01:00
|
|
|
$reflection = new Internal\Codebase\Reflection($providers->classlike_storage_provider, $this);
|
2018-01-21 19:38:51 +01:00
|
|
|
|
2018-11-12 16:46:55 +01:00
|
|
|
$this->scanner = new Internal\Codebase\Scanner(
|
2018-02-04 00:52:35 +01:00
|
|
|
$this,
|
|
|
|
$config,
|
2018-09-28 22:18:45 +02:00
|
|
|
$providers->file_storage_provider,
|
|
|
|
$providers->file_provider,
|
2020-01-03 19:10:24 +01:00
|
|
|
$reflection,
|
2018-09-28 22:18:45 +02:00
|
|
|
$providers->file_reference_provider,
|
2019-05-30 16:30:41 +02:00
|
|
|
$progress
|
2018-02-04 00:52:35 +01:00
|
|
|
);
|
2018-01-21 19:38:51 +01:00
|
|
|
|
2018-09-28 22:18:45 +02:00
|
|
|
$this->loadAnalyzer();
|
2018-01-21 19:38:51 +01:00
|
|
|
|
2020-01-03 19:10:24 +01:00
|
|
|
$this->functions = new Internal\Codebase\Functions($providers->file_storage_provider, $reflection);
|
2018-12-21 17:32:44 +01:00
|
|
|
|
2018-11-12 16:46:55 +01:00
|
|
|
$this->classlikes = new Internal\Codebase\ClassLikes(
|
2018-10-07 02:11:19 +02:00
|
|
|
$this->config,
|
2018-09-28 22:18:45 +02:00
|
|
|
$providers->classlike_storage_provider,
|
2019-04-13 00:28:07 +02:00
|
|
|
$providers->file_reference_provider,
|
2020-03-03 04:27:54 +01:00
|
|
|
$providers->statements_provider,
|
2018-12-21 17:32:44 +01:00
|
|
|
$this->scanner
|
2018-02-04 00:52:35 +01:00
|
|
|
);
|
2018-12-21 17:32:44 +01:00
|
|
|
|
2021-01-22 01:38:17 +01:00
|
|
|
$this->properties = new Internal\Codebase\Properties(
|
|
|
|
$providers->classlike_storage_provider,
|
|
|
|
$providers->file_reference_provider,
|
|
|
|
$this->classlikes
|
|
|
|
);
|
|
|
|
|
2018-12-21 17:32:44 +01:00
|
|
|
$this->methods = new Internal\Codebase\Methods(
|
|
|
|
$providers->classlike_storage_provider,
|
|
|
|
$providers->file_reference_provider,
|
|
|
|
$this->classlikes
|
|
|
|
);
|
|
|
|
|
2018-11-12 16:46:55 +01:00
|
|
|
$this->populator = new Internal\Codebase\Populator(
|
2018-02-04 00:52:35 +01:00
|
|
|
$config,
|
2018-09-28 22:18:45 +02:00
|
|
|
$providers->classlike_storage_provider,
|
|
|
|
$providers->file_storage_provider,
|
2018-02-04 00:52:35 +01:00
|
|
|
$this->classlikes,
|
2018-09-28 22:18:45 +02:00
|
|
|
$providers->file_reference_provider,
|
2019-05-30 16:30:41 +02:00
|
|
|
$progress
|
2018-02-04 00:52:35 +01:00
|
|
|
);
|
2018-10-07 02:11:19 +02:00
|
|
|
|
|
|
|
$this->loadAnalyzer();
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
private function loadAnalyzer(): void
|
2018-09-28 22:18:45 +02:00
|
|
|
{
|
2018-11-12 16:46:55 +01:00
|
|
|
$this->analyzer = new Internal\Codebase\Analyzer(
|
2018-09-28 22:18:45 +02:00
|
|
|
$this->config,
|
|
|
|
$this->file_provider,
|
|
|
|
$this->file_storage_provider,
|
2019-05-30 16:30:41 +02:00
|
|
|
$this->progress
|
2018-09-28 22:18:45 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-10-16 21:59:11 +02:00
|
|
|
* @param array<string> $candidate_files
|
2018-09-28 22:18:45 +02:00
|
|
|
*
|
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
public function reloadFiles(ProjectAnalyzer $project_analyzer, array $candidate_files): void
|
2018-09-28 22:18:45 +02:00
|
|
|
{
|
|
|
|
$this->loadAnalyzer();
|
|
|
|
|
2019-06-30 03:32:26 +02:00
|
|
|
$this->file_reference_provider->loadReferenceCache(false);
|
2018-09-28 22:18:45 +02:00
|
|
|
|
2019-07-28 16:05:45 +02:00
|
|
|
Internal\Analyzer\FunctionLikeAnalyzer::clearCache();
|
|
|
|
|
2018-10-16 21:59:11 +02:00
|
|
|
if (!$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);
|
2018-09-28 22:18:45 +02:00
|
|
|
|
|
|
|
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)) {
|
2018-09-28 22:18:45 +02:00
|
|
|
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));
|
2019-06-30 03:32:26 +02:00
|
|
|
|
2018-09-28 22:18:45 +02:00
|
|
|
$this->scanner->scanFiles($this->classlikes);
|
|
|
|
|
2018-11-11 18:19:53 +01:00
|
|
|
$this->file_reference_provider->updateReferenceCache($this, $referenced_files);
|
2018-09-28 22:18:45 +02:00
|
|
|
|
2019-09-14 20:26:31 +02:00
|
|
|
$this->populator->populateCodebase();
|
2018-09-28 22:18:45 +02:00
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function enterServerMode(): void
|
2018-10-17 21:52:26 +02:00
|
|
|
{
|
|
|
|
$this->server_mode = true;
|
2019-02-24 07:33:25 +01:00
|
|
|
$this->store_node_types = true;
|
2018-10-17 21:52:26 +02:00
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function collectLocations(): void
|
2019-04-13 00:28:07 +02:00
|
|
|
{
|
|
|
|
$this->collect_locations = true;
|
|
|
|
$this->classlikes->collect_locations = true;
|
|
|
|
$this->methods->collect_locations = true;
|
|
|
|
$this->properties->collect_locations = true;
|
|
|
|
}
|
|
|
|
|
2018-02-17 23:45:30 +01:00
|
|
|
/**
|
2019-04-17 19:15:06 +02:00
|
|
|
* @param 'always'|'auto' $find_unused_code
|
|
|
|
*
|
2018-02-17 23:45:30 +01:00
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
public function reportUnusedCode(string $find_unused_code = 'auto'): void
|
2018-02-17 23:45:30 +01:00
|
|
|
{
|
2019-04-17 19:15:06 +02:00
|
|
|
$this->collect_references = true;
|
|
|
|
$this->classlikes->collect_references = true;
|
|
|
|
$this->find_unused_code = $find_unused_code;
|
2019-03-05 21:45:09 +01:00
|
|
|
$this->find_unused_variables = true;
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function reportUnusedVariables(): void
|
2019-03-05 21:45:09 +01:00
|
|
|
{
|
|
|
|
$this->collect_references = true;
|
|
|
|
$this->find_unused_variables = true;
|
2018-02-17 23:45:30 +01:00
|
|
|
}
|
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
/**
|
2018-02-04 00:52:35 +01:00
|
|
|
* @param array<string, string> $files_to_analyze
|
2018-01-21 19:38:51 +01:00
|
|
|
*
|
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
public function addFilesToAnalyze(array $files_to_analyze): void
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
2018-02-04 00:52:35 +01:00
|
|
|
$this->scanner->addFilesToDeepScan($files_to_analyze);
|
2020-03-26 19:22:06 +01:00
|
|
|
$this->analyzer->addFilesToAnalyze($files_to_analyze);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
2018-02-01 07:10:27 +01:00
|
|
|
/**
|
2018-02-04 00:52:35 +01:00
|
|
|
* Scans all files their related files
|
2018-02-01 07:10:27 +01:00
|
|
|
*
|
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
public function scanFiles(int $threads = 1): void
|
2018-02-01 07:10:27 +01:00
|
|
|
{
|
2018-10-11 19:58:39 +02:00
|
|
|
$has_changes = $this->scanner->scanFiles($this->classlikes, $threads);
|
2018-02-01 07:10:27 +01:00
|
|
|
|
2018-02-04 00:52:35 +01:00
|
|
|
if ($has_changes) {
|
2019-09-14 20:26:31 +02:00
|
|
|
$this->populator->populateCodebase();
|
2018-02-01 07:10:27 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public function getFileContents(string $file_path): string
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
2018-02-04 00:52:35 +01:00
|
|
|
return $this->file_provider->getContents($file_path);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-09 01:06:16 +02:00
|
|
|
* @return list<PhpParser\Node\Stmt>
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
public function getStatementsForFile(string $file_path): array
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
2018-02-04 00:52:35 +01:00
|
|
|
return $this->statements_provider->getStatementsForFile(
|
|
|
|
$file_path,
|
2020-08-09 22:23:43 +02:00
|
|
|
$this->php_major_version . '.' . $this->php_minor_version,
|
2019-05-30 16:30:41 +02:00
|
|
|
$this->progress
|
2018-02-04 00:52:35 +01:00
|
|
|
);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public function createClassLikeStorage(string $fq_classlike_name): ClassLikeStorage
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
2018-02-04 00:52:35 +01:00
|
|
|
return $this->classlike_storage_provider->create($fq_classlike_name);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function cacheClassLikeStorage(ClassLikeStorage $classlike_storage, string $file_path): void
|
2018-02-19 06:27:39 +01:00
|
|
|
{
|
|
|
|
$file_contents = $this->file_provider->getContents($file_path);
|
2018-09-28 22:18:45 +02:00
|
|
|
|
|
|
|
if ($this->classlike_storage_provider->cache) {
|
|
|
|
$this->classlike_storage_provider->cache->writeToCache($classlike_storage, $file_path, $file_contents);
|
|
|
|
}
|
2018-02-19 06:27:39 +01:00
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02: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);
|
2020-02-15 02:54:26 +01:00
|
|
|
$storage = $this->classlike_storage_provider->exhume(
|
|
|
|
$fq_classlike_name,
|
|
|
|
$file_path,
|
|
|
|
$file_contents
|
|
|
|
);
|
2018-02-19 06:27:39 +01:00
|
|
|
|
|
|
|
if ($storage->is_trait) {
|
2020-02-15 02:54:26 +01:00
|
|
|
$this->classlikes->addFullyQualifiedTraitName($storage->name, $file_path);
|
2018-02-19 06:27:39 +01:00
|
|
|
} elseif ($storage->is_interface) {
|
2020-02-15 02:54:26 +01:00
|
|
|
$this->classlikes->addFullyQualifiedInterfaceName($storage->name, $file_path);
|
2018-02-19 06:27:39 +01:00
|
|
|
} else {
|
2020-02-15 02:54:26 +01:00
|
|
|
$this->classlikes->addFullyQualifiedClassName($storage->name, $file_path);
|
2018-02-19 06:27:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public static function getPsalmTypeFromReflection(?\ReflectionType $type) : Type\Union
|
2019-01-27 23:26:32 +01:00
|
|
|
{
|
|
|
|
return \Psalm\Internal\Codebase\Reflection::getPsalmTypeFromReflectionType($type);
|
|
|
|
}
|
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public function createFileStorageForPath(string $file_path): FileStorage
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
2018-02-04 00:52:35 +01:00
|
|
|
return $this->file_storage_provider->create($file_path);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-10-17 18:36:44 +02:00
|
|
|
* @return array<int, CodeLocation>
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2020-09-07 01:36:47 +02:00
|
|
|
public function findReferencesToSymbol(string $symbol): array
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
2019-04-13 00:28:07 +02:00
|
|
|
if (!$this->collect_locations) {
|
2018-02-04 00:52:35 +01:00
|
|
|
throw new \UnexpectedValueException('Should not be checking references');
|
|
|
|
}
|
|
|
|
|
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);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-10-17 18:36:44 +02:00
|
|
|
* @return array<int, CodeLocation>
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2020-09-07 01:36:47 +02:00
|
|
|
public function findReferencesToMethod(string $method_id): array
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
2019-04-13 00:28:07 +02:00
|
|
|
return $this->file_reference_provider->getClassMethodLocations(strtolower($method_id));
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
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
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function findReferencesToProperty(string $property_id): array
|
2019-01-22 17:10:37 +01:00
|
|
|
{
|
2020-09-02 06:17:41 +02:00
|
|
|
[$fq_class_name, $property_name] = explode('::', $property_id);
|
2019-07-05 22:24:00 +02:00
|
|
|
|
2019-04-13 00:28:07 +02:00
|
|
|
return $this->file_reference_provider->getClassPropertyLocations(
|
|
|
|
strtolower($fq_class_name) . '::' . $property_name
|
|
|
|
);
|
2019-01-22 17:10:37 +01:00
|
|
|
}
|
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
/**
|
2020-10-17 18:36:44 +02:00
|
|
|
* @return CodeLocation[]
|
|
|
|
*
|
|
|
|
* @psalm-return array<int, CodeLocation>
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2020-09-07 01:36:47 +02:00
|
|
|
public function findReferencesToClassLike(string $fq_class_name): array
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
2018-02-04 00:52:35 +01:00
|
|
|
$fq_class_name_lc = strtolower($fq_class_name);
|
2019-04-13 00:28:07 +02:00
|
|
|
$locations = $this->file_reference_provider->getClassLocations($fq_class_name_lc);
|
2018-01-21 19:38:51 +01:00
|
|
|
|
2018-02-04 00:52:35 +01:00
|
|
|
if (isset($this->use_referencing_locations[$fq_class_name_lc])) {
|
2019-04-13 00:28:07 +02:00
|
|
|
$locations = array_merge($locations, $this->use_referencing_locations[$fq_class_name_lc]);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
2019-04-13 00:28:07 +02:00
|
|
|
return $locations;
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
2020-10-17 18:36:44 +02:00
|
|
|
public function getClosureStorage(string $file_path, string $closure_id): Storage\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];
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new \UnexpectedValueException(
|
|
|
|
'Expecting ' . $closure_id . ' to have storage in ' . $file_path
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function addGlobalConstantType(string $const_id, Type\Union $type): void
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
|
|
|
self::$stubbed_constants[$const_id] = $type;
|
|
|
|
}
|
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public function getStubbedConstantType(string $const_id): ?Type\Union
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
2021-09-26 22:06:24 +02:00
|
|
|
return self::$stubbed_constants[$const_id] ?? null;
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
2020-01-09 05:48:42 +01:00
|
|
|
/**
|
|
|
|
* @return array<string, Type\Union>
|
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function getAllStubbedConstants(): array
|
2020-01-09 05:48:42 +01:00
|
|
|
{
|
|
|
|
return self::$stubbed_constants;
|
|
|
|
}
|
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public function fileExists(string $file_path): bool
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
2018-02-04 00:52:35 +01:00
|
|
|
return $this->file_provider->fileExists($file_path);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-02-04 00:52:35 +01:00
|
|
|
* Check whether a class/interface exists
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2020-03-26 17:35:27 +01:00
|
|
|
public function classOrInterfaceExists(
|
2020-09-07 01:36:47 +02:00
|
|
|
string $fq_class_name,
|
|
|
|
?CodeLocation $code_location = null,
|
2020-03-29 03:39:44 +02:00
|
|
|
?string $calling_fq_class_name = null,
|
2020-03-26 17:35:27 +01:00
|
|
|
?string $calling_method_id = null
|
2020-09-04 22:26:33 +02:00
|
|
|
): bool {
|
2020-03-29 03:39:44 +02:00
|
|
|
return $this->classlikes->classOrInterfaceExists(
|
|
|
|
$fq_class_name,
|
|
|
|
$code_location,
|
|
|
|
$calling_fq_class_name,
|
|
|
|
$calling_method_id
|
|
|
|
);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
2021-05-03 23:54:09 +02:00
|
|
|
/**
|
|
|
|
* 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,
|
|
|
|
$calling_method_id
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public function classExtendsOrImplements(string $fq_class_name, string $possible_parent): bool
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
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-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-02-04 00:52:35 +01:00
|
|
|
* Determine whether or not a given class exists
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2020-03-29 03:39:44 +02:00
|
|
|
public function classExists(
|
2020-09-07 01:36:47 +02:00
|
|
|
string $fq_class_name,
|
|
|
|
?CodeLocation $code_location = null,
|
2020-03-29 03:39:44 +02:00
|
|
|
?string $calling_fq_class_name = null,
|
|
|
|
?string $calling_method_id = null
|
2020-09-04 22:26:33 +02:00
|
|
|
): bool {
|
2020-03-29 03:39:44 +02:00
|
|
|
return $this->classlikes->classExists(
|
|
|
|
$fq_class_name,
|
|
|
|
$code_location,
|
|
|
|
$calling_fq_class_name,
|
|
|
|
$calling_method_id
|
|
|
|
);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-02-04 00:52:35 +01:00
|
|
|
* Determine whether or not a class extends a parent
|
2018-01-21 19:38:51 +01:00
|
|
|
*
|
2019-03-16 23:03:37 +01:00
|
|
|
* @throws \Psalm\Exception\UnpopulatedClasslikeException when called on unpopulated class
|
|
|
|
* @throws \InvalidArgumentException when class does not exist
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2020-09-07 01:36:47 +02:00
|
|
|
public function classExtends(string $fq_class_name, string $possible_parent): bool
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
2019-03-17 22:11:04 +01:00
|
|
|
return $this->classlikes->classExtends($fq_class_name, $possible_parent, true);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-02-04 00:52:35 +01:00
|
|
|
* Check whether a class implements an interface
|
2018-01-21 19:38:51 +01:00
|
|
|
*/
|
2020-09-07 01:36:47 +02:00
|
|
|
public function classImplements(string $fq_class_name, string $interface): bool
|
2018-01-21 19:38:51 +01:00
|
|
|
{
|
2018-02-04 00:52:35 +01:00
|
|
|
return $this->classlikes->classImplements($fq_class_name, $interface);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
2020-09-21 00:27:02 +02:00
|
|
|
|
2020-03-26 17:35:27 +01:00
|
|
|
public function interfaceExists(
|
2020-09-07 01:36:47 +02:00
|
|
|
string $fq_interface_name,
|
|
|
|
?CodeLocation $code_location = null,
|
2020-03-29 03:39:44 +02:00
|
|
|
?string $calling_fq_class_name = null,
|
2020-03-26 17:35:27 +01:00
|
|
|
?string $calling_method_id = null
|
2020-09-04 22:26:33 +02:00
|
|
|
): bool {
|
2020-03-29 03:39:44 +02:00
|
|
|
return $this->classlikes->interfaceExists(
|
|
|
|
$fq_interface_name,
|
|
|
|
$code_location,
|
|
|
|
$calling_fq_class_name,
|
|
|
|
$calling_method_id
|
|
|
|
);
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public function interfaceExtends(string $interface_name, string $possible_parent): bool
|
2018-01-31 22:08:52 +01:00
|
|
|
{
|
2018-02-04 00:52:35 +01:00
|
|
|
return $this->classlikes->interfaceExtends($interface_name, $possible_parent);
|
|
|
|
}
|
2018-01-31 22:08:52 +01:00
|
|
|
|
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
|
|
|
*/
|
2020-09-07 01:36:47 +02:00
|
|
|
public function getParentInterfaces(string $fq_interface_name): array
|
2018-02-04 00:52:35 +01:00
|
|
|
{
|
2020-02-15 02:54:26 +01:00
|
|
|
return $this->classlikes->getParentInterfaces(
|
|
|
|
$this->classlikes->getUnAliasedName($fq_interface_name)
|
|
|
|
);
|
2018-02-04 00:52:35 +01:00
|
|
|
}
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2018-02-04 00:52:35 +01:00
|
|
|
/**
|
|
|
|
* Determine whether or not a class has the correct casing
|
|
|
|
*/
|
2020-09-07 01:36:47 +02:00
|
|
|
public function classHasCorrectCasing(string $fq_class_name): bool
|
2018-02-04 00:52:35 +01:00
|
|
|
{
|
|
|
|
return $this->classlikes->classHasCorrectCasing($fq_class_name);
|
2018-01-31 22:08:52 +01:00
|
|
|
}
|
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public function interfaceHasCorrectCasing(string $fq_interface_name): bool
|
2018-01-31 22:08:52 +01:00
|
|
|
{
|
2018-02-04 00:52:35 +01:00
|
|
|
return $this->classlikes->interfaceHasCorrectCasing($fq_interface_name);
|
|
|
|
}
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public function traitHasCorrectCase(string $fq_trait_name): bool
|
2018-02-04 00:52:35 +01:00
|
|
|
{
|
|
|
|
return $this->classlikes->traitHasCorrectCase($fq_trait_name);
|
|
|
|
}
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2020-03-14 19:51:43 +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
|
2020-10-17 18:36:44 +02:00
|
|
|
*
|
|
|
|
* @return Storage\FunctionStorage|Storage\MethodStorage
|
2020-03-14 19:51:43 +01:00
|
|
|
*/
|
|
|
|
public function getFunctionLikeStorage(
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
|
|
|
string $function_id
|
|
|
|
): FunctionLikeStorage {
|
|
|
|
$doesMethodExist =
|
|
|
|
\Psalm\Internal\MethodIdentifier::isValidMethodIdReference($function_id)
|
|
|
|
&& $this->methodExists($function_id);
|
2020-03-16 16:45:49 +01:00
|
|
|
|
2020-03-14 19:51:43 +01:00
|
|
|
if ($doesMethodExist) {
|
2020-03-16 16:45:49 +01:00
|
|
|
$method_id = \Psalm\Internal\MethodIdentifier::wrap($function_id);
|
|
|
|
|
|
|
|
$declaring_method_id = $this->methods->getDeclaringMethodId($method_id);
|
|
|
|
|
|
|
|
if (!$declaring_method_id) {
|
|
|
|
throw new \UnexpectedValueException('Declaring method for ' . $method_id . ' cannot be found');
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->methods->getStorage($declaring_method_id);
|
2020-03-14 19:51:43 +01:00
|
|
|
}
|
|
|
|
|
2020-04-21 03:36:44 +02:00
|
|
|
return $this->functions->getStorage($statements_analyzer, strtolower($function_id));
|
2020-03-14 19:51:43 +01:00
|
|
|
}
|
|
|
|
|
2018-02-04 00:52:35 +01:00
|
|
|
/**
|
|
|
|
* Whether or not a given method exists
|
|
|
|
*
|
2020-02-15 02:54:26 +01:00
|
|
|
* @param string|\Psalm\Internal\MethodIdentifier $method_id
|
2020-03-26 17:35:27 +01:00
|
|
|
* @param string|\Psalm\Internal\MethodIdentifier|null $calling_method_id
|
2018-02-04 00:52:35 +01:00
|
|
|
*
|
2020-03-26 17:35:27 +01:00
|
|
|
@return bool
|
2018-02-04 00:52:35 +01:00
|
|
|
*/
|
2019-04-17 06:17:28 +02:00
|
|
|
public function methodExists(
|
2020-02-15 02:54:26 +01:00
|
|
|
$method_id,
|
2020-03-26 17:35:27 +01:00
|
|
|
?CodeLocation $code_location = null,
|
|
|
|
$calling_method_id = null,
|
2021-06-10 20:18:15 +02:00
|
|
|
?string $file_path = null,
|
|
|
|
bool $is_used = true
|
2020-09-04 22:26:33 +02:00
|
|
|
): bool {
|
2019-04-17 06:17:28 +02:00
|
|
|
return $this->methods->methodExists(
|
2020-03-15 00:10:20 +01:00
|
|
|
Internal\MethodIdentifier::wrap($method_id),
|
2020-03-26 17:35:27 +01:00
|
|
|
is_string($calling_method_id) ? strtolower($calling_method_id) : strtolower((string) $calling_method_id),
|
2019-04-17 06:17:28 +02:00
|
|
|
$code_location,
|
|
|
|
null,
|
2021-06-10 20:18:15 +02:00
|
|
|
$file_path,
|
|
|
|
true,
|
|
|
|
$is_used
|
2019-04-17 06:17:28 +02:00
|
|
|
);
|
2018-01-31 22:08:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-02-15 02:54:26 +01:00
|
|
|
* @param string|\Psalm\Internal\MethodIdentifier $method_id
|
2018-02-04 00:52:35 +01:00
|
|
|
*
|
2018-05-12 00:35:02 +02:00
|
|
|
* @return array<int, \Psalm\Storage\FunctionLikeParameter>
|
2018-01-31 22:08:52 +01:00
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function getMethodParams($method_id): array
|
2018-01-31 22:08:52 +01:00
|
|
|
{
|
2020-03-15 00:10:20 +01:00
|
|
|
return $this->methods->getMethodParams(Internal\MethodIdentifier::wrap($method_id));
|
2018-01-31 22:08:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-02-15 02:54:26 +01:00
|
|
|
* @param string|\Psalm\Internal\MethodIdentifier $method_id
|
2018-02-04 00:52:35 +01:00
|
|
|
*
|
2018-01-31 22:08:52 +01:00
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function isVariadic($method_id): bool
|
2018-01-31 22:08:52 +01:00
|
|
|
{
|
2020-03-15 00:10:20 +01:00
|
|
|
return $this->methods->isVariadic(Internal\MethodIdentifier::wrap($method_id));
|
2018-02-04 00:52:35 +01:00
|
|
|
}
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2018-02-04 00:52:35 +01:00
|
|
|
/**
|
2020-02-15 02:54:26 +01:00
|
|
|
* @param string|\Psalm\Internal\MethodIdentifier $method_id
|
2020-10-28 17:45:26 +01:00
|
|
|
* @param list<PhpParser\Node\Arg> $call_args
|
2018-02-04 00:52:35 +01:00
|
|
|
*
|
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function getMethodReturnType($method_id, ?string &$self_class, array $call_args = []): ?Type\Union
|
2018-02-04 00:52:35 +01:00
|
|
|
{
|
2020-02-15 02:54:26 +01:00
|
|
|
return $this->methods->getMethodReturnType(
|
2020-03-15 00:10:20 +01:00
|
|
|
Internal\MethodIdentifier::wrap($method_id),
|
2020-02-15 02:54:26 +01:00
|
|
|
$self_class,
|
|
|
|
null,
|
|
|
|
$call_args
|
|
|
|
);
|
2018-02-04 00:52:35 +01:00
|
|
|
}
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2018-02-04 00:52:35 +01:00
|
|
|
/**
|
2020-02-15 02:54:26 +01:00
|
|
|
* @param string|\Psalm\Internal\MethodIdentifier $method_id
|
2018-02-04 00:52:35 +01:00
|
|
|
*
|
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function getMethodReturnsByRef($method_id): bool
|
2018-02-04 00:52:35 +01:00
|
|
|
{
|
2020-03-15 00:10:20 +01:00
|
|
|
return $this->methods->getMethodReturnsByRef(Internal\MethodIdentifier::wrap($method_id));
|
2018-02-04 00:52:35 +01:00
|
|
|
}
|
2018-01-31 22:08:52 +01:00
|
|
|
|
2018-02-04 00:52:35 +01:00
|
|
|
/**
|
2020-02-15 02:54:26 +01:00
|
|
|
* @param string|\Psalm\Internal\MethodIdentifier $method_id
|
2018-02-04 00:52:35 +01:00
|
|
|
* @param CodeLocation|null $defined_location
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
public function getMethodReturnTypeLocation(
|
|
|
|
$method_id,
|
|
|
|
CodeLocation &$defined_location = null
|
2020-09-04 22:26:33 +02:00
|
|
|
): ?CodeLocation {
|
2020-02-15 02:54:26 +01:00
|
|
|
return $this->methods->getMethodReturnTypeLocation(
|
2020-03-15 00:10:20 +01:00
|
|
|
Internal\MethodIdentifier::wrap($method_id),
|
2020-02-15 02:54:26 +01:00
|
|
|
$defined_location
|
|
|
|
);
|
2018-01-31 22:08:52 +01:00
|
|
|
}
|
|
|
|
|
2018-01-31 23:09:09 +01:00
|
|
|
/**
|
2020-02-15 02:54:26 +01:00
|
|
|
* @param string|\Psalm\Internal\MethodIdentifier $method_id
|
2018-02-04 00:52:35 +01:00
|
|
|
*
|
2018-01-31 23:09:09 +01:00
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function getDeclaringMethodId($method_id): ?string
|
2018-01-31 23:09:09 +01:00
|
|
|
{
|
2020-11-24 06:18:24 +01:00
|
|
|
$new_method_id = $this->methods->getDeclaringMethodId(Internal\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)
|
|
|
|
*
|
2020-02-15 02:54:26 +01:00
|
|
|
* @param string|\Psalm\Internal\MethodIdentifier $method_id
|
2018-02-04 00:52:35 +01:00
|
|
|
*
|
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function getAppearingMethodId($method_id): ?string
|
2018-02-04 00:52:35 +01:00
|
|
|
{
|
2020-11-24 06:18:24 +01:00
|
|
|
$new_method_id = $this->methods->getAppearingMethodId(Internal\MethodIdentifier::wrap($method_id));
|
|
|
|
|
|
|
|
return $new_method_id ? (string) $new_method_id : null;
|
2018-01-31 23:09:09 +01:00
|
|
|
}
|
|
|
|
|
2018-01-31 22:08:52 +01:00
|
|
|
/**
|
2020-02-15 02:54:26 +01:00
|
|
|
* @param string|\Psalm\Internal\MethodIdentifier $method_id
|
2018-02-04 00:52:35 +01:00
|
|
|
*
|
2020-10-17 18:36:44 +02:00
|
|
|
* @return array<string, Internal\MethodIdentifier>
|
2018-01-31 22:08:52 +01:00
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function getOverriddenMethodIds($method_id): array
|
2018-01-31 22:08:52 +01:00
|
|
|
{
|
2020-03-15 00:10:20 +01:00
|
|
|
return $this->methods->getOverriddenMethodIds(Internal\MethodIdentifier::wrap($method_id));
|
2018-01-31 22:08:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-02-15 02:54:26 +01:00
|
|
|
* @param string|\Psalm\Internal\MethodIdentifier $method_id
|
2018-02-04 00:52:35 +01:00
|
|
|
*
|
2018-01-31 22:08:52 +01:00
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function getCasedMethodId($method_id): string
|
2018-01-31 22:08:52 +01:00
|
|
|
{
|
2020-03-15 00:10:20 +01:00
|
|
|
return $this->methods->getCasedMethodId(Internal\MethodIdentifier::wrap($method_id));
|
2018-01-31 22:08:52 +01:00
|
|
|
}
|
2018-09-28 22:18:45 +02:00
|
|
|
|
2020-10-12 21:02:52 +02:00
|
|
|
public function invalidateInformationForFile(string $file_path): void
|
2018-09-28 22:18:45 +02:00
|
|
|
{
|
|
|
|
$this->scanner->removeFile($file_path);
|
|
|
|
|
|
|
|
try {
|
|
|
|
$file_storage = $this->file_storage_provider->get($file_path);
|
|
|
|
} 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
|
|
|
|
2021-01-27 03:34:46 +01:00
|
|
|
public function getFunctionStorageForSymbol(string $file_path, string $symbol): ?FunctionLikeStorage
|
|
|
|
{
|
|
|
|
if (strpos($symbol, '::')) {
|
|
|
|
$symbol = substr($symbol, 0, -2);
|
|
|
|
/** @psalm-suppress ArgumentTypeCoercion */
|
|
|
|
$method_id = new \Psalm\Internal\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);
|
2021-01-27 03:34:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$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];
|
2021-01-27 03:34:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!$function_id) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2021-09-25 02:34:21 +02:00
|
|
|
return $this->functions->getStorage(null, $function_id);
|
2021-01-27 03:34:46 +01:00
|
|
|
}
|
|
|
|
|
2021-02-24 16:14:04 +01:00
|
|
|
/**
|
|
|
|
* @param string $file_path
|
|
|
|
* @param string $symbol
|
|
|
|
* @return array{ type: string, description?: string|null}|null
|
|
|
|
*/
|
|
|
|
public function getSymbolInformation(string $file_path, string $symbol): ?array
|
2018-10-26 22:17:15 +02:00
|
|
|
{
|
2019-07-02 00:48:33 +02:00
|
|
|
if (\is_numeric($symbol[0])) {
|
2021-02-24 16:14:04 +01:00
|
|
|
return ['type' => \preg_replace('/^[^:]*:/', '', $symbol)];
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
if (strpos($symbol, '::')) {
|
|
|
|
if (strpos($symbol, '()')) {
|
|
|
|
$symbol = substr($symbol, 0, -2);
|
|
|
|
|
2020-10-15 00:51:15 +02:00
|
|
|
/** @psalm-suppress ArgumentTypeCoercion */
|
2020-02-15 02:54:26 +01:00
|
|
|
$method_id = new \Psalm\Internal\MethodIdentifier(...explode('::', $symbol));
|
|
|
|
|
|
|
|
$declaring_method_id = $this->methods->getDeclaringMethodId($method_id);
|
2018-10-26 22:17:15 +02:00
|
|
|
|
|
|
|
if (!$declaring_method_id) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$storage = $this->methods->getStorage($declaring_method_id);
|
|
|
|
|
2021-02-24 16:14:04 +01:00
|
|
|
return [
|
|
|
|
'type' => '<?php ' . $storage->getSignature(true),
|
|
|
|
'description' => $storage->description,
|
|
|
|
];
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
|
|
|
|
2020-09-02 06:17:41 +02:00
|
|
|
[, $symbol_name] = explode('::', $symbol);
|
2018-10-26 22:17:15 +02:00
|
|
|
|
|
|
|
if (strpos($symbol, '$') !== false) {
|
|
|
|
$storage = $this->properties->getStorage($symbol);
|
|
|
|
|
2021-02-24 16:14:04 +01:00
|
|
|
return [
|
|
|
|
'type' => '<?php ' . $storage->getInfo() . ' ' . $symbol_name,
|
|
|
|
'description' => $storage->description,
|
|
|
|
];
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
|
|
|
|
2020-09-02 06:17:41 +02:00
|
|
|
[$fq_classlike_name, $const_name] = explode('::', $symbol);
|
2018-10-26 22:17:15 +02:00
|
|
|
|
|
|
|
$class_constants = $this->classlikes->getConstantsForClass(
|
|
|
|
$fq_classlike_name,
|
|
|
|
\ReflectionProperty::IS_PRIVATE
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!isset($class_constants[$const_name])) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2021-02-24 16:14:04 +01:00
|
|
|
return [
|
|
|
|
'type' => '<?php ' . $const_name,
|
|
|
|
'description' => $class_constants[$const_name]->description,
|
|
|
|
];
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (strpos($symbol, '()')) {
|
|
|
|
$function_id = strtolower(substr($symbol, 0, -2));
|
2020-06-08 17:20:54 +02:00
|
|
|
$file_storage = $this->file_storage_provider->get($file_path);
|
2018-10-26 22:17:15 +02:00
|
|
|
|
|
|
|
if (isset($file_storage->functions[$function_id])) {
|
|
|
|
$function_storage = $file_storage->functions[$function_id];
|
|
|
|
|
2021-02-24 16:14:04 +01:00
|
|
|
return [
|
|
|
|
'type' => '<?php ' . $function_storage->getSignature(true),
|
|
|
|
'description' => $function_storage->description,
|
|
|
|
];
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
2020-06-18 06:16:19 +02:00
|
|
|
|
2020-06-08 17:20:54 +02:00
|
|
|
if (!$function_id) {
|
|
|
|
return null;
|
|
|
|
}
|
2018-10-26 22:17:15 +02:00
|
|
|
|
2020-06-08 17:20:54 +02:00
|
|
|
$function = $this->functions->getStorage(null, $function_id);
|
2021-02-24 16:14:04 +01:00
|
|
|
return [
|
|
|
|
'type' => '<?php ' . $function->getSignature(true),
|
|
|
|
'description' => $function->description,
|
|
|
|
];
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
|
|
|
|
2021-01-29 00:58:02 +01:00
|
|
|
if (strpos($symbol, '$') === 0) {
|
|
|
|
$type = VariableFetchAnalyzer::getGlobalType($symbol);
|
|
|
|
if (!$type->isMixed()) {
|
2021-02-24 16:14:04 +01:00
|
|
|
return ['type' => '<?php ' . $type];
|
2021-01-29 00:58:02 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-28 17:18:28 +01:00
|
|
|
try {
|
|
|
|
$storage = $this->classlike_storage_provider->get($symbol);
|
2021-02-24 16:14:04 +01:00
|
|
|
return [
|
|
|
|
'type' => '<?php ' . ($storage->abstract ? 'abstract ' : '') . 'class ' . $storage->name,
|
|
|
|
'description' => $storage->description,
|
|
|
|
];
|
2021-01-28 17:18:28 +01:00
|
|
|
} 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];
|
2021-02-24 16:14:04 +01:00
|
|
|
return ['type' => '<?php const ' . $symbol . ' ' . $type];
|
2021-01-28 17:18:28 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$file_storage = $this->file_storage_provider->get($file_path);
|
|
|
|
if (isset($file_storage->constants[$symbol])) {
|
2021-02-24 16:14:04 +01:00
|
|
|
return ['type' => '<?php const ' . $symbol . ' ' . $file_storage->constants[$symbol]];
|
2021-01-28 17:18:28 +01:00
|
|
|
}
|
|
|
|
$constant = ConstFetchAnalyzer::getGlobalConstType($this, $symbol, $symbol);
|
2018-10-26 22:17:15 +02:00
|
|
|
|
2021-01-28 17:18:28 +01:00
|
|
|
if ($constant) {
|
2021-02-24 16:14:04 +01:00
|
|
|
return ['type' => '<?php const ' . $symbol . ' ' . $constant];
|
2021-01-28 17:18:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
2018-10-26 22:17:15 +02:00
|
|
|
} catch (\Exception $e) {
|
|
|
|
error_log($e->getMessage());
|
2019-07-05 22:24:00 +02:00
|
|
|
|
2018-10-26 22:17:15 +02:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-12 21:02:52 +02:00
|
|
|
public function getSymbolLocation(string $file_path, string $symbol): ?CodeLocation
|
2018-10-26 22:17:15 +02:00
|
|
|
{
|
2019-07-02 00:48:33 +02:00
|
|
|
if (\is_numeric($symbol[0])) {
|
|
|
|
$symbol = \preg_replace('/:.*/', '', $symbol);
|
|
|
|
$symbol_parts = explode('-', $symbol);
|
|
|
|
|
|
|
|
$file_contents = $this->getFileContents($file_path);
|
|
|
|
|
2019-08-18 20:27:50 +02:00
|
|
|
return new CodeLocation\Raw(
|
|
|
|
$file_contents,
|
|
|
|
$file_path,
|
|
|
|
$this->config->shortenFileName($file_path),
|
|
|
|
(int) $symbol_parts[0],
|
|
|
|
(int) $symbol_parts[1]
|
|
|
|
);
|
2019-07-02 00:48:33 +02:00
|
|
|
}
|
|
|
|
|
2018-10-26 22:17:15 +02:00
|
|
|
try {
|
|
|
|
if (strpos($symbol, '::')) {
|
|
|
|
if (strpos($symbol, '()')) {
|
|
|
|
$symbol = substr($symbol, 0, -2);
|
|
|
|
|
2020-10-15 00:51:15 +02:00
|
|
|
/** @psalm-suppress ArgumentTypeCoercion */
|
2020-02-15 02:54:26 +01:00
|
|
|
$method_id = new \Psalm\Internal\MethodIdentifier(...explode('::', $symbol));
|
|
|
|
|
|
|
|
$declaring_method_id = $this->methods->getDeclaringMethodId($method_id);
|
2018-10-26 22:17:15 +02:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2020-09-02 06:17:41 +02:00
|
|
|
[$fq_classlike_name, $const_name] = explode('::', $symbol);
|
2018-10-26 22:17:15 +02:00
|
|
|
|
|
|
|
$class_constants = $this->classlikes->getConstantsForClass(
|
|
|
|
$fq_classlike_name,
|
|
|
|
\ReflectionProperty::IS_PRIVATE
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!isset($class_constants[$const_name])) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2020-10-05 15:50:32 +02:00
|
|
|
return $class_constants[$const_name]->location;
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2020-06-18 06:16:19 +02:00
|
|
|
|
2020-06-08 17:20:54 +02:00
|
|
|
if (!$function_id) {
|
|
|
|
return null;
|
|
|
|
}
|
2018-10-26 22:17:15 +02:00
|
|
|
|
2021-09-25 02:34:21 +02:00
|
|
|
return $this->functions->getStorage(null, $function_id)->location;
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
|
|
|
|
2021-09-25 02:34:21 +02:00
|
|
|
return $this->classlike_storage_provider->get($symbol)->location;
|
2018-10-26 22:17:15 +02:00
|
|
|
} catch (\UnexpectedValueException $e) {
|
|
|
|
error_log($e->getMessage());
|
2019-07-05 22:24:00 +02:00
|
|
|
|
2018-10-26 22:17:15 +02:00
|
|
|
return null;
|
2019-05-15 05:49:15 +02:00
|
|
|
} catch (\InvalidArgumentException $e) {
|
|
|
|
return null;
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array{0: string, 1: Range}|null
|
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function getReferenceAtPosition(string $file_path, Position $position): ?array
|
2018-10-26 22:17:15 +02:00
|
|
|
{
|
2018-11-20 22:32:40 +01:00
|
|
|
$is_open = $this->file_provider->isOpen($file_path);
|
|
|
|
|
|
|
|
if (!$is_open) {
|
|
|
|
throw new \Psalm\Exception\UnanalyzedFileException($file_path . ' is not open');
|
|
|
|
}
|
|
|
|
|
2018-10-26 22:17:15 +02:00
|
|
|
$file_contents = $this->getFileContents($file_path);
|
|
|
|
|
|
|
|
$offset = $position->toOffset($file_contents);
|
|
|
|
|
2020-09-02 06:17:41 +02:00
|
|
|
[$reference_map, $type_map] = $this->analyzer->getMapsForFile($file_path);
|
2018-10-26 22:17:15 +02:00
|
|
|
|
|
|
|
$reference = null;
|
|
|
|
|
|
|
|
if (!$reference_map && !$type_map) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2020-07-09 22:24:51 +02:00
|
|
|
$reference_start_pos = null;
|
|
|
|
$reference_end_pos = null;
|
2018-10-26 22:17:15 +02:00
|
|
|
|
|
|
|
ksort($reference_map);
|
|
|
|
|
2020-09-02 06:17:41 +02:00
|
|
|
foreach ($reference_map as $start_pos => [$end_pos, $possible_reference]) {
|
2018-10-26 22:17:15 +02:00
|
|
|
if ($offset < $start_pos) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-02-24 07:33:25 +01:00
|
|
|
if ($offset > $end_pos) {
|
2018-10-26 22:17:15 +02:00
|
|
|
continue;
|
|
|
|
}
|
2020-07-09 22:24:51 +02:00
|
|
|
$reference_start_pos = $start_pos;
|
|
|
|
$reference_end_pos = $end_pos;
|
2018-10-26 22:17:15 +02:00
|
|
|
$reference = $possible_reference;
|
|
|
|
}
|
|
|
|
|
2020-07-09 22:24:51 +02:00
|
|
|
if ($reference === null || $reference_start_pos === null || $reference_end_pos === null) {
|
2019-07-02 00:48:33 +02:00
|
|
|
return null;
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$range = new Range(
|
2020-07-09 22:24:51 +02:00
|
|
|
self::getPositionFromOffset($reference_start_pos, $file_contents),
|
|
|
|
self::getPositionFromOffset($reference_end_pos, $file_contents)
|
2018-10-26 22:17:15 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
return [$reference, $range];
|
|
|
|
}
|
|
|
|
|
2019-07-01 21:54:33 +02:00
|
|
|
/**
|
2020-05-15 16:18:05 +02:00
|
|
|
* @return array{0: non-empty-string, 1: int, 2: Range}|null
|
2019-07-01 21:54:33 +02:00
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function getFunctionArgumentAtPosition(string $file_path, Position $position): ?array
|
2019-07-01 21:54:33 +02:00
|
|
|
{
|
|
|
|
$is_open = $this->file_provider->isOpen($file_path);
|
|
|
|
|
|
|
|
if (!$is_open) {
|
|
|
|
throw new \Psalm\Exception\UnanalyzedFileException($file_path . ' is not open');
|
|
|
|
}
|
|
|
|
|
|
|
|
$file_contents = $this->getFileContents($file_path);
|
|
|
|
|
|
|
|
$offset = $position->toOffset($file_contents);
|
|
|
|
|
2020-09-02 06:17:41 +02:00
|
|
|
[, , $argument_map] = $this->analyzer->getMapsForFile($file_path);
|
2019-07-01 21:54:33 +02:00
|
|
|
|
|
|
|
$reference = null;
|
|
|
|
$argument_number = null;
|
|
|
|
|
|
|
|
if (!$argument_map) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$start_pos = null;
|
|
|
|
$end_pos = null;
|
|
|
|
|
|
|
|
ksort($argument_map);
|
|
|
|
|
2020-09-02 06:17:41 +02:00
|
|
|
foreach ($argument_map as $start_pos => [$end_pos, $possible_reference, $possible_argument_number]) {
|
2019-07-01 21:54:33 +02:00
|
|
|
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),
|
|
|
|
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
|
|
|
*/
|
2021-02-24 16:14:04 +01:00
|
|
|
public function getSignatureInformation(
|
|
|
|
string $function_symbol,
|
|
|
|
string $file_path = null
|
|
|
|
): ?\LanguageServerProtocol\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 */
|
2020-02-15 02:54:26 +01:00
|
|
|
$method_id = new \Psalm\Internal\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;
|
2021-02-24 16:14:04 +01:00
|
|
|
$signature_label = $method_storage->cased_name;
|
|
|
|
$signature_documentation = $method_storage->description;
|
2019-07-19 05:08:54 +02:00
|
|
|
} else {
|
|
|
|
try {
|
2021-02-24 16:14:04 +01:00
|
|
|
if ($file_path) {
|
|
|
|
$function_storage = $this->functions->getStorage(
|
|
|
|
null,
|
|
|
|
strtolower($function_symbol),
|
|
|
|
dirname($file_path),
|
|
|
|
$file_path
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$function_storage = $this->functions->getStorage(null, strtolower($function_symbol));
|
|
|
|
}
|
2019-07-19 05:30:44 +02:00
|
|
|
$params = $function_storage->params;
|
2021-02-24 16:14:04 +01:00
|
|
|
$signature_label = $function_storage->cased_name;
|
|
|
|
$signature_documentation = $function_storage->description;
|
2019-07-19 05:08:54 +02:00
|
|
|
} catch (\Exception $exception) {
|
2020-05-25 19:10:06 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2021-02-24 16:14:04 +01: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-02-24 16:14:04 +01:00
|
|
|
$parameters[] = new \LanguageServerProtocol\ParameterInformation(
|
|
|
|
[
|
|
|
|
strlen($signature_label),
|
|
|
|
strlen($signature_label) + strlen($parameter_label),
|
|
|
|
],
|
|
|
|
$param->description ?? null
|
|
|
|
);
|
2019-07-19 05:08:54 +02:00
|
|
|
|
|
|
|
$signature_label .= $parameter_label;
|
|
|
|
|
|
|
|
if ($i < (count($params) - 1)) {
|
|
|
|
$signature_label .= ', ';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$signature_label .= ')';
|
|
|
|
|
|
|
|
return new \LanguageServerProtocol\SignatureInformation(
|
|
|
|
$signature_label,
|
2021-02-24 16:14:04 +01:00
|
|
|
$parameters,
|
|
|
|
$signature_documentation
|
2019-07-19 05:08:54 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-10-26 22:17:15 +02:00
|
|
|
/**
|
2021-01-27 03:34:46 +01:00
|
|
|
* @return array{0: string, 1: '->'|'::'|'['|'symbol', 2: int}|null
|
2018-10-26 22:17:15 +02:00
|
|
|
*/
|
2020-09-04 22:26:33 +02:00
|
|
|
public function getCompletionDataAtPosition(string $file_path, Position $position): ?array
|
2018-10-26 22:17:15 +02:00
|
|
|
{
|
2018-11-20 22:32:40 +01:00
|
|
|
$is_open = $this->file_provider->isOpen($file_path);
|
|
|
|
|
|
|
|
if (!$is_open) {
|
|
|
|
throw new \Psalm\Exception\UnanalyzedFileException($file_path . ' is not open');
|
|
|
|
}
|
|
|
|
|
2018-10-26 22:17:15 +02:00
|
|
|
$file_contents = $this->getFileContents($file_path);
|
|
|
|
|
|
|
|
$offset = $position->toOffset($file_contents);
|
|
|
|
|
2020-09-02 06:17:41 +02:00
|
|
|
[$reference_map, $type_map] = $this->analyzer->getMapsForFile($file_path);
|
2018-10-26 22:17:15 +02:00
|
|
|
|
2019-06-21 23:10:35 +02:00
|
|
|
if (!$reference_map && !$type_map) {
|
2018-10-26 22:17:15 +02:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
krsort($type_map);
|
|
|
|
|
2020-09-02 06:17:41 +02:00
|
|
|
foreach ($type_map as $start_pos => [$end_pos_excluding_whitespace, $possible_type]) {
|
2018-10-26 22:17:15 +02:00
|
|
|
if ($offset < $start_pos) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-06-12 12:30:48 +02:00
|
|
|
$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;
|
|
|
|
|
2021-01-27 03:34:46 +01:00
|
|
|
if ($offset - $end_pos === 1) {
|
|
|
|
$candidate_gap = substr($file_contents, $end_pos, 1);
|
|
|
|
|
|
|
|
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) {
|
2019-05-17 18:38:29 +02:00
|
|
|
$candidate_gap = substr($file_contents, $end_pos, 2);
|
2018-10-26 22:17:15 +02:00
|
|
|
|
2019-05-17 18:38:29 +02:00
|
|
|
if ($candidate_gap === '->' || $candidate_gap === '::') {
|
|
|
|
$gap = $candidate_gap;
|
|
|
|
$recent_type = $possible_type;
|
|
|
|
|
2019-06-21 23:10:35 +02:00
|
|
|
if ($recent_type === 'mixed') {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2019-06-30 03:32:26 +02:00
|
|
|
return [$recent_type, $gap, $offset];
|
2019-05-17 18:38:29 +02:00
|
|
|
}
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-02 06:17:41 +02:00
|
|
|
foreach ($reference_map as $start_pos => [$end_pos, $possible_reference]) {
|
2021-01-24 19:29:21 +01:00
|
|
|
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] !== '*') {
|
2019-06-21 23:10:35 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($offset - $end_pos === 0) {
|
|
|
|
$recent_type = $possible_reference;
|
|
|
|
|
2019-06-30 03:32:26 +02:00
|
|
|
return [$recent_type, 'symbol', $offset];
|
2019-06-21 23:10:35 +02:00
|
|
|
}
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
|
|
|
|
2019-06-21 23:10:35 +02:00
|
|
|
return null;
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
|
|
|
|
2021-01-27 03:34:46 +01:00
|
|
|
public function getTypeContextAtPosition(string $file_path, Position $position): ?Type\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) {
|
|
|
|
return null;
|
|
|
|
}
|
2021-09-25 02:34:21 +02:00
|
|
|
|
|
|
|
return $function_storage->params[$argument_num]->type;
|
2021-01-27 03:34:46 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2019-05-17 18:11:21 +02:00
|
|
|
/**
|
2020-10-17 18:36:44 +02:00
|
|
|
* @return list<\LanguageServerProtocol\CompletionItem>
|
2019-05-17 18:11:21 +02:00
|
|
|
*/
|
|
|
|
public function getCompletionItemsForClassishThing(string $type_string, string $gap) : array
|
|
|
|
{
|
2021-02-27 17:36:19 +01:00
|
|
|
$completion_items = [];
|
2019-05-17 18:11:21 +02:00
|
|
|
|
|
|
|
$type = Type::parseString($type_string);
|
|
|
|
|
2020-01-04 18:20:26 +01:00
|
|
|
foreach ($type->getAtomicTypes() as $atomic_type) {
|
2019-05-17 18:11:21 +02:00
|
|
|
if ($atomic_type instanceof Type\Atomic\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);
|
|
|
|
|
2021-02-27 17:36:19 +01:00
|
|
|
if ($method_storage->is_static || $gap === '->') {
|
|
|
|
$completion_item = new \LanguageServerProtocol\CompletionItem(
|
|
|
|
$method_storage->cased_name,
|
|
|
|
\LanguageServerProtocol\CompletionItemKind::METHOD,
|
|
|
|
(string)$method_storage,
|
|
|
|
$method_storage->description,
|
|
|
|
(string)$method_storage->visibility,
|
|
|
|
$method_storage->cased_name,
|
|
|
|
$method_storage->cased_name . (count($method_storage->params) !== 0 ? '($0)' : '()'),
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
new Command('Trigger parameter hints', 'editor.action.triggerParameterHints'),
|
|
|
|
null,
|
|
|
|
2
|
|
|
|
);
|
|
|
|
|
|
|
|
$completion_item->insertTextFormat = \LanguageServerProtocol\InsertTextFormat::SNIPPET;
|
|
|
|
|
|
|
|
$completion_items[] = $completion_item;
|
2019-06-13 15:41:42 +02:00
|
|
|
}
|
2019-05-17 18:11:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($class_storage->declaring_property_ids as $property_name => $declaring_class) {
|
|
|
|
$property_storage = $this->properties->getStorage(
|
|
|
|
$declaring_class . '::$' . $property_name
|
|
|
|
);
|
|
|
|
|
2021-02-27 17:36:19 +01:00
|
|
|
if ($property_storage->is_static || $gap === '->') {
|
|
|
|
$completion_items[] = new \LanguageServerProtocol\CompletionItem(
|
|
|
|
'$' . $property_name,
|
|
|
|
\LanguageServerProtocol\CompletionItemKind::PROPERTY,
|
|
|
|
$property_storage->getInfo(),
|
|
|
|
$property_storage->description,
|
|
|
|
(string)$property_storage->visibility,
|
|
|
|
$property_name,
|
|
|
|
($gap === '::' ? '$' : '') . $property_name
|
|
|
|
);
|
2019-06-13 15:41:42 +02:00
|
|
|
}
|
2019-05-17 18:11:21 +02:00
|
|
|
}
|
|
|
|
|
2021-02-24 16:14:04 +01:00
|
|
|
foreach ($class_storage->constants as $const_name => $const) {
|
2021-02-27 17:36:19 +01:00
|
|
|
$completion_items[] = new \LanguageServerProtocol\CompletionItem(
|
2019-06-21 23:10:35 +02:00
|
|
|
$const_name,
|
2019-05-17 18:11:21 +02:00
|
|
|
\LanguageServerProtocol\CompletionItemKind::VARIABLE,
|
2019-06-21 23:10:35 +02:00
|
|
|
'const ' . $const_name,
|
2021-02-24 16:14:04 +01:00
|
|
|
$const->description,
|
2019-05-17 18:11:21 +02:00
|
|
|
null,
|
2019-06-13 15:45:39 +02:00
|
|
|
$const_name,
|
2019-05-17 18:11:21 +02:00
|
|
|
$const_name
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} catch (\Exception $e) {
|
|
|
|
error_log($e->getMessage());
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2019-06-13 15:42:38 +02:00
|
|
|
}
|
2019-05-17 18:11:21 +02:00
|
|
|
|
|
|
|
return $completion_items;
|
|
|
|
}
|
|
|
|
|
2019-06-21 23:10:35 +02:00
|
|
|
/**
|
2020-10-17 18:36:44 +02:00
|
|
|
* @return list<\LanguageServerProtocol\CompletionItem>
|
2019-06-21 23:10:35 +02:00
|
|
|
*/
|
2019-06-30 03:32:26 +02:00
|
|
|
public function getCompletionItemsForPartialSymbol(
|
|
|
|
string $type_string,
|
|
|
|
int $offset,
|
|
|
|
string $file_path
|
|
|
|
) : array {
|
2021-02-15 05:25:13 +01:00
|
|
|
$fq_suggestion = false;
|
|
|
|
|
|
|
|
if (($type_string[1] ?? '') === '\\') {
|
|
|
|
$fq_suggestion = true;
|
|
|
|
}
|
|
|
|
|
2019-06-21 23:10:35 +02:00
|
|
|
$matching_classlike_names = $this->classlikes->getMatchingClassLikeNames($type_string);
|
|
|
|
|
|
|
|
$completion_items = [];
|
|
|
|
|
2019-06-30 03:32:26 +02:00
|
|
|
$file_storage = $this->file_storage_provider->get($file_path);
|
|
|
|
|
|
|
|
$aliases = null;
|
|
|
|
|
2020-02-15 02:54:26 +01:00
|
|
|
foreach ($file_storage->classlikes_in_file as $fq_class_name => $_) {
|
2019-06-30 03:32:26 +02:00
|
|
|
try {
|
|
|
|
$class_storage = $this->classlike_storage_provider->get($fq_class_name);
|
|
|
|
} 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;
|
|
|
|
}
|
|
|
|
}
|
2019-06-21 23:10:35 +02:00
|
|
|
|
2019-07-18 04:50:57 +02:00
|
|
|
foreach ($matching_classlike_names as $fq_class_name) {
|
2019-06-30 17:12:50 +02:00
|
|
|
$extra_edits = [];
|
|
|
|
|
|
|
|
$insertion_text = Type::getStringFromFQCLN(
|
2019-07-18 04:50:57 +02:00
|
|
|
$fq_class_name,
|
2019-06-30 17:12:50 +02:00
|
|
|
$aliases && $aliases->namespace ? $aliases->namespace : null,
|
|
|
|
$aliases ? $aliases->uses_flipped : [],
|
|
|
|
null
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($aliases
|
2021-02-15 05:25:13 +01:00
|
|
|
&& !$fq_suggestion
|
2019-06-30 17:12:50 +02:00
|
|
|
&& $aliases->namespace
|
2019-07-18 04:50:57 +02:00
|
|
|
&& $insertion_text === '\\' . $fq_class_name
|
2019-06-30 17:12:50 +02:00
|
|
|
&& $aliases->namespace_first_stmt_start
|
|
|
|
) {
|
|
|
|
$file_contents = $this->getFileContents($file_path);
|
|
|
|
|
2019-07-18 04:50:57 +02:00
|
|
|
$class_name = \preg_replace('/^.*\\\/', '', $fq_class_name);
|
2019-06-30 17:12:50 +02:00
|
|
|
|
|
|
|
if ($aliases->uses_end) {
|
|
|
|
$position = self::getPositionFromOffset($aliases->uses_end, $file_contents);
|
|
|
|
$extra_edits[] = new \LanguageServerProtocol\TextEdit(
|
|
|
|
new Range(
|
|
|
|
$position,
|
|
|
|
$position
|
|
|
|
),
|
2019-07-18 04:50:57 +02:00
|
|
|
"\n" . 'use ' . $fq_class_name . ';'
|
2019-06-30 17:12:50 +02:00
|
|
|
);
|
|
|
|
} else {
|
|
|
|
$position = self::getPositionFromOffset($aliases->namespace_first_stmt_start, $file_contents);
|
|
|
|
$extra_edits[] = new \LanguageServerProtocol\TextEdit(
|
|
|
|
new Range(
|
|
|
|
$position,
|
|
|
|
$position
|
|
|
|
),
|
2019-07-18 04:50:57 +02:00
|
|
|
'use ' . $fq_class_name . ';' . "\n" . "\n"
|
2019-06-30 17:12:50 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$insertion_text = $class_name;
|
|
|
|
}
|
|
|
|
|
2021-02-24 16:14:04 +01:00
|
|
|
try {
|
|
|
|
$class_storage = $this->classlike_storage_provider->get($fq_class_name);
|
|
|
|
$description = $class_storage->description;
|
|
|
|
} catch (\Exception $e) {
|
|
|
|
$description = null;
|
|
|
|
}
|
|
|
|
|
2019-06-21 23:10:35 +02:00
|
|
|
$completion_items[] = new \LanguageServerProtocol\CompletionItem(
|
2019-07-18 04:50:57 +02:00
|
|
|
$fq_class_name,
|
2019-06-21 23:10:35 +02:00
|
|
|
\LanguageServerProtocol\CompletionItemKind::CLASS_,
|
|
|
|
null,
|
2021-02-24 16:14:04 +01:00
|
|
|
$description,
|
2019-06-21 23:10:35 +02:00
|
|
|
null,
|
2019-07-18 04:50:57 +02:00
|
|
|
$fq_class_name,
|
2019-06-30 17:12:50 +02:00
|
|
|
$insertion_text,
|
|
|
|
null,
|
|
|
|
$extra_edits
|
2019-06-21 23:10:35 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-02-12 22:59:47 +01:00
|
|
|
$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;
|
|
|
|
}
|
|
|
|
$completion_items[] = new \LanguageServerProtocol\CompletionItem(
|
|
|
|
$function_name,
|
|
|
|
\LanguageServerProtocol\CompletionItemKind::FUNCTION,
|
|
|
|
$function->getSignature(false),
|
2021-02-24 16:14:04 +01:00
|
|
|
$function->description,
|
2021-02-12 22:59:47 +01:00
|
|
|
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,
|
2021-02-27 17:36:19 +01:00
|
|
|
new Command('Trigger parameter hints', 'editor.action.triggerParameterHints'),
|
2021-02-15 20:05:54 +01:00
|
|
|
null,
|
|
|
|
2
|
2021-02-12 22:59:47 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2019-06-21 23:10:35 +02:00
|
|
|
return $completion_items;
|
|
|
|
}
|
|
|
|
|
2021-01-27 03:34:46 +01:00
|
|
|
/**
|
|
|
|
* @return list<\LanguageServerProtocol\CompletionItem>
|
|
|
|
*/
|
|
|
|
public function getCompletionItemsForType(Type\Union $type): array
|
|
|
|
{
|
|
|
|
$completion_items = [];
|
|
|
|
foreach ($type->getAtomicTypes() as $atomic_type) {
|
|
|
|
if ($atomic_type instanceof Type\Atomic\TBool) {
|
|
|
|
$bools = (string) $atomic_type === 'bool' ? ['true', 'false'] : [(string) $atomic_type];
|
|
|
|
foreach ($bools as $property_name) {
|
|
|
|
$completion_items[] = new \LanguageServerProtocol\CompletionItem(
|
|
|
|
$property_name,
|
|
|
|
\LanguageServerProtocol\CompletionItemKind::VALUE,
|
|
|
|
'bool',
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
null,
|
2021-02-12 17:09:41 +01:00
|
|
|
$property_name
|
2021-01-27 03:34:46 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
} elseif ($atomic_type instanceof Type\Atomic\TLiteralString) {
|
|
|
|
$completion_items[] = new \LanguageServerProtocol\CompletionItem(
|
|
|
|
$atomic_type->value,
|
|
|
|
\LanguageServerProtocol\CompletionItemKind::VALUE,
|
|
|
|
$atomic_type->getId(),
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
null,
|
2021-02-12 17:09:41 +01:00
|
|
|
"'$atomic_type->value'"
|
2021-01-27 03:34:46 +01:00
|
|
|
);
|
|
|
|
} elseif ($atomic_type instanceof Type\Atomic\TLiteralInt) {
|
|
|
|
$completion_items[] = new \LanguageServerProtocol\CompletionItem(
|
|
|
|
(string) $atomic_type->value,
|
|
|
|
\LanguageServerProtocol\CompletionItemKind::VALUE,
|
|
|
|
$atomic_type->getId(),
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
null,
|
2021-02-12 17:09:41 +01:00
|
|
|
(string) $atomic_type->value
|
2021-01-27 03:34:46 +01:00
|
|
|
);
|
2021-05-03 23:54:09 +02:00
|
|
|
} elseif ($atomic_type instanceof Type\Atomic\TClassConstant) {
|
2021-01-27 03:34:46 +01:00
|
|
|
$const = $atomic_type->fq_classlike_name . '::' . $atomic_type->const_name;
|
|
|
|
$completion_items[] = new \LanguageServerProtocol\CompletionItem(
|
|
|
|
$const,
|
|
|
|
\LanguageServerProtocol\CompletionItemKind::VALUE,
|
|
|
|
$atomic_type->getId(),
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
null,
|
2021-02-12 17:09:41 +01:00
|
|
|
$const
|
2021-01-27 03:34:46 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $completion_items;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return list<\LanguageServerProtocol\CompletionItem>
|
|
|
|
*/
|
|
|
|
public function getCompletionItemsForArrayKeys(
|
|
|
|
string $type_string
|
|
|
|
) : array {
|
|
|
|
$completion_items = [];
|
|
|
|
$type = Type::parseString($type_string);
|
|
|
|
foreach ($type->getAtomicTypes() as $atomic_type) {
|
|
|
|
if ($atomic_type instanceof Type\Atomic\TKeyedArray) {
|
|
|
|
foreach ($atomic_type->properties as $property_name => $property) {
|
|
|
|
$completion_items[] = new \LanguageServerProtocol\CompletionItem(
|
|
|
|
(string) $property_name,
|
|
|
|
\LanguageServerProtocol\CompletionItemKind::PROPERTY,
|
|
|
|
(string) $property,
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
null,
|
2021-02-12 23:00:58 +01:00
|
|
|
"'$property_name'"
|
2021-01-27 03:34:46 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $completion_items;
|
|
|
|
}
|
|
|
|
|
2018-10-26 22:17:15 +02:00
|
|
|
private static function getPositionFromOffset(int $offset, string $file_contents) : Position
|
|
|
|
{
|
|
|
|
$file_contents = substr($file_contents, 0, $offset);
|
2019-06-30 17:12:50 +02:00
|
|
|
|
2021-08-03 01:49:09 +02:00
|
|
|
$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);
|
2019-06-30 17:12:50 +02:00
|
|
|
|
2018-10-26 22:17:15 +02:00
|
|
|
return new Position(
|
2018-11-21 19:53:39 +01:00
|
|
|
substr_count($file_contents, "\n"),
|
2019-06-30 17:12:50 +02:00
|
|
|
$offset - (int)$before_newline_count - 1
|
2018-10-26 22:17:15 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function addTemporaryFileChanges(string $file_path, string $new_content): void
|
2018-10-26 22:17:15 +02:00
|
|
|
{
|
2018-11-09 16:41:51 +01:00
|
|
|
$this->file_provider->addTemporaryFileChanges($file_path, $new_content);
|
|
|
|
}
|
2018-11-02 02:52:39 +01:00
|
|
|
|
2020-09-12 17:24:05 +02:00
|
|
|
public function removeTemporaryFileChanges(string $file_path): void
|
2018-10-26 22:17:15 +02:00
|
|
|
{
|
2018-11-20 21:51:47 +01:00
|
|
|
$this->file_provider->removeTemporaryFileChanges($file_path);
|
2018-10-26 22:17:15 +02:00
|
|
|
}
|
2019-02-20 12:54:01 +01:00
|
|
|
|
2019-02-18 21:54:23 +01:00
|
|
|
/**
|
2019-02-20 16:59:33 +01:00
|
|
|
* Checks if type is a subtype of other
|
|
|
|
*
|
|
|
|
* Given two types, checks if `$input_type` is a subtype of `$container_type`.
|
|
|
|
* If you consider `Type\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
|
|
|
*/
|
2019-02-18 21:40:47 +01:00
|
|
|
public function isTypeContainedByType(
|
|
|
|
Type\Union $input_type,
|
|
|
|
Type\Union $container_type
|
|
|
|
): bool {
|
2020-07-22 01:40:35 +02:00
|
|
|
return UnionTypeComparator::isContainedBy($this, $input_type, $container_type);
|
2019-02-18 21:40:47 +01:00
|
|
|
}
|
2019-02-20 12:54:01 +01:00
|
|
|
|
|
|
|
/**
|
2019-02-20 16:59:33 +01:00
|
|
|
* 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`.
|
|
|
|
* If you consider `Type\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.
|
2019-02-20 16:59:33 +01:00
|
|
|
*
|
|
|
|
* 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.
|
2019-02-20 12:54:01 +01:00
|
|
|
*/
|
|
|
|
public function canTypeBeContainedByType(
|
|
|
|
Type\Union $input_type,
|
|
|
|
Type\Union $container_type
|
|
|
|
): bool {
|
2020-07-22 01:40:35 +02:00
|
|
|
return UnionTypeComparator::canBeContainedBy($this, $input_type, $container_type);
|
2019-02-20 12:54:01 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-02-20 16:59:33 +01:00
|
|
|
* 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)]
|
|
|
|
* ```
|
|
|
|
*
|
2019-02-20 12:54:01 +01:00
|
|
|
* @return array{Type\Union,Type\Union}
|
|
|
|
*/
|
|
|
|
public function getKeyValueParamsForTraversableObject(Type\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(),
|
|
|
|
];
|
|
|
|
}
|
2020-04-05 23:25:47 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @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);
|
|
|
|
}
|
2020-06-18 06:16:19 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array<string> $taints
|
|
|
|
*
|
|
|
|
* @psalm-suppress PossiblyUnusedMethod
|
|
|
|
*/
|
|
|
|
public function addTaintSource(
|
|
|
|
Type\Union $expr_type,
|
|
|
|
string $taint_id,
|
|
|
|
array $taints = \Psalm\Type\TaintKindGroup::ALL_INPUT,
|
|
|
|
?CodeLocation $code_location = null
|
|
|
|
) : void {
|
2020-09-25 06:37:40 +02:00
|
|
|
if (!$this->taint_flow_graph) {
|
2020-06-18 06:16:19 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-10-13 22:49:03 +02:00
|
|
|
$source = new \Psalm\Internal\DataFlow\TaintSource(
|
2020-06-18 06:16:19 +02:00
|
|
|
$taint_id,
|
|
|
|
$taint_id,
|
|
|
|
$code_location,
|
|
|
|
null,
|
|
|
|
$taints
|
|
|
|
);
|
|
|
|
|
2020-09-25 06:37:40 +02:00
|
|
|
$this->taint_flow_graph->addSource($source);
|
2020-06-18 06:16:19 +02:00
|
|
|
|
|
|
|
$expr_type->parent_nodes = [
|
2020-09-25 06:37:40 +02:00
|
|
|
$source->id => $source,
|
2020-06-18 06:16:19 +02:00
|
|
|
];
|
|
|
|
}
|
2020-07-16 22:02:26 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array<string> $taints
|
|
|
|
*
|
|
|
|
* @psalm-suppress PossiblyUnusedMethod
|
|
|
|
*/
|
|
|
|
public function addTaintSink(
|
|
|
|
string $taint_id,
|
|
|
|
array $taints = \Psalm\Type\TaintKindGroup::ALL_INPUT,
|
|
|
|
?CodeLocation $code_location = null
|
|
|
|
) : void {
|
2020-09-25 06:37:40 +02:00
|
|
|
if (!$this->taint_flow_graph) {
|
2020-07-16 22:02:26 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-10-13 22:49:03 +02:00
|
|
|
$sink = new \Psalm\Internal\DataFlow\TaintSink(
|
2020-07-16 22:02:26 +02:00
|
|
|
$taint_id,
|
|
|
|
$taint_id,
|
|
|
|
$code_location,
|
|
|
|
null,
|
|
|
|
$taints
|
|
|
|
);
|
|
|
|
|
2020-09-25 06:37:40 +02:00
|
|
|
$this->taint_flow_graph->addSink($sink);
|
2020-07-16 22:02:26 +02:00
|
|
|
}
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|