1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-05 20:48:45 +01:00
psalm/src/Psalm/Codebase/CallMap.php

165 lines
4.3 KiB
PHP

<?php
namespace Psalm\Codebase;
use Psalm\Type;
use Psalm\Storage\FunctionLikeParameter;
/**
* @internal
*
* Gets values from the call map array, which stores data about native functions and methods
*/
class CallMap
{
/**
* @var array<array<string,string>>|null
*/
private static $call_map = null;
/**
* @param string $function_id
*
* @return array|null
* @psalm-return array<int, array<int, FunctionLikeParameter>>|null
*/
public static function getParamsFromCallMap($function_id)
{
$call_map = self::getCallMap();
$call_map_key = strtolower($function_id);
if (!isset($call_map[$call_map_key])) {
return null;
}
$call_map_functions = [];
$call_map_functions[] = $call_map[$call_map_key];
for ($i = 1; $i < 10; ++$i) {
if (!isset($call_map[$call_map_key . '\'' . $i])) {
break;
}
$call_map_functions[] = $call_map[$call_map_key . '\'' . $i];
}
$function_type_options = [];
foreach ($call_map_functions as $call_map_function_args) {
array_shift($call_map_function_args);
$function_types = [];
/** @var string $arg_name - key type changed with above array_shift */
foreach ($call_map_function_args as $arg_name => $arg_type) {
$by_reference = false;
$optional = false;
$variadic = false;
if ($arg_name[0] === '&') {
$arg_name = substr($arg_name, 1);
$by_reference = true;
}
if (substr($arg_name, -1) === '=') {
$arg_name = substr($arg_name, 0, -1);
$optional = true;
}
if (substr($arg_name, 0, 3) === '...') {
$arg_name = substr($arg_name, 3);
$variadic = true;
}
$param_type = $arg_type
? Type::parseString($arg_type)
: Type::getMixed();
if ($param_type->hasScalarType() || $param_type->hasObject()) {
$param_type->from_docblock = true;
}
$function_types[] = new FunctionLikeParameter(
$arg_name,
$by_reference,
$param_type,
null,
null,
$optional,
false,
$variadic
);
}
$function_type_options[] = $function_types;
}
return $function_type_options;
}
/**
* @param string $function_id
*
* @return Type\Union
*/
public static function getReturnTypeFromCallMap($function_id)
{
$call_map_key = strtolower($function_id);
$call_map = self::getCallMap();
if (!isset($call_map[$call_map_key])) {
throw new \InvalidArgumentException('Function ' . $function_id . ' was not found in callmap');
}
if (!$call_map[$call_map_key][0]) {
return Type::getMixed();
}
$call_map_type = Type::parseString($call_map[$call_map_key][0]);
if ($call_map_type->isNullable()) {
$call_map_type->from_docblock = true;
}
return $call_map_type;
}
/**
* Gets the method/function call map
*
* @return array<string, array<int|string, string>>
* @psalm-suppress MixedInferredReturnType as the use of require buggers things up
* @psalm-suppress MixedAssignment
* @psalm-suppress MixedTypeCoercion
*/
public static function getCallMap()
{
if (self::$call_map !== null) {
return self::$call_map;
}
/** @var array<string, array<int|string, string>> */
$call_map = require_once(__DIR__ . '/../CallMap.php');
self::$call_map = [];
foreach ($call_map as $key => $value) {
$cased_key = strtolower($key);
self::$call_map[$cased_key] = $value;
}
return self::$call_map;
}
/**
* @param string $key
*
* @return bool
*/
public static function inCallMap($key)
{
return isset(self::getCallMap()[strtolower($key)]);
}
}