forked from GNUsocial/gnu-social
		
	
		
			
				
	
	
		
			247 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			247 lines
		
	
	
		
			8.7 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
namespace PhpParser;
 | 
						|
 | 
						|
class LexerTest extends \PHPUnit_Framework_TestCase
 | 
						|
{
 | 
						|
    /* To allow overwriting in parent class */
 | 
						|
    protected function getLexer(array $options = array()) {
 | 
						|
        return new Lexer($options);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * @dataProvider provideTestError
 | 
						|
     */
 | 
						|
    public function testError($code, $message) {
 | 
						|
        if (defined('HHVM_VERSION')) {
 | 
						|
            $this->markTestSkipped('HHVM does not throw warnings from token_get_all()');
 | 
						|
        }
 | 
						|
 | 
						|
        $lexer = $this->getLexer();
 | 
						|
        try {
 | 
						|
            $lexer->startLexing($code);
 | 
						|
        } catch (Error $e) {
 | 
						|
            $this->assertSame($message, $e->getMessage());
 | 
						|
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        $this->fail('Expected PhpParser\Error');
 | 
						|
    }
 | 
						|
 | 
						|
    public function provideTestError() {
 | 
						|
        return array(
 | 
						|
            array('<?php /*', 'Unterminated comment on line 1'),
 | 
						|
            array('<?php ' . "\1", 'Unexpected character "' . "\1" . '" (ASCII 1) on unknown line'),
 | 
						|
            array('<?php ' . "\0", 'Unexpected null byte on unknown line'),
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * @dataProvider provideTestLex
 | 
						|
     */
 | 
						|
    public function testLex($code, $options, $tokens) {
 | 
						|
        $lexer = $this->getLexer($options);
 | 
						|
        $lexer->startLexing($code);
 | 
						|
        while ($id = $lexer->getNextToken($value, $startAttributes, $endAttributes)) {
 | 
						|
            $token = array_shift($tokens);
 | 
						|
 | 
						|
            $this->assertSame($token[0], $id);
 | 
						|
            $this->assertSame($token[1], $value);
 | 
						|
            $this->assertEquals($token[2], $startAttributes);
 | 
						|
            $this->assertEquals($token[3], $endAttributes);
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    public function provideTestLex() {
 | 
						|
        return array(
 | 
						|
            // tests conversion of closing PHP tag and drop of whitespace and opening tags
 | 
						|
            array(
 | 
						|
                '<?php tokens ?>plaintext',
 | 
						|
                array(),
 | 
						|
                array(
 | 
						|
                    array(
 | 
						|
                        Parser::T_STRING, 'tokens',
 | 
						|
                        array('startLine' => 1), array('endLine' => 1)
 | 
						|
                    ),
 | 
						|
                    array(
 | 
						|
                        ord(';'), '?>',
 | 
						|
                        array('startLine' => 1), array('endLine' => 1)
 | 
						|
                    ),
 | 
						|
                    array(
 | 
						|
                        Parser::T_INLINE_HTML, 'plaintext',
 | 
						|
                        array('startLine' => 1), array('endLine' => 1)
 | 
						|
                    ),
 | 
						|
                )
 | 
						|
            ),
 | 
						|
            // tests line numbers
 | 
						|
            array(
 | 
						|
                '<?php' . "\n" . '$ token /** doc' . "\n" . 'comment */ $',
 | 
						|
                array(),
 | 
						|
                array(
 | 
						|
                    array(
 | 
						|
                        ord('$'), '$',
 | 
						|
                        array('startLine' => 2), array('endLine' => 2)
 | 
						|
                    ),
 | 
						|
                    array(
 | 
						|
                        Parser::T_STRING, 'token',
 | 
						|
                        array('startLine' => 2), array('endLine' => 2)
 | 
						|
                    ),
 | 
						|
                    array(
 | 
						|
                        ord('$'), '$',
 | 
						|
                        array(
 | 
						|
                            'startLine' => 3,
 | 
						|
                            'comments' => array(new Comment\Doc('/** doc' . "\n" . 'comment */', 2))
 | 
						|
                        ),
 | 
						|
                        array('endLine' => 3)
 | 
						|
                    ),
 | 
						|
                )
 | 
						|
            ),
 | 
						|
            // tests comment extraction
 | 
						|
            array(
 | 
						|
                '<?php /* comment */ // comment' . "\n" . '/** docComment 1 *//** docComment 2 */ token',
 | 
						|
                array(),
 | 
						|
                array(
 | 
						|
                    array(
 | 
						|
                        Parser::T_STRING, 'token',
 | 
						|
                        array(
 | 
						|
                            'startLine' => 2,
 | 
						|
                            'comments' => array(
 | 
						|
                                new Comment('/* comment */', 1),
 | 
						|
                                new Comment('// comment' . "\n", 1),
 | 
						|
                                new Comment\Doc('/** docComment 1 */', 2),
 | 
						|
                                new Comment\Doc('/** docComment 2 */', 2),
 | 
						|
                            ),
 | 
						|
                        ),
 | 
						|
                        array('endLine' => 2)
 | 
						|
                    ),
 | 
						|
                )
 | 
						|
            ),
 | 
						|
            // tests differing start and end line
 | 
						|
            array(
 | 
						|
                '<?php "foo' . "\n" . 'bar"',
 | 
						|
                array(),
 | 
						|
                array(
 | 
						|
                    array(
 | 
						|
                        Parser::T_CONSTANT_ENCAPSED_STRING, '"foo' . "\n" . 'bar"',
 | 
						|
                        array('startLine' => 1), array('endLine' => 2)
 | 
						|
                    ),
 | 
						|
                )
 | 
						|
            ),
 | 
						|
            // tests exact file offsets
 | 
						|
            array(
 | 
						|
                '<?php "a";' . "\n" . '// foo' . "\n" . '"b";',
 | 
						|
                array('usedAttributes' => array('startFilePos', 'endFilePos')),
 | 
						|
                array(
 | 
						|
                    array(
 | 
						|
                        Parser::T_CONSTANT_ENCAPSED_STRING, '"a"',
 | 
						|
                        array('startFilePos' => 6), array('endFilePos' => 8)
 | 
						|
                    ),
 | 
						|
                    array(
 | 
						|
                        ord(';'), ';',
 | 
						|
                        array('startFilePos' => 9), array('endFilePos' => 9)
 | 
						|
                    ),
 | 
						|
                    array(
 | 
						|
                        Parser::T_CONSTANT_ENCAPSED_STRING, '"b"',
 | 
						|
                        array('startFilePos' => 18), array('endFilePos' => 20)
 | 
						|
                    ),
 | 
						|
                    array(
 | 
						|
                        ord(';'), ';',
 | 
						|
                        array('startFilePos' => 21), array('endFilePos' => 21)
 | 
						|
                    ),
 | 
						|
                )
 | 
						|
            ),
 | 
						|
            // tests token offsets
 | 
						|
            array(
 | 
						|
                '<?php "a";' . "\n" . '// foo' . "\n" . '"b";',
 | 
						|
                array('usedAttributes' => array('startTokenPos', 'endTokenPos')),
 | 
						|
                array(
 | 
						|
                    array(
 | 
						|
                        Parser::T_CONSTANT_ENCAPSED_STRING, '"a"',
 | 
						|
                        array('startTokenPos' => 1), array('endTokenPos' => 1)
 | 
						|
                    ),
 | 
						|
                    array(
 | 
						|
                        ord(';'), ';',
 | 
						|
                        array('startTokenPos' => 2), array('endTokenPos' => 2)
 | 
						|
                    ),
 | 
						|
                    array(
 | 
						|
                        Parser::T_CONSTANT_ENCAPSED_STRING, '"b"',
 | 
						|
                        array('startTokenPos' => 5), array('endTokenPos' => 5)
 | 
						|
                    ),
 | 
						|
                    array(
 | 
						|
                        ord(';'), ';',
 | 
						|
                        array('startTokenPos' => 6), array('endTokenPos' => 6)
 | 
						|
                    ),
 | 
						|
                )
 | 
						|
            ),
 | 
						|
            // tests all attributes being disabled
 | 
						|
            array(
 | 
						|
                '<?php /* foo */ $bar;',
 | 
						|
                array('usedAttributes' => array()),
 | 
						|
                array(
 | 
						|
                    array(
 | 
						|
                        Parser::T_VARIABLE, '$bar',
 | 
						|
                        array(), array()
 | 
						|
                    ),
 | 
						|
                    array(
 | 
						|
                        ord(';'), ';',
 | 
						|
                        array(), array()
 | 
						|
                    )
 | 
						|
                )
 | 
						|
            )
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * @dataProvider provideTestHaltCompiler
 | 
						|
     */
 | 
						|
    public function testHandleHaltCompiler($code, $remaining) {
 | 
						|
        $lexer = $this->getLexer();
 | 
						|
        $lexer->startLexing($code);
 | 
						|
 | 
						|
        while (Parser::T_HALT_COMPILER !== $lexer->getNextToken());
 | 
						|
 | 
						|
        $this->assertSame($remaining, $lexer->handleHaltCompiler());
 | 
						|
        $this->assertSame(0, $lexer->getNextToken());
 | 
						|
    }
 | 
						|
 | 
						|
    public function provideTestHaltCompiler() {
 | 
						|
        return array(
 | 
						|
            array('<?php ... __halt_compiler();Remaining Text', 'Remaining Text'),
 | 
						|
            array('<?php ... __halt_compiler ( ) ;Remaining Text', 'Remaining Text'),
 | 
						|
            array('<?php ... __halt_compiler() ?>Remaining Text', 'Remaining Text'),
 | 
						|
            //array('<?php ... __halt_compiler();' . "\0", "\0"),
 | 
						|
            //array('<?php ... __halt_compiler /* */ ( ) ;Remaining Text', 'Remaining Text'),
 | 
						|
        );
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * @expectedException \PhpParser\Error
 | 
						|
     * @expectedExceptionMessage __HALT_COMPILER must be followed by "();"
 | 
						|
     */
 | 
						|
    public function testHandleHaltCompilerError() {
 | 
						|
        $lexer = $this->getLexer();
 | 
						|
        $lexer->startLexing('<?php ... __halt_compiler invalid ();');
 | 
						|
 | 
						|
        while (Parser::T_HALT_COMPILER !== $lexer->getNextToken());
 | 
						|
        $lexer->handleHaltCompiler();
 | 
						|
    }
 | 
						|
 | 
						|
    public function testGetTokens() {
 | 
						|
        $code = '<?php "a";' . "\n" . '// foo' . "\n" . '"b";';
 | 
						|
        $expectedTokens = array(
 | 
						|
            array(T_OPEN_TAG, '<?php ', 1),
 | 
						|
            array(T_CONSTANT_ENCAPSED_STRING, '"a"', 1),
 | 
						|
            ';',
 | 
						|
            array(T_WHITESPACE, "\n", 1),
 | 
						|
            array(T_COMMENT, '// foo' . "\n", 2),
 | 
						|
            array(T_CONSTANT_ENCAPSED_STRING, '"b"', 3),
 | 
						|
            ';',
 | 
						|
        );
 | 
						|
 | 
						|
        $lexer = $this->getLexer();
 | 
						|
        $lexer->startLexing($code);
 | 
						|
        $this->assertSame($expectedTokens, $lexer->getTokens());
 | 
						|
    }
 | 
						|
}
 |