annotate vendor/nikic/php-parser/test/PhpParser/LexerTest.php @ 2:92f882872392

Trusted hosts, + remove migration modules
author Chris Cannam
date Tue, 05 Dec 2017 09:26:43 +0000
parents 4c8ae668cc8c
children 5fb285c0d0e3
rev   line source
Chris@0 1 <?php
Chris@0 2
Chris@0 3 namespace PhpParser;
Chris@0 4
Chris@0 5 use PhpParser\Parser\Tokens;
Chris@0 6
Chris@0 7 class LexerTest extends \PHPUnit_Framework_TestCase
Chris@0 8 {
Chris@0 9 /* To allow overwriting in parent class */
Chris@0 10 protected function getLexer(array $options = array()) {
Chris@0 11 return new Lexer($options);
Chris@0 12 }
Chris@0 13
Chris@0 14 /**
Chris@0 15 * @dataProvider provideTestError
Chris@0 16 */
Chris@0 17 public function testError($code, $messages) {
Chris@0 18 if (defined('HHVM_VERSION')) {
Chris@0 19 $this->markTestSkipped('HHVM does not throw warnings from token_get_all()');
Chris@0 20 }
Chris@0 21
Chris@0 22 $errorHandler = new ErrorHandler\Collecting();
Chris@0 23 $lexer = $this->getLexer(['usedAttributes' => [
Chris@0 24 'comments', 'startLine', 'endLine', 'startFilePos', 'endFilePos'
Chris@0 25 ]]);
Chris@0 26 $lexer->startLexing($code, $errorHandler);
Chris@0 27 $errors = $errorHandler->getErrors();
Chris@0 28
Chris@0 29 $this->assertSame(count($messages), count($errors));
Chris@0 30 for ($i = 0; $i < count($messages); $i++) {
Chris@0 31 $this->assertSame($messages[$i], $errors[$i]->getMessageWithColumnInfo($code));
Chris@0 32 }
Chris@0 33 }
Chris@0 34
Chris@0 35 public function provideTestError() {
Chris@0 36 return array(
Chris@0 37 array("<?php /*", array("Unterminated comment from 1:7 to 1:9")),
Chris@0 38 array("<?php \1", array("Unexpected character \"\1\" (ASCII 1) from 1:7 to 1:7")),
Chris@0 39 array("<?php \0", array("Unexpected null byte from 1:7 to 1:7")),
Chris@0 40 // Error with potentially emulated token
Chris@0 41 array("<?php ?? \0", array("Unexpected null byte from 1:10 to 1:10")),
Chris@0 42 array("<?php\n\0\1 foo /* bar", array(
Chris@0 43 "Unexpected null byte from 2:1 to 2:1",
Chris@0 44 "Unexpected character \"\1\" (ASCII 1) from 2:2 to 2:2",
Chris@0 45 "Unterminated comment from 2:8 to 2:14"
Chris@0 46 )),
Chris@0 47 );
Chris@0 48 }
Chris@0 49
Chris@0 50 /**
Chris@0 51 * @dataProvider provideTestLex
Chris@0 52 */
Chris@0 53 public function testLex($code, $options, $tokens) {
Chris@0 54 $lexer = $this->getLexer($options);
Chris@0 55 $lexer->startLexing($code);
Chris@0 56 while ($id = $lexer->getNextToken($value, $startAttributes, $endAttributes)) {
Chris@0 57 $token = array_shift($tokens);
Chris@0 58
Chris@0 59 $this->assertSame($token[0], $id);
Chris@0 60 $this->assertSame($token[1], $value);
Chris@0 61 $this->assertEquals($token[2], $startAttributes);
Chris@0 62 $this->assertEquals($token[3], $endAttributes);
Chris@0 63 }
Chris@0 64 }
Chris@0 65
Chris@0 66 public function provideTestLex() {
Chris@0 67 return array(
Chris@0 68 // tests conversion of closing PHP tag and drop of whitespace and opening tags
Chris@0 69 array(
Chris@0 70 '<?php tokens ?>plaintext',
Chris@0 71 array(),
Chris@0 72 array(
Chris@0 73 array(
Chris@0 74 Tokens::T_STRING, 'tokens',
Chris@0 75 array('startLine' => 1), array('endLine' => 1)
Chris@0 76 ),
Chris@0 77 array(
Chris@0 78 ord(';'), '?>',
Chris@0 79 array('startLine' => 1), array('endLine' => 1)
Chris@0 80 ),
Chris@0 81 array(
Chris@0 82 Tokens::T_INLINE_HTML, 'plaintext',
Chris@0 83 array('startLine' => 1, 'hasLeadingNewline' => false),
Chris@0 84 array('endLine' => 1)
Chris@0 85 ),
Chris@0 86 )
Chris@0 87 ),
Chris@0 88 // tests line numbers
Chris@0 89 array(
Chris@0 90 '<?php' . "\n" . '$ token /** doc' . "\n" . 'comment */ $',
Chris@0 91 array(),
Chris@0 92 array(
Chris@0 93 array(
Chris@0 94 ord('$'), '$',
Chris@0 95 array('startLine' => 2), array('endLine' => 2)
Chris@0 96 ),
Chris@0 97 array(
Chris@0 98 Tokens::T_STRING, 'token',
Chris@0 99 array('startLine' => 2), array('endLine' => 2)
Chris@0 100 ),
Chris@0 101 array(
Chris@0 102 ord('$'), '$',
Chris@0 103 array(
Chris@0 104 'startLine' => 3,
Chris@0 105 'comments' => array(
Chris@0 106 new Comment\Doc('/** doc' . "\n" . 'comment */', 2, 14),
Chris@0 107 )
Chris@0 108 ),
Chris@0 109 array('endLine' => 3)
Chris@0 110 ),
Chris@0 111 )
Chris@0 112 ),
Chris@0 113 // tests comment extraction
Chris@0 114 array(
Chris@0 115 '<?php /* comment */ // comment' . "\n" . '/** docComment 1 *//** docComment 2 */ token',
Chris@0 116 array(),
Chris@0 117 array(
Chris@0 118 array(
Chris@0 119 Tokens::T_STRING, 'token',
Chris@0 120 array(
Chris@0 121 'startLine' => 2,
Chris@0 122 'comments' => array(
Chris@0 123 new Comment('/* comment */', 1, 6),
Chris@0 124 new Comment('// comment' . "\n", 1, 20),
Chris@0 125 new Comment\Doc('/** docComment 1 */', 2, 31),
Chris@0 126 new Comment\Doc('/** docComment 2 */', 2, 50),
Chris@0 127 ),
Chris@0 128 ),
Chris@0 129 array('endLine' => 2)
Chris@0 130 ),
Chris@0 131 )
Chris@0 132 ),
Chris@0 133 // tests differing start and end line
Chris@0 134 array(
Chris@0 135 '<?php "foo' . "\n" . 'bar"',
Chris@0 136 array(),
Chris@0 137 array(
Chris@0 138 array(
Chris@0 139 Tokens::T_CONSTANT_ENCAPSED_STRING, '"foo' . "\n" . 'bar"',
Chris@0 140 array('startLine' => 1), array('endLine' => 2)
Chris@0 141 ),
Chris@0 142 )
Chris@0 143 ),
Chris@0 144 // tests exact file offsets
Chris@0 145 array(
Chris@0 146 '<?php "a";' . "\n" . '// foo' . "\n" . '"b";',
Chris@0 147 array('usedAttributes' => array('startFilePos', 'endFilePos')),
Chris@0 148 array(
Chris@0 149 array(
Chris@0 150 Tokens::T_CONSTANT_ENCAPSED_STRING, '"a"',
Chris@0 151 array('startFilePos' => 6), array('endFilePos' => 8)
Chris@0 152 ),
Chris@0 153 array(
Chris@0 154 ord(';'), ';',
Chris@0 155 array('startFilePos' => 9), array('endFilePos' => 9)
Chris@0 156 ),
Chris@0 157 array(
Chris@0 158 Tokens::T_CONSTANT_ENCAPSED_STRING, '"b"',
Chris@0 159 array('startFilePos' => 18), array('endFilePos' => 20)
Chris@0 160 ),
Chris@0 161 array(
Chris@0 162 ord(';'), ';',
Chris@0 163 array('startFilePos' => 21), array('endFilePos' => 21)
Chris@0 164 ),
Chris@0 165 )
Chris@0 166 ),
Chris@0 167 // tests token offsets
Chris@0 168 array(
Chris@0 169 '<?php "a";' . "\n" . '// foo' . "\n" . '"b";',
Chris@0 170 array('usedAttributes' => array('startTokenPos', 'endTokenPos')),
Chris@0 171 array(
Chris@0 172 array(
Chris@0 173 Tokens::T_CONSTANT_ENCAPSED_STRING, '"a"',
Chris@0 174 array('startTokenPos' => 1), array('endTokenPos' => 1)
Chris@0 175 ),
Chris@0 176 array(
Chris@0 177 ord(';'), ';',
Chris@0 178 array('startTokenPos' => 2), array('endTokenPos' => 2)
Chris@0 179 ),
Chris@0 180 array(
Chris@0 181 Tokens::T_CONSTANT_ENCAPSED_STRING, '"b"',
Chris@0 182 array('startTokenPos' => 5), array('endTokenPos' => 5)
Chris@0 183 ),
Chris@0 184 array(
Chris@0 185 ord(';'), ';',
Chris@0 186 array('startTokenPos' => 6), array('endTokenPos' => 6)
Chris@0 187 ),
Chris@0 188 )
Chris@0 189 ),
Chris@0 190 // tests all attributes being disabled
Chris@0 191 array(
Chris@0 192 '<?php /* foo */ $bar;',
Chris@0 193 array('usedAttributes' => array()),
Chris@0 194 array(
Chris@0 195 array(
Chris@0 196 Tokens::T_VARIABLE, '$bar',
Chris@0 197 array(), array()
Chris@0 198 ),
Chris@0 199 array(
Chris@0 200 ord(';'), ';',
Chris@0 201 array(), array()
Chris@0 202 )
Chris@0 203 )
Chris@0 204 ),
Chris@0 205 // tests no tokens
Chris@0 206 array(
Chris@0 207 '',
Chris@0 208 array(),
Chris@0 209 array()
Chris@0 210 ),
Chris@0 211 );
Chris@0 212 }
Chris@0 213
Chris@0 214 /**
Chris@0 215 * @dataProvider provideTestHaltCompiler
Chris@0 216 */
Chris@0 217 public function testHandleHaltCompiler($code, $remaining) {
Chris@0 218 $lexer = $this->getLexer();
Chris@0 219 $lexer->startLexing($code);
Chris@0 220
Chris@0 221 while (Tokens::T_HALT_COMPILER !== $lexer->getNextToken());
Chris@0 222
Chris@0 223 $this->assertSame($remaining, $lexer->handleHaltCompiler());
Chris@0 224 $this->assertSame(0, $lexer->getNextToken());
Chris@0 225 }
Chris@0 226
Chris@0 227 public function provideTestHaltCompiler() {
Chris@0 228 return array(
Chris@0 229 array('<?php ... __halt_compiler();Remaining Text', 'Remaining Text'),
Chris@0 230 array('<?php ... __halt_compiler ( ) ;Remaining Text', 'Remaining Text'),
Chris@0 231 array('<?php ... __halt_compiler() ?>Remaining Text', 'Remaining Text'),
Chris@0 232 //array('<?php ... __halt_compiler();' . "\0", "\0"),
Chris@0 233 //array('<?php ... __halt_compiler /* */ ( ) ;Remaining Text', 'Remaining Text'),
Chris@0 234 );
Chris@0 235 }
Chris@0 236
Chris@0 237 /**
Chris@0 238 * @expectedException \PhpParser\Error
Chris@0 239 * @expectedExceptionMessage __HALT_COMPILER must be followed by "();"
Chris@0 240 */
Chris@0 241 public function testHandleHaltCompilerError() {
Chris@0 242 $lexer = $this->getLexer();
Chris@0 243 $lexer->startLexing('<?php ... __halt_compiler invalid ();');
Chris@0 244
Chris@0 245 while (Tokens::T_HALT_COMPILER !== $lexer->getNextToken());
Chris@0 246 $lexer->handleHaltCompiler();
Chris@0 247 }
Chris@0 248
Chris@0 249 public function testGetTokens() {
Chris@0 250 $code = '<?php "a";' . "\n" . '// foo' . "\n" . '"b";';
Chris@0 251 $expectedTokens = array(
Chris@0 252 array(T_OPEN_TAG, '<?php ', 1),
Chris@0 253 array(T_CONSTANT_ENCAPSED_STRING, '"a"', 1),
Chris@0 254 ';',
Chris@0 255 array(T_WHITESPACE, "\n", 1),
Chris@0 256 array(T_COMMENT, '// foo' . "\n", 2),
Chris@0 257 array(T_CONSTANT_ENCAPSED_STRING, '"b"', 3),
Chris@0 258 ';',
Chris@0 259 );
Chris@0 260
Chris@0 261 $lexer = $this->getLexer();
Chris@0 262 $lexer->startLexing($code);
Chris@0 263 $this->assertSame($expectedTokens, $lexer->getTokens());
Chris@0 264 }
Chris@0 265 }