> */ protected static $function_return_types = []; /** * @var array> */ protected static $function_return_type_locations = []; /** * @var array> */ protected static $function_namespaces = []; /** * @var array> */ protected static $existing_functions = []; /** * @var array> */ protected static $deprecated_functions = []; /** * @var array> */ protected static $have_registered_function = []; /** * @var array>> */ protected static $file_function_params = []; /** * @var array */ protected static $variadic_functions = []; /** * @var array> */ protected static $builtin_function_params = []; /** * @var array */ protected static $builtin_functions = []; /** * @var array>|null */ protected static $call_map = null; /** * @param mixed $function * @param StatementsSource $source * @param string $base_file_name */ public function __construct($function, StatementsSource $source, $base_file_name) { if (!$function instanceof PhpParser\Node\Stmt\Function_) { throw new \InvalidArgumentException('Bad'); } parent::__construct($function, $source); $this->registerFunction($function, $base_file_name); } /** * @param string $function_id * @param string $file_name * @return boolean */ public static function functionExists($function_id, $file_name) { if (isset(self::$existing_functions[$file_name][$function_id])) { return true; } if (strpos($function_id, '::') !== false) { $function_id = strtolower(preg_replace('/^[^:]+::/', '', $function_id)); } if (!isset(self::$builtin_functions[$function_id])) { self::extractReflectionInfo($function_id); } return self::$builtin_functions[$function_id]; } /** * @param string $function_id * @param string $file_name * @return array */ public static function getParams($function_id, $file_name) { if (isset(self::$builtin_functions[$function_id]) && self::$builtin_functions[$function_id]) { return self::$builtin_function_params[$function_id]; } return self::$file_function_params[$file_name][$function_id]; } /** * @param string $function_id * @param string $file_name * @return boolean */ public static function isVariadic($function_id, $file_name) { return isset(self::$variadic_functions[$file_name][$function_id]); } /** * @param string $function_id * @return void */ protected static function extractReflectionInfo($function_id) { try { $reflection_function = new \ReflectionFunction($function_id); $reflection_params = $reflection_function->getParameters(); self::$builtin_function_params[$function_id] = []; /** @var \ReflectionParameter $param */ foreach ($reflection_params as $param) { self::$builtin_function_params[$function_id][] = self::getReflectionParamArray($param); } self::$builtin_functions[$function_id] = true; } catch (\ReflectionException $e) { self::$builtin_functions[$function_id] = false; } } /** * @param string $function_id * @param string $file_name * @return Type\Union|null */ public static function getFunctionReturnType($function_id, $file_name) { if (!isset(self::$function_return_types[$file_name][$function_id])) { throw new \InvalidArgumentException('Do not know function ' . $function_id . ' in file ' . $file_name); } $function_return_types = self::$function_return_types[$file_name][$function_id]; return $function_return_types ? clone $function_return_types : null; } /** * @param string $function_id * @param string $file_name * @return CodeLocation|null */ public static function getFunctionReturnTypeLocation($function_id, $file_name) { if (!isset(self::$function_return_type_locations[$file_name][$function_id])) { throw new \InvalidArgumentException('Do not know function ' . $function_id . ' in file ' . $file_name); } return self::$function_return_type_locations[$file_name][$function_id] ?: null; } /** * @param PhpParser\Node\Stmt\Function_ $function * @param string $file_name * @return null|false */ protected function registerFunction(PhpParser\Node\Stmt\Function_ $function, $file_name) { $function_id = ($this->namespace ? strtolower($this->namespace) . '\\' : '') . strtolower($function->name); if (isset(self::$have_registered_function[$file_name][$function_id])) { return null; } self::$have_registered_function[$file_name][$function_id] = true; self::$function_namespaces[$file_name][$function_id] = $this->namespace; self::$existing_functions[$file_name][$function_id] = true; self::$file_function_params[$file_name][$function_id] = []; $function_param_names = []; /** @var PhpParser\Node\Param $param */ foreach ($function->getParams() as $param) { $param_array = self::getTranslatedParam( $param, $this ); self::$file_function_params[$file_name][$function_id][] = $param_array; $function_param_names[$param->name] = $param_array->type; } $config = Config::getInstance(); $return_type = null; $return_type_location = null; $docblock_info = null; $this->suppressed_issues = []; $doc_comment = $function->getDocComment(); if ($function->returnType) { $parser_return_type = $function->returnType; $suffix = ''; if ($parser_return_type instanceof PhpParser\Node\NullableType) { $suffix = '|null'; $parser_return_type = $parser_return_type->type; } $return_type = Type::parseString( (is_string($parser_return_type) ? $parser_return_type : ClassLikeChecker::getFQCLNFromNameObject( $parser_return_type, $this->namespace, $this->getAliasedClasses() ) ) . $suffix ); $return_type_location = new CodeLocation($this->getSource(), $function, false, self::RETURN_TYPE_REGEX); } if ($doc_comment) { try { $docblock_info = CommentChecker::extractDocblockInfo( (string)$doc_comment, $doc_comment->getLine() ); } catch (DocblockParseException $e) { if (IssueBuffer::accepts( new InvalidDocblock( 'Invalid type passed in docblock for ' . $this->getMethodId(), new CodeLocation($this, $function, true) ) )) { return false; } } if ($docblock_info) { if ($docblock_info->deprecated) { self::$deprecated_functions[$file_name][$function_id] = true; } if ($docblock_info->variadic) { self::$variadic_functions[$file_name][$function_id] = true; } $this->suppressed_issues = $docblock_info->suppress; if ($config->use_docblock_types) { if ($docblock_info->return_type) { $return_type = Type::parseString( self::fixUpLocalType( (string)$docblock_info->return_type, null, $this->namespace, $this->getAliasedClasses() ) ); if (!$return_type_location) { $return_type_location = new CodeLocation($this->getSource(), $function, true); } $return_type_location->setCommentLine($docblock_info->return_type_line_number); } if ($docblock_info->params) { $this->improveParamsFromDocblock( $docblock_info->params, $function_param_names, self::$file_function_params[$file_name][$function_id], new CodeLocation($this, $function, false) ); } } } } self::$function_return_type_locations[$file_name][$function_id] = self::$function_return_types[$file_name][$function_id] = $return_type ?: false; return null; } /** * @param string $function_id * @return array|null * @psalm-return array>|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 = []; foreach ($call_map_function_args as $arg_name => $arg_type) { $by_reference = false; $optional = 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; } $function_types[] = new FunctionLikeParameter( $arg_name, $by_reference, $arg_type ? Type::parseString($arg_type) : Type::getMixed(), null, $optional, false, $arg_name === '...' ); } $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(); } return Type::parseString($call_map[$call_map_key][0]); } /** * @param string $function_id * @param array $call_args * @param CodeLocation $code_location * @param array $suppressed_issues * @return Type\Union */ public static function getReturnTypeFromCallMapWithArgs( $function_id, array $call_args, CodeLocation $code_location, array $suppressed_issues ) { $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_args) { if (in_array($call_map_key, ['str_replace', 'preg_replace', 'preg_replace_callback'])) { if (isset($call_args[2]->value->inferredType)) { /** @var Type\Union */ $subject_type = $call_args[2]->value->inferredType; if (!$subject_type->hasString() && $subject_type->hasArray()) { return Type::getArray(); } return Type::getString(); } } if (in_array($call_map_key, ['pathinfo'])) { if (isset($call_args[1])) { return Type::getString(); } return Type::getArray(); } if (substr($call_map_key, 0, 6) === 'array_') { $array_return_type = self::getArrayReturnType( $call_map_key, $call_args, $code_location, $suppressed_issues ); if ($array_return_type) { return $array_return_type; } } if ($call_map_key === 'explode' || $call_map_key === 'preg_split') { return Type::parseString('array'); } } if (!$call_map[$call_map_key][0]) { return Type::getMixed(); } return Type::parseString($call_map[$call_map_key][0]); } /** * @param string $call_map_key * @param array $call_args * @param CodeLocation $code_location * @param array $suppressed_issues * @return Type\Union|null */ protected static function getArrayReturnType( $call_map_key, $call_args, CodeLocation $code_location, array $suppressed_issues ) { if ($call_map_key === 'array_map' || $call_map_key === 'array_filter') { return self::getArrayMapReturnType($call_map_key, $call_args, $code_location, $suppressed_issues); } $first_arg = isset($call_args[0]->value) ? $call_args[0]->value : null; $first_arg_array = $first_arg && isset($first_arg->inferredType) && isset($first_arg->inferredType->types['array']) && $first_arg->inferredType->types['array'] instanceof Type\Generic ? $first_arg->inferredType->types['array'] : null; if ($call_map_key === 'array_values' || $call_map_key === 'array_unique') { if ($first_arg_array) { $inner_type = clone $first_arg_array->type_params[1]; return new Type\Union([new Type\Generic('array', [Type::getInt(), $inner_type])]); } } if ($call_map_key === 'array_keys') { if ($first_arg_array) { $inner_type = clone $first_arg_array->type_params[0]; return new Type\Union([new Type\Generic('array', [Type::getInt(), $inner_type])]); } } if ($call_map_key === 'array_merge') { $inner_value_types = []; $inner_key_types = []; foreach ($call_args as $offset => $call_arg) { if (!isset($call_arg->value->inferredType)) { return Type::getArray(); } foreach ($call_arg->value->inferredType->types as $type_part) { if (!$type_part instanceof Type\Generic) { return Type::getArray(); } if ($type_part->type_params[1]->isEmpty()) { continue; } $inner_key_types = array_merge(array_values($type_part->type_params[0]->types), $inner_key_types); $inner_value_types = array_merge( array_values($type_part->type_params[1]->types), $inner_value_types ); } if ($inner_value_types) { return new Type\Union([ new Type\Generic( 'array', [ Type::combineTypes($inner_key_types), Type::combineTypes($inner_value_types) ] ) ]); } } return Type::getArray(); } if ($call_map_key === 'array_diff') { if (!$first_arg_array) { return Type::getArray(); } return new Type\Union([ new Type\Generic( 'array', [ Type::getInt(), clone $first_arg_array->type_params[1] ] ) ]); } if ($call_map_key === 'array_diff_key') { if (!$first_arg_array) { return Type::getArray(); } return clone $first_arg->inferredType; } if ($call_map_key === 'array_shift' || $call_map_key === 'array_pop') { if (!$first_arg_array) { return Type::getMixed(); } return clone $first_arg_array->type_params[1]; } return null; } /** * @param string $call_map_key * @param array $call_args * @param CodeLocation $code_location * @param array $suppressed_issues * @return Type\Union */ protected static function getArrayMapReturnType( $call_map_key, $call_args, CodeLocation $code_location, array $suppressed_issues ) { $function_index = $call_map_key === 'array_map' ? 0 : 1; $array_index = $call_map_key === 'array_map' ? 1 : 0; $array_arg = isset($call_args[$array_index]->value) ? $call_args[$array_index]->value : null; $array_arg_type = $array_arg && isset($array_arg->inferredType) && isset($array_arg->inferredType->types['array']) && $array_arg->inferredType->types['array'] instanceof Type\Generic ? $array_arg->inferredType->types['array'] : null; if (isset($call_args[$function_index])) { $function_call_arg = $call_args[$function_index]; if ($function_call_arg->value instanceof PhpParser\Node\Expr\Closure) { $closure_yield_types = []; $closure_return_types = EffectsAnalyser::getReturnTypes( $function_call_arg->value->stmts, $closure_yield_types, true ); if (!$closure_return_types) { IssueBuffer::accepts( new InvalidReturnType( 'No return type could be found in the closure passed to ' . $call_map_key, $code_location ), $suppressed_issues ); return Type::getArray(); } if ($call_map_key === 'array_map') { $inner_type = new Type\Union($closure_return_types); return new Type\Union([new Type\Generic('array', [Type::getInt(), $inner_type])]); } if ($array_arg_type) { $inner_type = clone $array_arg_type->type_params[1]; return new Type\Union([new Type\Generic('array', [Type::getInt(), $inner_type])]); } } elseif ($function_call_arg->value instanceof PhpParser\Node\Scalar\String_) { $mapped_function_id = strtolower($function_call_arg->value->value); $call_map = self::getCallMap(); if (isset($call_map[$mapped_function_id][0])) { if ($call_map[$mapped_function_id][0]) { $mapped_function_return = Type::parseString($call_map[$mapped_function_id][0]); return new Type\Union([new Type\Generic('array', [Type::getInt(), $mapped_function_return])]); } } else { // @todo handle array_map('some_custom_function', $arr) } } } // where there's no function passed to array_filter if ($call_map_key === 'array_filter' && $array_arg_type) { $inner_type = clone $array_arg_type->type_params[1]; $second_arg = isset($call_args[1]->value) ? $call_args[1]->value : null; if (!$second_arg) { $inner_type->removeType('null'); $inner_type->removeType('false'); } return new Type\Union([new Type\Generic('array', [Type::getInt(), $inner_type])]); } return Type::getArray(); } /** * Gets the method/function call map * * @return array> * @psalm-suppress MixedInferredReturnType as the use of require buggers things up * @psalm-suppress MixedArgument also buggered */ protected static function getCallMap() { if (self::$call_map !== null) { return self::$call_map; } /** @var array> */ $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)]); } /** * @return void */ public static function clearCache() { self::$function_return_types = []; self::$function_namespaces = []; self::$existing_functions = []; self::$deprecated_functions = []; self::$have_registered_function = []; self::$file_function_params = []; self::$variadic_functions = []; self::$builtin_function_params = []; self::$builtin_functions = []; } }