mirror of
https://github.com/danog/PHP-Parser.git
synced 2024-12-15 02:47:04 +01:00
dd711f2a04
Now two arrays are fetched from the lexer: $startAttributes and $endAttributes. When constructing the attributes for a node, the $startAttributes from the first token of the node and the $endAttributes of the last token of the node are merged. Now the end line is saved in the endLine attribute.
214 lines
5.8 KiB
PHP
214 lines
5.8 KiB
PHP
<?php
|
|
|
|
class PHPParser_Tests_NodeVisitor_NameResolverTest extends PHPUnit_Framework_TestCase
|
|
{
|
|
/**
|
|
* @covers PHPParser_NodeVisitor_NameResolver
|
|
*/
|
|
public function testResolveNames() {
|
|
$code = <<<EOC
|
|
<?php
|
|
|
|
namespace Foo {
|
|
use Hallo as Hi;
|
|
|
|
new Bar();
|
|
new Hi();
|
|
new Hi\\Bar();
|
|
new \\Bar();
|
|
new namespace\\Bar();
|
|
|
|
bar();
|
|
hi();
|
|
Hi\\bar();
|
|
foo\\bar();
|
|
\\bar();
|
|
namespace\\bar();
|
|
}
|
|
namespace {
|
|
use Hallo as Hi;
|
|
|
|
new Bar();
|
|
new Hi();
|
|
new Hi\\Bar();
|
|
new \\Bar();
|
|
new namespace\\Bar();
|
|
|
|
bar();
|
|
hi();
|
|
Hi\\bar();
|
|
foo\\bar();
|
|
\\bar();
|
|
namespace\\bar();
|
|
}
|
|
EOC;
|
|
$expectedCode = <<<EOC
|
|
namespace Foo {
|
|
use Hallo as Hi;
|
|
new \\Foo\\Bar();
|
|
new \\Hallo();
|
|
new \\Hallo\\Bar();
|
|
new \\Bar();
|
|
new \\Foo\\Bar();
|
|
bar();
|
|
hi();
|
|
\\Hallo\\bar();
|
|
\\Foo\\foo\\bar();
|
|
\\bar();
|
|
\\Foo\\bar();
|
|
}
|
|
namespace {
|
|
use Hallo as Hi;
|
|
new \\Bar();
|
|
new \\Hallo();
|
|
new \\Hallo\\Bar();
|
|
new \\Bar();
|
|
new \\Bar();
|
|
bar();
|
|
hi();
|
|
\\Hallo\\bar();
|
|
\\foo\\bar();
|
|
\\bar();
|
|
\\bar();
|
|
}
|
|
EOC;
|
|
|
|
$parser = new PHPParser_Parser(new PHPParser_Lexer_Emulative);
|
|
$prettyPrinter = new PHPParser_PrettyPrinter_Zend;
|
|
$traverser = new PHPParser_NodeTraverser;
|
|
$traverser->addVisitor(new PHPParser_NodeVisitor_NameResolver);
|
|
|
|
$stmts = $parser->parse($code);
|
|
$stmts = $traverser->traverse($stmts);
|
|
|
|
$this->assertEquals($expectedCode, $prettyPrinter->prettyPrint($stmts));
|
|
}
|
|
|
|
/**
|
|
* @covers PHPParser_NodeVisitor_NameResolver
|
|
*/
|
|
public function testResolveLocations() {
|
|
$code = <<<EOC
|
|
<?php
|
|
namespace NS {
|
|
class A extends B implements C {
|
|
use A;
|
|
}
|
|
|
|
interface A extends C {
|
|
public function a(A \$a);
|
|
}
|
|
|
|
A::b();
|
|
A::\$b;
|
|
A::B;
|
|
new A;
|
|
\$a instanceof A;
|
|
|
|
namespace\a();
|
|
namespace\A;
|
|
}
|
|
EOC;
|
|
$expectedCode = <<<EOC
|
|
namespace NS {
|
|
class A extends \\NS\\B implements \\NS\\C
|
|
{
|
|
use \\NS\\A;
|
|
}
|
|
interface A extends \\NS\\C
|
|
{
|
|
public function a(\\NS\\A \$a);
|
|
}
|
|
\\NS\\A::b();
|
|
\\NS\\A::\$b;
|
|
\\NS\\A::B;
|
|
new \\NS\\A();
|
|
\$a instanceof \\NS\\A;
|
|
\\NS\\a();
|
|
\\NS\\A;
|
|
}
|
|
EOC;
|
|
|
|
$parser = new PHPParser_Parser(new PHPParser_Lexer_Emulative);
|
|
$prettyPrinter = new PHPParser_PrettyPrinter_Zend;
|
|
$traverser = new PHPParser_NodeTraverser;
|
|
$traverser->addVisitor(new PHPParser_NodeVisitor_NameResolver);
|
|
|
|
$stmts = $parser->parse($code);
|
|
$stmts = $traverser->traverse($stmts);
|
|
|
|
$this->assertEquals($expectedCode, $prettyPrinter->prettyPrint($stmts));
|
|
}
|
|
|
|
public function testNoResolveSpecialName() {
|
|
$stmts = array(new PHPParser_Node_Expr_New(new PHPParser_Node_Name('self')));
|
|
|
|
$traverser = new PHPParser_NodeTraverser;
|
|
$traverser->addVisitor(new PHPParser_NodeVisitor_NameResolver);
|
|
|
|
$this->assertEquals($stmts, $traverser->traverse($stmts));
|
|
}
|
|
|
|
protected function createNamespacedAndNonNamespaced(array $stmts) {
|
|
return array(
|
|
new PHPParser_Node_Stmt_Namespace(new PHPParser_Node_Name('NS'), $stmts),
|
|
new PHPParser_Node_Stmt_Namespace(null, $stmts),
|
|
);
|
|
}
|
|
|
|
public function testAddNamespacedName() {
|
|
$stmts = $this->createNamespacedAndNonNamespaced(array(
|
|
new PHPParser_Node_Stmt_Class('A'),
|
|
new PHPParser_Node_Stmt_Interface('B'),
|
|
new PHPParser_Node_Stmt_Function('C'),
|
|
new PHPParser_Node_Stmt_Const(array(
|
|
new PHPParser_Node_Const('D', new PHPParser_Node_Scalar_String('E'))
|
|
)),
|
|
));
|
|
|
|
$traverser = new PHPParser_NodeTraverser;
|
|
$traverser->addVisitor(new PHPParser_NodeVisitor_NameResolver);
|
|
|
|
$stmts = $traverser->traverse($stmts);
|
|
|
|
$this->assertEquals('NS\\A', (string) $stmts[0]->stmts[0]->namespacedName);
|
|
$this->assertEquals('NS\\B', (string) $stmts[0]->stmts[1]->namespacedName);
|
|
$this->assertEquals('NS\\C', (string) $stmts[0]->stmts[2]->namespacedName);
|
|
$this->assertEquals('NS\\D', (string) $stmts[0]->stmts[3]->consts[0]->namespacedName);
|
|
$this->assertEquals('A', (string) $stmts[1]->stmts[0]->namespacedName);
|
|
$this->assertEquals('B', (string) $stmts[1]->stmts[1]->namespacedName);
|
|
$this->assertEquals('C', (string) $stmts[1]->stmts[2]->namespacedName);
|
|
$this->assertEquals('D', (string) $stmts[1]->stmts[3]->consts[0]->namespacedName);
|
|
}
|
|
|
|
public function testAddTraitNamespacedName() {
|
|
$stmts = $this->createNamespacedAndNonNamespaced(array(
|
|
new PHPParser_Node_Stmt_Trait('A')
|
|
));
|
|
|
|
$traverser = new PHPParser_NodeTraverser;
|
|
$traverser->addVisitor(new PHPParser_NodeVisitor_NameResolver);
|
|
|
|
$stmts = $traverser->traverse($stmts);
|
|
|
|
$this->assertEquals('NS\\A', (string) $stmts[0]->stmts[0]->namespacedName);
|
|
$this->assertEquals('A', (string) $stmts[1]->stmts[0]->namespacedName);
|
|
}
|
|
|
|
/**
|
|
* @expectedException PHPParser_Error
|
|
* @expectedExceptionMessage Cannot use "C" as "B" because the name is already in use on line 2
|
|
*/
|
|
public function testAlreadyInUseError() {
|
|
$stmts = array(
|
|
new PHPParser_Node_Stmt_Use(array(
|
|
new PHPParser_Node_Stmt_UseUse(new PHPParser_Node_Name('A\B'), 'B', array('startLine' => 1)),
|
|
new PHPParser_Node_Stmt_UseUse(new PHPParser_Node_Name('C'), 'B', array('startLine' => 2)),
|
|
))
|
|
);
|
|
|
|
$traverser = new PHPParser_NodeTraverser;
|
|
$traverser->addVisitor(new PHPParser_NodeVisitor_NameResolver);
|
|
$traverser->traverse($stmts);
|
|
}
|
|
} |