1
0
mirror of https://github.com/danog/psalm.git synced 2024-12-15 10:57:08 +01:00
psalm/src/Psalm/Internal/Provider/FunctionExistenceProvider.php

98 lines
2.3 KiB
PHP
Raw Normal View History

<?php
namespace Psalm\Internal\Provider;
use PhpParser;
use Psalm\StatementsSource;
use \Psalm\Plugin\Hook\FunctionExistenceProviderInterface;
use function version_compare;
use const PHP_VERSION;
use function strtolower;
class FunctionExistenceProvider
{
/**
* @var array<
* string,
* array<\Closure(
* StatementsSource,
* string
* ) : ?bool>
* >
*/
private static $handlers = [];
public function __construct()
{
self::$handlers = [];
}
/**
* @param class-string<FunctionExistenceProviderInterface> $class
* @psalm-suppress PossiblyUnusedParam
* @return void
*/
public function registerClass(string $class)
{
if (version_compare(PHP_VERSION, '7.1.0') >= 0) {
/**
* @psalm-suppress UndefinedMethod
* @var \Closure
*/
$callable = \Closure::fromCallable([$class, 'doesFunctionExist']);
} else {
$callable = (new \ReflectionClass($class))->getMethod('doesFunctionExist')->getClosure(new $class);
if (!$callable) {
throw new \UnexpectedValueException('Callable must not be null');
}
}
foreach ($class::getFunctionIds() as $function_id) {
/** @psalm-suppress MixedTypeCoercion */
$this->registerClosure($function_id, $callable);
}
}
/**
* /**
* @param \Closure(
* StatementsSource,
* string
* ) : ?bool $c
*
* @return void
*/
public function registerClosure(string $function_id, \Closure $c)
{
self::$handlers[$function_id][] = $c;
}
public function has(string $function_id) : bool
{
return isset(self::$handlers[strtolower($function_id)]);
}
/**
* @param array<PhpParser\Node\Arg> $call_args
* @return ?bool
*/
public function doesFunctionExist(
StatementsSource $statements_source,
string $function_id
) {
foreach (self::$handlers[strtolower($function_id)] as $function_handler) {
$function_exists = $function_handler(
$statements_source,
$function_id
);
if ($function_exists !== null) {
return $function_exists;
}
}
return null;
}
}