1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-22 05:41:20 +01:00
psalm/src/Psalm/Checker/NamespaceChecker.php

179 lines
5.2 KiB
PHP
Raw Normal View History

<?php
namespace Psalm\Checker;
2016-10-15 00:12:57 -04:00
use PhpParser\Node\Stmt\Namespace_;
2016-02-04 09:22:46 -05:00
use PhpParser;
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-11-20 21:49:06 -05:00
class NamespaceChecker extends SourceChecker implements StatementsSource
{
2017-01-07 14:35:07 -05:00
use CanAlias;
/**
* @var FileChecker
*/
protected $source;
2016-10-15 00:12:57 -04:00
/**
* @var Namespace_
*/
protected $namespace;
2016-10-15 00:12:57 -04:00
/**
* @var string
*/
protected $namespace_name;
2016-10-15 00:12:57 -04:00
/**
* @var array<int, ClassChecker>
*/
protected $class_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-11-20 21:49:06 -05:00
protected static $public_namespace_constants = [];
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)
{
$this->source = $source;
$this->namespace = $namespace;
2016-10-15 00:12:57 -04:00
$this->namespace_name = $this->namespace->name ? implode('\\', $this->namespace->name->parts) : '';
}
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
*/
public function visit()
{
$leftover_stmts = [];
2016-11-20 21:49:06 -05:00
self::$public_namespace_constants[$this->namespace_name] = [];
$classlike_checkers = [];
2017-01-07 14:35:07 -05:00
$namespace_context = new Context($this->source->getFileName());
foreach ($this->namespace->stmts as $stmt) {
if ($stmt instanceof PhpParser\Node\Stmt\ClassLike) {
$this->visitClassLike($stmt, $classlike_checkers);
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-11-20 21:49:06 -05:00
$leftover_stmts[] = $stmt;
2016-11-02 02:29:00 -04:00
} else {
$leftover_stmts[] = $stmt;
}
}
$this->class_checkers = [];
foreach ($classlike_checkers as $classlike_checker) {
if ($classlike_checker instanceof ClassChecker) {
$classlike_checker->visit(null, null);
$this->class_checkers[] = $classlike_checker;
} elseif ($classlike_checker instanceof InterfaceChecker) {
$classlike_checker->visit();
}
}
if ($leftover_stmts) {
$statements_checker = new StatementsChecker($this);
2017-01-07 14:35:07 -05:00
$context = new Context($this->source->getFileName());
$statements_checker->analyze($leftover_stmts, $context);
}
}
/**
* @param Context $context
* @return void
*/
2017-01-07 15:57:25 -05:00
public function analyze(Context $context)
{
foreach ($this->class_checkers as $class_checker) {
2017-01-07 15:57:25 -05:00
$class_checker->analyze(null, $context);
}
$this->class_checkers = [];
}
2016-11-20 22:02:26 -05:00
/**
* @param PhpParser\Node\Stmt\ClassLike $stmt
* @param array<ClassLikeChecker> $classlike_checkers
2016-11-20 22:02:26 -05:00
* @return void
*/
public function visitClassLike(
PhpParser\Node\Stmt\ClassLike $stmt,
array &$classlike_checkers
) {
if (!$stmt->name) {
throw new \UnexpectedValueException('Did not expect anonymous class here');
}
2016-12-14 12:28:38 -05:00
2017-01-07 14:35:07 -05:00
$fq_class_name = ClassLikeChecker::getFQCLNFromString($stmt->name, $this);
2016-11-20 21:49:06 -05:00
if ($stmt instanceof PhpParser\Node\Stmt\Class_) {
$classlike_checkers[] = new ClassChecker($stmt, $this, $fq_class_name);
2016-11-20 21:49:06 -05:00
} elseif ($stmt instanceof PhpParser\Node\Stmt\Interface_) {
$classlike_checkers[] = new InterfaceChecker($stmt, $this, $fq_class_name);
2016-11-20 21:49:06 -05:00
} elseif ($stmt instanceof PhpParser\Node\Stmt\Trait_) {
// register the trait checker
new TraitChecker($stmt, $this, $fq_class_name);
2016-11-20 21:49: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-11-20 21:49:06 -05:00
return $this->namespace_name;
}
/**
2016-11-20 21:49:06 -05:00
* @param string $const_name
* @param Type\Union $const_type
2016-11-02 02:29:00 -04:00
* @return void
*/
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
* @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-11-20 21:49:06 -05:00
if ($visibility === \ReflectionProperty::IS_PUBLIC) {
return self::$public_namespace_constants[$namespace_name];
}
2016-11-20 21:49:06 -05:00
throw new \InvalidArgumentException('Given $visibility not supported');
}
}