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
|
|
|
|
2016-04-02 15:22:24 +02:00
|
|
|
use PhpParser\Node\Expr;
|
|
|
|
use PhpParser\Node\Scalar;
|
|
|
|
use PhpParser\Node\Scalar\String_;
|
2017-04-27 18:14:07 +02:00
|
|
|
use PHPUnit\Framework\TestCase;
|
2014-12-19 00:36:44 +01:00
|
|
|
|
2017-04-27 18:14:07 +02:00
|
|
|
abstract class ParserTest extends TestCase
|
2012-05-11 16:18:14 +02:00
|
|
|
{
|
2015-06-20 11:47:25 +02:00
|
|
|
/** @returns Parser */
|
2015-06-20 11:43:16 +02:00
|
|
|
abstract protected function getParser(Lexer $lexer);
|
2012-05-11 16:18:14 +02:00
|
|
|
|
2015-05-01 20:17:39 +02:00
|
|
|
/**
|
|
|
|
* @expectedException \PhpParser\Error
|
|
|
|
* @expectedExceptionMessage Syntax error, unexpected EOF on line 1
|
|
|
|
*/
|
|
|
|
public function testParserThrowsSyntaxError() {
|
2015-06-20 11:43:16 +02:00
|
|
|
$parser = $this->getParser(new Lexer());
|
2015-05-01 20:17:39 +02:00
|
|
|
$parser->parse('<?php foo');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @expectedException \PhpParser\Error
|
|
|
|
* @expectedExceptionMessage Cannot use foo as self because 'self' is a special class name on line 1
|
|
|
|
*/
|
|
|
|
public function testParserThrowsSpecialError() {
|
2015-06-20 11:43:16 +02:00
|
|
|
$parser = $this->getParser(new Lexer());
|
2015-05-01 20:17:39 +02:00
|
|
|
$parser->parse('<?php use foo as self;');
|
|
|
|
}
|
|
|
|
|
2016-09-30 18:28:35 +02:00
|
|
|
/**
|
|
|
|
* @expectedException \PhpParser\Error
|
|
|
|
* @expectedExceptionMessage Unterminated comment on line 1
|
|
|
|
*/
|
|
|
|
public function testParserThrowsLexerError() {
|
|
|
|
$parser = $this->getParser(new Lexer());
|
|
|
|
$parser->parse('<?php /*');
|
|
|
|
}
|
|
|
|
|
2014-12-19 00:36:44 +01:00
|
|
|
public function testAttributeAssignment() {
|
2017-08-13 14:35:03 +02:00
|
|
|
$lexer = new Lexer([
|
|
|
|
'usedAttributes' => [
|
2014-12-19 00:36:44 +01:00
|
|
|
'comments', 'startLine', 'endLine',
|
|
|
|
'startTokenPos', 'endTokenPos',
|
2017-08-13 14:35:03 +02:00
|
|
|
]
|
|
|
|
]);
|
2014-12-19 00:36:44 +01:00
|
|
|
|
|
|
|
$code = <<<'EOC'
|
|
|
|
<?php
|
|
|
|
/** Doc comment */
|
|
|
|
function test($a) {
|
|
|
|
// Line
|
|
|
|
// Comments
|
|
|
|
echo $a;
|
|
|
|
}
|
|
|
|
EOC;
|
2015-06-20 11:43:16 +02:00
|
|
|
$code = canonicalize($code);
|
2014-12-19 00:36:44 +01:00
|
|
|
|
2015-06-20 11:43:16 +02:00
|
|
|
$parser = $this->getParser($lexer);
|
2014-12-19 00:36:44 +01:00
|
|
|
$stmts = $parser->parse($code);
|
|
|
|
|
|
|
|
/** @var \PhpParser\Node\Stmt\Function_ $fn */
|
|
|
|
$fn = $stmts[0];
|
|
|
|
$this->assertInstanceOf('PhpParser\Node\Stmt\Function_', $fn);
|
2017-08-13 14:35:03 +02:00
|
|
|
$this->assertEquals([
|
|
|
|
'comments' => [
|
2016-04-02 00:54:01 +02:00
|
|
|
new Comment\Doc('/** Doc comment */', 2, 6),
|
2017-08-13 14:35:03 +02:00
|
|
|
],
|
2014-12-19 00:36:44 +01:00
|
|
|
'startLine' => 3,
|
|
|
|
'endLine' => 7,
|
|
|
|
'startTokenPos' => 3,
|
|
|
|
'endTokenPos' => 21,
|
2017-08-13 14:35:03 +02:00
|
|
|
], $fn->getAttributes());
|
2014-12-19 00:36:44 +01:00
|
|
|
|
|
|
|
$param = $fn->params[0];
|
|
|
|
$this->assertInstanceOf('PhpParser\Node\Param', $param);
|
2017-08-13 14:35:03 +02:00
|
|
|
$this->assertEquals([
|
2014-12-19 00:36:44 +01:00
|
|
|
'startLine' => 3,
|
|
|
|
'endLine' => 3,
|
|
|
|
'startTokenPos' => 7,
|
|
|
|
'endTokenPos' => 7,
|
2017-08-13 14:35:03 +02:00
|
|
|
], $param->getAttributes());
|
2014-12-19 00:36:44 +01:00
|
|
|
|
|
|
|
/** @var \PhpParser\Node\Stmt\Echo_ $echo */
|
|
|
|
$echo = $fn->stmts[0];
|
|
|
|
$this->assertInstanceOf('PhpParser\Node\Stmt\Echo_', $echo);
|
2017-08-13 14:35:03 +02:00
|
|
|
$this->assertEquals([
|
|
|
|
'comments' => [
|
2016-04-02 00:54:01 +02:00
|
|
|
new Comment("// Line\n", 4, 49),
|
|
|
|
new Comment("// Comments\n", 5, 61),
|
2017-08-13 14:35:03 +02:00
|
|
|
],
|
2014-12-19 00:36:44 +01:00
|
|
|
'startLine' => 6,
|
|
|
|
'endLine' => 6,
|
|
|
|
'startTokenPos' => 16,
|
|
|
|
'endTokenPos' => 19,
|
2017-08-13 14:35:03 +02:00
|
|
|
], $echo->getAttributes());
|
2014-12-19 00:36:44 +01:00
|
|
|
|
|
|
|
/** @var \PhpParser\Node\Expr\Variable $var */
|
|
|
|
$var = $echo->exprs[0];
|
|
|
|
$this->assertInstanceOf('PhpParser\Node\Expr\Variable', $var);
|
2017-08-13 14:35:03 +02:00
|
|
|
$this->assertEquals([
|
2014-12-19 00:36:44 +01:00
|
|
|
'startLine' => 6,
|
|
|
|
'endLine' => 6,
|
|
|
|
'startTokenPos' => 18,
|
|
|
|
'endTokenPos' => 18,
|
2017-08-13 14:35:03 +02:00
|
|
|
], $var->getAttributes());
|
2014-12-19 00:36:44 +01:00
|
|
|
}
|
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;
|
2015-06-20 11:43:16 +02:00
|
|
|
$parser = $this->getParser($lexer);
|
2015-01-17 23:46:28 +01:00
|
|
|
$parser->parse('dummy');
|
|
|
|
}
|
2016-04-02 15:22:24 +02:00
|
|
|
|
|
|
|
/**
|
2016-07-25 16:42:42 +02:00
|
|
|
* @dataProvider provideTestExtraAttributes
|
2016-04-02 15:22:24 +02:00
|
|
|
*/
|
2016-07-25 16:42:42 +02:00
|
|
|
public function testExtraAttributes($code, $expectedAttributes) {
|
2016-04-02 15:22:24 +02:00
|
|
|
$parser = $this->getParser(new Lexer);
|
|
|
|
$stmts = $parser->parse("<?php $code;");
|
2017-01-19 21:15:26 +01:00
|
|
|
$node = $stmts[0] instanceof Node\Stmt\Expression ? $stmts[0]->expr : $stmts[0];
|
|
|
|
$attributes = $node->getAttributes();
|
2016-04-02 15:22:24 +02:00
|
|
|
foreach ($expectedAttributes as $name => $value) {
|
|
|
|
$this->assertSame($value, $attributes[$name]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-25 16:42:42 +02:00
|
|
|
public function provideTestExtraAttributes() {
|
2017-08-13 14:35:03 +02:00
|
|
|
return [
|
|
|
|
['0', ['kind' => Scalar\LNumber::KIND_DEC]],
|
|
|
|
['9', ['kind' => Scalar\LNumber::KIND_DEC]],
|
|
|
|
['07', ['kind' => Scalar\LNumber::KIND_OCT]],
|
|
|
|
['0xf', ['kind' => Scalar\LNumber::KIND_HEX]],
|
|
|
|
['0XF', ['kind' => Scalar\LNumber::KIND_HEX]],
|
|
|
|
['0b1', ['kind' => Scalar\LNumber::KIND_BIN]],
|
|
|
|
['0B1', ['kind' => Scalar\LNumber::KIND_BIN]],
|
|
|
|
['[]', ['kind' => Expr\Array_::KIND_SHORT]],
|
|
|
|
['array()', ['kind' => Expr\Array_::KIND_LONG]],
|
|
|
|
["'foo'", ['kind' => String_::KIND_SINGLE_QUOTED]],
|
|
|
|
["b'foo'", ['kind' => String_::KIND_SINGLE_QUOTED]],
|
|
|
|
["B'foo'", ['kind' => String_::KIND_SINGLE_QUOTED]],
|
|
|
|
['"foo"', ['kind' => String_::KIND_DOUBLE_QUOTED]],
|
|
|
|
['b"foo"', ['kind' => String_::KIND_DOUBLE_QUOTED]],
|
|
|
|
['B"foo"', ['kind' => String_::KIND_DOUBLE_QUOTED]],
|
|
|
|
['"foo$bar"', ['kind' => String_::KIND_DOUBLE_QUOTED]],
|
|
|
|
['b"foo$bar"', ['kind' => String_::KIND_DOUBLE_QUOTED]],
|
|
|
|
['B"foo$bar"', ['kind' => String_::KIND_DOUBLE_QUOTED]],
|
|
|
|
["<<<'STR'\nSTR\n", ['kind' => String_::KIND_NOWDOC, 'docLabel' => 'STR']],
|
|
|
|
["<<<STR\nSTR\n", ['kind' => String_::KIND_HEREDOC, 'docLabel' => 'STR']],
|
|
|
|
["<<<\"STR\"\nSTR\n", ['kind' => String_::KIND_HEREDOC, 'docLabel' => 'STR']],
|
|
|
|
["b<<<'STR'\nSTR\n", ['kind' => String_::KIND_NOWDOC, 'docLabel' => 'STR']],
|
|
|
|
["B<<<'STR'\nSTR\n", ['kind' => String_::KIND_NOWDOC, 'docLabel' => 'STR']],
|
|
|
|
["<<< \t 'STR'\nSTR\n", ['kind' => String_::KIND_NOWDOC, 'docLabel' => 'STR']],
|
2016-04-07 05:27:08 +02:00
|
|
|
// HHVM doesn't support this due to a lexer bug
|
|
|
|
// (https://github.com/facebook/hhvm/issues/6970)
|
|
|
|
// array("<<<'\xff'\n\xff\n", ['kind' => String_::KIND_NOWDOC, 'docLabel' => "\xff"]),
|
2017-08-13 14:35:03 +02:00
|
|
|
["<<<\"STR\"\n\$a\nSTR\n", ['kind' => String_::KIND_HEREDOC, 'docLabel' => 'STR']],
|
|
|
|
["b<<<\"STR\"\n\$a\nSTR\n", ['kind' => String_::KIND_HEREDOC, 'docLabel' => 'STR']],
|
|
|
|
["B<<<\"STR\"\n\$a\nSTR\n", ['kind' => String_::KIND_HEREDOC, 'docLabel' => 'STR']],
|
|
|
|
["<<< \t \"STR\"\n\$a\nSTR\n", ['kind' => String_::KIND_HEREDOC, 'docLabel' => 'STR']],
|
|
|
|
["die", ['kind' => Expr\Exit_::KIND_DIE]],
|
|
|
|
["die('done')", ['kind' => Expr\Exit_::KIND_DIE]],
|
|
|
|
["exit", ['kind' => Expr\Exit_::KIND_EXIT]],
|
|
|
|
["exit(1)", ['kind' => Expr\Exit_::KIND_EXIT]],
|
|
|
|
["?>Foo", ['hasLeadingNewline' => false]],
|
|
|
|
["?>\nFoo", ['hasLeadingNewline' => true]],
|
2017-09-26 18:46:44 +02:00
|
|
|
["namespace Foo;", ['kind' => Node\Stmt\Namespace_::KIND_SEMICOLON]],
|
|
|
|
["namespace Foo {}", ['kind' => Node\Stmt\Namespace_::KIND_BRACED]],
|
|
|
|
["namespace {}", ['kind' => Node\Stmt\Namespace_::KIND_BRACED]],
|
2017-08-13 14:35:03 +02:00
|
|
|
];
|
2016-04-02 15:22:24 +02:00
|
|
|
}
|
2015-01-17 23:46:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
class InvalidTokenLexer extends Lexer {
|
2017-04-28 21:40:59 +02:00
|
|
|
public function getNextToken(&$value = null, &$startAttributes = null, &$endAttributes = null) : int {
|
2015-01-17 23:46:28 +01:00
|
|
|
$value = 'foobar';
|
|
|
|
return 999;
|
|
|
|
}
|
|
|
|
}
|