*/ private $tokens; /** * The number of tokens. * * @var int */ private $numTokens; /** * The current array pointer. * * @var int */ private $pointer = 0; /** @param string $contents */ public function __construct($contents) { $this->tokens = token_get_all($contents); // The PHP parser sets internal compiler globals for certain things. Annoyingly, the last docblock comment it // saw gets stored in doc_comment. When it comes to compile the next thing to be include()d this stored // doc_comment becomes owned by the first thing the compiler sees in the file that it considers might have a // docblock. If the first thing in the file is a class without a doc block this would cause calls to // getDocBlock() on said class to return our long lost doc_comment. Argh. // To workaround, cause the parser to parse an empty docblock. Sure getDocBlock() will return this, but at least // it's harmless to us. token_get_all("numTokens = count($this->tokens); } /** * Gets the next non whitespace and non comment token. * * @param bool $docCommentIsComment If TRUE then a doc comment is considered a comment and skipped. * If FALSE then only whitespace and normal comments are skipped. * * @return mixed[]|string|null The token if exists, null otherwise. */ public function next($docCommentIsComment = true) { for ($i = $this->pointer; $i < $this->numTokens; $i++) { $this->pointer++; if ( $this->tokens[$i][0] === T_WHITESPACE || $this->tokens[$i][0] === T_COMMENT || ($docCommentIsComment && $this->tokens[$i][0] === T_DOC_COMMENT) ) { continue; } return $this->tokens[$i]; } return null; } /** * Parses a single use statement. * * @return array A list with all found class names for a use statement. */ public function parseUseStatement() { $groupRoot = ''; $class = ''; $alias = ''; $statements = []; $explicitAlias = false; while (($token = $this->next())) { if (! $explicitAlias && $token[0] === T_STRING) { $class .= $token[1]; $alias = $token[1]; } elseif ($explicitAlias && $token[0] === T_STRING) { $alias = $token[1]; } elseif ( PHP_VERSION_ID >= 80000 && ($token[0] === T_NAME_QUALIFIED || $token[0] === T_NAME_FULLY_QUALIFIED) ) { $class .= $token[1]; $classSplit = explode('\\', $token[1]); $alias = $classSplit[count($classSplit) - 1]; } elseif ($token[0] === T_NS_SEPARATOR) { $class .= '\\'; $alias = ''; } elseif ($token[0] === T_AS) { $explicitAlias = true; $alias = ''; } elseif ($token === ',') { $statements[strtolower($alias)] = $groupRoot . $class; $class = ''; $alias = ''; $explicitAlias = false; } elseif ($token === ';') { $statements[strtolower($alias)] = $groupRoot . $class; break; } elseif ($token === '{') { $groupRoot = $class; $class = ''; } elseif ($token === '}') { continue; } else { break; } } return $statements; } /** * Gets all use statements. * * @param string $namespaceName The namespace name of the reflected class. * * @return array A list with all found use statements. */ public function parseUseStatements($namespaceName) { $statements = []; while (($token = $this->next())) { if ($token[0] === T_USE) { $statements = array_merge($statements, $this->parseUseStatement()); continue; } if ($token[0] !== T_NAMESPACE || $this->parseNamespace() !== $namespaceName) { continue; } // Get fresh array for new namespace. This is to prevent the parser to collect the use statements // for a previous namespace with the same name. This is the case if a namespace is defined twice // or if a namespace with the same name is commented out. $statements = []; } return $statements; } /** * Gets the namespace. * * @return string The found namespace. */ public function parseNamespace() { $name = ''; while ( ($token = $this->next()) && ($token[0] === T_STRING || $token[0] === T_NS_SEPARATOR || ( PHP_VERSION_ID >= 80000 && ($token[0] === T_NAME_QUALIFIED || $token[0] === T_NAME_FULLY_QUALIFIED) )) ) { $name .= $token[1]; } return $name; } /** * Gets the class name. * * @return string The found class name. */ public function parseClass() { // Namespaces and class names are tokenized the same: T_STRINGs // separated by T_NS_SEPARATOR so we can use one function to provide // both. return $this->parseNamespace(); } } __halt_compiler();----SIGNATURE:----KbGKVw/Sp/4yiWTx3n6Wlt9R7mN7+5G9zMeRKC/Ssy5twbH0yCtqAWmYgwnY1RXtgMCEa6TP4Xde9YtZRC/B3HQ4R6uV2IFL7he7amkeN7r7Qgq92c7T6Yeg5rguZOU6vXKgJ1Zi0DtzjIXzJPOrOhem2lJ2G8WrveOL8M3REwwYX7FjYdOSjzlqXVShGyXoVrlMqfyZibBzxOPjjpUnCKkJg8+mjD0HNF+hCW8k9nA2tzXlQyZpXILmZPiM3X8DF7/vmoFldEeRisO5IRYHQAPcd68SGPk07POujhLRKzk+t+8ERuN9zHYIgKItEV8m/yBfyPvrpchEadlTL2s1FvGf6++v4gS+XvvHii2cZ6z0fGb59k2OKIa6K04Weyfg/i8pZQVNwkQVpMV9D8Cra1yeHJkCqxQHMiG0m9XkQXKbogkcvlUYBpXi4wNa2CCS3eWGFfLRilEiWjTzc4UJ8GTU+GLfXQFMoViSGRDOy7XhMpYgbh1AmgXpU0pyGclKMwolzqrG75R1wdAI6/EHwYSLAk1wO6dh2nzY5qhSj6bsoVzYI/FAcqpzp5h1t3Ja9ORDQDQ0xR89MjavFpkn4DD7Vwr6hL3DYMq1TS8yos3N4A7NhH2wkEwm8dG91IEOIGjhRSs8MEYZ72UmAMY1Ug8OXKfttDMQZsUif+pPTmU=----ATTACHMENT:----MjI2NjU5MDA2OTMwNDQyIDEzMzc1MDk1Mzk0MTI1NjYgNTYxNDU1OTk3MDY0MjIwNw==