mirror of
https://github.com/phabelio/PHP-Parser.git
synced 2025-01-10 06:38:19 +01:00
f66a32e2df
Perform emulation by replacing #[ with %[, then patching % back to # and coalescing #[ into T_ATTRIBUTE if it is a freestanding token.
326 lines
11 KiB
PHP
326 lines
11 KiB
PHP
<?php declare(strict_types=1);
|
|
|
|
namespace PhpParser\Lexer;
|
|
|
|
use PhpParser\ErrorHandler;
|
|
use PhpParser\Lexer;
|
|
use PhpParser\LexerTest;
|
|
use PhpParser\Parser\Tokens;
|
|
|
|
class EmulativeTest extends LexerTest
|
|
{
|
|
protected function getLexer(array $options = []) {
|
|
return new Emulative($options);
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideTestReplaceKeywords
|
|
*/
|
|
public function testReplaceKeywords($keyword, $expectedToken) {
|
|
$lexer = $this->getLexer();
|
|
$lexer->startLexing('<?php ' . $keyword);
|
|
|
|
$this->assertSame($expectedToken, $lexer->getNextToken());
|
|
$this->assertSame(0, $lexer->getNextToken());
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideTestReplaceKeywords
|
|
*/
|
|
public function testNoReplaceKeywordsAfterObjectOperator(string $keyword) {
|
|
$lexer = $this->getLexer();
|
|
$lexer->startLexing('<?php ->' . $keyword);
|
|
|
|
$this->assertSame(Tokens::T_OBJECT_OPERATOR, $lexer->getNextToken());
|
|
$this->assertSame(Tokens::T_STRING, $lexer->getNextToken());
|
|
$this->assertSame(0, $lexer->getNextToken());
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideTestReplaceKeywords
|
|
*/
|
|
public function testNoReplaceKeywordsAfterObjectOperatorWithSpaces(string $keyword) {
|
|
$lexer = $this->getLexer();
|
|
$lexer->startLexing('<?php -> ' . $keyword);
|
|
|
|
$this->assertSame(Tokens::T_OBJECT_OPERATOR, $lexer->getNextToken());
|
|
$this->assertSame(Tokens::T_STRING, $lexer->getNextToken());
|
|
$this->assertSame(0, $lexer->getNextToken());
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideTestReplaceKeywords
|
|
*/
|
|
public function testNoReplaceKeywordsAfterNullsafeObjectOperator(string $keyword) {
|
|
$lexer = $this->getLexer();
|
|
$lexer->startLexing('<?php ?->' . $keyword);
|
|
|
|
$this->assertSame(Tokens::T_NULLSAFE_OBJECT_OPERATOR, $lexer->getNextToken());
|
|
$this->assertSame(Tokens::T_STRING, $lexer->getNextToken());
|
|
$this->assertSame(0, $lexer->getNextToken());
|
|
}
|
|
|
|
public function provideTestReplaceKeywords() {
|
|
return [
|
|
// PHP 8.0
|
|
['match', Tokens::T_MATCH],
|
|
|
|
// PHP 7.4
|
|
['fn', Tokens::T_FN],
|
|
|
|
// PHP 5.5
|
|
['finally', Tokens::T_FINALLY],
|
|
['yield', Tokens::T_YIELD],
|
|
|
|
// PHP 5.4
|
|
['callable', Tokens::T_CALLABLE],
|
|
['insteadof', Tokens::T_INSTEADOF],
|
|
['trait', Tokens::T_TRAIT],
|
|
['__TRAIT__', Tokens::T_TRAIT_C],
|
|
|
|
// PHP 5.3
|
|
['__DIR__', Tokens::T_DIR],
|
|
['goto', Tokens::T_GOTO],
|
|
['namespace', Tokens::T_NAMESPACE],
|
|
['__NAMESPACE__', Tokens::T_NS_C],
|
|
];
|
|
}
|
|
|
|
private function assertSameTokens(array $expectedTokens, Lexer $lexer) {
|
|
$tokens = [];
|
|
while (0 !== $token = $lexer->getNextToken($text)) {
|
|
$tokens[] = [$token, $text];
|
|
}
|
|
$this->assertSame($expectedTokens, $tokens);
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideTestLexNewFeatures
|
|
*/
|
|
public function testLexNewFeatures($code, array $expectedTokens) {
|
|
$lexer = $this->getLexer();
|
|
$lexer->startLexing('<?php ' . $code);
|
|
$this->assertSameTokens($expectedTokens, $lexer);
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideTestLexNewFeatures
|
|
*/
|
|
public function testLeaveStuffAloneInStrings($code) {
|
|
$stringifiedToken = '"' . addcslashes($code, '"\\') . '"';
|
|
|
|
$lexer = $this->getLexer();
|
|
$lexer->startLexing('<?php ' . $stringifiedToken);
|
|
|
|
$this->assertSame(Tokens::T_CONSTANT_ENCAPSED_STRING, $lexer->getNextToken($text));
|
|
$this->assertSame($stringifiedToken, $text);
|
|
$this->assertSame(0, $lexer->getNextToken());
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideTestLexNewFeatures
|
|
*/
|
|
public function testErrorAfterEmulation($code) {
|
|
$errorHandler = new ErrorHandler\Collecting;
|
|
$lexer = $this->getLexer();
|
|
$lexer->startLexing('<?php ' . $code . "\0", $errorHandler);
|
|
|
|
$errors = $errorHandler->getErrors();
|
|
$this->assertCount(1, $errors);
|
|
|
|
$error = $errors[0];
|
|
$this->assertSame('Unexpected null byte', $error->getRawMessage());
|
|
|
|
$attrs = $error->getAttributes();
|
|
$expPos = strlen('<?php ' . $code);
|
|
$expLine = 1 + substr_count('<?php ' . $code, "\n");
|
|
$this->assertSame($expPos, $attrs['startFilePos']);
|
|
$this->assertSame($expPos, $attrs['endFilePos']);
|
|
$this->assertSame($expLine, $attrs['startLine']);
|
|
$this->assertSame($expLine, $attrs['endLine']);
|
|
}
|
|
|
|
public function provideTestLexNewFeatures() {
|
|
return [
|
|
['yield from', [
|
|
[Tokens::T_YIELD_FROM, 'yield from'],
|
|
]],
|
|
["yield\r\nfrom", [
|
|
[Tokens::T_YIELD_FROM, "yield\r\nfrom"],
|
|
]],
|
|
['...', [
|
|
[Tokens::T_ELLIPSIS, '...'],
|
|
]],
|
|
['**', [
|
|
[Tokens::T_POW, '**'],
|
|
]],
|
|
['**=', [
|
|
[Tokens::T_POW_EQUAL, '**='],
|
|
]],
|
|
['??', [
|
|
[Tokens::T_COALESCE, '??'],
|
|
]],
|
|
['<=>', [
|
|
[Tokens::T_SPACESHIP, '<=>'],
|
|
]],
|
|
['0b1010110', [
|
|
[Tokens::T_LNUMBER, '0b1010110'],
|
|
]],
|
|
['0b1011010101001010110101010010101011010101010101101011001110111100', [
|
|
[Tokens::T_DNUMBER, '0b1011010101001010110101010010101011010101010101101011001110111100'],
|
|
]],
|
|
['\\', [
|
|
[Tokens::T_NS_SEPARATOR, '\\'],
|
|
]],
|
|
["<<<'NOWDOC'\nNOWDOC;\n", [
|
|
[Tokens::T_START_HEREDOC, "<<<'NOWDOC'\n"],
|
|
[Tokens::T_END_HEREDOC, 'NOWDOC'],
|
|
[ord(';'), ';'],
|
|
]],
|
|
["<<<'NOWDOC'\nFoobar\nNOWDOC;\n", [
|
|
[Tokens::T_START_HEREDOC, "<<<'NOWDOC'\n"],
|
|
[Tokens::T_ENCAPSED_AND_WHITESPACE, "Foobar\n"],
|
|
[Tokens::T_END_HEREDOC, 'NOWDOC'],
|
|
[ord(';'), ';'],
|
|
]],
|
|
|
|
// PHP 7.3: Flexible heredoc/nowdoc
|
|
["<<<LABEL\nLABEL,", [
|
|
[Tokens::T_START_HEREDOC, "<<<LABEL\n"],
|
|
[Tokens::T_END_HEREDOC, "LABEL"],
|
|
[ord(','), ','],
|
|
]],
|
|
["<<<LABEL\n LABEL,", [
|
|
[Tokens::T_START_HEREDOC, "<<<LABEL\n"],
|
|
[Tokens::T_END_HEREDOC, " LABEL"],
|
|
[ord(','), ','],
|
|
]],
|
|
["<<<LABEL\n Foo\n LABEL;", [
|
|
[Tokens::T_START_HEREDOC, "<<<LABEL\n"],
|
|
[Tokens::T_ENCAPSED_AND_WHITESPACE, " Foo\n"],
|
|
[Tokens::T_END_HEREDOC, " LABEL"],
|
|
[ord(';'), ';'],
|
|
]],
|
|
["<<<A\n A,<<<A\n A,", [
|
|
[Tokens::T_START_HEREDOC, "<<<A\n"],
|
|
[Tokens::T_END_HEREDOC, " A"],
|
|
[ord(','), ','],
|
|
[Tokens::T_START_HEREDOC, "<<<A\n"],
|
|
[Tokens::T_END_HEREDOC, " A"],
|
|
[ord(','), ','],
|
|
]],
|
|
["<<<LABEL\nLABELNOPE\nLABEL\n", [
|
|
[Tokens::T_START_HEREDOC, "<<<LABEL\n"],
|
|
[Tokens::T_ENCAPSED_AND_WHITESPACE, "LABELNOPE\n"],
|
|
[Tokens::T_END_HEREDOC, "LABEL"],
|
|
]],
|
|
// Interpretation changed
|
|
["<<<LABEL\n LABEL\nLABEL\n", [
|
|
[Tokens::T_START_HEREDOC, "<<<LABEL\n"],
|
|
[Tokens::T_END_HEREDOC, " LABEL"],
|
|
[Tokens::T_STRING, "LABEL"],
|
|
]],
|
|
|
|
// PHP 7.4: Null coalesce equal
|
|
['??=', [
|
|
[Tokens::T_COALESCE_EQUAL, '??='],
|
|
]],
|
|
|
|
// PHP 7.4: Number literal separator
|
|
['1_000', [
|
|
[Tokens::T_LNUMBER, '1_000'],
|
|
]],
|
|
['0xCAFE_F00D', [
|
|
[Tokens::T_LNUMBER, '0xCAFE_F00D'],
|
|
]],
|
|
['0b0101_1111', [
|
|
[Tokens::T_LNUMBER, '0b0101_1111'],
|
|
]],
|
|
['0137_041', [
|
|
[Tokens::T_LNUMBER, '0137_041'],
|
|
]],
|
|
['1_000.0', [
|
|
[Tokens::T_DNUMBER, '1_000.0'],
|
|
]],
|
|
['1_0.0', [
|
|
[Tokens::T_DNUMBER, '1_0.0']
|
|
]],
|
|
['1_000_000_000.0', [
|
|
[Tokens::T_DNUMBER, '1_000_000_000.0']
|
|
]],
|
|
['0e1_0', [
|
|
[Tokens::T_DNUMBER, '0e1_0']
|
|
]],
|
|
['1_0e+10', [
|
|
[Tokens::T_DNUMBER, '1_0e+10']
|
|
]],
|
|
['1_0e-10', [
|
|
[Tokens::T_DNUMBER, '1_0e-10']
|
|
]],
|
|
['0b1011010101001010_110101010010_10101101010101_0101101011001_110111100', [
|
|
[Tokens::T_DNUMBER, '0b1011010101001010_110101010010_10101101010101_0101101011001_110111100'],
|
|
]],
|
|
['0xFFFF_FFFF_FFFF_FFFF', [
|
|
[Tokens::T_DNUMBER, '0xFFFF_FFFF_FFFF_FFFF'],
|
|
]],
|
|
['1_000+1', [
|
|
[Tokens::T_LNUMBER, '1_000'],
|
|
[ord('+'), '+'],
|
|
[Tokens::T_LNUMBER, '1'],
|
|
]],
|
|
['1_0abc', [
|
|
[Tokens::T_LNUMBER, '1_0'],
|
|
[Tokens::T_STRING, 'abc'],
|
|
]],
|
|
['?->', [
|
|
[Tokens::T_NULLSAFE_OBJECT_OPERATOR, '?->'],
|
|
]],
|
|
['#[Attr]', [
|
|
[Tokens::T_ATTRIBUTE, '#['],
|
|
[Tokens::T_STRING, 'Attr'],
|
|
[ord(']'), ']'],
|
|
]],
|
|
["#[\nAttr\n]", [
|
|
[Tokens::T_ATTRIBUTE, '#['],
|
|
[Tokens::T_STRING, 'Attr'],
|
|
[ord(']'), ']'],
|
|
]],
|
|
// Test interaction of two patch-based emulators
|
|
["<<<LABEL\n LABEL, #[Attr]", [
|
|
[Tokens::T_START_HEREDOC, "<<<LABEL\n"],
|
|
[Tokens::T_END_HEREDOC, " LABEL"],
|
|
[ord(','), ','],
|
|
[Tokens::T_ATTRIBUTE, '#['],
|
|
[Tokens::T_STRING, 'Attr'],
|
|
[ord(']'), ']'],
|
|
]],
|
|
["#[Attr] <<<LABEL\n LABEL,", [
|
|
[Tokens::T_ATTRIBUTE, '#['],
|
|
[Tokens::T_STRING, 'Attr'],
|
|
[ord(']'), ']'],
|
|
[Tokens::T_START_HEREDOC, "<<<LABEL\n"],
|
|
[Tokens::T_END_HEREDOC, " LABEL"],
|
|
[ord(','), ','],
|
|
]],
|
|
];
|
|
}
|
|
|
|
/**
|
|
* @dataProvider provideTestTargetVersion
|
|
*/
|
|
public function testTargetVersion(string $phpVersion, string $code, array $expectedTokens) {
|
|
$lexer = $this->getLexer(['phpVersion' => $phpVersion]);
|
|
$lexer->startLexing('<?php ' . $code);
|
|
$this->assertSameTokens($expectedTokens, $lexer);
|
|
}
|
|
|
|
public function provideTestTargetVersion() {
|
|
return [
|
|
['8.0', 'match', [[Tokens::T_MATCH, 'match']]],
|
|
['7.4', 'match', [[Tokens::T_STRING, 'match']]],
|
|
['7.4', 'fn', [[Tokens::T_FN, 'fn']]],
|
|
['7.3', 'fn', [[Tokens::T_STRING, 'fn']]],
|
|
];
|
|
}
|
|
}
|