2016-01-19 18:27:06 -05:00
|
|
|
<?php
|
2016-08-13 14:20:46 -04:00
|
|
|
namespace Psalm\Checker;
|
|
|
|
|
2016-02-04 09:22:46 -05:00
|
|
|
use PhpParser;
|
2017-05-19 00:48:26 -04:00
|
|
|
use PhpParser\Node\Stmt\Namespace_;
|
2016-11-02 02:29:00 -04:00
|
|
|
use Psalm\Context;
|
|
|
|
use Psalm\StatementsSource;
|
2016-11-20 21:49:06 -05:00
|
|
|
use Psalm\Type;
|
2016-01-19 18:27:06 -05:00
|
|
|
|
2016-11-20 21:49:06 -05:00
|
|
|
class NamespaceChecker extends SourceChecker implements StatementsSource
|
2016-01-19 18:27:06 -05:00
|
|
|
{
|
2017-01-07 14:35:07 -05:00
|
|
|
use CanAlias;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var FileChecker
|
|
|
|
*/
|
|
|
|
protected $source;
|
|
|
|
|
2016-10-15 00:12:57 -04:00
|
|
|
/**
|
|
|
|
* @var Namespace_
|
|
|
|
*/
|
2017-02-11 18:56:38 -05:00
|
|
|
private $namespace;
|
2016-10-15 00:12:57 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
2017-02-11 18:56:38 -05:00
|
|
|
private $namespace_name;
|
2016-10-15 00:12:57 -04:00
|
|
|
|
2017-03-19 15:37:07 -04:00
|
|
|
/**
|
|
|
|
* @var array<int, FunctionChecker>
|
|
|
|
*/
|
|
|
|
public $function_checkers = [];
|
|
|
|
|
2017-01-02 15:31:18 -05:00
|
|
|
/**
|
|
|
|
* @var array<int, ClassChecker>
|
|
|
|
*/
|
2017-01-12 00:54:41 -05:00
|
|
|
public $class_checkers = [];
|
2017-01-02 15:31:18 -05:00
|
|
|
|
2017-02-01 18:11:00 -05:00
|
|
|
/**
|
|
|
|
* @var array<int, ClassChecker>
|
|
|
|
*/
|
|
|
|
public $interface_checkers = [];
|
|
|
|
|
2016-10-15 00:12:57 -04:00
|
|
|
/**
|
2016-11-20 21:49:06 -05:00
|
|
|
* A lookup table for public namespace constants
|
|
|
|
*
|
|
|
|
* @var array<string, array<string, Type\Union>>
|
2016-07-22 13:29:46 -04:00
|
|
|
*/
|
2016-11-20 21:49:06 -05:00
|
|
|
protected static $public_namespace_constants = [];
|
2016-07-22 13:29:46 -04:00
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
|
|
|
* @param Namespace_ $namespace
|
2017-01-07 14:35:07 -05:00
|
|
|
* @param FileChecker $source
|
2016-11-02 02:29:00 -04:00
|
|
|
*/
|
2017-01-07 14:35:07 -05:00
|
|
|
public function __construct(Namespace_ $namespace, FileChecker $source)
|
2016-01-19 18:27:06 -05:00
|
|
|
{
|
2017-01-02 15:31:18 -05:00
|
|
|
$this->source = $source;
|
2016-08-13 18:54:49 -04:00
|
|
|
$this->namespace = $namespace;
|
2016-10-15 00:12:57 -04:00
|
|
|
$this->namespace_name = $this->namespace->name ? implode('\\', $this->namespace->name->parts) : '';
|
2016-01-19 18:27:06 -05:00
|
|
|
}
|
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
2016-11-12 18:51:48 -05:00
|
|
|
* @return void
|
2016-11-02 02:29:00 -04:00
|
|
|
*/
|
2017-07-25 16:11:02 -04:00
|
|
|
public function collectAnalyzableInformation()
|
2016-01-19 18:27:06 -05:00
|
|
|
{
|
|
|
|
$leftover_stmts = [];
|
2017-03-19 15:37:07 -04:00
|
|
|
$function_stmts = [];
|
2016-01-19 18:27:06 -05:00
|
|
|
|
2017-01-17 23:55:08 -05:00
|
|
|
if (!isset(self::$public_namespace_constants[$this->namespace_name])) {
|
|
|
|
self::$public_namespace_constants[$this->namespace_name] = [];
|
|
|
|
}
|
2016-11-20 21:49:06 -05:00
|
|
|
|
2017-01-16 18:33:04 -05:00
|
|
|
$namespace_context = new Context();
|
2017-02-26 23:09:18 -05:00
|
|
|
$namespace_context->collect_references = $this->getFileChecker()->project_checker->collect_references;
|
2017-01-02 15:31:18 -05:00
|
|
|
|
2016-08-13 18:54:49 -04:00
|
|
|
foreach ($this->namespace->stmts as $stmt) {
|
|
|
|
if ($stmt instanceof PhpParser\Node\Stmt\ClassLike) {
|
2017-07-25 16:11:02 -04:00
|
|
|
$this->collectAnalyzableClassLike($stmt);
|
2016-11-02 02:29:00 -04:00
|
|
|
} elseif ($stmt instanceof PhpParser\Node\Stmt\Use_) {
|
2016-11-20 21:49:06 -05:00
|
|
|
$this->visitUse($stmt);
|
2016-11-20 22:40:19 -05:00
|
|
|
} elseif ($stmt instanceof PhpParser\Node\Stmt\GroupUse) {
|
|
|
|
$this->visitGroupUse($stmt);
|
2016-11-20 21:49:06 -05:00
|
|
|
} elseif ($stmt instanceof PhpParser\Node\Stmt\Const_) {
|
|
|
|
foreach ($stmt->consts as $const) {
|
|
|
|
self::$public_namespace_constants[$this->namespace_name][$const->name] = Type::getMixed();
|
2016-01-19 18:27:06 -05:00
|
|
|
}
|
2017-01-02 15:31:18 -05:00
|
|
|
|
2016-11-20 21:49:06 -05:00
|
|
|
$leftover_stmts[] = $stmt;
|
2017-03-19 15:37:07 -04:00
|
|
|
} elseif ($stmt instanceof PhpParser\Node\Stmt\Function_) {
|
|
|
|
$function_stmts[] = $stmt;
|
2016-11-02 02:29:00 -04:00
|
|
|
} else {
|
2016-01-19 18:27:06 -05:00
|
|
|
$leftover_stmts[] = $stmt;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-19 15:37:07 -04:00
|
|
|
$function_checkers = [];
|
|
|
|
|
|
|
|
// hoist functions to the top
|
|
|
|
foreach ($function_stmts as $stmt) {
|
|
|
|
$function_checker = new FunctionChecker($stmt, $this);
|
|
|
|
|
|
|
|
$this->source->addNamespacedFunctionChecker(
|
|
|
|
(string)$function_checker->getMethodId(),
|
|
|
|
$function_checker
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-01-19 18:27:06 -05:00
|
|
|
if ($leftover_stmts) {
|
2017-01-02 15:31:18 -05:00
|
|
|
$statements_checker = new StatementsChecker($this);
|
2017-01-16 18:33:04 -05:00
|
|
|
$context = new Context();
|
2017-02-26 23:09:18 -05:00
|
|
|
$context->collect_references = $this->getFileChecker()->project_checker->collect_references;
|
2017-01-07 15:09:47 -05:00
|
|
|
$statements_checker->analyze($leftover_stmts, $context);
|
2016-01-19 18:27:06 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-20 22:02:26 -05:00
|
|
|
/**
|
|
|
|
* @param PhpParser\Node\Stmt\ClassLike $stmt
|
2017-05-26 20:16:18 -04:00
|
|
|
*
|
2016-11-20 22:02:26 -05:00
|
|
|
* @return void
|
|
|
|
*/
|
2017-07-25 16:11:02 -04:00
|
|
|
public function collectAnalyzableClassLike(PhpParser\Node\Stmt\ClassLike $stmt)
|
2017-02-01 18:11:00 -05:00
|
|
|
{
|
2016-12-03 22:41:45 -05:00
|
|
|
if (!$stmt->name) {
|
|
|
|
throw new \UnexpectedValueException('Did not expect anonymous class here');
|
|
|
|
}
|
2016-12-14 12:28:38 -05:00
|
|
|
|
2017-07-25 16:11:02 -04:00
|
|
|
$fq_class_name = ClassLikeChecker::getFQCLNFromString($stmt->name, $this->getAliases());
|
2017-01-31 19:21:33 -05:00
|
|
|
|
2016-11-20 21:49:06 -05:00
|
|
|
if ($stmt instanceof PhpParser\Node\Stmt\Class_) {
|
2017-02-01 18:11:00 -05:00
|
|
|
$this->source->addNamespacedClassChecker(
|
|
|
|
$fq_class_name,
|
|
|
|
new ClassChecker($stmt, $this, $fq_class_name)
|
|
|
|
);
|
2016-11-20 21:49:06 -05:00
|
|
|
} elseif ($stmt instanceof PhpParser\Node\Stmt\Interface_) {
|
2017-02-01 18:11:00 -05:00
|
|
|
$this->source->addNamespacedInterfaceChecker(
|
|
|
|
$fq_class_name,
|
|
|
|
new InterfaceChecker($stmt, $this, $fq_class_name)
|
|
|
|
);
|
2016-11-20 21:49:06 -05:00
|
|
|
}
|
2016-01-19 18:27:06 -05:00
|
|
|
}
|
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
2016-11-20 21:49:06 -05:00
|
|
|
public function getNamespace()
|
2016-01-19 18:27:06 -05:00
|
|
|
{
|
2016-11-20 21:49:06 -05:00
|
|
|
return $this->namespace_name;
|
2016-01-19 18:27:06 -05:00
|
|
|
}
|
|
|
|
|
2016-10-09 17:54:58 -04:00
|
|
|
/**
|
2016-11-20 21:49:06 -05:00
|
|
|
* @param string $const_name
|
|
|
|
* @param Type\Union $const_type
|
2017-05-26 20:16:18 -04:00
|
|
|
*
|
2016-11-02 02:29:00 -04:00
|
|
|
* @return void
|
2016-10-09 17:54:58 -04:00
|
|
|
*/
|
2016-11-20 21:49:06 -05:00
|
|
|
public function setConstType($const_name, Type\Union $const_type)
|
2016-08-24 19:00:44 -04:00
|
|
|
{
|
2016-11-20 21:49:06 -05:00
|
|
|
self::$public_namespace_constants[$this->namespace_name][$const_name] = $const_type;
|
2016-08-24 19:00:44 -04:00
|
|
|
}
|
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
2016-11-20 21:49:06 -05:00
|
|
|
* @param string $namespace_name
|
|
|
|
* @param mixed $visibility
|
2017-05-26 20:16:18 -04:00
|
|
|
*
|
2016-11-20 21:49:06 -05:00
|
|
|
* @return array<string,Type\Union>
|
2016-11-02 02:29:00 -04:00
|
|
|
*/
|
2016-11-20 21:49:06 -05:00
|
|
|
public static function getConstantsForNamespace($namespace_name, $visibility)
|
2016-08-24 19:00:44 -04:00
|
|
|
{
|
2016-11-20 21:49:06 -05:00
|
|
|
// remove for PHP 7.1 support
|
|
|
|
$visibility = \ReflectionProperty::IS_PUBLIC;
|
2016-08-24 19:00:44 -04:00
|
|
|
|
2016-11-20 21:49:06 -05:00
|
|
|
// @todo this does not allow for loading in namespace constants not already defined in the current sweep
|
2016-11-20 22:40:19 -05:00
|
|
|
if (!isset(self::$public_namespace_constants[$namespace_name])) {
|
|
|
|
self::$public_namespace_constants[$namespace_name] = [];
|
|
|
|
}
|
2016-05-15 23:06:03 -04:00
|
|
|
|
2016-11-20 21:49:06 -05:00
|
|
|
if ($visibility === \ReflectionProperty::IS_PUBLIC) {
|
|
|
|
return self::$public_namespace_constants[$namespace_name];
|
|
|
|
}
|
2016-07-22 13:29:46 -04:00
|
|
|
|
2016-11-20 21:49:06 -05:00
|
|
|
throw new \InvalidArgumentException('Given $visibility not supported');
|
2016-07-22 13:29:46 -04:00
|
|
|
}
|
2017-07-29 15:05:06 -04:00
|
|
|
|
|
|
|
public function getFileChecker()
|
|
|
|
{
|
|
|
|
return $this->source;
|
|
|
|
}
|
2016-01-19 18:27:06 -05:00
|
|
|
}
|