2016-11-20 21:49:06 -05:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Checker;
|
|
|
|
|
|
|
|
use PhpParser\Node\Stmt\Namespace_;
|
|
|
|
use PhpParser;
|
|
|
|
use Psalm\Context;
|
|
|
|
use Psalm\StatementsSource;
|
|
|
|
use Psalm\Type;
|
|
|
|
|
|
|
|
abstract class SourceChecker implements StatementsSource
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @var array<string, string>
|
|
|
|
*/
|
|
|
|
protected $aliased_classes = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var array<string, string>
|
|
|
|
*/
|
|
|
|
protected $aliased_classes_flipped = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var array<string, string>
|
|
|
|
*/
|
|
|
|
protected $aliased_functions = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var array<string, string>
|
|
|
|
*/
|
|
|
|
protected $aliased_constants = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $file_name;
|
|
|
|
|
2016-12-03 19:11:30 -05:00
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
protected $file_path;
|
|
|
|
|
2016-11-20 21:49:06 -05:00
|
|
|
/**
|
|
|
|
* @var string|null
|
|
|
|
*/
|
|
|
|
protected $include_file_name;
|
|
|
|
|
2016-12-03 19:11:30 -05:00
|
|
|
/**
|
|
|
|
* @var string|null
|
|
|
|
*/
|
|
|
|
protected $include_file_path;
|
|
|
|
|
2016-11-20 21:49:06 -05:00
|
|
|
/**
|
2017-01-02 01:07:44 -05:00
|
|
|
* @var array<int, string>
|
2016-11-20 21:49:06 -05:00
|
|
|
*/
|
|
|
|
protected $suppressed_issues = [];
|
|
|
|
|
|
|
|
/**
|
2016-12-17 00:48:31 -05:00
|
|
|
* @var array<string, bool>
|
2016-11-20 21:49:06 -05:00
|
|
|
*/
|
|
|
|
protected $declared_classes = [];
|
|
|
|
|
2016-11-20 23:45:10 -05:00
|
|
|
/**
|
|
|
|
* @param PhpParser\Node\Stmt\Use_ $stmt
|
|
|
|
* @return void
|
|
|
|
*/
|
2016-11-20 21:49:06 -05:00
|
|
|
public function visitUse(PhpParser\Node\Stmt\Use_ $stmt)
|
|
|
|
{
|
2016-11-20 22:40:19 -05:00
|
|
|
foreach ($stmt->uses as $use) {
|
|
|
|
$use_path = implode('\\', $use->name->parts);
|
|
|
|
|
|
|
|
switch ($use->type !== PhpParser\Node\Stmt\Use_::TYPE_UNKNOWN ? $use->type : $stmt->type) {
|
|
|
|
case PhpParser\Node\Stmt\Use_::TYPE_FUNCTION:
|
|
|
|
$this->aliased_functions[strtolower($use->alias)] = $use_path;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PhpParser\Node\Stmt\Use_::TYPE_CONSTANT:
|
|
|
|
$this->aliased_constants[$use->alias] = $use_path;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PhpParser\Node\Stmt\Use_::TYPE_NORMAL:
|
|
|
|
$this->aliased_classes[strtolower($use->alias)] = $use_path;
|
2016-12-10 13:07:12 -05:00
|
|
|
$this->aliased_classes_flipped[strtolower($use_path)] = $use->alias;
|
2016-11-20 22:40:19 -05:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-20 23:45:10 -05:00
|
|
|
/**
|
|
|
|
* @param PhpParser\Node\Stmt\GroupUse $stmt
|
|
|
|
* @return void
|
|
|
|
*/
|
2016-11-20 22:40:19 -05:00
|
|
|
public function visitGroupUse(PhpParser\Node\Stmt\GroupUse $stmt)
|
|
|
|
{
|
|
|
|
$use_prefix = implode('\\', $stmt->prefix->parts);
|
|
|
|
|
|
|
|
foreach ($stmt->uses as $use) {
|
|
|
|
$use_path = $use_prefix . '\\' . implode('\\', $use->name->parts);
|
|
|
|
|
|
|
|
switch ($use->type !== PhpParser\Node\Stmt\Use_::TYPE_UNKNOWN ? $use->type : $stmt->type) {
|
|
|
|
case PhpParser\Node\Stmt\Use_::TYPE_FUNCTION:
|
|
|
|
$this->aliased_functions[strtolower($use->alias)] = $use_path;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PhpParser\Node\Stmt\Use_::TYPE_CONSTANT:
|
|
|
|
$this->aliased_constants[$use->alias] = $use_path;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PhpParser\Node\Stmt\Use_::TYPE_NORMAL:
|
|
|
|
$this->aliased_classes[strtolower($use->alias)] = $use_path;
|
2016-12-10 13:07:12 -05:00
|
|
|
$this->aliased_classes_flipped[strtolower($use_path)] = $use->alias;
|
2016-11-20 22:40:19 -05:00
|
|
|
break;
|
|
|
|
}
|
2016-11-20 21:49:06 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $class_name
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function containsClass($class_name)
|
|
|
|
{
|
|
|
|
return isset($this->declared_classes[$class_name]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-12-24 18:23:22 +00:00
|
|
|
* @return array<string, string>
|
2016-11-20 21:49:06 -05:00
|
|
|
*/
|
|
|
|
public function getAliasedClasses()
|
|
|
|
{
|
|
|
|
return $this->aliased_classes;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getAliasedClassesFlipped()
|
|
|
|
{
|
|
|
|
return $this->aliased_classes_flipped;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a list of all aliased constants
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getAliasedConstants()
|
|
|
|
{
|
|
|
|
return $this->aliased_constants;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a list of all aliased functions
|
|
|
|
*
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function getAliasedFunctions()
|
|
|
|
{
|
|
|
|
return $this->aliased_functions;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a list of the classes declared
|
|
|
|
*
|
2016-12-17 00:48:31 -05:00
|
|
|
* @return array<string, bool>
|
2016-11-20 21:49:06 -05:00
|
|
|
*/
|
|
|
|
public function getDeclaredClasses()
|
|
|
|
{
|
|
|
|
return $this->declared_classes;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return null
|
|
|
|
*/
|
|
|
|
public function getFQCLN()
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return null
|
|
|
|
*/
|
|
|
|
public function getClassName()
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return null
|
|
|
|
*/
|
|
|
|
public function getClassLikeChecker()
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string|null
|
|
|
|
*/
|
|
|
|
public function getParentClass()
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getFileName()
|
|
|
|
{
|
|
|
|
return $this->file_name;
|
|
|
|
}
|
|
|
|
|
2016-12-03 19:11:30 -05:00
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getFilePath()
|
|
|
|
{
|
|
|
|
return $this->file_path;
|
|
|
|
}
|
|
|
|
|
2016-11-20 21:49:06 -05:00
|
|
|
/**
|
|
|
|
* @return null|string
|
|
|
|
*/
|
|
|
|
public function getIncludeFileName()
|
|
|
|
{
|
|
|
|
return $this->include_file_name;
|
|
|
|
}
|
|
|
|
|
2016-12-03 19:11:30 -05:00
|
|
|
/**
|
|
|
|
* @return null|string
|
|
|
|
*/
|
|
|
|
public function getIncludeFilePath()
|
|
|
|
{
|
|
|
|
return $this->include_file_path;
|
|
|
|
}
|
|
|
|
|
2016-11-20 21:49:06 -05:00
|
|
|
/**
|
|
|
|
* @param string|null $file_name
|
2016-12-03 19:11:30 -05:00
|
|
|
* @param string|null $file_path
|
2016-11-20 21:49:06 -05:00
|
|
|
* @return void
|
|
|
|
*/
|
2016-12-03 19:11:30 -05:00
|
|
|
public function setIncludeFileName($file_name, $file_path)
|
2016-11-20 21:49:06 -05:00
|
|
|
{
|
|
|
|
$this->include_file_name = $file_name;
|
2016-12-03 19:11:30 -05:00
|
|
|
$this->include_file_path = $file_path;
|
2016-11-20 21:49:06 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getCheckedFileName()
|
|
|
|
{
|
|
|
|
return $this->include_file_name ?: $this->file_name;
|
|
|
|
}
|
|
|
|
|
2016-12-03 19:11:30 -05:00
|
|
|
/**
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function getCheckedFilePath()
|
|
|
|
{
|
|
|
|
return $this->include_file_path ?: $this->file_path;
|
|
|
|
}
|
|
|
|
|
2016-11-20 21:49:06 -05:00
|
|
|
/**
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function isStatic()
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return null
|
|
|
|
*/
|
|
|
|
public function getSource()
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a list of suppressed issues
|
|
|
|
*
|
|
|
|
* @return array<string>
|
|
|
|
*/
|
|
|
|
public function getSuppressedIssues()
|
|
|
|
{
|
|
|
|
return $this->suppressed_issues;
|
|
|
|
}
|
2016-12-17 00:48:31 -05:00
|
|
|
|
|
|
|
public function getNamespace()
|
|
|
|
{
|
|
|
|
return '';
|
|
|
|
}
|
2016-11-20 21:49:06 -05:00
|
|
|
}
|