170 lines
5.4 KiB
PHP
Raw Normal View History

2011-04-18 19:02:30 +02:00
<?php
2011-06-05 18:40:04 +02:00
class PHPParser_Lexer
2011-04-18 19:02:30 +02:00
{
2011-06-03 22:02:02 +02:00
protected $code;
2011-04-18 19:02:30 +02:00
protected $tokens;
protected $pos;
protected $line;
2011-04-18 19:02:30 +02:00
2011-07-13 12:24:10 +02:00
protected static $tokenMap;
protected static $dropTokens = array(
T_WHITESPACE => 1, T_COMMENT => 1, T_OPEN_TAG => 1
2011-04-18 19:02:30 +02:00
);
/**
* Creates a Lexer.
*
* @param string $code
2011-07-13 23:07:05 +02:00
*
* @throws PHPParser_Error on lexing errors (unterminated comment or unexpected character)
*/
2011-04-18 19:02:30 +02:00
public function __construct($code) {
self::initTokenMap();
// Reset the error message in error_get_last()
// Still hoping for a better solution to be found.
@$errorGetLastResetUndefinedVariable;
2011-06-03 22:02:02 +02:00
$this->code = $code;
$this->tokens = @token_get_all($code);
2011-04-18 19:02:30 +02:00
$this->pos = -1;
2011-07-13 13:27:14 +02:00
$this->line = 1;
$error = error_get_last();
if (preg_match(
2011-07-13 12:24:10 +02:00
'~^Unterminated comment starting line ([0-9]+)$~',
$error['message'],
$matches
)
) {
2011-07-13 12:24:10 +02:00
throw new PHPParser_Error('Unterminated comment', $matches[1]);
}
if (preg_match(
2011-07-13 12:24:10 +02:00
'~^Unexpected character in input: \'(.)\' \(ASCII=([0-9]+)\)~s',
$error['message'],
$matches
)
) {
2011-07-13 12:24:10 +02:00
throw new PHPParser_Error(sprintf(
'Unexpected character "%s" (ASCII %d)',
$matches[1], $matches[2]
));
}
// PHP cuts error message after null byte, so need special case
if (preg_match('~^Unexpected character in input: \'$~', $error['message'])) {
throw new PHPParser_Error('Unexpected null byte');
}
2011-04-18 19:02:30 +02:00
}
/**
* Returns the next token id.
*
* @param mixed $value Variable to store token content in
* @param mixed $line Variable to store line in
* @param mixed $docComment Variable to store doc comment in
*
* @return int Token id
*/
2011-07-13 12:24:10 +02:00
public function lex(&$value = null, &$line = null, &$docComment = null) {
$docComment = null;
2011-04-18 19:02:30 +02:00
while (isset($this->tokens[++$this->pos])) {
$token = $this->tokens[$this->pos];
2011-06-03 22:02:02 +02:00
2011-04-18 19:02:30 +02:00
if (is_string($token)) {
$line = $this->line;
// bug in token_get_all
if ('b"' === $token) {
$value = 'b"';
return ord('"');
} else {
$value = $token;
return ord($token);
}
2011-07-13 13:27:14 +02:00
} else {
$this->line += substr_count($token[1], "\n");
if (T_DOC_COMMENT === $token[0]) {
$docComment = $token[1];
} elseif (!isset(self::$dropTokens[$token[0]])) {
$value = $token[1];
$line = $token[2];
return self::$tokenMap[$token[0]];
}
2011-04-18 19:02:30 +02:00
}
}
return 0;
}
2011-06-03 22:02:02 +02:00
/**
* Handles __halt_compiler() by returning the text after it.
*
* @return string Remaining text
*/
public function handleHaltCompiler() {
// get the length of the text before the T_HALT_COMPILER token
$textBefore = '';
for ($i = 0; $i <= $this->pos; ++$i) {
if (is_string($this->tokens[$i])) {
$textBefore .= $this->tokens[$i];
} else {
$textBefore .= $this->tokens[$i][1];
}
}
// text after T_HALT_COMPILER, still including ();
$textAfter = substr($this->code, strlen($textBefore));
// ensure that it is followed by ();
// this simplifies the situation, by not allowing any comments
// in between of the tokens.
if (!preg_match('~\s*\(\s*\)\s*(?:;|\?>)~', $textAfter, $matches)) {
throw new PHPParser_Error('__halt_compiler must be followed by "();"');
2011-06-03 22:02:02 +02:00
}
// prevent the lexer from returning any further tokens
$this->pos = count($this->tokens);
// return with (); removed
return substr($textAfter, strlen($matches[0]));
}
/**
* Initializes the token map.
*
* The token map maps the PHP internal token identifiers
* to the identifiers used by the Parser. Additionally it
* maps T_OPEN_TAG_WITH_ECHO to T_ECHO and T_CLOSE_TAG to ';'.
*/
2011-07-13 12:24:10 +02:00
protected static function initTokenMap() {
2011-04-18 19:02:30 +02:00
if (!self::$tokenMap) {
self::$tokenMap = array();
// 256 is the minimum possible token number, as everything below
// it is an ASCII value
for ($i = 256; $i < 1000; ++$i) {
// T_DOUBLE_COLON is equivalent to T_PAAMAYIM_NEKUDOTAYIM
if (T_DOUBLE_COLON === $i) {
2011-06-05 18:40:04 +02:00
self::$tokenMap[$i] = PHPParser_Parser::T_PAAMAYIM_NEKUDOTAYIM;
2011-04-18 19:02:30 +02:00
// T_OPEN_TAG_WITH_ECHO with dropped T_OPEN_TAG results in T_ECHO
} elseif(T_OPEN_TAG_WITH_ECHO === $i) {
2011-06-05 18:40:04 +02:00
self::$tokenMap[$i] = PHPParser_Parser::T_ECHO;
2011-04-18 19:02:30 +02:00
// T_CLOSE_TAG is equivalent to ';'
} elseif(T_CLOSE_TAG === $i) {
self::$tokenMap[$i] = ord(';');
// and the others can be mapped directly
} elseif ('UNKNOWN' !== ($name = token_name($i))
&& defined($name = 'PHPParser_Parser::' . $name)
) {
self::$tokenMap[$i] = constant($name);
2011-04-18 19:02:30 +02:00
}
}
}
}
}