class StaticReflectionParser

Parses a file for namespaces/use/class declarations.

@author Karoly Negyesi <karoly@negyesi.net>

Hierarchy

Expanded class hierarchy of StaticReflectionParser

3 files declare their use of StaticReflectionParser
AbstractReaderTest.php in drupal/core/vendor/doctrine/common/tests/Doctrine/Tests/Common/Annotations/AbstractReaderTest.php
AnnotatedClassDiscovery.php in drupal/core/lib/Drupal/Component/Plugin/Discovery/AnnotatedClassDiscovery.php
Contains Drupal\Component\Plugin\Discovery\AnnotatedClassDiscovery.
StaticReflectionParserTest.php in drupal/core/vendor/doctrine/common/tests/Doctrine/Tests/Common/Reflection/StaticReflectionParserTest.php

File

drupal/core/vendor/doctrine/common/lib/Doctrine/Common/Reflection/StaticReflectionParser.php, line 30

Namespace

Doctrine\Common\Reflection
View source
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 . '"');
  }

}

Members

Namesort descending Modifiers Type Description Overrides
StaticReflectionParser::$classAnnotationOptimize protected property TRUE if the caller only wants class annotations.
StaticReflectionParser::$className protected property The name of the class.
StaticReflectionParser::$docComment protected property The docComment of the class.
StaticReflectionParser::$namespace protected property The namespace of the class
StaticReflectionParser::$parentClassName protected property The name of the class this class extends, if any.
StaticReflectionParser::$parentStaticReflectionParser protected property The parent PSR-0 Parser.
StaticReflectionParser::$parsed protected property TRUE when the parser has ran.
StaticReflectionParser::$useStatements protected property The use statements of this class.
StaticReflectionParser::getClassName public function
StaticReflectionParser::getDocComment public function Get docComment.
StaticReflectionParser::getNamespaceName public function
StaticReflectionParser::getParentStaticReflectionParser protected function
StaticReflectionParser::getReflectionClass public function Get the ReflectionClass equivalent for this file / class. Overrides ReflectionProviderInterface::getReflectionClass
StaticReflectionParser::getReflectionMethod public function Get the ReflectionMethod equivalent for the method of this file / class. Overrides ReflectionProviderInterface::getReflectionMethod
StaticReflectionParser::getReflectionProperty public function Get the ReflectionProperty equivalent for the method of this file / class. Overrides ReflectionProviderInterface::getReflectionProperty
StaticReflectionParser::getStaticReflectionParserForDeclaringClass public function Get the PSR-0 parser for the declaring class.
StaticReflectionParser::getUseStatements public function Get the use statements from this file.
StaticReflectionParser::parse protected function
StaticReflectionParser::__construct public function Parses a class residing in a PSR-0 hierarchy.