diff --git a/core/lib/Drupal/Core/TypedData/TypedDataManager.php b/core/lib/Drupal/Core/TypedData/TypedDataManager.php old mode 100644 new mode 100755 index d42a3d2..143636d --- a/core/lib/Drupal/Core/TypedData/TypedDataManager.php +++ b/core/lib/Drupal/Core/TypedData/TypedDataManager.php @@ -322,7 +322,7 @@ public function getValidator() { $this->validator = Validation::createValidatorBuilder() ->setMetadataFactory(new MetadataFactory()) ->setTranslator(new DrupalTranslator()) - ->setApiVersion(Validation::API_VERSION_2_4) + ->setApiVersion(Validation::API_VERSION_2_5) ->getValidator(); } return $this->validator; diff --git a/core/lib/Drupal/Core/TypedData/Validation/Metadata.php b/core/lib/Drupal/Core/TypedData/Validation/Metadata.php old mode 100644 new mode 100755 index 73bfc8b..d76af50 --- a/core/lib/Drupal/Core/TypedData/Validation/Metadata.php +++ b/core/lib/Drupal/Core/TypedData/Validation/Metadata.php @@ -7,98 +7,32 @@ namespace Drupal\Core\TypedData\Validation; -use Drupal\Core\TypedData\TypedDataInterface; -use Symfony\Component\Validator\ValidationVisitorInterface; -use Symfony\Component\Validator\PropertyMetadataInterface; +use Symfony\Component\Validator\Exception\BadMethodCallException; /** - * Typed data implementation of the validator MetadataInterface. + * Validator metadata for typed data objects containing no properties. */ -class Metadata implements PropertyMetadataInterface { +class Metadata extends MetadataBase { /** - * The name of the property, or empty if this is the root. - * - * @var string + * {@inheritdoc} */ - protected $name; - - /** - * The typed data object the metadata is about. - * - * @var \Drupal\Core\TypedData\TypedDataInterface - */ - protected $typedData; - - /** - * The metadata factory used. - * - * @var \Drupal\Core\TypedData\Validation\MetadataFactory - */ - protected $factory; - - /** - * Constructs the object. - * - * @param \Drupal\Core\TypedData\TypedDataInterface $typed_data - * The typed data object the metadata is about. - * @param $name - * The name of the property to get metadata for. Leave empty, if - * the data is the root of the typed data tree. - * @param \Drupal\Core\TypedData\Validation\MetadataFactory $factory - * The factory to use for instantiating property metadata. - */ - public function __construct(TypedDataInterface $typed_data, $name = '', MetadataFactory $factory) { - $this->typedData = $typed_data; - $this->name = $name; - $this->factory = $factory; - } - - /** - * Implements MetadataInterface::accept(). - */ - public function accept(ValidationVisitorInterface $visitor, $typed_data, $group, $propertyPath) { - - // @todo: Do we have to care about groups? Symfony class metadata has - // $propagatedGroup. - - $visitor->visit($this, $typed_data->getValue(), $group, $propertyPath); + public function getConstrainedProperties() { + return []; } /** - * Implements MetadataInterface::findConstraints(). + * {@inheritdoc} */ - public function findConstraints($group) { - return $this->typedData->getConstraints(); + public function hasPropertyMetadata($property_name) { + return []; } /** - * Returns the name of the property. - * - * @return string The property name. + * {@inheritdoc} */ - public function getPropertyName() { - return $this->name; + public function getPropertyMetadata($property_name) { + throw new BadMethodCallException("The data does not contain any properties."); } - /** - * Extracts the value of the property from the given container. - * - * @param mixed $container The container to extract the property value from. - * - * @return mixed The value of the property. - */ - public function getPropertyValue($container) { - return $this->typedData->getValue(); - } - - /** - * Returns the typed data object. - * - * @return \Drupal\Core\TypedData\TypedDataInterface - * The typed data object. - */ - public function getTypedData() { - return $this->typedData; - } } diff --git a/core/lib/Drupal/Core/TypedData/Validation/MetadataBase.php b/core/lib/Drupal/Core/TypedData/Validation/MetadataBase.php new file mode 100755 index 0000000..5236257 --- /dev/null +++ b/core/lib/Drupal/Core/TypedData/Validation/MetadataBase.php @@ -0,0 +1,151 @@ +typedData = $typed_data; + $this->factory = $factory; + } + + /** + * Returns the typed data object. + * + * @return \Drupal\Core\TypedData\TypedDataInterface + * The typed data object. + */ + public function getTypedData() { + return $this->typedData; + } + + /** + * {@inheritdoc} + */ + public function accept(ValidationVisitorInterface $visitor, $typed_data, $group, $propertyPath) { + + // @todo: Do we have to care about groups? Symfony class metadata has + // $propagatedGroup. + throw new BadMethodCallException('Not supported.'); + + $visitor->visit($this, $typed_data->getValue(), $group, $propertyPath); + } + + + /** + * {@inheritdoc} + */ + public function findConstraints($group) { + return $this->typedData->getConstraints(); + } + + /** + * {@inheritdoc} + */ + public function getConstraints() { + return $this->typedData->getConstraints(); + } + + /** + * {@inheritdoc} + */ + public function getCascadingStrategy() { + return CascadingStrategy::NONE; + } + + /** + * {@inheritdoc} + */ + public function getTraversalStrategy() { + return TraversalStrategy::IMPLICIT; + } + + /** + * {@inheritdoc} + */ + public function getClassName() { + return get_class($this->typedData); + } + + /** + * {@inheritdoc} + */ + public function getGroupSequence() { + return NULL; + } + + /** + * {@inheritdoc} + */ + public function hasGroupSequence() { + return FALSE; + } + + /** + * {@inheritdoc} + */ + public function isGroupSequenceProvider() { + return FALSE; + } + + /** + * {@inheritdoc} + */ + public function getPropertyName() { + return $this->typedData->getName(); + } + + /** + * {@inheritdoc} + */ + public function getPropertyValue($container) { + $property = $container->get($this->getPropertyName()); + if ($property instanceof ListInterface || $property instanceof ComplexDataInterface) { + // To let all constraints properly handle empty structures, pass on NULL + // if the data structure is empty. That way existing NotNull or NotBlank + // constraints work as expected. + return !$property->isEmpty() ? $property : NULL; + } + else { + return $property->getValue(); + } + } + +} diff --git a/core/lib/Drupal/Core/TypedData/Validation/MetadataFactory.php b/core/lib/Drupal/Core/TypedData/Validation/MetadataFactory.php old mode 100644 new mode 100755 index 2858daf..2d52915 --- a/core/lib/Drupal/Core/TypedData/Validation/MetadataFactory.php +++ b/core/lib/Drupal/Core/TypedData/Validation/MetadataFactory.php @@ -10,7 +10,7 @@ use Drupal\Core\TypedData\ComplexDataInterface; use Drupal\Core\TypedData\ListInterface; use Drupal\Core\TypedData\TypedDataInterface; -use Symfony\Component\Validator\MetadataFactoryInterface; +use Symfony\Component\Validator\Mapping\Factory\MetadataFactoryInterface; /** * Typed data implementation of the validator MetadataFactoryInterface. @@ -18,27 +18,25 @@ class MetadataFactory implements MetadataFactoryInterface { /** - * Implements MetadataFactoryInterface::getMetadataFor(). + * {@inheritdoc} * * @param \Drupal\Core\TypedData\TypedDataInterface $typed_data * Some typed data object containing the value to validate. - * @param $name - * (optional) The name of the property to get metadata for. Leave empty, if - * the data is the root of the typed data tree. */ - public function getMetadataFor($typed_data, $name = '') { + public function getMetadataFor($typed_data) { if (!$typed_data instanceof TypedDataInterface) { throw new \InvalidArgumentException('The passed value must be a typed data object.'); } $is_container = $typed_data instanceof ComplexDataInterface || $typed_data instanceof ListInterface; $class = '\Drupal\Core\TypedData\Validation\\' . ($is_container ? 'PropertyContainerMetadata' : 'Metadata'); - return new $class($typed_data, $name, $this); + return new $class($typed_data, $this); } /** - * Implements MetadataFactoryInterface::hasMetadataFor(). + * {@inheritdoc} */ public function hasMetadataFor($value) { return $value instanceof TypedDataInterface; } + } diff --git a/core/lib/Drupal/Core/TypedData/Validation/PropertyContainerMetadata.php b/core/lib/Drupal/Core/TypedData/Validation/PropertyContainerMetadata.php index f5850eb..50dd6dc 100644 --- a/core/lib/Drupal/Core/TypedData/Validation/PropertyContainerMetadata.php +++ b/core/lib/Drupal/Core/TypedData/Validation/PropertyContainerMetadata.php @@ -9,18 +9,19 @@ use Drupal\Core\TypedData\ComplexDataInterface; use Drupal\Core\TypedData\ListInterface; -use Symfony\Component\Validator\PropertyMetadataContainerInterface; use Symfony\Component\Validator\ValidationVisitorInterface; /** - * Typed data implementation of the validator MetadataInterface. + * Validator metadata for typed data property containers. */ -class PropertyContainerMetadata extends Metadata implements PropertyMetadataContainerInterface { +class PropertyContainerMetadata extends MetadataBase { /** - * Overrides Metadata::accept(). + * {@inheritdoc} */ public function accept(ValidationVisitorInterface $visitor, $typed_data, $group, $propertyPath) { + throw new BadMethodCallException('Not supported.'); + // To let all constraints properly handle empty structures, pass on NULL // if the data structure is empty. That way existing NotNull or NotBlank // constraints work as expected. @@ -39,7 +40,7 @@ public function accept(ValidationVisitorInterface $visitor, $typed_data, $group, } /** - * Implements PropertyMetadataContainerInterface::hasPropertyMetadata(). + * {@inheritdoc} */ public function hasPropertyMetadata($property_name) { try { @@ -52,17 +53,32 @@ public function hasPropertyMetadata($property_name) { } /** - * Implements PropertyMetadataContainerInterface::getPropertyMetadata(). + * {@inheritdoc} */ public function getPropertyMetadata($property_name) { + if ($this->typedData instanceof ListInterface || $this->typedData instanceof ComplexDataInterface) { + return array($this->factory->getMetadataFor($this->typedData->get($property_name))); + } + else { + throw new \LogicException("There are no known properties."); + } + } + + /** + * {@inheritdoc} + */ + public function getConstrainedProperties() { if ($this->typedData instanceof ListInterface) { - return array(new Metadata($this->typedData[$property_name], $property_name)); + // @todo: When https://www.drupal.org/node/2164601 gets committed, + // simplify this to return the list indexes based on the list count. + return array_keys(iterator_to_array($this->typedData)); } elseif ($this->typedData instanceof ComplexDataInterface) { - return array(new Metadata($this->typedData->get($property_name), $property_name)); + return array_keys($this->typedData->getProperties()); } else { throw new \LogicException("There are no known properties."); } } + } diff --git a/core/lib/Drupal/Core/Validation/Plugin/Validation/Constraint/ComplexDataConstraintValidator.php b/core/lib/Drupal/Core/Validation/Plugin/Validation/Constraint/ComplexDataConstraintValidator.php old mode 100644 new mode 100755 index 31ffc5b..6cc5b27 --- a/core/lib/Drupal/Core/Validation/Plugin/Validation/Constraint/ComplexDataConstraintValidator.php +++ b/core/lib/Drupal/Core/Validation/Plugin/Validation/Constraint/ComplexDataConstraintValidator.php @@ -38,10 +38,14 @@ public function validate($value, Constraint $constraint) { $property = $property->getValue(); } elseif ($property->isEmpty()) { - // @see \Drupal\Core\TypedData\Validation\PropertyContainerMetadata::accept(); + // @see \Drupal\Core\TypedData\Validation\ClassMetadata::accept(); $property = NULL; } - $this->context->validateValue($property, $constraints, $name, $group); + + $this->context + ->getValidator() + ->inContext($this->context) + ->validate($property, $constraints, $group); } } }