Parses a file for namespaces/use/class declarations.
@author Karoly Negyesi <karoly@negyesi.net>
Expanded class hierarchy of StaticReflectionParser
class StaticReflectionParser implements ReflectionProviderInterface {
/**
* The name of the class.
*
* @var string
*/
protected $className;
/**
* TRUE if the caller only wants class annotations.
*
* @var boolean.
*/
protected $classAnnotationOptimize;
/**
* TRUE when the parser has ran.
*
* @var boolean
*/
protected $parsed = false;
/**
* The namespace of the class
*
* @var string
*/
protected $namespace = '';
/**
* The use statements of this class.
*
* @var array
*/
protected $useStatements = array();
/**
* The docComment of the class.
*
* @var string
*/
protected $docComment = array(
'class' => '',
'property' => array(),
'method' => array(),
);
/**
* The name of the class this class extends, if any.
*
* @var string
*/
protected $parentClassName = '';
/**
* The parent PSR-0 Parser.
*
* @var \Doctrine\Common\Annotations\StaticReflectionParser
*/
protected $parentStaticReflectionParser;
/**
* Parses a class residing in a PSR-0 hierarchy.
*
* @param string $class
* The full, namespaced class name.
* @param ClassFinder $finder
* A ClassFinder object which finds the class.
* @param boolean $classAnnotationOptimize
* Only retrieve the class docComment. Presumes there is only one
* statement per line.
*/
public function __construct($className, $finder, $classAnnotationOptimize = false) {
$this->className = ltrim($className, '\\');
if ($lastNsPos = strrpos($this->className, '\\')) {
$this->namespace = substr($this->className, 0, $lastNsPos);
}
$this->finder = $finder;
$this->classAnnotationOptimize = $classAnnotationOptimize;
}
protected function parse() {
if ($this->parsed || !($fileName = $this->finder
->findFile($this->className))) {
return;
}
$this->parsed = true;
$contents = file_get_contents($fileName);
if ($this->classAnnotationOptimize) {
if (preg_match("/(\\A.*)^\\s+(abstract|final)?\\s+class\\s+{$className}\\s+{/sm", $contents, $matches)) {
$contents = $matches[1];
}
}
$tokenParser = new TokenParser($contents);
$docComment = '';
while ($token = $tokenParser
->next(false)) {
if (is_array($token)) {
switch ($token[0]) {
case T_USE:
$this->useStatements = array_merge($this->useStatements, $tokenParser
->parseUseStatement());
break;
case T_DOC_COMMENT:
$docComment = $token[1];
break;
case T_CLASS:
$this->docComment['class'] = $docComment;
$docComment = '';
break;
case T_VAR:
case T_PRIVATE:
case T_PROTECTED:
case T_PUBLIC:
$token = $tokenParser
->next();
if ($token[0] === T_VARIABLE) {
$propertyName = substr($token[1], 1);
$this->docComment['property'][$propertyName] = $docComment;
continue 2;
}
if ($token[0] !== T_FUNCTION) {
// For example, it can be T_FINAL.
continue 2;
}
// No break.
case T_FUNCTION:
// The next string after function is the name, but
// there can be & before the function name so find the
// string.
while (($token = $tokenParser
->next()) && $token[0] !== T_STRING) {
}
$methodName = $token[1];
$this->docComment['method'][$methodName] = $docComment;
$docComment = '';
break;
case T_EXTENDS:
$this->parentClassName = $tokenParser
->parseClass();
$nsPos = strpos($this->parentClassName, '\\');
$fullySpecified = false;
if ($nsPos === 0) {
$fullySpecified = true;
}
else {
if ($nsPos) {
$prefix = strtolower(substr($this->parentClassName, 0, $nsPos));
$postfix = substr($this->parentClassName, $nsPos);
}
else {
$prefix = strtolower($this->parentClassName);
$postfix = '';
}
foreach ($this->useStatements as $alias => $use) {
if ($alias == $prefix) {
$this->parentClassName = '\\' . $use . $postfix;
$fullySpecified = true;
}
}
}
if (!$fullySpecified) {
$this->parentClassName = '\\' . $this->namespace . '\\' . $this->parentClassName;
}
break;
}
}
}
}
protected function getParentStaticReflectionParser() {
if (empty($this->parentStaticReflectionParser)) {
$this->parentStaticReflectionParser = new static($this->parentClassName, $this->finder);
}
return $this->parentStaticReflectionParser;
}
public function getClassName() {
return $this->className;
}
public function getNamespaceName() {
return $this->namespace;
}
/**
* Get the ReflectionClass equivalent for this file / class.
*/
public function getReflectionClass() {
return new StaticReflectionClass($this);
}
/**
* Get the ReflectionMethod equivalent for the method of this file / class.
*/
public function getReflectionMethod($methodName) {
return new StaticReflectionMethod($this, $methodName);
}
/**
* Get the ReflectionProperty equivalent for the method of this file / class.
*/
public function getReflectionProperty($propertyName) {
return new StaticReflectionProperty($this, $propertyName);
}
/**
* Get the use statements from this file.
*/
public function getUseStatements() {
$this
->parse();
return $this->useStatements;
}
/**
* Get docComment.
*
* @param string $type class, property or method.
* @param string $name Name of the property or method, not needed for class.
*
* @return string the doc comment or empty string if none.
*/
public function getDocComment($type = 'class', $name = '') {
$this
->parse();
return $name ? $this->docComment[$type][$name] : $this->docComment[$type];
}
/**
* Get the PSR-0 parser for the declaring class.
*
* @param string $type property or method.
* @param string $name Name of the property or method.
*
* @return StaticReflectionParser A static reflection parser for the declaring class.
*/
public function getStaticReflectionParserForDeclaringClass($type, $name) {
$this
->parse();
if (isset($this->docComment[$type][$name])) {
return $this;
}
if (!empty($this->parentClassName)) {
return $this
->getParentStaticReflectionParser()
->getStaticReflectionParserForDeclaringClass($type, $name);
}
throw new ReflectionException('Invalid ' . $type . ' "' . $name . '"');
}
}
Name | Modifiers | Type | Description | Overrides |
---|---|---|---|---|
StaticReflectionParser:: |
protected | property | TRUE if the caller only wants class annotations. | |
StaticReflectionParser:: |
protected | property | The name of the class. | |
StaticReflectionParser:: |
protected | property | The docComment of the class. | |
StaticReflectionParser:: |
protected | property | The namespace of the class | |
StaticReflectionParser:: |
protected | property | The name of the class this class extends, if any. | |
StaticReflectionParser:: |
protected | property | The parent PSR-0 Parser. | |
StaticReflectionParser:: |
protected | property | TRUE when the parser has ran. | |
StaticReflectionParser:: |
protected | property | The use statements of this class. | |
StaticReflectionParser:: |
public | function | ||
StaticReflectionParser:: |
public | function | Get docComment. | |
StaticReflectionParser:: |
public | function | ||
StaticReflectionParser:: |
protected | function | ||
StaticReflectionParser:: |
public | function |
Get the ReflectionClass equivalent for this file / class. Overrides ReflectionProviderInterface:: |
|
StaticReflectionParser:: |
public | function |
Get the ReflectionMethod equivalent for the method of this file / class. Overrides ReflectionProviderInterface:: |
|
StaticReflectionParser:: |
public | function |
Get the ReflectionProperty equivalent for the method of this file / class. Overrides ReflectionProviderInterface:: |
|
StaticReflectionParser:: |
public | function | Get the PSR-0 parser for the declaring class. | |
StaticReflectionParser:: |
public | function | Get the use statements from this file. | |
StaticReflectionParser:: |
protected | function | ||
StaticReflectionParser:: |
public | function | Parses a class residing in a PSR-0 hierarchy. |