2019-11-25 17:44:54 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Psalm\Internal\Provider;
|
|
|
|
|
|
|
|
use PhpParser;
|
2019-11-26 23:48:47 +01:00
|
|
|
use SplObjectStorage;
|
2019-11-25 17:44:54 +01:00
|
|
|
use Psalm\Type\Union;
|
|
|
|
|
|
|
|
class NodeDataProvider implements \Psalm\NodeTypeProvider
|
|
|
|
{
|
2019-11-26 23:48:47 +01:00
|
|
|
/** @var SplObjectStorage<PhpParser\Node, Union> */
|
|
|
|
private $node_types;
|
2019-11-25 17:44:54 +01:00
|
|
|
|
2019-11-26 23:48:47 +01:00
|
|
|
/** @var SplObjectStorage<PhpParser\Node, array<string, non-empty-list<non-empty-list<string>>>|null> */
|
|
|
|
private $node_assertions;
|
2019-11-25 17:44:54 +01:00
|
|
|
|
2019-11-26 23:48:47 +01:00
|
|
|
/** @var SplObjectStorage<PhpParser\Node, array<int, \Psalm\Storage\Assertion>> */
|
|
|
|
private $node_if_true_assertions;
|
2019-11-25 17:44:54 +01:00
|
|
|
|
2019-11-26 23:48:47 +01:00
|
|
|
/** @var SplObjectStorage<PhpParser\Node, array<int, \Psalm\Storage\Assertion>> */
|
|
|
|
private $node_if_false_assertions;
|
2019-11-25 17:44:54 +01:00
|
|
|
|
|
|
|
/** @var bool */
|
|
|
|
public $cache_assertions = true;
|
|
|
|
|
2019-11-26 23:48:47 +01:00
|
|
|
public function __construct()
|
|
|
|
{
|
|
|
|
/** @psalm-suppress PropertyTypeCoercion */
|
|
|
|
$this->node_types = new SplObjectStorage();
|
|
|
|
/** @psalm-suppress PropertyTypeCoercion */
|
|
|
|
$this->node_assertions = new SplObjectStorage();
|
|
|
|
/** @psalm-suppress PropertyTypeCoercion */
|
|
|
|
$this->node_if_true_assertions = new SplObjectStorage();
|
|
|
|
/** @psalm-suppress PropertyTypeCoercion */
|
|
|
|
$this->node_if_false_assertions = new SplObjectStorage();
|
|
|
|
}
|
|
|
|
|
2019-11-25 17:44:54 +01:00
|
|
|
/**
|
|
|
|
* @param PhpParser\Node\Expr|PhpParser\Node\Name|PhpParser\Node\Stmt\Return_ $node
|
|
|
|
*/
|
2020-10-12 21:46:47 +02:00
|
|
|
public function setType(PhpParser\NodeAbstract $node, Union $type) : void
|
2019-11-25 17:44:54 +01:00
|
|
|
{
|
2019-11-26 23:48:47 +01:00
|
|
|
$this->node_types[$node] = $type;
|
2019-11-25 17:44:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param PhpParser\Node\Expr|PhpParser\Node\Name|PhpParser\Node\Stmt\Return_ $node
|
|
|
|
*/
|
2020-10-12 21:46:47 +02:00
|
|
|
public function getType(PhpParser\NodeAbstract $node) : ?Union
|
2019-11-25 17:44:54 +01:00
|
|
|
{
|
2019-11-26 23:48:47 +01:00
|
|
|
return $this->node_types[$node] ?? null;
|
2019-11-25 17:44:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param array<string, non-empty-list<non-empty-list<string>>>|null $assertions
|
|
|
|
*/
|
2020-09-07 01:36:47 +02:00
|
|
|
public function setAssertions(PhpParser\Node\Expr $node, ?array $assertions) : void
|
2019-11-25 17:44:54 +01:00
|
|
|
{
|
|
|
|
if (!$this->cache_assertions) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-26 23:48:47 +01:00
|
|
|
$this->node_assertions[$node] = $assertions;
|
2019-11-25 17:44:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return array<string, non-empty-list<non-empty-list<string>>>|null
|
|
|
|
*/
|
2020-09-07 01:36:47 +02:00
|
|
|
public function getAssertions(PhpParser\Node\Expr $node) : ?array
|
2019-11-25 17:44:54 +01:00
|
|
|
{
|
|
|
|
if (!$this->cache_assertions) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2019-11-26 23:48:47 +01:00
|
|
|
return $this->node_assertions[$node] ?? null;
|
2019-11-25 17:44:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param PhpParser\Node\Expr\FuncCall|PhpParser\Node\Expr\MethodCall|PhpParser\Node\Expr\StaticCall $node
|
|
|
|
* @param array<int, \Psalm\Storage\Assertion> $assertions
|
|
|
|
*/
|
2020-10-12 21:46:47 +02:00
|
|
|
public function setIfTrueAssertions(PhpParser\Node\Expr $node, array $assertions) : void
|
2019-11-25 17:44:54 +01:00
|
|
|
{
|
2019-11-26 23:48:47 +01:00
|
|
|
$this->node_if_true_assertions[$node] = $assertions;
|
2019-11-25 17:44:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param PhpParser\Node\Expr\FuncCall|PhpParser\Node\Expr\MethodCall|PhpParser\Node\Expr\StaticCall $node
|
|
|
|
* @return array<int, \Psalm\Storage\Assertion>|null
|
|
|
|
*/
|
2020-10-12 21:46:47 +02:00
|
|
|
public function getIfTrueAssertions(PhpParser\Node\Expr $node) : ?array
|
2019-11-25 17:44:54 +01:00
|
|
|
{
|
2019-11-26 23:48:47 +01:00
|
|
|
return $this->node_if_true_assertions[$node] ?? null;
|
2019-11-25 17:44:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param PhpParser\Node\Expr\FuncCall|PhpParser\Node\Expr\MethodCall|PhpParser\Node\Expr\StaticCall $node
|
|
|
|
* @param array<int, \Psalm\Storage\Assertion> $assertions
|
|
|
|
*/
|
2020-10-12 21:46:47 +02:00
|
|
|
public function setIfFalseAssertions(PhpParser\Node\Expr $node, array $assertions) : void
|
2019-11-25 17:44:54 +01:00
|
|
|
{
|
2019-11-26 23:48:47 +01:00
|
|
|
$this->node_if_false_assertions[$node] = $assertions;
|
2019-11-25 17:44:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param PhpParser\Node\Expr\FuncCall|PhpParser\Node\Expr\MethodCall|PhpParser\Node\Expr\StaticCall $node
|
|
|
|
* @return array<int, \Psalm\Storage\Assertion>|null
|
|
|
|
*/
|
2020-10-12 21:46:47 +02:00
|
|
|
public function getIfFalseAssertions(PhpParser\Node\Expr $node) : ?array
|
2019-11-25 17:44:54 +01:00
|
|
|
{
|
2019-11-26 23:48:47 +01:00
|
|
|
return $this->node_if_false_assertions[$node] ?? null;
|
2019-11-25 17:44:54 +01:00
|
|
|
}
|
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public function isPureCompatible(PhpParser\Node\Expr $node) : bool
|
2019-11-25 17:44:54 +01:00
|
|
|
{
|
|
|
|
$node_type = self::getType($node);
|
|
|
|
|
2020-03-14 06:09:12 +01:00
|
|
|
return ($node_type && $node_type->reference_free) || isset($node->pure);
|
2019-11-25 17:44:54 +01:00
|
|
|
}
|
|
|
|
|
2020-09-07 01:36:47 +02:00
|
|
|
public function clearNodeOfTypeAndAssertions(PhpParser\Node\Expr $node) : void
|
2019-11-25 17:44:54 +01:00
|
|
|
{
|
2019-11-26 23:48:47 +01:00
|
|
|
unset($this->node_types[$node], $this->node_assertions[$node]);
|
2019-11-25 17:44:54 +01:00
|
|
|
}
|
|
|
|
}
|