radarrplexorganizrnginxsonarrdashboardembycouchpotatonzbgetbookmarkapplication-dashboardmuximuxlandingpagestartpagelandinghtpcserverhomepagesabnzbdheimdall
		
		
		
		
			You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							384 lines
						
					
					
						
							12 KiB
						
					
					
				
			
		
		
		
			
			
			
				
					
				
				
					
				
			
		
		
	
	
							384 lines
						
					
					
						
							12 KiB
						
					
					
				| <?php | |
| 
 | |
| /* | |
|  * This file is part of the Symfony package. | |
|  * | |
|  * (c) Fabien Potencier <fabien@symfony.com> | |
|  * | |
|  * For the full copyright and license information, please view the LICENSE | |
|  * file that was distributed with this source code. | |
|  */ | |
| 
 | |
| namespace Symfony\Component\CssSelector\Parser; | |
| 
 | |
| use Symfony\Component\CssSelector\Exception\SyntaxErrorException; | |
| use Symfony\Component\CssSelector\Node; | |
| use Symfony\Component\CssSelector\Parser\Tokenizer\Tokenizer; | |
| 
 | |
| /** | |
|  * CSS selector parser. | |
|  * | |
|  * This component is a port of the Python cssselect library, | |
|  * which is copyright Ian Bicking, @see https://github.com/SimonSapin/cssselect. | |
|  * | |
|  * @author Jean-François Simon <jeanfrancois.simon@sensiolabs.com> | |
|  * | |
|  * @internal | |
|  */ | |
| class Parser implements ParserInterface | |
| { | |
|     private $tokenizer; | |
| 
 | |
|     public function __construct(Tokenizer $tokenizer = null) | |
|     { | |
|         $this->tokenizer = $tokenizer ?: new Tokenizer(); | |
|     } | |
| 
 | |
|     /** | |
|      * {@inheritdoc} | |
|      */ | |
|     public function parse($source) | |
|     { | |
|         $reader = new Reader($source); | |
|         $stream = $this->tokenizer->tokenize($reader); | |
| 
 | |
|         return $this->parseSelectorList($stream); | |
|     } | |
| 
 | |
|     /** | |
|      * Parses the arguments for ":nth-child()" and friends. | |
|      * | |
|      * @param Token[] $tokens | |
|      * | |
|      * @return array | |
|      * | |
|      * @throws SyntaxErrorException | |
|      */ | |
|     public static function parseSeries(array $tokens) | |
|     { | |
|         foreach ($tokens as $token) { | |
|             if ($token->isString()) { | |
|                 throw SyntaxErrorException::stringAsFunctionArgument(); | |
|             } | |
|         } | |
| 
 | |
|         $joined = trim(implode('', array_map(function (Token $token) { | |
|             return $token->getValue(); | |
|         }, $tokens))); | |
| 
 | |
|         $int = function ($string) { | |
|             if (!is_numeric($string)) { | |
|                 throw SyntaxErrorException::stringAsFunctionArgument(); | |
|             } | |
| 
 | |
|             return (int) $string; | |
|         }; | |
| 
 | |
|         switch (true) { | |
|             case 'odd' === $joined: | |
|                 return array(2, 1); | |
|             case 'even' === $joined: | |
|                 return array(2, 0); | |
|             case 'n' === $joined: | |
|                 return array(1, 0); | |
|             case false === strpos($joined, 'n'): | |
|                 return array(0, $int($joined)); | |
|         } | |
| 
 | |
|         $split = explode('n', $joined); | |
|         $first = isset($split[0]) ? $split[0] : null; | |
| 
 | |
|         return array( | |
|             $first ? ('-' === $first || '+' === $first ? $int($first.'1') : $int($first)) : 1, | |
|             isset($split[1]) && $split[1] ? $int($split[1]) : 0, | |
|         ); | |
|     } | |
| 
 | |
|     /** | |
|      * Parses selector nodes. | |
|      * | |
|      * @return array | |
|      */ | |
|     private function parseSelectorList(TokenStream $stream) | |
|     { | |
|         $stream->skipWhitespace(); | |
|         $selectors = array(); | |
| 
 | |
|         while (true) { | |
|             $selectors[] = $this->parserSelectorNode($stream); | |
| 
 | |
|             if ($stream->getPeek()->isDelimiter(array(','))) { | |
|                 $stream->getNext(); | |
|                 $stream->skipWhitespace(); | |
|             } else { | |
|                 break; | |
|             } | |
|         } | |
| 
 | |
|         return $selectors; | |
|     } | |
| 
 | |
|     /** | |
|      * Parses next selector or combined node. | |
|      * | |
|      * @return Node\SelectorNode | |
|      * | |
|      * @throws SyntaxErrorException | |
|      */ | |
|     private function parserSelectorNode(TokenStream $stream) | |
|     { | |
|         list($result, $pseudoElement) = $this->parseSimpleSelector($stream); | |
| 
 | |
|         while (true) { | |
|             $stream->skipWhitespace(); | |
|             $peek = $stream->getPeek(); | |
| 
 | |
|             if ($peek->isFileEnd() || $peek->isDelimiter(array(','))) { | |
|                 break; | |
|             } | |
| 
 | |
|             if (null !== $pseudoElement) { | |
|                 throw SyntaxErrorException::pseudoElementFound($pseudoElement, 'not at the end of a selector'); | |
|             } | |
| 
 | |
|             if ($peek->isDelimiter(array('+', '>', '~'))) { | |
|                 $combinator = $stream->getNext()->getValue(); | |
|                 $stream->skipWhitespace(); | |
|             } else { | |
|                 $combinator = ' '; | |
|             } | |
| 
 | |
|             list($nextSelector, $pseudoElement) = $this->parseSimpleSelector($stream); | |
|             $result = new Node\CombinedSelectorNode($result, $combinator, $nextSelector); | |
|         } | |
| 
 | |
|         return new Node\SelectorNode($result, $pseudoElement); | |
|     } | |
| 
 | |
|     /** | |
|      * Parses next simple node (hash, class, pseudo, negation). | |
|      * | |
|      * @param TokenStream $stream | |
|      * @param bool        $insideNegation | |
|      * | |
|      * @return array | |
|      * | |
|      * @throws SyntaxErrorException | |
|      */ | |
|     private function parseSimpleSelector(TokenStream $stream, $insideNegation = false) | |
|     { | |
|         $stream->skipWhitespace(); | |
| 
 | |
|         $selectorStart = count($stream->getUsed()); | |
|         $result = $this->parseElementNode($stream); | |
|         $pseudoElement = null; | |
| 
 | |
|         while (true) { | |
|             $peek = $stream->getPeek(); | |
|             if ($peek->isWhitespace() | |
|                 || $peek->isFileEnd() | |
|                 || $peek->isDelimiter(array(',', '+', '>', '~')) | |
|                 || ($insideNegation && $peek->isDelimiter(array(')'))) | |
|             ) { | |
|                 break; | |
|             } | |
| 
 | |
|             if (null !== $pseudoElement) { | |
|                 throw SyntaxErrorException::pseudoElementFound($pseudoElement, 'not at the end of a selector'); | |
|             } | |
| 
 | |
|             if ($peek->isHash()) { | |
|                 $result = new Node\HashNode($result, $stream->getNext()->getValue()); | |
|             } elseif ($peek->isDelimiter(array('.'))) { | |
|                 $stream->getNext(); | |
|                 $result = new Node\ClassNode($result, $stream->getNextIdentifier()); | |
|             } elseif ($peek->isDelimiter(array('['))) { | |
|                 $stream->getNext(); | |
|                 $result = $this->parseAttributeNode($result, $stream); | |
|             } elseif ($peek->isDelimiter(array(':'))) { | |
|                 $stream->getNext(); | |
| 
 | |
|                 if ($stream->getPeek()->isDelimiter(array(':'))) { | |
|                     $stream->getNext(); | |
|                     $pseudoElement = $stream->getNextIdentifier(); | |
| 
 | |
|                     continue; | |
|                 } | |
| 
 | |
|                 $identifier = $stream->getNextIdentifier(); | |
|                 if (in_array(strtolower($identifier), array('first-line', 'first-letter', 'before', 'after'))) { | |
|                     // Special case: CSS 2.1 pseudo-elements can have a single ':'. | |
|                     // Any new pseudo-element must have two. | |
|                     $pseudoElement = $identifier; | |
| 
 | |
|                     continue; | |
|                 } | |
| 
 | |
|                 if (!$stream->getPeek()->isDelimiter(array('('))) { | |
|                     $result = new Node\PseudoNode($result, $identifier); | |
| 
 | |
|                     continue; | |
|                 } | |
| 
 | |
|                 $stream->getNext(); | |
|                 $stream->skipWhitespace(); | |
| 
 | |
|                 if ('not' === strtolower($identifier)) { | |
|                     if ($insideNegation) { | |
|                         throw SyntaxErrorException::nestedNot(); | |
|                     } | |
| 
 | |
|                     list($argument, $argumentPseudoElement) = $this->parseSimpleSelector($stream, true); | |
|                     $next = $stream->getNext(); | |
| 
 | |
|                     if (null !== $argumentPseudoElement) { | |
|                         throw SyntaxErrorException::pseudoElementFound($argumentPseudoElement, 'inside ::not()'); | |
|                     } | |
| 
 | |
|                     if (!$next->isDelimiter(array(')'))) { | |
|                         throw SyntaxErrorException::unexpectedToken('")"', $next); | |
|                     } | |
| 
 | |
|                     $result = new Node\NegationNode($result, $argument); | |
|                 } else { | |
|                     $arguments = array(); | |
|                     $next = null; | |
| 
 | |
|                     while (true) { | |
|                         $stream->skipWhitespace(); | |
|                         $next = $stream->getNext(); | |
| 
 | |
|                         if ($next->isIdentifier() | |
|                             || $next->isString() | |
|                             || $next->isNumber() | |
|                             || $next->isDelimiter(array('+', '-')) | |
|                         ) { | |
|                             $arguments[] = $next; | |
|                         } elseif ($next->isDelimiter(array(')'))) { | |
|                             break; | |
|                         } else { | |
|                             throw SyntaxErrorException::unexpectedToken('an argument', $next); | |
|                         } | |
|                     } | |
| 
 | |
|                     if (empty($arguments)) { | |
|                         throw SyntaxErrorException::unexpectedToken('at least one argument', $next); | |
|                     } | |
| 
 | |
|                     $result = new Node\FunctionNode($result, $identifier, $arguments); | |
|                 } | |
|             } else { | |
|                 throw SyntaxErrorException::unexpectedToken('selector', $peek); | |
|             } | |
|         } | |
| 
 | |
|         if (count($stream->getUsed()) === $selectorStart) { | |
|             throw SyntaxErrorException::unexpectedToken('selector', $stream->getPeek()); | |
|         } | |
| 
 | |
|         return array($result, $pseudoElement); | |
|     } | |
| 
 | |
|     /** | |
|      * Parses next element node. | |
|      * | |
|      * @return Node\ElementNode | |
|      */ | |
|     private function parseElementNode(TokenStream $stream) | |
|     { | |
|         $peek = $stream->getPeek(); | |
| 
 | |
|         if ($peek->isIdentifier() || $peek->isDelimiter(array('*'))) { | |
|             if ($peek->isIdentifier()) { | |
|                 $namespace = $stream->getNext()->getValue(); | |
|             } else { | |
|                 $stream->getNext(); | |
|                 $namespace = null; | |
|             } | |
| 
 | |
|             if ($stream->getPeek()->isDelimiter(array('|'))) { | |
|                 $stream->getNext(); | |
|                 $element = $stream->getNextIdentifierOrStar(); | |
|             } else { | |
|                 $element = $namespace; | |
|                 $namespace = null; | |
|             } | |
|         } else { | |
|             $element = $namespace = null; | |
|         } | |
| 
 | |
|         return new Node\ElementNode($namespace, $element); | |
|     } | |
| 
 | |
|     /** | |
|      * Parses next attribute node. | |
|      * | |
|      * @return Node\AttributeNode | |
|      * | |
|      * @throws SyntaxErrorException | |
|      */ | |
|     private function parseAttributeNode(Node\NodeInterface $selector, TokenStream $stream) | |
|     { | |
|         $stream->skipWhitespace(); | |
|         $attribute = $stream->getNextIdentifierOrStar(); | |
| 
 | |
|         if (null === $attribute && !$stream->getPeek()->isDelimiter(array('|'))) { | |
|             throw SyntaxErrorException::unexpectedToken('"|"', $stream->getPeek()); | |
|         } | |
| 
 | |
|         if ($stream->getPeek()->isDelimiter(array('|'))) { | |
|             $stream->getNext(); | |
| 
 | |
|             if ($stream->getPeek()->isDelimiter(array('='))) { | |
|                 $namespace = null; | |
|                 $stream->getNext(); | |
|                 $operator = '|='; | |
|             } else { | |
|                 $namespace = $attribute; | |
|                 $attribute = $stream->getNextIdentifier(); | |
|                 $operator = null; | |
|             } | |
|         } else { | |
|             $namespace = $operator = null; | |
|         } | |
| 
 | |
|         if (null === $operator) { | |
|             $stream->skipWhitespace(); | |
|             $next = $stream->getNext(); | |
| 
 | |
|             if ($next->isDelimiter(array(']'))) { | |
|                 return new Node\AttributeNode($selector, $namespace, $attribute, 'exists', null); | |
|             } elseif ($next->isDelimiter(array('='))) { | |
|                 $operator = '='; | |
|             } elseif ($next->isDelimiter(array('^', '$', '*', '~', '|', '!')) | |
|                 && $stream->getPeek()->isDelimiter(array('=')) | |
|             ) { | |
|                 $operator = $next->getValue().'='; | |
|                 $stream->getNext(); | |
|             } else { | |
|                 throw SyntaxErrorException::unexpectedToken('operator', $next); | |
|             } | |
|         } | |
| 
 | |
|         $stream->skipWhitespace(); | |
|         $value = $stream->getNext(); | |
| 
 | |
|         if ($value->isNumber()) { | |
|             // if the value is a number, it's casted into a string | |
|             $value = new Token(Token::TYPE_STRING, (string) $value->getValue(), $value->getPosition()); | |
|         } | |
| 
 | |
|         if (!($value->isIdentifier() || $value->isString())) { | |
|             throw SyntaxErrorException::unexpectedToken('string or identifier', $value); | |
|         } | |
| 
 | |
|         $stream->skipWhitespace(); | |
|         $next = $stream->getNext(); | |
| 
 | |
|         if (!$next->isDelimiter(array(']'))) { | |
|             throw SyntaxErrorException::unexpectedToken('"]"', $next); | |
|         } | |
| 
 | |
|         return new Node\AttributeNode($selector, $namespace, $attribute, $operator, $value->getValue()); | |
|     } | |
| }
 | |
| 
 |