TokenStream.php 3.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167
  1. <?php
  2. /*
  3. * This file is part of the Symfony package.
  4. *
  5. * (c) Fabien Potencier <fabien@symfony.com>
  6. *
  7. * For the full copyright and license information, please view the LICENSE
  8. * file that was distributed with this source code.
  9. */
  10. namespace Symfony\Component\CssSelector\Parser;
  11. use Symfony\Component\CssSelector\Exception\InternalErrorException;
  12. use Symfony\Component\CssSelector\Exception\SyntaxErrorException;
  13. /**
  14. * CSS selector token stream.
  15. *
  16. * This component is a port of the Python cssselect library,
  17. * which is copyright Ian Bicking, @see https://github.com/SimonSapin/cssselect.
  18. *
  19. * @author Jean-François Simon <jeanfrancois.simon@sensiolabs.com>
  20. *
  21. * @internal
  22. */
  23. class TokenStream
  24. {
  25. /**
  26. * @var Token[]
  27. */
  28. private $tokens = [];
  29. /**
  30. * @var Token[]
  31. */
  32. private $used = [];
  33. /**
  34. * @var int
  35. */
  36. private $cursor = 0;
  37. /**
  38. * @var Token|null
  39. */
  40. private $peeked;
  41. /**
  42. * @var bool
  43. */
  44. private $peeking = false;
  45. /**
  46. * Pushes a token.
  47. *
  48. * @return $this
  49. */
  50. public function push(Token $token): self
  51. {
  52. $this->tokens[] = $token;
  53. return $this;
  54. }
  55. /**
  56. * Freezes stream.
  57. *
  58. * @return $this
  59. */
  60. public function freeze(): self
  61. {
  62. return $this;
  63. }
  64. /**
  65. * Returns next token.
  66. *
  67. * @throws InternalErrorException If there is no more token
  68. */
  69. public function getNext(): Token
  70. {
  71. if ($this->peeking) {
  72. $this->peeking = false;
  73. $this->used[] = $this->peeked;
  74. return $this->peeked;
  75. }
  76. if (!isset($this->tokens[$this->cursor])) {
  77. throw new InternalErrorException('Unexpected token stream end.');
  78. }
  79. return $this->tokens[$this->cursor++];
  80. }
  81. /**
  82. * Returns peeked token.
  83. */
  84. public function getPeek(): Token
  85. {
  86. if (!$this->peeking) {
  87. $this->peeked = $this->getNext();
  88. $this->peeking = true;
  89. }
  90. return $this->peeked;
  91. }
  92. /**
  93. * Returns used tokens.
  94. *
  95. * @return Token[]
  96. */
  97. public function getUsed(): array
  98. {
  99. return $this->used;
  100. }
  101. /**
  102. * Returns next identifier token.
  103. *
  104. * @throws SyntaxErrorException If next token is not an identifier
  105. */
  106. public function getNextIdentifier(): string
  107. {
  108. $next = $this->getNext();
  109. if (!$next->isIdentifier()) {
  110. throw SyntaxErrorException::unexpectedToken('identifier', $next);
  111. }
  112. return $next->getValue();
  113. }
  114. /**
  115. * Returns next identifier or null if star delimiter token is found.
  116. *
  117. * @throws SyntaxErrorException If next token is not an identifier or a star delimiter
  118. */
  119. public function getNextIdentifierOrStar(): ?string
  120. {
  121. $next = $this->getNext();
  122. if ($next->isIdentifier()) {
  123. return $next->getValue();
  124. }
  125. if ($next->isDelimiter(['*'])) {
  126. return null;
  127. }
  128. throw SyntaxErrorException::unexpectedToken('identifier or "*"', $next);
  129. }
  130. /**
  131. * Skips next whitespace if any.
  132. */
  133. public function skipWhitespace()
  134. {
  135. $peek = $this->getPeek();
  136. if ($peek->isWhitespace()) {
  137. $this->getNext();
  138. }
  139. }
  140. }