123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225 |
- <?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\PropertyAccess;
- use Symfony\Component\PropertyAccess\Exception\InvalidPropertyPathException;
- use Symfony\Component\PropertyAccess\Exception\OutOfBoundsException;
- use Symfony\Component\PropertyAccess\Exception\UnexpectedTypeException;
- /**
- * Default implementation of {@link PropertyPathInterface}.
- *
- * @author Bernhard Schussek <bschussek@gmail.com>
- */
- class PropertyPath implements \IteratorAggregate, PropertyPathInterface
- {
- /**
- * Character used for separating between plural and singular of an element.
- * @var string
- */
- const SINGULAR_SEPARATOR = '|';
- /**
- * The elements of the property path
- * @var array
- */
- private $elements = array();
- /**
- * The singular forms of the elements in the property path.
- * @var array
- */
- private $singulars = array();
- /**
- * The number of elements in the property path
- * @var integer
- */
- private $length;
- /**
- * Contains a Boolean for each property in $elements denoting whether this
- * element is an index. It is a property otherwise.
- * @var array
- */
- private $isIndex = array();
- /**
- * String representation of the path
- * @var string
- */
- private $pathAsString;
- /**
- * Constructs a property path from a string.
- *
- * @param PropertyPath|string $propertyPath The property path as string or instance
- *
- * @throws UnexpectedTypeException If the given path is not a string
- * @throws InvalidPropertyPathException If the syntax of the property path is not valid
- */
- public function __construct($propertyPath)
- {
- // Can be used as copy constructor
- if ($propertyPath instanceof PropertyPath) {
- /* @var PropertyPath $propertyPath */
- $this->elements = $propertyPath->elements;
- $this->singulars = $propertyPath->singulars;
- $this->length = $propertyPath->length;
- $this->isIndex = $propertyPath->isIndex;
- $this->pathAsString = $propertyPath->pathAsString;
- return;
- }
- if (!is_string($propertyPath)) {
- throw new UnexpectedTypeException($propertyPath, 'string or Symfony\Component\PropertyAccess\PropertyPath');
- }
- if ('' === $propertyPath) {
- throw new InvalidPropertyPathException('The property path should not be empty.');
- }
- $this->pathAsString = $propertyPath;
- $position = 0;
- $remaining = $propertyPath;
- // first element is evaluated differently - no leading dot for properties
- $pattern = '/^(([^\.\[]+)|\[([^\]]+)\])(.*)/';
- while (preg_match($pattern, $remaining, $matches)) {
- if ('' !== $matches[2]) {
- $element = $matches[2];
- $this->isIndex[] = false;
- } else {
- $element = $matches[3];
- $this->isIndex[] = true;
- }
- // Disabled this behaviour as the syntax is not yet final
- //$pos = strpos($element, self::SINGULAR_SEPARATOR);
- $pos = false;
- $singular = null;
- if (false !== $pos) {
- $singular = substr($element, $pos + 1);
- $element = substr($element, 0, $pos);
- }
- $this->elements[] = $element;
- $this->singulars[] = $singular;
- $position += strlen($matches[1]);
- $remaining = $matches[4];
- $pattern = '/^(\.(\w+)|\[([^\]]+)\])(.*)/';
- }
- if ('' !== $remaining) {
- throw new InvalidPropertyPathException(sprintf(
- 'Could not parse property path "%s". Unexpected token "%s" at position %d',
- $propertyPath,
- $remaining{0},
- $position
- ));
- }
- $this->length = count($this->elements);
- }
- /**
- * {@inheritdoc}
- */
- public function __toString()
- {
- return $this->pathAsString;
- }
- /**
- * {@inheritdoc}
- */
- public function getLength()
- {
- return $this->length;
- }
- /**
- * {@inheritdoc}
- */
- public function getParent()
- {
- if ($this->length <= 1) {
- return null;
- }
- $parent = clone $this;
- --$parent->length;
- $parent->pathAsString = substr($parent->pathAsString, 0, max(strrpos($parent->pathAsString, '.'), strrpos($parent->pathAsString, '[')));
- array_pop($parent->elements);
- array_pop($parent->singulars);
- array_pop($parent->isIndex);
- return $parent;
- }
- /**
- * Returns a new iterator for this path
- *
- * @return PropertyPathIteratorInterface
- */
- public function getIterator()
- {
- return new PropertyPathIterator($this);
- }
- /**
- * {@inheritdoc}
- */
- public function getElements()
- {
- return $this->elements;
- }
- /**
- * {@inheritdoc}
- */
- public function getElement($index)
- {
- if (!isset($this->elements[$index])) {
- throw new OutOfBoundsException(sprintf('The index %s is not within the property path', $index));
- }
- return $this->elements[$index];
- }
- /**
- * {@inheritdoc}
- */
- public function isProperty($index)
- {
- if (!isset($this->isIndex[$index])) {
- throw new OutOfBoundsException(sprintf('The index %s is not within the property path', $index));
- }
- return !$this->isIndex[$index];
- }
- /**
- * {@inheritdoc}
- */
- public function isIndex($index)
- {
- if (!isset($this->isIndex[$index])) {
- throw new OutOfBoundsException(sprintf('The index %s is not within the property path', $index));
- }
- return $this->isIndex[$index];
- }
- }
|