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;
|
2016-04-02 22:22:24 +09:00
|
|
|
use PhpParser\Node\Expr;
|
|
|
|
use PhpParser\Node\Scalar;
|
|
|
|
use PhpParser\Node\Scalar\String_;
|
2017-04-27 19:14:07 +03:00
|
|
|
use PHPUnit\Framework\TestCase;
|
2014-12-19 00:36:44 +01:00
|
|
|
|
2017-04-27 19:14:07 +03: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() {
|
|
|
|
$lexer = new Lexer(array(
|
|
|
|
'usedAttributes' => array(
|
|
|
|
'comments', 'startLine', 'endLine',
|
|
|
|
'startTokenPos', 'endTokenPos',
|
|
|
|
)
|
|
|
|
));
|
|
|
|
|
|
|
|
$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);
|
|
|
|
$this->assertEquals(array(
|
|
|
|
'comments' => array(
|
2016-04-02 07:54:01 +09:00
|
|
|
new Comment\Doc('/** Doc comment */', 2, 6),
|
2014-12-19 00:36:44 +01:00
|
|
|
),
|
|
|
|
'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(
|
2016-04-02 07:54:01 +09:00
|
|
|
new Comment("// Line\n", 4, 49),
|
|
|
|
new Comment("// Comments\n", 5, 61),
|
2014-12-19 00:36:44 +01:00
|
|
|
),
|
|
|
|
'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;
|
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 22:22:24 +09:00
|
|
|
|
|
|
|
/**
|
2016-07-25 16:42:42 +02:00
|
|
|
* @dataProvider provideTestExtraAttributes
|
2016-04-02 22:22:24 +09:00
|
|
|
*/
|
2016-07-25 16:42:42 +02:00
|
|
|
public function testExtraAttributes($code, $expectedAttributes) {
|
2016-04-02 22:22:24 +09: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 22:22:24 +09:00
|
|
|
foreach ($expectedAttributes as $name => $value) {
|
|
|
|
$this->assertSame($value, $attributes[$name]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-25 16:42:42 +02:00
|
|
|
public function provideTestExtraAttributes() {
|
2016-04-02 22:22:24 +09:00
|
|
|
return array(
|
|
|
|
array('0', ['kind' => Scalar\LNumber::KIND_DEC]),
|
|
|
|
array('9', ['kind' => Scalar\LNumber::KIND_DEC]),
|
|
|
|
array('07', ['kind' => Scalar\LNumber::KIND_OCT]),
|
|
|
|
array('0xf', ['kind' => Scalar\LNumber::KIND_HEX]),
|
|
|
|
array('0XF', ['kind' => Scalar\LNumber::KIND_HEX]),
|
|
|
|
array('0b1', ['kind' => Scalar\LNumber::KIND_BIN]),
|
|
|
|
array('0B1', ['kind' => Scalar\LNumber::KIND_BIN]),
|
|
|
|
array('[]', ['kind' => Expr\Array_::KIND_SHORT]),
|
|
|
|
array('array()', ['kind' => Expr\Array_::KIND_LONG]),
|
|
|
|
array("'foo'", ['kind' => String_::KIND_SINGLE_QUOTED]),
|
|
|
|
array("b'foo'", ['kind' => String_::KIND_SINGLE_QUOTED]),
|
|
|
|
array("B'foo'", ['kind' => String_::KIND_SINGLE_QUOTED]),
|
|
|
|
array('"foo"', ['kind' => String_::KIND_DOUBLE_QUOTED]),
|
|
|
|
array('b"foo"', ['kind' => String_::KIND_DOUBLE_QUOTED]),
|
|
|
|
array('B"foo"', ['kind' => String_::KIND_DOUBLE_QUOTED]),
|
|
|
|
array('"foo$bar"', ['kind' => String_::KIND_DOUBLE_QUOTED]),
|
|
|
|
array('b"foo$bar"', ['kind' => String_::KIND_DOUBLE_QUOTED]),
|
|
|
|
array('B"foo$bar"', ['kind' => String_::KIND_DOUBLE_QUOTED]),
|
|
|
|
array("<<<'STR'\nSTR\n", ['kind' => String_::KIND_NOWDOC, 'docLabel' => 'STR']),
|
|
|
|
array("<<<STR\nSTR\n", ['kind' => String_::KIND_HEREDOC, 'docLabel' => 'STR']),
|
|
|
|
array("<<<\"STR\"\nSTR\n", ['kind' => String_::KIND_HEREDOC, 'docLabel' => 'STR']),
|
|
|
|
array("b<<<'STR'\nSTR\n", ['kind' => String_::KIND_NOWDOC, 'docLabel' => 'STR']),
|
|
|
|
array("B<<<'STR'\nSTR\n", ['kind' => String_::KIND_NOWDOC, 'docLabel' => 'STR']),
|
|
|
|
array("<<< \t 'STR'\nSTR\n", ['kind' => String_::KIND_NOWDOC, 'docLabel' => 'STR']),
|
2016-04-07 12:27:08 +09: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"]),
|
2016-04-02 22:22:24 +09:00
|
|
|
array("<<<\"STR\"\n\$a\nSTR\n", ['kind' => String_::KIND_HEREDOC, 'docLabel' => 'STR']),
|
|
|
|
array("b<<<\"STR\"\n\$a\nSTR\n", ['kind' => String_::KIND_HEREDOC, 'docLabel' => 'STR']),
|
|
|
|
array("B<<<\"STR\"\n\$a\nSTR\n", ['kind' => String_::KIND_HEREDOC, 'docLabel' => 'STR']),
|
|
|
|
array("<<< \t \"STR\"\n\$a\nSTR\n", ['kind' => String_::KIND_HEREDOC, 'docLabel' => 'STR']),
|
2016-04-19 14:49:18 +02:00
|
|
|
array("die", ['kind' => Expr\Exit_::KIND_DIE]),
|
|
|
|
array("die('done')", ['kind' => Expr\Exit_::KIND_DIE]),
|
|
|
|
array("exit", ['kind' => Expr\Exit_::KIND_EXIT]),
|
|
|
|
array("exit(1)", ['kind' => Expr\Exit_::KIND_EXIT]),
|
2016-07-25 16:42:42 +02:00
|
|
|
array("?>Foo", ['hasLeadingNewline' => false]),
|
|
|
|
array("?>\nFoo", ['hasLeadingNewline' => true]),
|
2016-04-02 22:22:24 +09: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;
|
|
|
|
}
|
|
|
|
}
|