1
0
mirror of https://github.com/danog/PHP-Parser.git synced 2025-01-10 06:58:22 +01:00
PHP-Parser/test/PhpParser/NodeAbstractTest.php

325 lines
9.7 KiB
PHP
Raw Normal View History

<?php declare(strict_types=1);
2011-11-27 21:43:27 +01:00
namespace PhpParser;
2017-04-27 18:14:07 +02:00
use PHPUnit\Framework\TestCase;
class DummyNode extends NodeAbstract {
public $subNode1;
public $subNode2;
public function __construct($subNode1, $subNode2, $attributes) {
2015-05-02 22:17:34 +02:00
parent::__construct($attributes);
$this->subNode1 = $subNode1;
$this->subNode2 = $subNode2;
}
2017-04-28 21:40:59 +02:00
public function getSubNodeNames() : array {
2017-08-13 14:35:03 +02:00
return ['subNode1', 'subNode2'];
}
// This method is only overwritten because the node is located in an unusual namespace
2017-04-28 21:40:59 +02:00
public function getType() : string {
return 'Dummy';
}
}
2017-04-27 18:14:07 +02:00
class NodeAbstractTest extends TestCase
2011-11-27 21:43:27 +01:00
{
public function provideNodes() {
2017-08-13 14:35:03 +02:00
$attributes = [
'startLine' => 10,
'endLine' => 11,
'startTokenPos' => 12,
'endTokenPos' => 13,
'startFilePos' => 14,
'endFilePos' => 15,
2017-08-13 14:35:03 +02:00
'comments' => [
new Comment('// Comment' . "\n"),
new Comment\Doc('/** doc comment */'),
2017-08-13 14:35:03 +02:00
],
];
2015-05-02 22:17:34 +02:00
$node = new DummyNode('value1', 'value2', $attributes);
$node->notSubNode = 'value3';
2017-08-13 14:35:03 +02:00
return [
[$attributes, $node],
];
}
2011-11-27 21:43:27 +01:00
/**
* @dataProvider provideNodes
*/
public function testConstruct(array $attributes, Node $node) {
2014-09-30 20:38:09 +02:00
$this->assertSame('Dummy', $node->getType());
2017-08-13 14:35:03 +02:00
$this->assertSame(['subNode1', 'subNode2'], $node->getSubNodeNames());
2014-09-30 20:38:09 +02:00
$this->assertSame(10, $node->getLine());
$this->assertSame(10, $node->getStartLine());
$this->assertSame(11, $node->getEndLine());
$this->assertSame(12, $node->getStartTokenPos());
$this->assertSame(13, $node->getEndTokenPos());
$this->assertSame(14, $node->getStartFilePos());
$this->assertSame(15, $node->getEndFilePos());
2014-09-30 20:38:09 +02:00
$this->assertSame('/** doc comment */', $node->getDocComment()->getText());
$this->assertSame('value1', $node->subNode1);
$this->assertSame('value2', $node->subNode2);
$this->assertTrue(isset($node->subNode1));
$this->assertTrue(isset($node->subNode2));
$this->assertFalse(isset($node->subNode3));
2014-09-30 20:38:09 +02:00
$this->assertSame($attributes, $node->getAttributes());
2017-09-29 17:09:16 +02:00
$this->assertSame($attributes['comments'], $node->getComments());
2011-11-27 21:43:27 +01:00
return $node;
}
/**
* @dataProvider provideNodes
*/
public function testGetDocComment(array $attributes, Node $node) {
2014-09-30 20:38:09 +02:00
$this->assertSame('/** doc comment */', $node->getDocComment()->getText());
$comments = $node->getComments();
array_pop($comments); // remove doc comment
$node->setAttribute('comments', $comments);
$this->assertNull($node->getDocComment());
array_pop($comments); // remove comment
$node->setAttribute('comments', $comments);
$this->assertNull($node->getDocComment());
}
2016-09-17 20:48:51 +02:00
public function testSetDocComment() {
$node = new DummyNode(null, null, []);
// Add doc comment to node without comments
$docComment = new Comment\Doc('/** doc */');
$node->setDocComment($docComment);
$this->assertSame($docComment, $node->getDocComment());
// Replace it
$docComment = new Comment\Doc('/** doc 2 */');
$node->setDocComment($docComment);
$this->assertSame($docComment, $node->getDocComment());
// Add docmment to node with other comments
$c1 = new Comment('/* foo */');
$c2 = new Comment('/* bar */');
$docComment = new Comment\Doc('/** baz */');
$node->setAttribute('comments', [$c1, $c2]);
$node->setDocComment($docComment);
$this->assertSame([$c1, $c2, $docComment], $node->getAttribute('comments'));
}
2011-11-27 21:43:27 +01:00
/**
* @dataProvider provideNodes
2011-11-27 21:43:27 +01:00
*/
public function testChange(array $attributes, Node $node) {
// direct modification
2011-11-27 21:43:27 +01:00
$node->subNode = 'newValue';
2014-09-30 20:38:09 +02:00
$this->assertSame('newValue', $node->subNode);
2011-11-27 21:43:27 +01:00
// indirect modification
$subNode =& $node->subNode;
$subNode = 'newNewValue';
2014-09-30 20:38:09 +02:00
$this->assertSame('newNewValue', $node->subNode);
// removal
2011-11-27 21:43:27 +01:00
unset($node->subNode);
$this->assertFalse(isset($node->subNode));
}
/**
* @dataProvider provideNodes
*/
public function testIteration(array $attributes, Node $node) {
// Iteration is simple object iteration over properties,
// not over subnodes
$i = 0;
foreach ($node as $key => $value) {
if ($i === 0) {
$this->assertSame('subNode1', $key);
$this->assertSame('value1', $value);
} else if ($i === 1) {
$this->assertSame('subNode2', $key);
$this->assertSame('value2', $value);
} else if ($i === 2) {
$this->assertSame('notSubNode', $key);
$this->assertSame('value3', $value);
} else {
throw new \Exception;
}
$i++;
}
$this->assertSame(3, $i);
}
public function testAttributes() {
/** @var $node Node */
$node = $this->getMockForAbstractClass('PhpParser\NodeAbstract');
$this->assertEmpty($node->getAttributes());
$node->setAttribute('key', 'value');
$this->assertTrue($node->hasAttribute('key'));
2014-09-30 20:38:09 +02:00
$this->assertSame('value', $node->getAttribute('key'));
$this->assertFalse($node->hasAttribute('doesNotExist'));
$this->assertNull($node->getAttribute('doesNotExist'));
2014-09-30 20:38:09 +02:00
$this->assertSame('default', $node->getAttribute('doesNotExist', 'default'));
$node->setAttribute('null', null);
$this->assertTrue($node->hasAttribute('null'));
$this->assertNull($node->getAttribute('null'));
$this->assertNull($node->getAttribute('null', 'default'));
2014-09-30 20:38:09 +02:00
$this->assertSame(
2017-08-13 14:35:03 +02:00
[
'key' => 'value',
'null' => null,
2017-08-13 14:35:03 +02:00
],
$node->getAttributes()
);
2017-05-05 18:18:44 +02:00
$node->setAttributes(
2017-08-13 14:35:03 +02:00
[
2017-05-05 18:18:44 +02:00
'a' => 'b',
'c' => null,
2017-08-13 14:35:03 +02:00
]
2017-05-05 18:18:44 +02:00
);
$this->assertSame(
2017-08-13 14:35:03 +02:00
[
2017-05-05 18:18:44 +02:00
'a' => 'b',
'c' => null,
2017-08-13 14:35:03 +02:00
],
2017-05-05 18:18:44 +02:00
$node->getAttributes()
);
}
public function testJsonSerialization() {
$code = <<<'PHP'
<?php
// comment
/** doc comment */
function functionName(&$a = 0, $b = 1.0) {
echo 'Foo';
}
PHP;
$expected = <<<'JSON'
[
{
"nodeType": "Stmt_Function",
"byRef": false,
"name": {
"nodeType": "Identifier",
"name": "functionName",
"attributes": {
"startLine": 4,
"endLine": 4
}
},
"params": [
{
"nodeType": "Param",
"type": null,
"byRef": true,
"variadic": false,
"var": {
2017-01-19 23:23:38 +01:00
"nodeType": "Expr_Variable",
"name": "a",
"attributes": {
"startLine": 4,
"endLine": 4
}
},
"default": {
"nodeType": "Scalar_LNumber",
"value": 0,
"attributes": {
"startLine": 4,
"endLine": 4,
"kind": 10
}
},
"attributes": {
"startLine": 4,
"endLine": 4
}
},
{
"nodeType": "Param",
"type": null,
"byRef": false,
"variadic": false,
"var": {
2017-01-19 23:23:38 +01:00
"nodeType": "Expr_Variable",
"name": "b",
"attributes": {
"startLine": 4,
"endLine": 4
}
},
"default": {
"nodeType": "Scalar_DNumber",
"value": 1,
"attributes": {
"startLine": 4,
"endLine": 4
}
},
"attributes": {
"startLine": 4,
"endLine": 4
}
}
],
"returnType": null,
"stmts": [
{
"nodeType": "Stmt_Echo",
"exprs": [
{
"nodeType": "Scalar_String",
"value": "Foo",
"attributes": {
"startLine": 5,
"endLine": 5,
"kind": 1
}
}
],
"attributes": {
"startLine": 5,
"endLine": 5
}
}
],
"attributes": {
"startLine": 4,
"comments": [
{
"nodeType": "Comment",
2016-07-25 21:02:53 +02:00
"text": "\/\/ comment\n",
"line": 2,
2016-07-25 21:02:53 +02:00
"filePos": 6
},
{
"nodeType": "Comment_Doc",
"text": "\/** doc comment *\/",
"line": 3,
2016-07-25 21:02:53 +02:00
"filePos": 17
}
],
"endLine": 6
}
}
]
JSON;
$parser = new Parser\Php7(new Lexer());
2016-07-25 21:02:53 +02:00
$stmts = $parser->parse(canonicalize($code));
$json = json_encode($stmts, JSON_PRETTY_PRINT);
$this->assertEquals(canonicalize($expected), canonicalize($json));
}
}