> */ 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' */ public $find_unused_code; /** * @var FileProvider */ public $file_provider; /** * @var FileReferenceProvider */ public $file_reference_provider; /** * @var StatementsProvider */ public $statements_provider; private Progress $progress; /** * @var array */ private static $stubbed_constants = []; /** * Whether to register autoloaded information * * @var bool */ public $register_autoload_files = false; /** * 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; /** * @var Scanner */ public $scanner; /** * @var Analyzer */ public $analyzer; /** * @var Functions */ public $functions; /** * @var ClassLikes */ public $classlikes; /** * @var Methods */ public $methods; /** * @var Properties */ public $properties; /** * @var Populator */ public $populator; /** * @var ?TaintFlowGraph */ public $taint_flow_graph; /** * @var bool */ public $server_mode = false; /** * @var bool */ public $store_node_types = false; /** * 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 */ public $methods_to_move = []; /** * @var array */ public $methods_to_rename = []; /** * @var array */ public $properties_to_move = []; /** * @var array */ public $properties_to_rename = []; /** * @var array */ public $class_constants_to_move = []; /** * @var array */ public $class_constants_to_rename = []; /** * @var array */ public $classes_to_move = []; /** * @var array */ public $call_transforms = []; /** * @var array */ public $property_transforms = []; /** * @var array */ public $class_constant_transforms = []; /** * @var array */ 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'; /** * @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 = []; $reflection = new Reflection($providers->classlike_storage_provider, $this); $this->scanner = new Scanner( $this, $config, $providers->file_storage_provider, $providers->file_provider, $reflection, $providers->file_reference_provider, $progress, ); $this->loadAnalyzer(); $this->functions = new Functions($providers->file_storage_provider, $reflection); $this->classlikes = new ClassLikes( $this->config, $providers->classlike_storage_provider, $providers->file_reference_provider, $providers->statements_provider, $this->scanner, ); $this->properties = new Properties( $providers->classlike_storage_provider, $providers->file_reference_provider, $this->classlikes, ); $this->methods = new Methods( $providers->classlike_storage_provider, $providers->file_reference_provider, $this->classlikes, ); $this->populator = new Populator( $providers->classlike_storage_provider, $providers->file_storage_provider, $this->classlikes, $providers->file_reference_provider, $progress, ); $this->loadAnalyzer(); } private function loadAnalyzer(): void { $this->analyzer = new Analyzer( $this->config, $this->file_provider, $this->file_storage_provider, $this->progress, ); } /** * @param array $candidate_files */ public function reloadFiles(ProjectAnalyzer $project_analyzer, array $candidate_files, bool $force = false): void { $this->loadAnalyzer(); if ($force) { FileReferenceProvider::clearCache(); } $this->file_reference_provider->loadReferenceCache(false); FunctionLikeAnalyzer::clearCache(); 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; } } } $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) { 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); $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; $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; } /** * @param array $files_to_analyze */ public function addFilesToAnalyze(array $files_to_analyze): void { $this->scanner->addFilesToDeepScan($files_to_analyze); $this->analyzer->addFilesToAnalyze($files_to_analyze); } /** * Scans all files their related files */ public function scanFiles(int $threads = 1): void { $has_changes = $this->scanner->scanFiles($this->classlikes, $threads); if ($has_changes) { $this->populator->populateCodebase(); } } public function getFileContents(string $file_path): string { return $this->file_provider->getContents($file_path); } /** * @return list */ public function getStatementsForFile(string $file_path): array { return $this->statements_provider->getStatementsForFile( $file_path, $this->analysis_php_version_id, $this->progress, ); } public function createClassLikeStorage(string $fq_classlike_name): ClassLikeStorage { return $this->classlike_storage_provider->create($fq_classlike_name); } public function cacheClassLikeStorage(ClassLikeStorage $classlike_storage, string $file_path): void { 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); } public function exhumeClassLikeStorage(string $fq_classlike_name, string $file_path): void { $file_contents = $this->file_provider->getContents($file_path); $storage = $this->classlike_storage_provider->exhume( $fq_classlike_name, $file_path, $file_contents, ); if ($storage->is_trait) { $this->classlikes->addFullyQualifiedTraitName($storage->name, $file_path); } elseif ($storage->is_interface) { $this->classlikes->addFullyQualifiedInterfaceName($storage->name, $file_path); } else { $this->classlikes->addFullyQualifiedClassName($storage->name, $file_path); } } public static function getPsalmTypeFromReflection(?ReflectionType $type): Union { return Reflection::getPsalmTypeFromReflectionType($type); } public function createFileStorageForPath(string $file_path): FileStorage { return $this->file_storage_provider->create($file_path); } /** * @return array */ public function findReferencesToSymbol(string $symbol): array { if (!$this->collect_locations) { throw new UnexpectedValueException('Should not be checking references'); } if (strpos($symbol, '::$') !== false) { return $this->findReferencesToProperty($symbol); } if (strpos($symbol, '::') !== false) { return $this->findReferencesToMethod($symbol); } return $this->findReferencesToClassLike($symbol); } /** * @return array */ public function findReferencesToMethod(string $method_id): array { return $this->file_reference_provider->getClassMethodLocations(strtolower($method_id)); } /** * @return array */ public function findReferencesToProperty(string $property_id): array { /** @psalm-suppress PossiblyUndefinedIntArrayOffset */ [$fq_class_name, $property_name] = explode('::', $property_id); return $this->file_reference_provider->getClassPropertyLocations( strtolower($fq_class_name) . '::' . $property_name, ); } /** * @return CodeLocation[] * @psalm-return array */ public function findReferencesToClassLike(string $fq_class_name): array { $fq_class_name_lc = strtolower($fq_class_name); $locations = $this->file_reference_provider->getClassLocations($fq_class_name_lc); 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 { $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, ); } public function addGlobalConstantType(string $const_id, Union $type): void { self::$stubbed_constants[$const_id] = $type; } public function getStubbedConstantType(string $const_id): ?Union { return self::$stubbed_constants[$const_id] ?? null; } /** * @return array */ public function getAllStubbedConstants(): array { return self::$stubbed_constants; } public function fileExists(string $file_path): bool { return $this->file_provider->fileExists($file_path); } /** * 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, $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, $calling_method_id, ); } public function classExtendsOrImplements(string $fq_class_name, string $possible_parent): bool { return $this->classlikes->classExtends($fq_class_name, $possible_parent) || $this->classlikes->classImplements($fq_class_name, $possible_parent); } /** * 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, $calling_method_id, ); } /** * Determine whether or not a class extends a parent * * @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); } /** * Check whether a class implements an interface */ public function classImplements(string $fq_class_name, string $interface): bool { return $this->classlikes->classImplements($fq_class_name, $interface); } 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, $calling_method_id, ); } public function interfaceExtends(string $interface_name, string $possible_parent): bool { return $this->classlikes->interfaceExtends($interface_name, $possible_parent); } /** * @return array all interfaces extended by $interface_name */ public function getParentInterfaces(string $fq_interface_name): array { return $this->classlikes->getParentInterfaces( $this->classlikes->getUnAliasedName($fq_interface_name), ); } /** * Determine whether or not a class has the correct casing */ public function classHasCorrectCasing(string $fq_class_name): bool { return $this->classlikes->classHasCorrectCasing($fq_class_name); } public function interfaceHasCorrectCasing(string $fq_interface_name): bool { return $this->classlikes->interfaceHasCorrectCasing($fq_interface_name); } public function traitHasCorrectCasing(string $fq_trait_name): bool { return $this->classlikes->traitHasCorrectCasing($fq_trait_name); } /** * Given a function id, return the function like storage for * a method, closure, or function. * * @param non-empty-string $function_id * @return FunctionStorage|MethodStorage */ public function getFunctionLikeStorage( StatementsAnalyzer $statements_analyzer, string $function_id ): FunctionLikeStorage { $doesMethodExist = MethodIdentifier::isValidMethodIdReference($function_id) && $this->methodExists($function_id); if ($doesMethodExist) { $method_id = 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); } return $this->functions->getStorage($statements_analyzer, strtolower($function_id)); } /** * Whether or not a given method exists * * @param string|MethodIdentifier $method_id * @param string|MethodIdentifier|null $calling_method_id */ 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( MethodIdentifier::wrap($method_id), is_string($calling_method_id) ? strtolower($calling_method_id) : strtolower((string) $calling_method_id), $code_location, null, $file_path, true, $is_used, ); } /** * @param string|MethodIdentifier $method_id * @return array */ public function getMethodParams($method_id): array { return $this->methods->getMethodParams(MethodIdentifier::wrap($method_id)); } /** * @param string|MethodIdentifier $method_id */ public function isVariadic($method_id): bool { return $this->methods->isVariadic(MethodIdentifier::wrap($method_id)); } /** * @param string|MethodIdentifier $method_id * @param list $call_args */ public function getMethodReturnType($method_id, ?string &$self_class, array $call_args = []): ?Union { return $this->methods->getMethodReturnType( MethodIdentifier::wrap($method_id), $self_class, null, $call_args, ); } /** * @param string|MethodIdentifier $method_id */ public function getMethodReturnsByRef($method_id): bool { return $this->methods->getMethodReturnsByRef(MethodIdentifier::wrap($method_id)); } /** * @param string|MethodIdentifier $method_id */ public function getMethodReturnTypeLocation( $method_id, CodeLocation &$defined_location = null ): ?CodeLocation { return $this->methods->getMethodReturnTypeLocation( MethodIdentifier::wrap($method_id), $defined_location, ); } /** * @param string|MethodIdentifier $method_id */ public function getDeclaringMethodId($method_id): ?string { $new_method_id = $this->methods->getDeclaringMethodId(MethodIdentifier::wrap($method_id)); return $new_method_id ? (string) $new_method_id : null; } /** * Get the class this method appears in (vs is declared in, which could give a trait) * * @param string|MethodIdentifier $method_id */ public function getAppearingMethodId($method_id): ?string { $new_method_id = $this->methods->getAppearingMethodId(MethodIdentifier::wrap($method_id)); return $new_method_id ? (string) $new_method_id : null; } /** * @param string|MethodIdentifier $method_id * @return array */ public function getOverriddenMethodIds($method_id): array { return $this->methods->getOverriddenMethodIds(MethodIdentifier::wrap($method_id)); } /** * @param string|MethodIdentifier $method_id */ public function getCasedMethodId($method_id): string { 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); } 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); } public function getFunctionStorageForSymbol(string $file_path, string $symbol): ?FunctionLikeStorage { 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; } 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])) { return $file_storage->functions[$function_id]; } if (!$function_id) { return null; } return $this->functions->getStorage(null, $function_id); } /** * Get Markup content from Reference */ public function getMarkupContentForSymbolByReference( Reference $reference ): ?PHPMarkdownContent { //Direct Assignment if (is_numeric($reference->symbol[0])) { return new PHPMarkdownContent( preg_replace( '/^[^:]*:/', '', $reference->symbol, ), ); } //Class if (strpos($reference->symbol, '::')) { //Class Method if (strpos($reference->symbol, '()')) { $symbol = substr($reference->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 new PHPMarkdownContent( $storage->getHoverMarkdown(), "{$storage->defining_fqcln}::{$storage->cased_name}", $storage->description, ); } /** @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, ); } } //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, ); } //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, ); } return null; } /** @psalm-suppress PossiblyUndefinedIntArrayOffset */ [$fq_classlike_name, $const_name] = explode( '::', $reference->symbol, ); $class_constants = $this->classlikes->getConstantsForClass( $fq_classlike_name, ReflectionProperty::IS_PRIVATE, ); if (!isset($class_constants[$const_name])) { return null; } //Class Constant return new PHPMarkdownContent( $class_constants[$const_name]->getHoverMarkdown($const_name), $fq_classlike_name . '::' . $const_name, $class_constants[$const_name]->description, ); } //Procedural Function if (strpos($reference->symbol, '()')) { $function_id = strtolower(substr($reference->symbol, 0, -2)); $file_storage = $this->file_storage_provider->get( $reference->file_path, ); 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, ); } if (!$function_id) { return null; } $function = $this->functions->getStorage(null, $function_id); return new PHPMarkdownContent( $function->getHoverMarkdown(), $function_id, $function->description, ); } //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, ); } } 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, ); } } 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, ); //Global Constant if ($type) { return new PHPMarkdownContent( 'const ' . $reference->symbol . ' ' . $type, $reference->symbol, ); } } return new PHPMarkdownContent($reference->symbol); } /** * @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' => 'getCompletionSignature(), 'description' => $storage->description, ]; } [, $symbol_name] = explode('::', $symbol); if (strpos($symbol, '$') !== false) { $storage = $this->properties->getStorage($symbol); return [ 'type' => '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' => ' $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' => 'getCompletionSignature(), 'description' => $function_storage->description, ]; } if (!$function_id) { return null; } $function = $this->functions->getStorage(null, $function_id); return [ 'type' => 'getCompletionSignature(), 'description' => $function->description, ]; } if (strpos($symbol, '$') === 0) { $type = VariableFetchAnalyzer::getGlobalType($symbol, $this->analysis_php_version_id); if (!$type->isMixed()) { return ['type' => 'classlike_storage_provider->get($symbol); return [ 'type' => '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' => 'file_storage_provider->get($file_path); if (isset($file_storage->constants[$symbol])) { return ['type' => 'constants[$symbol]]; } $constant = ConstFetchAnalyzer::getGlobalConstType($this, $symbol, $symbol); if ($constant) { return ['type' => '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 { if (is_numeric($reference->symbol[0])) { $symbol = preg_replace('/:.*/', '', $reference->symbol); $symbol_parts = explode('-', $symbol); if (!isset($symbol_parts[0]) || !isset($symbol_parts[1])) { return null; } $file_contents = $this->getFileContents($reference->file_path); return new Raw( $file_contents, $reference->file_path, $this->config->shortenFileName($reference->file_path), (int) $symbol_parts[0], (int) $symbol_parts[1], ); } try { if (strpos($reference->symbol, '::')) { if (strpos($reference->symbol, '()')) { $symbol = substr($reference->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($reference->symbol, '$') !== false) { $storage = $this->properties->getStorage( $reference->symbol, ); return $storage->location; } /** @psalm-suppress PossiblyUndefinedIntArrayOffset */ [$fq_classlike_name, $const_name] = explode( '::', $reference->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($reference->symbol, '()')) { $file_storage = $this->file_storage_provider->get( $reference->file_path, ); $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; } return $this->functions->getStorage(null, $function_id) ->location; } return $this->classlike_storage_provider->get( $reference->symbol, )->location; } catch (UnexpectedValueException $e) { error_log($e->getMessage()); return null; } catch (InvalidArgumentException $e) { return null; } } /** * @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); if ($ref === null) { return null; } return [$ref->symbol, $ref->range]; } /** * Get Reference from Position */ public function getReferenceAtPositionAsReference( string $file_path, Position $position ): ?Reference { $is_open = $this->file_provider->isOpen($file_path); if (!$is_open) { throw new UnanalyzedFileException($file_path . ' is not open'); } $file_contents = $this->getFileContents($file_path); $offset = $position->toOffset($file_contents); $reference_maps = $this->analyzer->getMapsForFile($file_path); $reference_start_pos = null; $reference_end_pos = null; $symbol = null; foreach ($reference_maps as $reference_map) { ksort($reference_map); 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; } if ($symbol !== null && $reference_start_pos !== null && $reference_end_pos !== null ) { break; } } if ($symbol === null || $reference_start_pos === null || $reference_end_pos === null) { return null; } $range = new Range( self::getPositionFromOffset($reference_start_pos, $file_contents), self::getPositionFromOffset($reference_end_pos, $file_contents), ); 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) { 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), self::getPositionFromOffset($end_pos, $file_contents), ); return [$reference, $argument_number, $range]; } /** * @param non-empty-string $function_symbol */ public function getSignatureInformation( string $function_symbol, string $file_path = null ): ?SignatureInformation { $signature_label = ''; $signature_documentation = null; if (strpos($function_symbol, '::') !== false) { /** @psalm-suppress ArgumentTypeCoercion */ $method_id = new MethodIdentifier(...explode('::', $function_symbol)); $declaring_method_id = $this->methods->getDeclaringMethodId($method_id); 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; } else { try { 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)); } $params = $function_storage->params; $signature_label = $function_storage->cased_name; $signature_documentation = $function_storage->description; } catch (Exception $exception) { if (InternalCallMapHandler::inCallMap($function_symbol)) { $callables = InternalCallMapHandler::getCallablesFromCallMap($function_symbol); if (!$callables || !$callables[0]->params) { return null; } $params = $callables[0]->params; } else { return null; } } } $signature_label .= '('; $parameters = []; foreach ($params as $i => $param) { $parameter_label = ($param->type ?: 'mixed') . ' $' . $param->name; $parameters[] = new ParameterInformation( [ strlen($signature_label), strlen($signature_label) + strlen($parameter_label), ], $param->description ?? null, ); $signature_label .= $parameter_label; if ($i < (count($params) - 1)) { $signature_label .= ', '; } } $signature_label .= ')'; return new SignatureInformation( $signature_label, $parameters, $signature_documentation, ); } /** * @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) { 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; } /** @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); if ($candidate_gap === '[') { $gap = $candidate_gap; $recent_type = $possible_type; if ($recent_type === 'mixed') { return null; } return [$recent_type, $gap, $offset]; } } 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; } 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; } return $function_storage->params[$argument_num]->type; } return null; } /** * @return list */ 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) { 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 === '->') { $completion_item = new CompletionItem( $method_storage->cased_name, CompletionItemKind::METHOD, $method_storage->getCompletionSignature(), $method_storage->description, (string)$method_storage->visibility, $method_storage->cased_name, $method_storage->cased_name, null, null, new Command('Trigger parameter hints', 'editor.action.triggerParameterHints'), null, 2, ); 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; } } $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( $declaring_class . '::$' . $property_name, ); if ($property_storage->is_static || $gap === '->') { $completion_items[] = new CompletionItem( '$' . $property_name, CompletionItemKind::PROPERTY, $property_storage->getInfo(), $property_storage->description, (string)$property_storage->visibility, $property_name, ($gap === '::' ? '$' : '') . $property_name, ); } } foreach ($class_storage->constants as $const_name => $const) { $completion_items[] = new CompletionItem( $const_name, CompletionItemKind::VARIABLE, 'const ' . $const_name, $const->description, null, $const_name, $const_name, ); } } catch (Exception $e) { error_log($e->getMessage()); continue; } } } return $completion_items; } /** * @return list */ 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); } 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, $aliases->uses_flipped ?? [], 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); $extra_edits[] = new TextEdit( new Range( $position, $position, ), "\n" . 'use ' . $fq_class_name . ';', ); } else { $position = self::getPositionFromOffset($aliases->namespace_first_stmt_start, $file_contents); $extra_edits[] = new TextEdit( new Range( $position, $position, ), '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; } catch (Exception $e) { $description = null; } $completion_items[] = new CompletionItem( $fq_class_name, CompletionItemKind::CLASS_, null, $description, null, $fq_class_name, $insertion_text, null, $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; } $completion_items[] = new CompletionItem( $function_name, CompletionItemKind::FUNCTION, $function->getCompletionSignature(), $function->description, null, $function_name, $function_name . (count($function->params) !== 0 ? '($0)' : '()'), null, null, new Command('Trigger parameter hints', 'editor.action.triggerParameterHints'), null, 2, ); } return $completion_items; } /** * @return list */ public function getCompletionItemsForType(Union $type): array { $completion_items = []; foreach ($type->getAtomicTypes() as $atomic_type) { if ($atomic_type instanceof TBool) { $bools = (string) $atomic_type === 'bool' ? ['true', 'false'] : [(string) $atomic_type]; foreach ($bools as $property_name) { $completion_items[] = new CompletionItem( $property_name, CompletionItemKind::VALUE, 'bool', null, null, null, $property_name, ); } } elseif ($atomic_type instanceof TLiteralString) { $completion_items[] = new CompletionItem( $atomic_type->value, CompletionItemKind::VALUE, $atomic_type->getId(), null, null, null, "'$atomic_type->value'", ); } elseif ($atomic_type instanceof TLiteralInt) { $completion_items[] = new CompletionItem( (string) $atomic_type->value, CompletionItemKind::VALUE, $atomic_type->getId(), null, null, null, (string) $atomic_type->value, ); } elseif ($atomic_type instanceof TClassConstant) { $const = $atomic_type->fq_classlike_name . '::' . $atomic_type->const_name; $completion_items[] = new CompletionItem( $const, CompletionItemKind::VALUE, $atomic_type->getId(), null, null, null, $const, ); } } return $completion_items; } /** * @return list */ public function getCompletionItemsForArrayKeys( string $type_string ): array { $completion_items = []; $type = Type::parseString($type_string); foreach ($type->getAtomicTypes() as $atomic_type) { if ($atomic_type instanceof TKeyedArray) { foreach ($atomic_type->properties as $property_name => $property) { $completion_items[] = new CompletionItem( (string) $property_name, CompletionItemKind::PROPERTY, (string) $property, null, null, null, "'$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"), $offset - (int)$before_newline_count - 1, ); } public function addTemporaryFileChanges(string $file_path, string $new_content, ?int $version = null): void { $this->file_provider->addTemporaryFileChanges($file_path, $new_content, $version); } public function removeTemporaryFileChanges(string $file_path): void { $this->file_provider->removeTemporaryFileChanges($file_path); } /** * Checks if type is a subtype of other * * Given two types, checks if `$input_type` is a subtype of `$container_type`. * 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. */ public function isTypeContainedByType( Union $input_type, Union $container_type ): bool { 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`. * If you consider `Union` as a set of types, this will tell you if intersection * of `$input_type` with `$container_type` is not empty. * * $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( Union $input_type, Union $container_type ): bool { 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')) * // returns [Union(TInt), Union(TString)] * ``` * * @return array{Union, Union} */ 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 $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 $taints * @psalm-suppress PossiblyUnusedMethod */ public function addTaintSource( Union $expr_type, string $taint_id, array $taints = TaintKindGroup::ALL_INPUT, ?CodeLocation $code_location = null ): Union { if (!$this->taint_flow_graph) { return $expr_type; } $source = new TaintSource( $taint_id, $taint_id, $code_location, null, $taints, ); $this->taint_flow_graph->addSource($source); return $expr_type->addParentNodes([$source->id => $source]); } /** * @param array $taints * @psalm-suppress PossiblyUnusedMethod */ public function addTaintSink( string $taint_id, array $taints = TaintKindGroup::ALL_INPUT, ?CodeLocation $code_location = null ): void { if (!$this->taint_flow_graph) { return; } $sink = new TaintSink( $taint_id, $taint_id, $code_location, null, $taints, ); $this->taint_flow_graph->addSink($sink); } public function getMinorAnalysisPhpVersion(): int { return self::transformPhpVersionId($this->analysis_php_version_id % 10_000, 100); } public function getMajorAnalysisPhpVersion(): int { return self::transformPhpVersionId($this->analysis_php_version_id, 10_000); } public static function transformPhpVersionId(int $php_version_id, int $div): int { return intdiv($php_version_id, $div); } }