320 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			320 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
/**
 | 
						|
 * This file is part of the Peast package
 | 
						|
 *
 | 
						|
 * (c) Marco Marchiò <marco.mm89@gmail.com>
 | 
						|
 *
 | 
						|
 * For the full copyright and license information refer to the LICENSE file
 | 
						|
 * distributed with this source code
 | 
						|
 */
 | 
						|
namespace Peast\Syntax;
 | 
						|
 | 
						|
/**
 | 
						|
 * Comments registry class. Internal class used to manage comments
 | 
						|
 * 
 | 
						|
 * @author Marco Marchiò <marco.mm89@gmail.com>
 | 
						|
 */
 | 
						|
class CommentsRegistry
 | 
						|
{
 | 
						|
    /**
 | 
						|
     * Map of the indices where nodes start
 | 
						|
     * 
 | 
						|
     * @var int 
 | 
						|
     */
 | 
						|
    protected $nodesStartMap = array();
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Map of the indices where nodes end
 | 
						|
     * 
 | 
						|
     * @var int 
 | 
						|
     */
 | 
						|
    protected $nodesEndMap = array();
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Comments buffer
 | 
						|
     * 
 | 
						|
     * @var array
 | 
						|
     */
 | 
						|
    protected $buffer = null;
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Last token index
 | 
						|
     * 
 | 
						|
     * @var int
 | 
						|
     */
 | 
						|
    protected $lastTokenIndex = null;
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Comments registry
 | 
						|
     * 
 | 
						|
     * @var array
 | 
						|
     */
 | 
						|
    protected $registry = array();
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Class constructor
 | 
						|
     * 
 | 
						|
     * @param Parser    $parser     Parser
 | 
						|
     */
 | 
						|
    public function __construct(Parser $parser)
 | 
						|
    {
 | 
						|
        $parser->getEventsEmitter()
 | 
						|
               ->addListener("NodeCompleted", array($this, "onNodeCompleted"))
 | 
						|
               ->addListener("EndParsing", array($this, "onEndParsing"));
 | 
						|
        
 | 
						|
        $parser->getScanner()->getEventsEmitter()
 | 
						|
               ->addListener("TokenConsumed", array($this, "onTokenConsumed"))
 | 
						|
               ->addListener("EndReached", array($this, "onTokenConsumed"))
 | 
						|
               ->addListener("FreezeState", array($this, "onScannerFreezeState"))
 | 
						|
               ->addListener("ResetState", array($this, "onScannerResetState"));
 | 
						|
    }
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Listener called every time the scanner compose the array that represents
 | 
						|
     * its current state
 | 
						|
     * 
 | 
						|
     * @param array   $state   State
 | 
						|
     * 
 | 
						|
     * @return void
 | 
						|
     */
 | 
						|
    public function onScannerFreezeState(&$state)
 | 
						|
    {
 | 
						|
        //Register the current last token index
 | 
						|
        $state["commentsLastTokenIndex"] = $this->lastTokenIndex;
 | 
						|
    }
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Listener called every time the scanner reset its state using the given
 | 
						|
     * array
 | 
						|
     * 
 | 
						|
     * @param array   $state   State
 | 
						|
     * 
 | 
						|
     * @return void
 | 
						|
     */
 | 
						|
    public function onScannerResetState(&$state)
 | 
						|
    {
 | 
						|
        //Reset the last token index and delete it from the state array
 | 
						|
        $this->lastTokenIndex = $state["commentsLastTokenIndex"];
 | 
						|
        unset($state["commentsLastTokenIndex"]);
 | 
						|
    }
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Listener called every time a token is consumed and when the scanner
 | 
						|
     * reaches the end of the source
 | 
						|
     * 
 | 
						|
     * @param Token|null   $token   Consumed token or null if the end has
 | 
						|
     *                              been reached
 | 
						|
     * 
 | 
						|
     * @return void
 | 
						|
     */
 | 
						|
    public function onTokenConsumed($token = null)
 | 
						|
    {
 | 
						|
        //Check if it's a comment
 | 
						|
        if ($token && $token->type === Token::TYPE_COMMENT) {
 | 
						|
            //If there is not an open comments buffer, create it
 | 
						|
            if (!$this->buffer) {
 | 
						|
                $this->buffer = array(
 | 
						|
                    "prev" => $this->lastTokenIndex,
 | 
						|
                    "next" => null,
 | 
						|
                    "comments" => array()
 | 
						|
                );
 | 
						|
            }
 | 
						|
            //Add the comment token to the buffer
 | 
						|
            $this->buffer["comments"][] = $token;
 | 
						|
        } else {
 | 
						|
            
 | 
						|
            if ($token) {
 | 
						|
                $loc = $token->location;
 | 
						|
                //Store the token end position
 | 
						|
                $this->lastTokenIndex = $loc->end->getIndex();
 | 
						|
                if ($this->buffer) {
 | 
						|
                    //Fill the "next" key on the comments buffer with the token
 | 
						|
                    //start position
 | 
						|
                    $this->buffer["next"] = $loc->start->getIndex();
 | 
						|
                }
 | 
						|
            }
 | 
						|
            
 | 
						|
            //If there is an open comment buffer, close it and move it to the
 | 
						|
            //registry
 | 
						|
            if ($buffer = $this->buffer) {
 | 
						|
                //Use the location as key to add the group of comments to the
 | 
						|
                //registry, in this way if comments are reprocessed they won't
 | 
						|
                //be duplicated
 | 
						|
                $key = implode("-", array(
 | 
						|
                    $buffer["prev"] !== null ? $buffer["prev"] : "",
 | 
						|
                    $buffer["next"] !== null ? $buffer["next"] : ""
 | 
						|
                ));
 | 
						|
                $this->registry[$key] = $this->buffer;
 | 
						|
                $this->buffer = null;
 | 
						|
            }
 | 
						|
            
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Listener called every time a node is completed by the parser
 | 
						|
     * 
 | 
						|
     * @param Node\Node   $node     Completed node
 | 
						|
     * 
 | 
						|
     * @return void
 | 
						|
     */
 | 
						|
    public function onNodeCompleted(Node\Node $node)
 | 
						|
    {
 | 
						|
        //Every time a node is completed, register its start and end indices
 | 
						|
        //in the relative properties
 | 
						|
        $loc = $node->location;
 | 
						|
        foreach (array("Start", "End") as $pos) {
 | 
						|
            $val = $loc->{"get$pos"}()->getIndex();
 | 
						|
            $map = &$this->{"nodes{$pos}Map"};
 | 
						|
            if (!isset($map[$val])) {
 | 
						|
                $map[$val] = array();
 | 
						|
            }
 | 
						|
            $map[$val][] = $node;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Listener called when parsing process ends
 | 
						|
     * 
 | 
						|
     * @return void
 | 
						|
     */
 | 
						|
    public function onEndParsing()
 | 
						|
    {
 | 
						|
        //Return if there are no comments to process
 | 
						|
        if ($this->registry) {
 | 
						|
            
 | 
						|
            //Make sure nodes start indices map is sorted
 | 
						|
            ksort($this->nodesStartMap);
 | 
						|
            
 | 
						|
            //Loop all comment groups in the registry
 | 
						|
            foreach ($this->registry as $group) {
 | 
						|
                $this->findNodeForCommentsGroup($group);
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Finds the node to attach the given comments group
 | 
						|
     * 
 | 
						|
     * @param array    $group   Comments group
 | 
						|
     * 
 | 
						|
     * @return void
 | 
						|
     */
 | 
						|
    public function findNodeForCommentsGroup($group)
 | 
						|
    {
 | 
						|
        $next = $group["next"];
 | 
						|
        $prev = $group["prev"];
 | 
						|
        $comments = $group["comments"];
 | 
						|
        $leading = true;
 | 
						|
        
 | 
						|
        //If the group of comments has a next token index that appears
 | 
						|
        //in the map of start node indices, add the group to the
 | 
						|
        //corresponding node's leading comments. This associates
 | 
						|
        //comments that appear immediately before a node.
 | 
						|
        //For example: /*comment*/ for (;;){}
 | 
						|
        if (isset($this->nodesStartMap[$next])) {
 | 
						|
            $nodes = $this->nodesStartMap[$next];
 | 
						|
        }
 | 
						|
        //If the group of comments has a previous token index that appears
 | 
						|
        //in the map of end node indices, add the group to the
 | 
						|
        //corresponding node's trailing comments. This associates
 | 
						|
        //comments that appear immediately after a node.
 | 
						|
        //For example: for (;;){} /*comment*/ 
 | 
						|
        elseif (isset($this->nodesEndMap[$prev])) {
 | 
						|
            $nodes = $this->nodesEndMap[$prev];
 | 
						|
            $leading = false;
 | 
						|
        }
 | 
						|
        //Otherwise, find a node that wraps the comments position.
 | 
						|
        //This associates inner comments:
 | 
						|
        //For example: for /*comment*/ (;;){}
 | 
						|
        else {
 | 
						|
            //Calculate comments group boundaries
 | 
						|
            $start = $comments[0]->location->start->getIndex();
 | 
						|
            $end = $comments[count($comments) -1]->location->end->getIndex();
 | 
						|
            $nodes = array();
 | 
						|
            
 | 
						|
            //Loop all the entries in the start index map
 | 
						|
            foreach ($this->nodesStartMap as $idx => $ns) {
 | 
						|
                //If the index is higher than the start index of the comments
 | 
						|
                //group, stop
 | 
						|
                if ($idx > $start) {
 | 
						|
                    break;
 | 
						|
                }
 | 
						|
                foreach ($ns as $node) {
 | 
						|
                    //Check if the comments group is inside node indices range
 | 
						|
                    if ($node->location->end->getIndex() >= $end) {
 | 
						|
                        $nodes[] = $node;
 | 
						|
                    }
 | 
						|
                }
 | 
						|
            }
 | 
						|
            
 | 
						|
            //If comments can't be associated with any node, associate it as
 | 
						|
            //leading comments of the program, this happens when the source is
 | 
						|
            //empty
 | 
						|
            if (!$nodes) {
 | 
						|
                $firstNode = array_values($this->nodesStartMap);
 | 
						|
                $nodes = array($firstNode[0][0]);
 | 
						|
            }
 | 
						|
        }
 | 
						|
        
 | 
						|
        //If there are multiple possible nodes to associate the comments to,
 | 
						|
        //find the shortest one
 | 
						|
        if (count($nodes) > 1) {
 | 
						|
            usort($nodes, array($this, "compareNodesLength"));
 | 
						|
        }
 | 
						|
        $this->associateComments($nodes[0], $comments, $leading);
 | 
						|
    }
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Compares node length 
 | 
						|
     * 
 | 
						|
     * @param Node\Node  $node1     First node
 | 
						|
     * @param Node\Node  $node2     Second node
 | 
						|
     * 
 | 
						|
     * @return int
 | 
						|
     * 
 | 
						|
     * @codeCoverageIgnore
 | 
						|
     */
 | 
						|
    public function compareNodesLength($node1, $node2)
 | 
						|
    {
 | 
						|
        $loc1 = $node1->location;
 | 
						|
        $length1 = $loc1->end->getIndex() - $loc1->start->getIndex();
 | 
						|
        $loc2 = $node2->location;
 | 
						|
        $length2 = $loc2->end->getIndex() - $loc2->start->getIndex();
 | 
						|
        //If the nodes have the same length make sure to choose nodes
 | 
						|
        //different from Program nodes
 | 
						|
        if ($length1 === $length2) {
 | 
						|
            if ($node1 instanceof Node\Program) {
 | 
						|
                $length1 += 1000;
 | 
						|
            } elseif ($node2 instanceof Node\Program) {
 | 
						|
                $length2 += 1000;
 | 
						|
            }
 | 
						|
        }
 | 
						|
        return $length1 < $length2 ? -1 : 1;
 | 
						|
    }
 | 
						|
    
 | 
						|
    /**
 | 
						|
     * Adds comments to the given node
 | 
						|
     * 
 | 
						|
     * @param Node\Node     $node       Node
 | 
						|
     * @param array         $comments   Array of comments to add
 | 
						|
     * @param bool          $leading    True to add comments as leading comments
 | 
						|
     *                                  or false to add them as trailing comments
 | 
						|
     * 
 | 
						|
     * @return void
 | 
						|
     */
 | 
						|
    public function associateComments($node, $comments, $leading)
 | 
						|
    {
 | 
						|
        $fn = ($leading ? "Leading" : "Trailing") . "Comments";
 | 
						|
        $currentComments = $node->{"get$fn"}();
 | 
						|
        foreach ($comments as $comment) {
 | 
						|
            $loc = $comment->location;
 | 
						|
            $commentNode = new Node\Comment;
 | 
						|
            $commentNode->location->start = $loc->start;
 | 
						|
            $commentNode->location->end = $loc->end;
 | 
						|
            $commentNode->setRawText($comment->value);
 | 
						|
            $currentComments[] = $commentNode;
 | 
						|
        }
 | 
						|
        $node->{"set$fn"}($currentComments);
 | 
						|
    }
 | 
						|
} |