2012-05-11 16:18:14 +02:00
|
|
|
<?php
|
|
|
|
|
2014-02-06 14:44:16 +01:00
|
|
|
namespace PhpParser;
|
2012-05-11 16:18:14 +02:00
|
|
|
|
2014-12-19 00:36:44 +01:00
|
|
|
use PhpParser\Comment;
|
|
|
|
|
2014-02-06 14:44:16 +01:00
|
|
|
require_once __DIR__ . '/CodeTestAbstract.php';
|
|
|
|
|
|
|
|
class ParserTest extends CodeTestAbstract
|
2012-05-11 16:18:14 +02:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* @dataProvider provideTestParse
|
|
|
|
*/
|
|
|
|
public function testParse($name, $code, $dump) {
|
2014-02-06 14:44:16 +01:00
|
|
|
$parser = new Parser(new Lexer\Emulative);
|
|
|
|
$dumper = new NodeDumper;
|
2012-05-11 16:18:14 +02:00
|
|
|
|
|
|
|
$stmts = $parser->parse($code);
|
2014-09-30 20:38:09 +02:00
|
|
|
$this->assertSame(
|
2012-05-11 16:18:14 +02:00
|
|
|
$this->canonicalize($dump),
|
|
|
|
$this->canonicalize($dumper->dump($stmts)),
|
|
|
|
$name
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideTestParse() {
|
2014-02-06 14:44:16 +01:00
|
|
|
return $this->getTests(__DIR__ . '/../code/parser', 'test');
|
2012-05-11 16:18:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @dataProvider provideTestParseFail
|
|
|
|
*/
|
|
|
|
public function testParseFail($name, $code, $msg) {
|
2014-02-06 14:44:16 +01:00
|
|
|
$parser = new Parser(new Lexer\Emulative);
|
2012-05-11 16:18:14 +02:00
|
|
|
|
|
|
|
try {
|
|
|
|
$parser->parse($code);
|
|
|
|
|
2014-02-06 14:44:16 +01:00
|
|
|
$this->fail(sprintf('"%s": Expected Error', $name));
|
|
|
|
} catch (Error $e) {
|
2014-09-30 20:38:09 +02:00
|
|
|
$this->assertSame($msg, $e->getMessage(), $name);
|
2012-05-11 16:18:14 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideTestParseFail() {
|
2014-02-06 14:44:16 +01:00
|
|
|
return $this->getTests(__DIR__ . '/../code/parser', 'test-fail');
|
2012-05-11 16:18:14 +02:00
|
|
|
}
|
2014-12-19 00:36:44 +01:00
|
|
|
|
|
|
|
public function testAttributeAssignment() {
|
|
|
|
$lexer = new Lexer(array(
|
|
|
|
'usedAttributes' => array(
|
|
|
|
'comments', 'startLine', 'endLine',
|
|
|
|
'startTokenPos', 'endTokenPos',
|
|
|
|
)
|
|
|
|
));
|
|
|
|
|
|
|
|
$code = <<<'EOC'
|
|
|
|
<?php
|
|
|
|
/** Doc comment */
|
|
|
|
function test($a) {
|
|
|
|
// Line
|
|
|
|
// Comments
|
|
|
|
echo $a;
|
|
|
|
}
|
|
|
|
EOC;
|
2014-12-19 17:27:53 +01:00
|
|
|
$code = $this->canonicalize($code);
|
2014-12-19 00:36:44 +01:00
|
|
|
|
|
|
|
$parser = new Parser($lexer);
|
|
|
|
$stmts = $parser->parse($code);
|
|
|
|
|
|
|
|
/** @var \PhpParser\Node\Stmt\Function_ $fn */
|
|
|
|
$fn = $stmts[0];
|
|
|
|
$this->assertInstanceOf('PhpParser\Node\Stmt\Function_', $fn);
|
|
|
|
$this->assertEquals(array(
|
|
|
|
'comments' => array(
|
|
|
|
new Comment\Doc('/** Doc comment */', 2),
|
|
|
|
),
|
|
|
|
'startLine' => 3,
|
|
|
|
'endLine' => 7,
|
|
|
|
'startTokenPos' => 3,
|
|
|
|
'endTokenPos' => 21,
|
|
|
|
), $fn->getAttributes());
|
|
|
|
|
|
|
|
$param = $fn->params[0];
|
|
|
|
$this->assertInstanceOf('PhpParser\Node\Param', $param);
|
|
|
|
$this->assertEquals(array(
|
|
|
|
'startLine' => 3,
|
|
|
|
'endLine' => 3,
|
|
|
|
'startTokenPos' => 7,
|
|
|
|
'endTokenPos' => 7,
|
|
|
|
), $param->getAttributes());
|
|
|
|
|
|
|
|
/** @var \PhpParser\Node\Stmt\Echo_ $echo */
|
|
|
|
$echo = $fn->stmts[0];
|
|
|
|
$this->assertInstanceOf('PhpParser\Node\Stmt\Echo_', $echo);
|
|
|
|
$this->assertEquals(array(
|
|
|
|
'comments' => array(
|
|
|
|
new Comment("// Line\n", 4),
|
|
|
|
new Comment("// Comments\n", 5),
|
|
|
|
),
|
|
|
|
'startLine' => 6,
|
|
|
|
'endLine' => 6,
|
|
|
|
'startTokenPos' => 16,
|
|
|
|
'endTokenPos' => 19,
|
|
|
|
), $echo->getAttributes());
|
|
|
|
|
|
|
|
/** @var \PhpParser\Node\Expr\Variable $var */
|
|
|
|
$var = $echo->exprs[0];
|
|
|
|
$this->assertInstanceOf('PhpParser\Node\Expr\Variable', $var);
|
|
|
|
$this->assertEquals(array(
|
|
|
|
'startLine' => 6,
|
|
|
|
'endLine' => 6,
|
|
|
|
'startTokenPos' => 18,
|
|
|
|
'endTokenPos' => 18,
|
|
|
|
), $var->getAttributes());
|
|
|
|
}
|
2015-01-17 23:46:28 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @expectedException \RangeException
|
|
|
|
* @expectedExceptionMessage The lexer returned an invalid token (id=999, value=foobar)
|
|
|
|
*/
|
|
|
|
public function testInvalidToken() {
|
|
|
|
$lexer = new InvalidTokenLexer;
|
|
|
|
$parser = new Parser($lexer);
|
|
|
|
$parser->parse('dummy');
|
|
|
|
}
|
2015-02-26 16:26:29 +01:00
|
|
|
|
|
|
|
public function testInvalidOctals() {
|
|
|
|
if (version_compare(PHP_VERSION, '7.0-dev', '>=')) {
|
|
|
|
$this->markTestSkipped('Cannot parse invalid octal numbers on PHP 7');
|
|
|
|
}
|
|
|
|
|
|
|
|
$parser = new Parser(new Lexer);
|
|
|
|
$stmts = $parser->parse('<?php 0787; 0177777777777777777777787;');
|
|
|
|
$this->assertInstanceof($stmts[0], 'PhpParser\Scalar\LNumber');
|
|
|
|
$this->assertInstanceof($stmts[1], 'PhpParser\Scalar\DNumber');
|
|
|
|
$this->assertSame(7, $stmts[0]->value);
|
|
|
|
$this->assertSame(0xFFFFFFFFFFFFFFFF, $stmts[0]->value);
|
|
|
|
}
|
2015-01-17 23:46:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
class InvalidTokenLexer extends Lexer {
|
|
|
|
public function getNextToken(&$value = null, &$startAttributes = null, &$endAttributes = null) {
|
|
|
|
$value = 'foobar';
|
|
|
|
return 999;
|
|
|
|
}
|
|
|
|
}
|