2018-10-26 06:59:14 +02:00
|
|
|
<?php declare(strict_types=1);
|
|
|
|
|
2018-11-06 03:57:36 +01:00
|
|
|
namespace Psalm\Internal\Traverser;
|
2018-10-26 06:59:14 +02:00
|
|
|
|
|
|
|
use PhpParser\Node;
|
|
|
|
use PhpParser\NodeTraverserInterface;
|
2019-06-26 22:52:29 +02:00
|
|
|
use function gettype;
|
|
|
|
use function array_pop;
|
|
|
|
use function array_splice;
|
2018-10-26 06:59:14 +02:00
|
|
|
|
2018-12-02 00:37:49 +01:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2018-10-26 06:59:14 +02:00
|
|
|
class CustomTraverser extends \PhpParser\NodeTraverser implements NodeTraverserInterface
|
|
|
|
{
|
|
|
|
public function __construct()
|
|
|
|
{
|
|
|
|
$this->stopTraversal = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Recursively traverse a node.
|
|
|
|
*
|
|
|
|
* @param Node $node Node to traverse.
|
|
|
|
*
|
|
|
|
* @return Node Result of traversal (may be original node or new one)
|
|
|
|
*/
|
|
|
|
protected function traverseNode(Node $node) : Node
|
|
|
|
{
|
|
|
|
foreach ($node->getSubNodeNames() as $name) {
|
|
|
|
$subNode =& $node->$name;
|
|
|
|
|
|
|
|
if (\is_array($subNode)) {
|
|
|
|
$subNode = $this->traverseArray($subNode);
|
|
|
|
if ($this->stopTraversal) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} elseif ($subNode instanceof Node) {
|
|
|
|
$traverseChildren = true;
|
|
|
|
foreach ($this->visitors as $visitor) {
|
|
|
|
$return = $visitor->enterNode($subNode, $traverseChildren);
|
|
|
|
if (null !== $return) {
|
|
|
|
if ($return instanceof Node) {
|
|
|
|
$subNode = $return;
|
|
|
|
} elseif (self::DONT_TRAVERSE_CHILDREN === $return) {
|
|
|
|
$traverseChildren = false;
|
|
|
|
} elseif (self::STOP_TRAVERSAL === $return) {
|
|
|
|
$this->stopTraversal = true;
|
|
|
|
break 2;
|
|
|
|
} else {
|
|
|
|
throw new \LogicException(
|
|
|
|
'enterNode() returned invalid value of type ' . gettype($return)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($traverseChildren) {
|
|
|
|
$subNode = $this->traverseNode($subNode);
|
|
|
|
if ($this->stopTraversal) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($this->visitors as $visitor) {
|
|
|
|
$return = $visitor->leaveNode($subNode);
|
|
|
|
if (null !== $return) {
|
|
|
|
if ($return instanceof Node) {
|
|
|
|
$subNode = $return;
|
|
|
|
} elseif (self::STOP_TRAVERSAL === $return) {
|
|
|
|
$this->stopTraversal = true;
|
|
|
|
break 2;
|
|
|
|
} elseif (\is_array($return)) {
|
|
|
|
throw new \LogicException(
|
|
|
|
'leaveNode() may only return an array ' .
|
|
|
|
'if the parent structure is an array'
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
throw new \LogicException(
|
|
|
|
'leaveNode() returned invalid value of type ' . gettype($return)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $node;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Recursively traverse array (usually of nodes).
|
|
|
|
*
|
|
|
|
* @param array $nodes Array to traverse
|
|
|
|
*
|
|
|
|
* @return array Result of traversal (may be original array or changed one)
|
|
|
|
*/
|
|
|
|
protected function traverseArray(array $nodes) : array
|
|
|
|
{
|
|
|
|
$doNodes = [];
|
|
|
|
|
|
|
|
foreach ($nodes as $i => &$node) {
|
|
|
|
if ($node instanceof Node) {
|
|
|
|
$traverseChildren = true;
|
|
|
|
foreach ($this->visitors as $visitor) {
|
|
|
|
$return = $visitor->enterNode($node, $traverseChildren);
|
|
|
|
if (null !== $return) {
|
|
|
|
if ($return instanceof Node) {
|
|
|
|
$node = $return;
|
|
|
|
} elseif (self::DONT_TRAVERSE_CHILDREN === $return) {
|
|
|
|
$traverseChildren = false;
|
|
|
|
} elseif (self::STOP_TRAVERSAL === $return) {
|
|
|
|
$this->stopTraversal = true;
|
|
|
|
break 2;
|
|
|
|
} else {
|
|
|
|
throw new \LogicException(
|
|
|
|
'enterNode() returned invalid value of type ' . gettype($return)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($traverseChildren) {
|
|
|
|
$node = $this->traverseNode($node);
|
|
|
|
if ($this->stopTraversal) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($this->visitors as $visitor) {
|
|
|
|
$return = $visitor->leaveNode($node);
|
|
|
|
if (null !== $return) {
|
|
|
|
if ($return instanceof Node) {
|
|
|
|
$node = $return;
|
|
|
|
} elseif (\is_array($return)) {
|
|
|
|
$doNodes[] = [$i, $return];
|
|
|
|
break;
|
|
|
|
} elseif (self::REMOVE_NODE === $return) {
|
|
|
|
$doNodes[] = [$i, []];
|
|
|
|
break;
|
|
|
|
} elseif (self::STOP_TRAVERSAL === $return) {
|
|
|
|
$this->stopTraversal = true;
|
|
|
|
break 2;
|
|
|
|
} elseif (false === $return) {
|
|
|
|
throw new \LogicException(
|
|
|
|
'bool(false) return from leaveNode() no longer supported. ' .
|
|
|
|
'Return NodeTraverser::REMOVE_NODE instead'
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
throw new \LogicException(
|
|
|
|
'leaveNode() returned invalid value of type ' . gettype($return)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} elseif (\is_array($node)) {
|
|
|
|
throw new \LogicException('Invalid node structure: Contains nested arrays');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!empty($doNodes)) {
|
|
|
|
while (list($i, $replace) = array_pop($doNodes)) {
|
|
|
|
array_splice($nodes, $i, 1, $replace);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $nodes;
|
|
|
|
}
|
|
|
|
}
|