Symfony2 Form Validator - 在刷新之前比较旧值和新值

时间:2013-06-25 20:12:23

标签: php symfony doctrine-orm entity validation

我想知道是否有一种方法可以在刷新之前比较实体内验证器中的旧值和新值。

我有一个Server实体可以将表单呈现为正常。该实体与status(N> 1)有关系,当状态从Unracked更改为Racked时,需要检查对服务器的SSH和FTP访问。如果未实现访问,验证器应该失败。

我已将验证程序回调映射到isServerValid()实体中的方法Server,如此处所述 http://symfony.com/doc/current/reference/constraints/Callback.html。我显然可以通过$this->status访问“新”值,但我怎样才能获得原始值?

在伪代码中,像这样:

public function isAuthorValid(ExecutionContextInterface $context)
{
    $original = ... ; // get old values
    if( $this->status !== $original->status && $this->status === 'Racked' && $original->status === 'Unracked' )
    {
        // check ftp and ssh connection
        // $context->addViolationAt('status', 'Unable to connect etc etc');
    }
}

提前致谢!

4 个答案:

答案 0 :(得分:32)

Symfony 2.5(http://symfony.com/doc/current/cookbook/validation/custom_constraint.html

的完整示例

在此示例中,字段的新值" integerField"实体" NoDecreasingInteger"必须高于储值。

创建约束:

// src/Acme/AcmeBundle/Validator/Constraints/IncrementOnly.php;
<?php
namespace Acme\AcmeBundle\Validator\Constraints;

use Symfony\Component\Validator\Constraint;

/**
 * @Annotation
 */
class IncrementOnly extends Constraint
{
  public $message = 'The new value %new% is least than the old %old%';

  public function getTargets()
  {
    return self::CLASS_CONSTRAINT;
  }

  public function validatedBy()
  {
    return 'increment_only';
  }
}

创建约束验证器:

// src/Acme/AcmeBundle/Validator/Constraints/IncrementOnlyValidator.php
<?php
namespace Acme\AcmeBundle\Validator\Constraints;

use Symfony\Component\Validator\Constraint;
use Symfony\Component\Validator\ConstraintValidator;

use Doctrine\ORM\EntityManager;

class IncrementOnlyValidator extends ConstraintValidator
{
  protected $em;

  public function __construct(EntityManager $em)
  {
    $this->em = $em;
  }

  public function validate($object, Constraint $constraint)
  {
    $new_value = $object->getIntegerField();

    $old_data = $this->em
      ->getUnitOfWork()
      ->getOriginalEntityData($object);

    // $old_data is empty if we create a new NoDecreasingInteger object.
    if (is_array($old_data) and !empty($old_data))
      {
        $old_value = $old_data['integerField'];

        if ($new_value < $old_value)
          {
            $this->context->buildViolation($constraint->message)
              ->setParameter("%new%", $new_value)
              ->setParameter('%old%', $old_value)
              ->addViolation();
          }
      }
  }
}

将验证器绑定到实体:

// src/Acme/AcmeBundle/Resources/config/validator.yml
Acme\AcmeBundle\Entity\NoDecreasingInteger:
  constraints:
     - Acme\AcmeBundle\Validator\Constraints\IncrementOnly: ~

将EntityManager注入IncrementOnlyValidator:

// src/Acme/AcmeBundle/Resources/config/services.yml
services:
   validator.increment_only:
        class: Acme\AcmeBundle\Validator\Constraints\IncrementOnlyValidator
        arguments: ["@doctrine.orm.entity_manager"]
        tags:
            - { name: validator.constraint_validator, alias: increment_only }

答案 1 :(得分:6)

在symfony2

中的自定义验证程序中访问EntityManager

您可以检查控制器操作中的先前值...但这不是一个干净的解决方案!

普通表单验证只能访问绑定到表单的数据...默认情况下不能访问“以前的”数据。

您尝试使用的回调约束无法访问容器或任何其他服务...因此您无法轻松访问实体管理器(或任何以前的数据提供程序)以检查以前的值。

您需要的是custom validator上的class level。需要类级别,因为如果要获取实体,则需要访问整个对象而不仅仅是单个值。

验证器本身可能如下所示:

namespace Vendor\YourBundle\Validation\Constraints;

use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\Validator\Constraint;
use Symfony\Component\Validator\ConstraintValidator;

class StatusValidator extends ConstraintValidator
{
    protected $container;

    public function __construct(ContainerInterface $container)
    {
        $this->container = $container;
    }

    public function validate($status, Constraint $constraint)
    {

        $em = $this->container->get('doctrine')->getEntityManager('default');

        $previousStatus = $em->getRepository('YourBundle:Status')->findOneBy(array('id' => $status->getId()));

        // ... do something with the previous status here

        if ( $previousStatus->getValue() != $status->getValue() ) {
            $this->context->addViolationAt('whatever', $constraint->message, array(), null);
        }
    }

    public function getTargets()
    {
        return self::CLASS_CONSTRAINT;
    }

    public function validatedBy()
    {
       return 'previous_value';
    }
}

...然后将验证器注册为服务并将其标记为验证器

services:
    validator.previous_value:
        class: Vendor\YourBundle\Validation\Constraints\StatusValidator

        # example! better inject only the services you need ... 
        # i.e. ... @doctrine.orm.entity_manager

        arguments: [ @service_container ]         
        tags:
            - { name: validator.constraint_validator, alias: previous_value }

最后使用约束为您的状态实体(即使用注释)

use Vendor\YourBundle\Validation\Constraints as MyValidation;

/**
 * @MyValidation\StatusValidator
 */
class Status 
{

答案 2 :(得分:0)

以前的答案完全有效,可能适合您的使用案例。

对于&#34;简单&#34;用例,虽然可能会填满。 如果实体可以通过(仅)表单进行编辑,则可以在FormBuilder上添加约束:

<?php

namespace AppBundle\Form\Type;

// ...

use Symfony\Component\Validator\Constraints\GreaterThanOrEqual;

/**
 * Class MyFormType
 */
class MyFormType extends AbstractType
{
    /**
     * {@inheritdoc}
     */
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder
            ->add('fooField', IntegerType::class, [
                'constraints' => [
                    new GreaterThanOrEqual(['value' => $builder->getData()->getFooField()])
                ]
            ])
        ;
    }
}

这适用于任何Symfony 2+版本。

答案 3 :(得分:0)

为了记录,这是使用Symfony5的方法。

首先,您需要在验证器的构造函数中注入EntityManagerInterface服务。 然后,使用它来检索原始实体。

/** @var EntityManagerInterface */
private $entityManager;

/**
 * MyValidator constructor.
 * @param EntityManagerInterface $entityManager
 */
public function __construct(EntityManagerInterface $entityManager)
{
    $this->entityManager = $entityManager;
}

/**
 * @param string $value
 * @param Constraint $constraint
 */
public function validate($value, Constraint $constraint)
{    
    $originalEntity = $this->entityManager
        ->getUnitOfWork()
        ->getOriginalEntityData($this->context->getObject());

    // ...
}