1
0
mirror of https://github.com/danog/psalm.git synced 2024-12-14 18:36:58 +01:00
psalm/lib/ClassChecker.php

176 lines
4.8 KiB
PHP
Raw Normal View History

2016-01-08 00:28:27 +01:00
<?php
2016-01-08 00:36:55 +01:00
namespace CodeInspector;
2016-01-08 00:28:27 +01:00
use \PhpParser;
use \PhpParser\Error;
use \PhpParser\ParserFactory;
class ClassChecker implements StatementsSource
2016-01-08 00:28:27 +01:00
{
protected $_file_name;
protected $_class;
protected $_namespace;
protected $_aliased_classes;
protected $_absolute_class;
protected $_class_properties = [];
protected $_has_custom_get = false;
2016-01-08 00:28:27 +01:00
protected static $_existing_classes = [];
public function __construct(PhpParser\Node\Stmt\Class_ $class, StatementsSource $source, $absolute_class)
2016-01-08 00:28:27 +01:00
{
$this->_class = $class;
$this->_namespace = $source->getNamespace();
$this->_aliased_classes = $source->getAliasedClasses();
$this->_file_name = $source->getFileName();
$this->_absolute_class = $absolute_class;
2016-01-08 00:28:27 +01:00
self::$_existing_classes[$absolute_class] = 1;
2016-01-08 00:28:27 +01:00
}
public function check()
{
if ($this->_class->extends instanceof PhpParser\Node\Name) {
self::checkClassName($this->_class->extends, $this->_namespace, $this->_aliased_classes, $this->_file_name);
}
2016-01-26 20:13:04 +01:00
$leftover_stmts = [];
try {
new \ReflectionMethod($this->_absolute_class . '::__get');
$this->_has_custom_get = true;
} catch (\ReflectionException $e) {}
2016-01-08 00:28:27 +01:00
foreach ($this->_class->stmts as $stmt) {
if ($stmt instanceof PhpParser\Node\Stmt\ClassMethod) {
$method_checker = new ClassMethodChecker($stmt, $this);
2016-01-08 00:28:27 +01:00
$method_checker->check();
2016-01-26 20:13:04 +01:00
} else {
if ($stmt instanceof PhpParser\Node\Stmt\Property) {
foreach ($stmt->props as $property) {
$this->_class_properties[] = $property->name;
}
}
2016-01-26 20:13:04 +01:00
$leftover_stmts[] = $stmt;
2016-01-08 00:28:27 +01:00
}
}
2016-01-26 20:13:04 +01:00
if ($leftover_stmts) {
$scope_vars = [];
$possibly_in_scope_vars = [];
(new StatementsChecker($this))->check($leftover_stmts, $scope_vars, $possibly_in_scope_vars);
}
2016-01-08 00:28:27 +01:00
}
public static function checkClassName(PhpParser\Node\Name $class_name, $namespace, array $aliased_classes, $file_name)
{
if ($class_name->parts[0] === 'static') {
return;
}
$absolute_class = self::getAbsoluteClassFromName($class_name, $namespace, $aliased_classes);
if (!isset(self::$_existing_classes[$absolute_class]) && !class_exists($absolute_class, true) && !interface_exists($absolute_class, true)) {
throw new CodeException('Class ' . $absolute_class . ' does not exist', $file_name, $class_name->getLine());
}
self::$_existing_classes[$absolute_class] = 1;
}
public static function getAbsoluteClassFromName(PhpParser\Node\Name $class_name, $namespace, array $aliased_classes)
{
if ($class_name instanceof PhpParser\Node\Name\FullyQualified) {
return '\\' . implode('\\', $class_name->parts);
}
return self::getAbsoluteClassFromString(implode('\\', $class_name->parts), $namespace, $aliased_classes);
2016-01-08 00:28:27 +01:00
}
public static function getAbsoluteClassFromString($class, $namespace, array $imported_namespaces)
{
2016-01-08 00:28:27 +01:00
if ($class[0] === '\\') {
return $class;
}
if (strpos($class, '\\') !== false) {
$class_parts = explode('\\', $class);
$first_namespace = array_shift($class_parts);
if (isset($imported_namespaces[$first_namespace])) {
return self::_addSlash($imported_namespaces[$first_namespace] . '\\' . implode('\\', $class_parts));
}
} elseif (isset($imported_namespaces[$class])) {
2016-01-08 00:28:27 +01:00
return self::_addSlash($imported_namespaces[$class]);
}
if ($namespace && substr($namespace, -1) !== '\\') {
$namespace .= '\\';
}
return self::_addSlash($namespace . $class);
}
protected static function _addSlash($class)
{
if ($class[0] === '\\') {
return $class;
}
return '\\' . $class;
}
public function getNamespace()
{
return $this->_namespace;
}
public function getAliasedClasses()
{
return $this->_aliased_classes;
}
public function getAbsoluteClass()
{
return $this->_absolute_class;
}
public function getClassName()
{
return $this->_class->name;
}
public function getClassExtends()
{
return $this->_class->extends;
}
public function getFileName()
{
return $this->_file_name;
}
public function getClassChecker()
{
return $this;
}
public function isStatic()
{
return false;
}
public function hasCustomGet()
{
return $this->_has_custom_get;
}
public function getPropertyNames()
{
return $this->_class_properties;
}
2016-01-08 00:28:27 +01:00
}