2019-03-01 14:57:10 +01:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Internal\Provider;
|
|
|
|
|
|
|
|
use Psalm\CodeLocation;
|
2021-01-06 15:05:53 +01:00
|
|
|
use Psalm\Plugin\EventHandler\Event\MethodExistenceProviderEvent;
|
|
|
|
use Psalm\Plugin\Hook\MethodExistenceProviderInterface as LegacyMethodExistenceProviderInterface;
|
|
|
|
use Psalm\Plugin\EventHandler\MethodExistenceProviderInterface;
|
2019-07-05 22:24:00 +02:00
|
|
|
use Psalm\StatementsSource;
|
2019-06-26 22:52:29 +02:00
|
|
|
use function strtolower;
|
2021-01-06 15:05:53 +01:00
|
|
|
use function is_subclass_of;
|
2019-03-01 14:57:10 +01:00
|
|
|
|
|
|
|
class MethodExistenceProvider
|
|
|
|
{
|
2021-01-06 15:05:53 +01:00
|
|
|
/**
|
|
|
|
* @var array<
|
|
|
|
* lowercase-string,
|
|
|
|
* array<\Closure(MethodExistenceProviderEvent) : ?bool>
|
|
|
|
* >
|
|
|
|
*/
|
|
|
|
private static $handlers = [];
|
|
|
|
|
2019-03-01 14:57:10 +01:00
|
|
|
/**
|
|
|
|
* @var array<
|
2020-12-29 12:42:12 +01:00
|
|
|
* lowercase-string,
|
2019-03-01 14:57:10 +01:00
|
|
|
* array<\Closure(
|
|
|
|
* string,
|
|
|
|
* string,
|
|
|
|
* ?StatementsSource=,
|
|
|
|
* ?CodeLocation
|
|
|
|
* ) : ?bool>
|
|
|
|
* >
|
|
|
|
*/
|
2021-01-06 15:05:53 +01:00
|
|
|
private static $legacy_handlers = [];
|
2019-03-01 14:57:10 +01:00
|
|
|
|
|
|
|
public function __construct()
|
|
|
|
{
|
|
|
|
self::$handlers = [];
|
2021-01-06 15:05:53 +01:00
|
|
|
self::$legacy_handlers = [];
|
2019-03-01 14:57:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-01-06 15:05:53 +01:00
|
|
|
* @param class-string<LegacyMethodExistenceProviderInterface>|class-string<MethodExistenceProviderInterface> $class
|
2019-03-01 14:57:10 +01:00
|
|
|
*/
|
2020-09-12 17:24:05 +02:00
|
|
|
public function registerClass(string $class): void
|
2019-03-01 14:57:10 +01:00
|
|
|
{
|
2021-01-06 15:05:53 +01:00
|
|
|
if (is_subclass_of($class, LegacyMethodExistenceProviderInterface::class, true)) {
|
|
|
|
$callable = \Closure::fromCallable([$class, 'doesMethodExist']);
|
2019-03-01 14:57:10 +01:00
|
|
|
|
2021-01-06 15:05:53 +01:00
|
|
|
foreach ($class::getClassLikeNames() as $fq_classlike_name) {
|
|
|
|
$this->registerLegacyClosure($fq_classlike_name, $callable);
|
|
|
|
}
|
|
|
|
} elseif (is_subclass_of($class, MethodExistenceProviderInterface::class, true)) {
|
|
|
|
$callable = \Closure::fromCallable([$class, 'doesMethodExist']);
|
|
|
|
|
|
|
|
foreach ($class::getClassLikeNames() as $fq_classlike_name) {
|
|
|
|
$this->registerClosure($fq_classlike_name, $callable);
|
|
|
|
}
|
2019-03-01 14:57:10 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-01-06 15:05:53 +01:00
|
|
|
* @param \Closure(MethodExistenceProviderEvent) : ?bool $c
|
|
|
|
*/
|
|
|
|
public function registerClosure(string $fq_classlike_name, \Closure $c): void
|
|
|
|
{
|
|
|
|
self::$handlers[strtolower($fq_classlike_name)][] = $c;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-03-01 14:57:10 +01:00
|
|
|
* @param \Closure(
|
|
|
|
* string,
|
|
|
|
* string,
|
|
|
|
* ?StatementsSource=,
|
|
|
|
* ?CodeLocation
|
|
|
|
* ) : ?bool $c
|
|
|
|
*/
|
2021-01-06 15:05:53 +01:00
|
|
|
public function registerLegacyClosure(string $fq_classlike_name, \Closure $c): void
|
2019-03-01 14:57:10 +01:00
|
|
|
{
|
2021-01-06 15:05:53 +01:00
|
|
|
self::$legacy_handlers[strtolower($fq_classlike_name)][] = $c;
|
2019-03-01 14:57:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public function has(string $fq_classlike_name) : bool
|
|
|
|
{
|
2021-01-06 15:05:53 +01:00
|
|
|
return isset(self::$handlers[strtolower($fq_classlike_name)]) ||
|
|
|
|
isset(self::$legacy_handlers[strtolower($fq_classlike_name)]);
|
2019-03-01 14:57:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public function doesMethodExist(
|
|
|
|
string $fq_classlike_name,
|
|
|
|
string $method_name_lowercase,
|
2020-09-07 01:36:47 +02:00
|
|
|
?StatementsSource $source = null,
|
|
|
|
?CodeLocation $code_location = null
|
2020-09-04 22:26:33 +02:00
|
|
|
): ?bool {
|
2021-01-06 15:05:53 +01:00
|
|
|
foreach (self::$handlers[strtolower($fq_classlike_name)] ?? [] as $method_handler) {
|
|
|
|
$event = new MethodExistenceProviderEvent(
|
|
|
|
$fq_classlike_name,
|
|
|
|
$method_name_lowercase,
|
|
|
|
$source,
|
|
|
|
$code_location
|
|
|
|
);
|
|
|
|
$method_exists = $method_handler($event);
|
|
|
|
|
|
|
|
if ($method_exists !== null) {
|
|
|
|
return $method_exists;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (self::$legacy_handlers[strtolower($fq_classlike_name)] ?? [] as $method_handler) {
|
2019-03-01 14:57:10 +01:00
|
|
|
$method_exists = $method_handler(
|
|
|
|
$fq_classlike_name,
|
|
|
|
$method_name_lowercase,
|
|
|
|
$source,
|
|
|
|
$code_location
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($method_exists !== null) {
|
|
|
|
return $method_exists;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|