1
0
mirror of https://github.com/danog/psalm.git synced 2024-12-14 02:07:37 +01:00
psalm/src/Psalm/Internal/Codebase/Functions.php
2019-02-17 13:17:45 -05:00

235 lines
7.2 KiB
PHP

<?php
namespace Psalm\Internal\Codebase;
use Psalm\Codebase;
use Psalm\Internal\Analyzer\StatementsAnalyzer;
use Psalm\Internal\Provider\{FileStorageProvider, FunctionReturnTypeProvider};
use Psalm\StatementsSource;
use Psalm\Storage\FunctionLikeStorage;
/**
* @internal
*/
class Functions
{
/**
* @var FileStorageProvider
*/
private $file_storage_provider;
/**
* @var array<string, FunctionLikeStorage>
*/
private static $stubbed_functions;
/** @var FunctionReturnTypeProvider */
public $return_type_provider;
/**
* @var Reflection
*/
private $reflection;
public function __construct(FileStorageProvider $storage_provider, Reflection $reflection)
{
$this->file_storage_provider = $storage_provider;
$this->reflection = $reflection;
$this->return_type_provider = new FunctionReturnTypeProvider();
self::$stubbed_functions = [];
}
/**
* @param StatementsAnalyzer|null $statements_analyzer
* @param string $function_id
*
* @return FunctionLikeStorage
*/
public function getStorage($statements_analyzer, $function_id)
{
if (isset(self::$stubbed_functions[strtolower($function_id)])) {
return self::$stubbed_functions[strtolower($function_id)];
}
if ($this->reflection->hasFunction($function_id)) {
return $this->reflection->getFunctionStorage($function_id);
}
if (!$statements_analyzer) {
throw new \UnexpectedValueException('$statements_analyzer must not be null here');
}
$file_path = $statements_analyzer->getRootFilePath();
$checked_file_path = $statements_analyzer->getFilePath();
$file_storage = $this->file_storage_provider->get($file_path);
$function_analyzers = $statements_analyzer->getFunctionAnalyzers();
if (isset($function_analyzers[$function_id])) {
$function_id = $function_analyzers[$function_id]->getMethodId();
if (isset($file_storage->functions[$function_id])) {
return $file_storage->functions[$function_id];
}
}
// closures can be returned here
if (isset($file_storage->functions[$function_id])) {
return $file_storage->functions[$function_id];
}
if (!isset($file_storage->declaring_function_ids[$function_id])) {
if ($checked_file_path !== $file_path) {
$file_storage = $this->file_storage_provider->get($checked_file_path);
if (isset($file_storage->functions[$function_id])) {
return $file_storage->functions[$function_id];
}
}
throw new \UnexpectedValueException(
'Expecting ' . $function_id . ' to have storage in ' . $file_path
);
}
$declaring_file_path = $file_storage->declaring_function_ids[$function_id];
$declaring_file_storage = $this->file_storage_provider->get($declaring_file_path);
if (!isset($declaring_file_storage->functions[$function_id])) {
throw new \UnexpectedValueException(
'Not expecting ' . $function_id . ' to not have storage in ' . $declaring_file_path
);
}
return $declaring_file_storage->functions[$function_id];
}
/**
* @param string $function_id
* @param FunctionLikeStorage $storage
*
* @return void
*/
public function addGlobalFunction($function_id, FunctionLikeStorage $storage)
{
self::$stubbed_functions[strtolower($function_id)] = $storage;
}
/**
* @param string $function_id
*
* @return bool
*/
public function hasStubbedFunction($function_id)
{
return isset(self::$stubbed_functions[strtolower($function_id)]);
}
/**
* @param string $function_id
*
* @return bool
*/
public function functionExists(StatementsAnalyzer $statements_analyzer, $function_id)
{
$file_storage = $this->file_storage_provider->get($statements_analyzer->getRootFilePath());
if (isset($file_storage->declaring_function_ids[$function_id])) {
return true;
}
if ($this->reflection->hasFunction($function_id)) {
return true;
}
if (isset(self::$stubbed_functions[strtolower($function_id)])) {
return true;
}
if (isset($statements_analyzer->getFunctionAnalyzers()[$function_id])) {
return true;
}
$predefined_functions = $statements_analyzer->getCodebase()->config->getPredefinedFunctions();
if (isset($predefined_functions[$function_id])) {
if ($this->reflection->registerFunction($function_id) === false) {
return false;
}
return true;
}
return false;
}
/**
* @param string $function_name
* @param StatementsSource $source
*
* @return string
*/
public function getFullyQualifiedFunctionNameFromString($function_name, StatementsSource $source)
{
if (empty($function_name)) {
throw new \InvalidArgumentException('$function_name cannot be empty');
}
if ($function_name[0] === '\\') {
return substr($function_name, 1);
}
$function_name_lcase = strtolower($function_name);
$aliases = $source->getAliases();
$imported_function_namespaces = $aliases->functions;
$imported_namespaces = $aliases->uses;
if (strpos($function_name, '\\') !== false) {
$function_name_parts = explode('\\', $function_name);
$first_namespace = array_shift($function_name_parts);
$first_namespace_lcase = strtolower($first_namespace);
if (isset($imported_namespaces[$first_namespace_lcase])) {
return $imported_namespaces[$first_namespace_lcase] . '\\' . implode('\\', $function_name_parts);
}
if (isset($imported_function_namespaces[$first_namespace_lcase])) {
return $imported_function_namespaces[$first_namespace_lcase] . '\\' .
implode('\\', $function_name_parts);
}
} elseif (isset($imported_function_namespaces[$function_name_lcase])) {
return $imported_function_namespaces[$function_name_lcase];
}
$namespace = $source->getNamespace();
return ($namespace ? $namespace . '\\' : '') . $function_name;
}
/**
* @param string $function_id
* @param string $file_path
*
* @return bool
*/
public static function isVariadic(Codebase $codebase, $function_id, $file_path)
{
$file_storage = $codebase->file_storage_provider->get($file_path);
if (!isset($file_storage->declaring_function_ids[$function_id])) {
return false;
}
$declaring_file_path = $file_storage->declaring_function_ids[$function_id];
$file_storage = $declaring_file_path === $file_path
? $file_storage
: $codebase->file_storage_provider->get($declaring_file_path);
return isset($file_storage->functions[$function_id]) && $file_storage->functions[$function_id]->variadic;
}
}