2016-10-29 23:07:13 -04:00
|
|
|
<?php
|
2017-12-06 01:05:51 -05:00
|
|
|
namespace Psalm\Examples\Template;
|
|
|
|
|
|
|
|
use PhpParser;
|
2016-10-29 23:07:13 -04:00
|
|
|
use Psalm;
|
2018-11-05 21:57:36 -05:00
|
|
|
use Psalm\Codebase;
|
|
|
|
use Psalm\Internal\Analyzer\ClassAnalyzer;
|
|
|
|
use Psalm\Internal\Analyzer\ClassLikeAnalyzer;
|
|
|
|
use Psalm\Internal\Analyzer\MethodAnalyzer;
|
|
|
|
use Psalm\Internal\Analyzer\StatementsAnalyzer;
|
2017-12-06 01:05:51 -05:00
|
|
|
use Psalm\CodeLocation;
|
2016-10-29 23:07:13 -04:00
|
|
|
use Psalm\Context;
|
2018-11-05 21:57:36 -05:00
|
|
|
use Psalm\DocComment;
|
2016-10-29 23:07:13 -04:00
|
|
|
use Psalm\Type;
|
|
|
|
|
2018-11-05 21:57:36 -05:00
|
|
|
class TemplateAnalyzer extends Psalm\Internal\Analyzer\FileAnalyzer
|
2016-10-29 23:07:13 -04:00
|
|
|
{
|
2017-12-06 01:05:51 -05:00
|
|
|
const VIEW_CLASS = 'Your\\View\\Class';
|
2016-10-29 23:07:13 -04:00
|
|
|
|
2018-05-30 16:19:18 -04:00
|
|
|
public function analyze(Context $context = null, $update_docblocks = false, Context $global_context = null)
|
2017-12-06 01:05:51 -05:00
|
|
|
{
|
2018-11-11 12:01:14 -05:00
|
|
|
$codebase = $this->project_analyzer->getCodebase();
|
2018-01-21 13:38:51 -05:00
|
|
|
$stmts = $codebase->getStatementsForFile($this->file_path);
|
2017-12-06 01:05:51 -05:00
|
|
|
|
|
|
|
if (empty($stmts)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
$first_stmt = $stmts[0];
|
|
|
|
|
|
|
|
$this_params = null;
|
|
|
|
|
|
|
|
if (($first_stmt instanceof PhpParser\Node\Stmt\Nop) && ($doc_comment = $first_stmt->getDocComment())) {
|
2020-05-28 22:14:41 -04:00
|
|
|
$comment_block = DocComment::parsePreservingLength($doc_comment);
|
2017-12-06 01:05:51 -05:00
|
|
|
|
2020-05-28 22:14:41 -04:00
|
|
|
if (isset($comment_block->tags['variablesfrom'])) {
|
|
|
|
$variables_from = trim($comment_block->tags['variablesfrom'][0]);
|
2017-12-06 01:05:51 -05:00
|
|
|
|
|
|
|
$first_line_regex = '/([A-Za-z\\\0-9]+::[a-z_A-Z]+)(\s+weak)?/';
|
|
|
|
|
|
|
|
$matches = [];
|
|
|
|
|
|
|
|
if (!preg_match($first_line_regex, $variables_from, $matches)) {
|
|
|
|
throw new \InvalidArgumentException('Could not interpret doc comment correctly');
|
|
|
|
}
|
|
|
|
|
2020-02-14 20:54:26 -05:00
|
|
|
$method_id = new \Psalm\Internal\MethodIdentifier(...explode('::', $matches[1]));
|
|
|
|
|
|
|
|
$this_params = $this->checkMethod($method_id, $first_stmt, $codebase);
|
2016-10-29 23:07:13 -04:00
|
|
|
|
|
|
|
if ($this_params === false) {
|
2017-12-06 01:05:51 -05:00
|
|
|
return;
|
2016-10-29 23:07:13 -04:00
|
|
|
}
|
|
|
|
|
2017-12-06 01:05:51 -05:00
|
|
|
$this_params->vars_in_scope['$this'] = new Type\Union([
|
2018-01-18 17:41:14 -05:00
|
|
|
new Type\Atomic\TNamedObject(self::VIEW_CLASS),
|
2017-12-06 01:05:51 -05:00
|
|
|
]);
|
2016-10-29 23:07:13 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$this_params) {
|
2017-12-06 01:05:51 -05:00
|
|
|
$this_params = new Context();
|
2016-10-29 23:07:13 -04:00
|
|
|
$this_params->check_variables = false;
|
2017-12-06 01:05:51 -05:00
|
|
|
$this_params->self = self::VIEW_CLASS;
|
|
|
|
$this_params->vars_in_scope['$this'] = new Type\Union([
|
2018-01-18 17:41:14 -05:00
|
|
|
new Type\Atomic\TNamedObject(self::VIEW_CLASS),
|
2017-12-06 01:05:51 -05:00
|
|
|
]);
|
2016-10-29 23:07:13 -04:00
|
|
|
}
|
|
|
|
|
2018-01-21 12:44:46 -05:00
|
|
|
$this->checkWithViewClass($this_params, $stmts);
|
2016-10-29 23:07:13 -04:00
|
|
|
}
|
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
2020-02-14 20:54:26 -05:00
|
|
|
* @param \Psalm\Internal\MethodIdentifier $method_id
|
2017-12-06 01:05:51 -05:00
|
|
|
* @param PhpParser\Node $stmt
|
2018-01-18 17:41:14 -05:00
|
|
|
*
|
2017-12-06 01:05:51 -05:00
|
|
|
* @return Context|false
|
2016-11-02 02:29:00 -04:00
|
|
|
*/
|
2020-02-14 20:54:26 -05:00
|
|
|
private function checkMethod(\Psalm\Internal\MethodIdentifier $method_id, PhpParser\Node $stmt, Codebase $codebase)
|
2016-10-29 23:07:13 -04:00
|
|
|
{
|
2018-11-05 21:57:36 -05:00
|
|
|
if (ClassLikeAnalyzer::checkFullyQualifiedClassLikeName(
|
2018-01-01 20:04:03 -05:00
|
|
|
$this,
|
2020-02-14 20:54:26 -05:00
|
|
|
$method_id->fq_class_name,
|
2017-12-06 01:05:51 -05:00
|
|
|
new CodeLocation($this, $stmt),
|
2020-02-11 16:39:33 -05:00
|
|
|
null,
|
2020-03-26 12:35:27 -04:00
|
|
|
null,
|
2017-12-06 01:05:51 -05:00
|
|
|
[],
|
|
|
|
true
|
|
|
|
) === false
|
|
|
|
) {
|
2016-10-29 23:07:13 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-12-06 01:05:51 -05:00
|
|
|
$this_context = new Context();
|
2020-02-14 20:54:26 -05:00
|
|
|
$this_context->self = $method_id->fq_class_name;
|
|
|
|
|
|
|
|
$class_storage = $codebase->classlike_storage_provider->get($method_id->fq_class_name);
|
2016-10-29 23:07:13 -04:00
|
|
|
|
2020-02-14 20:54:26 -05:00
|
|
|
$this_context->vars_in_scope['$this'] = new Type\Union([new Type\Atomic\TNamedObject($class_storage->name)]);
|
2016-10-29 23:07:13 -04:00
|
|
|
|
2019-03-02 15:07:26 -05:00
|
|
|
$this->project_analyzer->getMethodMutations(
|
2020-02-14 20:54:26 -05:00
|
|
|
new \Psalm\Internal\MethodIdentifier($method_id->fq_class_name, '__construct'),
|
2019-03-02 15:07:26 -05:00
|
|
|
$this_context,
|
|
|
|
$this->getRootFilePath(),
|
|
|
|
$this->getRootFileName()
|
|
|
|
);
|
2016-10-29 23:07:13 -04:00
|
|
|
|
2020-02-14 20:54:26 -05:00
|
|
|
$this_context->vars_in_scope['$this'] = new Type\Union([new Type\Atomic\TNamedObject($class_storage->name)]);
|
2016-10-29 23:07:13 -04:00
|
|
|
|
|
|
|
// check the actual method
|
2019-03-02 15:07:26 -05:00
|
|
|
$this->project_analyzer->getMethodMutations(
|
|
|
|
$method_id,
|
|
|
|
$this_context,
|
|
|
|
$this->getRootFilePath(),
|
|
|
|
$this->getRootFileName()
|
|
|
|
);
|
2016-10-29 23:07:13 -04:00
|
|
|
|
2017-12-06 01:05:51 -05:00
|
|
|
$view_context = new Context();
|
2020-02-14 20:54:26 -05:00
|
|
|
$view_context->self = strtolower(self::VIEW_CLASS);
|
2016-10-29 23:07:13 -04:00
|
|
|
|
|
|
|
// add all $this-> vars to scope
|
2018-01-28 12:01:51 -05:00
|
|
|
foreach ($this_context->vars_possibly_in_scope as $var => $_) {
|
2016-10-29 23:07:13 -04:00
|
|
|
$view_context->vars_in_scope[str_replace('$this->', '$', $var)] = Type::getMixed();
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($this_context->vars_in_scope as $var => $type) {
|
|
|
|
$view_context->vars_in_scope[str_replace('$this->', '$', $var)] = $type;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $view_context;
|
|
|
|
}
|
|
|
|
|
2016-11-02 02:29:00 -04:00
|
|
|
/**
|
2017-12-06 01:05:51 -05:00
|
|
|
* @param Context $context
|
2018-01-21 12:44:46 -05:00
|
|
|
* @param array<PhpParser\Node\Stmt> $stmts
|
2018-01-18 17:41:14 -05:00
|
|
|
*
|
2017-12-06 01:05:51 -05:00
|
|
|
* @return void
|
2016-11-02 02:29:00 -04:00
|
|
|
*/
|
2018-01-21 12:44:46 -05:00
|
|
|
protected function checkWithViewClass(Context $context, array $stmts)
|
2016-10-29 23:07:13 -04:00
|
|
|
{
|
2017-12-06 01:05:51 -05:00
|
|
|
$pseudo_method_stmts = [];
|
2016-11-21 15:51:38 -05:00
|
|
|
|
2017-12-06 01:05:51 -05:00
|
|
|
foreach ($stmts as $stmt) {
|
|
|
|
if ($stmt instanceof PhpParser\Node\Stmt\Use_) {
|
|
|
|
$this->visitUse($stmt);
|
|
|
|
} else {
|
|
|
|
$pseudo_method_stmts[] = $stmt;
|
|
|
|
}
|
2016-11-21 15:51:38 -05:00
|
|
|
}
|
|
|
|
|
2017-12-06 01:05:51 -05:00
|
|
|
$pseudo_method_name = preg_replace('/[^a-zA-Z0-9_]+/', '_', $this->file_name);
|
2016-10-29 23:07:13 -04:00
|
|
|
|
2017-12-06 01:05:51 -05:00
|
|
|
$class_method = new PhpParser\Node\Stmt\ClassMethod($pseudo_method_name, ['stmts' => []]);
|
2016-10-29 23:07:13 -04:00
|
|
|
|
2017-12-06 01:05:51 -05:00
|
|
|
$class = new PhpParser\Node\Stmt\Class_(self::VIEW_CLASS);
|
2016-10-29 23:07:13 -04:00
|
|
|
|
2018-11-11 12:01:14 -05:00
|
|
|
$class_analyzer = new ClassAnalyzer($class, $this, self::VIEW_CLASS);
|
2017-12-06 01:05:51 -05:00
|
|
|
|
2018-11-11 12:01:14 -05:00
|
|
|
$view_method_analyzer = new MethodAnalyzer($class_method, $class_analyzer);
|
2017-12-06 01:05:51 -05:00
|
|
|
|
|
|
|
if (!$context->check_variables) {
|
2018-11-11 12:01:14 -05:00
|
|
|
$view_method_analyzer->addSuppressedIssue('UndefinedVariable');
|
2017-12-06 01:05:51 -05:00
|
|
|
}
|
2016-10-29 23:07:13 -04:00
|
|
|
|
2019-11-25 11:44:54 -05:00
|
|
|
$statements_source = new StatementsAnalyzer(
|
|
|
|
$view_method_analyzer,
|
|
|
|
new \Psalm\Internal\Provider\NodeDataProvider()
|
|
|
|
);
|
2016-10-29 23:07:13 -04:00
|
|
|
|
2018-11-05 21:57:36 -05:00
|
|
|
$statements_source->analyze($pseudo_method_stmts, $context);
|
2016-10-29 23:07:13 -04:00
|
|
|
}
|
|
|
|
}
|