1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-10 23:18:40 +01:00
psalm/src/Psalm/Internal/Analyzer/InterfaceAnalyzer.php

184 lines
6.4 KiB
PHP
Raw Normal View History

<?php
2018-11-06 03:57:36 +01:00
namespace Psalm\Internal\Analyzer;
2021-12-03 21:40:18 +01:00
use InvalidArgumentException;
use LogicException;
use PhpParser;
use Psalm\CodeLocation;
2021-12-03 20:11:20 +01:00
use Psalm\Context;
use Psalm\FileManipulation;
use Psalm\Internal\Analyzer\Statements\Expression\ClassConstAnalyzer;
use Psalm\Internal\FileManipulation\FileManipulationBuffer;
2021-12-03 20:11:20 +01:00
use Psalm\Internal\Provider\NodeDataProvider;
use Psalm\Issue\ParseError;
use Psalm\Issue\UndefinedInterface;
2021-12-03 20:11:20 +01:00
use Psalm\IssueBuffer;
2021-12-03 21:40:18 +01:00
use UnexpectedValueException;
use function strtolower;
/**
* @internal
*/
2018-11-06 03:57:36 +01:00
class InterfaceAnalyzer extends ClassLikeAnalyzer
{
public function __construct(
PhpParser\Node\Stmt\Interface_ $interface,
SourceAnalyzer $source,
string $fq_interface_name
) {
2017-01-09 05:58:06 +01:00
parent::__construct($interface, $source, $fq_interface_name);
}
public function analyze(): void
{
if (!$this->class instanceof PhpParser\Node\Stmt\Interface_) {
2021-12-03 21:40:18 +01:00
throw new LogicException('Something went badly wrong');
}
$interface_context = new Context($this->fq_class_name);
$project_analyzer = $this->file_analyzer->project_analyzer;
$codebase = $project_analyzer->getCodebase();
$config = $project_analyzer->getConfig();
if ($this->class->extends) {
foreach ($this->class->extends as $extended_interface) {
$extended_interface_name = self::getFQCLNFromNameObject(
$extended_interface,
$this->getAliases()
);
$parent_reference_location = new CodeLocation($this, $extended_interface);
2018-11-06 03:57:36 +01:00
if (!$codebase->classOrInterfaceExists(
$extended_interface_name,
$parent_reference_location
)) {
// we should not normally get here
return;
}
try {
$extended_interface_storage = $codebase->classlike_storage_provider->get($extended_interface_name);
2021-12-03 21:40:18 +01:00
} catch (InvalidArgumentException $e) {
continue;
}
if (!$extended_interface_storage->is_interface) {
$code_location = new CodeLocation(
$this,
$extended_interface
);
2021-12-03 20:11:20 +01:00
IssueBuffer::maybeAdd(
new UndefinedInterface(
$extended_interface_name . ' is not an interface',
$code_location,
$extended_interface_name
),
$this->getSuppressedIssues()
);
}
2019-02-24 07:33:25 +01:00
if ($codebase->store_node_types && $extended_interface_name) {
$bounds = $parent_reference_location->getSelectionBounds();
$codebase->analyzer->addOffsetReference(
$this->getFilePath(),
$bounds[0],
$bounds[1],
$extended_interface_name
);
}
}
}
2019-01-19 18:19:51 +01:00
$fq_interface_name = $this->getFQCLN();
if (!$fq_interface_name) {
2021-12-03 21:40:18 +01:00
throw new UnexpectedValueException('bad');
}
$class_storage = $codebase->classlike_storage_provider->get($fq_interface_name);
2020-11-22 07:15:52 +01:00
foreach ($class_storage->attributes as $attribute) {
AttributeAnalyzer::analyze(
$this,
$attribute,
$class_storage->suppressed_issues + $this->getSuppressedIssues(),
1,
$class_storage
);
}
$member_stmts = [];
2019-01-19 18:19:51 +01:00
foreach ($this->class->stmts as $stmt) {
if ($stmt instanceof PhpParser\Node\Stmt\ClassMethod) {
$method_analyzer = new MethodAnalyzer($stmt, $this);
2021-12-03 20:11:20 +01:00
$type_provider = new NodeDataProvider();
2021-12-03 20:11:20 +01:00
$method_analyzer->analyze(new Context($this->getFQCLN()), $type_provider);
$actual_method_id = $method_analyzer->getMethodId();
if ($stmt->name->name !== '__construct'
&& $config->reportIssueInFile('InvalidReturnType', $this->getFilePath())
) {
ClassAnalyzer::analyzeClassMethodReturnType(
$stmt,
$method_analyzer,
$this,
$type_provider,
$codebase,
$class_storage,
$fq_interface_name,
$actual_method_id,
$actual_method_id,
false
);
}
} elseif ($stmt instanceof PhpParser\Node\Stmt\Property) {
2021-12-03 20:11:20 +01:00
IssueBuffer::add(
new ParseError(
'Interfaces cannot have properties',
new CodeLocation($this, $stmt)
)
);
return;
} elseif ($stmt instanceof PhpParser\Node\Stmt\ClassConst) {
$member_stmts[] = $stmt;
foreach ($stmt->consts as $const) {
$const_id = strtolower($this->fq_class_name) . '::' . $const->name;
foreach ($codebase->class_constants_to_rename as $original_const_id => $new_const_name) {
if ($const_id === $original_const_id) {
$file_manipulations = [
new FileManipulation(
(int) $const->name->getAttribute('startFilePos'),
(int) $const->name->getAttribute('endFilePos') + 1,
$new_const_name
)
];
FileManipulationBuffer::add(
$this->getFilePath(),
$file_manipulations
);
}
}
}
2019-01-19 18:19:51 +01:00
}
}
$statements_analyzer = new StatementsAnalyzer($this, new NodeDataProvider());
$statements_analyzer->analyze($member_stmts, $interface_context, null, true);
ClassConstAnalyzer::analyze($this->storage, $this->getCodebase());
}
}