2022-01-23 19:25:36 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace Psalm\Internal\Provider;
|
|
|
|
|
|
|
|
use Closure;
|
|
|
|
use PhpParser;
|
|
|
|
use Psalm\CodeLocation;
|
|
|
|
use Psalm\Context;
|
|
|
|
use Psalm\Internal\Analyzer\StatementsAnalyzer;
|
2022-01-25 23:16:34 +01:00
|
|
|
use Psalm\Plugin\ArgTypeInferer;
|
|
|
|
use Psalm\Plugin\DynamicFunctionStorage;
|
|
|
|
use Psalm\Plugin\DynamicTemplateProvider;
|
2022-01-23 19:25:36 +01:00
|
|
|
use Psalm\Plugin\EventHandler\Event\FunctionDynamicStorageProviderEvent;
|
|
|
|
use Psalm\Plugin\EventHandler\FunctionDynamicStorageProviderInterface;
|
|
|
|
use Psalm\Storage\FunctionStorage;
|
|
|
|
|
|
|
|
use function strtolower;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
final class FunctionDynamicStorageProvider
|
|
|
|
{
|
2022-01-25 23:16:34 +01:00
|
|
|
/** @var array<lowercase-string, array<Closure(FunctionDynamicStorageProviderEvent): ?DynamicFunctionStorage>> */
|
2022-01-23 19:25:36 +01:00
|
|
|
private static $handlers = [];
|
|
|
|
|
2022-01-23 21:33:14 +01:00
|
|
|
/** @var array<lowercase-string, ?FunctionStorage> */
|
2022-01-23 19:25:36 +01:00
|
|
|
private static $dynamic_storages = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param class-string<FunctionDynamicStorageProviderInterface> $class
|
|
|
|
*/
|
|
|
|
public function registerClass(string $class): void
|
|
|
|
{
|
|
|
|
$callable = Closure::fromCallable([$class, 'getFunctionStorage']);
|
|
|
|
|
|
|
|
foreach ($class::getFunctionIds() as $function_id) {
|
|
|
|
$this->registerClosure($function_id, $callable);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2022-01-25 23:16:34 +01:00
|
|
|
* @param Closure(FunctionDynamicStorageProviderEvent): ?DynamicFunctionStorage $c
|
2022-01-23 19:25:36 +01:00
|
|
|
*/
|
|
|
|
public function registerClosure(string $fq_function_name, Closure $c): void
|
|
|
|
{
|
|
|
|
self::$handlers[strtolower($fq_function_name)][] = $c;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function has(string $fq_function_name): bool
|
|
|
|
{
|
|
|
|
return isset(self::$handlers[strtolower($fq_function_name)]);
|
|
|
|
}
|
|
|
|
|
2022-01-23 22:36:07 +01:00
|
|
|
public function getFunctionStorage(
|
2022-01-23 19:25:36 +01:00
|
|
|
PhpParser\Node\Expr\FuncCall $stmt,
|
|
|
|
StatementsAnalyzer $statements_analyzer,
|
|
|
|
string $function_id,
|
|
|
|
Context $context,
|
|
|
|
CodeLocation $code_location
|
|
|
|
): ?FunctionStorage {
|
2022-01-26 12:14:24 +01:00
|
|
|
if ($stmt->isFirstClassCallable()) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2022-01-23 19:25:36 +01:00
|
|
|
$dynamic_storage_id = strtolower($statements_analyzer->getFilePath())
|
|
|
|
. ':' . $stmt->getLine()
|
|
|
|
. ':' . (int)$stmt->getAttribute('startFilePos')
|
|
|
|
. ':dynamic-storage'
|
|
|
|
. ':-:' . strtolower($function_id);
|
|
|
|
|
|
|
|
if (isset(self::$dynamic_storages[$dynamic_storage_id])) {
|
|
|
|
return self::$dynamic_storages[$dynamic_storage_id];
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (self::$handlers[strtolower($function_id)] ?? [] as $class_handler) {
|
|
|
|
$event = new FunctionDynamicStorageProviderEvent(
|
2022-01-25 23:16:34 +01:00
|
|
|
new ArgTypeInferer($context, $statements_analyzer),
|
|
|
|
new DynamicTemplateProvider('fn-' . strtolower($function_id)),
|
2022-01-23 19:25:36 +01:00
|
|
|
$statements_analyzer,
|
|
|
|
$function_id,
|
2022-01-23 22:28:25 +01:00
|
|
|
$stmt,
|
2022-01-23 19:25:36 +01:00
|
|
|
$context,
|
2022-01-25 23:16:34 +01:00
|
|
|
$code_location,
|
2022-01-23 19:25:36 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
$result = $class_handler($event);
|
|
|
|
|
2022-01-25 23:16:34 +01:00
|
|
|
return self::$dynamic_storages[$dynamic_storage_id] = $result
|
|
|
|
? $result->toFunctionStorage($function_id)
|
|
|
|
: null;
|
2022-01-23 19:25:36 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|